├── .gitattributes
├── .gitignore
├── .npmignore
├── LICENSE
├── README.md
├── controllers.js
├── eslint.config.mjs
├── library.js
├── package.json
├── plugin.json
├── screenshots
├── desktop.png
└── mobile.png
├── static
├── lib
│ ├── admin.js
│ ├── client.js
│ ├── composer.js
│ └── composer
│ │ ├── autocomplete.js
│ │ ├── categoryList.js
│ │ ├── controls.js
│ │ ├── drafts.js
│ │ ├── formatting.js
│ │ ├── post-queue.js
│ │ ├── preview.js
│ │ ├── resize.js
│ │ ├── scheduler.js
│ │ ├── tags.js
│ │ └── uploads.js
├── scss
│ ├── composer.scss
│ ├── page-compose.scss
│ ├── textcomplete.scss
│ └── zen-mode.scss
└── templates
│ ├── admin
│ └── plugins
│ │ └── composer-default.tpl
│ ├── compose.tpl
│ ├── composer.tpl
│ ├── modals
│ └── topic-scheduler.tpl
│ └── partials
│ ├── composer-formatting.tpl
│ ├── composer-tags.tpl
│ ├── composer-title-container.tpl
│ └── composer-write-preview.tpl
└── websockets.js
/.gitattributes:
--------------------------------------------------------------------------------
1 | # Auto detect text files and perform LF normalization
2 | * text=auto
3 |
4 | # Custom for Visual Studio
5 | *.cs diff=csharp
6 | *.sln merge=union
7 | *.csproj merge=union
8 | *.vbproj merge=union
9 | *.fsproj merge=union
10 | *.dbproj merge=union
11 |
12 | # Standard to msysgit
13 | *.doc diff=astextplain
14 | *.DOC diff=astextplain
15 | *.docx diff=astextplain
16 | *.DOCX diff=astextplain
17 | *.dot diff=astextplain
18 | *.DOT diff=astextplain
19 | *.pdf diff=astextplain
20 | *.PDF diff=astextplain
21 | *.rtf diff=astextplain
22 | *.RTF diff=astextplain
23 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | #################
2 | ## Eclipse
3 | #################
4 |
5 | *.pydevproject
6 | .project
7 | .metadata
8 | bin/
9 | tmp/
10 | *.tmp
11 | *.bak
12 | *.swp
13 | *~.nib
14 | local.properties
15 | .classpath
16 | .settings/
17 | .loadpath
18 |
19 | # External tool builders
20 | .externalToolBuilders/
21 |
22 | # Locally stored "Eclipse launch configurations"
23 | *.launch
24 |
25 | # CDT-specific
26 | .cproject
27 |
28 | # PDT-specific
29 | .buildpath
30 |
31 |
32 | #################
33 | ## Visual Studio
34 | #################
35 |
36 | .vscode
37 |
38 | ## Ignore Visual Studio temporary files, build results, and
39 | ## files generated by popular Visual Studio add-ons.
40 |
41 | # User-specific files
42 | *.suo
43 | *.user
44 | *.sln.docstates
45 |
46 | # Build results
47 |
48 | [Dd]ebug/
49 | [Rr]elease/
50 | x64/
51 | build/
52 | [Bb]in/
53 | [Oo]bj/
54 |
55 | # MSTest test Results
56 | [Tt]est[Rr]esult*/
57 | [Bb]uild[Ll]og.*
58 |
59 | *_i.c
60 | *_p.c
61 | *.ilk
62 | *.meta
63 | *.obj
64 | *.pch
65 | *.pdb
66 | *.pgc
67 | *.pgd
68 | *.rsp
69 | *.sbr
70 | *.tlb
71 | *.tli
72 | *.tlh
73 | *.tmp
74 | *.tmp_proj
75 | *.log
76 | *.vspscc
77 | *.vssscc
78 | .builds
79 | *.pidb
80 | *.log
81 | *.scc
82 |
83 | # Visual C++ cache files
84 | ipch/
85 | *.aps
86 | *.ncb
87 | *.opensdf
88 | *.sdf
89 | *.cachefile
90 |
91 | # Visual Studio profiler
92 | *.psess
93 | *.vsp
94 | *.vspx
95 |
96 | # Guidance Automation Toolkit
97 | *.gpState
98 |
99 | # ReSharper is a .NET coding add-in
100 | _ReSharper*/
101 | *.[Rr]e[Ss]harper
102 |
103 | # TeamCity is a build add-in
104 | _TeamCity*
105 |
106 | # DotCover is a Code Coverage Tool
107 | *.dotCover
108 |
109 | # NCrunch
110 | *.ncrunch*
111 | .*crunch*.local.xml
112 |
113 | # Installshield output folder
114 | [Ee]xpress/
115 |
116 | # DocProject is a documentation generator add-in
117 | DocProject/buildhelp/
118 | DocProject/Help/*.HxT
119 | DocProject/Help/*.HxC
120 | DocProject/Help/*.hhc
121 | DocProject/Help/*.hhk
122 | DocProject/Help/*.hhp
123 | DocProject/Help/Html2
124 | DocProject/Help/html
125 |
126 | # Click-Once directory
127 | publish/
128 |
129 | # Publish Web Output
130 | *.Publish.xml
131 | *.pubxml
132 |
133 | # NuGet Packages Directory
134 | ## TODO: If you have NuGet Package Restore enabled, uncomment the next line
135 | #packages/
136 |
137 | # Windows Azure Build Output
138 | csx
139 | *.build.csdef
140 |
141 | # Windows Store app package directory
142 | AppPackages/
143 |
144 | # Others
145 | sql/
146 | *.Cache
147 | ClientBin/
148 | [Ss]tyle[Cc]op.*
149 | ~$*
150 | *~
151 | *.dbmdl
152 | *.[Pp]ublish.xml
153 | *.pfx
154 | *.publishsettings
155 |
156 | # RIA/Silverlight projects
157 | Generated_Code/
158 |
159 | # Backup & report files from converting an old project file to a newer
160 | # Visual Studio version. Backup files are not needed, because we have git ;-)
161 | _UpgradeReport_Files/
162 | Backup*/
163 | UpgradeLog*.XML
164 | UpgradeLog*.htm
165 |
166 | # SQL Server files
167 | App_Data/*.mdf
168 | App_Data/*.ldf
169 |
170 | #############
171 | ## Windows detritus
172 | #############
173 |
174 | # Windows image file caches
175 | Thumbs.db
176 | ehthumbs.db
177 |
178 | # Folder config file
179 | Desktop.ini
180 |
181 | # Recycle Bin used on file shares
182 | $RECYCLE.BIN/
183 |
184 | # Mac crap
185 | .DS_Store
186 |
187 | # can't have it committed because it interferes with the package-lock.json
188 | # generated by each individual install
189 | package-lock.json
190 | yarn.lock
191 |
192 |
193 | #############
194 | ## Python
195 | #############
196 |
197 | *.py[co]
198 |
199 | # Packages
200 | *.egg
201 | *.egg-info
202 | dist/
203 | build/
204 | eggs/
205 | parts/
206 | var/
207 | sdist/
208 | develop-eggs/
209 | .installed.cfg
210 |
211 | # Installer logs
212 | pip-log.txt
213 |
214 | # Unit test / coverage reports
215 | .coverage
216 | .tox
217 |
218 | #Translations
219 | *.mo
220 |
221 | #Mr Developer
222 | .mr.developer.cfg
223 |
224 | sftp-config.json
225 | node_modules/
226 |
227 | *.sublime-project
228 | *.sublime-workspace
--------------------------------------------------------------------------------
/.npmignore:
--------------------------------------------------------------------------------
1 | sftp-config.json
2 | node_modules/
3 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | Copyright (c) 2016 NodeBB Inc.
2 |
3 | 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:
4 |
5 | The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
6 |
7 | 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.
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Default Composer for NodeBB
2 |
3 | This plugin activates the default composer for NodeBB. It is activated by default, but can be swapped out as necessary.
4 |
5 | ## Screenshots
6 |
7 | ### Desktop
8 | 
9 |
10 | ### Mobile Devices
11 | 
--------------------------------------------------------------------------------
/controllers.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | const Controllers = {};
4 |
5 | Controllers.renderAdminPage = function (req, res) {
6 | res.render('admin/plugins/composer-default', {
7 | title: 'Composer (Default)',
8 | });
9 | };
10 |
11 | module.exports = Controllers;
12 |
--------------------------------------------------------------------------------
/eslint.config.mjs:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | import serverConfig from 'eslint-config-nodebb';
4 | import publicConfig from 'eslint-config-nodebb/public';
5 |
6 | export default [
7 | ...publicConfig,
8 | ...serverConfig,
9 | ];
10 |
11 |
--------------------------------------------------------------------------------
/library.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | const url = require('url');
4 |
5 | const nconf = require.main.require('nconf');
6 | const validator = require('validator');
7 |
8 | const plugins = require.main.require('./src/plugins');
9 | const topics = require.main.require('./src/topics');
10 | const categories = require.main.require('./src/categories');
11 | const posts = require.main.require('./src/posts');
12 | const user = require.main.require('./src/user');
13 | const meta = require.main.require('./src/meta');
14 | const privileges = require.main.require('./src/privileges');
15 | const translator = require.main.require('./src/translator');
16 | const utils = require.main.require('./src/utils');
17 | const helpers = require.main.require('./src/controllers/helpers');
18 | const SocketPlugins = require.main.require('./src/socket.io/plugins');
19 | const socketMethods = require('./websockets');
20 |
21 | const plugin = module.exports;
22 |
23 | plugin.socketMethods = socketMethods;
24 |
25 | plugin.init = async function (data) {
26 | const { router } = data;
27 | const routeHelpers = require.main.require('./src/routes/helpers');
28 | const controllers = require('./controllers');
29 | SocketPlugins.composer = socketMethods;
30 | routeHelpers.setupAdminPageRoute(router, '/admin/plugins/composer-default', controllers.renderAdminPage);
31 | };
32 |
33 | plugin.appendConfig = async function (config) {
34 | config['composer-default'] = await meta.settings.get('composer-default');
35 | return config;
36 | };
37 |
38 | plugin.addAdminNavigation = async function (header) {
39 | header.plugins.push({
40 | route: '/plugins/composer-default',
41 | icon: 'fa-edit',
42 | name: 'Composer (Default)',
43 | });
44 | return header;
45 | };
46 |
47 | plugin.addPrefetchTags = async function (hookData) {
48 | const prefetch = [
49 | '/assets/src/modules/composer.js', '/assets/src/modules/composer/uploads.js', '/assets/src/modules/composer/drafts.js',
50 | '/assets/src/modules/composer/tags.js', '/assets/src/modules/composer/categoryList.js', '/assets/src/modules/composer/resize.js',
51 | '/assets/src/modules/composer/autocomplete.js', '/assets/templates/composer.tpl',
52 | `/assets/language/${meta.config.defaultLang || 'en-GB'}/topic.json`,
53 | `/assets/language/${meta.config.defaultLang || 'en-GB'}/modules.json`,
54 | `/assets/language/${meta.config.defaultLang || 'en-GB'}/tags.json`,
55 | ];
56 |
57 | hookData.links = hookData.links.concat(prefetch.map(path => ({
58 | rel: 'prefetch',
59 | href: `${nconf.get('relative_path') + path}?${meta.config['cache-buster']}`,
60 | })));
61 |
62 | return hookData;
63 | };
64 |
65 | plugin.getFormattingOptions = async function () {
66 | const defaultVisibility = {
67 | mobile: true,
68 | desktop: true,
69 |
70 | // op or reply
71 | main: true,
72 | reply: true,
73 | };
74 | let payload = {
75 | defaultVisibility,
76 | options: [
77 | {
78 | name: 'tags',
79 | title: '[[global:tags.tags]]',
80 | className: 'fa fa-tags',
81 | visibility: {
82 | ...defaultVisibility,
83 | desktop: false,
84 | },
85 | },
86 | {
87 | name: 'zen',
88 | title: '[[modules:composer.zen-mode]]',
89 | className: 'fa fa-arrows-alt',
90 | visibility: defaultVisibility,
91 | },
92 | ],
93 | };
94 | if (parseInt(meta.config.allowTopicsThumbnail, 10) === 1) {
95 | payload.options.push({
96 | name: 'thumbs',
97 | title: '[[topic:composer.thumb-title]]',
98 | className: 'fa fa-address-card-o',
99 | badge: true,
100 | visibility: {
101 | ...defaultVisibility,
102 | reply: false,
103 | },
104 | });
105 | }
106 |
107 | payload = await plugins.hooks.fire('filter:composer.formatting', payload);
108 |
109 | payload.options.forEach((option) => {
110 | option.visibility = {
111 | ...defaultVisibility,
112 | ...option.visibility || {},
113 | };
114 | });
115 |
116 | return payload ? payload.options : null;
117 | };
118 |
119 | plugin.filterComposerBuild = async function (hookData) {
120 | const { req } = hookData;
121 | const { res } = hookData;
122 |
123 | if (req.query.p) {
124 | try {
125 | const a = url.parse(req.query.p, true, true);
126 | return helpers.redirect(res, `/${(a.path || '').replace(/^\/*/, '')}`);
127 | } catch (e) {
128 | return helpers.redirect(res, '/');
129 | }
130 | } else if (!req.query.pid && !req.query.tid && !req.query.cid) {
131 | return helpers.redirect(res, '/');
132 | }
133 |
134 | await checkPrivileges(req, res);
135 |
136 | const [
137 | isMainPost,
138 | postData,
139 | topicData,
140 | categoryData,
141 | isAdmin,
142 | isMod,
143 | formatting,
144 | tagWhitelist,
145 | globalPrivileges,
146 | canTagTopics,
147 | canScheduleTopics,
148 | ] = await Promise.all([
149 | posts.isMain(req.query.pid),
150 | getPostData(req),
151 | getTopicData(req),
152 | categories.getCategoryFields(req.query.cid, [
153 | 'name', 'icon', 'color', 'bgColor', 'backgroundImage', 'imageClass', 'minTags', 'maxTags',
154 | ]),
155 | user.isAdministrator(req.uid),
156 | isModerator(req),
157 | plugin.getFormattingOptions(),
158 | getTagWhitelist(req.query, req.uid),
159 | privileges.global.get(req.uid),
160 | canTag(req),
161 | canSchedule(req),
162 | ]);
163 |
164 | const isEditing = !!req.query.pid;
165 | const isGuestPost = postData && parseInt(postData.uid, 10) === 0;
166 | const save_id = utils.generateSaveId(req.uid);
167 | const discardRoute = generateDiscardRoute(req, topicData);
168 | const body = await generateBody(req, postData);
169 |
170 | let action = 'topics.post';
171 | let isMain = isMainPost;
172 | if (req.query.tid) {
173 | action = 'posts.reply';
174 | } else if (req.query.pid) {
175 | action = 'posts.edit';
176 | } else {
177 | isMain = true;
178 | }
179 | globalPrivileges['topics:tag'] = canTagTopics;
180 | const cid = parseInt(req.query.cid, 10);
181 | const topicTitle = topicData && topicData.title ? topicData.title.replace(/%/g, '%').replace(/,/g, ',') : validator.escape(String(req.query.title || ''));
182 | return {
183 | req: req,
184 | res: res,
185 | templateData: {
186 | disabled: !req.query.pid && !req.query.tid && !req.query.cid,
187 | pid: parseInt(req.query.pid, 10),
188 | tid: parseInt(req.query.tid, 10),
189 | cid: cid || (topicData ? topicData.cid : null),
190 | action: action,
191 | toPid: parseInt(req.query.toPid, 10),
192 | discardRoute: discardRoute,
193 |
194 | resizable: false,
195 | allowTopicsThumbnail: parseInt(meta.config.allowTopicsThumbnail, 10) === 1 && isMain,
196 |
197 | // can't use title property as that is used for page title
198 | topicTitle: topicTitle,
199 | titleLength: topicTitle ? topicTitle.length : 0,
200 | topic: topicData,
201 | thumb: topicData ? topicData.thumb : '',
202 | body: body,
203 |
204 | isMain: isMain,
205 | isTopicOrMain: !!req.query.cid || isMain,
206 | maximumTitleLength: meta.config.maximumTitleLength,
207 | maximumPostLength: meta.config.maximumPostLength,
208 | minimumTagLength: meta.config.minimumTagLength || 3,
209 | maximumTagLength: meta.config.maximumTagLength || 15,
210 | tagWhitelist: tagWhitelist,
211 | selectedCategory: cid ? categoryData : null,
212 | minTags: categoryData.minTags,
213 | maxTags: categoryData.maxTags,
214 |
215 | isTopic: !!req.query.cid,
216 | isEditing: isEditing,
217 | canSchedule: canScheduleTopics,
218 | showHandleInput: meta.config.allowGuestHandles === 1 &&
219 | (req.uid === 0 || (isEditing && isGuestPost && (isAdmin || isMod))),
220 | handle: postData ? postData.handle || '' : undefined,
221 | formatting: formatting,
222 | isAdminOrMod: isAdmin || isMod,
223 | save_id: save_id,
224 | privileges: globalPrivileges,
225 | 'composer:showHelpTab': meta.config['composer:showHelpTab'] === 1,
226 | },
227 | };
228 | };
229 |
230 | async function checkPrivileges(req, res) {
231 | const notAllowed = (
232 | (req.query.cid && !await privileges.categories.can('topics:create', req.query.cid, req.uid)) ||
233 | (req.query.tid && !await privileges.topics.can('topics:reply', req.query.tid, req.uid)) ||
234 | (req.query.pid && !await privileges.posts.can('posts:edit', req.query.pid, req.uid))
235 | );
236 |
237 | if (notAllowed) {
238 | await helpers.notAllowed(req, res);
239 | }
240 | }
241 |
242 | function generateDiscardRoute(req, topicData) {
243 | if (req.query.cid) {
244 | return `${nconf.get('relative_path')}/category/${validator.escape(String(req.query.cid))}`;
245 | } else if ((req.query.tid || req.query.pid)) {
246 | if (topicData) {
247 | return `${nconf.get('relative_path')}/topic/${topicData.slug}`;
248 | }
249 | return `${nconf.get('relative_path')}/`;
250 | }
251 | }
252 |
253 | async function generateBody(req, postData) {
254 | let body = '';
255 | // Quoted reply
256 | if (req.query.toPid && parseInt(req.query.quoted, 10) === 1 && postData) {
257 | const username = await user.getUserField(postData.uid, 'username');
258 | const translated = await translator.translate(`[[modules:composer.user-said, ${username}]]`);
259 | body = `${translated}\n` +
260 | `> ${postData ? `${postData.content.replace(/\n/g, '\n> ')}\n\n` : ''}`;
261 | } else if (req.query.body || req.query.content) {
262 | body = validator.escape(String(req.query.body || req.query.content));
263 | }
264 | body = postData ? postData.content : '';
265 | return translator.escape(body);
266 | }
267 |
268 | async function getPostData(req) {
269 | if (!req.query.pid && !req.query.toPid) {
270 | return null;
271 | }
272 |
273 | return await posts.getPostData(req.query.pid || req.query.toPid);
274 | }
275 |
276 | async function getTopicData(req) {
277 | if (req.query.tid) {
278 | return await topics.getTopicData(req.query.tid);
279 | } else if (req.query.pid) {
280 | return await topics.getTopicDataByPid(req.query.pid);
281 | }
282 | return null;
283 | }
284 |
285 | async function isModerator(req) {
286 | if (!req.loggedIn) {
287 | return false;
288 | }
289 | const cid = cidFromQuery(req.query);
290 | return await user.isModerator(req.uid, cid);
291 | }
292 |
293 | async function canTag(req) {
294 | if (parseInt(req.query.cid, 10)) {
295 | return await privileges.categories.can('topics:tag', req.query.cid, req.uid);
296 | }
297 | return true;
298 | }
299 |
300 | async function canSchedule(req) {
301 | if (parseInt(req.query.cid, 10)) {
302 | return await privileges.categories.can('topics:schedule', req.query.cid, req.uid);
303 | }
304 | return false;
305 | }
306 |
307 | async function getTagWhitelist(query, uid) {
308 | const cid = await cidFromQuery(query);
309 | const [tagWhitelist, isAdminOrMod] = await Promise.all([
310 | categories.getTagWhitelist([cid]),
311 | privileges.categories.isAdminOrMod(cid, uid),
312 | ]);
313 | return categories.filterTagWhitelist(tagWhitelist[0], isAdminOrMod);
314 | }
315 |
316 | async function cidFromQuery(query) {
317 | if (query.cid) {
318 | return query.cid;
319 | } else if (query.tid) {
320 | return await topics.getTopicField(query.tid, 'cid');
321 | } else if (query.pid) {
322 | return await posts.getCidByPid(query.pid);
323 | }
324 | return null;
325 | }
326 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "nodebb-plugin-composer-default",
3 | "version": "10.2.50",
4 | "description": "Default composer for NodeBB",
5 | "main": "library.js",
6 | "repository": {
7 | "type": "git",
8 | "url": "https://github.com/NodeBB/nodebb-plugin-composer-default"
9 | },
10 | "scripts": {
11 | "lint": "eslint ."
12 | },
13 | "keywords": [
14 | "nodebb",
15 | "plugin",
16 | "composer",
17 | "markdown"
18 | ],
19 | "author": {
20 | "name": "NodeBB Team",
21 | "email": "sales@nodebb.org"
22 | },
23 | "license": "MIT",
24 | "bugs": {
25 | "url": "https://github.com/NodeBB/nodebb-plugin-composer-default/issues"
26 | },
27 | "readmeFilename": "README.md",
28 | "nbbpm": {
29 | "compatibility": "^4.0.0"
30 | },
31 | "dependencies": {
32 | "screenfull": "^5.0.2",
33 | "validator": "^13.7.0"
34 | },
35 | "devDependencies": {
36 | "eslint": "^9.25.1",
37 | "eslint-config-nodebb": "^1.1.4",
38 | "eslint-plugin-import": "^2.31.0"
39 | }
40 | }
41 |
--------------------------------------------------------------------------------
/plugin.json:
--------------------------------------------------------------------------------
1 | {
2 | "id": "nodebb-plugin-composer-default",
3 | "url": "https://github.com/NodeBB/nodebb-plugin-composer-default",
4 | "library": "library.js",
5 | "hooks": [
6 | { "hook": "static:app.load", "method": "init" },
7 | { "hook": "filter:config.get", "method": "appendConfig" },
8 | { "hook": "filter:composer.build", "method": "filterComposerBuild" },
9 | { "hook": "filter:admin.header.build", "method": "addAdminNavigation" },
10 | { "hook": "filter:meta.getLinkTags", "method": "addPrefetchTags" }
11 | ],
12 | "scss": [
13 | "./static/scss/composer.scss"
14 | ],
15 | "scripts": [
16 | "./static/lib/client.js",
17 | "./node_modules/screenfull/dist/screenfull.js"
18 | ],
19 | "modules": {
20 | "composer.js": "./static/lib/composer.js",
21 | "composer/categoryList.js": "./static/lib/composer/categoryList.js",
22 | "composer/controls.js": "./static/lib/composer/controls.js",
23 | "composer/drafts.js": "./static/lib/composer/drafts.js",
24 | "composer/formatting.js": "./static/lib/composer/formatting.js",
25 | "composer/preview.js": "./static/lib/composer/preview.js",
26 | "composer/resize.js": "./static/lib/composer/resize.js",
27 | "composer/scheduler.js": "./static/lib/composer/scheduler.js",
28 | "composer/tags.js": "./static/lib/composer/tags.js",
29 | "composer/uploads.js": "./static/lib/composer/uploads.js",
30 | "composer/autocomplete.js": "./static/lib/composer/autocomplete.js",
31 | "composer/post-queue.js": "./static/lib/composer/post-queue.js",
32 | "../admin/plugins/composer-default.js": "./static/lib/admin.js"
33 | },
34 | "templates": "static/templates"
35 | }
--------------------------------------------------------------------------------
/screenshots/desktop.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/NodeBB/nodebb-plugin-composer-default/6c5c0f50abfd12d32bd349ccd3c31b1f95006578/screenshots/desktop.png
--------------------------------------------------------------------------------
/screenshots/mobile.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/NodeBB/nodebb-plugin-composer-default/6c5c0f50abfd12d32bd349ccd3c31b1f95006578/screenshots/mobile.png
--------------------------------------------------------------------------------
/static/lib/admin.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | define('admin/plugins/composer-default', ['settings'], function (Settings) {
4 | const ACP = {};
5 |
6 | ACP.init = function () {
7 | Settings.load('composer-default', $('.composer-default-settings'));
8 |
9 | $('#save').on('click', function () {
10 | Settings.save('composer-default', $('.composer-default-settings'));
11 | });
12 | };
13 |
14 | return ACP;
15 | });
16 |
--------------------------------------------------------------------------------
/static/lib/client.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | $(document).ready(function () {
4 | $(window).on('action:app.load', function () {
5 | require(['composer/drafts'], function (drafts) {
6 | drafts.migrateGuest();
7 | drafts.loadOpen();
8 | });
9 | });
10 |
11 | $(window).on('action:composer.topic.new', function (ev, data) {
12 | if (config['composer-default'].composeRouteEnabled !== 'on') {
13 | require(['composer'], function (composer) {
14 | composer.newTopic({
15 | cid: data.cid,
16 | title: data.title || '',
17 | body: data.body || '',
18 | tags: data.tags || [],
19 | });
20 | });
21 | } else {
22 | ajaxify.go(
23 | 'compose?cid=' + data.cid +
24 | (data.title ? '&title=' + encodeURIComponent(data.title) : '') +
25 | (data.body ? '&body=' + encodeURIComponent(data.body) : '')
26 | );
27 | }
28 | });
29 |
30 | $(window).on('action:composer.post.edit', function (ev, data) {
31 | if (config['composer-default'].composeRouteEnabled !== 'on') {
32 | require(['composer'], function (composer) {
33 | composer.editPost({ pid: data.pid });
34 | });
35 | } else {
36 | ajaxify.go('compose?pid=' + data.pid);
37 | }
38 | });
39 |
40 | $(window).on('action:composer.post.new', function (ev, data) {
41 | // backwards compatibility
42 | data.body = data.body || data.text;
43 | data.title = data.title || data.topicName;
44 | if (config['composer-default'].composeRouteEnabled !== 'on') {
45 | require(['composer'], function (composer) {
46 | composer.newReply({
47 | tid: data.tid,
48 | toPid: data.pid,
49 | title: data.title,
50 | body: data.body,
51 | });
52 | });
53 | } else {
54 | ajaxify.go(
55 | 'compose?tid=' + data.tid +
56 | (data.pid ? '&toPid=' + data.pid : '') +
57 | (data.title ? '&title=' + encodeURIComponent(data.title) : '') +
58 | (data.body ? '&body=' + encodeURIComponent(data.body) : '')
59 | );
60 | }
61 | });
62 |
63 | $(window).on('action:composer.addQuote', function (ev, data) {
64 | data.body = data.body || data.text;
65 | data.title = data.title || data.topicName;
66 | if (config['composer-default'].composeRouteEnabled !== 'on') {
67 | require(['composer'], function (composer) {
68 | var topicUUID = composer.findByTid(data.tid);
69 | composer.addQuote({
70 | tid: data.tid,
71 | toPid: data.pid,
72 | selectedPid: data.selectedPid,
73 | title: data.title,
74 | username: data.username,
75 | body: data.body,
76 | uuid: topicUUID,
77 | });
78 | });
79 | } else {
80 | ajaxify.go('compose?tid=' + data.tid + '&toPid=' + data.pid + '"ed=1&username=' + data.username);
81 | }
82 | });
83 |
84 | $(window).on('action:composer.enhance', function (ev, data) {
85 | require(['composer'], function (composer) {
86 | composer.enhance(data.container);
87 | });
88 | });
89 | });
90 |
--------------------------------------------------------------------------------
/static/lib/composer.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | define('composer', [
4 | 'taskbar',
5 | 'translator',
6 | 'composer/uploads',
7 | 'composer/formatting',
8 | 'composer/drafts',
9 | 'composer/tags',
10 | 'composer/categoryList',
11 | 'composer/preview',
12 | 'composer/resize',
13 | 'composer/autocomplete',
14 | 'composer/scheduler',
15 | 'composer/post-queue',
16 | 'scrollStop',
17 | 'topicThumbs',
18 | 'api',
19 | 'bootbox',
20 | 'alerts',
21 | 'hooks',
22 | 'messages',
23 | 'search',
24 | 'screenfull',
25 | ], function (taskbar, translator, uploads, formatting, drafts, tags,
26 | categoryList, preview, resize, autocomplete, scheduler, postQueue, scrollStop,
27 | topicThumbs, api, bootbox, alerts, hooks, messagesModule, search, screenfull) {
28 | var composer = {
29 | active: undefined,
30 | posts: {},
31 | bsEnvironment: undefined,
32 | formatting: undefined,
33 | };
34 |
35 | $(window).off('resize', onWindowResize).on('resize', onWindowResize);
36 | onWindowResize();
37 |
38 | $(window).on('action:composer.topics.post', function (ev, data) {
39 | localStorage.removeItem('category:' + data.data.cid + ':bookmark');
40 | localStorage.removeItem('category:' + data.data.cid + ':bookmark:clicked');
41 | });
42 |
43 | $(window).on('popstate', function () {
44 | var env = utils.findBootstrapEnvironment();
45 | if (composer.active && (env === 'xs' || env === 'sm')) {
46 | if (!composer.posts[composer.active].modified) {
47 | composer.discard(composer.active);
48 | if (composer.discardConfirm && composer.discardConfirm.length) {
49 | composer.discardConfirm.modal('hide');
50 | delete composer.discardConfirm;
51 | }
52 | return;
53 | }
54 |
55 | translator.translate('[[modules:composer.discard]]', function (translated) {
56 | composer.discardConfirm = bootbox.confirm(translated, function (confirm) {
57 | if (confirm) {
58 | composer.discard(composer.active);
59 | } else {
60 | composer.posts[composer.active].modified = true;
61 | }
62 | });
63 | composer.posts[composer.active].modified = false;
64 | });
65 | }
66 | });
67 |
68 | function removeComposerHistory() {
69 | var env = composer.bsEnvironment;
70 | if (ajaxify.data.template.compose === true || env === 'xs' || env === 'sm') {
71 | history.back();
72 | }
73 | }
74 |
75 | function onWindowResize() {
76 | var env = utils.findBootstrapEnvironment();
77 | var isMobile = env === 'xs' || env === 'sm';
78 |
79 | if (preview.toggle) {
80 | if (preview.env !== env && isMobile) {
81 | preview.env = env;
82 | preview.toggle(false);
83 | }
84 | preview.env = env;
85 | }
86 |
87 | if (composer.active !== undefined) {
88 | resize.reposition($('.composer[data-uuid="' + composer.active + '"]'));
89 |
90 | if (!isMobile && window.location.pathname.startsWith(config.relative_path + '/compose')) {
91 | /*
92 | * If this conditional is met, we're no longer in mobile/tablet
93 | * resolution but we've somehow managed to have a mobile
94 | * composer load, so let's go back to the topic
95 | */
96 | history.back();
97 | } else if (isMobile && !window.location.pathname.startsWith(config.relative_path + '/compose')) {
98 | /*
99 | * In this case, we're in mobile/tablet resolution but the composer
100 | * that loaded was a regular composer, so let's fix the address bar
101 | */
102 | mobileHistoryAppend();
103 | }
104 | }
105 | composer.bsEnvironment = env;
106 | }
107 |
108 | function alreadyOpen(post) {
109 | // If a composer for the same cid/tid/pid is already open, return the uuid, else return bool false
110 | var type;
111 | var id;
112 |
113 | if (post.hasOwnProperty('cid')) {
114 | type = 'cid';
115 | } else if (post.hasOwnProperty('tid')) {
116 | type = 'tid';
117 | } else if (post.hasOwnProperty('pid')) {
118 | type = 'pid';
119 | }
120 |
121 | id = post[type];
122 |
123 | // Find a match
124 | for (const uuid of Object.keys(composer.posts)) {
125 | if (composer.posts[uuid].hasOwnProperty(type) && id === composer.posts[uuid][type]) {
126 | return uuid;
127 | }
128 | }
129 |
130 | // No matches...
131 | return false;
132 | }
133 |
134 | function push(post) {
135 | if (!post) {
136 | return;
137 | }
138 |
139 | var uuid = utils.generateUUID();
140 | var existingUUID = alreadyOpen(post);
141 |
142 | if (existingUUID) {
143 | taskbar.updateActive(existingUUID);
144 | return composer.load(existingUUID);
145 | }
146 |
147 | var actionText = '[[topic:composer.new-topic]]';
148 | if (post.action === 'posts.reply') {
149 | actionText = '[[topic:composer.replying-to]]';
150 | } else if (post.action === 'posts.edit') {
151 | actionText = '[[topic:composer.editing-in]]';
152 | }
153 |
154 | translator.translate(actionText, function (translatedAction) {
155 | taskbar.push('composer', uuid, {
156 | title: translatedAction.replace('%1', '"' + post.title + '"'),
157 | });
158 | });
159 |
160 | composer.posts[uuid] = post;
161 | composer.load(uuid);
162 | }
163 |
164 | async function composerAlert(post_uuid, message) {
165 | $('.composer[data-uuid="' + post_uuid + '"]').find('.composer-submit').removeAttr('disabled');
166 |
167 | const { showAlert } = await hooks.fire('filter:composer.error', { post_uuid, message, showAlert: true });
168 |
169 | if (showAlert) {
170 | alerts.alert({
171 | type: 'danger',
172 | timeout: 10000,
173 | title: '',
174 | message: message,
175 | alert_id: 'post_error',
176 | });
177 | }
178 | }
179 |
180 | composer.findByTid = function (tid) {
181 | // Iterates through the initialised composers and returns the uuid of the matching composer
182 | for (const uuid of Object.keys(composer.posts)) {
183 | if (composer.posts[uuid].hasOwnProperty('tid') && String(composer.posts[uuid].tid) === String(tid)) {
184 | return uuid;
185 | }
186 | }
187 |
188 | return null;
189 | };
190 |
191 | composer.addButton = function (iconClass, onClick, title) {
192 | formatting.addButton(iconClass, onClick, title);
193 | };
194 |
195 | composer.newTopic = async (data) => {
196 | let pushData = {
197 | save_id: data.save_id,
198 | action: 'topics.post',
199 | cid: data.cid,
200 | handle: data.handle,
201 | title: data.title || '',
202 | body: data.body || '',
203 | tags: data.tags || [],
204 | modified: !!((data.title && data.title.length) || (data.body && data.body.length)),
205 | isMain: true,
206 | };
207 |
208 | ({ pushData } = await hooks.fire('filter:composer.topic.push', {
209 | data: data,
210 | pushData: pushData,
211 | }));
212 |
213 | push(pushData);
214 | };
215 |
216 | composer.addQuote = function (data) {
217 | // tid, toPid, selectedPid, title, username, text, uuid
218 | data.uuid = data.uuid || composer.active;
219 |
220 | var escapedTitle = (data.title || '')
221 | .replace(/([\\`*_{}[\]()#+\-.!])/g, '\\$1')
222 | .replace(/\[/g, '[')
223 | .replace(/\]/g, ']')
224 | .replace(/%/g, '%')
225 | .replace(/,/g, ',');
226 |
227 | if (data.body) {
228 | data.body = '> ' + data.body.replace(/\n/g, '\n> ') + '\n\n';
229 | }
230 | var link = '[' + escapedTitle + '](' + config.relative_path + '/post/' + encodeURIComponent(data.selectedPid || data.toPid) + ')';
231 | if (data.uuid === undefined) {
232 | if (data.title && (data.selectedPid || data.toPid)) {
233 | composer.newReply({
234 | tid: data.tid,
235 | toPid: data.toPid,
236 | title: data.title,
237 | body: '[[modules:composer.user-said-in, ' + data.username + ', ' + link + ']]\n' + data.body,
238 | });
239 | } else {
240 | composer.newReply({
241 | tid: data.tid,
242 | toPid: data.toPid,
243 | title: data.title,
244 | body: '[[modules:composer.user-said, ' + data.username + ']]\n' + data.body,
245 | });
246 | }
247 | return;
248 | } else if (data.uuid !== composer.active) {
249 | // If the composer is not currently active, activate it
250 | composer.load(data.uuid);
251 | }
252 |
253 | var postContainer = $('.composer[data-uuid="' + data.uuid + '"]');
254 | var bodyEl = postContainer.find('textarea');
255 | var prevText = bodyEl.val();
256 | if (data.title && (data.selectedPid || data.toPid)) {
257 | translator.translate('[[modules:composer.user-said-in, ' + data.username + ', ' + link + ']]\n', config.defaultLang, onTranslated);
258 | } else {
259 | translator.translate('[[modules:composer.user-said, ' + data.username + ']]\n', config.defaultLang, onTranslated);
260 | }
261 |
262 | function onTranslated(translated) {
263 | composer.posts[data.uuid].body = (prevText.length ? prevText + '\n\n' : '') + translated + data.body;
264 | bodyEl.val(composer.posts[data.uuid].body);
265 | focusElements(postContainer);
266 | preview.render(postContainer);
267 | }
268 | };
269 |
270 | composer.newReply = function (data) {
271 | translator.translate(data.body, config.defaultLang, function (translated) {
272 | push({
273 | save_id: data.save_id,
274 | action: 'posts.reply',
275 | tid: data.tid,
276 | toPid: data.toPid,
277 | title: data.title,
278 | body: translated,
279 | modified: !!(translated && translated.length),
280 | isMain: false,
281 | });
282 | });
283 | };
284 |
285 | composer.editPost = function (data) {
286 | // pid, text
287 | socket.emit('plugins.composer.push', data.pid, function (err, postData) {
288 | if (err) {
289 | return alerts.error(err);
290 | }
291 | postData.save_id = data.save_id;
292 | postData.action = 'posts.edit';
293 | postData.pid = data.pid;
294 | postData.modified = false;
295 | if (data.body) {
296 | postData.body = data.body;
297 | postData.modified = true;
298 | }
299 | if (data.title) {
300 | postData.title = data.title;
301 | postData.modified = true;
302 | }
303 | push(postData);
304 | });
305 | };
306 |
307 | composer.load = function (post_uuid) {
308 | var postContainer = $('.composer[data-uuid="' + post_uuid + '"]');
309 | if (postContainer.length) {
310 | activate(post_uuid);
311 | resize.reposition(postContainer);
312 | focusElements(postContainer);
313 | onShow();
314 | } else if (composer.formatting) {
315 | createNewComposer(post_uuid);
316 | } else {
317 | socket.emit('plugins.composer.getFormattingOptions', function (err, options) {
318 | if (err) {
319 | return alerts.error(err);
320 | }
321 | composer.formatting = options;
322 | createNewComposer(post_uuid);
323 | });
324 | }
325 | };
326 |
327 | composer.enhance = function (postContainer, post_uuid, postData) {
328 | /*
329 | This method enhances a composer container with client-side sugar (preview, etc)
330 | Everything in here also applies to the /compose route
331 | */
332 |
333 | if (!post_uuid && !postData) {
334 | post_uuid = utils.generateUUID();
335 | composer.posts[post_uuid] = ajaxify.data;
336 | postData = ajaxify.data;
337 | postContainer.attr('data-uuid', post_uuid);
338 | }
339 |
340 | categoryList.init(postContainer, composer.posts[post_uuid]);
341 | scheduler.init(postContainer, composer.posts);
342 |
343 | formatting.addHandler(postContainer);
344 | formatting.addComposerButtons();
345 | preview.handleToggler(postContainer);
346 | postQueue.showAlert(postContainer, postData);
347 | uploads.initialize(post_uuid);
348 | tags.init(postContainer, composer.posts[post_uuid]);
349 | autocomplete.init(postContainer, post_uuid);
350 |
351 | postContainer.on('change', 'input, textarea', function () {
352 | composer.posts[post_uuid].modified = true;
353 | });
354 |
355 | postContainer.on('click', '.composer-submit', function (e) {
356 | e.preventDefault();
357 | e.stopPropagation(); // Other click events bring composer back to active state which is undesired on submit
358 |
359 | $(this).attr('disabled', true);
360 | post(post_uuid);
361 | });
362 |
363 | require(['mousetrap'], function (mousetrap) {
364 | mousetrap(postContainer.get(0)).bind('mod+enter', function () {
365 | postContainer.find('.composer-submit').attr('disabled', true);
366 | post(post_uuid);
367 | });
368 | });
369 |
370 | postContainer.find('.composer-discard').on('click', function (e) {
371 | e.preventDefault();
372 |
373 | if (!composer.posts[post_uuid].modified) {
374 | composer.discard(post_uuid);
375 | return removeComposerHistory();
376 | }
377 |
378 | formatting.exitFullscreen();
379 |
380 | var btn = $(this).prop('disabled', true);
381 | translator.translate('[[modules:composer.discard]]', function (translated) {
382 | bootbox.confirm(translated, function (confirm) {
383 | if (confirm) {
384 | composer.discard(post_uuid);
385 | removeComposerHistory();
386 | }
387 | btn.prop('disabled', false);
388 | });
389 | });
390 | });
391 |
392 | postContainer.find('.composer-minimize, .minimize .trigger').on('click', function (e) {
393 | e.preventDefault();
394 | e.stopPropagation();
395 | composer.minimize(post_uuid);
396 | });
397 |
398 | const textareaEl = postContainer.find('textarea');
399 | textareaEl.on('input propertychange', utils.debounce(function () {
400 | preview.render(postContainer);
401 | }, 250));
402 |
403 | textareaEl.on('scroll', function () {
404 | preview.matchScroll(postContainer);
405 | });
406 |
407 | drafts.init(postContainer, postData);
408 | const draft = drafts.get(postData.save_id);
409 |
410 | preview.render(postContainer, function () {
411 | preview.matchScroll(postContainer);
412 | });
413 |
414 | if (postData.action === 'posts.edit' && !utils.isNumber(postData.pid)) {
415 | handleRemotePid(postContainer);
416 | }
417 | handleHelp(postContainer);
418 | handleSearch(postContainer);
419 | focusElements(postContainer);
420 | if (postData.action === 'posts.edit') {
421 | composer.updateThumbCount(post_uuid, postContainer);
422 | }
423 |
424 | // Hide "zen mode" if fullscreen API is not enabled/available (ahem, iOS...)
425 | if (!screenfull.isEnabled) {
426 | $('[data-format="zen"]').parent().addClass('hidden');
427 | }
428 |
429 | hooks.fire('action:composer.enhanced', { postContainer, postData, draft });
430 | };
431 |
432 | async function getSelectedCategory(postData) {
433 | const { template } = ajaxify.data;
434 | const { cid } = postData;
435 | if ((template.category || template.world) && String(cid) === String(ajaxify.data.cid)) {
436 | // no need to load data if we are already on the category page
437 | return ajaxify.data;
438 | } else if (cid) {
439 | const categoryUrl = cid !== -1 ? `/api/category/${encodeURIComponent(postData.cid)}` : `/api/world`;
440 | return await api.get(categoryUrl, {});
441 | }
442 | return null;
443 | }
444 |
445 | async function createNewComposer(post_uuid) {
446 | var postData = composer.posts[post_uuid];
447 |
448 | var isTopic = postData ? postData.hasOwnProperty('cid') : false;
449 | var isMain = postData ? !!postData.isMain : false;
450 | var isEditing = postData ? !!postData.pid : false;
451 | var isGuestPost = postData ? parseInt(postData.uid, 10) === 0 : false;
452 | const isScheduled = postData.timestamp > Date.now();
453 |
454 | // see
455 | // https://github.com/NodeBB/NodeBB/issues/2994 and
456 | // https://github.com/NodeBB/NodeBB/issues/1951
457 | // remove when 1951 is resolved
458 |
459 | var title = postData.title.replace(/%/g, '%').replace(/,/g, ',');
460 | postData.category = await getSelectedCategory(postData);
461 | const privileges = postData.category ? postData.category.privileges : ajaxify.data.privileges;
462 | var data = {
463 | topicTitle: title,
464 | titleLength: title.length,
465 | body: translator.escape(utils.escapeHTML(postData.body)),
466 | mobile: composer.bsEnvironment === 'xs' || composer.bsEnvironment === 'sm',
467 | resizable: true,
468 | thumb: postData.thumb,
469 | isTopicOrMain: isTopic || isMain,
470 | maximumTitleLength: config.maximumTitleLength,
471 | maximumPostLength: config.maximumPostLength,
472 | minimumTagLength: config.minimumTagLength,
473 | maximumTagLength: config.maximumTagLength,
474 | 'composer:showHelpTab': config['composer:showHelpTab'],
475 | isTopic: isTopic,
476 | isEditing: isEditing,
477 | canSchedule: !!(isMain && privileges &&
478 | ((privileges['topics:schedule'] && !isEditing) || (isScheduled && privileges.view_scheduled))),
479 | canUploadImage: app.user.privileges['upload:post:image'] && (config.maximumFileSize > 0 || app.user.isAdmin),
480 | canUploadFile: app.user.privileges['upload:post:file'] && (config.maximumFileSize > 0 || app.user.isAdmin),
481 | showHandleInput: config.allowGuestHandles &&
482 | (app.user.uid === 0 || (isEditing && isGuestPost && app.user.isAdmin)),
483 | handle: postData ? postData.handle || '' : undefined,
484 | formatting: composer.formatting,
485 | tagWhitelist: postData.category ? postData.category.tagWhitelist : ajaxify.data.tagWhitelist,
486 | privileges: app.user.privileges,
487 | selectedCategory: postData.category,
488 | submitOptions: [
489 | // Add items using `filter:composer.create`, or just add them to the
in DOM
490 | // {
491 | // action: 'foobar',
492 | // text: 'Text Label',
493 | // }
494 | ],
495 | };
496 |
497 | if (data.mobile) {
498 | mobileHistoryAppend();
499 |
500 | app.toggleNavbar(false);
501 | }
502 |
503 | postData.mobile = composer.bsEnvironment === 'xs' || composer.bsEnvironment === 'sm';
504 |
505 | ({ postData, createData: data } = await hooks.fire('filter:composer.create', {
506 | postData: postData,
507 | createData: data,
508 | }));
509 |
510 | app.parseAndTranslate('composer', data, function (composerTemplate) {
511 | if ($('.composer.composer[data-uuid="' + post_uuid + '"]').length) {
512 | return;
513 | }
514 | composerTemplate = $(composerTemplate);
515 |
516 | composerTemplate.find('.title').each(function () {
517 | $(this).text(translator.unescape($(this).text()));
518 | });
519 |
520 | composerTemplate.attr('data-uuid', post_uuid);
521 |
522 | $(document.body).append(composerTemplate);
523 |
524 | var postContainer = $(composerTemplate[0]);
525 |
526 | resize.reposition(postContainer);
527 | composer.enhance(postContainer, post_uuid, postData);
528 | /*
529 | Everything after this line is applied to the resizable composer only
530 | Want something done to both resizable composer and the one in /compose?
531 | Put it in composer.enhance().
532 |
533 | Eventually, stuff after this line should be moved into composer.enhance().
534 | */
535 |
536 | activate(post_uuid);
537 |
538 | postContainer.on('click', function () {
539 | if (!taskbar.isActive(post_uuid)) {
540 | taskbar.updateActive(post_uuid);
541 | }
542 | });
543 |
544 | resize.handleResize(postContainer);
545 |
546 | if (composer.bsEnvironment === 'xs' || composer.bsEnvironment === 'sm') {
547 | var submitBtns = postContainer.find('.composer-submit');
548 | var mobileSubmitBtn = postContainer.find('.mobile-navbar .composer-submit');
549 | var textareaEl = postContainer.find('.write');
550 | var idx = textareaEl.attr('tabindex');
551 |
552 | submitBtns.removeAttr('tabindex');
553 | mobileSubmitBtn.attr('tabindex', parseInt(idx, 10) + 1);
554 | }
555 |
556 | $(window).trigger('action:composer.loaded', {
557 | postContainer: postContainer,
558 | post_uuid: post_uuid,
559 | composerData: composer.posts[post_uuid],
560 | formatting: composer.formatting,
561 | });
562 |
563 | scrollStop.apply(postContainer.find('.write'));
564 | focusElements(postContainer);
565 | onShow();
566 | });
567 | }
568 |
569 | function mobileHistoryAppend() {
570 | var path = 'compose?p=' + window.location.pathname;
571 | var returnPath = window.location.pathname.slice(1) + window.location.search;
572 |
573 | // Remove relative path from returnPath
574 | if (returnPath.startsWith(config.relative_path.slice(1))) {
575 | returnPath = returnPath.slice(config.relative_path.length);
576 | }
577 |
578 | // Add in return path to be caught by ajaxify when post is completed, or if back is pressed
579 | window.history.replaceState({
580 | url: null,
581 | returnPath: returnPath,
582 | }, returnPath, config.relative_path + '/' + returnPath);
583 |
584 | // Update address bar in case f5 is pressed
585 | window.history.pushState({
586 | url: path,
587 | }, path, `${config.relative_path}/${returnPath}`);
588 | }
589 |
590 | function handleRemotePid(postContainer) {
591 | alerts.alert({
592 | title: '[[modules:composer.remote-pid-editing]]',
593 | message: '[[modules:composer.remote-pid-content-immutable]]',
594 | timeout: 15000,
595 | });
596 | var container = postContainer.find('.write-container');
597 | container.addClass('hidden');
598 | }
599 |
600 | function handleHelp(postContainer) {
601 | const helpBtn = postContainer.find('[data-action="help"]');
602 | helpBtn.on('click', async function () {
603 | const html = await socket.emit('plugins.composer.renderHelp');
604 | if (html && html.length > 0) {
605 | bootbox.dialog({
606 | size: 'large',
607 | message: html,
608 | onEscape: true,
609 | backdrop: true,
610 | onHidden: function () {
611 | helpBtn.focus();
612 | },
613 | });
614 | }
615 | });
616 | }
617 |
618 | function handleSearch(postContainer) {
619 | var uuid = postContainer.attr('data-uuid');
620 | var isEditing = composer.posts[uuid] && composer.posts[uuid].action === 'posts.edit';
621 | var env = utils.findBootstrapEnvironment();
622 | var isMobile = env === 'xs' || env === 'sm';
623 | if (isEditing || isMobile) {
624 | return;
625 | }
626 |
627 | search.enableQuickSearch({
628 | searchElements: {
629 | inputEl: postContainer.find('input.title'),
630 | resultEl: postContainer.find('.quick-search-container'),
631 | },
632 | searchOptions: {
633 | composer: 1,
634 | },
635 | hideOnNoMatches: true,
636 | hideDuringSearch: true,
637 | });
638 | }
639 |
640 | function activate(post_uuid) {
641 | if (composer.active && composer.active !== post_uuid) {
642 | composer.minimize(composer.active);
643 | }
644 |
645 | composer.active = post_uuid;
646 | const postContainer = $('.composer[data-uuid="' + post_uuid + '"]');
647 | postContainer.css('visibility', 'visible');
648 | $(window).trigger('action:composer.activate', {
649 | post_uuid: post_uuid,
650 | postContainer: postContainer,
651 | });
652 | }
653 |
654 | function focusElements(postContainer) {
655 | setTimeout(function () {
656 | var title = postContainer.find('input.title');
657 |
658 | if (title.length) {
659 | title.focus();
660 | } else {
661 | postContainer.find('textarea').focus().putCursorAtEnd();
662 | }
663 | }, 20);
664 | }
665 |
666 | async function post(post_uuid) {
667 | var postData = composer.posts[post_uuid];
668 | var postContainer = $('.composer[data-uuid="' + post_uuid + '"]');
669 | var handleEl = postContainer.find('.handle');
670 | var titleEl = postContainer.find('.title');
671 | var bodyEl = postContainer.find('textarea');
672 | var thumbEl = postContainer.find('input#topic-thumb-url');
673 | var onComposeRoute = postData.hasOwnProperty('template') && postData.template.compose === true;
674 | const submitBtn = postContainer.find('.composer-submit');
675 |
676 | titleEl.val(titleEl.val().trim());
677 | bodyEl.val(utils.rtrim(bodyEl.val()));
678 | if (thumbEl.length) {
679 | thumbEl.val(thumbEl.val().trim());
680 | }
681 |
682 | var action = postData.action;
683 |
684 | var checkTitle = (postData.hasOwnProperty('cid') || parseInt(postData.pid, 10)) && postContainer.find('input.title').length;
685 | var isCategorySelected = !checkTitle || (checkTitle && postData.cid);
686 |
687 | // Specifically for checking title/body length via plugins
688 | var payload = {
689 | post_uuid: post_uuid,
690 | postData: postData,
691 | postContainer: postContainer,
692 | titleEl: titleEl,
693 | titleLen: titleEl.val().length,
694 | bodyEl: bodyEl,
695 | bodyLen: bodyEl.val().length,
696 | };
697 |
698 | await hooks.fire('filter:composer.check', payload);
699 | $(window).trigger('action:composer.check', payload);
700 |
701 | if (payload.error) {
702 | return composerAlert(post_uuid, payload.error);
703 | }
704 |
705 | if (uploads.inProgress[post_uuid] && uploads.inProgress[post_uuid].length) {
706 | return composerAlert(post_uuid, '[[error:still-uploading]]');
707 | } else if (checkTitle && payload.titleLen < parseInt(config.minimumTitleLength, 10)) {
708 | return composerAlert(post_uuid, '[[error:title-too-short, ' + config.minimumTitleLength + ']]');
709 | } else if (checkTitle && payload.titleLen > parseInt(config.maximumTitleLength, 10)) {
710 | return composerAlert(post_uuid, '[[error:title-too-long, ' + config.maximumTitleLength + ']]');
711 | } else if (action === 'topics.post' && !isCategorySelected) {
712 | return composerAlert(post_uuid, '[[error:category-not-selected]]');
713 | } else if (payload.bodyLen < parseInt(config.minimumPostLength, 10)) {
714 | return composerAlert(post_uuid, '[[error:content-too-short, ' + config.minimumPostLength + ']]');
715 | } else if (payload.bodyLen > parseInt(config.maximumPostLength, 10)) {
716 | return composerAlert(post_uuid, '[[error:content-too-long, ' + config.maximumPostLength + ']]');
717 | } else if (checkTitle && !tags.isEnoughTags(post_uuid)) {
718 | return composerAlert(post_uuid, '[[error:not-enough-tags, ' + tags.minTagCount() + ']]');
719 | } else if (scheduler.isActive() && scheduler.getTimestamp() <= Date.now()) {
720 | return composerAlert(post_uuid, '[[error:scheduling-to-past]]');
721 | }
722 |
723 | let composerData = {
724 | uuid: post_uuid,
725 | };
726 | let method = 'post';
727 | let route = '';
728 |
729 | if (action === 'topics.post') {
730 | route = '/topics';
731 | composerData = {
732 | ...composerData,
733 | handle: handleEl ? handleEl.val() : undefined,
734 | title: titleEl.val(),
735 | content: bodyEl.val(),
736 | thumb: thumbEl.val() || '',
737 | cid: categoryList.getSelectedCid(),
738 | tags: tags.getTags(post_uuid),
739 | timestamp: scheduler.getTimestamp(),
740 | };
741 | } else if (action === 'posts.reply') {
742 | route = `/topics/${postData.tid}`;
743 | composerData = {
744 | ...composerData,
745 | tid: postData.tid,
746 | handle: handleEl ? handleEl.val() : undefined,
747 | content: bodyEl.val(),
748 | toPid: postData.toPid,
749 | };
750 | } else if (action === 'posts.edit') {
751 | method = 'put';
752 | route = `/posts/${encodeURIComponent(postData.pid)}`;
753 | composerData = {
754 | ...composerData,
755 | pid: postData.pid,
756 | handle: handleEl ? handleEl.val() : undefined,
757 | content: bodyEl.val(),
758 | title: titleEl.val(),
759 | thumb: thumbEl.val() || '',
760 | tags: tags.getTags(post_uuid),
761 | timestamp: scheduler.getTimestamp(),
762 | };
763 | }
764 | var submitHookData = {
765 | composerEl: postContainer,
766 | action: action,
767 | composerData: composerData,
768 | postData: postData,
769 | redirect: true,
770 | };
771 |
772 | await hooks.fire('filter:composer.submit', submitHookData);
773 | hooks.fire('action:composer.submit', Object.freeze(submitHookData));
774 |
775 | // Minimize composer (and set textarea as readonly) while submitting
776 | var taskbarIconEl = $('#taskbar .composer[data-uuid="' + post_uuid + '"] i');
777 | var textareaEl = postContainer.find('.write');
778 | taskbarIconEl.removeClass('fa-plus').addClass('fa-circle-o-notch fa-spin');
779 | composer.minimize(post_uuid);
780 | textareaEl.prop('readonly', true);
781 |
782 | api[method](route, composerData)
783 | .then((data) => {
784 | submitBtn.removeAttr('disabled');
785 | postData.submitted = true;
786 |
787 | composer.discard(post_uuid);
788 | drafts.removeDraft(postData.save_id);
789 |
790 | if (data.queued) {
791 | alerts.alert({
792 | type: 'success',
793 | title: '[[global:alert.success]]',
794 | message: data.message,
795 | timeout: 10000,
796 | clickfn: function () {
797 | ajaxify.go(`/post-queue/${data.id}`);
798 | },
799 | });
800 | } else if (action === 'topics.post') {
801 | if (submitHookData.redirect) {
802 | ajaxify.go('topic/' + data.slug, undefined, (onComposeRoute || composer.bsEnvironment === 'xs' || composer.bsEnvironment === 'sm'));
803 | }
804 | } else if (action === 'posts.reply') {
805 | if (onComposeRoute || composer.bsEnvironment === 'xs' || composer.bsEnvironment === 'sm') {
806 | window.history.back();
807 | } else if (submitHookData.redirect &&
808 | ((ajaxify.data.template.name !== 'topic') ||
809 | (ajaxify.data.template.topic && parseInt(postData.tid, 10) !== parseInt(ajaxify.data.tid, 10)))
810 | ) {
811 | ajaxify.go('post/' + data.pid);
812 | }
813 | } else {
814 | removeComposerHistory();
815 | }
816 |
817 | hooks.fire('action:composer.' + action, { composerData: composerData, data: data });
818 | })
819 | .catch((err) => {
820 | // Restore composer on error
821 | composer.load(post_uuid);
822 | textareaEl.prop('readonly', false);
823 | if (err.message === '[[error:email-not-confirmed]]') {
824 | return messagesModule.showEmailConfirmWarning(err.message);
825 | }
826 | composerAlert(post_uuid, err.message);
827 | });
828 | }
829 |
830 | function onShow() {
831 | $('html').addClass('composing');
832 | }
833 |
834 | function onHide() {
835 | $('#content').css({ paddingBottom: 0 });
836 | $('html').removeClass('composing');
837 | app.toggleNavbar(true);
838 | formatting.exitFullscreen();
839 | }
840 |
841 | composer.discard = function (post_uuid) {
842 | if (composer.posts[post_uuid]) {
843 | var postData = composer.posts[post_uuid];
844 | var postContainer = $('.composer[data-uuid="' + post_uuid + '"]');
845 | postContainer.remove();
846 | drafts.removeDraft(postData.save_id);
847 | topicThumbs.deleteAll(post_uuid);
848 |
849 | taskbar.discard('composer', post_uuid);
850 | $('[data-action="post"]').removeAttr('disabled');
851 |
852 | hooks.fire('action:composer.discard', {
853 | post_uuid: post_uuid,
854 | postData: postData,
855 | });
856 | delete composer.posts[post_uuid];
857 | composer.active = undefined;
858 | }
859 | scheduler.reset();
860 | onHide();
861 | };
862 |
863 | // Alias to .discard();
864 | composer.close = composer.discard;
865 |
866 | composer.minimize = function (post_uuid) {
867 | var postContainer = $('.composer[data-uuid="' + post_uuid + '"]');
868 | postContainer.css('visibility', 'hidden');
869 | composer.active = undefined;
870 | taskbar.minimize('composer', post_uuid);
871 | $(window).trigger('action:composer.minimize', {
872 | post_uuid: post_uuid,
873 | });
874 |
875 | onHide();
876 | };
877 |
878 | composer.minimizeActive = function () {
879 | if (composer.active) {
880 | composer.miminize(composer.active);
881 | }
882 | };
883 |
884 | composer.updateThumbCount = function (uuid, postContainer) {
885 | const composerObj = composer.posts[uuid];
886 | if (composerObj.action === 'topics.post' || (composerObj.action === 'posts.edit' && composerObj.isMain)) {
887 | const calls = [
888 | topicThumbs.get(uuid),
889 | ];
890 | if (composerObj.pid) {
891 | calls.push(topicThumbs.getByPid(composerObj.pid));
892 | }
893 | Promise.all(calls).then((thumbs) => {
894 | const thumbCount = thumbs.flat().length;
895 | const formatEl = postContainer.find('[data-format="thumbs"]');
896 | formatEl.find('.badge')
897 | .text(thumbCount)
898 | .toggleClass('hidden', !thumbCount);
899 | });
900 | }
901 | };
902 |
903 | return composer;
904 | });
905 |
--------------------------------------------------------------------------------
/static/lib/composer/autocomplete.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | define('composer/autocomplete', [
4 | 'composer/preview', 'autocomplete',
5 | ], function (preview, Autocomplete) {
6 | var autocomplete = {
7 | _active: {},
8 | };
9 |
10 | $(window).on('action:composer.discard', function (evt, data) {
11 | if (autocomplete._active.hasOwnProperty(data.post_uuid)) {
12 | autocomplete._active[data.post_uuid].destroy();
13 | delete autocomplete._active[data.post_uuid];
14 | }
15 | });
16 |
17 | autocomplete.init = function (postContainer, post_uuid) {
18 | var element = postContainer.find('.write');
19 | var dropdownClass = 'composer-autocomplete-dropdown-' + post_uuid;
20 | var timer;
21 |
22 | if (!element.length) {
23 | /**
24 | * Some composers do their own thing before calling autocomplete.init() again.
25 | * One reason is because they want to override the textarea with their own element.
26 | * In those scenarios, they don't specify the "write" class, and this conditional
27 | * looks for that and stops the autocomplete init process.
28 | */
29 | return;
30 | }
31 |
32 | var data = {
33 | element: element,
34 | strategies: [],
35 | options: {
36 | style: {
37 | 'z-index': 20000,
38 | },
39 | className: dropdownClass + ' dropdown-menu textcomplete-dropdown',
40 | },
41 | };
42 |
43 | element.on('keyup', function () {
44 | clearTimeout(timer);
45 | timer = setTimeout(function () {
46 | var dropdown = document.querySelector('.' + dropdownClass);
47 | if (dropdown) {
48 | var pos = dropdown.getBoundingClientRect();
49 |
50 | var margin = parseFloat(dropdown.style.marginTop, 10) || 0;
51 |
52 | var offset = window.innerHeight + margin - 10 - pos.bottom;
53 | dropdown.style.marginTop = Math.min(offset, 0) + 'px';
54 | }
55 | }, 0);
56 | });
57 |
58 | $(window).trigger('composer:autocomplete:init', data);
59 |
60 | autocomplete._active[post_uuid] = Autocomplete.setup(data);
61 |
62 | data.element.on('textComplete:select', function () {
63 | preview.render(postContainer);
64 | });
65 | };
66 |
67 | return autocomplete;
68 | });
69 |
--------------------------------------------------------------------------------
/static/lib/composer/categoryList.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | define('composer/categoryList', [
4 | 'categorySelector', 'taskbar', 'api',
5 | ], function (categorySelector, taskbar, api) {
6 | var categoryList = {};
7 |
8 | var selector;
9 |
10 | categoryList.init = function (postContainer, postData) {
11 | var listContainer = postContainer.find('.category-list-container');
12 | if (!listContainer.length) {
13 | return;
14 | }
15 |
16 | postContainer.on('action:composer.resize', function () {
17 | toggleDropDirection(postContainer);
18 | });
19 |
20 | categoryList.updateTaskbar(postContainer, postData);
21 |
22 | selector = categorySelector.init(listContainer.find('[component="category-selector"]'), {
23 | privilege: 'topics:create',
24 | states: ['watching', 'tracking', 'notwatching', 'ignoring'],
25 | onSelect: function (selectedCategory) {
26 | if (postData.hasOwnProperty('cid')) {
27 | changeCategory(postContainer, postData, selectedCategory);
28 | }
29 | },
30 | });
31 | if (!selector) {
32 | return;
33 | }
34 | if (postData.cid && postData.category) {
35 | selector.selectedCategory = { cid: postData.cid, name: postData.category.name };
36 | } else if (ajaxify.data.template.compose && ajaxify.data.selectedCategory) {
37 | // separate composer route
38 | selector.selectedCategory = { cid: ajaxify.data.cid, name: ajaxify.data.selectedCategory };
39 | }
40 |
41 | // this is the mobile category selector
42 | postContainer.find('.category-name')
43 | .translateHtml(selector.selectedCategory ? selector.selectedCategory.name : '[[modules:composer.select-category]]')
44 | .on('click', function () {
45 | categorySelector.modal({
46 | privilege: 'topics:create',
47 | states: ['watching', 'tracking', 'notwatching', 'ignoring'],
48 | openOnLoad: true,
49 | showLinks: false,
50 | onSubmit: function (selectedCategory) {
51 | postContainer.find('.category-name').text(selectedCategory.name);
52 | selector.selectCategory(selectedCategory.cid);
53 | if (postData.hasOwnProperty('cid')) {
54 | changeCategory(postContainer, postData, selectedCategory);
55 | }
56 | },
57 | });
58 | });
59 |
60 | toggleDropDirection(postContainer);
61 | };
62 |
63 | function toggleDropDirection(postContainer) {
64 | postContainer.find('.category-list-container [component="category-selector"]').toggleClass('dropup', postContainer.outerHeight() < $(window).height() / 2);
65 | }
66 |
67 | categoryList.getSelectedCid = function () {
68 | var selectedCategory;
69 | if (selector) {
70 | selectedCategory = selector.getSelectedCategory();
71 | }
72 | return selectedCategory ? selectedCategory.cid : 0;
73 | };
74 |
75 | categoryList.updateTaskbar = function (postContainer, postData) {
76 | if (parseInt(postData.cid, 10)) {
77 | api.get(`/categories/${postData.cid}`, {}).then(function (category) {
78 | updateTaskbarByCategory(postContainer, category);
79 | });
80 | }
81 | };
82 |
83 | function updateTaskbarByCategory(postContainer, category) {
84 | if (category) {
85 | var uuid = postContainer.attr('data-uuid');
86 | taskbar.update('composer', uuid, {
87 | image: category.backgroundImage,
88 | color: category.color,
89 | 'background-color': category.bgColor,
90 | icon: category.icon && category.icon.slice(3),
91 | });
92 | }
93 | }
94 |
95 | async function changeCategory(postContainer, postData, selectedCategory) {
96 | postData.cid = selectedCategory.cid;
97 | const categoryData = await window.fetch(`${config.relative_path}/api/category/${encodeURIComponent(selectedCategory.cid)}`).then(r => r.json());
98 | postData.category = categoryData;
99 | updateTaskbarByCategory(postContainer, categoryData);
100 | require(['composer/scheduler', 'composer/tags', 'composer/post-queue'], function (scheduler, tags, postQueue) {
101 | scheduler.onChangeCategory(categoryData);
102 | tags.onChangeCategory(postContainer, postData, selectedCategory.cid, categoryData);
103 | postQueue.onChangeCategory(postContainer, postData);
104 |
105 | $(window).trigger('action:composer.changeCategory', {
106 | postContainer: postContainer,
107 | postData: postData,
108 | selectedCategory: selectedCategory,
109 | categoryData: categoryData,
110 | });
111 | });
112 | }
113 |
114 | return categoryList;
115 | });
116 |
--------------------------------------------------------------------------------
/static/lib/composer/controls.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | define('composer/controls', ['composer/preview'], function (preview) {
4 | var controls = {};
5 |
6 | /** ********************************************** */
7 | /* Rich Textarea Controls */
8 | /** ********************************************** */
9 | controls.insertIntoTextarea = function (textarea, value) {
10 | var payload = {
11 | context: this,
12 | textarea: textarea,
13 | value: value,
14 | preventDefault: false,
15 | };
16 | $(window).trigger('action:composer.insertIntoTextarea', payload);
17 |
18 | if (payload.preventDefault) {
19 | return;
20 | }
21 |
22 | var $textarea = $(payload.textarea);
23 | var currentVal = $textarea.val();
24 | var postContainer = $textarea.parents('[component="composer"]');
25 |
26 | $textarea.val(
27 | currentVal.slice(0, payload.textarea.selectionStart) +
28 | payload.value +
29 | currentVal.slice(payload.textarea.selectionStart)
30 | );
31 |
32 | preview.render(postContainer);
33 | };
34 |
35 | controls.replaceSelectionInTextareaWith = function (textarea, value) {
36 | var payload = {
37 | context: this,
38 | textarea: textarea,
39 | value: value,
40 | preventDefault: false,
41 | };
42 | $(window).trigger('action:composer.replaceSelectionInTextareaWith', payload);
43 |
44 | if (payload.preventDefault) {
45 | return;
46 | }
47 |
48 | var $textarea = $(payload.textarea);
49 | var currentVal = $textarea.val();
50 | var postContainer = $textarea.parents('[component="composer"]');
51 |
52 | $textarea.val(
53 | currentVal.slice(0, payload.textarea.selectionStart) +
54 | payload.value +
55 | currentVal.slice(payload.textarea.selectionEnd)
56 | );
57 |
58 | preview.render(postContainer);
59 | };
60 |
61 | controls.wrapSelectionInTextareaWith = function (textarea, leading, trailing) {
62 | var payload = {
63 | context: this,
64 | textarea: textarea,
65 | leading: leading,
66 | trailing: trailing,
67 | preventDefault: false,
68 | };
69 | $(window).trigger('action:composer.wrapSelectionInTextareaWith', payload);
70 |
71 | if (payload.preventDefault) {
72 | return;
73 | }
74 |
75 | if (trailing === undefined) {
76 | trailing = leading;
77 | }
78 |
79 | var $textarea = $(textarea);
80 | var currentVal = $textarea.val();
81 |
82 | var matches = /^(\s*)([\s\S]*?)(\s*)$/.exec(currentVal.slice(textarea.selectionStart, textarea.selectionEnd));
83 |
84 | if (!matches[2]) {
85 | // selection is entirely whitespace
86 | matches = [null, '', currentVal.slice(textarea.selectionStart, textarea.selectionEnd), ''];
87 | }
88 |
89 | $textarea.val(
90 | currentVal.slice(0, textarea.selectionStart) +
91 | matches[1] +
92 | leading +
93 | matches[2] +
94 | trailing +
95 | matches[3] +
96 | currentVal.slice(textarea.selectionEnd)
97 | );
98 |
99 | return [matches[1].length, matches[3].length];
100 | };
101 |
102 | controls.updateTextareaSelection = function (textarea, start, end) {
103 | var payload = {
104 | context: this,
105 | textarea: textarea,
106 | start: start,
107 | end: end,
108 | preventDefault: false,
109 | };
110 | $(window).trigger('action:composer.updateTextareaSelection', payload);
111 |
112 | if (payload.preventDefault) {
113 | return;
114 | }
115 |
116 | textarea.setSelectionRange(payload.start, payload.end);
117 | $(payload.textarea).focus();
118 | };
119 |
120 | controls.getBlockData = function (textareaEl, query, selectionStart) {
121 | // Determines whether the cursor is sitting inside a block-type element (bold, italic, etc.)
122 | var value = textareaEl.value;
123 | query = query.replace(/[-[\]/{}()*+?.\\^$|]/g, '\\$&');
124 | var regex = new RegExp(query, 'g');
125 | var match;
126 | var matchIndices = [];
127 | var payload;
128 |
129 | // Isolate the line the cursor is on
130 | value = value.split('\n').reduce(function (memo, line) {
131 | if (memo !== null) {
132 | return memo;
133 | }
134 |
135 | memo = selectionStart <= line.length ? line : null;
136 |
137 | if (memo === null) {
138 | selectionStart -= (line.length + 1);
139 | }
140 |
141 | return memo;
142 | }, null);
143 |
144 | // Find query characters and determine return payload
145 | while ((match = regex.exec(value)) !== null) {
146 | matchIndices.push(match.index);
147 | }
148 |
149 | payload = {
150 | in: !!(matchIndices.reduce(function (memo, cur) {
151 | if (selectionStart >= cur + 2) {
152 | memo += 1;
153 | }
154 |
155 | return memo;
156 | }, 0) % 2),
157 | atEnd: matchIndices.reduce(function (memo, cur) {
158 | if (memo) {
159 | return memo;
160 | }
161 |
162 | return selectionStart === cur;
163 | }, false),
164 | };
165 |
166 | payload.atEnd = payload.in ? payload.atEnd : false;
167 | return payload;
168 | };
169 |
170 | return controls;
171 | });
172 |
--------------------------------------------------------------------------------
/static/lib/composer/drafts.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | define('composer/drafts', ['api', 'alerts'], function (api, alerts) {
4 | const drafts = {};
5 | const draftSaveDelay = 1000;
6 | drafts.init = function (postContainer, postData) {
7 | const draftIconEl = postContainer.find('.draft-icon');
8 | const uuid = postContainer.attr('data-uuid');
9 | function doSaveDraft() {
10 | // check if composer is still around,
11 | // it might have been gone by the time this timeout triggers
12 | if (!$(`[component="composer"][data-uuid="${uuid}"]`).length) {
13 | return;
14 | }
15 |
16 | if (!postData.save_id) {
17 | postData.save_id = utils.generateSaveId(app.user.uid);
18 | }
19 | // Post is modified, save to list of opened drafts
20 | drafts.addToDraftList('available', postData.save_id);
21 | drafts.addToDraftList('open', postData.save_id);
22 | saveDraft(postContainer, draftIconEl, postData);
23 | }
24 |
25 | postContainer.on('keyup', 'textarea, input.handle, input.title', utils.debounce(doSaveDraft, draftSaveDelay));
26 | postContainer.on('click', 'input[type="checkbox"]', utils.debounce(doSaveDraft, draftSaveDelay));
27 | postContainer.on('click', '[component="category/list"] [data-cid]', utils.debounce(doSaveDraft, draftSaveDelay));
28 | postContainer.on('itemAdded', '.tags', utils.debounce(doSaveDraft, draftSaveDelay));
29 | postContainer.on('thumb.uploaded', doSaveDraft);
30 |
31 | draftIconEl.on('animationend', function () {
32 | $(this).toggleClass('active', false);
33 | });
34 |
35 | $(window).on('unload', function () {
36 | // remove all drafts from the open list
37 | const open = drafts.getList('open');
38 | if (open.length) {
39 | open.forEach(save_id => drafts.removeFromDraftList('open', save_id));
40 | }
41 | });
42 |
43 | drafts.migrateGuest();
44 | drafts.migrateThumbs(...arguments);
45 | };
46 |
47 | function getStorage(uid) {
48 | return parseInt(uid, 10) > 0 ? localStorage : sessionStorage;
49 | }
50 |
51 | drafts.get = function (save_id) {
52 | if (!save_id) {
53 | return null;
54 | }
55 | const uid = save_id.split(':')[1];
56 | const storage = getStorage(uid);
57 | try {
58 | const draftJson = storage.getItem(save_id);
59 | const draft = JSON.parse(draftJson) || null;
60 | if (!draft) {
61 | throw new Error(`can't parse draft json for ${save_id}`);
62 | }
63 | draft.save_id = save_id;
64 | if (draft.timestamp) {
65 | draft.timestampISO = utils.toISOString(draft.timestamp);
66 | }
67 | $(window).trigger('action:composer.drafts.get', {
68 | save_id: save_id,
69 | draft: draft,
70 | storage: storage,
71 | });
72 | return draft;
73 | } catch (e) {
74 | console.warn(`[composer/drafts] Could not get draft ${save_id}, removing`);
75 | drafts.removeFromDraftList('available');
76 | drafts.removeFromDraftList('open');
77 | return null;
78 | }
79 | };
80 |
81 | function saveDraft(postContainer, draftIconEl, postData) {
82 | if (canSave(app.user.uid ? 'localStorage' : 'sessionStorage') && postData && postData.save_id && postContainer.length) {
83 | const titleEl = postContainer.find('input.title');
84 | const title = titleEl && titleEl.length && titleEl.val();
85 | const raw = postContainer.find('textarea').val();
86 | const storage = getStorage(app.user.uid);
87 |
88 | if (raw.length || (title && title.length)) {
89 | const draftData = {
90 | save_id: postData.save_id,
91 | action: postData.action,
92 | text: raw,
93 | uuid: postContainer.attr('data-uuid'),
94 | timestamp: Date.now(),
95 | };
96 |
97 | if (postData.action === 'topics.post') {
98 | // New topic only
99 | const tags = postContainer.find('input.tags').val();
100 | draftData.tags = tags;
101 | draftData.title = title;
102 | draftData.cid = postData.cid;
103 | } else if (postData.action === 'posts.reply') {
104 | // new reply only
105 | draftData.title = postData.title;
106 | draftData.tid = postData.tid;
107 | draftData.toPid = postData.toPid;
108 | } else if (postData.action === 'posts.edit') {
109 | draftData.pid = postData.pid;
110 | draftData.title = title || postData.title;
111 | }
112 | if (!app.user.uid) {
113 | draftData.handle = postContainer.find('input.handle').val();
114 | }
115 |
116 | // save all draft data into single item as json
117 | storage.setItem(postData.save_id, JSON.stringify(draftData));
118 |
119 | $(window).trigger('action:composer.drafts.save', {
120 | storage: storage,
121 | postData: postData,
122 | postContainer: postContainer,
123 | });
124 | draftIconEl.toggleClass('active', true);
125 | } else {
126 | drafts.removeDraft(postData.save_id);
127 | }
128 | }
129 | }
130 |
131 | drafts.removeDraft = function (save_id) {
132 | if (!save_id) {
133 | return;
134 | }
135 |
136 | // Remove save_id from list of open and available drafts
137 | drafts.removeFromDraftList('available', save_id);
138 | drafts.removeFromDraftList('open', save_id);
139 | const uid = save_id.split(':')[1];
140 | const storage = getStorage(uid);
141 | storage.removeItem(save_id);
142 |
143 | $(window).trigger('action:composer.drafts.remove', {
144 | storage: storage,
145 | save_id: save_id,
146 | });
147 | };
148 |
149 | drafts.getList = function (set) {
150 | try {
151 | const draftIds = localStorage.getItem(`drafts:${set}`);
152 | return JSON.parse(draftIds) || [];
153 | } catch (e) {
154 | console.warn('[composer/drafts] Could not read list of available drafts');
155 | return [];
156 | }
157 | };
158 |
159 | drafts.addToDraftList = function (set, save_id) {
160 | if (!canSave(app.user.uid ? 'localStorage' : 'sessionStorage') || !save_id) {
161 | return;
162 | }
163 | const list = drafts.getList(set);
164 | if (!list.includes(save_id)) {
165 | list.push(save_id);
166 | localStorage.setItem('drafts:' + set, JSON.stringify(list));
167 | }
168 | };
169 |
170 | drafts.removeFromDraftList = function (set, save_id) {
171 | if (!canSave(app.user.uid ? 'localStorage' : 'sessionStorage') || !save_id) {
172 | return;
173 | }
174 | const list = drafts.getList(set);
175 | if (list.includes(save_id)) {
176 | list.splice(list.indexOf(save_id), 1);
177 | localStorage.setItem('drafts:' + set, JSON.stringify(list));
178 | }
179 | };
180 |
181 | drafts.migrateGuest = function () {
182 | // If any drafts are made while as guest, and user then logs in, assume control of those drafts
183 | if (canSave('localStorage') && app.user.uid) {
184 | // composer::
185 | const test = /^composer:\d+:\d$/;
186 | const keys = Object.keys(sessionStorage).filter(function (key) {
187 | return test.test(key);
188 | });
189 | const migrated = new Set([]);
190 | const renamed = keys.map(function (key) {
191 | const parts = key.split(':');
192 | parts[1] = app.user.uid;
193 |
194 | migrated.add(parts.join(':'));
195 | return parts.join(':');
196 | });
197 |
198 | keys.forEach(function (key, idx) {
199 | localStorage.setItem(renamed[idx], sessionStorage.getItem(key));
200 | sessionStorage.removeItem(key);
201 | });
202 |
203 | migrated.forEach(function (save_id) {
204 | drafts.addToDraftList('available', save_id);
205 | });
206 |
207 | return migrated;
208 | }
209 | };
210 |
211 | drafts.migrateThumbs = function (postContainer, postData) {
212 | if (!app.uid) {
213 | return;
214 | }
215 |
216 | // If any thumbs were uploaded, migrate them to this new composer's uuid
217 | const newUUID = postContainer.attr('data-uuid');
218 | const draft = drafts.get(postData.save_id);
219 |
220 | if (draft && draft.uuid) {
221 | api.put(`/topics/${draft.uuid}/thumbs`, {
222 | tid: newUUID,
223 | }).then(() => {
224 | require(['composer'], function (composer) {
225 | composer.updateThumbCount(newUUID, postContainer);
226 | });
227 | });
228 | }
229 | };
230 |
231 | drafts.listAvailable = function () {
232 | const available = drafts.getList('available');
233 | return available.map(drafts.get).filter(Boolean);
234 | };
235 |
236 | drafts.getAvailableCount = function () {
237 | return drafts.listAvailable().length;
238 | };
239 |
240 | drafts.open = function (save_id) {
241 | if (!save_id) {
242 | return;
243 | }
244 | const draft = drafts.get(save_id);
245 | openComposer(save_id, draft);
246 | };
247 |
248 | drafts.loadOpen = function () {
249 | if (ajaxify.data.template.login || ajaxify.data.template.register || (config.hasOwnProperty('openDraftsOnPageLoad') && !config.openDraftsOnPageLoad)) {
250 | return;
251 | }
252 | // Load drafts if they were open
253 | const available = drafts.getList('available');
254 | const open = drafts.getList('open');
255 |
256 | if (available.length) {
257 | // Deconstruct each save_id and open up composer
258 | available.forEach(function (save_id) {
259 | if (!save_id || open.includes(save_id)) {
260 | return;
261 | }
262 | const draft = drafts.get(save_id);
263 | if (!draft || (!draft.text && !draft.title)) {
264 | drafts.removeFromDraftList('available', save_id);
265 | drafts.removeFromDraftList('open', save_id);
266 | return;
267 | }
268 | openComposer(save_id, draft);
269 | });
270 | }
271 | };
272 |
273 | function openComposer(save_id, draft) {
274 | const saveObj = save_id.split(':');
275 | const uid = saveObj[1];
276 | // Don't open other peoples' drafts
277 | if (parseInt(app.user.uid, 10) !== parseInt(uid, 10)) {
278 | return;
279 | }
280 | require(['composer'], function (composer) {
281 | if (draft.action === 'topics.post') {
282 | composer.newTopic({
283 | save_id: draft.save_id,
284 | cid: draft.cid,
285 | handle: app.user && app.user.uid ? undefined : utils.escapeHTML(draft.handle),
286 | title: utils.escapeHTML(draft.title),
287 | body: draft.text,
288 | tags: String(draft.tags || '').split(','),
289 | });
290 | } else if (draft.action === 'posts.reply') {
291 | api.get('/topics/' + draft.tid, {}, function (err, topicObj) {
292 | if (err) {
293 | return alerts.error(err);
294 | }
295 |
296 | composer.newReply({
297 | save_id: draft.save_id,
298 | tid: draft.tid,
299 | toPid: draft.toPid,
300 | title: topicObj.title,
301 | body: draft.text,
302 | });
303 | });
304 | } else if (draft.action === 'posts.edit') {
305 | composer.editPost({
306 | save_id: draft.save_id,
307 | pid: draft.pid,
308 | title: draft.title ? utils.escapeHTML(draft.title) : undefined,
309 | body: draft.text,
310 | });
311 | }
312 | });
313 | }
314 |
315 | // Feature detection courtesy of: https://developer.mozilla.org/en-US/docs/Web/API/Web_Storage_API/Using_the_Web_Storage_API
316 | function canSave(type) {
317 | var storage;
318 | try {
319 | storage = window[type];
320 | var x = '__storage_test__';
321 | storage.setItem(x, x);
322 | storage.removeItem(x);
323 | return true;
324 | } catch (e) {
325 | return e instanceof DOMException && (
326 | // everything except Firefox
327 | e.code === 22 ||
328 | // Firefox
329 | e.code === 1014 ||
330 | // test name field too, because code might not be present
331 | // everything except Firefox
332 | e.name === 'QuotaExceededError' ||
333 | // Firefox
334 | e.name === 'NS_ERROR_DOM_QUOTA_REACHED') &&
335 | // acknowledge QuotaExceededError only if there's something already stored
336 | (storage && storage.length !== 0);
337 | }
338 | }
339 |
340 | return drafts;
341 | });
342 |
--------------------------------------------------------------------------------
/static/lib/composer/formatting.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | define('composer/formatting', [
4 | 'composer/preview', 'composer/resize', 'topicThumbs', 'screenfull',
5 | ], function (preview, resize, topicThumbs, screenfull) {
6 | var formatting = {};
7 |
8 | var formattingDispatchTable = {
9 | picture: function () {
10 | var postContainer = this;
11 | postContainer.find('#files')
12 | .attr('accept', 'image/*')
13 | .click();
14 | },
15 |
16 | upload: function () {
17 | var postContainer = this;
18 | postContainer.find('#files')
19 | .attr('accept', '')
20 | .click();
21 | },
22 |
23 | thumbs: function () {
24 | formatting.exitFullscreen();
25 | var postContainer = this;
26 | require(['composer'], function (composer) {
27 | const uuid = postContainer.get(0).getAttribute('data-uuid');
28 | const composerObj = composer.posts[uuid];
29 |
30 | if (composerObj.action === 'topics.post' || (composerObj.action === 'posts.edit' && composerObj.isMain)) {
31 | topicThumbs.modal.open({ id: uuid, pid: composerObj.pid }).then(() => {
32 | postContainer.trigger('thumb.uploaded'); // toggle draft save
33 |
34 | // Update client-side with count
35 | composer.updateThumbCount(uuid, postContainer);
36 | });
37 | }
38 | });
39 | },
40 |
41 | tags: function () {
42 | var postContainer = this;
43 | postContainer.find('.tags-container').toggleClass('hidden');
44 | },
45 |
46 | zen: function () {
47 | var postContainer = this;
48 | $(window).one('resize', function () {
49 | function onResize() {
50 | if (!screenfull.isFullscreen) {
51 | app.toggleNavbar(true);
52 | $('html').removeClass('zen-mode');
53 | resize.reposition(postContainer);
54 | $(window).off('resize', onResize);
55 | }
56 | }
57 |
58 | if (screenfull.isFullscreen) {
59 | app.toggleNavbar(false);
60 | $('html').addClass('zen-mode');
61 | postContainer.find('.write').focus();
62 |
63 | $(window).on('resize', onResize);
64 | $(window).one('action:composer.topics.post action:composer.posts.reply action:composer.posts.edit action:composer.discard', screenfull.exit);
65 | }
66 | });
67 |
68 | screenfull.toggle(postContainer.get(0));
69 | $(window).trigger('action:composer.fullscreen', { postContainer: postContainer });
70 | },
71 | };
72 |
73 | var buttons = [];
74 |
75 | formatting.exitFullscreen = function () {
76 | if (screenfull.isEnabled && screenfull.isFullscreen) {
77 | screenfull.exit();
78 | }
79 | };
80 |
81 | formatting.addComposerButtons = function () {
82 | const formattingBarEl = $('.formatting-bar');
83 | const fileForm = formattingBarEl.find('.formatting-group #fileForm');
84 | buttons.forEach((btn) => {
85 | let markup = ``;
86 | if (Array.isArray(btn.dropdownItems) && btn.dropdownItems.length) {
87 | markup = generateFormattingDropdown(btn);
88 | } else {
89 | markup = `
90 | -
91 |
95 |
96 | `;
97 | }
98 | fileForm.before(markup);
99 | });
100 |
101 | const els = formattingBarEl.find('.formatting-group>li');
102 | els.tooltip({
103 | container: '#content',
104 | animation: false,
105 | trigger: 'manual',
106 | }).on('mouseenter', function (ev) {
107 | const target = $(ev.target);
108 | const isDropdown = target.hasClass('dropdown-menu') || !!target.parents('.dropdown-menu').length;
109 | if (!isDropdown) {
110 | $(this).tooltip('show');
111 | }
112 | }).on('click mouseleave', function () {
113 | $(this).tooltip('hide');
114 | });
115 | };
116 |
117 | function generateBadgetHtml(btn) {
118 | let badgeHtml = '';
119 | if (btn.badge) {
120 | badgeHtml = ``;
121 | }
122 | return badgeHtml;
123 | }
124 |
125 | function generateFormattingDropdown(btn) {
126 | const dropdownItemsHtml = btn.dropdownItems.map(function (btn) {
127 | return `
128 | -
129 |
130 | ${btn.text}
131 | ${generateBadgetHtml(btn)}
132 |
133 |
134 | `;
135 | });
136 | return `
137 | -
138 |
141 |
144 |
145 | `;
146 | }
147 |
148 | formatting.addButton = function (iconClass, onClick, title, name) {
149 | name = name || iconClass.replace('fa fa-', '');
150 | formattingDispatchTable[name] = onClick;
151 | buttons.push({
152 | name,
153 | iconClass,
154 | title,
155 | });
156 | };
157 |
158 | formatting.addDropdown = function (data) {
159 | buttons.push({
160 | iconClass: data.iconClass,
161 | title: data.title,
162 | dropdownItems: data.dropdownItems,
163 | });
164 | data.dropdownItems.forEach((btn) => {
165 | if (btn.name && btn.onClick) {
166 | formattingDispatchTable[btn.name] = btn.onClick;
167 | }
168 | });
169 | };
170 |
171 | formatting.getDispatchTable = function () {
172 | return formattingDispatchTable;
173 | };
174 |
175 | formatting.addButtonDispatch = function (name, onClick) {
176 | formattingDispatchTable[name] = onClick;
177 | };
178 |
179 | formatting.addHandler = function (postContainer) {
180 | postContainer.on('click', '.formatting-bar [data-format]', function (event) {
181 | var format = $(this).attr('data-format');
182 | var textarea = $(this).parents('[component="composer"]').find('textarea')[0];
183 |
184 | if (formattingDispatchTable.hasOwnProperty(format)) {
185 | formattingDispatchTable[format].call(
186 | postContainer, textarea, textarea.selectionStart, textarea.selectionEnd, event
187 | );
188 | preview.render(postContainer);
189 | }
190 | });
191 | };
192 |
193 | return formatting;
194 | });
195 |
--------------------------------------------------------------------------------
/static/lib/composer/post-queue.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | define('composer/post-queue', [], function () {
4 | const postQueue = {};
5 |
6 | postQueue.showAlert = async function (postContainer, postData) {
7 | const alertEl = postContainer.find('[component="composer/post-queue/alert"]');
8 | if (!config.postQueue || app.user.isAdmin || app.user.isGlobalMod || app.user.isMod) {
9 | alertEl.remove();
10 | return;
11 | }
12 | const shouldQueue = await socket.emit('plugins.composer.shouldQueue', { postData: postData });
13 | alertEl.toggleClass('show', shouldQueue);
14 | alertEl.toggleClass('pe-none', !shouldQueue);
15 | };
16 |
17 | postQueue.onChangeCategory = async function (postContainer, postData) {
18 | if (!config.postQueue) {
19 | return;
20 | }
21 | postQueue.showAlert(postContainer, postData);
22 | };
23 |
24 | return postQueue;
25 | });
26 |
--------------------------------------------------------------------------------
/static/lib/composer/preview.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | define('composer/preview', ['hooks'], function (hooks) {
4 | var preview = {};
5 |
6 | preview.render = function (postContainer, callback) {
7 | callback = callback || function () {};
8 | if (!postContainer.find('.preview-container').is(':visible')) {
9 | return callback();
10 | }
11 |
12 | var textarea = postContainer.find('textarea');
13 |
14 | socket.emit('plugins.composer.renderPreview', textarea.val(), function (err, preview) {
15 | if (err) {
16 | return;
17 | }
18 | preview = $('' + preview + '
');
19 | preview.find('img:not(.not-responsive)').addClass('img-fluid');
20 | postContainer.find('.preview').html(preview);
21 | hooks.fire('action:composer.preview', { postContainer, preview });
22 | callback();
23 | });
24 | };
25 |
26 | preview.matchScroll = function (postContainer) {
27 | if (!postContainer.find('.preview-container').is(':visible')) {
28 | return;
29 | }
30 | var textarea = postContainer.find('textarea');
31 | var preview = postContainer.find('.preview');
32 |
33 | if (textarea.length && preview.length) {
34 | var diff = textarea[0].scrollHeight - textarea.height();
35 |
36 | if (diff === 0) {
37 | return;
38 | }
39 |
40 | var scrollPercent = textarea.scrollTop() / diff;
41 |
42 | preview.scrollTop(Math.max(preview[0].scrollHeight - preview.height(), 0) * scrollPercent);
43 | }
44 | };
45 |
46 | preview.handleToggler = function ($postContainer) {
47 | const postContainer = $postContainer.get(0);
48 | preview.env = utils.findBootstrapEnvironment();
49 | const isMobile = ['xs', 'sm'].includes(preview.env);
50 | const toggler = postContainer.querySelector('.formatting-bar [data-action="preview"]');
51 | const showText = toggler.querySelector('.show-text');
52 | const hideText = toggler.querySelector('.hide-text');
53 | const previewToggled = localStorage.getItem('composer:previewToggled');
54 | const hidePreviewOnOpen = config['composer-default'] && config['composer-default'].hidePreviewOnOpen === 'on';
55 | let show = !isMobile && (
56 | ((previewToggled === null && !hidePreviewOnOpen) || previewToggled === 'true')
57 | );
58 | const previewContainer = postContainer.querySelector('.preview-container');
59 | const writeContainer = postContainer.querySelector('.write-container');
60 |
61 | if (!toggler) {
62 | return;
63 | }
64 |
65 | function togglePreview(show) {
66 | if (isMobile) {
67 | previewContainer.classList.toggle('hide', false);
68 | writeContainer.classList.toggle('maximized', false);
69 |
70 | previewContainer.classList.toggle('d-none', !show);
71 | previewContainer.classList.toggle('d-flex', show);
72 | previewContainer.classList.toggle('w-100', show);
73 |
74 | writeContainer.classList.toggle('d-flex', !show);
75 | writeContainer.classList.toggle('d-none', show);
76 | writeContainer.classList.toggle('w-100', !show);
77 | } else {
78 | previewContainer.classList.toggle('hide', !show);
79 | writeContainer.classList.toggle('w-50', show);
80 | writeContainer.classList.toggle('w-100', !show);
81 | localStorage.setItem('composer:previewToggled', show);
82 | }
83 | showText.classList.toggle('hide', show);
84 | hideText.classList.toggle('hide', !show);
85 | if (show) {
86 | preview.render($postContainer);
87 | }
88 | preview.matchScroll($postContainer);
89 | }
90 | preview.toggle = togglePreview;
91 |
92 | toggler.addEventListener('click', (e) => {
93 | if (e.button !== 0) {
94 | return;
95 | }
96 |
97 | show = !show;
98 | togglePreview(show);
99 | });
100 |
101 | togglePreview(show);
102 | };
103 |
104 | return preview;
105 | });
106 |
--------------------------------------------------------------------------------
/static/lib/composer/resize.js:
--------------------------------------------------------------------------------
1 |
2 | 'use strict';
3 |
4 | define('composer/resize', ['taskbar'], function (taskbar) {
5 | var resize = {};
6 | var oldRatio = 0;
7 | var minimumRatio = 0.3;
8 | var snapMargin = 0.05;
9 | var smallMin = 768;
10 |
11 | var $body = $('body');
12 | var $window = $(window);
13 | var $headerMenu = $('[component="navbar"]');
14 | const content = document.getElementById('content');
15 |
16 | var header = $headerMenu[0];
17 |
18 | function getSavedRatio() {
19 | return localStorage.getItem('composer:resizeRatio') || 0.5;
20 | }
21 |
22 | function saveRatio(ratio) {
23 | localStorage.setItem('composer:resizeRatio', Math.min(ratio, 1));
24 | }
25 |
26 | function getBounds() {
27 | var headerRect;
28 | if (header) {
29 | headerRect = header.getBoundingClientRect();
30 | } else {
31 | // Mock data
32 | headerRect = { bottom: 0 };
33 | }
34 |
35 | var headerBottom = Math.max(headerRect.bottom, 0);
36 |
37 | var rect = {
38 | top: 0,
39 | left: 0,
40 | right: window.innerWidth,
41 | bottom: window.innerHeight,
42 | };
43 |
44 | rect.width = rect.right;
45 | rect.height = rect.bottom;
46 |
47 | rect.boundedTop = headerBottom;
48 | rect.boundedHeight = rect.bottom - headerBottom;
49 |
50 | return rect;
51 | }
52 |
53 | function doResize(postContainer, ratio) {
54 | var bounds = getBounds();
55 | var elem = postContainer[0];
56 | var style = window.getComputedStyle(elem);
57 |
58 | // Adjust minimumRatio for shorter viewports
59 | var minHeight = parseInt(style.getPropertyValue('min-height'), 10);
60 | var adjustedMinimum = Math.max(minHeight / window.innerHeight, minimumRatio);
61 |
62 | if (bounds.width >= smallMin) {
63 | const boundedDifference = (bounds.height - bounds.boundedHeight) / bounds.height;
64 | ratio = Math.min(Math.max(ratio, adjustedMinimum + boundedDifference), 1);
65 |
66 | var top = ratio * bounds.boundedHeight / bounds.height;
67 | elem.style.top = ((1 - top) * 100).toString() + '%';
68 |
69 | // Add some extra space at the bottom of the body so that
70 | // the user can still scroll to the last post w/ composer open
71 | var rect = elem.getBoundingClientRect();
72 | content.style.paddingBottom = (rect.bottom - rect.top).toString() + 'px';
73 | } else {
74 | elem.style.top = 0;
75 | content.style.paddingBottom = 0;
76 | }
77 |
78 | postContainer.ratio = ratio;
79 |
80 | taskbar.updateActive(postContainer.attr('data-uuid'));
81 | }
82 |
83 | var resizeIt = doResize;
84 | var raf = window.requestAnimationFrame ||
85 | window.webkitRequestAnimationFrame ||
86 | window.mozRequestAnimationFrame;
87 |
88 | if (raf) {
89 | resizeIt = function (postContainer, ratio) {
90 | raf(function () {
91 | doResize(postContainer, ratio);
92 |
93 | setTimeout(function () {
94 | $window.trigger('action:composer.resize');
95 | postContainer.trigger('action:composer.resize');
96 | }, 0);
97 | });
98 | };
99 | }
100 |
101 | resize.reposition = function (postContainer) {
102 | var ratio = getSavedRatio();
103 |
104 | if (ratio >= 1 - snapMargin) {
105 | ratio = 1;
106 | postContainer.addClass('maximized');
107 | }
108 |
109 | resizeIt(postContainer, ratio);
110 | };
111 |
112 | resize.maximize = function (postContainer, state) {
113 | if (state) {
114 | resizeIt(postContainer, 1);
115 | } else {
116 | resize.reposition(postContainer);
117 | }
118 | };
119 |
120 | resize.handleResize = function (postContainer) {
121 | var resizeOffset = 0;
122 | var resizeBegin = 0;
123 | var resizeEnd = 0;
124 | var $resizer = postContainer.find('.resizer');
125 | var resizer = $resizer[0];
126 |
127 | function resizeStart(e) {
128 | var resizeRect = resizer.getBoundingClientRect();
129 | var resizeCenterY = (resizeRect.top + resizeRect.bottom) / 2;
130 |
131 | resizeOffset = (resizeCenterY - e.clientY) / 2;
132 | resizeBegin = e.clientY;
133 |
134 | $window.on('mousemove', resizeAction);
135 | $window.on('mouseup', resizeStop);
136 | $body.on('touchmove', resizeTouchAction);
137 | }
138 |
139 | function resizeAction(e) {
140 | var position = e.clientY - resizeOffset;
141 | var bounds = getBounds();
142 | var ratio = (bounds.height - position) / (bounds.boundedHeight);
143 |
144 | resizeIt(postContainer, ratio);
145 | }
146 |
147 | function resizeStop(e) {
148 | e.preventDefault();
149 | resizeEnd = e.clientY;
150 |
151 | postContainer.find('textarea').focus();
152 | $window.off('mousemove', resizeAction);
153 | $window.off('mouseup', resizeStop);
154 | $body.off('touchmove', resizeTouchAction);
155 |
156 | var position = resizeEnd - resizeOffset;
157 | var bounds = getBounds();
158 | var ratio = (bounds.height - position) / (bounds.boundedHeight);
159 |
160 | if (resizeEnd - resizeBegin === 0 && postContainer.hasClass('maximized')) {
161 | postContainer.removeClass('maximized');
162 | ratio = (!oldRatio || oldRatio >= 1 - snapMargin) ? 0.5 : oldRatio;
163 | resizeIt(postContainer, ratio);
164 | } else if (resizeEnd - resizeBegin === 0 || ratio >= 1 - snapMargin) {
165 | resizeIt(postContainer, 1);
166 | postContainer.addClass('maximized');
167 | oldRatio = ratio;
168 | } else {
169 | postContainer.removeClass('maximized');
170 | }
171 |
172 | saveRatio(ratio);
173 | }
174 |
175 | function resizeTouchAction(e) {
176 | e.preventDefault();
177 | resizeAction(e.touches[0]);
178 | }
179 |
180 | $resizer
181 | .on('mousedown', function (e) {
182 | if (e.button !== 0) {
183 | return;
184 | }
185 |
186 | e.preventDefault();
187 | resizeStart(e);
188 | })
189 | .on('touchstart', function (e) {
190 | e.preventDefault();
191 | resizeStart(e.touches[0]);
192 | })
193 | .on('touchend', resizeStop);
194 | };
195 |
196 | return resize;
197 | });
198 |
--------------------------------------------------------------------------------
/static/lib/composer/scheduler.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | define('composer/scheduler', ['benchpress', 'bootbox', 'alerts', 'translator'], function (
4 | Benchpress,
5 | bootbox,
6 | alerts,
7 | translator
8 | ) {
9 | const scheduler = {};
10 | const state = {
11 | timestamp: 0,
12 | open: false,
13 | edit: false,
14 | posts: {},
15 | };
16 | let displayBtnCons = [];
17 | let displayBtns;
18 | let cancelBtn;
19 | let submitContainer;
20 | let submitOptionsCon;
21 |
22 | const dropdownDisplayBtn = {
23 | el: null,
24 | defaultText: '',
25 | activeText: '',
26 | };
27 |
28 | const submitBtn = {
29 | el: null,
30 | icon: null,
31 | defaultText: '',
32 | activeText: '',
33 | };
34 | let dateInput;
35 | let timeInput;
36 |
37 | $(window).on('action:composer.activate', handleOnActivate);
38 |
39 | scheduler.init = function ($postContainer, posts) {
40 | state.timestamp = 0;
41 | state.posts = posts;
42 |
43 | translator.translateKeys(['[[topic:composer.post-later]]', '[[modules:composer.change-schedule-date]]']).then((translated) => {
44 | dropdownDisplayBtn.defaultText = translated[0];
45 | dropdownDisplayBtn.activeText = translated[1];
46 | });
47 |
48 | displayBtnCons = $postContainer[0].querySelectorAll('.display-scheduler');
49 | displayBtns = $postContainer[0].querySelectorAll('.display-scheduler i');
50 | dropdownDisplayBtn.el = $postContainer[0].querySelector('.dropdown-item.display-scheduler');
51 | cancelBtn = $postContainer[0].querySelector('.dropdown-item.cancel-scheduling');
52 | submitContainer = $postContainer.find('[component="composer/submit/container"]');
53 | submitOptionsCon = $postContainer.find('[component="composer/submit/options/container"]');
54 |
55 | submitBtn.el = $postContainer[0].querySelector('.composer-submit:not(.btn-sm)');
56 | submitBtn.icon = submitBtn.el.querySelector('i');
57 | submitBtn.defaultText = submitBtn.el.lastChild.textContent;
58 | submitBtn.activeText = submitBtn.el.getAttribute('data-text-variant');
59 |
60 | cancelBtn.addEventListener('click', cancelScheduling);
61 | displayBtnCons.forEach(el => el.addEventListener('click', openModal));
62 | };
63 |
64 | scheduler.getTimestamp = function () {
65 | if (!scheduler.isActive() || isNaN(state.timestamp)) {
66 | return 0;
67 | }
68 | return state.timestamp;
69 | };
70 |
71 | scheduler.isActive = function () {
72 | return state.timestamp > 0;
73 | };
74 |
75 | scheduler.isOpen = function () {
76 | return state.open;
77 | };
78 |
79 | scheduler.reset = function () {
80 | state.timestamp = 0;
81 | };
82 |
83 | scheduler.onChangeCategory = function (categoryData) {
84 | toggleDisplayButtons(categoryData.privileges['topics:schedule']);
85 | toggleItems(false);
86 | const optionsVisible = categoryData.privileges['topics:schedule'] || submitOptionsCon.attr('data-submit-options') > 0;
87 | submitContainer.find('.composer-submit').toggleClass('rounded-1', !optionsVisible);
88 | submitOptionsCon.toggleClass('hidden', !optionsVisible);
89 | scheduler.reset();
90 | };
91 |
92 | async function openModal() {
93 | const html = await Benchpress.render('modals/topic-scheduler');
94 | bootbox.dialog({
95 | message: html,
96 | title: '[[modules:composer.schedule-for]]',
97 | className: 'topic-scheduler',
98 | onShown: initModal,
99 | onHidden: handleOnHidden,
100 | onEscape: true,
101 | buttons: {
102 | cancel: {
103 | label: state.timestamp ? '[[modules:composer.cancel-scheduling]]' : '[[modules:bootbox.cancel]]',
104 | className: (state.timestamp ? 'btn-warning' : 'btn-outline-secondary') + (state.edit ? ' hidden' : ''),
105 | callback: cancelScheduling,
106 | },
107 | set: {
108 | label: '[[modules:composer.set-schedule-date]]',
109 | className: 'btn-primary',
110 | callback: setTimestamp,
111 | },
112 | },
113 | });
114 | }
115 |
116 | function initModal(ev) {
117 | state.open = true;
118 | const schedulerContainer = ev.target.querySelector('.datetime-picker');
119 | dateInput = schedulerContainer.querySelector('input[type="date"]');
120 | timeInput = schedulerContainer.querySelector('input[type="time"]');
121 | initDateTimeInputs();
122 | }
123 |
124 | function handleOnHidden() {
125 | state.open = false;
126 | }
127 |
128 | function handleOnActivate(ev, { post_uuid }) {
129 | state.edit = false;
130 |
131 | const postData = state.posts[post_uuid];
132 | if (postData && postData.isMain && postData.timestamp > Date.now()) {
133 | state.timestamp = postData.timestamp;
134 | state.edit = true;
135 | toggleItems();
136 | }
137 | }
138 |
139 | function initDateTimeInputs() {
140 | const d = new Date();
141 | // Update min. selectable date and time
142 | const nowLocalISO = new Date(d.getTime() - (d.getTimezoneOffset() * 60000)).toJSON();
143 | dateInput.setAttribute('min', nowLocalISO.slice(0, 10));
144 | timeInput.setAttribute('min', nowLocalISO.slice(11, -8));
145 |
146 | if (scheduler.isActive()) {
147 | const scheduleDate = new Date(state.timestamp - (d.getTimezoneOffset() * 60000)).toJSON();
148 | dateInput.value = scheduleDate.slice(0, 10);
149 | timeInput.value = scheduleDate.slice(11, -8);
150 | }
151 | }
152 |
153 | function setTimestamp() {
154 | const bothFilled = dateInput.value && timeInput.value;
155 | const timestamp = new Date(`${dateInput.value} ${timeInput.value}`).getTime();
156 | if (!bothFilled || isNaN(timestamp) || timestamp < Date.now()) {
157 | state.timestamp = 0;
158 | const message = timestamp < Date.now() ? '[[error:scheduling-to-past]]' : '[[error:invalid-schedule-date]]';
159 | alerts.alert({
160 | type: 'danger',
161 | timeout: 3000,
162 | title: '',
163 | alert_id: 'post_error',
164 | message,
165 | });
166 | return false;
167 | }
168 | if (!state.timestamp) {
169 | toggleItems(true);
170 | }
171 | state.timestamp = timestamp;
172 | }
173 |
174 | function cancelScheduling() {
175 | if (!state.timestamp) {
176 | return;
177 | }
178 | toggleItems(false);
179 | state.timestamp = 0;
180 | }
181 |
182 | function toggleItems(active = true) {
183 | displayBtns.forEach(btn => btn.classList.toggle('active', active));
184 | if (submitBtn.icon) {
185 | submitBtn.icon.classList.toggle('fa-check', !active);
186 | submitBtn.icon.classList.toggle('fa-clock-o', active);
187 | }
188 | if (dropdownDisplayBtn.el) {
189 | dropdownDisplayBtn.el.textContent = active ? dropdownDisplayBtn.activeText : dropdownDisplayBtn.defaultText;
190 | cancelBtn.classList.toggle('hidden', !active);
191 | }
192 | // Toggle submit button text
193 | submitBtn.el.lastChild.textContent = active ? submitBtn.activeText : submitBtn.defaultText;
194 | }
195 |
196 | function toggleDisplayButtons(show) {
197 | displayBtnCons.forEach(btn => btn.classList.toggle('hidden', !show));
198 | }
199 |
200 | return scheduler;
201 | });
202 |
--------------------------------------------------------------------------------
/static/lib/composer/tags.js:
--------------------------------------------------------------------------------
1 |
2 | 'use strict';
3 |
4 | define('composer/tags', ['alerts'], function (alerts) {
5 | var tags = {};
6 |
7 | var minTags;
8 | var maxTags;
9 |
10 | tags.init = function (postContainer, postData) {
11 | var tagEl = postContainer.find('.tags');
12 | if (!tagEl.length) {
13 | return;
14 | }
15 |
16 | minTags = ajaxify.data.hasOwnProperty('minTags') ? ajaxify.data.minTags : config.minimumTagsPerTopic;
17 | maxTags = ajaxify.data.hasOwnProperty('maxTags') ? ajaxify.data.maxTags : config.maximumTagsPerTopic;
18 |
19 | tagEl.tagsinput({
20 | tagClass: 'badge bg-info rounded-1',
21 | confirmKeys: [13, 44],
22 | trimValue: true,
23 | });
24 | var input = postContainer.find('.bootstrap-tagsinput input');
25 |
26 | toggleTagInput(postContainer, postData, ajaxify.data);
27 |
28 | app.loadJQueryUI(function () {
29 | input.autocomplete({
30 | delay: 100,
31 | position: { my: 'left bottom', at: 'left top', collision: 'flip' },
32 | appendTo: postContainer.find('.bootstrap-tagsinput'),
33 | open: function () {
34 | $(this).autocomplete('widget').css('z-index', 20000);
35 | },
36 | source: function (request, response) {
37 | socket.emit('topics.autocompleteTags', {
38 | query: request.term,
39 | cid: postData.cid,
40 | }, function (err, tags) {
41 | if (err) {
42 | return alerts.error(err);
43 | }
44 | if (tags) {
45 | response(tags);
46 | }
47 | $('.ui-autocomplete a').attr('data-ajaxify', 'false');
48 | });
49 | },
50 | select: function (/* event, ui */) {
51 | // when autocomplete is selected from the dropdown simulate a enter key down to turn it into a tag
52 | triggerEnter(input);
53 | },
54 | });
55 |
56 | addTags(postData.tags, tagEl);
57 |
58 | tagEl.on('beforeItemAdd', function (event) {
59 | var reachedMaxTags = maxTags && maxTags <= tags.getTags(postContainer.attr('data-uuid')).length;
60 | var cleanTag = utils.cleanUpTag(event.item, config.maximumTagLength);
61 | var different = cleanTag !== event.item;
62 | event.cancel = different ||
63 | event.item.length < config.minimumTagLength ||
64 | event.item.length > config.maximumTagLength ||
65 | reachedMaxTags;
66 |
67 | if (event.item.length < config.minimumTagLength) {
68 | return alerts.error('[[error:tag-too-short, ' + config.minimumTagLength + ']]');
69 | } else if (event.item.length > config.maximumTagLength) {
70 | return alerts.error('[[error:tag-too-long, ' + config.maximumTagLength + ']]');
71 | } else if (reachedMaxTags) {
72 | return alerts.error('[[error:too-many-tags, ' + maxTags + ']]');
73 | }
74 | var cid = postData.hasOwnProperty('cid') ? postData.cid : ajaxify.data.cid;
75 | $(window).trigger('action:tag.beforeAdd', {
76 | cid,
77 | tagEl,
78 | tag: event.item,
79 | event,
80 | inputAutocomplete: input,
81 | });
82 | if (different) {
83 | tagEl.tagsinput('add', cleanTag);
84 | }
85 | if (event.cancel && input.length) {
86 | input.autocomplete('close');
87 | }
88 | });
89 |
90 | tagEl.on('itemRemoved', function (event) {
91 | if (!event.item || (event.options && event.options.skipRemoveCheck)) {
92 | return;
93 | }
94 |
95 | socket.emit('topics.canRemoveTag', { tag: event.item }, function (err, allowed) {
96 | if (err) {
97 | return alerts.error(err);
98 | }
99 | if (!allowed) {
100 | alerts.error('[[error:cant-remove-system-tag]]');
101 | tagEl.tagsinput('add', event.item, { skipAddCheck: true });
102 | }
103 | });
104 | });
105 |
106 | tagEl.on('itemAdded', function (event) {
107 | if (event.options && event.options.skipAddCheck) {
108 | return;
109 | }
110 | var cid = postData.hasOwnProperty('cid') ? postData.cid : ajaxify.data.cid;
111 | socket.emit('topics.isTagAllowed', { tag: event.item, cid: cid || 0 }, function (err, allowed) {
112 | if (err) {
113 | return alerts.error(err);
114 | }
115 | if (!allowed) {
116 | return tagEl.tagsinput('remove', event.item, { skipRemoveCheck: true });
117 | }
118 | $(window).trigger('action:tag.added', {
119 | cid,
120 | tagEl,
121 | tag: event.item,
122 | inputAutocomplete: input,
123 | });
124 | if (input.length) {
125 | input.autocomplete('close');
126 | }
127 | });
128 | });
129 | });
130 |
131 | input.attr('tabIndex', tagEl.attr('tabIndex'));
132 | input.on('blur', function () {
133 | triggerEnter(input);
134 | });
135 |
136 | $('[component="composer/tag/dropdown"]').on('click', 'li', function () {
137 | var tag = $(this).attr('data-tag');
138 | if (tag) {
139 | addTags([tag], tagEl);
140 | }
141 | return false;
142 | });
143 | };
144 |
145 | tags.isEnoughTags = function (post_uuid) {
146 | return tags.getTags(post_uuid).length >= minTags;
147 | };
148 |
149 | tags.minTagCount = function () {
150 | return minTags;
151 | };
152 |
153 | tags.onChangeCategory = function (postContainer, postData, cid, categoryData) {
154 | var tagDropdown = postContainer.find('[component="composer/tag/dropdown"]');
155 | if (!tagDropdown.length) {
156 | return;
157 | }
158 |
159 | toggleTagInput(postContainer, postData, categoryData);
160 | tagDropdown.toggleClass('hidden', !categoryData.tagWhitelist || !categoryData.tagWhitelist.length);
161 | if (categoryData.tagWhitelist) {
162 | app.parseAndTranslate('composer', 'tagWhitelist', { tagWhitelist: categoryData.tagWhitelist }, function (html) {
163 | tagDropdown.find('.dropdown-menu').html(html);
164 | });
165 | }
166 | };
167 |
168 | function toggleTagInput(postContainer, postData, data) {
169 | var tagEl = postContainer.find('.tags');
170 | var input = postContainer.find('.bootstrap-tagsinput input');
171 | if (!input.length) {
172 | return;
173 | }
174 |
175 | if (data.hasOwnProperty('minTags')) {
176 | minTags = data.minTags;
177 | }
178 | if (data.hasOwnProperty('maxTags')) {
179 | maxTags = data.maxTags;
180 | }
181 |
182 | if (data.tagWhitelist && data.tagWhitelist.length) {
183 | input.attr('readonly', '');
184 | input.attr('placeholder', '');
185 |
186 | tagEl.tagsinput('items').slice().forEach(function (tag) {
187 | if (data.tagWhitelist.indexOf(tag) === -1) {
188 | tagEl.tagsinput('remove', tag);
189 | }
190 | });
191 | } else {
192 | input.removeAttr('readonly');
193 | input.attr('placeholder', postContainer.find('input.tags').attr('placeholder'));
194 | }
195 | postContainer.find('.tags-container').toggleClass('haswhitelist', !!(data.tagWhitelist && data.tagWhitelist.length));
196 | postContainer.find('.tags-container').toggleClass('hidden', (
197 | data.privileges && data.privileges.hasOwnProperty('topics:tag') && !data.privileges['topics:tag']) ||
198 | (maxTags === 0 && !postData && !postData.tags && !postData.tags.length));
199 |
200 | if (data.privileges && data.privileges.hasOwnProperty('topics:tag') && !data.privileges['topics:tag']) {
201 | tagEl.tagsinput('removeAll');
202 | }
203 |
204 | $(window).trigger('action:tag.toggleInput', {
205 | postContainer: postContainer,
206 | tagWhitelist: data.tagWhitelist,
207 | tagsInput: input,
208 | });
209 | }
210 |
211 | function triggerEnter(input) {
212 | // http://stackoverflow.com/a/3276819/583363
213 | var e = jQuery.Event('keypress');
214 | e.which = 13;
215 | e.keyCode = 13;
216 | setTimeout(function () {
217 | input.trigger(e);
218 | }, 100);
219 | }
220 |
221 | function addTags(tags, tagEl) {
222 | if (tags && tags.length) {
223 | for (var i = 0; i < tags.length; ++i) {
224 | tagEl.tagsinput('add', tags[i]);
225 | }
226 | }
227 | }
228 |
229 | tags.getTags = function (post_uuid) {
230 | return $('.composer[data-uuid="' + post_uuid + '"] .tags').tagsinput('items');
231 | };
232 |
233 | return tags;
234 | });
235 |
--------------------------------------------------------------------------------
/static/lib/composer/uploads.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | define('composer/uploads', [
4 | 'composer/preview',
5 | 'composer/categoryList',
6 | 'translator',
7 | 'alerts',
8 | 'uploadHelpers',
9 | 'jquery-form',
10 | ], function (preview, categoryList, translator, alerts, uploadHelpers) {
11 | var uploads = {
12 | inProgress: {},
13 | };
14 |
15 | var uploadingText = '';
16 |
17 | uploads.initialize = function (post_uuid) {
18 | initializeDragAndDrop(post_uuid);
19 | initializePaste(post_uuid);
20 |
21 | addChangeHandlers(post_uuid);
22 | addTopicThumbHandlers(post_uuid);
23 | translator.translate('[[modules:composer.uploading, ' + 0 + '%]]', function (translated) {
24 | uploadingText = translated;
25 | });
26 | };
27 |
28 | function addChangeHandlers(post_uuid) {
29 | var postContainer = $('.composer[data-uuid="' + post_uuid + '"]');
30 |
31 | postContainer.find('#files').on('change', function (e) {
32 | var files = (e.target || {}).files ||
33 | ($(this).val() ? [{ name: $(this).val(), type: utils.fileMimeType($(this).val()) }] : null);
34 | if (files) {
35 | uploadContentFiles({ files: files, post_uuid: post_uuid, route: '/api/post/upload' });
36 | }
37 | });
38 | }
39 |
40 | function addTopicThumbHandlers(post_uuid) {
41 | var postContainer = $('.composer[data-uuid="' + post_uuid + '"]');
42 |
43 | postContainer.on('click', '.topic-thumb-clear-btn', function (e) {
44 | postContainer.find('input#topic-thumb-url').val('').trigger('change');
45 | resetInputFile(postContainer.find('input#topic-thumb-file'));
46 | $(this).addClass('hide');
47 | e.preventDefault();
48 | });
49 |
50 | postContainer.on('paste change keypress', 'input#topic-thumb-url', function () {
51 | var urlEl = $(this);
52 | setTimeout(function () {
53 | var url = urlEl.val();
54 | if (url) {
55 | postContainer.find('.topic-thumb-clear-btn').removeClass('hide');
56 | } else {
57 | resetInputFile(postContainer.find('input#topic-thumb-file'));
58 | postContainer.find('.topic-thumb-clear-btn').addClass('hide');
59 | }
60 | postContainer.find('img.topic-thumb-preview').attr('src', url);
61 | }, 100);
62 | });
63 | }
64 |
65 | function resetInputFile($el) {
66 | $el.wrap('').closest('form').get(0).reset();
67 | $el.unwrap();
68 | }
69 |
70 | function initializeDragAndDrop(post_uuid) {
71 | var postContainer = $('.composer[data-uuid="' + post_uuid + '"]');
72 | uploadHelpers.handleDragDrop({
73 | container: postContainer,
74 | callback: function (upload) {
75 | uploadContentFiles({
76 | files: upload.files,
77 | post_uuid: post_uuid,
78 | route: '/api/post/upload',
79 | formData: upload.formData,
80 | });
81 | },
82 | });
83 | }
84 |
85 | function initializePaste(post_uuid) {
86 | var postContainer = $('.composer[data-uuid="' + post_uuid + '"]');
87 | uploadHelpers.handlePaste({
88 | container: postContainer,
89 | callback: function (upload) {
90 | uploadContentFiles({
91 | files: upload.files,
92 | fileNames: upload.fileNames,
93 | post_uuid: post_uuid,
94 | route: '/api/post/upload',
95 | formData: upload.formData,
96 | });
97 | },
98 | });
99 | }
100 |
101 | function escapeRegExp(text) {
102 | return text.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, '\\$&');
103 | }
104 |
105 | function insertText(str, index, insert) {
106 | return str.slice(0, index) + insert + str.slice(index);
107 | }
108 |
109 | function uploadContentFiles(params) {
110 | var files = [...params.files];
111 | var post_uuid = params.post_uuid;
112 | var postContainer = $('.composer[data-uuid="' + post_uuid + '"]');
113 | var textarea = postContainer.find('textarea');
114 | var text = textarea.val();
115 | var uploadForm = postContainer.find('#fileForm');
116 | var doneUploading = false;
117 | uploadForm.attr('action', config.relative_path + params.route);
118 |
119 | var cid = categoryList.getSelectedCid();
120 | if (!cid && ajaxify.data.cid) {
121 | cid = ajaxify.data.cid;
122 | }
123 | var i = 0;
124 | var isImage = false;
125 | for (i = 0; i < files.length; ++i) {
126 | isImage = files[i].type.match(/image./);
127 | if ((isImage && !app.user.privileges['upload:post:image']) || (!isImage && !app.user.privileges['upload:post:file'])) {
128 | return alerts.error('[[error:no-privileges]]');
129 | }
130 | }
131 |
132 | var filenameMapping = [];
133 | let filesText = '';
134 | for (i = 0; i < files.length; ++i) {
135 | // The filename map has datetime and iterator prepended so that they can be properly tracked even if the
136 | // filenames are identical.
137 | filenameMapping.push(i + '_' + Date.now() + '_' + (params.fileNames ? params.fileNames[i] : files[i].name));
138 | isImage = files[i].type.match(/image./);
139 |
140 | if (!app.user.isAdmin && files[i].size > parseInt(config.maximumFileSize, 10) * 1024) {
141 | uploadForm[0].reset();
142 | return alerts.error('[[error:file-too-big, ' + config.maximumFileSize + ']]');
143 | }
144 | filesText += (isImage ? '!' : '') + '[' + filenameMapping[i] + '](' + uploadingText + ') ';
145 | }
146 |
147 | const cursorPosition = textarea.getCursorPosition();
148 | const textLen = text.length;
149 | text = insertText(text, cursorPosition, filesText);
150 |
151 | if (uploadForm.length) {
152 | postContainer.find('[data-action="post"]').prop('disabled', true);
153 | }
154 | textarea.val(text);
155 |
156 | $(window).trigger('action:composer.uploadStart', {
157 | post_uuid: post_uuid,
158 | files: filenameMapping.map(function (filename, i) {
159 | return {
160 | filename: filename.replace(/^\d+_\d{13}_/, ''),
161 | isImage: /image./.test(files[i].type),
162 | };
163 | }),
164 | text: uploadingText,
165 | });
166 |
167 | uploadForm.off('submit').submit(function () {
168 | function updateTextArea(filename, text, trim) {
169 | var newFilename;
170 | if (trim) {
171 | newFilename = filename.replace(/^\d+_\d{13}_/, '');
172 | }
173 | var current = textarea.val();
174 | var re = new RegExp(escapeRegExp(filename) + ']\\([^)]+\\)', 'g');
175 | textarea.val(current.replace(re, (newFilename || filename) + '](' + text + ')'));
176 |
177 | $(window).trigger('action:composer.uploadUpdate', {
178 | post_uuid: post_uuid,
179 | filename: filename,
180 | text: text,
181 | });
182 | }
183 |
184 | uploads.inProgress[post_uuid] = uploads.inProgress[post_uuid] || [];
185 | uploads.inProgress[post_uuid].push(1);
186 |
187 | if (params.formData) {
188 | params.formData.append('cid', cid);
189 | }
190 |
191 | $(this).ajaxSubmit({
192 | headers: {
193 | 'x-csrf-token': config.csrf_token,
194 | },
195 | resetForm: true,
196 | clearForm: true,
197 | formData: params.formData,
198 | data: { cid: cid },
199 |
200 | error: function (xhr) {
201 | doneUploading = true;
202 | postContainer.find('[data-action="post"]').prop('disabled', false);
203 | const errorMsg = onUploadError(xhr, post_uuid);
204 | for (var i = 0; i < files.length; ++i) {
205 | updateTextArea(filenameMapping[i], errorMsg, true);
206 | }
207 | preview.render(postContainer);
208 | },
209 |
210 | uploadProgress: function (event, position, total, percent) {
211 | translator.translate('[[modules:composer.uploading, ' + percent + '%]]', function (translated) {
212 | if (doneUploading) {
213 | return;
214 | }
215 | for (var i = 0; i < files.length; ++i) {
216 | updateTextArea(filenameMapping[i], translated);
217 | }
218 | });
219 | },
220 |
221 | success: function (res) {
222 | const uploads = res.response.images;
223 | doneUploading = true;
224 | if (uploads && uploads.length) {
225 | for (var i = 0; i < uploads.length; ++i) {
226 | uploads[i].filename = filenameMapping[i].replace(/^\d+_\d{13}_/, '');
227 | uploads[i].isImage = /image./.test(files[i].type);
228 | updateTextArea(filenameMapping[i], uploads[i].url, true);
229 | }
230 | }
231 | preview.render(postContainer);
232 | textarea.prop('selectionEnd', cursorPosition + textarea.val().length - textLen);
233 | textarea.focus();
234 | postContainer.find('[data-action="post"]').prop('disabled', false);
235 | $(window).trigger('action:composer.upload', {
236 | post_uuid: post_uuid,
237 | files: uploads,
238 | });
239 | },
240 |
241 | complete: function () {
242 | uploadForm[0].reset();
243 | uploads.inProgress[post_uuid].pop();
244 | },
245 | });
246 |
247 | return false;
248 | });
249 |
250 | uploadForm.submit();
251 | }
252 |
253 | function onUploadError(xhr, post_uuid) {
254 | var msg = (xhr.responseJSON &&
255 | (xhr.responseJSON.error || (xhr.responseJSON.status && xhr.responseJSON.status.message))) ||
256 | '[[error:parse-error]]';
257 |
258 | if (xhr && xhr.status === 413) {
259 | msg = xhr.statusText || 'Request Entity Too Large';
260 | }
261 | alerts.error(msg);
262 | $(window).trigger('action:composer.uploadError', {
263 | post_uuid: post_uuid,
264 | message: msg,
265 | });
266 | return msg;
267 | }
268 |
269 | return uploads;
270 | });
271 |
272 |
--------------------------------------------------------------------------------
/static/scss/composer.scss:
--------------------------------------------------------------------------------
1 | .composer {
2 | background-color: var(--bs-body-bg);
3 | color: var(--bs-body-color);
4 | z-index: $zindex-dropdown;
5 | visibility: hidden;
6 | padding: 0;
7 | position: fixed;
8 | bottom: 0;
9 | top: 0;
10 | right: 0;
11 | left: 0;
12 |
13 | .mobile-navbar {
14 | position: static;
15 | min-height: 40px;
16 | margin: 0;
17 |
18 | .btn-group {
19 | flex-shrink: 0;
20 | }
21 |
22 | button {
23 | font-size: 20px;
24 | }
25 |
26 | display: flex;
27 |
28 | .category-name-container, .title {
29 | text-align: center;
30 | text-overflow: ellipsis;
31 | overflow: hidden;
32 | white-space: nowrap;
33 | flex-grow: 2;
34 | font-size: 16px;
35 | line-height: inherit;
36 | padding: 9px 5px;
37 | margin: 0;
38 | }
39 | }
40 |
41 | .title-container {
42 | > div[data-component="composer/handle"] {
43 | flex: 0.33;
44 | }
45 |
46 | .category-list-container {
47 |
48 | [component="category-selector"] {
49 | .category-dropdown-menu {
50 | max-height: 300px;
51 | }
52 | }
53 | }
54 |
55 | .category-list {
56 | padding: 0 2rem;
57 | }
58 |
59 | .action-bar {
60 | .dropdown-menu:empty {
61 | & ~ .dropdown-toggle {
62 | display: none;
63 | }
64 | }
65 | }
66 | }
67 |
68 | .formatting-bar {
69 | .spacer {
70 | &:before {
71 | content: ' | ';
72 | color: $gray-200;
73 | }
74 | }
75 | }
76 |
77 | .tags-container {
78 | [component="composer/tag/dropdown"] {
79 | .dropdown-menu {
80 | max-height: 400px;
81 | overflow-y: auto;
82 | }
83 |
84 | > button {
85 | border: 0;
86 | }
87 | }
88 | // if picking tags from taglist dropdown hide the input
89 | &.haswhitelist .bootstrap-tagsinput {
90 | input {
91 | display: none;
92 | }
93 | }
94 | .bootstrap-tagsinput {
95 | background: transparent;
96 | flex-grow: 1;
97 | border: 0;
98 | padding: 0;
99 | box-shadow: none;
100 | max-height: 80px;
101 | overflow: auto;
102 |
103 | input {
104 | &::placeholder{
105 | color: $input-placeholder-color;
106 | }
107 | color: $body-color;
108 | font-size: 16px;
109 | width: 50%;
110 | @include media-breakpoint-down(md) {
111 | width: 100%;
112 | }
113 |
114 |
115 | height: 28px;
116 | padding: 4px 6px;
117 | }
118 |
119 | .ui-autocomplete {
120 | max-height: 350px;
121 | overflow-x: hidden;
122 | overflow-y: auto;
123 | }
124 | }
125 | }
126 |
127 | .resizer {
128 | background: linear-gradient(transparent, var(--bs-body-bg));
129 | margin-left: calc($spacer * -0.5);
130 | padding-left: $spacer;
131 |
132 | .trigger {
133 | cursor: ns-resize;
134 |
135 | .handle {
136 | border-top-left-radius: 50%;
137 | border-top-right-radius: 50%;
138 | border-bottom: 0 !important;
139 | }
140 | }
141 | }
142 |
143 | .minimize {
144 | display: none;
145 | position: absolute;
146 | top: 0px;
147 | right: 10px;
148 | height: 0;
149 |
150 | @include pointer;
151 |
152 | .trigger {
153 | position: relative;
154 | display: block;
155 | top: -20px;
156 | right: 0px;
157 | margin: 0 auto;
158 | margin-left: 20px;
159 | line-height: 26px;
160 | @include transition(filter .15s linear);
161 |
162 | &:hover {
163 | filter: invert(100%);
164 | }
165 |
166 | i {
167 | width: 32px;
168 | height: 32px;
169 | background: #333;
170 | border: 1px solid #333;
171 | border-radius: 50%;
172 |
173 | position: relative;
174 |
175 | color: #FFF;
176 | font-size: 16px;
177 |
178 | &:before {
179 | position: relative;
180 | top: 25%;
181 | }
182 | }
183 | }
184 | }
185 |
186 | &.reply {
187 | .title-container {
188 | display: none;
189 | }
190 | }
191 |
192 | &.resizable.maximized {
193 | .resizer {
194 | top: 0 !important;
195 | background: transparent;
196 |
197 | .trigger {
198 | height: $spacer * 0.5;
199 |
200 | .handle {
201 | border-top-left-radius: 0%;
202 | border-top-right-radius: 0%;
203 | border-bottom-left-radius: 50%;
204 | border-bottom-right-radius: 50%;
205 | border-bottom: var(--bs-border-width) var(--bs-border-style) var(--bs-border-color) !important;
206 | }
207 |
208 | i {
209 | &:before {
210 | content: fa-content($fa-var-chevron-down);
211 | }
212 | }
213 | }
214 | }
215 | }
216 |
217 | .draft-icon {
218 | font-family: 'FontAwesome';
219 | color: $success;
220 | opacity: 0;
221 |
222 | &::before {
223 | content: fa-content($fa-var-save);
224 | }
225 |
226 | &.active {
227 | animation: draft-saved 3s ease;
228 | }
229 | }
230 |
231 | textarea {
232 | resize: none;
233 | }
234 |
235 | .preview {
236 | padding: $input-padding-y $input-padding-x;
237 | }
238 | }
239 |
240 | .datetime-picker {
241 | display: flex;
242 | justify-content: center;
243 | flex-direction: row;
244 | min-width: 310px;
245 | max-width: 310px;
246 | margin: 0 auto;
247 |
248 | input {
249 | flex: 3;
250 | line-height: inherit;
251 | }
252 |
253 | input + input {
254 | border-left: none;
255 | flex: 2;
256 | }
257 | }
258 |
259 | .modal.topic-scheduler {
260 | z-index: 1070;
261 | & + .modal-backdrop {
262 | z-index: 1060;
263 | }
264 | }
265 |
266 | @keyframes draft-saved {
267 | 0%, 100% {
268 | opacity: 0;
269 | }
270 |
271 | 15% {
272 | opacity: 1;
273 | }
274 |
275 | 30% {
276 | opacity: 0.5;
277 | }
278 |
279 | 45% {
280 | opacity: 1;
281 | }
282 |
283 | 85% {
284 | opacity: 1;
285 | }
286 | }
287 |
288 | @keyframes pulse {
289 | from {
290 | transform: scale(1);
291 | color: inherit;
292 | }
293 | 50% {
294 | transform: scale(.9);
295 | }
296 | to {
297 | transform: scale(1);
298 | color: #00adff;
299 | }
300 | }
301 |
302 | @include media-breakpoint-down(lg) {
303 | html.composing .composer { z-index: $zindex-modal; }
304 | }
305 |
306 | @include media-breakpoint-down(sm) {
307 | html.composing {
308 | .composer {
309 | height: 100%;
310 |
311 | .draft-icon {
312 | position: absolute;
313 | bottom: 1em;
314 | right: 0em;
315 |
316 | &::after {
317 | top: 7px;
318 | }
319 | }
320 |
321 | .preview-container {
322 | max-width: initial;
323 | }
324 | }
325 |
326 | body {
327 | padding-bottom: 0 !important;
328 | }
329 | }
330 | }
331 |
332 | @include media-breakpoint-up(lg) {
333 | html.composing {
334 | .composer {
335 | left: 15%;
336 | width: 70%;
337 | min-height: 400px;
338 |
339 | .resizer {
340 | display: block;
341 | }
342 |
343 | .minimize {
344 | display: block;
345 | }
346 | }
347 | }
348 | }
349 |
350 | @include media-breakpoint-up(md) {
351 | // without this formatting elements that are dropdowns are not visible on desktop.
352 | // on mobile dropdowns use bottom-sheet and overflow is auto
353 | .formatting-group {
354 | overflow: visible!important;
355 | }
356 | }
357 |
358 | @import './zen-mode';
359 | @import './page-compose';
360 | @import './textcomplete';
361 |
362 |
363 | .skin-noskin, .skin-cosmo, .skin-flatly,
364 | .skin-journal, .skin-litera, .skin-minty, .skin-pulse,
365 | .skin-sandstone, .skin-sketchy, .skin-spacelab, .skin-united {
366 | .composer {
367 | color: var(--bs-secondary) !important;
368 | background-color: var(--bs-light) !important;
369 | }
370 | }
371 |
372 | .skin-cerulean, .skin-lumen, .skin-lux, .skin-morph,
373 | .skin-simplex, .skin-yeti, .skin-zephyr {
374 | .composer {
375 | color: var(--bs-body) !important;
376 | background-color: var(--bs-light) !important;
377 | }
378 | }
379 |
380 | @include color-mode(dark) {
381 | .skin-noskin .composer {
382 | color: var(--bs-secondary)!important;
383 | background-color: var(--bs-body-bg)!important;
384 | }
385 | }
--------------------------------------------------------------------------------
/static/scss/page-compose.scss:
--------------------------------------------------------------------------------
1 | .page-compose .composer {
2 | z-index: initial;
3 | position: static;
4 | [data-action="hide"] {
5 | display: none;
6 | }
7 |
8 | @include media-breakpoint-down(md) {
9 | .title-container {
10 | flex-wrap: wrap;
11 | }
12 | .category-list-container {
13 | [component="category-selector-selected"] > span {
14 | display: inline!important;
15 | }
16 | width: 100%;
17 | }
18 | }
19 | }
20 |
21 | .zen-mode .page-compose .composer {
22 | position: absolute;
23 | }
24 | .page-compose {
25 | &.skin-noskin, &.skin-cosmo, &.skin-flatly,
26 | &.skin-journal, &.skin-litera, &.skin-minty, &.skin-pulse,
27 | &.skin-sandstone, &.skin-sketchy, &.skin-spacelab, &.skin-united,
28 | &.skin-cerulean, &.skin-lumen, &.skin-lux, &.skin-morph,
29 | &.skin-simplex, &.skin-yeti, &.skin-zephyr {
30 | .composer {
31 | color: var(--bs-body-color) !important;
32 | background-color: var(--bs-body-bg) !important;
33 | }
34 | }
35 | }
36 |
--------------------------------------------------------------------------------
/static/scss/textcomplete.scss:
--------------------------------------------------------------------------------
1 | .textcomplete-dropdown {
2 | border: 1px solid $border-color;
3 | background-color: $body-bg;
4 | color: $body-color;
5 | list-style: none;
6 | padding: 0;
7 | margin: 0;
8 |
9 | li {
10 | margin: 0;
11 | }
12 |
13 | .textcomplete-footer, .textcomplete-item {
14 | border-top: 1px solid $border-color;
15 | }
16 |
17 | .textcomplete-item {
18 | padding: 2px 5px;
19 | cursor: pointer;
20 |
21 | &:hover, &.active {
22 | color: $dropdown-link-hover-color;
23 | background-color: $dropdown-link-hover-bg;
24 | }
25 | }
26 | }
--------------------------------------------------------------------------------
/static/scss/zen-mode.scss:
--------------------------------------------------------------------------------
1 | html.zen-mode {
2 | overflow: hidden;
3 | }
4 |
5 | .zen-mode .composer {
6 | &.resizable {
7 | padding-top: 0;
8 | }
9 |
10 | .composer-container {
11 | padding-top: 5px;
12 | }
13 |
14 | .tag-row {
15 | display: none;
16 | }
17 |
18 | .title-container .category-list-container {
19 | margin-top: 3px;
20 | }
21 |
22 | .write, .preview {
23 | border: none;
24 | outline: none;
25 | }
26 |
27 | .resizer {
28 | display: none;
29 | }
30 |
31 | &.reply {
32 | .title-container {
33 | display: none;
34 | }
35 | }
36 |
37 | @include media-breakpoint-up(md) {
38 | & {
39 | padding-left: 15px;
40 | padding-right: 15px;
41 | }
42 | .write-preview-container {
43 | margin-bottom: 0;
44 |
45 | > div {
46 | padding: 0;
47 | margin: 0;
48 | }
49 | }
50 | }
51 | }
--------------------------------------------------------------------------------
/static/templates/admin/plugins/composer-default.tpl:
--------------------------------------------------------------------------------
1 |
22 |
23 |
--------------------------------------------------------------------------------
/static/templates/compose.tpl:
--------------------------------------------------------------------------------
1 |
2 |
3 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 | {{{ if isTopicOrMain }}}
24 |
25 | {{{ end }}}
26 |
27 |
28 |
--------------------------------------------------------------------------------
/static/templates/composer.tpl:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 | {{{ if isTopicOrMain }}}
32 |
33 | {{{ end }}}
34 |
35 |
[[topic:composer.drag-and-drop-images]]
36 |
37 |
44 |
45 |
46 |
47 |
--------------------------------------------------------------------------------
/static/templates/modals/topic-scheduler.tpl:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
--------------------------------------------------------------------------------
/static/templates/partials/composer-formatting.tpl:
--------------------------------------------------------------------------------
1 |
2 |
59 |
60 |
61 |
62 |
67 | {{{ if composer:showHelpTab }}}
68 |
72 | {{{ end }}}
73 |
74 |
75 |
76 |
--------------------------------------------------------------------------------
/static/templates/partials/composer-tags.tpl:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/static/templates/partials/composer-title-container.tpl:
--------------------------------------------------------------------------------
1 |
2 | {{{ if isTopic }}}
3 |
4 |
5 |
6 | {{{ end }}}
7 |
8 | {{{ if showHandleInput }}}
9 |
10 |
11 |
12 | {{{ end }}}
13 |
14 |
15 | {{{ if isTopicOrMain }}}
16 |
17 | {{{ else }}}
18 | {{{ if isEditing }}}[[topic:composer.editing-in, "{topicTitle}"]]{{{ else }}}[[topic:composer.replying-to, "{topicTitle}"]]{{{ end }}}
19 | {{{ end }}}
20 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
36 |
43 |
44 |
45 |
46 |
47 |
--------------------------------------------------------------------------------
/static/templates/partials/composer-write-preview.tpl:
--------------------------------------------------------------------------------
1 |
2 |
3 |
[[modules:composer.post-queue-alert]]
4 |
5 |
6 |
7 |
10 |
--------------------------------------------------------------------------------
/websockets.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | const meta = require.main.require('./src/meta');
4 | const privileges = require.main.require('./src/privileges');
5 | const posts = require.main.require('./src/posts');
6 | const topics = require.main.require('./src/topics');
7 | const plugins = require.main.require('./src/plugins');
8 |
9 | const Sockets = module.exports;
10 |
11 | Sockets.push = async function (socket, pid) {
12 | const canRead = await privileges.posts.can('topics:read', pid, socket.uid);
13 | if (!canRead) {
14 | throw new Error('[[error:no-privileges]]');
15 | }
16 |
17 | const postData = await posts.getPostFields(pid, ['content', 'sourceContent', 'tid', 'uid', 'handle', 'timestamp']);
18 | if (!postData && !postData.content) {
19 | throw new Error('[[error:invalid-pid]]');
20 | }
21 |
22 | const [topic, tags, isMain] = await Promise.all([
23 | topics.getTopicDataByPid(pid),
24 | topics.getTopicTags(postData.tid),
25 | posts.isMain(pid),
26 | ]);
27 |
28 | if (!topic) {
29 | throw new Error('[[error:no-topic]]');
30 | }
31 |
32 | const result = await plugins.hooks.fire('filter:composer.push', {
33 | pid: pid,
34 | uid: postData.uid,
35 | handle: parseInt(meta.config.allowGuestHandles, 10) ? postData.handle : undefined,
36 | body: postData.sourceContent || postData.content,
37 | title: topic.title,
38 | thumb: topic.thumb,
39 | tags: tags,
40 | isMain: isMain,
41 | timestamp: postData.timestamp,
42 | });
43 | return result;
44 | };
45 |
46 | Sockets.editCheck = async function (socket, pid) {
47 | const isMain = await posts.isMain(pid);
48 | return { titleEditable: isMain };
49 | };
50 |
51 | Sockets.renderPreview = async function (socket, content) {
52 | return await plugins.hooks.fire('filter:parse.raw', content);
53 | };
54 |
55 | Sockets.renderHelp = async function () {
56 | const helpText = meta.config['composer:customHelpText'] || '';
57 | if (!meta.config['composer:showHelpTab']) {
58 | throw new Error('help-hidden');
59 | }
60 |
61 | const parsed = await plugins.hooks.fire('filter:parse.raw', helpText);
62 | if (meta.config['composer:allowPluginHelp'] && plugins.hooks.hasListeners('filter:composer.help')) {
63 | return await plugins.hooks.fire('filter:composer.help', parsed) || helpText;
64 | }
65 | return helpText;
66 | };
67 |
68 | Sockets.getFormattingOptions = async function () {
69 | return await require('./library').getFormattingOptions();
70 | };
71 |
72 | Sockets.shouldQueue = async function (socket, data) {
73 | if (!data || !data.postData) {
74 | throw new Error('[[error:invalid-data]]');
75 | }
76 | if (socket.uid <= 0) {
77 | return false;
78 | }
79 |
80 | let shouldQueue = false;
81 | const { postData } = data;
82 | if (postData.action === 'posts.reply') {
83 | shouldQueue = await posts.shouldQueue(socket.uid, {
84 | tid: postData.tid,
85 | content: postData.content || '',
86 | });
87 | } else if (postData.action === 'topics.post') {
88 | shouldQueue = await posts.shouldQueue(socket.uid, {
89 | cid: postData.cid,
90 | content: postData.content || '',
91 | });
92 | }
93 | return shouldQueue;
94 | };
95 |
--------------------------------------------------------------------------------