DEV: Add to-markdown decorator functions (#16943)

To be used in discourse-spoiler-alert
This commit is contained in:
Jarek Radosz 2022-05-31 11:06:41 +02:00 committed by GitHub
parent 69bab5e5a0
commit 711cd7c85d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 104 additions and 2 deletions

View File

@ -100,7 +100,7 @@ import { addSectionLink } from "discourse/lib/sidebar/custom-topics-section-link
// based on Semantic Versioning 2.0.0. Please update the changelog at // based on Semantic Versioning 2.0.0. Please update the changelog at
// docs/CHANGELOG-JAVASCRIPT-PLUGIN-API.md whenever you change the version // docs/CHANGELOG-JAVASCRIPT-PLUGIN-API.md whenever you change the version
// using the format described at https://keepachangelog.com/en/1.0.0/. // using the format described at https://keepachangelog.com/en/1.0.0/.
const PLUGIN_API_VERSION = "1.2.0"; const PLUGIN_API_VERSION = "1.3.0";
// This helper prevents us from applying the same `modifyClass` over and over in test mode. // This helper prevents us from applying the same `modifyClass` over and over in test mode.
function canModify(klass, type, resolverName, changes) { function canModify(klass, type, resolverName, changes) {

View File

@ -11,6 +11,50 @@ const hasChild = (e, n) => {
return (e.children || []).some((c) => c.name === n); return (e.children || []).some((c) => c.name === n);
}; };
let tagDecorateCallbacks = [];
let blockDecorateCallbacks = [];
/**
* Allows to add support for custom inline markdown/bbcode
*
* ```
* addTagDecorateCallback(function (text) {
* if (this.element.attributes.class === "loud") {
* this.prefix = "^^";
* this.suffix = "^^";
* return text.toLowerCase();
* }
* });
* ```
*/
export function addTagDecorateCallback(callback) {
tagDecorateCallbacks.push(callback);
}
export function clearTagDecorateCallbacks() {
tagDecorateCallbacks = [];
}
/**
* Allows to add support for custom block markdown/bbcode
*
* ```
* addBlockDecorateCallback(function (text) {
* if (this.element.attributes.class === "spoiled") {
* this.prefix = "[spoiler]";
* this.suffix = "[/spoiler]";
* }
* });
* ```
*/
export function addBlockDecorateCallback(callback) {
blockDecorateCallbacks.push(callback);
}
export function clearBlockDecorateCallbacks() {
blockDecorateCallbacks = [];
}
export class Tag { export class Tag {
constructor(name, prefix = "", suffix = "", inline = false) { constructor(name, prefix = "", suffix = "", inline = false) {
this.name = name; this.name = name;
@ -20,6 +64,14 @@ export class Tag {
} }
decorate(text) { decorate(text) {
for (const callback of tagDecorateCallbacks) {
const result = callback.call(this, text);
if (typeof result !== "undefined") {
text = result;
}
}
if (this.prefix || this.suffix) { if (this.prefix || this.suffix) {
text = [this.prefix, text, this.suffix].join(""); text = [this.prefix, text, this.suffix].join("");
} }
@ -137,6 +189,14 @@ export class Tag {
decorate(text) { decorate(text) {
const parent = this.element.parent; const parent = this.element.parent;
for (const callback of blockDecorateCallbacks) {
const result = callback.call(this, text);
if (typeof result !== "undefined") {
text = result;
}
}
if (this.name === "p" && parent && parent.name === "li") { if (this.name === "p" && parent && parent.name === "li") {
// fix for google docs // fix for google docs
this.gap = ""; this.gap = "";

View File

@ -64,6 +64,10 @@ import {
} from "discourse/lib/user-presence"; } from "discourse/lib/user-presence";
import PreloadStore from "discourse/lib/preload-store"; import PreloadStore from "discourse/lib/preload-store";
import { resetDefaultSectionLinks as resetTopicsSectionLinks } from "discourse/lib/sidebar/custom-topics-section-links"; import { resetDefaultSectionLinks as resetTopicsSectionLinks } from "discourse/lib/sidebar/custom-topics-section-links";
import {
clearBlockDecorateCallbacks,
clearTagDecorateCallbacks,
} from "discourse/lib/to-markdown";
const LEGACY_ENV = !setupApplicationTest; const LEGACY_ENV = !setupApplicationTest;
@ -188,6 +192,8 @@ function testCleanup(container, app) {
} }
restoreBaseUri(); restoreBaseUri();
resetTopicsSectionLinks(); resetTopicsSectionLinks();
clearTagDecorateCallbacks();
clearBlockDecorateCallbacks();
} }
export function discourseModule(name, options) { export function discourseModule(name, options) {

View File

@ -1,5 +1,8 @@
import { module, test } from "qunit"; import { module, test } from "qunit";
import toMarkdown from "discourse/lib/to-markdown"; import toMarkdown, {
addBlockDecorateCallback,
addTagDecorateCallback,
} from "discourse/lib/to-markdown";
module("Unit | Utility | to-markdown", function () { module("Unit | Utility | to-markdown", function () {
test("converts styles between normal words", function (assert) { test("converts styles between normal words", function (assert) {
@ -458,4 +461,31 @@ test2
'<img src="" />'; '<img src="" />';
assert.strictEqual(toMarkdown(html), "[image]"); assert.strictEqual(toMarkdown(html), "[image]");
}); });
test("addTagDecorateCallback", function (assert) {
const html = `<span class="loud">HELLO THERE</span>`;
addTagDecorateCallback(function (text) {
if (this.element.attributes.class === "loud") {
this.prefix = "^^";
this.suffix = "^^";
return text.toLowerCase();
}
});
assert.strictEqual(toMarkdown(html), "^^hello there^^");
});
test("addBlockDecorateCallback", function (assert) {
const html = `<div class="quiet">hey<br>there</div>`;
addBlockDecorateCallback(function () {
if (this.element.attributes.class === "quiet") {
this.prefix = "[quiet]";
this.suffix = "[/quiet]";
}
});
assert.strictEqual(toMarkdown(html), "[quiet]hey\nthere[/quiet]");
});
}); });

View File

@ -7,6 +7,12 @@ in this file.
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
## [1.3.0] - 2022-05-29
### Added
- Adds `beforeToMarkdownTagDecorate` and `beforeToMarkdownBlockDecorate` that allow to modify to-markdown behavior.
## [1.2.0] - 2022-03-18 ## [1.2.0] - 2022-03-18
### Added ### Added