From 6ff7c0d8ff2f54b83b6b7ad4c51497be830de05f Mon Sep 17 00:00:00 2001 From: aisha kh Date: Fri, 15 Sep 2023 13:44:07 +0300 Subject: [PATCH 01/27] Created the IF statement at line 51 to make user anonymous --- install/data/footer.json | 2 +- src/posts/create.js | 7 +++++++ 2 files changed, 8 insertions(+), 1 deletion(-) diff --git a/install/data/footer.json b/install/data/footer.json index 53b2176a..2479c373 100644 --- a/install/data/footer.json +++ b/install/data/footer.json @@ -2,7 +2,7 @@ { "widget": "html", "data" : { - "html": "", + "html": "", "title":"", "container":"" } diff --git a/src/posts/create.js b/src/posts/create.js index 094ae1c6..73ebd1f4 100644 --- a/src/posts/create.js +++ b/src/posts/create.js @@ -47,6 +47,13 @@ module.exports = function (Posts) { postData.handle = data.handle; } + // Adding the Anonymous feature + if (data.postVisibility === "anonymous"){ + postData.uid = null; + postData.handle = "Anonymous" + postData.isAnonymous = true; + } + let result = await plugins.hooks.fire('filter:post.create', { post: postData, data: data }); postData = result.post; await db.setObject(`post:${postData.pid}`, postData); From 912cd68beed31038ec2b053a5e313c50c639b526 Mon Sep 17 00:00:00 2001 From: aisha kh Date: Fri, 15 Sep 2023 13:45:19 +0300 Subject: [PATCH 02/27] Created the ELSE statement at line 56 to make user visible if they dont choose to be anonymous --- src/posts/create.js | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/posts/create.js b/src/posts/create.js index 73ebd1f4..6a95d0ca 100644 --- a/src/posts/create.js +++ b/src/posts/create.js @@ -52,6 +52,8 @@ module.exports = function (Posts) { postData.uid = null; postData.handle = "Anonymous" postData.isAnonymous = true; + } else { + postData.isAnonymous = false; } let result = await plugins.hooks.fire('filter:post.create', { post: postData, data: data }); From a4d7b8eae22e9d6dd4e0e9692d267b81d5ddbb54 Mon Sep 17 00:00:00 2001 From: aisha kh Date: Fri, 15 Sep 2023 15:31:49 +0300 Subject: [PATCH 03/27] Added the visibility dropdown in the post.tpl as a test for visual purposes (line 52 - line 60) --- src/posts/create.js | 2 +- .../templates/partials/topic/post.tpl | 9 +++++++++ 2 files changed, 10 insertions(+), 1 deletion(-) diff --git a/src/posts/create.js b/src/posts/create.js index 6a95d0ca..feb4fb0d 100644 --- a/src/posts/create.js +++ b/src/posts/create.js @@ -47,7 +47,7 @@ module.exports = function (Posts) { postData.handle = data.handle; } - // Adding the Anonymous feature + // Adding and checking if Anonymous feature is enabled if (data.postVisibility === "anonymous"){ postData.uid = null; postData.handle = "Anonymous" diff --git a/themes/nodebb-theme-persona/templates/partials/topic/post.tpl b/themes/nodebb-theme-persona/templates/partials/topic/post.tpl index aeb745c1..62f9ddc3 100644 --- a/themes/nodebb-theme-persona/templates/partials/topic/post.tpl +++ b/themes/nodebb-theme-persona/templates/partials/topic/post.tpl @@ -50,6 +50,15 @@
+
+ + +
+
+ {posts.content}
From 26efcacdb362511cde45e04b501974def956d9c1 Mon Sep 17 00:00:00 2001 From: aisha kh Date: Fri, 15 Sep 2023 19:39:08 +0300 Subject: [PATCH 04/27] Removed visibility dropdown from post.tpl because it is in the wrong place --- .../templates/partials/topic/post.tpl | 9 --------- 1 file changed, 9 deletions(-) diff --git a/themes/nodebb-theme-persona/templates/partials/topic/post.tpl b/themes/nodebb-theme-persona/templates/partials/topic/post.tpl index 62f9ddc3..aeb745c1 100644 --- a/themes/nodebb-theme-persona/templates/partials/topic/post.tpl +++ b/themes/nodebb-theme-persona/templates/partials/topic/post.tpl @@ -50,15 +50,6 @@
-
- - -
-
- {posts.content}
From 7a30223606ed438f7a8686a4a92a4b7cf9641934 Mon Sep 17 00:00:00 2001 From: aisha kh Date: Sun, 17 Sep 2023 15:28:46 +0300 Subject: [PATCH 05/27] Removed anonymous feature in create.js --- src/posts/create.js | 9 --------- 1 file changed, 9 deletions(-) diff --git a/src/posts/create.js b/src/posts/create.js index feb4fb0d..094ae1c6 100644 --- a/src/posts/create.js +++ b/src/posts/create.js @@ -47,15 +47,6 @@ module.exports = function (Posts) { postData.handle = data.handle; } - // Adding and checking if Anonymous feature is enabled - if (data.postVisibility === "anonymous"){ - postData.uid = null; - postData.handle = "Anonymous" - postData.isAnonymous = true; - } else { - postData.isAnonymous = false; - } - let result = await plugins.hooks.fire('filter:post.create', { post: postData, data: data }); postData = result.post; await db.setObject(`post:${postData.pid}`, postData); From 380e2d8fa8d0b5de19e2dd3a20579bafa4cb128d Mon Sep 17 00:00:00 2001 From: aisha kh Date: Sun, 17 Sep 2023 15:40:40 +0300 Subject: [PATCH 06/27] Changed .gitignore and added composer.tpl to be ignored + Added dropdown feature line 96 in composer.tpl --- .gitignore | 1 + 1 file changed, 1 insertion(+) diff --git a/.gitignore b/.gitignore index 5a176ec4..1cd00afd 100644 --- a/.gitignore +++ b/.gitignore @@ -2,6 +2,7 @@ dist/ yarn.lock npm-debug.log node_modules/ +!node_modules/nodebb-plugin-composer-default/static/templates/composer.tpl sftp-config.json config.json jsconfig.json From e614d70d90b31201e4cef7b43fe6565b5d64354f Mon Sep 17 00:00:00 2001 From: aisha kh Date: Sun, 17 Sep 2023 17:20:52 +0300 Subject: [PATCH 07/27] Re-added the postVisibility statement in create.js line 52 --- src/posts/create.js | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/src/posts/create.js b/src/posts/create.js index 094ae1c6..bb91893d 100644 --- a/src/posts/create.js +++ b/src/posts/create.js @@ -47,6 +47,16 @@ module.exports = function (Posts) { postData.handle = data.handle; } + // Adding and checking if Anonymous feature is enabled + // making anonymous users' uid -4 + if (data.postVisibility === "anonymous"){ + postData.uid = -4; + postData.handle = "Anonymous" + postData.isAnonymous = true; + } else { + postData.isAnonymous = false; + } + let result = await plugins.hooks.fire('filter:post.create', { post: postData, data: data }); postData = result.post; await db.setObject(`post:${postData.pid}`, postData); From c9422c8571bd34e94075f80017a4de1b30e22b0c Mon Sep 17 00:00:00 2001 From: Gulnaz Serikbay Date: Thu, 28 Sep 2023 17:57:53 +0300 Subject: [PATCH 08/27] some backend support --- src/posts/create.js | 10 +++------- src/topics/create.js | 1 + src/topics/index.js | 5 +++++ src/topics/posts.js | 15 +++++++++++++-- 4 files changed, 22 insertions(+), 9 deletions(-) diff --git a/src/posts/create.js b/src/posts/create.js index bb91893d..a80cfd9b 100644 --- a/src/posts/create.js +++ b/src/posts/create.js @@ -19,6 +19,8 @@ module.exports = function (Posts) { const content = data.content.toString(); const timestamp = data.timestamp || Date.now(); const isMain = data.isMain || false; + const { postType } = data; + if (!uid && parseInt(uid, 10) !== 0) { throw new Error('[[error:invalid-uid]]'); @@ -35,6 +37,7 @@ module.exports = function (Posts) { tid: tid, content: content, timestamp: timestamp, + postType: postType, }; if (data.toPid) { @@ -49,13 +52,6 @@ module.exports = function (Posts) { // Adding and checking if Anonymous feature is enabled // making anonymous users' uid -4 - if (data.postVisibility === "anonymous"){ - postData.uid = -4; - postData.handle = "Anonymous" - postData.isAnonymous = true; - } else { - postData.isAnonymous = false; - } let result = await plugins.hooks.fire('filter:post.create', { post: postData, data: data }); postData = result.post; diff --git a/src/topics/create.js b/src/topics/create.js index c366c219..d37a2476 100644 --- a/src/topics/create.js +++ b/src/topics/create.js @@ -33,6 +33,7 @@ module.exports = function (Topics) { lastposttime: 0, postcount: 0, viewcount: 0, + postType: data.postType, }; if (Array.isArray(data.tags) && data.tags.length) { diff --git a/src/topics/index.js b/src/topics/index.js index 0287cb62..71df5515 100644 --- a/src/topics/index.js +++ b/src/topics/index.js @@ -213,7 +213,12 @@ Topics.getTopicWithPosts = async function (topicData, set, uid, start, stop, rev topicData.unreplied = topicData.postcount === 1; topicData.icons = []; + const result = await plugins.hooks.fire('filter:topic.get', { topic: topicData, uid: uid }); + // hide the user identity + if (topicData.postType === 'anon') { + topicData.uid = 0; + } return result.topic; }; diff --git a/src/topics/posts.js b/src/topics/posts.js index d045d568..cefe5c2f 100644 --- a/src/topics/posts.js +++ b/src/topics/posts.js @@ -140,6 +140,17 @@ module.exports = function (Topics) { postObj.replies = replies[i]; postObj.selfPost = parseInt(uid, 10) > 0 && parseInt(uid, 10) === postObj.uid; + // allow instructors to see the anon + if (postObj.postType === 'anon' && !postObj.selfPost) { + postObj.uid = 0; + postObj.user = { + username: 'Anonymous', + anon: true, + displayname: 'Anonymous', + }; + postObj.postTypeAnon = true; + } + // Username override for guests, if enabled if (meta.config.allowGuestHandles && postObj.uid === 0 && postObj.handle) { postObj.user.username = validator.escape(String(postObj.handle)); @@ -321,7 +332,7 @@ module.exports = function (Topics) { const uniquePids = _.uniq(_.flatten(arrayOfReplyPids)); - let replyData = await posts.getPostsFields(uniquePids, ['pid', 'uid', 'timestamp']); + let replyData = await posts.getPostsFields(uniquePids, ['pid', 'uid', 'timestamp', 'postType']); const result = await plugins.hooks.fire('filter:topics.getPostReplies', { uid: callerUid, replies: replyData, @@ -352,7 +363,7 @@ module.exports = function (Topics) { replyPids.forEach((replyPid) => { const replyData = pidMap[replyPid]; - if (!uidsUsed[replyData.uid] && currentData.users.length < 6) { + if (!uidsUsed[replyData.uid] && currentData.users.length < 6 && replyData.postType !== 'anon') { currentData.users.push(uidMap[replyData.uid]); uidsUsed[replyData.uid] = true; } From 57c1eb3b38011a85d1e94a20f1d46f390d04e571 Mon Sep 17 00:00:00 2001 From: Gulnaz Serikbay Date: Sat, 30 Sep 2023 01:28:55 +0300 Subject: [PATCH 09/27] added working backend and frontend for anonymous posting --- src/categories/topics.js | 9 +++++++++ themes/nodebb-theme-persona/less/topic.less | 5 +++++ themes/nodebb-theme-persona/templates/topic.tpl | 2 +- 3 files changed, 15 insertions(+), 1 deletion(-) diff --git a/src/categories/topics.js b/src/categories/topics.js index 00addbe0..aa732161 100644 --- a/src/categories/topics.js +++ b/src/categories/topics.js @@ -160,6 +160,15 @@ module.exports = function (Categories) { } topics.forEach((topic) => { + // set user credentials to Anonymous + if (topic.postType === 'anon') { + topic.uid = 0; + topic.user = { + username: 'Anonymous', + anon: true, + displayname: 'Anonymous', + }; + } if (!topic.scheduled && topic.deleted && !topic.isOwner) { topic.title = '[[topic:topic_is_deleted]]'; if (topic.hasOwnProperty('titleRaw')) { diff --git a/themes/nodebb-theme-persona/less/topic.less b/themes/nodebb-theme-persona/less/topic.less index b41ab690..5a946a55 100644 --- a/themes/nodebb-theme-persona/less/topic.less +++ b/themes/nodebb-theme-persona/less/topic.less @@ -17,6 +17,10 @@ line-height: 24px; margin-top: 0px; } + + } + .icon-public{ + opacity: 0%; } > span { @@ -29,6 +33,7 @@ .topic-title { text-transform: initial; } + } [component="topic/labels"] { diff --git a/themes/nodebb-theme-persona/templates/topic.tpl b/themes/nodebb-theme-persona/templates/topic.tpl index 44c66c2c..8712b220 100644 --- a/themes/nodebb-theme-persona/templates/topic.tpl +++ b/themes/nodebb-theme-persona/templates/topic.tpl @@ -16,7 +16,7 @@ {{{each icons}}}{@value}{{{end}}} {title} - + Anonymous Post
From 7def5522de816318f4381428eadf43b15ffc2577 Mon Sep 17 00:00:00 2001 From: Gulnaz Serikbay Date: Sat, 30 Sep 2023 01:32:53 +0300 Subject: [PATCH 10/27] changed configs and frontend in nodebb-plugin-composer --- .gitignore | 4 +- .../nodebb-plugin-composer-default/.eslintrc | 3 + .../.gitattributes | 22 + .../nodebb-plugin-composer-default/.jshintrc | 86 ++ .../nodebb-plugin-composer-default/LICENSE | 7 + .../nodebb-plugin-composer-default/README.md | 11 + .../controllers.js | 9 + .../nodebb-plugin-composer-default/library.js | 330 +++++++ .../package.json | 43 + .../plugin.json | 34 + .../screenshots/desktop.png | Bin 0 -> 17071 bytes .../screenshots/mobile.png | Bin 0 -> 7693 bytes .../static/less/composer.less | 573 ++++++++++++ .../static/less/medium.less | 66 ++ .../static/less/page-compose.less | 24 + .../static/less/textcomplete.less | 24 + .../static/less/zen-mode.less | 55 ++ .../static/lib/.eslintrc | 6 + .../static/lib/admin.js | 25 + .../static/lib/client.js | 71 ++ .../static/lib/composer.js | 877 ++++++++++++++++++ .../static/lib/composer/autocomplete.js | 97 ++ .../static/lib/composer/categoryList.js | 113 +++ .../static/lib/composer/controls.js | 145 +++ .../static/lib/composer/drafts.js | 309 ++++++ .../static/lib/composer/formatting.js | 116 +++ .../static/lib/composer/preview.js | 106 +++ .../static/lib/composer/resize.js | 194 ++++ .../static/lib/composer/scheduler.js | 169 ++++ .../static/lib/composer/tags.js | 226 +++++ .../static/lib/composer/uploads.js | 266 ++++++ .../admin/plugins/composer-default.tpl | 17 + .../static/templates/compose.tpl | 126 +++ .../static/templates/composer.tpl | 145 +++ .../templates/modals/topic-scheduler.tpl | 4 + .../websockets.js | 70 ++ 36 files changed, 4371 insertions(+), 2 deletions(-) create mode 100644 node_modules/nodebb-plugin-composer-default/.eslintrc create mode 100644 node_modules/nodebb-plugin-composer-default/.gitattributes create mode 100644 node_modules/nodebb-plugin-composer-default/.jshintrc create mode 100644 node_modules/nodebb-plugin-composer-default/LICENSE create mode 100644 node_modules/nodebb-plugin-composer-default/README.md create mode 100644 node_modules/nodebb-plugin-composer-default/controllers.js create mode 100644 node_modules/nodebb-plugin-composer-default/library.js create mode 100644 node_modules/nodebb-plugin-composer-default/package.json create mode 100644 node_modules/nodebb-plugin-composer-default/plugin.json create mode 100644 node_modules/nodebb-plugin-composer-default/screenshots/desktop.png create mode 100644 node_modules/nodebb-plugin-composer-default/screenshots/mobile.png create mode 100644 node_modules/nodebb-plugin-composer-default/static/less/composer.less create mode 100644 node_modules/nodebb-plugin-composer-default/static/less/medium.less create mode 100644 node_modules/nodebb-plugin-composer-default/static/less/page-compose.less create mode 100644 node_modules/nodebb-plugin-composer-default/static/less/textcomplete.less create mode 100644 node_modules/nodebb-plugin-composer-default/static/less/zen-mode.less create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/.eslintrc create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/admin.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/client.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/composer.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/composer/autocomplete.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/composer/categoryList.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/composer/controls.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/composer/drafts.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/composer/formatting.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/composer/preview.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/composer/resize.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/composer/scheduler.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/composer/tags.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/lib/composer/uploads.js create mode 100644 node_modules/nodebb-plugin-composer-default/static/templates/admin/plugins/composer-default.tpl create mode 100644 node_modules/nodebb-plugin-composer-default/static/templates/compose.tpl create mode 100644 node_modules/nodebb-plugin-composer-default/static/templates/composer.tpl create mode 100644 node_modules/nodebb-plugin-composer-default/static/templates/modals/topic-scheduler.tpl create mode 100644 node_modules/nodebb-plugin-composer-default/websockets.js diff --git a/.gitignore b/.gitignore index 1cd00afd..1200b713 100644 --- a/.gitignore +++ b/.gitignore @@ -1,8 +1,8 @@ dist/ yarn.lock npm-debug.log -node_modules/ -!node_modules/nodebb-plugin-composer-default/static/templates/composer.tpl +node_modules/* +!node_modules/nodebb-plugin-composer-default sftp-config.json config.json jsconfig.json diff --git a/node_modules/nodebb-plugin-composer-default/.eslintrc b/node_modules/nodebb-plugin-composer-default/.eslintrc new file mode 100644 index 00000000..74e8dc06 --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/.eslintrc @@ -0,0 +1,3 @@ +{ + "extends": "nodebb/lib" +} \ No newline at end of file diff --git a/node_modules/nodebb-plugin-composer-default/.gitattributes b/node_modules/nodebb-plugin-composer-default/.gitattributes new file mode 100644 index 00000000..412eeda7 --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/.gitattributes @@ -0,0 +1,22 @@ +# Auto detect text files and perform LF normalization +* text=auto + +# Custom for Visual Studio +*.cs diff=csharp +*.sln merge=union +*.csproj merge=union +*.vbproj merge=union +*.fsproj merge=union +*.dbproj merge=union + +# Standard to msysgit +*.doc diff=astextplain +*.DOC diff=astextplain +*.docx diff=astextplain +*.DOCX diff=astextplain +*.dot diff=astextplain +*.DOT diff=astextplain +*.pdf diff=astextplain +*.PDF diff=astextplain +*.rtf diff=astextplain +*.RTF diff=astextplain diff --git a/node_modules/nodebb-plugin-composer-default/.jshintrc b/node_modules/nodebb-plugin-composer-default/.jshintrc new file mode 100644 index 00000000..1981c254 --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/.jshintrc @@ -0,0 +1,86 @@ +{ + // JSHint Default Configuration File (as on JSHint website) + // See http://jshint.com/docs/ for more details + + "maxerr" : 50, // {int} Maximum error before stopping + + // Enforcing + "bitwise" : true, // true: Prohibit bitwise operators (&, |, ^, etc.) + "camelcase" : false, // true: Identifiers must be in camelCase + "curly" : true, // true: Require {} for every new block or scope + "eqeqeq" : true, // true: Require triple equals (===) for comparison + "forin" : true, // true: Require filtering for..in loops with obj.hasOwnProperty() + "immed" : false, // true: Require immediate invocations to be wrapped in parens e.g. `(function () { } ());` + "indent" : 4, // {int} Number of spaces to use for indentation + "latedef" : false, // true: Require variables/functions to be defined before being used + "newcap" : false, // true: Require capitalization of all constructor functions e.g. `new F()` + "noarg" : true, // true: Prohibit use of `arguments.caller` and `arguments.callee` + "noempty" : true, // true: Prohibit use of empty blocks + "nonew" : false, // true: Prohibit use of constructors for side-effects (without assignment) + "plusplus" : false, // true: Prohibit use of `++` & `--` + "quotmark" : false, // Quotation mark consistency: + // false : do nothing (default) + // true : ensure whatever is used is consistent + // "single" : require single quotes + // "double" : require double quotes + "undef" : true, // true: Require all non-global variables to be declared (prevents global leaks) + "unused" : true, // true: Require all defined variables be used + "strict" : true, // true: Requires all functions run in ES5 Strict Mode + "trailing" : false, // true: Prohibit trailing whitespaces + "maxparams" : false, // {int} Max number of formal params allowed per function + "maxdepth" : false, // {int} Max depth of nested blocks (within functions) + "maxstatements" : false, // {int} Max number statements per function + "maxcomplexity" : false, // {int} Max cyclomatic complexity per function + "maxlen" : false, // {int} Max number of characters per line + + // Relaxing + "asi" : false, // true: Tolerate Automatic Semicolon Insertion (no semicolons) + "boss" : false, // true: Tolerate assignments where comparisons would be expected + "debug" : false, // true: Allow debugger statements e.g. browser breakpoints. + "eqnull" : false, // true: Tolerate use of `== null` + "es5" : false, // true: Allow ES5 syntax (ex: getters and setters) + "esnext" : false, // true: Allow ES.next (ES6) syntax (ex: `const`) + "moz" : false, // true: Allow Mozilla specific syntax (extends and overrides esnext features) + // (ex: `for each`, multiple try/catch, function expression…) + "evil" : false, // true: Tolerate use of `eval` and `new Function()` + "expr" : false, // true: Tolerate `ExpressionStatement` as Programs + "funcscope" : false, // true: Tolerate defining variables inside control statements" + "globalstrict" : false, // true: Allow global "use strict" (also enables 'strict') + "iterator" : false, // true: Tolerate using the `__iterator__` property + "lastsemic" : false, // true: Tolerate omitting a semicolon for the last statement of a 1-line block + "laxbreak" : false, // true: Tolerate possibly unsafe line breakings + "laxcomma" : false, // true: Tolerate comma-first style coding + "loopfunc" : false, // true: Tolerate functions being defined in loops + "multistr" : false, // true: Tolerate multi-line strings + "proto" : false, // true: Tolerate using the `__proto__` property + "scripturl" : false, // true: Tolerate script-targeted URLs + "smarttabs" : false, // true: Tolerate mixed tabs/spaces when used for alignment + "shadow" : false, // true: Allows re-define variables later in code e.g. `var x=1; x=2;` + "sub" : false, // true: Tolerate using `[]` notation when it can still be expressed in dot notation + "supernew" : false, // true: Tolerate `new function () { ... };` and `new Object;` + "validthis" : false, // true: Tolerate using this in a non-constructor function + + // Environments + "browser" : true, // Web Browser (window, document, etc) + "couch" : false, // CouchDB + "devel" : true, // Development/debugging (alert, confirm, etc) + "dojo" : false, // Dojo Toolkit + "jquery" : true, // jQuery + "mootools" : false, // MooTools + "node" : true, // Node.js + "nonstandard" : false, // Widely adopted globals (escape, unescape, etc) + "prototypejs" : false, // Prototype and Scriptaculous + "rhino" : false, // Rhino + "worker" : false, // Web Workers + "wsh" : false, // Windows Scripting Host + "yui" : false, // Yahoo User Interface + + // Legacy + "nomen" : false, // true: Prohibit dangling `_` in variables + "onevar" : false, // true: Allow only one `var` statement per function + "passfail" : false, // true: Stop on first error + "white" : false, // true: Check against strict whitespace and indentation rules + + // Custom Globals + "globals" : {} // additional predefined global variables +} \ No newline at end of file diff --git a/node_modules/nodebb-plugin-composer-default/LICENSE b/node_modules/nodebb-plugin-composer-default/LICENSE new file mode 100644 index 00000000..b8658d3a --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/LICENSE @@ -0,0 +1,7 @@ +Copyright (c) 2016 NodeBB Inc. + +Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. \ No newline at end of file diff --git a/node_modules/nodebb-plugin-composer-default/README.md b/node_modules/nodebb-plugin-composer-default/README.md new file mode 100644 index 00000000..7bcfff9a --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/README.md @@ -0,0 +1,11 @@ +# Default Composer for NodeBB + +This plugin activates the default composer for NodeBB. It is activated by default, but can be swapped out as necessary. + +## Screenshots + +### Desktop +![Desktop Composer](screenshots/desktop.png?raw=true) + +### Mobile Devices +![Mobile Composer](screenshots/mobile.png?raw=true) \ No newline at end of file diff --git a/node_modules/nodebb-plugin-composer-default/controllers.js b/node_modules/nodebb-plugin-composer-default/controllers.js new file mode 100644 index 00000000..ef76d10c --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/controllers.js @@ -0,0 +1,9 @@ +'use strict'; + +const Controllers = {}; + +Controllers.renderAdminPage = function (req, res) { + res.render('admin/plugins/composer-default', {}); +}; + +module.exports = Controllers; diff --git a/node_modules/nodebb-plugin-composer-default/library.js b/node_modules/nodebb-plugin-composer-default/library.js new file mode 100644 index 00000000..a635ee11 --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/library.js @@ -0,0 +1,330 @@ +'use strict'; + +const url = require('url'); + +const nconf = require.main.require('nconf'); +const winston = require.main.require('winston'); +const validator = require('validator'); + +const plugins = require.main.require('./src/plugins'); +const topics = require.main.require('./src/topics'); +const categories = require.main.require('./src/categories'); +const posts = require.main.require('./src/posts'); +const user = require.main.require('./src/user'); +const meta = require.main.require('./src/meta'); +const privileges = require.main.require('./src/privileges'); +const translator = require.main.require('./src/translator'); +const helpers = require.main.require('./src/controllers/helpers'); +const SocketPlugins = require.main.require('./src/socket.io/plugins'); +const socketMethods = require('./websockets'); + +const plugin = module.exports; + +plugin.socketMethods = socketMethods; + +plugin.init = async function (data) { + const controllers = require('./controllers'); + SocketPlugins.composer = socketMethods; + + data.router.get('/admin/plugins/composer-default', data.middleware.admin.buildHeader, controllers.renderAdminPage); + data.router.get('/api/admin/plugins/composer-default', controllers.renderAdminPage); +}; + +plugin.appendConfig = async function (config) { + config['composer-default'] = await meta.settings.get('composer-default'); + return config; +}; + +plugin.addAdminNavigation = async function (header) { + header.plugins.push({ + route: '/plugins/composer-default', + icon: 'fa-edit', + name: 'Composer (Default)', + }); + return header; +}; + +plugin.addPrefetchTags = async function (hookData) { + const prefetch = [ + '/assets/src/modules/composer.js', '/assets/src/modules/composer/uploads.js', '/assets/src/modules/composer/drafts.js', + '/assets/src/modules/composer/tags.js', '/assets/src/modules/composer/categoryList.js', '/assets/src/modules/composer/resize.js', + '/assets/src/modules/composer/autocomplete.js', '/assets/templates/composer.tpl', + `/assets/language/${meta.config.defaultLang || 'en-GB'}/topic.json`, + `/assets/language/${meta.config.defaultLang || 'en-GB'}/modules.json`, + `/assets/language/${meta.config.defaultLang || 'en-GB'}/tags.json`, + ]; + + hookData.links = hookData.links.concat(prefetch.map(path => ({ + rel: 'prefetch', + href: `${nconf.get('relative_path') + path}?${meta.config['cache-buster']}`, + }))); + + return hookData; +}; + +plugin.getFormattingOptions = async function () { + const defaultVisibility = { + mobile: true, + desktop: true, + + // op or reply + main: true, + reply: true, + }; + let payload = { + defaultVisibility, + options: [ + { + name: 'tags', + title: '[[global:tags.tags]]', + className: 'fa fa-tags', + visibility: { + ...defaultVisibility, + desktop: false, + }, + }, + { + name: 'zen', + title: '[[modules:composer.zen_mode]]', + className: 'fa fa-arrows-alt', + visibility: defaultVisibility, + }, + ], + }; + if (parseInt(meta.config.allowTopicsThumbnail, 10) === 1) { + payload.options.push({ + name: 'thumbs', + title: '[[topic:composer.thumb_title]]', + className: 'fa fa-address-card-o', + visibility: { + ...defaultVisibility, + reply: false, + }, + }); + } + + payload = await plugins.hooks.fire('filter:composer.formatting', payload); + + // TODO: Backwards compatibility -- remove in v1.16.0 + payload.options = payload.options.map((option) => { + option.visibility = { + ...defaultVisibility, + ...option.visibility || {}, + }; + if (option.hasOwnProperty('mobile')) { + winston.warn('[composer/formatting] `mobile` is no longer supported as a formatting option, use `visibility` instead (default values are passed in payload)'); + option.visibility.mobile = option.mobile; + option.visibility.desktop = !option.mobile; + } + + return option; + }); + // end + + return payload ? payload.options : null; +}; + +plugin.filterComposerBuild = async function (hookData) { + const { req } = hookData; + const { res } = hookData; + + if (req.query.p) { + if (!res.locals.isAPI) { + let a; + try { + a = url.parse(req.query.p, true, true); + } catch (e) { + return helpers.redirect(res, '/'); + } + return helpers.redirect(res, `/${(a.path || '').replace(/^\/*/, '')}`); + } + res.render('', {}); + return; + } else if (!req.query.pid && !req.query.tid && !req.query.cid) { + return helpers.redirect(res, '/'); + } + const [ + isMainPost, + postData, + topicData, + categoryData, + isAdmin, + isMod, + formatting, + tagWhitelist, + globalPrivileges, + canTagTopics, + canScheduleTopics, + ] = await Promise.all([ + posts.isMain(req.query.pid), + getPostData(req), + getTopicData(req), + categories.getCategoryFields(req.query.cid, [ + 'name', 'icon', 'color', 'bgColor', 'backgroundImage', 'imageClass', 'minTags', 'maxTags', + ]), + user.isAdministrator(req.uid), + isModerator(req), + plugin.getFormattingOptions(), + getTagWhitelist(req.query, req.uid), + privileges.global.get(req.uid), + canTag(req), + canSchedule(req), + ]); + + const isEditing = !!req.query.pid; + const isGuestPost = postData && parseInt(postData.uid, 10) === 0; + const save_id = generateSaveId(req); + const discardRoute = generateDiscardRoute(req, topicData); + const body = await generateBody(req, postData); + + let action = 'topics.post'; + let isMain = isMainPost; + if (req.query.tid) { + action = 'posts.reply'; + } else if (req.query.pid) { + action = 'posts.edit'; + } else { + isMain = true; + } + globalPrivileges['topics:tag'] = canTagTopics; + const cid = parseInt(req.query.cid, 10); + const topicTitle = topicData && topicData.title ? topicData.title.replace(/%/g, '%').replace(/,/g, ',') : validator.escape(String(req.query.title || '')); + return { + req: req, + res: res, + templateData: { + disabled: !req.query.pid && !req.query.tid && !req.query.cid, + pid: parseInt(req.query.pid, 10), + tid: parseInt(req.query.tid, 10), + cid: cid || (topicData ? topicData.cid : null), + action: action, + toPid: parseInt(req.query.toPid, 10), + discardRoute: discardRoute, + + resizable: false, + allowTopicsThumbnail: parseInt(meta.config.allowTopicsThumbnail, 10) === 1 && isMain, + + // can't use title property as that is used for page title + topicTitle: topicTitle, + titleLength: topicTitle ? topicTitle.length : 0, + topic: topicData, + thumb: topicData ? topicData.thumb : '', + body: body, + + isMain: isMain, + isTopicOrMain: !!req.query.cid || isMain, + maximumTitleLength: meta.config.maximumTitleLength, + maximumPostLength: meta.config.maximumPostLength, + minimumTagLength: meta.config.minimumTagLength || 3, + maximumTagLength: meta.config.maximumTagLength || 15, + tagWhitelist: tagWhitelist, + selectedCategory: cid ? categoryData : null, + minTags: categoryData.minTags, + maxTags: categoryData.maxTags, + + isTopic: !!req.query.cid, + isEditing: isEditing, + canSchedule: canScheduleTopics, + showHandleInput: meta.config.allowGuestHandles === 1 && + (req.uid === 0 || (isEditing && isGuestPost && (isAdmin || isMod))), + handle: postData ? postData.handle || '' : undefined, + formatting: formatting, + isAdminOrMod: isAdmin || isMod, + save_id: save_id, + privileges: globalPrivileges, + }, + }; +}; + +function generateDiscardRoute(req, topicData) { + if (req.query.cid) { + return `${nconf.get('relative_path')}/category/${validator.escape(String(req.query.cid))}`; + } else if ((req.query.tid || req.query.pid)) { + if (topicData) { + return `${nconf.get('relative_path')}/topic/${topicData.slug}`; + } + return `${nconf.get('relative_path')}/`; + } +} + +async function generateBody(req, postData) { + // Quoted reply + if (req.query.toPid && parseInt(req.query.quoted, 10) === 1 && postData) { + const username = await user.getUserField(postData.uid, 'username'); + const translated = await translator.translate(`[[modules:composer.user_said, ${username}]]`); + return `${translated}\n` + + `> ${postData ? `${postData.content.replace(/\n/g, '\n> ')}\n\n` : ''}`; + } else if (req.query.body || req.query.content) { + return validator.escape(String(req.query.body || req.query.content)); + } + return postData ? postData.content : ''; +} + +function generateSaveId(req) { + if (req.query.cid) { + return ['composer', req.uid, 'cid', req.query.cid].join(':'); + } else if (req.query.tid) { + return ['composer', req.uid, 'tid', req.query.tid].join(':'); + } else if (req.query.pid) { + return ['composer', req.uid, 'pid', req.query.pid].join(':'); + } +} + +async function getPostData(req) { + if (!req.query.pid && !req.query.toPid) { + return null; + } + + return await posts.getPostData(req.query.pid || req.query.toPid); +} + +async function getTopicData(req) { + if (req.query.tid) { + return await topics.getTopicData(req.query.tid); + } else if (req.query.pid) { + return await topics.getTopicDataByPid(req.query.pid); + } + return null; +} + +async function isModerator(req) { + if (!req.loggedIn) { + return false; + } + const cid = cidFromQuery(req.query); + return await user.isModerator(req.uid, cid); +} + +async function canTag(req) { + if (parseInt(req.query.cid, 10)) { + return await privileges.categories.can('topics:tag', req.query.cid, req.uid); + } + return true; +} + +async function canSchedule(req) { + if (parseInt(req.query.cid, 10)) { + return await privileges.categories.can('topics:schedule', req.query.cid, req.uid); + } + return false; +} + +async function getTagWhitelist(query, uid) { + const cid = await cidFromQuery(query); + const [tagWhitelist, isAdminOrMod] = await Promise.all([ + categories.getTagWhitelist([cid]), + privileges.categories.isAdminOrMod(cid, uid), + ]); + return categories.filterTagWhitelist(tagWhitelist[0], isAdminOrMod); +} + +async function cidFromQuery(query) { + if (query.cid) { + return query.cid; + } else if (query.tid) { + return await topics.getTopicField(query.tid, 'cid'); + } else if (query.pid) { + return await posts.getCidByPid(query.pid); + } + return null; +} diff --git a/node_modules/nodebb-plugin-composer-default/package.json b/node_modules/nodebb-plugin-composer-default/package.json new file mode 100644 index 00000000..85300c75 --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/package.json @@ -0,0 +1,43 @@ +{ + "name": "nodebb-plugin-composer-default", + "version": "9.2.4", + "description": "Default composer for NodeBB", + "main": "library.js", + "repository": { + "type": "git", + "url": "https://github.com/NodeBB/nodebb-plugin-composer-default" + }, + "scripts": { + "lint": "eslint ." + }, + "keywords": [ + "nodebb", + "plugin", + "composer", + "markdown" + ], + "author": { + "name": "NodeBB Team", + "email": "sales@nodebb.org" + }, + "license": "MIT", + "bugs": { + "url": "https://github.com/NodeBB/nodebb-plugin-composer-default/issues" + }, + "readmeFilename": "README.md", + "nbbpm": { + "compatibility": "^2.5.0" + }, + "dependencies": { + "@textcomplete/contenteditable": "^0.1.12", + "@textcomplete/core": "^0.1.12", + "@textcomplete/textarea": "^0.1.12", + "screenfull": "^5.0.2", + "validator": "^13.7.0" + }, + "devDependencies": { + "eslint": "^7.32.0", + "eslint-config-nodebb": "^0.0.1", + "eslint-plugin-import": "^2.23.4" + } +} diff --git a/node_modules/nodebb-plugin-composer-default/plugin.json b/node_modules/nodebb-plugin-composer-default/plugin.json new file mode 100644 index 00000000..1c793c57 --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/plugin.json @@ -0,0 +1,34 @@ +{ + "id": "nodebb-plugin-composer-default", + "url": "https://github.com/NodeBB/nodebb-plugin-composer-default", + "library": "library.js", + "hooks": [ + { "hook": "static:app.load", "method": "init" }, + { "hook": "filter:config.get", "method": "appendConfig" }, + { "hook": "filter:composer.build", "method": "filterComposerBuild" }, + { "hook": "filter:admin.header.build", "method": "addAdminNavigation" }, + { "hook": "filter:meta.getLinkTags", "method": "addPrefetchTags" } + ], + "less": [ + "./static/less/composer.less" + ], + "scripts": [ + "./static/lib/client.js", + "./node_modules/screenfull/dist/screenfull.js" + ], + "modules": { + "composer.js": "./static/lib/composer.js", + "composer/categoryList.js": "./static/lib/composer/categoryList.js", + "composer/controls.js": "./static/lib/composer/controls.js", + "composer/drafts.js": "./static/lib/composer/drafts.js", + "composer/formatting.js": "./static/lib/composer/formatting.js", + "composer/preview.js": "./static/lib/composer/preview.js", + "composer/resize.js": "./static/lib/composer/resize.js", + "composer/scheduler.js": "./static/lib/composer/scheduler.js", + "composer/tags.js": "./static/lib/composer/tags.js", + "composer/uploads.js": "./static/lib/composer/uploads.js", + "composer/autocomplete.js": "./static/lib/composer/autocomplete.js", + "../admin/plugins/composer-default.js": "./static/lib/admin.js" + }, + "templates": "static/templates" +} \ No newline at end of file diff --git a/node_modules/nodebb-plugin-composer-default/screenshots/desktop.png b/node_modules/nodebb-plugin-composer-default/screenshots/desktop.png new file mode 100644 index 0000000000000000000000000000000000000000..880af9552805f5dcf4fa5cad8efc3b457d6677b8 GIT binary patch literal 17071 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 G@Bcr+pxG?| literal 0 HcmV?d00001 diff --git a/node_modules/nodebb-plugin-composer-default/screenshots/mobile.png b/node_modules/nodebb-plugin-composer-default/screenshots/mobile.png new file mode 100644 index 0000000000000000000000000000000000000000..1776f4df0c53d3e4323a92d3352d2517530ab144 GIT binary patch literal 7693 zcmb_h2UJt(wniNVMT8j?R6t}bFdzY>N*z&Q5Gevu63Pe)gd)|Dmi6Ad_pAkkt{n&%4%+~+%`sYl)mpre$LwEw;m;R36d3{r`h_FwU*pf2xO!XIh z-@msJz!+Q~8*4u!Gn+uDbr0UHVY!C4WwOpIsZ1x^ob|GmOl<%9HJjh2<6YDlb=F@E zXtL?wBfQIL{(S9t-L-U5wUt=LKuOcE3+8LH%kNV$ZJ1VNz|tMv76}9YB)tmUBg&HTPwQ%K} z*;oBe<3Txj(|^c(b|I=bI!=`r^JPSJ;tkj_>uAdS@|T{y)jntERd@WsQ0V$-MnROA zsS)9PH!|&_-_u9o^PSIf^Vkf}dlZi;ch)iI=B4!nR=4lkIGOH`ARSL~4N3L&qhiG> z)4k06aAhaskElICjuO^C;;Bo_)V*Un^KaYn+6K2h5;GnL`?GG6f8S~gzWswgw-jb+ zGZ_{C?m%7HIk4QlqF?+0-*t);<(BT~nHQFZm%2-m@yCkMNJbpd^?Ui*=uCzHq39*a z;pcDaN1C-144r@OLw-di_15>`7ITz`>A_`)=7-Z#lnQBzd3b5j|_&($`}czjB(nquKN{jz_wWtF!+3 z`aOcC`*<-2GuK^fmy>))y^dS&Yl*LX@9~QYtvNkdQYmyLF;uSksKnAg`rcusGKyLD z=11}AP|LJmAu*oRomrca$};ng)Y4Lkq-*9$2c9fTFOSooI(3Tb79JF&k+VQ7Fw)h1 zmzHM2HWZYantGg`2%|2rgd?hMcYx;mhRc=%|C$S5j4#eX-D&mFzh;HPt=fB!4}kzpw9IZ0v;>AI#bl zf`Woho;-;*G!PRLgD!1oXc!uDZwS87=_e{G`tadHkKOf?baZbJw)XaxO^Ly9_Nh~y z#HVO9+S;13EiIRypC5lgu}l9G}$r%$^>5;A5--Il7d)OpGs=c6Md zwY0T+Q|<%=1_mBEc47!u+molm39$^|^ep`NF#@4i^rR_6Ii?qDfYR07?nT+Oy%REN zeJAwj(W9;`HAY58NCQOVVo0Xj)}pXk`@**rALF6lAN5JjErGR)q9I; z1i7W|oCsi|S6AFSDO5FE7U;arfy`x=4ow;_K@hFK#n1I2hpT`{146r@p>Y zTV$v3<*QfINfnuvMC`qL_Y!422KxG($7=r&sNPPMPx;1~rl6>3RPNAJS65eGZ!*`N zbLPw$J{{}>Wo2ghtJe>Y>XinzJN~EjpcHa6v&5|WF8lTsqr0j-_O$bLKiEz-R8~r( zP=n!D6jkFyl;pw{U0q!r92_1$PWkw8ZK^54#KgovUw_1Nk6f{+s~jzC*?Ysu$%!;i zdxQJ>byJ8iJ%`}vm6a99xU!y}USlXnmRj-%-o}VzHa|b_v^0_kiP6%U8XFUI-(C)4=AH`^ z^g5tyF|dgx$hh?m3`j~y=%#g~r>FlWzCMOoqf-z)MNbc3C6mb+Heno6n_*u2^>EtH zKi3Tox)HGk^SziZBfSib#9Q_=Q96b}!NK`CIXmlJY66CpZiXygUS5f+TZ1+=N~7A^ z+A}jVySuy1Pl_7wmpD0--@d(R(VaauHda|#`6y4Dba2MHC-+fPQxj!xi; zOZmKfdGqGYXoLY$Q&Sk-neH^|K?K6|I-o+KJxiALxF}G9#Xf6#do46HlpwQ}AfFN# z8mg+M7DMF|7k8R&j)G&dF^p5h{F^@m8`iMOwWfkN&=M>7SblGloR*q8H#e7lMny%% z$a7~}PTvNH!x_1+_*GQweI&?QK7D$QgQLoMWunS$EA+gO@ldHP)Fq#GK97Jv47TzG zc>sgLyE;2Vo74{vdq8<~baZrj4VT%QtxS9|BP21LJb6Lb^Z^oyG&VNYEKy}6SIo`N zE8oALF6z9qI?azN`8sWfewPA!pzLlK8X7KDZ6#3!?I4ebyV1yrJVRzd=<~$1_%Xo= zZe$$M*u-SEEuOd>ec}we`Lkz9;o(2Bv)}Oj+T5I*m$$aHRWmzW^zkFA#3~j5g{iK! z^@f;ONp|+vrX~`=CbwD=yVQ#XWr3=z*4=;rS?TFR8a`6*^$}jvQ982J?Mc9lGaQl$61K*> z>!ePLgF9<8ZT3!1tnxq5(h?G-wuZ{=A&VIz3B8aNg4}_$l$6xv%kOh@d8MS>_P1BM zGF2i@MBKlBAFgut#~-EMZCaDOhKGlp7KdVEVz_%U)Opg2eVG7>ph*Z#P3X!y%NrXi zYH9^oteCK{W{ws_w^KM@=$CsWOOotaR#vH7w-TjYG)~>|YJ++d#m*;%a!6hGJul~3 zNqwsnas5kUV^&56t`IvnXAyP(+}X2Qnuo`gVY4#*r1*HY^XJVUKj!k4fTnrFbtgTo z4KmXZkhYb^Df^+gSc^C6%a7rWte2!rz9?lgCf@5YH*UZlQ{k4M4#*47w( zgq-7CB$Q?6TP%co7MS5+yceBTz5sA^B^#OvYHDk)N7<_qScJ!SdnVpyarZF=v4;R! z=Z(t6Vi&OwiCIamyw|Q>6Bh0w5Cq^iZ1|V@qRx~whp!K{w5~=p#cj#n22A$z^P{4# zhMeL(dy3S7yK8G}YhW-8Ti99}9d)Mo`um$ced@6@9docttbuko&&2d?JH0rkaiyVQ zV`D=;u(MMGqdf@V-}vO#TUKc75b=>xyXof64CN)4+JiEq6%SQ(Rs)2ys9 z{&M-8XmfW#Yb&dv?P-@tksL0OF4q4;rQR7KDy?4UY6GXt;>}+c+fXvSZ z#+G|toXhE2yh~d=mnt1EeR`JyC0kHnV`o?7phX@Ze{5=6Vo5}h3Lnuu)YdMs9uy<_ z-5=PQGr~d`A|oQ~YnXGry}hA>RG zOw-E`PEz)X<<}1aaR<<@s-)Cm^E*bn;_$j6j=&nq(Kk9cZ*MFPqv%qMB0yK{x^1iP zC4g{B&&(VtcT_GCVBm|jr~1z9-+s1+2XRR77Rfi?i-*o6Q;`M|~D3fMMzq^bhM2gs-X z{{GI`c^M^ItQ=)MA|Rl#?}G_|7VzojLcb;wIZ)xWw794B{_t6k1rC&S+upLY&y87#SG<_@+yHKVdAuIJ4y&1%oP_P7 zwqM_HvUY`1eA*IySXgm!b4EXkTnx#{A$<+MGC46(y*=^z$14hb{rwr~>42g0UM*8c zP_tMRT)l}v@LFpV-|(>j?E@Oc`S2g&&i)z!A}+Ez)VfvSy8eB9d^|id@+w2up(q=v zbb+J-Sp%fVxNYuFMX0S;%z!wNrL1>>zJO<+@=S5_xS~*$_0T$4^H-&}S3>7mpre|ivre)d2y5uv&KSc`|jeMRLK$W!i_30k; z>d{-nA|TD9o_p5JqXsS-J-xlE@nZZE5)wCWuG;FhU)lOZ%NZFJB{MKuXgOcAR1S(9 zf`hb^=j7nPa4KW+@|2a8KjQHR^qG&;)t6fYYQERV*i8j8rGlD*W((7Xj?&VK^jxS0 z@Y~;63)_EKJZyC2EDOseX=lM}*Y4$LiB({mo10;umn0p`!Bd)=ng%d(R(n#)Q$iQ1 z`p|1))&tN|M7+%?v#&uY=7&8|X!-c;@R5&PzIcH(E1?)lAPOdm(su z1cimyHa9~91IFt)76jvk!>4&dapF){h* zIIn#y^O|oV4oB|$kd~8!Zrpp5i4}d%>A5`)hDFAG`#C6NL&K5JLCk|=Y76PbV$X(j z)zz8Kp1s<}6VhlqdX4HdQVpt!>u$vha^{0ryY1z1P>oFy-1vs7*7kN5`R%=)jL46#RkNg?kJTQVS+FPrn?cXz?%}~uJq4VCKo;}x@R$PxY*p65 zf*V8F%xY2Ylno-JXBTAACaH`2=FK9=j6u0Whq(y2F$_M|!`*#oc$oEu?qv?w?VTN^ zwZzm%qH6*NjF+S(Ks0A)T)22KE-5J~CI%1l6-K%J!fAV^QBClbFmlP!&fBn$zHnbj zsR?A5jI=bA`eGSxSB(dF9D$pfNXpWom1L7I$%)r}7A zUBr7{SBdMsh;q?eV zfB%awBI!;@EWIs%+w_EC>kk<0Fl0Rzw=&?5RbTor4T^7W4(ytw^YU18^L^5<9fjgT zc2mEh+a*B$m?gO=q^MVJ^r9f35mcwih7gt516FI-C+AsNxvSVMa05_%=-+VV+N?u7wm@_GF@gQ69Nl%<>^DPgjB)Yvw+#&J;)sSY8e3go*UIa`*orvE zxU9|Gx^?T`-Mh~t`*P!ib3cIX0+>PS>gvME^z;A$Z%If5bIMUkS;=BnJ75;~21luf z$Y^9#&|zK;4E1cdT*gJ(?#doM#sntor=IIOPl_ZLJ?%j{xhg11M?A zx#k>l;pq0A{UokGLFHcoj5n?DFZ}#Z`Drj3F-&!!uXhtpQ{Vn1_}m~fGZUtW_YD*j z6d2@gGyK{@a*$LI_3kxpJ;GI6O~1{&SOH`0-+u%30J7c!R~{8@Q^(t2)J5vbf;k3& zNnT!_l+z+RJG+&!vAltEDnK;&pxF3$9UUFoAASJ%A}JZ6P<(A-zQ@bb2~W zzCQkOjJDy+7r^WuA`yi`xw*NSfprI!;2^1}s0igp4^xHG=bvD`*CTdzb||xQ2jQ~v z`YGz`^F%c|+l{#%&{oiIF0QT?B5|dqrJ#dhzVVPXi3h0M=L7>-YQA57DTDiddCi(( zV>4QPNWY(aXr&}2!Gs)#<$+vLxS}$#`2D-Yldnu%clGu4L27^`ed*)l$3%OE9fk%3 z0)df^uJFT$G6#}|ib@D@G&Vm5Y@}Zm^dU@1Bqb$58@%}bEydrXC8m5i5G`^MeeoIX z;SrFZe}jbblYeMvi13A(9Y~$*oSdfS<~fL<2#21n?Mg?QA`Xek$}-c_i><1vDi(oI z;E><~3&E*D7%R%kz-IUM_JRh|qJy&;xw*MlYBW7(r>1}lRzUrAijZMI+8Ox;2KBz6 zZ;+T09FCu#ACsM3SW<%EFEJep)daIo)ccOzk{y;@lnv zpfCXhU%otwHqJHjqCCpcYMx%LnDMfAaM-78jruVG_qMmSMH!1(A-nw)2}Gi}jLh)R zP-SLj0|W_%6`=b-qQS5g6lbR+$gr-?&PDNqYz)R45QdqVIm61z+S+EQ^ib8sw7>`l zWYNYR0`iaMHGrH0K}W^Jz^1|6UAc0FVtwJwLnWmk`4mAhQ}at+{Ln7kstG*2yhX*u z>+90Rpyb6hir2f8#$ M_f+p<<$r(qUvfo3!2kdN literal 0 HcmV?d00001 diff --git a/node_modules/nodebb-plugin-composer-default/static/less/composer.less b/node_modules/nodebb-plugin-composer-default/static/less/composer.less new file mode 100644 index 00000000..e9a8ca66 --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/static/less/composer.less @@ -0,0 +1,573 @@ +.define-if-not-set() { + @btn-primary-bg: darken(#428bca, 6.5%); + @btn-primary-color: #fff; + @gray-light: lighten(#000, 46.7%); + @padding-large-vertical: 10px; + @padding-large-horizontal: 16px; +} + +.define-if-not-set(); + +.composer { + .no-select; + + z-index: @zindex-modal; + + background: #fff; + visibility: hidden; + + padding: 0; + + position: fixed; + bottom: 0; + top: 0; + right: 0; + left: 0; + + .composer-container { + height: 100%; + display: flex; + flex-direction: column; + } + + .mobile-navbar { + position: static; + background: @btn-primary-bg; + color: @btn-primary-color; + min-height: 40px; + margin: 0; + + .btn-group { + flex-shrink: 0; + } + + button { + font-size: 20px; + } + + display: flex; + + .category-name-container, .title { + text-align: center; + text-overflow: ellipsis; + overflow: hidden; + white-space: nowrap; + flex-grow: 2; + font-size: 16px; + line-height: inherit; + padding: 9px 5px; + margin: 0; + } + } + + .title-container { + display: flex; + border-bottom: 1px solid #eee; + margin: 0; + + > div[data-component="composer/title"] { + flex: 1; + } + + > div[data-component="composer/handle"] { + flex: 0.33; + } + + .title, .handle { + display: block; + margin: 0; + padding: 8px; + font-size: 18px; + border: 0; + .box-shadow(none); + overflow: hidden; + } + + .category-list-container { + + [component="category-selector"] { + margin-right: 0.5em; + + .category-dropdown-menu { + max-height: 300px; + } + } + } + + .category-list { + padding: 0 2rem; + } + + .quick-search-container { + top: 70px; + right: auto; + margin-bottom: 0px; + } + + .composer-submit, .composer-discard { + min-width: 106px; + } + + .action-bar { + .dropdown-menu:empty { + & ~ .dropdown-toggle { + display: none; + } + } + } + } + + .display-scheduler { + display: flex; + align-items: center; + margin-right: 1em; + + > i { + font-size: 30px; + display: inline-block; + cursor: pointer; + + &.active { + animation: 300ms ease forwards pulse; + } + } + } + + .category-tag-row { + margin: 0; + } + + .formatting-bar { + margin: 0; + + span { + color: #000; + } + + .spacer { + &:before { + content: ' | '; + color: @gray-light; + } + } + + .formatting-group { + padding: 0; + margin: 0; + overflow-x: auto; + white-space: nowrap; + display: block; + list-style: none; + + li { + display: inline-block; + padding: 10px 15px; + cursor: pointer; + color: #333; + position: relative; + + &:focus, &:hover { + outline: none; + background-color: darken(#fff, 10%); + } + + &[data-format="thumbs"][data-count]:after { + content: attr(data-count); + background: @brand-info; + color: white; + font-weight: 600; + position: absolute; + top: 5px; + left: 2.5em; + padding: 0px 5px; + border-radius: 5px; + font-size: 0.75em; + } + } + } + } + + .write-preview-container { + flex: 2; + display: flex; + overflow: hidden; + } + + .write-container, .preview-container { + display: flex; + flex: 1; + margin: 0 15px; + position: relative; + + .help-text { + text-transform: uppercase; + position: absolute; + right: 35px; + top: 8px; + font-size: 10px; + color: #999; + z-index: 1; + } + } + + .write-container { + &.maximized { + width: 100%; + } + } + + .preview-container { + word-wrap: break-word; + max-width: 50%; + max-width: ~"calc(50% - 30px)"; + } + + .write, .preview { + width: 100%; + font-size: 16px; + .border-radius(0); + resize: none; + overflow: auto; + padding: 25px 10px; + margin: 0; + } + + .write { + border: none; + border-top: 1px solid #EDEDED; + border-bottom: 1px solid #EDEDED; + .box-shadow(inset 0 1px 1px rgba(0, 0, 0, 0.05)); + } + + .preview { + -webkit-touch-callout: default; + user-select: text; + + p { + margin: 0 0 18px; + } + } + + .help { + .pointer; + } + + .toggle-preview { + margin-left: 20px; + .pointer; + } + + .tags-container { + [component="composer/tag/dropdown"] { + display: inherit; + top: -8px; + left: -8px; + .dropdown-menu { + max-height: 400px; + overflow-y: auto; + } + + > button { + border: 0; + } + } + + .bootstrap-tagsinput { + border: 0; + padding: 4px 6px; + box-shadow: none; + display: block; + max-height: 80px; + overflow: auto; + + input { + font-size: 16px; + width: 50%; + height: 28px; + padding: 4px 6px; + } + + .label { + color: white; + font-size: 13px; + } + } + } + + .category-selector { + display: block; + visibility: hidden; + + position: fixed; + left: 0; + right: 0; + bottom: 0; + + margin: 0; + padding: 0 5px; + max-height: ~"calc(100% - 86px)"; + + background: #fff; + box-shadow: 0 2px 6px rgba(0,0,0,0.35); + overflow: auto; + -webkit-overflow-scrolling: touch; + transform: translate3d(0, 350px, 0); + transition: transform 0.3s, visibility 0s 0.3s; + z-index: 2; + + &.open { + transform: none; + visibility: visible; + transition-delay: 0s; + } + + li { + padding: 10px; + color: #333; + + &.active { + background-color: @btn-primary-bg; + color: @btn-primary-color; + } + } + } + + .resizer { + display: none; + position: absolute; + left: 10%; + width: 80%; + top: 0px; + height: 0; + + .pointer; + + .trigger { + position: relative; + display: block; + top: -20px; + margin: 0 auto; + margin-left: 20px; + line-height: 26px; + .transition(filter .15s linear); + + &:hover { + filter: invert(100%); + cursor: ns-resize; + } + + i { + width: 32px; + height: 32px; + background: #333; + border: 1px solid #333; + .border-radius(50%); + + position: relative; + + color: #FFF; + font-size: 16px; + + &:before { + content: @fa-var-arrows-v; + position: relative; + top: 25%; + } + } + } + } + + .minimize { + display: none; + position: absolute; + top: 0px; + right: 10px; + height: 0; + + .pointer; + + .trigger { + position: relative; + display: block; + top: -20px; + right: 0px; + margin: 0 auto; + margin-left: 20px; + line-height: 26px; + .transition(filter .15s linear); + + &:hover { + filter: invert(100%); + } + + i { + width: 32px; + height: 32px; + background: #333; + border: 1px solid #333; + .border-radius(50%); + + position: relative; + + color: #FFF; + font-size: 16px; + + &:before { + position: relative; + top: 25%; + } + } + } + } + + &.reply { + .title-container { + display: none; + } + } + + &.resizable.maximized { + .resizer .trigger i { + &:before { + content: @fa-var-chevron-down; + } + } + + box-shadow: none; + } + + .draft-icon { + font-family: 'FontAwesome'; + color: @state-success-text; + margin: 0 1em; + opacity: 0; + + &::before { + content: @fa-var-save; + position: relative; + top: 25%; + } + + &::after { + content: @fa-var-check; + position: relative; + top: 18px; + font-size: 0.7em; + left: -15%; + } + + &.active { + animation: draft-saved 3s ease; + } + } +} + +.datetime-picker { + display: flex; + justify-content: center; + flex-direction: row; + min-width: 310px; + max-width: 310px; + margin: 0 auto; + + input { + flex: 3; + line-height: inherit; + } + + input + input { + border-left: none; + flex: 2; + } +} + +.modal.topic-scheduler { + z-index: 1070; + & + .modal-backdrop { + z-index: 1060; + } +} + +@keyframes draft-saved { + 0%, 100% { + opacity: 0; + } + + 15% { + opacity: 1; + } + + 30% { + opacity: 0.5; + } + + 45% { + opacity: 1; + } + + 85% { + opacity: 1; + } +} + +@keyframes pulse { + from { + transform: scale(1); + color: inherit; + } + 50% { + transform: scale(.9); + } + to { + transform: scale(1); + color: #00adff; + } +} + +@media (min-width: @screen-md-max) { + html.composing { + .composer { + left: 15%; + width: 70%; + } + } +} + +@media (max-width: @screen-sm-max) { + html.composing { + .composer { + height: 100%; + z-index: @zindex-modal; + + .draft-icon { + position: absolute; + bottom: 1em; + right: 0em; + + &::after { + top: 7px; + } + } + + .toggle-preview.hide { + display: inline-block !important; + } + + .preview-container { + max-width: initial; + } + } + + body { + padding-bottom: 0 !important; + } + } +} + +@media (min-width: @screen-md-min) { + @import './medium.less'; +} + +@import './zen-mode.less'; +@import './page-compose.less'; +@import './textcomplete.less'; \ No newline at end of file diff --git a/node_modules/nodebb-plugin-composer-default/static/less/medium.less b/node_modules/nodebb-plugin-composer-default/static/less/medium.less new file mode 100644 index 00000000..b2856fba --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/static/less/medium.less @@ -0,0 +1,66 @@ +.composer { + min-height: 400px; + + .resizer { + display: block; + } + + .minimize { + display: block; + } + + &.reply .title-container { + display: flex; + } + + .title-container { + border: 0; + + .title, .handle { + font-size: 22px; + padding: 4px 18px; + } + } + + .formatting-bar .formatting-group { + display: inline-block; + } + + &.resizable { + .box-shadow(0px 6px 12px rgba(0, 0, 0, 0.5)); + + padding-top: 30px; + padding-left: 15px; + padding-right: 15px; + } + + .category-tag-row { + margin-top: 5px; + margin-bottom: 8px; + } + + .write-preview-container { + margin-bottom: 15px; + } + + .write, .preview { + padding: 20px; + } + + .write { + border: 1px solid #EDEDED; + } + + .tags-container { + margin-top: 8px - 15px; + margin-bottom: 8px; + + .bootstrap-tagsinput { + padding: 0; + + input { + margin-left: -6px; + } + } + } +} diff --git a/node_modules/nodebb-plugin-composer-default/static/less/page-compose.less b/node_modules/nodebb-plugin-composer-default/static/less/page-compose.less new file mode 100644 index 00000000..e223c7f1 --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/static/less/page-compose.less @@ -0,0 +1,24 @@ +html, +body.page-compose, +.page-compose #content, +.page-compose #panel { + height: 100%; +} + +body.page-compose { + padding-bottom: 0 !important; +} + +.page-compose .composer { + height: 100%; + z-index: initial; + position: static; + + .display-scheduler { + margin: 0 0 0 .6em; + } +} + +.zen-mode .page-compose .composer { + position: absolute; +} \ No newline at end of file diff --git a/node_modules/nodebb-plugin-composer-default/static/less/textcomplete.less b/node_modules/nodebb-plugin-composer-default/static/less/textcomplete.less new file mode 100644 index 00000000..8e9a8731 --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/static/less/textcomplete.less @@ -0,0 +1,24 @@ +.textcomplete-dropdown { + border: 1px solid #ddd; + background-color: white; + list-style: none; + padding: 0; + margin: 0; + + li { + margin: 0; + } + + .textcomplete-footer, .textcomplete-item { + border-top: 1px solid #ddd; + } + + .textcomplete-item { + padding: 2px 5px; + cursor: pointer; + + &:hover, &.active { + background-color: rgb(110, 183, 219); + } + } +} \ No newline at end of file diff --git a/node_modules/nodebb-plugin-composer-default/static/less/zen-mode.less b/node_modules/nodebb-plugin-composer-default/static/less/zen-mode.less new file mode 100644 index 00000000..a1408eb4 --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/static/less/zen-mode.less @@ -0,0 +1,55 @@ +html.zen-mode { + overflow: hidden; +} + +.zen-mode .composer { + &.resizable { + padding-top: 0; + } + + .composer-container { + padding-top: 5px; + } + + .tag-row { + display: none; + } + + .title-container .category-list-container { + margin-top: 3px; + } + + .write, .preview { + border: none; + outline: none; + } + + .resizer { + display: none; + } + + &.reply { + .title-container { + display: none; + } + + .category-tag-row { + margin-top: 3px; + } + } + + @media (min-width: @screen-md-min) { + & { + padding-left: 15px; + padding-right: 15px; + } + .write-preview-container { + margin-bottom: 0; + + > div { + padding: 0; + margin: 0; + } + } + } +} \ No newline at end of file diff --git a/node_modules/nodebb-plugin-composer-default/static/lib/.eslintrc b/node_modules/nodebb-plugin-composer-default/static/lib/.eslintrc new file mode 100644 index 00000000..9fc9d47e --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/static/lib/.eslintrc @@ -0,0 +1,6 @@ +{ + "extends": "nodebb/public", + "rules": { + "no-cond-assign": ["error", "except-parens"] + } +} \ No newline at end of file diff --git a/node_modules/nodebb-plugin-composer-default/static/lib/admin.js b/node_modules/nodebb-plugin-composer-default/static/lib/admin.js new file mode 100644 index 00000000..eccc801f --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/static/lib/admin.js @@ -0,0 +1,25 @@ +'use strict'; + +define('admin/plugins/composer-default', ['settings', 'alerts'], function (Settings, alerts) { + var ACP = {}; + + ACP.init = function () { + Settings.load('composer-default', $('.composer-default-settings')); + + $('#save').on('click', function () { + Settings.save('composer-default', $('.composer-default-settings'), function () { + alerts.alert({ + type: 'success', + alert_id: 'composer-default-saved', + title: 'Settings Saved', + message: 'Please reload your NodeBB to apply these settings', + clickfn: function () { + socket.emit('admin.reload'); + }, + }); + }); + }); + }; + + return ACP; +}); diff --git a/node_modules/nodebb-plugin-composer-default/static/lib/client.js b/node_modules/nodebb-plugin-composer-default/static/lib/client.js new file mode 100644 index 00000000..3cc41b9b --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/static/lib/client.js @@ -0,0 +1,71 @@ +'use strict'; + +$(document).ready(function () { + $(window).on('action:app.load', function () { + require(['composer/drafts'], function (drafts) { + drafts.migrateGuest(); + drafts.loadOpen(); + }); + }); + + $(window).on('action:composer.topic.new', function (ev, data) { + if (config['composer-default'].composeRouteEnabled !== 'on') { + require(['composer'], function (composer) { + composer.newTopic({ + cid: data.cid, + title: data.title || '', + body: data.body || '', + tags: data.tags || [], + }); + }); + } else { + ajaxify.go( + 'compose?cid=' + data.cid + + (data.title ? '&title=' + encodeURIComponent(data.title) : '') + + (data.body ? '&body=' + encodeURIComponent(data.body) : '') + ); + } + }); + + $(window).on('action:composer.post.edit', function (ev, data) { + if (config['composer-default'].composeRouteEnabled !== 'on') { + require(['composer'], function (composer) { + composer.editPost(data.pid); + }); + } else { + ajaxify.go('compose?pid=' + data.pid); + } + }); + + $(window).on('action:composer.post.new', function (ev, data) { + if (config['composer-default'].composeRouteEnabled !== 'on') { + require(['composer'], function (composer) { + composer.newReply(data.tid, data.pid, data.topicName, data.text); + }); + } else { + ajaxify.go( + 'compose?tid=' + data.tid + + (data.pid ? '&toPid=' + data.pid : '') + + (data.topicName ? '&title=' + encodeURIComponent(data.topicName) : '') + + (data.text ? '&body=' + encodeURIComponent(data.text) : '') + ); + } + }); + + $(window).on('action:composer.addQuote', function (ev, data) { + if (config['composer-default'].composeRouteEnabled !== 'on') { + require(['composer'], function (composer) { + var topicUUID = composer.findByTid(data.tid); + composer.addQuote(data.tid, data.pid, data.selectedPid, data.topicName, data.username, data.text, topicUUID); + }); + } else { + ajaxify.go('compose?tid=' + data.tid + '&toPid=' + data.pid + '"ed=1&username=' + data.username); + } + }); + + $(window).on('action:composer.enhance', function (ev, data) { + require(['composer'], function (composer) { + composer.enhance(data.container); + }); + }); +}); diff --git a/node_modules/nodebb-plugin-composer-default/static/lib/composer.js b/node_modules/nodebb-plugin-composer-default/static/lib/composer.js new file mode 100644 index 00000000..01e5e6c3 --- /dev/null +++ b/node_modules/nodebb-plugin-composer-default/static/lib/composer.js @@ -0,0 +1,877 @@ +'use strict'; + +define('composer', [ + 'taskbar', + 'translator', + 'composer/uploads', + 'composer/formatting', + 'composer/drafts', + 'composer/tags', + 'composer/categoryList', + 'composer/preview', + 'composer/resize', + 'composer/autocomplete', + 'composer/scheduler', + 'scrollStop', + 'topicThumbs', + 'api', + 'bootbox', + 'alerts', + 'hooks', + 'messages', + 'search', + 'screenfull', +], function (taskbar, translator, uploads, formatting, drafts, tags, + categoryList, preview, resize, autocomplete, scheduler, scrollStop, + topicThumbs, api, bootbox, alerts, hooks, messagesModule, search, screenfull) { + var composer = { + active: undefined, + posts: {}, + bsEnvironment: undefined, + formatting: undefined, + }; + + $(window).off('resize', onWindowResize).on('resize', onWindowResize); + onWindowResize(); + + $(window).on('action:composer.topics.post', function (ev, data) { + localStorage.removeItem('category:' + data.data.cid + ':bookmark'); + localStorage.removeItem('category:' + data.data.cid + ':bookmark:clicked'); + }); + + $(window).on('popstate', function () { + var env = utils.findBootstrapEnvironment(); + if (composer.active && (env === 'xs' || env === 'sm')) { + if (!composer.posts[composer.active].modified) { + composer.discard(composer.active); + if (composer.discardConfirm && composer.discardConfirm.length) { + composer.discardConfirm.modal('hide'); + delete composer.discardConfirm; + } + return; + } + + translator.translate('[[modules:composer.discard]]', function (translated) { + composer.discardConfirm = bootbox.confirm(translated, function (confirm) { + if (confirm) { + composer.discard(composer.active); + } else { + composer.posts[composer.active].modified = true; + } + }); + composer.posts[composer.active].modified = false; + }); + } + }); + + function removeComposerHistory() { + var env = composer.bsEnvironment; + if (ajaxify.data.template.compose === true || env === 'xs' || env === 'sm') { + history.back(); + } + } + + function onWindowResize() { + var env = utils.findBootstrapEnvironment(); + var isMobile = env === 'xs' || env === 'sm'; + + if (preview.toggle) { + if (preview.env !== env && isMobile) { + preview.env = env; + preview.toggle(false); + } + preview.env = env; + } + + if (composer.active !== undefined) { + resize.reposition($('.composer[data-uuid="' + composer.active + '"]')); + + if (!isMobile && window.location.pathname.startsWith(config.relative_path + '/compose')) { + /* + * If this conditional is met, we're no longer in mobile/tablet + * resolution but we've somehow managed to have a mobile + * composer load, so let's go back to the topic + */ + history.back(); + } else if (isMobile && !window.location.pathname.startsWith(config.relative_path + '/compose')) { + /* + * In this case, we're in mobile/tablet resolution but the composer + * that loaded was a regular composer, so let's fix the address bar + */ + mobileHistoryAppend(); + } + } + composer.bsEnvironment = env; + } + + function alreadyOpen(post) { + // If a composer for the same cid/tid/pid is already open, return the uuid, else return bool false + var type; + var id; + + if (post.hasOwnProperty('cid')) { + type = 'cid'; + } else if (post.hasOwnProperty('tid')) { + type = 'tid'; + } else if (post.hasOwnProperty('pid')) { + type = 'pid'; + } + + id = post[type]; + + // Find a match + for (var uuid in composer.posts) { + if (composer.posts[uuid].hasOwnProperty(type) && id === composer.posts[uuid][type]) { + return uuid; + } + } + + // No matches... + return false; + } + + function push(post) { + var uuid = utils.generateUUID(); + var existingUUID = alreadyOpen(post); + + if (existingUUID) { + taskbar.updateActive(existingUUID); + return composer.load(existingUUID); + } + + var actionText = '[[topic:composer.new_topic]]'; + if (post.action === 'posts.reply') { + actionText = '[[topic:composer.replying_to]]'; + } else if (post.action === 'posts.edit') { + actionText = '[[topic:composer.editing]]'; + } + + translator.translate(actionText, function (translatedAction) { + taskbar.push('composer', uuid, { + title: translatedAction.replace('%1', '"' + post.title + '"'), + }); + }); + + // Construct a save_id + if (post.hasOwnProperty('cid')) { + post.save_id = ['composer', app.user.uid, 'cid', post.cid].join(':'); + } else if (post.hasOwnProperty('tid')) { + post.save_id = ['composer', app.user.uid, 'tid', post.tid].join(':'); + } else if (post.hasOwnProperty('pid')) { + post.save_id = ['composer', app.user.uid, 'pid', post.pid].join(':'); + } + + composer.posts[uuid] = post; + composer.load(uuid); + } + + async function composerAlert(post_uuid, message) { + $('.composer[data-uuid="' + post_uuid + '"]').find('.composer-submit').removeAttr('disabled'); + + const { showAlert } = await hooks.fire('filter:composer.error', { post_uuid, message, showAlert: true }); + + if (showAlert) { + alerts.alert({ + type: 'danger', + timeout: 10000, + title: '', + message: message, + alert_id: 'post_error', + }); + } + } + + composer.findByTid = function (tid) { + // Iterates through the initialised composers and returns the uuid of the matching composer + for (var uuid in composer.posts) { + if (composer.posts.hasOwnProperty(uuid) && composer.posts[uuid].hasOwnProperty('tid') && parseInt(composer.posts[uuid].tid, 10) === parseInt(tid, 10)) { + return uuid; + } + } + + return null; + }; + + composer.addButton = function (iconClass, onClick, title) { + formatting.addButton(iconClass, onClick, title); + }; + + composer.newTopic = async (data) => { + var pushData = { + action: 'topics.post', + cid: data.cid, + handle: data.handle, + title: data.title || '', + body: data.body || '', + tags: data.tags || [], + modified: !!((data.title && data.title.length) || (data.body && data.body.length)), + isMain: true, + }; + + ({ pushData } = await hooks.fire('filter:composer.topic.push', { + data: data, + pushData: pushData, + })); + + push(pushData); + }; + + composer.addQuote = function (tid, toPid, selectedPid, title, username, text, uuid) { + uuid = uuid || composer.active; + + var escapedTitle = (title || '') + .replace(/([\\`*_{}[\]()#+\-.!])/g, '\\$1') + .replace(/\[/g, '[') + .replace(/\]/g, ']') + .replace(/%/g, '%') + .replace(/,/g, ','); + + if (text) { + text = '> ' + text.replace(/\n/g, '\n> ') + '\n\n'; + } + var link = '[' + escapedTitle + '](' + config.relative_path + '/post/' + (selectedPid || toPid) + ')'; + if (uuid === undefined) { + if (title && (selectedPid || toPid)) { + composer.newReply(tid, toPid, title, '[[modules:composer.user_said_in, ' + username + ', ' + link + ']]\n' + text); + } else { + composer.newReply(tid, toPid, title, '[[modules:composer.user_said, ' + username + ']]\n' + text); + } + return; + } else if (uuid !== composer.active) { + // If the composer is not currently active, activate it + composer.load(uuid); + } + + var postContainer = $('.composer[data-uuid="' + uuid + '"]'); + var bodyEl = postContainer.find('textarea'); + var prevText = bodyEl.val(); + if (title && (selectedPid || toPid)) { + translator.translate('[[modules:composer.user_said_in, ' + username + ', ' + link + ']]\n', config.defaultLang, onTranslated); + } else { + translator.translate('[[modules:composer.user_said, ' + username + ']]\n', config.defaultLang, onTranslated); + } + + function onTranslated(translated) { + composer.posts[uuid].body = (prevText.length ? prevText + '\n\n' : '') + translated + text; + bodyEl.val(composer.posts[uuid].body); + focusElements(postContainer); + preview.render(postContainer); + } + }; + + composer.newReply = function (tid, toPid, title, text) { + translator.translate(text, config.defaultLang, function (translated) { + push({ + action: 'posts.reply', + tid: tid, + toPid: toPid, + title: title, + body: translated, + modified: !!((title && title.length) || (translated && translated.length)), + isMain: false, + }); + }); + }; + + composer.editPost = function (pid) { + socket.emit('plugins.composer.push', pid, function (err, threadData) { + if (err) { + return alerts.error(err); + } + threadData.action = 'posts.edit'; + threadData.pid = pid; + threadData.modified = false; + push(threadData); + }); + }; + + composer.load = function (post_uuid) { + var postContainer = $('.composer[data-uuid="' + post_uuid + '"]'); + if (postContainer.length) { + activate(post_uuid); + resize.reposition(postContainer); + focusElements(postContainer); + onShow(); + } else if (composer.formatting) { + createNewComposer(post_uuid); + } else { + socket.emit('plugins.composer.getFormattingOptions', function (err, options) { + if (err) { + return alerts.error(err); + } + composer.formatting = options; + createNewComposer(post_uuid); + }); + } + }; + + composer.enhance = function (postContainer, post_uuid, postData) { + /* + This method enhances a composer container with client-side sugar (preview, etc) + Everything in here also applies to the /compose route + */ + + if (!post_uuid && !postData) { + post_uuid = utils.generateUUID(); + composer.posts[post_uuid] = ajaxify.data; + postData = ajaxify.data; + postContainer.attr('data-uuid', post_uuid); + } + + var bodyEl = postContainer.find('textarea'); + var submitBtn = postContainer.find('.composer-submit'); + + categoryList.init(postContainer, composer.posts[post_uuid]); + scheduler.init(postContainer, composer.posts); + + formatting.addHandler(postContainer); + formatting.addComposerButtons(); + preview.handleToggler(postContainer); + + uploads.initialize(post_uuid); + tags.init(postContainer, composer.posts[post_uuid]); + autocomplete.init(postContainer, post_uuid); + + postContainer.on('change', 'input, textarea', function () { + composer.posts[post_uuid].modified = true; + + // Post is modified, save to list of opened drafts + drafts.updateVisibility('available', composer.posts[post_uuid].save_id, true); + drafts.updateVisibility('open', composer.posts[post_uuid].save_id, true); + }); + + submitBtn.on('click', function (e) { + e.preventDefault(); + e.stopPropagation(); // Other click events bring composer back to active state which is undesired on submit + + $(this).attr('disabled', true); + post(post_uuid); + }); + + require(['mousetrap'], function (mousetrap) { + mousetrap(postContainer.get(0)).bind('mod+enter', function () { + submitBtn.attr('disabled', true); + post(post_uuid); + }); + }); + + postContainer.find('.composer-discard').on('click', function (e) { + e.preventDefault(); + + if (!composer.posts[post_uuid].modified) { + composer.discard(post_uuid); + return removeComposerHistory(); + } + + formatting.exitFullscreen(); + + var btn = $(this).prop('disabled', true); + translator.translate('[[modules:composer.discard]]', function (translated) { + bootbox.confirm(translated, function (confirm) { + if (confirm) { + composer.discard(post_uuid); + removeComposerHistory(); + } + btn.prop('disabled', false); + }); + }); + }); + + postContainer.find('.composer-minimize, .minimize .trigger').on('click', function (e) { + e.preventDefault(); + e.stopPropagation(); + composer.minimize(post_uuid); + }); + + bodyEl.on('input propertychange', utils.debounce(function () { + preview.render(postContainer); + }, 250)); + + bodyEl.on('scroll', function () { + preview.matchScroll(postContainer); + }); + + drafts.init(postContainer, postData); + const draft = drafts.get(postData.save_id); + + preview.render(postContainer, function () { + preview.matchScroll(postContainer); + }); + + handleHelp(postContainer); + handleSearch(postContainer); + focusElements(postContainer); + if (postData.action === 'posts.edit') { + composer.updateThumbCount(post_uuid, postContainer); + } + + // Hide "zen mode" if fullscreen API is not enabled/available (ahem, iOS...) + if (!screenfull.isEnabled) { + $('[data-format="zen"]').addClass('hidden'); + } + + hooks.fire('action:composer.enhanced', { postContainer, postData, draft }); + }; + + async function getSelectedCategory(postData) { + if (ajaxify.data.template.category) { + // no need to load data if we are already on the category page + return ajaxify.data; + } else if (parseInt(postData.cid, 10)) { + return await api.get(`/api/category/${postData.cid}`, {}); + } + return null; + } + + async function createNewComposer(post_uuid) { + var postData = composer.posts[post_uuid]; + + var isTopic = postData ? postData.hasOwnProperty('cid') : false; + var isMain = postData ? !!postData.isMain : false; + var isEditing = postData ? !!postData.pid : false; + var isGuestPost = postData ? parseInt(postData.uid, 10) === 0 : false; + const isScheduled = postData.timestamp > Date.now(); + + // see + // https://github.com/NodeBB/NodeBB/issues/2994 and + // https://github.com/NodeBB/NodeBB/issues/1951 + // remove when 1951 is resolved + + var title = postData.title.replace(/%/g, '%').replace(/,/g, ','); + postData.category = await getSelectedCategory(postData); + const privileges = postData.category ? postData.category.privileges : ajaxify.data.privileges; + var data = { + title: title, + titleLength: title.length, + body: postData.body, + mobile: composer.bsEnvironment === 'xs' || composer.bsEnvironment === 'sm', + resizable: true, + thumb: postData.thumb, + isTopicOrMain: isTopic || isMain, + maximumTitleLength: config.maximumTitleLength, + maximumPostLength: config.maximumPostLength, + minimumTagLength: config.minimumTagLength, + maximumTagLength: config.maximumTagLength, + isTopic: isTopic, + isEditing: isEditing, + canSchedule: !!(isMain && privileges && + ((privileges['topics:schedule'] && !isEditing) || (isScheduled && privileges.view_scheduled))), + showHandleInput: config.allowGuestHandles && + (app.user.uid === 0 || (isEditing && isGuestPost && app.user.isAdmin)), + handle: postData ? postData.handle || '' : undefined, + formatting: composer.formatting, + tagWhitelist: postData.category ? postData.category.tagWhitelist : ajaxify.data.tagWhitelist, + privileges: app.user.privileges, + selectedCategory: postData.category, + submitOptions: [ + // Add items using `filter:composer.create`, or just add them to the