2016-06-14 14:31:51 -04:00
import Quote from 'discourse/lib/quote';
import Post from 'discourse/models/post';
import { default as PrettyText, buildOptions } from 'pretty-text/pretty-text';
import { IMAGE_VERSION as v} from 'pretty-text/emoji';
2017-06-14 13:57:58 -04:00
QUnit.module("lib:pretty-text");
2016-06-14 14:31:51 -04:00
2017-07-14 08:27:28 -04:00
const rawOpts = {
2016-06-14 14:31:51 -04:00
siteSettings: {
enable_emoji: true,
2018-01-23 21:23:27 -05:00
enable_emoji_shortcuts: true,
2017-12-07 11:30:47 -05:00
enable_mentions: true,
2016-06-14 14:31:51 -04:00
emoji_set: 'emoji_one',
highlighted_languages: 'json|ruby|javascript',
default_code_lang: 'auto',
2018-02-01 00:16:14 -05:00
enable_markdown_linkify: true,
markdown_linkify_tlds: 'com'
2016-06-14 14:31:51 -04:00
},
2018-01-09 15:51:47 -05:00
censoredWords: 'shucks|whiz|whizzer|a**le|badword*',
2016-06-14 14:31:51 -04:00
getURL: url => url
2017-07-14 08:27:28 -04:00
};
const defaultOpts = buildOptions(rawOpts);
2016-06-14 14:31:51 -04:00
2017-06-14 13:57:58 -04:00
QUnit.assert.cooked = function(input, expected, message) {
const actual = new PrettyText(defaultOpts).cook(input);
this.pushResult({
result: actual === expected.replace(/\/>/g, ">"),
actual,
expected,
message
});
2016-06-14 14:31:51 -04:00
};
2017-06-14 13:57:58 -04:00
QUnit.assert.cookedOptions = function(input, opts, expected, message) {
2017-07-14 08:27:28 -04:00
const merged = _.merge({}, rawOpts, opts);
const actual = new PrettyText(buildOptions(merged)).cook(input);
2017-06-14 13:57:58 -04:00
this.pushResult({
result: actual === expected,
actual,
expected,
message
});
2013-06-18 18:03:36 -04:00
};
2017-06-14 13:57:58 -04:00
QUnit.assert.cookedPara = function(input, expected, message) {
QUnit.assert.cooked(input, `<p>${expected}</p>`, message);
2013-06-21 14:06:20 -04:00
};
2013-06-18 18:03:36 -04:00
2017-07-12 18:10:51 -04:00
2017-07-14 08:27:28 -04:00
QUnit.skip("Pending Engine fixes and spec fixes", assert => {
assert.cooked("Derpy: http://derp.com?_test_=1",
'<p>Derpy: <a href=https://derp.com?_test_=1"http://derp.com?_test_=1">http://derp.com?_test_=1</a></p>',
"works with underscores in urls");
assert.cooked("**a*_b**", "<p><strong>a*_b</strong></p>", "allows for characters within bold");
});
QUnit.test("buildOptions", assert => {
assert.ok(buildOptions({ siteSettings: { enable_emoji: true } }).discourse.features.emoji, 'emoji enabled');
assert.ok(!buildOptions({ siteSettings: { enable_emoji: false } }).discourse.features.emoji, 'emoji disabled');
2016-06-14 14:31:51 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("basic cooking", assert => {
assert.cooked("hello", "<p>hello</p>", "surrounds text with paragraphs");
assert.cooked("**evil**", "<p><strong>evil</strong></p>", "it bolds text.");
assert.cooked("__bold__", "<p><strong>bold</strong></p>", "it bolds text.");
assert.cooked("*trout*", "<p><em>trout</em></p>", "it italicizes text.");
assert.cooked("_trout_", "<p><em>trout</em></p>", "it italicizes text.");
2018-03-07 20:19:32 -05:00
assert.cooked("***hello***", "<p><em><strong>hello</strong></em></p>", "it can do bold and italics at once.");
2017-06-14 13:57:58 -04:00
assert.cooked("word_with_underscores", "<p>word_with_underscores</p>", "it doesn't do intraword italics");
assert.cooked("common/_special_font_face.html.erb", "<p>common/_special_font_face.html.erb</p>", "it doesn't intraword with a slash");
assert.cooked("hello \\*evil\\*", "<p>hello *evil*</p>", "it supports escaping of asterisks");
assert.cooked("hello \\_evil\\_", "<p>hello _evil_</p>", "it supports escaping of italics");
assert.cooked("brussels sprouts are *awful*.", "<p>brussels sprouts are <em>awful</em>.</p>", "it doesn't swallow periods.");
2013-06-21 11:36:33 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("Nested bold and italics", assert => {
assert.cooked("*this is italic **with some bold** inside*", "<p><em>this is italic <strong>with some bold</strong> inside</em></p>", "it handles nested bold in italics");
2014-09-22 16:51:48 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("Traditional Line Breaks", assert => {
2016-06-14 14:31:51 -04:00
const input = "1\n2\n3";
2017-07-14 08:27:28 -04:00
assert.cooked(input, "<p>1<br>\n2<br>\n3</p>", "automatically handles trivial newlines");
assert.cookedOptions(input, { siteSettings: {traditional_markdown_linebreaks: true} }, "<p>1\n2\n3</p>");
2013-08-26 15:21:23 -04:00
});
2013-06-21 11:36:33 -04:00
2017-06-14 13:57:58 -04:00
QUnit.test("Unbalanced underscores", assert => {
assert.cooked("[evil_trout][1] hello_\n\n[1]: http://eviltrout.com", "<p><a href=\"http://eviltrout.com\">evil_trout</a> hello_</p>");
2013-11-20 11:53:06 -05:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("Line Breaks", assert => {
assert.cooked("[] first choice\n[] second choice",
2017-07-14 08:27:28 -04:00
"<p>[] first choice<br>\n[] second choice</p>",
2013-08-26 15:21:23 -04:00
"it handles new lines correctly with [] options");
2013-10-18 15:20:27 -04:00
2017-07-14 08:27:28 -04:00
// note this is a change from previous engine but is correct
// we have an html block and behavior is defined per common mark
// spec
// ole engine would wrap trout in a <p>
2017-06-14 13:57:58 -04:00
assert.cooked("<blockquote>evil</blockquote>\ntrout",
2017-07-14 08:27:28 -04:00
"<blockquote>evil</blockquote>\ntrout",
2013-10-18 15:20:27 -04:00
"it doesn't insert <br> after blockquotes");
2017-06-14 13:57:58 -04:00
assert.cooked("leading<blockquote>evil</blockquote>\ntrout",
2017-07-14 08:27:28 -04:00
"<p>leading<blockquote>evil</blockquote><br>\ntrout</p>",
2013-10-18 15:20:27 -04:00
"it doesn't insert <br> after blockquotes with leading text");
2013-06-18 18:03:36 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("Paragraphs for HTML", assert => {
assert.cooked("<div>hello world</div>", "<div>hello world</div>", "it doesn't surround <div> with paragraphs");
assert.cooked("<p>hello world</p>", "<p>hello world</p>", "it doesn't surround <p> with paragraphs");
assert.cooked("<i>hello world</i>", "<p><i>hello world</i></p>", "it surrounds inline <i> html tags with paragraphs");
assert.cooked("<b>hello world</b>", "<p><b>hello world</b></p>", "it surrounds inline <b> html tags with paragraphs");
2013-10-18 15:20:27 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("Links", assert => {
2013-08-08 18:14:12 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("EvilTrout: http://eviltrout.com",
2013-08-22 18:03:01 -04:00
'<p>EvilTrout: <a href="http://eviltrout.com">http://eviltrout.com</a></p>',
"autolinks a URL");
2017-06-14 13:57:58 -04:00
assert.cooked("Youtube: http://www.youtube.com/watch?v=1MrpeBRkM5A",
2013-06-18 18:03:36 -04:00
'<p>Youtube: <a href="http://www.youtube.com/watch?v=1MrpeBRkM5A">http://www.youtube.com/watch?v=1MrpeBRkM5A</a></p>',
"allows links to contain query params");
2017-06-14 13:57:58 -04:00
assert.cooked("Derpy: http://derp.com?__test=1",
2013-08-08 18:14:12 -04:00
'<p>Derpy: <a href="http://derp.com?__test=1">http://derp.com?__test=1</a></p>',
"works with double underscores in urls");
2017-06-14 13:57:58 -04:00
assert.cooked("Atwood: www.codinghorror.com",
2013-06-18 18:03:36 -04:00
'<p>Atwood: <a href="http://www.codinghorror.com">www.codinghorror.com</a></p>',
"autolinks something that begins with www");
2017-06-14 13:57:58 -04:00
assert.cooked("Atwood: http://www.codinghorror.com",
2013-06-18 18:03:36 -04:00
'<p>Atwood: <a href="http://www.codinghorror.com">http://www.codinghorror.com</a></p>',
"autolinks a URL with http://www");
2017-06-14 13:57:58 -04:00
assert.cooked("EvilTrout: http://eviltrout.com hello",
2013-08-21 22:49:55 -04:00
'<p>EvilTrout: <a href="http://eviltrout.com">http://eviltrout.com</a> hello</p>',
"autolinks with trailing text");
2017-06-14 13:57:58 -04:00
assert.cooked("here is [an example](http://twitter.com)",
2013-06-18 18:03:36 -04:00
'<p>here is <a href="http://twitter.com">an example</a></p>',
"supports markdown style links");
2017-06-14 13:57:58 -04:00
assert.cooked("Batman: http://en.wikipedia.org/wiki/The_Dark_Knight_(film)",
2013-06-18 18:03:36 -04:00
'<p>Batman: <a href="http://en.wikipedia.org/wiki/The_Dark_Knight_(film)">http://en.wikipedia.org/wiki/The_Dark_Knight_(film)</a></p>',
"autolinks a URL with parentheses (like Wikipedia)");
2013-08-08 18:14:12 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("Here's a tweet:\nhttps://twitter.com/evil_trout/status/345954894420787200",
2017-07-14 08:27:28 -04:00
"<p>Here's a tweet:<br>\n<a href=\"https://twitter.com/evil_trout/status/345954894420787200\" class=\"onebox\" target=\"_blank\">https://twitter.com/evil_trout/status/345954894420787200</a></p>",
2013-08-08 18:14:12 -04:00
"It doesn't strip the new line.");
2017-06-14 13:57:58 -04:00
assert.cooked("1. View @eviltrout's profile here: http://meta.discourse.org/u/eviltrout/activity<br/>next line.",
2017-07-14 08:27:28 -04:00
"<ol>\n<li>View <span class=\"mention\">@eviltrout</span>'s profile here: <a href=\"http://meta.discourse.org/u/eviltrout/activity\">http://meta.discourse.org/u/eviltrout/activity</a><br>next line.</li>\n</ol>",
2013-08-22 18:03:01 -04:00
"allows autolinking within a list without inserting a paragraph.");
2017-06-14 13:57:58 -04:00
assert.cooked("[3]: http://eviltrout.com", "", "It doesn't autolink markdown link references");
2013-08-08 18:14:12 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("[]: http://eviltrout.com", "<p>[]: <a href=\"http://eviltrout.com\">http://eviltrout.com</a></p>", "It doesn't accept empty link references");
2014-08-30 20:55:43 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("[b]label[/b]: description", "<p><span class=\"bbcode-b\">label</span>: description</p>", "It doesn't accept BBCode as link references");
2014-08-30 20:55:43 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("http://discourse.org and http://discourse.org/another_url and http://www.imdb.com/name/nm2225369",
2013-08-08 18:14:12 -04:00
"<p><a href=\"http://discourse.org\">http://discourse.org</a> and " +
"<a href=\"http://discourse.org/another_url\">http://discourse.org/another_url</a> and " +
"<a href=\"http://www.imdb.com/name/nm2225369\">http://www.imdb.com/name/nm2225369</a></p>",
'allows multiple links on one line');
2017-07-14 08:27:28 -04:00
assert.cooked("* [Evil Trout][1]\n\n[1]: http://eviltrout.com",
"<ul>\n<li><a href=\"http://eviltrout.com\">Evil Trout</a></li>\n</ul>",
2013-08-28 13:06:41 -04:00
"allows markdown link references in a list");
2017-06-14 13:57:58 -04:00
assert.cooked("User [MOD]: Hello!",
2013-10-21 14:16:23 -04:00
"<p>User [MOD]: Hello!</p>",
"It does not consider references that are obviously not URLs");
2014-07-03 16:54:56 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("<small>http://eviltrout.com</small>", "<p><small><a href=\"http://eviltrout.com\">http://eviltrout.com</a></small></p>", "Links within HTML tags");
2014-07-14 14:26:48 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("[http://google.com ... wat](http://discourse.org)",
2014-07-14 14:26:48 -04:00
"<p><a href=\"http://discourse.org\">http://google.com ... wat</a></p>",
2016-06-14 14:31:51 -04:00
"it supports links within links");
2014-08-23 16:46:13 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("[http://google.com](http://discourse.org)",
2016-10-29 20:30:36 -04:00
"<p><a href=\"http://discourse.org\">http://google.com</a></p>",
"it supports markdown links where the name and link match");
2017-06-14 13:57:58 -04:00
assert.cooked("[Link](http://www.example.com) (with an outer \"description\")",
2017-07-14 08:27:28 -04:00
"<p><a href=\"http://www.example.com\">Link</a> (with an outer "description")</p>",
2014-09-22 16:51:48 -04:00
"it doesn't consume closing parens as part of the url");
2014-12-31 11:44:56 -05:00
2017-06-14 13:57:58 -04:00
assert.cooked("A link inside parentheses (http://www.example.com)",
2016-10-29 20:30:36 -04:00
"<p>A link inside parentheses (<a href=\"http://www.example.com\">http://www.example.com</a>)</p>",
"it auto-links a url within parentheses");
2017-06-14 13:57:58 -04:00
assert.cooked("[ul][1]\n\n[1]: http://eviltrout.com",
2014-12-31 11:44:56 -05:00
"<p><a href=\"http://eviltrout.com\">ul</a></p>",
"it can use `ul` as a link name");
2013-06-18 18:03:36 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("simple quotes", assert => {
2017-07-14 08:27:28 -04:00
assert.cooked("> nice!", "<blockquote>\n<p>nice!</p>\n</blockquote>", "it supports simple quotes");
assert.cooked(" > nice!", "<blockquote>\n<p>nice!</p>\n</blockquote>", "it allows quotes with preceding spaces");
2017-06-14 13:57:58 -04:00
assert.cooked("> level 1\n> > level 2",
2017-07-14 08:27:28 -04:00
"<blockquote>\n<p>level 1</p>\n<blockquote>\n<p>level 2</p>\n</blockquote>\n</blockquote>",
2013-08-29 15:18:27 -04:00
"it allows nesting of blockquotes");
2017-06-14 13:57:58 -04:00
assert.cooked("> level 1\n> > level 2",
2017-07-14 08:27:28 -04:00
"<blockquote>\n<p>level 1</p>\n<blockquote>\n<p>level 2</p>\n</blockquote>\n</blockquote>",
2013-08-29 15:18:27 -04:00
"it allows nesting of blockquotes with spaces");
2013-09-05 17:03:35 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("- hello\n\n > world\n > eviltrout",
2017-07-14 08:27:28 -04:00
`<ul>
<li>
<p>hello</p>
<blockquote>
<p>world<br>
eviltrout</p>
</blockquote>
</li>
</ul>`,
2013-09-05 17:03:35 -04:00
"it allows quotes within a list.");
2014-07-15 16:41:53 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("- <p>eviltrout</p>",
2017-07-14 08:27:28 -04:00
"<ul>\n<li>\n<p>eviltrout</p></li>\n</ul>",
2014-07-15 16:41:53 -04:00
"it allows paragraphs within a list.");
2017-07-14 08:27:28 -04:00
assert.cooked(" > indent 1\n > indent 2", "<blockquote>\n<p>indent 1<br>\nindent 2</p>\n</blockquote>", "allow multiple spaces to indent");
2013-09-05 17:03:35 -04:00
2013-08-29 15:18:27 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("Quotes", assert => {
2013-08-21 22:36:02 -04:00
2017-07-14 08:27:28 -04:00
assert.cookedOptions("[quote=\"eviltrout, post: 1\"]\na quote\n\nsecond line\n\nthird line\n[/quote]",
2013-08-21 22:36:02 -04:00
{ topicId: 2 },
2018-04-02 14:18:10 -04:00
`<aside class=\"quote no-group\" data-post=\"1\">
2017-07-14 08:27:28 -04:00
<div class=\"title\">
<div class=\"quote-controls\"></div>
eviltrout:</div>
<blockquote>
<p>a quote</p>
<p>second line</p>
<p>third line</p>
</blockquote>
</aside>`,
2013-08-21 22:36:02 -04:00
"works with multiple lines");
2017-07-14 08:27:28 -04:00
assert.cookedOptions("[quote=\"bob, post:1\"]\nmy quote\n[/quote]",
2013-12-30 13:29:52 -05:00
{ topicId: 2, lookupAvatar: function() { } },
2018-04-02 14:18:10 -04:00
`<aside class=\"quote no-group\" data-post=\"1\">
2017-07-14 08:27:28 -04:00
<div class=\"title\">
<div class=\"quote-controls\"></div>
bob:</div>
<blockquote>
<p>my quote</p>
</blockquote>
</aside>`,
2013-06-18 18:03:36 -04:00
"includes no avatar if none is found");
2017-04-03 14:38:30 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked(`[quote]\na\n\n[quote]\nb\n[/quote]\n[/quote]`,
2018-04-02 14:18:10 -04:00
`<aside class=\"quote no-group\">
2017-07-14 08:27:28 -04:00
<blockquote>
<p>a</p>
2018-04-02 14:18:10 -04:00
<aside class=\"quote no-group\">
2017-07-14 08:27:28 -04:00
<blockquote>
<p>b</p>
</blockquote>
</aside>
</blockquote>
</aside>`,
2017-04-03 14:38:30 -04:00
"handles nested quotes properly");
2017-11-03 09:51:40 -04:00
assert.cookedOptions(`[quote="bob, post:1, topic:1"]\ntest quote\n[/quote]`, { lookupPrimaryUserGroupByPostNumber: () => "aUserGroup" },
`<aside class="quote group-aUserGroup" data-post="1" data-topic="1">
<div class="title">
<div class="quote-controls"></div>
bob:</div>
<blockquote>
<p>test quote</p>
</blockquote>
</aside>`,
"quote has group class");
2013-06-26 18:41:48 -04:00
});
2013-06-18 18:03:36 -04:00
2017-06-14 13:57:58 -04:00
QUnit.test("Mentions", assert => {
2016-06-14 14:31:51 -04:00
const alwaysTrue = { mentionLookup: (function() { return "user"; }) };
2013-08-24 15:00:18 -04:00
2017-06-14 13:57:58 -04:00
assert.cookedOptions("Hello @sam", alwaysTrue,
2017-03-28 12:16:58 -04:00
"<p>Hello <a class=\"mention\" href=\"/u/sam\">@sam</a></p>",
2013-06-18 18:03:36 -04:00
"translates mentions to links");
2017-06-14 13:57:58 -04:00
assert.cooked("[@codinghorror](https://twitter.com/codinghorror)",
2014-05-06 17:48:11 -04:00
"<p><a href=\"https://twitter.com/codinghorror\">@codinghorror</a></p>",
"it doesn't do mentions within links");
2017-06-14 13:57:58 -04:00
assert.cookedOptions("[@codinghorror](https://twitter.com/codinghorror)", alwaysTrue,
2014-05-06 17:48:11 -04:00
"<p><a href=\"https://twitter.com/codinghorror\">@codinghorror</a></p>",
"it doesn't do link mentions within links");
2017-06-14 13:57:58 -04:00
assert.cooked("Hello @EvilTrout",
2015-03-09 07:32:37 -04:00
"<p>Hello <span class=\"mention\">@EvilTrout</span></p>",
"adds a mention class");
2017-06-14 13:57:58 -04:00
assert.cooked("robin@email.host",
2015-03-09 07:32:37 -04:00
"<p>robin@email.host</p>",
"won't add mention class to an email address");
2017-06-14 13:57:58 -04:00
assert.cooked("hanzo55@yahoo.com",
2017-07-14 08:27:28 -04:00
"<p><a href=\"mailto:hanzo55@yahoo.com\">hanzo55@yahoo.com</a></p>",
2015-03-09 07:32:37 -04:00
"won't be affected by email addresses that have a number before the @ symbol");
2017-06-14 13:57:58 -04:00
assert.cooked("@EvilTrout yo",
2015-03-09 07:32:37 -04:00
"<p><span class=\"mention\">@EvilTrout</span> yo</p>",
"it handles mentions at the beginning of a string");
2017-06-14 13:57:58 -04:00
assert.cooked("yo\n@EvilTrout",
2017-07-14 08:27:28 -04:00
"<p>yo<br>\n<span class=\"mention\">@EvilTrout</span></p>",
2015-03-09 07:32:37 -04:00
"it handles mentions at the beginning of a new line");
2017-06-14 13:57:58 -04:00
assert.cooked("`evil` @EvilTrout `trout`",
2013-08-08 18:14:12 -04:00
"<p><code>evil</code> <span class=\"mention\">@EvilTrout</span> <code>trout</code></p>",
2013-06-18 18:03:36 -04:00
"deals correctly with multiple <code> blocks");
2015-03-09 07:32:37 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("```\na @test\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-auto\">a @test\n</code></pre>",
2015-03-09 07:32:37 -04:00
"should not do mentions within a code block.");
2013-06-18 18:03:36 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("> foo bar baz @eviltrout",
2017-07-14 08:27:28 -04:00
"<blockquote>\n<p>foo bar baz <span class=\"mention\">@eviltrout</span></p>\n</blockquote>",
2013-08-21 16:10:16 -04:00
"handles mentions in simple quotes");
2017-06-14 13:57:58 -04:00
assert.cooked("> foo bar baz @eviltrout ohmagerd\nlook at this",
2017-07-14 08:27:28 -04:00
"<blockquote>\n<p>foo bar baz <span class=\"mention\">@eviltrout</span> ohmagerd<br>\nlook at this</p>\n</blockquote>",
2013-08-21 16:10:16 -04:00
"does mentions properly with trailing text within a simple quote");
2017-06-14 13:57:58 -04:00
assert.cooked("`code` is okay before @mention",
2013-08-22 12:18:03 -04:00
"<p><code>code</code> is okay before <span class=\"mention\">@mention</span></p>",
"Does not mention in an inline code block");
2017-06-14 13:57:58 -04:00
assert.cooked("@mention is okay before `code`",
2013-08-22 12:18:03 -04:00
"<p><span class=\"mention\">@mention</span> is okay before <code>code</code></p>",
"Does not mention in an inline code block");
2017-06-14 13:57:58 -04:00
assert.cooked("don't `@mention`",
2013-08-22 12:18:03 -04:00
"<p>don't <code>@mention</code></p>",
"Does not mention in an inline code block");
2017-06-14 13:57:58 -04:00
assert.cooked("Yes `@this` should be code @eviltrout",
2013-08-22 15:54:41 -04:00
"<p>Yes <code>@this</code> should be code <span class=\"mention\">@eviltrout</span></p>",
"Does not mention in an inline code block");
2017-06-14 13:57:58 -04:00
assert.cooked("@eviltrout and `@eviltrout`",
2013-08-22 16:13:02 -04:00
"<p><span class=\"mention\">@eviltrout</span> and <code>@eviltrout</code></p>",
"you can have a mention in an inline code block following a real mention.");
2017-06-14 13:57:58 -04:00
assert.cooked("1. this is a list\n\n2. this is an @eviltrout mention\n",
2017-07-14 08:27:28 -04:00
"<ol>\n<li>\n<p>this is a list</p>\n</li>\n<li>\n<p>this is an <span class=\"mention\">@eviltrout</span> mention</p>\n</li>\n</ol>",
2013-08-22 17:50:07 -04:00
"it mentions properly in a list.");
2017-06-14 13:57:58 -04:00
assert.cooked("Hello @foo/@bar",
2015-10-16 19:17:54 -04:00
"<p>Hello <span class=\"mention\">@foo</span>/<span class=\"mention\">@bar</span></p>",
"handles mentions separated by a slash.");
2017-06-14 13:57:58 -04:00
assert.cookedOptions("@eviltrout", alwaysTrue,
2017-03-28 12:16:58 -04:00
"<p><a class=\"mention\" href=\"/u/eviltrout\">@eviltrout</a></p>",
2013-08-24 15:00:18 -04:00
"it doesn't onebox mentions");
2017-06-14 13:57:58 -04:00
assert.cookedOptions("<small>a @sam c</small>", alwaysTrue,
2017-03-28 12:16:58 -04:00
"<p><small>a <a class=\"mention\" href=\"/u/sam\">@sam</a> c</small></p>",
2014-07-03 16:54:56 -04:00
"it allows mentions within HTML tags");
2013-06-18 18:03:36 -04:00
});
2017-12-07 11:30:47 -05:00
QUnit.test("Mentions - disabled", assert => {
assert.cookedOptions("@eviltrout",
{ siteSettings : { enable_mentions: false }},
"<p>@eviltrout</p>");
});
2017-06-14 13:57:58 -04:00
QUnit.test("Category hashtags", assert => {
2016-06-14 14:31:51 -04:00
const alwaysTrue = { categoryHashtagLookup: (function() { return ["http://test.discourse.org/category-hashtag", "category-hashtag"]; }) };
2015-12-28 01:28:16 -05:00
2017-06-14 13:57:58 -04:00
assert.cookedOptions("Check out #category-hashtag", alwaysTrue,
2016-01-21 03:21:54 -05:00
"<p>Check out <a class=\"hashtag\" href=\"http://test.discourse.org/category-hashtag\">#<span>category-hashtag</span></a></p>",
2015-12-28 01:28:16 -05:00
"it translates category hashtag into links");
2017-06-14 13:57:58 -04:00
assert.cooked("Check out #category-hashtag",
2015-12-28 01:28:16 -05:00
"<p>Check out <span class=\"hashtag\">#category-hashtag</span></p>",
"it does not translate category hashtag into links if it is not a valid category hashtag");
2017-06-14 13:57:58 -04:00
assert.cookedOptions("[#category-hashtag](http://www.test.com)", alwaysTrue,
2015-12-28 01:28:16 -05:00
"<p><a href=\"http://www.test.com\">#category-hashtag</a></p>",
"it does not translate category hashtag within links");
2017-06-14 13:57:58 -04:00
assert.cooked("```\n# #category-hashtag\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-auto\"># #category-hashtag\n</code></pre>",
2015-12-28 01:28:16 -05:00
"it does not translate category hashtags to links in code blocks");
2017-06-14 13:57:58 -04:00
assert.cooked("># #category-hashtag\n",
2017-07-14 08:27:28 -04:00
"<blockquote>\n<h1><span class=\"hashtag\">#category-hashtag</span></h1>\n</blockquote>",
2015-12-28 01:28:16 -05:00
"it handles category hashtags in simple quotes");
2017-06-14 13:57:58 -04:00
assert.cooked("# #category-hashtag",
2015-12-28 01:28:16 -05:00
"<h1><span class=\"hashtag\">#category-hashtag</span></h1>",
"it works within ATX-style headers");
2017-06-14 13:57:58 -04:00
assert.cooked("don't `#category-hashtag`",
2015-12-28 01:28:16 -05:00
"<p>don't <code>#category-hashtag</code></p>",
"it does not mention in an inline code block");
2017-06-14 13:57:58 -04:00
assert.cooked("<small>#category-hashtag</small>",
2015-12-28 01:28:16 -05:00
"<p><small><span class=\"hashtag\">#category-hashtag</span></small></p>",
"it works between HTML tags");
});
2013-10-16 10:39:06 -04:00
2017-06-14 13:57:58 -04:00
QUnit.test("Heading", assert => {
assert.cooked("**Bold**\n----------", "<h2><strong>Bold</strong></h2>", "It will bold the heading");
2014-07-25 15:59:16 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("bold and italics", assert => {
2017-07-14 08:27:28 -04:00
assert.cooked("a \"**hello**\"", "<p>a "<strong>hello</strong>"</p>", "bolds in quotes");
2017-06-14 13:57:58 -04:00
assert.cooked("(**hello**)", "<p>(<strong>hello</strong>)</p>", "bolds in parens");
2017-07-14 08:27:28 -04:00
assert.cooked("**hello**\nworld", "<p><strong>hello</strong><br>\nworld</p>", "allows newline after bold");
assert.cooked("**hello**\n**world**", "<p><strong>hello</strong><br>\n<strong>world</strong></p>", "newline between two bolds");
2017-06-14 13:57:58 -04:00
assert.cooked("** hello**", "<p>** hello**</p>", "does not bold on a space boundary");
assert.cooked("**hello **", "<p>**hello **</p>", "does not bold on a space boundary");
assert.cooked("**你hello**", "<p><strong>你hello</strong></p>", "allows bolded chinese");
2013-10-16 10:39:06 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("Escaping", assert => {
assert.cooked("*\\*laughs\\**", "<p><em>*laughs*</em></p>", "allows escaping strong");
assert.cooked("*\\_laughs\\_*", "<p><em>_laughs_</em></p>", "allows escaping em");
2014-09-22 16:51:48 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("New Lines", assert => {
2017-07-14 08:27:28 -04:00
// historically we would not continue inline em or b across lines,
// however commonmark gives us no switch to do so and we would be very non compliant.
// turning softbreaks into a newline is just a renderer option, not a parser switch.
assert.cooked("_abc\ndef_", "<p><em>abc<br>\ndef</em></p>", "it does allow inlines to span new lines");
assert.cooked("_abc\n\ndef_", "<p>_abc</p>\n<p>def_</p>", "it does not allow inlines to span new paragraphs");
2014-07-25 16:07:42 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("Oneboxing", assert => {
2013-06-18 18:03:36 -04:00
2016-06-14 14:31:51 -04:00
function matches(input, regexp) {
return new PrettyText(defaultOpts).cook(input).match(regexp);
2013-06-18 18:03:36 -04:00
};
2017-06-14 13:57:58 -04:00
assert.ok(!matches("- http://www.textfiles.com/bbs/MINDVOX/FORUMS/ethics\n\n- http://drupal.org", /onebox/),
2013-08-08 18:14:12 -04:00
"doesn't onebox a link within a list");
2017-06-14 13:57:58 -04:00
assert.ok(matches("http://test.com", /onebox/), "adds a onebox class to a link on its own line");
assert.ok(matches("http://test.com\nhttp://test2.com", /onebox[\s\S]+onebox/m), "supports multiple links");
assert.ok(!matches("http://test.com bob", /onebox/), "doesn't onebox links that have trailing text");
2013-06-18 18:03:36 -04:00
2017-06-14 13:57:58 -04:00
assert.ok(!matches("[Tom Cruise](http://www.tomcruise.com/)", "onebox"), "Markdown links with labels are not oneboxed");
2017-07-14 08:27:28 -04:00
assert.ok(!matches("[http://www.tomcruise.com/](http://www.tomcruise.com/)",
2013-09-06 16:46:55 -04:00
"onebox"),
2017-07-14 08:27:28 -04:00
"Markdown links where the label is the same as the url but link is explicit");
2013-09-06 16:46:55 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("http://en.wikipedia.org/wiki/Homicide:_Life_on_the_Street",
2013-08-08 18:14:12 -04:00
"<p><a href=\"http://en.wikipedia.org/wiki/Homicide:_Life_on_the_Street\" class=\"onebox\"" +
2013-09-11 15:52:37 -04:00
" target=\"_blank\">http://en.wikipedia.org/wiki/Homicide:_Life_on_the_Street</a></p>",
2013-06-18 18:03:36 -04:00
"works with links that have underscores in them");
});
2017-06-14 13:57:58 -04:00
QUnit.test("links with full urls", assert => {
assert.cooked("[http://eviltrout.com][1] is a url\n\n[1]: http://eviltrout.com",
2013-11-04 14:24:40 -05:00
"<p><a href=\"http://eviltrout.com\">http://eviltrout.com</a> is a url</p>",
"it supports links that are full URLs");
});
2017-06-14 13:57:58 -04:00
QUnit.test("Code Blocks", assert => {
2013-08-08 18:14:12 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("<pre>\nhello\n</pre>\n",
2017-07-14 08:27:28 -04:00
"<pre>\nhello\n</pre>",
2014-06-23 15:21:07 -04:00
"pre blocks don't include extra lines");
2017-06-14 13:57:58 -04:00
assert.cooked("```\na\nb\nc\n\nd\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-auto\">a\nb\nc\n\nd\n</code></pre>",
2013-08-22 13:46:51 -04:00
"it treats new lines properly");
2017-06-14 13:57:58 -04:00
assert.cooked("```\ntest\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-auto\">test\n</code></pre>",
2013-08-08 18:14:12 -04:00
"it supports basic code blocks");
2017-06-14 13:57:58 -04:00
assert.cooked("```json\n{hello: 'world'}\n```\ntrailing",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-json\">{hello: 'world'}\n</code></pre>\n<p>trailing</p>",
2013-08-08 18:14:12 -04:00
"It does not truncate text after a code block.");
2017-06-14 13:57:58 -04:00
assert.cooked("```json\nline 1\n\nline 2\n\n\nline3\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-json\">line 1\n\nline 2\n\n\nline3\n</code></pre>",
2013-08-08 18:14:12 -04:00
"it maintains new lines inside a code block.");
2017-06-14 13:57:58 -04:00
assert.cooked("hello\nworld\n```json\nline 1\n\nline 2\n\n\nline3\n```",
2017-07-14 08:27:28 -04:00
"<p>hello<br>\nworld</p>\n<pre><code class=\"lang-json\">line 1\n\nline 2\n\n\nline3\n</code></pre>",
2013-08-08 18:14:12 -04:00
"it maintains new lines inside a code block with leading content.");
2017-06-14 13:57:58 -04:00
assert.cooked("```ruby\n<header>hello</header>\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-ruby\"><header>hello</header>\n</code></pre>",
2013-08-08 18:14:12 -04:00
"it escapes code in the code block");
2017-06-14 13:57:58 -04:00
assert.cooked("```text\ntext\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-nohighlight\">text\n</code></pre>",
2015-03-09 07:32:37 -04:00
"handles text by adding nohighlight");
2014-09-18 01:26:45 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("```ruby\n# cool\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-ruby\"># cool\n</code></pre>",
2013-08-08 18:14:12 -04:00
"it supports changing the language");
2017-06-14 13:57:58 -04:00
assert.cooked(" ```\n hello\n ```",
2017-07-14 08:27:28 -04:00
"<pre><code>```\nhello\n```</code></pre>",
2014-10-19 18:55:19 -04:00
"only detect ``` at the beginning of lines");
2013-08-24 13:24:27 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("```ruby\ndef self.parse(text)\n\n text\nend\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-ruby\">def self.parse(text)\n\n text\nend\n</code></pre>",
2013-08-24 13:24:27 -04:00
"it allows leading spaces on lines in a code block.");
2013-08-26 16:53:10 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("```ruby\nhello `eviltrout`\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-ruby\">hello `eviltrout`\n</code></pre>",
2013-08-26 16:53:10 -04:00
"it allows code with backticks in it");
2013-08-29 14:42:31 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("```eviltrout\nhello\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-auto\">hello\n</code></pre>",
2013-10-21 13:10:19 -04:00
"it doesn't not whitelist all classes");
2013-08-29 14:42:31 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("```\n[quote=\"sam, post:1, topic:9441, full:true\"]This is `<not>` a bug.[/quote]\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-auto\">[quote="sam, post:1, topic:9441, full:true"]This is `<not>` a bug.[/quote]\n</code></pre>",
2013-08-29 14:42:31 -04:00
"it allows code with backticks in it");
2017-06-14 13:57:58 -04:00
assert.cooked(" hello\n<blockquote>test</blockquote>",
2017-07-14 08:27:28 -04:00
"<pre><code>hello\n</code></pre>\n<blockquote>test</blockquote>",
2014-01-21 16:18:20 -05:00
"it allows an indented code block to by followed by a `<blockquote>`");
2014-08-13 19:58:01 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("``` foo bar ```",
2014-08-13 19:58:01 -04:00
"<p><code>foo bar</code></p>",
"it tolerates misuse of code block tags as inline code");
2014-08-26 08:11:23 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("```\nline1\n```\n```\nline2\n\nline3\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-auto\">line1\n</code></pre>\n<pre><code class=\"lang-auto\">line2\n\nline3\n</code></pre>",
2014-08-26 08:11:23 -04:00
"it does not consume next block's trailing newlines");
2014-09-10 06:59:21 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked(" <pre>test</pre>",
2014-09-10 06:59:21 -04:00
"<pre><code><pre>test</pre></code></pre>",
"it does not parse other block types in markdown code blocks");
2017-06-14 13:57:58 -04:00
assert.cooked(" [quote]test[/quote]",
2014-09-10 06:59:21 -04:00
"<pre><code>[quote]test[/quote]</code></pre>",
"it does not parse other block types in markdown code blocks");
2014-12-29 15:59:23 -05:00
2017-06-14 13:57:58 -04:00
assert.cooked("## a\nb\n```\nc\n```",
2017-07-14 08:27:28 -04:00
"<h2>a</h2>\n<p>b</p>\n<pre><code class=\"lang-auto\">c\n</code></pre>",
2015-03-09 07:32:37 -04:00
"it handles headings with code blocks after them.");
2013-08-08 18:14:12 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("URLs in BBCode tags", assert => {
2013-06-26 18:41:48 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("[img]http://eviltrout.com/eviltrout.png[/img][img]http://samsaffron.com/samsaffron.png[/img]",
2017-07-14 08:27:28 -04:00
"<p><img src=\"http://eviltrout.com/eviltrout.png\" alt/><img src=\"http://samsaffron.com/samsaffron.png\" alt/></p>",
2013-06-26 18:41:48 -04:00
"images are properly parsed");
2017-06-14 13:57:58 -04:00
assert.cooked("[url]http://discourse.org[/url]",
2017-07-14 08:27:28 -04:00
"<p><a href=\"http://discourse.org\" data-bbcode=\"true\">http://discourse.org</a></p>",
2013-06-26 18:41:48 -04:00
"links are properly parsed");
2017-06-14 13:57:58 -04:00
assert.cooked("[url=http://discourse.org]discourse[/url]",
2017-07-14 08:27:28 -04:00
"<p><a href=\"http://discourse.org\" data-bbcode=\"true\">discourse</a></p>",
2013-06-26 18:41:48 -04:00
"named links are properly parsed");
});
2013-10-11 16:24:27 -04:00
2017-06-14 13:57:58 -04:00
QUnit.test("images", assert => {
assert.cooked("[![folksy logo](http://folksy.com/images/folksy-colour.png)](http://folksy.com/)",
2014-02-20 15:23:10 -05:00
"<p><a href=\"http://folksy.com/\"><img src=\"http://folksy.com/images/folksy-colour.png\" alt=\"folksy logo\"/></a></p>",
"It allows images with links around them");
2014-07-18 14:38:01 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("<img src=\"data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAUAAAAFCAYAAACNbyblAAAAHElEQVQI12P4//8/w38GIAXDIBKE0DHxgljNBAAO9TXL0Y4OHwAAAABJRU5ErkJggg==\" alt=\"Red dot\">",
2017-07-17 18:06:11 -04:00
"<p><img src=\"data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAUAAAAFCAYAAACNbyblAAAAHElEQVQI12P4//8/w38GIAXDIBKE0DHxgljNBAAO9TXL0Y4OHwAAAABJRU5ErkJggg==\" alt=\"Red dot\"></p>",
2014-07-18 14:38:01 -04:00
"It allows data images");
2013-10-11 16:24:27 -04:00
});
2014-10-02 13:58:36 -04:00
2017-06-14 13:57:58 -04:00
QUnit.test("censoring", assert => {
assert.cooked("aw shucks, golly gee whiz.",
2017-07-14 08:27:28 -04:00
"<p>aw ■■■■■■, golly gee ■■■■.</p>",
2014-10-02 13:58:36 -04:00
"it censors words in the Site Settings");
2017-01-30 21:39:45 -05:00
2017-06-14 13:57:58 -04:00
assert.cooked("you are a whizzard! I love cheesewhiz. Whiz.",
2017-07-14 08:27:28 -04:00
"<p>you are a whizzard! I love cheesewhiz. ■■■■.</p>",
2014-10-09 16:35:27 -04:00
"it doesn't censor words unless they have boundaries.");
2017-01-30 21:39:45 -05:00
2017-06-14 13:57:58 -04:00
assert.cooked("you are a whizzer! I love cheesewhiz. Whiz.",
2017-07-14 08:27:28 -04:00
"<p>you are a ■■■■■■■! I love cheesewhiz. ■■■■.</p>",
2014-10-19 18:55:19 -04:00
"it censors words even if previous partial matches exist.");
2017-01-30 21:39:45 -05:00
2017-06-14 13:57:58 -04:00
assert.cooked("The link still works. [whiz](http://www.whiz.com)",
2017-07-14 08:27:28 -04:00
"<p>The link still works. <a href=\"http://www.whiz.com\">■■■■</a></p>",
2016-06-29 10:41:54 -04:00
"it won't break links by censoring them.");
2017-01-30 21:39:45 -05:00
2017-06-14 13:57:58 -04:00
assert.cooked("Call techapj the computer whiz at 555-555-1234 for free help.",
2018-02-26 15:48:59 -05:00
"<p>Call techapj the computer ■■■■ at 555-555-1234 for free help.</p>",
2016-11-08 16:36:34 -05:00
"uses both censored words and patterns from site settings");
2017-01-30 21:39:45 -05:00
2017-06-14 13:57:58 -04:00
assert.cooked("I have a pen, I have an a**le",
2017-07-14 08:27:28 -04:00
"<p>I have a pen, I have an ■■■■■</p>",
2017-01-30 21:39:45 -05:00
"it escapes regexp chars");
2018-01-09 15:51:47 -05:00
assert.cooked("No badword or apple here plz.",
"<p>No ■■■■■■■ or ■■■■■ here plz.</p>",
"it handles * as wildcard");
2018-01-10 14:11:14 -05:00
assert.cookedOptions(
"Pleased to meet you, but pleeeease call me later, xyz123",
{ siteSettings: {
2018-02-26 15:48:59 -05:00
watched_words_regular_expressions: true
2018-01-10 14:11:14 -05:00
},
censoredWords: 'xyz*|plee+ase'
},
2018-01-24 12:33:38 -05:00
"<p>Pleased to meet you, but ■■■■ call me later, ■■■■123</p>",
"supports words as regular expressions");
assert.cookedOptions(
"Meet downtown in your town at the townhouse on Main St.",
{ siteSettings: {
2018-02-26 15:48:59 -05:00
watched_words_regular_expressions: true
2018-01-24 12:33:38 -05:00
},
censoredWords: '\\btown\\b'
},
"<p>Meet downtown in your ■■■■ at the townhouse on Main St.</p>",
2018-01-10 14:11:14 -05:00
"supports words as regular expressions");
2014-10-02 13:58:36 -04:00
});
2015-03-12 11:20:14 -04:00
2017-06-14 13:57:58 -04:00
QUnit.test("code blocks/spans hoisting", assert => {
assert.cooked("```\n\n some code\n```",
2017-07-14 08:27:28 -04:00
"<pre><code class=\"lang-auto\">\n some code\n</code></pre>",
2015-03-12 11:20:14 -04:00
"it works when nesting standard markdown code blocks within a fenced code block");
2015-03-23 11:33:41 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("`$&`",
2015-03-23 11:33:41 -04:00
"<p><code>$&</code></p>",
"it works even when hoisting special replacement patterns");
2015-03-12 11:20:14 -04:00
});
2016-06-14 14:31:51 -04:00
2017-06-14 13:57:58 -04:00
QUnit.test('basic bbcode', assert => {
assert.cookedPara("[b]strong[/b]", "<span class=\"bbcode-b\">strong</span>", "bolds text");
assert.cookedPara("[i]emphasis[/i]", "<span class=\"bbcode-i\">emphasis</span>", "italics text");
assert.cookedPara("[u]underlined[/u]", "<span class=\"bbcode-u\">underlined</span>", "underlines text");
assert.cookedPara("[s]strikethrough[/s]", "<span class=\"bbcode-s\">strikethrough</span>", "strikes-through text");
2017-07-14 08:27:28 -04:00
assert.cookedPara("[img]http://eviltrout.com/eviltrout.png[/img]", "<img src=\"http://eviltrout.com/eviltrout.png\" alt>", "links images");
assert.cookedPara("[email]eviltrout@mailinator.com[/email]", "<a href=\"mailto:eviltrout@mailinator.com\" data-bbcode=\"true\">eviltrout@mailinator.com</a>", "supports [email] without a title");
2017-06-14 13:57:58 -04:00
assert.cookedPara("[b]evil [i]trout[/i][/b]",
2016-06-14 14:31:51 -04:00
"<span class=\"bbcode-b\">evil <span class=\"bbcode-i\">trout</span></span>",
"allows embedding of tags");
2017-07-14 08:27:28 -04:00
assert.cookedPara("[EMAIL]eviltrout@mailinator.com[/EMAIL]", "<a href=\"mailto:eviltrout@mailinator.com\" data-bbcode=\"true\">eviltrout@mailinator.com</a>", "supports upper case bbcode");
2017-06-14 13:57:58 -04:00
assert.cookedPara("[b]strong [b]stronger[/b][/b]", "<span class=\"bbcode-b\">strong <span class=\"bbcode-b\">stronger</span></span>", "accepts nested bbcode tags");
2016-06-14 14:31:51 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test('urls', assert => {
assert.cookedPara("[url]not a url[/url]", "not a url", "supports [url] that isn't a url");
2017-07-14 08:27:28 -04:00
assert.cookedPara("[url]abc.com[/url]", "<a href=\"http://abc.com\">abc.com</a>", "it magically links using linkify");
assert.cookedPara("[url]http://bettercallsaul.com[/url]", "<a href=\"http://bettercallsaul.com\" data-bbcode=\"true\">http://bettercallsaul.com</a>", "supports [url] without parameter");
assert.cookedPara("[url=http://example.com]example[/url]", "<a href=\"http://example.com\" data-bbcode=\"true\">example</a>", "supports [url] with given href");
2017-06-14 13:57:58 -04:00
assert.cookedPara("[url=http://www.example.com][img]http://example.com/logo.png[/img][/url]",
2017-07-14 08:27:28 -04:00
"<a href=\"http://www.example.com\" data-bbcode=\"true\"><img src=\"http://example.com/logo.png\" alt></a>",
2016-06-14 14:31:51 -04:00
"supports [url] with an embedded [img]");
});
2017-06-14 13:57:58 -04:00
QUnit.test('invalid bbcode', assert => {
2017-07-14 08:27:28 -04:00
assert.cooked("[code]I am not closed\n\nThis text exists.",
"<p>[code]I am not closed</p>\n<p>This text exists.</p>",
"does not raise an error with an open bbcode tag.");
2016-06-14 14:31:51 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test('code', assert => {
2017-07-14 08:27:28 -04:00
assert.cooked("[code]\nx++\n[/code]", "<pre><code class=\"lang-auto\">x++</code></pre>", "makes code into pre");
assert.cooked("[code]\nx++\ny++\nz++\n[/code]", "<pre><code class=\"lang-auto\">x++\ny++\nz++</code></pre>", "makes code into pre");
assert.cooked("[code]\nabc\n#def\n[/code]", '<pre><code class=\"lang-auto\">abc\n#def</code></pre>', 'it handles headings in a [code] block');
assert.cooked("[code]\n s\n[/code]",
2016-06-14 14:31:51 -04:00
"<pre><code class=\"lang-auto\"> s</code></pre>",
"it doesn't trim leading whitespace");
});
2017-06-14 13:57:58 -04:00
QUnit.test('tags with arguments', assert => {
2017-07-14 08:27:28 -04:00
assert.cookedPara("[url=http://bettercallsaul.com]better call![/url]", "<a href=\"http://bettercallsaul.com\" data-bbcode=\"true\">better call!</a>", "supports [url] with a title");
assert.cookedPara("[email=eviltrout@mailinator.com]evil trout[/email]", "<a href=\"mailto:eviltrout@mailinator.com\" data-bbcode=\"true\">evil trout</a>", "supports [email] with a title");
2017-06-14 13:57:58 -04:00
assert.cookedPara("[u][i]abc[/i][/u]", "<span class=\"bbcode-u\"><span class=\"bbcode-i\">abc</span></span>", "can nest tags");
assert.cookedPara("[b]first[/b] [b]second[/b]", "<span class=\"bbcode-b\">first</span> <span class=\"bbcode-b\">second</span>", "can bold two things on the same line");
2016-06-14 14:31:51 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("quotes", assert => {
2016-06-14 14:31:51 -04:00
const post = Post.create({
cooked: "<p><b>lorem</b> ipsum</p>",
username: "eviltrout",
post_number: 1,
topic_id: 2
});
function formatQuote(val, expected, text) {
2017-06-14 13:57:58 -04:00
assert.equal(Quote.build(post, val), expected, text);
2016-06-14 14:31:51 -04:00
};
formatQuote(undefined, "", "empty string for undefined content");
formatQuote(null, "", "empty string for null content");
formatQuote("", "", "empty string for empty string content");
formatQuote("lorem", "[quote=\"eviltrout, post:1, topic:2\"]\nlorem\n[/quote]\n\n", "correctly formats quotes");
formatQuote(" lorem \t ",
"[quote=\"eviltrout, post:1, topic:2\"]\nlorem\n[/quote]\n\n",
"trims white spaces before & after the quoted contents");
formatQuote("lorem ipsum",
"[quote=\"eviltrout, post:1, topic:2, full:true\"]\nlorem ipsum\n[/quote]\n\n",
"marks quotes as full when the quote is the full message");
formatQuote("**lorem** ipsum",
"[quote=\"eviltrout, post:1, topic:2, full:true\"]\n**lorem** ipsum\n[/quote]\n\n",
"keeps BBCode formatting");
2017-07-14 08:27:28 -04:00
assert.cooked("[quote]\ntest\n[/quote]",
2018-04-02 14:18:10 -04:00
"<aside class=\"quote no-group\">\n<blockquote>\n<p>test</p>\n</blockquote>\n</aside>",
2016-06-14 14:31:51 -04:00
"it supports quotes without params");
2017-07-14 08:27:28 -04:00
assert.cooked("[quote]\n*test*\n[/quote]",
2018-04-02 14:18:10 -04:00
"<aside class=\"quote no-group\">\n<blockquote>\n<p><em>test</em></p>\n</blockquote>\n</aside>",
2016-06-14 14:31:51 -04:00
"it doesn't insert a new line for italics");
2017-07-14 08:27:28 -04:00
assert.cooked("[quote=,script='a'><script>alert('test');//':a]\n[/quote]",
2018-04-02 14:18:10 -04:00
"<aside class=\"quote no-group\">\n<blockquote></blockquote>\n</aside>",
2016-06-14 14:31:51 -04:00
"It will not create a script tag within an attribute");
});
2017-06-14 13:57:58 -04:00
QUnit.test("quote formatting", assert => {
2016-06-14 14:31:51 -04:00
2017-07-14 08:27:28 -04:00
assert.cooked("[quote=\"EvilTrout, post:123, topic:456, full:true\"]\n[sam]\n[/quote]",
2018-04-02 14:18:10 -04:00
`<aside class=\"quote no-group\" data-post=\"123\" data-topic=\"456\" data-full=\"true\">
2017-07-14 08:27:28 -04:00
<div class=\"title\">
<div class=\"quote-controls\"></div>
EvilTrout:</div>
<blockquote>
<p>[sam]</p>
</blockquote>
</aside>`,
2016-06-14 14:31:51 -04:00
"it allows quotes with [] inside");
2017-07-14 08:27:28 -04:00
assert.cooked("[quote=\"eviltrout, post:1, topic:1\"]\nabc\n[/quote]",
2018-04-02 14:18:10 -04:00
`<aside class=\"quote no-group\" data-post=\"1\" data-topic=\"1\">
2017-07-14 08:27:28 -04:00
<div class=\"title\">
<div class=\"quote-controls\"></div>
eviltrout:</div>
<blockquote>
<p>abc</p>
</blockquote>
</aside>`,
2016-06-14 14:31:51 -04:00
"renders quotes properly");
2017-07-14 08:27:28 -04:00
assert.cooked("[quote=\"eviltrout, post:1, topic:1\"]\nabc\n[/quote]\nhello",
2018-04-02 14:18:10 -04:00
`<aside class=\"quote no-group\" data-post=\"1\" data-topic=\"1\">
2017-07-14 08:27:28 -04:00
<div class=\"title\">
<div class=\"quote-controls\"></div>
eviltrout:</div>
<blockquote>
<p>abc</p>
</blockquote>
</aside>
<p>hello</p>`,
2016-06-14 14:31:51 -04:00
"handles new lines properly");
2017-06-14 13:57:58 -04:00
assert.cooked("[quote=\"Alice, post:1, topic:1\"]\n[quote=\"Bob, post:2, topic:1\"]\n[/quote]\n[/quote]",
2018-04-02 14:18:10 -04:00
`<aside class=\"quote no-group\" data-post=\"1\" data-topic=\"1\">
2017-07-14 08:27:28 -04:00
<div class=\"title\">
<div class=\"quote-controls\"></div>
Alice:</div>
<blockquote>
2018-04-02 14:18:10 -04:00
<aside class=\"quote no-group\" data-post=\"2\" data-topic=\"1\">
2017-07-14 08:27:28 -04:00
<div class=\"title\">
<div class=\"quote-controls\"></div>
Bob:</div>
<blockquote></blockquote>
</aside>
</blockquote>
</aside>`,
2016-06-14 14:31:51 -04:00
"quotes can be nested");
2017-06-14 13:57:58 -04:00
assert.cooked("[quote=\"Alice, post:1, topic:1\"]\n[quote=\"Bob, post:2, topic:1\"]\n[/quote]",
2017-07-14 08:27:28 -04:00
`<p>[quote="Alice, post:1, topic:1"]</p>
2018-04-02 14:18:10 -04:00
<aside class=\"quote no-group\" data-post=\"2\" data-topic=\"1\">
2017-07-14 08:27:28 -04:00
<div class=\"title\">
<div class=\"quote-controls\"></div>
Bob:</div>
<blockquote></blockquote>
</aside>`,
"handles mismatched nested quote tags (non greedy)");
2016-06-14 14:31:51 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("[quote=\"Alice, post:1, topic:1\"]\n```javascript\nvar foo ='foo';\nvar bar = 'bar';\n```\n[/quote]",
2018-04-02 14:18:10 -04:00
`<aside class=\"quote no-group\" data-post=\"1\" data-topic=\"1\">
2017-07-14 08:27:28 -04:00
<div class=\"title\">
<div class=\"quote-controls\"></div>
Alice:</div>
<blockquote>
<pre><code class=\"lang-javascript\">var foo ='foo';
var bar = 'bar';
</code></pre>
</blockquote>
</aside>`,
2016-06-14 14:31:51 -04:00
"quotes can have code blocks without leading newline");
2017-07-14 08:27:28 -04:00
2017-06-14 13:57:58 -04:00
assert.cooked("[quote=\"Alice, post:1, topic:1\"]\n\n```javascript\nvar foo ='foo';\nvar bar = 'bar';\n```\n[/quote]",
2018-04-02 14:18:10 -04:00
`<aside class=\"quote no-group\" data-post=\"1\" data-topic=\"1\">
2017-07-14 08:27:28 -04:00
<div class=\"title\">
<div class=\"quote-controls\"></div>
Alice:</div>
<blockquote>
<pre><code class=\"lang-javascript\">var foo ='foo';
var bar = 'bar';
</code></pre>
</blockquote>
</aside>`,
2016-06-14 14:31:51 -04:00
"quotes can have code blocks with leading newline");
});
2017-06-14 13:57:58 -04:00
QUnit.test("quotes with trailing formatting", assert => {
2016-06-14 14:31:51 -04:00
const result = new PrettyText(defaultOpts).cook("[quote=\"EvilTrout, post:123, topic:456, full:true\"]\nhello\n[/quote]\n*Test*");
2017-06-14 13:57:58 -04:00
assert.equal(result,
2018-04-02 14:18:10 -04:00
`<aside class=\"quote no-group\" data-post=\"123\" data-topic=\"456\" data-full=\"true\">
2017-07-14 08:27:28 -04:00
<div class=\"title\">
<div class=\"quote-controls\"></div>
EvilTrout:</div>
<blockquote>
<p>hello</p>
</blockquote>
</aside>
<p><em>Test</em></p>`,
2016-06-14 14:31:51 -04:00
"it allows trailing formatting");
});
2017-06-14 13:57:58 -04:00
QUnit.test("enable/disable features", assert => {
2016-06-14 14:31:51 -04:00
2017-07-14 08:27:28 -04:00
assert.cookedOptions('|a|\n--\n|a|', { features: {table: false} }, '');
assert.cooked('|a|\n--\n|a|',
2017-11-13 02:09:24 -05:00
`<div class="md-table">
<table>
2017-07-14 08:27:28 -04:00
<thead>
<tr>
<th>a</th>
</tr>
</thead>
<tbody>
<tr>
<td>a</td>
</tr>
</tbody>
2017-11-13 02:09:24 -05:00
</table>
</div>`);
2016-06-14 14:31:51 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("emoji", assert => {
assert.cooked(":smile:", `<p><img src="/images/emoji/emoji_one/smile.png?v=${v}" title=":smile:" class="emoji" alt=":smile:"></p>`);
assert.cooked(":(", `<p><img src="/images/emoji/emoji_one/frowning.png?v=${v}" title=":frowning:" class="emoji" alt=":frowning:"></p>`);
assert.cooked("8-)", `<p><img src="/images/emoji/emoji_one/sunglasses.png?v=${v}" title=":sunglasses:" class="emoji" alt=":sunglasses:"></p>`);
2016-06-14 14:31:51 -04:00
});
2017-06-14 13:57:58 -04:00
QUnit.test("emoji - emojiSet", assert => {
assert.cookedOptions(":smile:",
2017-07-14 08:27:28 -04:00
{ siteSettings : { emoji_set: 'twitter' }},
2016-06-14 14:31:51 -04:00
`<p><img src="/images/emoji/twitter/smile.png?v=${v}" title=":smile:" class="emoji" alt=":smile:"></p>`);
});