client+server: implement code autoformatting using prettier and black

This commit is contained in:
Shyam Sunder 2020-06-05 18:03:37 -04:00
parent c06aaa63af
commit 57193b5715
312 changed files with 15512 additions and 12825 deletions

View file

@ -1,17 +1,45 @@
repos:
- repo: https://github.com/pre-commit/pre-commit-hooks
rev: v2.4.0
hooks:
- id: trailing-whitespace
- id: end-of-file-fixer
- id: check-yaml
- id: check-added-large-files
- repo: https://github.com/psf/black
rev: stable
hooks:
- id: black
files: server/szurubooru/
language_version: python3.8
- repo: https://github.com/timothycrosley/isort
rev: '4.3.21-2'
hooks:
- id: isort
files: server/szurubooru/
exclude: server/szurubooru/migrations/env.py
additional_dependencies:
- toml
- repo: https://github.com/prettier/prettier
rev: '2.0.5'
hooks:
- id: prettier
files: client/js/
exclude: client/js/.gitignore
args: ['--config', 'client/.prettierrc.yml']
- repo: https://github.com/pre-commit/mirrors-eslint
rev: v7.1.0
hooks:
- id: eslint
files: client/js/
args: ['--fix']
additional_dependencies:
- eslint-config-prettier
- repo: https://gitlab.com/pycqa/flake8
rev: '3.8.2'
hooks:
@ -19,32 +47,45 @@ repos:
files: server/szurubooru/
additional_dependencies:
- flake8-print
args: ['--config=server/setup.cfg']
args: ['--config=server/.flake8']
- repo: local
hooks:
- id: pytest
name: pytest
entry: >-
bash -c
'docker build -f server/Dockerfile.test -t $(git rev-parse --short HEAD)-test server/
&& docker run --rm -t $(git rev-parse --short HEAD)-test szurubooru/
&& docker rmi --no-prune $(git rev-parse --short HEAD)-test'
language: system
types: [python]
files: server/szurubooru/
pass_filenames: false
- id: docker-build-client
name: Test building the client in Docker
entry: bash -c 'docker build -t szurubooru-client:$(git rev-parse --short HEAD) client/'
name: Docker - build client
entry: bash -c 'docker build client/'
language: system
types: [file]
files: client/
pass_filenames: false
- id: docker-build-server
name: Test building the server in Docker
entry: bash -c 'docker build -t szurubooru-server:$(git rev-parse --short HEAD) server/'
name: Docker - build server
entry: bash -c 'docker build server/'
language: system
types: [file]
files: server/
pass_filenames: false
- id: pytest
name: pytest
entry: bash -c 'docker run --rm -t $(docker build -f server/Dockerfile.test -q server/) szurubooru/'
language: system
types: [python]
files: server/szurubooru/
exclude: server/szurubooru/migrations/
pass_filenames: false
- id: pytest-cov
name: pytest
entry: bash -c 'docker run --rm -t $(docker build -f server/Dockerfile.test -q server/) --cov-report=term-missing:skip-covered --cov=szurubooru szurubooru/'
language: system
types: [python]
files: server/szurubooru/
exclude: server/szurubooru/migrations/
pass_filenames: false
verbose: true
stages: [manual]
fail_fast: true
exclude: LICENSE.md

View file

@ -2,7 +2,7 @@ env:
browser: true
commonjs: true
es6: true
extends: 'eslint:recommended'
extends: 'prettier'
globals:
Atomics: readonly
SharedArrayBuffer: readonly
@ -10,284 +10,3 @@ ignorePatterns:
- build.js
parserOptions:
ecmaVersion: 11
rules:
accessor-pairs: error
array-bracket-newline: error
array-bracket-spacing:
- error
- never
array-callback-return: error
array-element-newline: 'off'
arrow-body-style: 'off'
arrow-parens:
- error
- as-needed
arrow-spacing:
- error
- after: true
before: true
block-scoped-var: error
block-spacing: error
brace-style:
- error
- 1tbs
callback-return: 'off'
camelcase: error
class-methods-use-this: 'off'
comma-dangle: 'off'
comma-spacing:
- error
- after: true
before: false
comma-style:
- error
- last
complexity: 'off'
computed-property-spacing:
- error
- never
consistent-return: 'off'
consistent-this: 'off'
curly: error
default-case: error
default-case-last: error
default-param-last: error
dot-location:
- error
- property
dot-notation:
- error
- allowKeywords: true
eol-last: error
eqeqeq: error
func-call-spacing: error
func-name-matching: error
func-names: error
func-style:
- error
- declaration
- allowArrowFunctions: true
function-call-argument-newline:
- error
- consistent
function-paren-newline: 'off'
generator-star-spacing: error
global-require: 'off'
grouped-accessor-pairs: 'off'
guard-for-in: error
handle-callback-err: error
id-blacklist: error
id-length: 'off'
id-match: error
implicit-arrow-linebreak:
- error
- beside
indent:
- error
- 4
indent-legacy: 'off'
init-declarations: error
jsx-quotes: error
key-spacing: error
keyword-spacing:
- error
- after: true
before: true
line-comment-position: 'off'
linebreak-style:
- error
- unix
lines-around-comment: error
lines-around-directive: error
lines-between-class-members:
- error
- always
max-classes-per-file: 'off'
max-depth: error
max-len: 'off'
max-lines: 'off'
max-lines-per-function: 'off'
max-nested-callbacks: error
max-params: 'off'
max-statements: 'off'
max-statements-per-line: error
multiline-comment-style:
- error
- separate-lines
multiline-ternary: 'off'
new-cap: error
new-parens: error
newline-after-var: 'off'
newline-before-return: 'off'
newline-per-chained-call: 'off'
no-alert: 'off'
no-array-constructor: error
no-await-in-loop: error
no-bitwise: 'off'
no-buffer-constructor: 'off'
no-caller: error
no-catch-shadow: error
no-confusing-arrow: error
no-console: error
no-constructor-return: error
no-continue: 'off'
no-div-regex: 'off'
no-duplicate-imports: error
no-else-return: 'off'
no-empty-function: 'off'
no-eq-null: error
no-eval: error
no-extend-native: error
no-extra-bind: error
no-extra-label: error
no-extra-parens: 'off'
no-floating-decimal: error
no-implicit-globals: error
no-implied-eval: error
no-inline-comments: 'off'
no-invalid-this: error
no-iterator: error
no-label-var: error
no-labels: error
no-lone-blocks: error
no-lonely-if: error
no-loop-func: 'off'
no-loss-of-precision: error
no-magic-numbers: 'off'
no-mixed-operators: error
no-mixed-requires: error
no-multi-assign: error
no-multi-spaces:
- error
- ignoreEOLComments: true
no-multi-str: error
no-multiple-empty-lines: error
no-native-reassign: error
no-negated-condition: 'off'
no-negated-in-lhs: error
no-nested-ternary: error
no-new: 'off'
no-new-func: error
no-new-object: error
no-new-require: error
no-new-wrappers: error
no-octal-escape: error
no-param-reassign: 'off'
no-path-concat: error
no-plusplus: 'off'
no-process-env: error
no-process-exit: error
no-proto: error
no-restricted-exports: error
no-restricted-globals: error
no-restricted-imports: error
no-restricted-modules: error
no-restricted-properties: error
no-restricted-syntax: error
no-return-assign: error
no-return-await: error
no-script-url: error
no-self-compare: error
no-sequences: error
no-shadow: 'off'
no-spaced-func: error
no-sync: error
no-tabs: error
no-template-curly-in-string: error
no-ternary: 'off'
no-throw-literal: 'off'
no-trailing-spaces: error
no-undef-init: error
no-undefined: 'off'
no-underscore-dangle: 'off'
no-unmodified-loop-condition: error
no-unneeded-ternary: error
no-unused-expressions: error
no-unused-vars: 'off'
no-use-before-define: 'off'
no-useless-backreference: error
no-useless-call: error
no-useless-computed-key: error
no-useless-concat: error
no-useless-constructor: error
no-useless-escape: 'off'
no-useless-rename: error
no-useless-return: error
no-var: 'off'
no-void: error
no-warning-comments: warn
no-whitespace-before-property: error
nonblock-statement-body-position: error
object-curly-newline: error
object-curly-spacing:
- error
- never
object-shorthand: 'off'
one-var: 'off'
one-var-declaration-per-line: error
operator-assignment:
- error
- always
operator-linebreak: 'off'
padded-blocks: 'off'
padding-line-between-statements: error
prefer-arrow-callback: error
prefer-const: 'off'
prefer-destructuring: 'off'
prefer-exponentiation-operator: 'off'
prefer-named-capture-group: 'off'
prefer-numeric-literals: error
prefer-object-spread: 'off'
prefer-promise-reject-errors: 'off'
prefer-reflect: 'off'
prefer-regex-literals: warn
prefer-rest-params: 'off'
prefer-spread: 'off'
prefer-template: 'off'
quote-props: 'off'
quotes: 'off'
radix:
- error
- as-needed
require-atomic-updates: error
require-await: error
require-jsdoc: 'off'
require-unicode-regexp: 'off'
rest-spread-spacing: error
semi: 'off'
semi-spacing:
- error
- after: true
before: false
semi-style:
- error
- last
sort-imports: error
sort-keys: 'off'
sort-vars: error
space-before-blocks: error
space-before-function-paren: 'off'
space-in-parens:
- error
- never
space-infix-ops: error
space-unary-ops: error
spaced-comment:
- error
- always
strict: error
switch-colon-spacing: error
symbol-description: error
template-curly-spacing:
- error
- never
template-tag-spacing: error
unicode-bom:
- error
- never
valid-jsdoc: error
vars-on-top: error
wrap-iife: error
wrap-regex: error
yield-star-spacing: error
yoda: 'off'

4
client/.prettierrc.yml Normal file
View file

@ -0,0 +1,4 @@
parser: babel
printWidth: 79
tabWidth: 4
quoteProps: consistent

View file

@ -1,10 +1,10 @@
'use strict';
"use strict";
const cookies = require('js-cookie');
const request = require('superagent');
const events = require('./events.js');
const progress = require('./util/progress.js');
const uri = require('./util/uri.js');
const cookies = require("js-cookie");
const request = require("superagent");
const events = require("./events.js");
const progress = require("./util/progress.js");
const uri = require("./util/uri.js");
let fileTokens = {};
let remoteConfig = null;
@ -18,22 +18,22 @@ class Api extends events.EventTarget {
this.token = null;
this.cache = {};
this.allRanks = [
'anonymous',
'restricted',
'regular',
'power',
'moderator',
'administrator',
'nobody',
"anonymous",
"restricted",
"regular",
"power",
"moderator",
"administrator",
"nobody",
];
this.rankNames = new Map([
['anonymous', 'Anonymous'],
['restricted', 'Restricted user'],
['regular', 'Regular user'],
['power', 'Power user'],
['moderator', 'Moderator'],
['administrator', 'Administrator'],
['nobody', 'Nobody'],
["anonymous", "Anonymous"],
["restricted", "Restricted user"],
["regular", "Regular user"],
["power", "Power user"],
["moderator", "Moderator"],
["administrator", "Administrator"],
["nobody", "Nobody"],
]);
}
@ -43,11 +43,12 @@ class Api extends events.EventTarget {
resolve(this.cache[url]);
});
}
return this._wrappedRequest(url, request.get, {}, {}, options)
.then(response => {
return this._wrappedRequest(url, request.get, {}, {}, options).then(
(response) => {
this.cache[url] = response;
return Promise.resolve(response);
});
}
);
}
post(url, data, files, options) {
@ -67,8 +68,7 @@ class Api extends events.EventTarget {
fetchConfig() {
if (remoteConfig === null) {
return this.get(uri.formatApiLink('info'))
.then(response => {
return this.get(uri.formatApiLink("info")).then((response) => {
remoteConfig = response.config;
});
} else {
@ -115,7 +115,8 @@ class Api extends events.EventTarget {
continue;
}
const rankIndex = this.allRanks.indexOf(
remoteConfig.privileges[p]);
remoteConfig.privileges[p]
);
if (minViableRank === null || rankIndex < minViableRank) {
minViableRank = rankIndex;
}
@ -123,17 +124,16 @@ class Api extends events.EventTarget {
if (minViableRank === null) {
throw `Bad privilege name: ${lookup}`;
}
let myRank = this.user !== null ?
this.allRanks.indexOf(this.user.rank) :
0;
let myRank =
this.user !== null ? this.allRanks.indexOf(this.user.rank) : 0;
return myRank >= minViableRank;
}
loginFromCookies() {
const auth = cookies.getJSON('auth');
return auth && auth.user && auth.token ?
this.loginWithToken(auth.user, auth.token, true) :
Promise.resolve();
const auth = cookies.getJSON("auth");
return auth && auth.user && auth.token
? this.loginWithToken(auth.user, auth.token, true)
: Promise.resolve();
}
loginWithToken(userName, token, doRemember) {
@ -141,63 +141,74 @@ class Api extends events.EventTarget {
return new Promise((resolve, reject) => {
this.userName = userName;
this.token = token;
this.get('/user/' + userName + '?bump-login=true')
.then(response => {
this.get("/user/" + userName + "?bump-login=true").then(
(response) => {
const options = {};
if (doRemember) {
options.expires = 365;
}
cookies.set(
'auth',
{'user': userName, 'token': token},
options);
"auth",
{ user: userName, token: token },
options
);
this.user = response;
resolve();
this.dispatchEvent(new CustomEvent('login'));
}, error => {
this.dispatchEvent(new CustomEvent("login"));
},
(error) => {
reject(error);
this.logout();
});
}
);
});
}
createToken(userName, options) {
let userTokenRequest = {
enabled: true,
note: 'Web Login Token'
note: "Web Login Token",
};
if (typeof options.expires !== 'undefined') {
userTokenRequest.expirationTime = new Date().addDays(options.expires).toISOString()
if (typeof options.expires !== "undefined") {
userTokenRequest.expirationTime = new Date()
.addDays(options.expires)
.toISOString();
}
return new Promise((resolve, reject) => {
this.post('/user-token/' + userName, userTokenRequest)
.then(response => {
this.post("/user-token/" + userName, userTokenRequest).then(
(response) => {
cookies.set(
'auth',
{'user': userName, 'token': response.token},
options);
"auth",
{ user: userName, token: response.token },
options
);
this.userName = userName;
this.token = response.token;
this.userPassword = null;
}, error => {
},
(error) => {
reject(error);
});
}
);
});
}
deleteToken(userName, userToken) {
return new Promise((resolve, reject) => {
this.delete('/user-token/' + userName + '/' + userToken, {})
.then(response => {
this.delete("/user-token/" + userName + "/" + userToken, {}).then(
(response) => {
const options = {};
cookies.set(
'auth',
{'user': userName, 'token': null},
options);
"auth",
{ user: userName, token: null },
options
);
resolve();
}, error => {
},
(error) => {
reject(error);
});
}
);
});
}
@ -206,8 +217,8 @@ class Api extends events.EventTarget {
return new Promise((resolve, reject) => {
this.userName = userName;
this.userPassword = userPassword;
this.get('/user/' + userName + '?bump-login=true')
.then(response => {
this.get("/user/" + userName + "?bump-login=true").then(
(response) => {
const options = {};
if (doRemember) {
options.expires = 365;
@ -215,22 +226,26 @@ class Api extends events.EventTarget {
this.createToken(this.userName, options);
this.user = response;
resolve();
this.dispatchEvent(new CustomEvent('login'));
}, error => {
this.dispatchEvent(new CustomEvent("login"));
},
(error) => {
reject(error);
this.logout();
});
}
);
});
}
logout() {
let self = this;
this.deleteToken(this.userName, this.token)
.then(response => {
this.deleteToken(this.userName, this.token).then(
(response) => {
self._logout();
}, error => {
},
(error) => {
self._logout();
});
}
);
}
_logout() {
@ -238,17 +253,19 @@ class Api extends events.EventTarget {
this.userName = null;
this.userPassword = null;
this.token = null;
this.dispatchEvent(new CustomEvent('logout'));
this.dispatchEvent(new CustomEvent("logout"));
}
forget() {
cookies.remove('auth');
cookies.remove("auth");
}
isLoggedIn(user) {
if (user) {
return this.userName !== null &&
this.userName.toLowerCase() === user.name.toLowerCase();
return (
this.userName !== null &&
this.userName.toLowerCase() === user.name.toLowerCase()
);
} else {
return this.userName !== null;
}
@ -259,8 +276,7 @@ class Api extends events.EventTarget {
}
_getFullUrl(url) {
const fullUrl =
('api/' + url).replace(/([^:])\/+/g, '$1/');
const fullUrl = ("api/" + url).replace(/([^:])\/+/g, "$1/");
const matches = fullUrl.match(/^([^?]*)\??(.*)$/);
const baseUrl = matches[1];
const request = matches[2];
@ -285,7 +301,7 @@ class Api extends events.EventTarget {
const file = files[key];
const fileId = this._getFileId(file);
if (fileTokens[fileId]) {
data[key + 'Token'] = fileTokens[fileId];
data[key + "Token"] = fileTokens[fileId];
} else {
promise = promise
.then(() => {
@ -293,33 +309,40 @@ class Api extends events.EventTarget {
abortFunction = () => uploadPromise.abort();
return uploadPromise;
})
.then(token => {
.then((token) => {
abortFunction = () => {};
fileTokens[fileId] = token;
data[key + 'Token'] = token;
data[key + "Token"] = token;
return Promise.resolve();
});
}
}
}
promise = promise.then(
() => {
promise = promise
.then(() => {
let requestPromise = this._rawRequest(
url, requestFactory, data, {}, options);
url,
requestFactory,
data,
{},
options
);
abortFunction = () => requestPromise.abort();
return requestPromise;
})
.catch(error => {
if (error.response && error.response.name ===
'MissingOrExpiredRequiredFileError') {
.catch((error) => {
if (
error.response &&
error.response.name === "MissingOrExpiredRequiredFileError"
) {
for (let key of Object.keys(files)) {
const file = files[key];
const fileId = this._getFileId(file);
fileTokens[fileId] = null;
}
error.message =
'The uploaded file has expired; ' +
'please resend the form to reupload.';
"The uploaded file has expired; " +
"please resend the form to reupload.";
}
return Promise.reject(error);
});
@ -331,10 +354,14 @@ class Api extends events.EventTarget {
let abortFunction = () => {};
let returnedPromise = new Promise((resolve, reject) => {
let uploadPromise = this._rawRequest(
'uploads', request.post, {}, {content: file}, options);
"uploads",
request.post,
{},
{ content: file },
options
);
abortFunction = () => uploadPromise.abort();
return uploadPromise.then(
response => {
return uploadPromise.then((response) => {
abortFunction = () => {};
return resolve(response.token);
}, reject);
@ -352,7 +379,7 @@ class Api extends events.EventTarget {
let returnedPromise = new Promise((resolve, reject) => {
let req = requestFactory(fullUrl);
req.set('Accept', 'application/json');
req.set("Accept", "application/json");
if (query) {
req.query(query);
@ -362,7 +389,7 @@ class Api extends events.EventTarget {
for (let key of Object.keys(files)) {
const value = files[key];
if (value.constructor === String) {
data[key + 'Url'] = value;
data[key + "Url"] = value;
} else {
req.attach(key, value || new Blob());
}
@ -371,9 +398,9 @@ class Api extends events.EventTarget {
if (data) {
if (files && Object.keys(files).length) {
req.attach('metadata', new Blob([JSON.stringify(data)]));
req.attach("metadata", new Blob([JSON.stringify(data)]));
} else {
req.set('Content-Type', 'application/json');
req.set("Content-Type", "application/json");
req.send(data);
}
}
@ -382,19 +409,28 @@ class Api extends events.EventTarget {
if (this.userName && this.token) {
req.auth = null;
// eslint-disable-next-line no-undef
req.set('Authorization', 'Token ' + new Buffer(
this.userName + ":" + this.token).toString('base64'))
req.set(
"Authorization",
"Token " +
new Buffer(
this.userName + ":" + this.token
).toString("base64")
);
} else if (this.userName && this.userPassword) {
req.auth(
this.userName,
encodeURIComponent(this.userPassword)
.replace(/%([0-9A-F]{2})/g, (match, p1) => {
return String.fromCharCode('0x' + p1);
}));
encodeURIComponent(this.userPassword).replace(
/%([0-9A-F]{2})/g,
(match, p1) => {
return String.fromCharCode("0x" + p1);
}
)
);
}
} catch (e) {
reject(
new Error('Authentication error (malformed credentials)'));
new Error("Authentication error (malformed credentials)")
);
}
if (!options.noProgress) {
@ -405,7 +441,8 @@ class Api extends events.EventTarget {
req.abort(); // does *NOT* call the callback passed in .end()
progress.done();
reject(
new Error('The request was aborted due to user cancel.'));
new Error("The request was aborted due to user cancel.")
);
};
req.end((error, response) => {
@ -414,7 +451,8 @@ class Api extends events.EventTarget {
if (error) {
if (response && response.body) {
error = new Error(
response.body.description || 'Unknown error');
response.body.description || "Unknown error"
);
error.response = response.body;
}
reject(error);

View file

@ -1,4 +1,4 @@
'use strict';
"use strict";
const config = require('./.config.autogen.json');
const config = require("./.config.autogen.json");
module.exports = config;

View file

@ -1,38 +1,40 @@
'use strict';
"use strict";
const router = require('../router.js');
const api = require('../api.js');
const tags = require('../tags.js');
const pools = require('../pools.js');
const uri = require('../util/uri.js');
const topNavigation = require('../models/top_navigation.js');
const LoginView = require('../views/login_view.js');
const router = require("../router.js");
const api = require("../api.js");
const tags = require("../tags.js");
const pools = require("../pools.js");
const uri = require("../util/uri.js");
const topNavigation = require("../models/top_navigation.js");
const LoginView = require("../views/login_view.js");
class LoginController {
constructor() {
api.forget();
topNavigation.activate('login');
topNavigation.setTitle('Login');
topNavigation.activate("login");
topNavigation.setTitle("Login");
this._loginView = new LoginView();
this._loginView.addEventListener('submit', e => this._evtLogin(e));
this._loginView.addEventListener("submit", (e) => this._evtLogin(e));
}
_evtLogin(e) {
this._loginView.clearMessages();
this._loginView.disableForm();
api.forget();
api.login(e.detail.name, e.detail.password, e.detail.remember)
.then(() => {
api.login(e.detail.name, e.detail.password, e.detail.remember).then(
() => {
const ctx = router.show(uri.formatClientLink());
ctx.controller.showSuccess('Logged in');
ctx.controller.showSuccess("Logged in");
// reload tag category color map, this is required when `tag_categories:list` has a permission other than anonymous
tags.refreshCategoryColorMap();
pools.refreshCategoryColorMap();
}, error => {
},
(error) => {
this._loginView.showError(error.message);
this._loginView.enableForm();
});
}
);
}
}
@ -41,15 +43,15 @@ class LogoutController {
api.forget();
api.logout();
const ctx = router.show(uri.formatClientLink());
ctx.controller.showSuccess('Logged out');
ctx.controller.showSuccess("Logged out");
}
}
module.exports = router => {
router.enter(['login'], (ctx, next) => {
module.exports = (router) => {
router.enter(["login"], (ctx, next) => {
ctx.controller = new LoginController();
});
router.enter(['logout'], (ctx, next) => {
router.enter(["logout"], (ctx, next) => {
ctx.controller = new LogoutController();
});
};

View file

@ -1,19 +1,19 @@
'use strict';
"use strict";
const api = require('../api.js');
const topNavigation = require('../models/top_navigation.js');
const EmptyView = require('../views/empty_view.js');
const api = require("../api.js");
const topNavigation = require("../models/top_navigation.js");
const EmptyView = require("../views/empty_view.js");
class BasePostController {
constructor(ctx) {
if (!api.hasPrivilege('posts:view')) {
if (!api.hasPrivilege("posts:view")) {
this._view = new EmptyView();
this._view.showError('You don\'t have privileges to view posts.');
this._view.showError("You don't have privileges to view posts.");
return;
}
topNavigation.activate('posts');
topNavigation.setTitle('Post #' + ctx.parameters.id.toString());
topNavigation.activate("posts");
topNavigation.setTitle("Post #" + ctx.parameters.id.toString());
}
}

View file

@ -1,51 +1,55 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const PostList = require('../models/post_list.js');
const topNavigation = require('../models/top_navigation.js');
const PageController = require('../controllers/page_controller.js');
const CommentsPageView = require('../views/comments_page_view.js');
const EmptyView = require('../views/empty_view.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const PostList = require("../models/post_list.js");
const topNavigation = require("../models/top_navigation.js");
const PageController = require("../controllers/page_controller.js");
const CommentsPageView = require("../views/comments_page_view.js");
const EmptyView = require("../views/empty_view.js");
const fields = ['id', 'comments', 'commentCount', 'thumbnailUrl'];
const fields = ["id", "comments", "commentCount", "thumbnailUrl"];
class CommentsController {
constructor(ctx) {
if (!api.hasPrivilege('comments:list')) {
if (!api.hasPrivilege("comments:list")) {
this._view = new EmptyView();
this._view.showError(
'You don\'t have privileges to view comments.');
"You don't have privileges to view comments."
);
return;
}
topNavigation.activate('comments');
topNavigation.setTitle('Listing comments');
topNavigation.activate("comments");
topNavigation.setTitle("Listing comments");
this._pageController = new PageController();
this._pageController.run({
parameters: ctx.parameters,
defaultLimit: 10,
getClientUrlForPage: (offset, limit) => {
const parameters = Object.assign(
{}, ctx.parameters, {offset: offset, limit: limit});
return uri.formatClientLink('comments', parameters);
const parameters = Object.assign({}, ctx.parameters, {
offset: offset,
limit: limit,
});
return uri.formatClientLink("comments", parameters);
},
requestPage: (offset, limit) => {
return PostList.search(
'sort:comment-date comment-count-min:1',
"sort:comment-date comment-count-min:1",
offset,
limit,
fields);
fields
);
},
pageRenderer: pageCtx => {
pageRenderer: (pageCtx) => {
Object.assign(pageCtx, {
canViewPosts: api.hasPrivilege('posts:view'),
canViewPosts: api.hasPrivilege("posts:view"),
});
const view = new CommentsPageView(pageCtx);
view.addEventListener('submit', e => this._evtUpdate(e));
view.addEventListener('score', e => this._evtScore(e));
view.addEventListener('delete', e => this._evtDelete(e));
view.addEventListener("submit", (e) => this._evtUpdate(e));
view.addEventListener("score", (e) => this._evtScore(e));
view.addEventListener("delete", (e) => this._evtDelete(e));
return view;
},
});
@ -54,26 +58,27 @@ class CommentsController {
_evtUpdate(e) {
// TODO: disable form
e.detail.comment.text = e.detail.text;
e.detail.comment.save()
.catch(error => {
e.detail.comment.save().catch((error) => {
e.detail.target.showError(error.message);
// TODO: enable form
});
}
_evtScore(e) {
e.detail.comment.setScore(e.detail.score)
.catch(error => window.alert(error.message));
e.detail.comment
.setScore(e.detail.score)
.catch((error) => window.alert(error.message));
}
_evtDelete(e) {
e.detail.comment.delete()
.catch(error => window.alert(error.message));
e.detail.comment
.delete()
.catch((error) => window.alert(error.message));
}
}
module.exports = router => {
router.enter(['comments'], (ctx, next) => {
module.exports = (router) => {
router.enter(["comments"], (ctx, next) => {
new CommentsController(ctx);
});
}
};

View file

@ -1,24 +1,24 @@
'use strict';
"use strict";
const topNavigation = require('../models/top_navigation.js');
const HelpView = require('../views/help_view.js');
const topNavigation = require("../models/top_navigation.js");
const HelpView = require("../views/help_view.js");
class HelpController {
constructor(section, subsection) {
topNavigation.activate('help');
topNavigation.setTitle('Help');
topNavigation.activate("help");
topNavigation.setTitle("Help");
this._helpView = new HelpView(section, subsection);
}
}
module.exports = router => {
router.enter(['help'], (ctx, next) => {
module.exports = (router) => {
router.enter(["help"], (ctx, next) => {
new HelpController();
});
router.enter(['help', ':section'], (ctx, next) => {
router.enter(["help", ":section"], (ctx, next) => {
new HelpController(ctx.parameters.section);
});
router.enter(['help', ':section', ':subsection'], (ctx, next) => {
router.enter(["help", ":section", ":subsection"], (ctx, next) => {
new HelpController(ctx.parameters.section, ctx.parameters.subsection);
});
};

View file

@ -1,26 +1,26 @@
'use strict';
"use strict";
const api = require('../api.js');
const config = require('../config.js');
const Info = require('../models/info.js');
const topNavigation = require('../models/top_navigation.js');
const HomeView = require('../views/home_view.js');
const api = require("../api.js");
const config = require("../config.js");
const Info = require("../models/info.js");
const topNavigation = require("../models/top_navigation.js");
const HomeView = require("../views/home_view.js");
class HomeController {
constructor() {
topNavigation.activate('home');
topNavigation.setTitle('Home');
topNavigation.activate("home");
topNavigation.setTitle("Home");
this._homeView = new HomeView({
name: api.getName(),
version: config.meta.version,
buildDate: config.meta.buildDate,
canListSnapshots: api.hasPrivilege('snapshots:list'),
canListPosts: api.hasPrivilege('posts:list'),
canListSnapshots: api.hasPrivilege("snapshots:list"),
canListPosts: api.hasPrivilege("posts:list"),
});
Info.get()
.then(info => {
Info.get().then(
(info) => {
this._homeView.setStats({
diskUsage: info.diskUsage,
postCount: info.postCount,
@ -31,7 +31,8 @@ class HomeController {
featuringTime: info.featuringTime,
});
},
error => this._homeView.showError(error.message));
(error) => this._homeView.showError(error.message)
);
}
showSuccess(message) {
@ -43,8 +44,8 @@ class HomeController {
}
}
module.exports = router => {
module.exports = (router) => {
router.enter([], (ctx, next) => {
ctx.controller = new HomeController();
});
}
};

View file

@ -1,18 +1,18 @@
'use strict';
"use strict";
const topNavigation = require('../models/top_navigation.js');
const NotFoundView = require('../views/not_found_view.js');
const topNavigation = require("../models/top_navigation.js");
const NotFoundView = require("../views/not_found_view.js");
class NotFoundController {
constructor(path) {
topNavigation.activate('');
topNavigation.setTitle('Not found');
topNavigation.activate("");
topNavigation.setTitle("Not found");
this._notFoundView = new NotFoundView(path);
}
}
module.exports = router => {
module.exports = (router) => {
router.enter(null, (ctx, next) => {
ctx.controller = new NotFoundController(ctx.canonicalPath);
});
}
};

View file

@ -1,8 +1,8 @@
'use strict';
"use strict";
const settings = require('../models/settings.js');
const EndlessPageView = require('../views/endless_page_view.js');
const ManualPageView = require('../views/manual_page_view.js');
const settings = require("../models/settings.js");
const EndlessPageView = require("../views/endless_page_view.js");
const ManualPageView = require("../views/manual_page_view.js");
class PageController {
constructor(ctx) {

View file

@ -1,19 +1,20 @@
'use strict';
"use strict";
const router = require('../router.js');
const api = require('../api.js');
const uri = require('../util/uri.js');
const topNavigation = require('../models/top_navigation.js');
const PasswordResetView = require('../views/password_reset_view.js');
const router = require("../router.js");
const api = require("../api.js");
const uri = require("../util/uri.js");
const topNavigation = require("../models/top_navigation.js");
const PasswordResetView = require("../views/password_reset_view.js");
class PasswordResetController {
constructor() {
topNavigation.activate('login');
topNavigation.setTitle('Password reminder');
topNavigation.activate("login");
topNavigation.setTitle("Password reminder");
this._passwordResetView = new PasswordResetView();
this._passwordResetView.addEventListener(
'submit', e => this._evtReset(e));
this._passwordResetView.addEventListener("submit", (e) =>
this._evtReset(e)
);
}
_evtReset(e) {
@ -21,15 +22,20 @@ class PasswordResetController {
this._passwordResetView.disableForm();
api.forget();
api.logout();
api.get(uri.formatApiLink('password-reset', e.detail.userNameOrEmail))
.then(() => {
api.get(
uri.formatApiLink("password-reset", e.detail.userNameOrEmail)
).then(
() => {
this._passwordResetView.showSuccess(
'E-mail has been sent. To finish the procedure, ' +
'please click the link it contains.');
}, error => {
"E-mail has been sent. To finish the procedure, " +
"please click the link it contains."
);
},
(error) => {
this._passwordResetView.showError(error.message);
this._passwordResetView.enableForm();
});
}
);
}
}
@ -38,26 +44,30 @@ class PasswordResetFinishController {
api.forget();
api.logout();
let password = null;
api.post(uri.formatApiLink('password-reset', name), {token: token})
.then(response => {
api.post(uri.formatApiLink("password-reset", name), { token: token })
.then((response) => {
password = response.password;
return api.login(name, password, false);
}).then(() => {
})
.then(
() => {
const ctx = router.show(uri.formatClientLink());
ctx.controller.showSuccess('New password: ' + password);
}, error => {
ctx.controller.showSuccess("New password: " + password);
},
(error) => {
const ctx = router.show(uri.formatClientLink());
ctx.controller.showError(error.message);
});
}
);
}
}
module.exports = router => {
router.enter(['password-reset'], (ctx, next) => {
module.exports = (router) => {
router.enter(["password-reset"], (ctx, next) => {
ctx.controller = new PasswordResetController();
});
router.enter(['password-reset', ':descriptor'], (ctx, next) => {
const [name, token] = ctx.parameters.descriptor.split(':', 2);
router.enter(["password-reset", ":descriptor"], (ctx, next) => {
const [name, token] = ctx.parameters.descriptor.split(":", 2);
ctx.controller = new PasswordResetFinishController(name, token);
});
};

View file

@ -1,57 +1,69 @@
'use strict';
"use strict";
const api = require('../api.js');
const pools = require('../pools.js');
const PoolCategoryList = require('../models/pool_category_list.js');
const topNavigation = require('../models/top_navigation.js');
const PoolCategoriesView = require('../views/pool_categories_view.js');
const EmptyView = require('../views/empty_view.js');
const api = require("../api.js");
const pools = require("../pools.js");
const PoolCategoryList = require("../models/pool_category_list.js");
const topNavigation = require("../models/top_navigation.js");
const PoolCategoriesView = require("../views/pool_categories_view.js");
const EmptyView = require("../views/empty_view.js");
class PoolCategoriesController {
constructor() {
if (!api.hasPrivilege('poolCategories:list')) {
if (!api.hasPrivilege("poolCategories:list")) {
this._view = new EmptyView();
this._view.showError(
'You don\'t have privileges to view pool categories.');
"You don't have privileges to view pool categories."
);
return;
}
topNavigation.activate('pools');
topNavigation.setTitle('Listing pools');
PoolCategoryList.get().then(response => {
topNavigation.activate("pools");
topNavigation.setTitle("Listing pools");
PoolCategoryList.get().then(
(response) => {
this._poolCategories = response.results;
this._view = new PoolCategoriesView({
poolCategories: this._poolCategories,
canEditName: api.hasPrivilege('poolCategories:edit:name'),
canEditColor: api.hasPrivilege('poolCategories:edit:color'),
canDelete: api.hasPrivilege('poolCategories:delete'),
canCreate: api.hasPrivilege('poolCategories:create'),
canSetDefault: api.hasPrivilege('poolCategories:setDefault'),
canEditName: api.hasPrivilege("poolCategories:edit:name"),
canEditColor: api.hasPrivilege(
"poolCategories:edit:color"
),
canDelete: api.hasPrivilege("poolCategories:delete"),
canCreate: api.hasPrivilege("poolCategories:create"),
canSetDefault: api.hasPrivilege(
"poolCategories:setDefault"
),
});
this._view.addEventListener('submit', e => this._evtSubmit(e));
}, error => {
this._view.addEventListener("submit", (e) =>
this._evtSubmit(e)
);
},
(error) => {
this._view = new EmptyView();
this._view.showError(error.message);
});
}
);
}
_evtSubmit(e) {
this._view.clearMessages();
this._view.disableForm();
this._poolCategories.save()
.then(() => {
this._poolCategories.save().then(
() => {
pools.refreshCategoryColorMap();
this._view.enableForm();
this._view.showSuccess('Changes saved.');
}, error => {
this._view.showSuccess("Changes saved.");
},
(error) => {
this._view.enableForm();
this._view.showError(error.message);
});
}
);
}
}
module.exports = router => {
router.enter(['pool-categories'], (ctx, next) => {
module.exports = (router) => {
router.enter(["pool-categories"], (ctx, next) => {
ctx.controller = new PoolCategoriesController(ctx, next);
});
};

View file

@ -1,35 +1,38 @@
'use strict';
"use strict";
const router = require('../router.js');
const api = require('../api.js');
const misc = require('../util/misc.js');
const uri = require('../util/uri.js');
const Pool = require('../models/pool.js');
const Post = require('../models/post.js');
const PoolCategoryList = require('../models/pool_category_list.js');
const topNavigation = require('../models/top_navigation.js');
const PoolView = require('../views/pool_view.js');
const EmptyView = require('../views/empty_view.js');
const router = require("../router.js");
const api = require("../api.js");
const misc = require("../util/misc.js");
const uri = require("../util/uri.js");
const Pool = require("../models/pool.js");
const Post = require("../models/post.js");
const PoolCategoryList = require("../models/pool_category_list.js");
const topNavigation = require("../models/top_navigation.js");
const PoolView = require("../views/pool_view.js");
const EmptyView = require("../views/empty_view.js");
class PoolController {
constructor(ctx, section) {
if (!api.hasPrivilege('pools:view')) {
if (!api.hasPrivilege("pools:view")) {
this._view = new EmptyView();
this._view.showError('You don\'t have privileges to view pools.');
this._view.showError("You don't have privileges to view pools.");
return;
}
Promise.all([
PoolCategoryList.get(),
Pool.get(ctx.parameters.id)
]).then(responses => {
Pool.get(ctx.parameters.id),
]).then(
(responses) => {
const [poolCategoriesResponse, pool] = responses;
topNavigation.activate('pools');
topNavigation.setTitle('Pool #' + pool.names[0]);
topNavigation.activate("pools");
topNavigation.setTitle("Pool #" + pool.names[0]);
this._name = ctx.parameters.name;
pool.addEventListener('change', e => this._evtSaved(e, section));
pool.addEventListener("change", (e) =>
this._evtSaved(e, section)
);
const categories = {};
for (let category of poolCategoriesResponse.results) {
@ -39,25 +42,35 @@ class PoolController {
this._view = new PoolView({
pool: pool,
section: section,
canEditAnything: api.hasPrivilege('pools:edit'),
canEditNames: api.hasPrivilege('pools:edit:names'),
canEditCategory: api.hasPrivilege('pools:edit:category'),
canEditDescription: api.hasPrivilege('pools:edit:description'),
canEditPosts: api.hasPrivilege('pools:edit:posts'),
canMerge: api.hasPrivilege('pools:merge'),
canDelete: api.hasPrivilege('pools:delete'),
canEditAnything: api.hasPrivilege("pools:edit"),
canEditNames: api.hasPrivilege("pools:edit:names"),
canEditCategory: api.hasPrivilege("pools:edit:category"),
canEditDescription: api.hasPrivilege(
"pools:edit:description"
),
canEditPosts: api.hasPrivilege("pools:edit:posts"),
canMerge: api.hasPrivilege("pools:merge"),
canDelete: api.hasPrivilege("pools:delete"),
categories: categories,
escapeColons: uri.escapeColons,
});
this._view.addEventListener('change', e => this._evtChange(e));
this._view.addEventListener('submit', e => this._evtUpdate(e));
this._view.addEventListener('merge', e => this._evtMerge(e));
this._view.addEventListener('delete', e => this._evtDelete(e));
}, error => {
this._view.addEventListener("change", (e) =>
this._evtChange(e)
);
this._view.addEventListener("submit", (e) =>
this._evtUpdate(e)
);
this._view.addEventListener("merge", (e) => this._evtMerge(e));
this._view.addEventListener("delete", (e) =>
this._evtDelete(e)
);
},
(error) => {
this._view = new EmptyView();
this._view.showError(error.message);
});
}
);
}
_evtChange(e) {
@ -67,7 +80,11 @@ class PoolController {
_evtSaved(e, section) {
misc.disableExitConfirmation();
if (this._name !== e.detail.pool.names[0]) {
router.replace(uri.formatClientLink('pool', e.detail.pool.id, section), null, false);
router.replace(
uri.formatClientLink("pool", e.detail.pool.id, section),
null,
false
);
}
}
@ -86,62 +103,74 @@ class PoolController {
if (e.detail.posts !== undefined) {
e.detail.pool.posts.clear();
for (let postId of e.detail.posts) {
e.detail.pool.posts.add(Post.fromResponse({id: parseInt(postId)}));
e.detail.pool.posts.add(
Post.fromResponse({ id: parseInt(postId) })
);
}
}
e.detail.pool.save().then(() => {
this._view.showSuccess('Pool saved.');
e.detail.pool.save().then(
() => {
this._view.showSuccess("Pool saved.");
this._view.enableForm();
}, error => {
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
_evtMerge(e) {
this._view.clearMessages();
this._view.disableForm();
e.detail.pool
.merge(e.detail.targetPoolId, e.detail.addAlias)
.then(() => {
this._view.showSuccess('Pool merged.');
e.detail.pool.merge(e.detail.targetPoolId, e.detail.addAlias).then(
() => {
this._view.showSuccess("Pool merged.");
this._view.enableForm();
router.replace(
uri.formatClientLink(
'pool', e.detail.targetPoolId, 'merge'),
"pool",
e.detail.targetPoolId,
"merge"
),
null,
false);
}, error => {
false
);
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
_evtDelete(e) {
this._view.clearMessages();
this._view.disableForm();
e.detail.pool.delete()
.then(() => {
const ctx = router.show(uri.formatClientLink('pools'));
ctx.controller.showSuccess('Pool deleted.');
}, error => {
e.detail.pool.delete().then(
() => {
const ctx = router.show(uri.formatClientLink("pools"));
ctx.controller.showSuccess("Pool deleted.");
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
}
module.exports = router => {
router.enter(['pool', ':id', 'edit'], (ctx, next) => {
ctx.controller = new PoolController(ctx, 'edit');
module.exports = (router) => {
router.enter(["pool", ":id", "edit"], (ctx, next) => {
ctx.controller = new PoolController(ctx, "edit");
});
router.enter(['pool', ':id', 'merge'], (ctx, next) => {
ctx.controller = new PoolController(ctx, 'merge');
router.enter(["pool", ":id", "merge"], (ctx, next) => {
ctx.controller = new PoolController(ctx, "merge");
});
router.enter(['pool', ':id', 'delete'], (ctx, next) => {
ctx.controller = new PoolController(ctx, 'delete');
router.enter(["pool", ":id", "delete"], (ctx, next) => {
ctx.controller = new PoolController(ctx, "delete");
});
router.enter(['pool', ':id'], (ctx, next) => {
ctx.controller = new PoolController(ctx, 'summary');
router.enter(["pool", ":id"], (ctx, next) => {
ctx.controller = new PoolController(ctx, "summary");
});
};

View file

@ -1,58 +1,65 @@
'use strict';
"use strict";
const router = require('../router.js');
const api = require('../api.js');
const misc = require('../util/misc.js');
const uri = require('../util/uri.js');
const PoolCategoryList = require('../models/pool_category_list.js');
const PoolCreateView = require('../views/pool_create_view.js');
const EmptyView = require('../views/empty_view.js');
const router = require("../router.js");
const api = require("../api.js");
const misc = require("../util/misc.js");
const uri = require("../util/uri.js");
const PoolCategoryList = require("../models/pool_category_list.js");
const PoolCreateView = require("../views/pool_create_view.js");
const EmptyView = require("../views/empty_view.js");
class PoolCreateController {
constructor(ctx) {
if (!api.hasPrivilege('pools:create')) {
if (!api.hasPrivilege("pools:create")) {
this._view = new EmptyView();
this._view.showError('You don\'t have privileges to create pools.');
this._view.showError("You don't have privileges to create pools.");
return;
}
PoolCategoryList.get().then(poolCategoriesResponse => {
PoolCategoryList.get().then(
(poolCategoriesResponse) => {
const categories = {};
for (let category of poolCategoriesResponse.results) {
categories[category.name] = category.name;
}
this._view = new PoolCreateView({
canCreate: api.hasPrivilege('pools:create'),
canCreate: api.hasPrivilege("pools:create"),
categories: categories,
escapeColons: uri.escapeColons,
});
this._view.addEventListener('submit', e => this._evtCreate(e));
}, error => {
this._view.addEventListener("submit", (e) =>
this._evtCreate(e)
);
},
(error) => {
this._view = new EmptyView();
this._view.showError(error.message);
});
}
);
}
_evtCreate(e) {
this._view.clearMessages();
this._view.disableForm();
api.post(uri.formatApiLink('pool'), e.detail)
.then(() => {
api.post(uri.formatApiLink("pool"), e.detail).then(
() => {
this._view.clearMessages();
misc.disableExitConfirmation();
const ctx = router.show(uri.formatClientLink('pools'));
ctx.controller.showSuccess('Pool created.');
}, error => {
const ctx = router.show(uri.formatClientLink("pools"));
ctx.controller.showSuccess("Pool created.");
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
}
module.exports = router => {
router.enter(['pool', 'create'], (ctx, next) => {
ctx.controller = new PoolCreateController(ctx, 'create');
module.exports = (router) => {
router.enter(["pool", "create"], (ctx, next) => {
ctx.controller = new PoolCreateController(ctx, "create");
});
};

View file

@ -1,47 +1,51 @@
'use strict';
"use strict";
const router = require('../router.js');
const api = require('../api.js');
const uri = require('../util/uri.js');
const PoolList = require('../models/pool_list.js');
const topNavigation = require('../models/top_navigation.js');
const PageController = require('../controllers/page_controller.js');
const PoolsHeaderView = require('../views/pools_header_view.js');
const PoolsPageView = require('../views/pools_page_view.js');
const EmptyView = require('../views/empty_view.js');
const router = require("../router.js");
const api = require("../api.js");
const uri = require("../util/uri.js");
const PoolList = require("../models/pool_list.js");
const topNavigation = require("../models/top_navigation.js");
const PageController = require("../controllers/page_controller.js");
const PoolsHeaderView = require("../views/pools_header_view.js");
const PoolsPageView = require("../views/pools_page_view.js");
const EmptyView = require("../views/empty_view.js");
const fields = [
'id',
'names',
'posts',
'creationTime',
'postCount',
'category'
"id",
"names",
"posts",
"creationTime",
"postCount",
"category",
];
class PoolListController {
constructor(ctx) {
this._pageController = new PageController();
if (!api.hasPrivilege('pools:list')) {
if (!api.hasPrivilege("pools:list")) {
this._view = new EmptyView();
this._view.showError('You don\'t have privileges to view pools.');
this._view.showError("You don't have privileges to view pools.");
return;
}
this._ctx = ctx;
topNavigation.activate('pools');
topNavigation.setTitle('Listing pools');
topNavigation.activate("pools");
topNavigation.setTitle("Listing pools");
this._headerView = new PoolsHeaderView({
hostNode: this._pageController.view.pageHeaderHolderNode,
parameters: ctx.parameters,
canCreate: api.hasPrivilege('pools:create'),
canEditPoolCategories: api.hasPrivilege('poolCategories:edit'),
canCreate: api.hasPrivilege("pools:create"),
canEditPoolCategories: api.hasPrivilege("poolCategories:edit"),
});
this._headerView.addEventListener(
'submit', e => this._evtSubmit(e), 'navigate', e => this._evtNavigate(e));
"submit",
(e) => this._evtSubmit(e),
"navigate",
(e) => this._evtNavigate(e)
);
this._syncPageController();
}
@ -57,24 +61,27 @@ class PoolListController {
_evtSubmit(e) {
this._view.clearMessages();
this._view.disableForm();
e.detail.pool.save()
.then(() => {
this._installView(e.detail.pool, 'edit');
this._view.showSuccess('Pool created.');
e.detail.pool.save().then(
() => {
this._installView(e.detail.pool, "edit");
this._view.showSuccess("Pool created.");
router.replace(
uri.formatClientLink(
'pool', e.detail.pool.id, 'edit'),
uri.formatClientLink("pool", e.detail.pool.id, "edit"),
null,
false);
}, error => {
false
);
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
_evtNavigate(e) {
router.showNoDispatch(
uri.formatClientLink('pools', e.detail.parameters));
uri.formatClientLink("pools", e.detail.parameters)
);
Object.assign(this._ctx.parameters, e.detail.parameters);
this._syncPageController();
}
@ -84,25 +91,29 @@ class PoolListController {
parameters: this._ctx.parameters,
defaultLimit: 50,
getClientUrlForPage: (offset, limit) => {
const parameters = Object.assign(
{}, this._ctx.parameters, {offset: offset, limit: limit});
return uri.formatClientLink('pools', parameters);
const parameters = Object.assign({}, this._ctx.parameters, {
offset: offset,
limit: limit,
});
return uri.formatClientLink("pools", parameters);
},
requestPage: (offset, limit) => {
return PoolList.search(
this._ctx.parameters.query, offset, limit, fields);
this._ctx.parameters.query,
offset,
limit,
fields
);
},
pageRenderer: pageCtx => {
pageRenderer: (pageCtx) => {
return new PoolsPageView(pageCtx);
},
});
}
}
module.exports = router => {
router.enter(
['pools'],
(ctx, next) => {
module.exports = (router) => {
router.enter(["pools"], (ctx, next) => {
ctx.controller = new PoolListController(ctx);
});
};

View file

@ -1,28 +1,33 @@
'use strict';
"use strict";
const router = require('../router.js');
const api = require('../api.js');
const misc = require('../util/misc.js');
const uri = require('../util/uri.js');
const settings = require('../models/settings.js');
const Post = require('../models/post.js');
const PostList = require('../models/post_list.js');
const PostDetailView = require('../views/post_detail_view.js');
const BasePostController = require('./base_post_controller.js');
const EmptyView = require('../views/empty_view.js');
const router = require("../router.js");
const api = require("../api.js");
const misc = require("../util/misc.js");
const uri = require("../util/uri.js");
const settings = require("../models/settings.js");
const Post = require("../models/post.js");
const PostList = require("../models/post_list.js");
const PostDetailView = require("../views/post_detail_view.js");
const BasePostController = require("./base_post_controller.js");
const EmptyView = require("../views/empty_view.js");
class PostDetailController extends BasePostController {
constructor(ctx, section) {
super(ctx);
Post.get(ctx.parameters.id).then(post => {
Post.get(ctx.parameters.id).then(
(post) => {
this._id = ctx.parameters.id;
post.addEventListener('change', e => this._evtSaved(e, section));
post.addEventListener("change", (e) =>
this._evtSaved(e, section)
);
this._installView(post, section);
}, error => {
},
(error) => {
this._view = new EmptyView();
this._view.showError(error.message);
});
}
);
}
showSuccess(message) {
@ -33,58 +38,68 @@ class PostDetailController extends BasePostController {
this._view = new PostDetailView({
post: post,
section: section,
canMerge: api.hasPrivilege('posts:merge'),
canMerge: api.hasPrivilege("posts:merge"),
});
this._view.addEventListener('select', e => this._evtSelect(e));
this._view.addEventListener('merge', e => this._evtMerge(e));
this._view.addEventListener("select", (e) => this._evtSelect(e));
this._view.addEventListener("merge", (e) => this._evtMerge(e));
}
_evtSelect(e) {
this._view.clearMessages();
this._view.disableForm();
Post.get(e.detail.postId).then(post => {
Post.get(e.detail.postId).then(
(post) => {
this._view.selectPost(post);
this._view.enableForm();
}, error => {
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
_evtSaved(e, section) {
misc.disableExitConfirmation();
if (this._id !== e.detail.post.id) {
router.replace(
uri.formatClientLink('post', e.detail.post.id, section),
uri.formatClientLink("post", e.detail.post.id, section),
null,
false);
false
);
}
}
_evtMerge(e) {
this._view.clearMessages();
this._view.disableForm();
e.detail.post.merge(e.detail.targetPost.id, e.detail.useOldContent)
.then(() => {
this._installView(e.detail.post, 'merge');
this._view.showSuccess('Post merged.');
e.detail.post
.merge(e.detail.targetPost.id, e.detail.useOldContent)
.then(
() => {
this._installView(e.detail.post, "merge");
this._view.showSuccess("Post merged.");
router.replace(
uri.formatClientLink(
'post', e.detail.targetPost.id, 'merge'),
"post",
e.detail.targetPost.id,
"merge"
),
null,
false);
}, error => {
false
);
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
}
module.exports = router => {
router.enter(
['post', ':id', 'merge'],
(ctx, next) => {
ctx.controller = new PostDetailController(ctx, 'merge');
module.exports = (router) => {
router.enter(["post", ":id", "merge"], (ctx, next) => {
ctx.controller = new PostDetailController(ctx, "merge");
});
};

View file

@ -1,48 +1,56 @@
'use strict';
"use strict";
const router = require('../router.js');
const api = require('../api.js');
const settings = require('../models/settings.js');
const uri = require('../util/uri.js');
const PostList = require('../models/post_list.js');
const topNavigation = require('../models/top_navigation.js');
const PageController = require('../controllers/page_controller.js');
const PostsHeaderView = require('../views/posts_header_view.js');
const PostsPageView = require('../views/posts_page_view.js');
const EmptyView = require('../views/empty_view.js');
const router = require("../router.js");
const api = require("../api.js");
const settings = require("../models/settings.js");
const uri = require("../util/uri.js");
const PostList = require("../models/post_list.js");
const topNavigation = require("../models/top_navigation.js");
const PageController = require("../controllers/page_controller.js");
const PostsHeaderView = require("../views/posts_header_view.js");
const PostsPageView = require("../views/posts_page_view.js");
const EmptyView = require("../views/empty_view.js");
const fields = [
'id', 'thumbnailUrl', 'type', 'safety',
'score', 'favoriteCount', 'commentCount', 'tags', 'version'
"id",
"thumbnailUrl",
"type",
"safety",
"score",
"favoriteCount",
"commentCount",
"tags",
"version",
];
class PostListController {
constructor(ctx) {
this._pageController = new PageController();
if (!api.hasPrivilege('posts:list')) {
if (!api.hasPrivilege("posts:list")) {
this._view = new EmptyView();
this._view.showError('You don\'t have privileges to view posts.');
this._view.showError("You don't have privileges to view posts.");
return;
}
this._ctx = ctx;
topNavigation.activate('posts');
topNavigation.setTitle('Listing posts');
topNavigation.activate("posts");
topNavigation.setTitle("Listing posts");
this._headerView = new PostsHeaderView({
hostNode: this._pageController.view.pageHeaderHolderNode,
parameters: ctx.parameters,
enableSafety: api.safetyEnabled(),
canBulkEditTags: api.hasPrivilege('posts:bulk-edit:tags'),
canBulkEditSafety: api.hasPrivilege('posts:bulk-edit:safety'),
canBulkEditTags: api.hasPrivilege("posts:bulk-edit:tags"),
canBulkEditSafety: api.hasPrivilege("posts:bulk-edit:safety"),
bulkEdit: {
tags: this._bulkEditTags
tags: this._bulkEditTags,
},
});
this._headerView.addEventListener(
'navigate', e => this._evtNavigate(e));
this._headerView.addEventListener("navigate", (e) =>
this._evtNavigate(e)
);
this._syncPageController();
}
@ -52,33 +60,35 @@ class PostListController {
}
get _bulkEditTags() {
return (this._ctx.parameters.tag || '').split(/\s+/).filter(s => s);
return (this._ctx.parameters.tag || "").split(/\s+/).filter((s) => s);
}
_evtNavigate(e) {
router.showNoDispatch(
uri.formatClientLink('posts', e.detail.parameters));
uri.formatClientLink("posts", e.detail.parameters)
);
Object.assign(this._ctx.parameters, e.detail.parameters);
this._syncPageController();
}
_evtTag(e) {
Promise.all(
this._bulkEditTags.map(tag => e.detail.post.tags.addByName(tag)))
this._bulkEditTags.map((tag) => e.detail.post.tags.addByName(tag))
)
.then(e.detail.post.save())
.catch(error => window.alert(error.message));
.catch((error) => window.alert(error.message));
}
_evtUntag(e) {
for (let tag of this._bulkEditTags) {
e.detail.post.tags.removeByName(tag);
}
e.detail.post.save().catch(error => window.alert(error.message));
e.detail.post.save().catch((error) => window.alert(error.message));
}
_evtChangeSafety(e) {
e.detail.post.safety = e.detail.safety;
e.detail.post.save().catch(error => window.alert(error.message));
e.detail.post.save().catch((error) => window.alert(error.message));
}
_syncPageController() {
@ -86,39 +96,45 @@ class PostListController {
parameters: this._ctx.parameters,
defaultLimit: parseInt(settings.get().postsPerPage),
getClientUrlForPage: (offset, limit) => {
const parameters = Object.assign(
{}, this._ctx.parameters, {offset: offset, limit: limit});
return uri.formatClientLink('posts', parameters);
const parameters = Object.assign({}, this._ctx.parameters, {
offset: offset,
limit: limit,
});
return uri.formatClientLink("posts", parameters);
},
requestPage: (offset, limit) => {
return PostList.search(
this._ctx.parameters.query, offset, limit, fields);
this._ctx.parameters.query,
offset,
limit,
fields
);
},
pageRenderer: pageCtx => {
pageRenderer: (pageCtx) => {
Object.assign(pageCtx, {
canViewPosts: api.hasPrivilege('posts:view'),
canBulkEditTags: api.hasPrivilege('posts:bulk-edit:tags'),
canBulkEditSafety:
api.hasPrivilege('posts:bulk-edit:safety'),
canViewPosts: api.hasPrivilege("posts:view"),
canBulkEditTags: api.hasPrivilege("posts:bulk-edit:tags"),
canBulkEditSafety: api.hasPrivilege(
"posts:bulk-edit:safety"
),
bulkEdit: {
tags: this._bulkEditTags,
},
});
const view = new PostsPageView(pageCtx);
view.addEventListener('tag', e => this._evtTag(e));
view.addEventListener('untag', e => this._evtUntag(e));
view.addEventListener(
'changeSafety', e => this._evtChangeSafety(e));
view.addEventListener("tag", (e) => this._evtTag(e));
view.addEventListener("untag", (e) => this._evtUntag(e));
view.addEventListener("changeSafety", (e) =>
this._evtChangeSafety(e)
);
return view;
},
});
}
}
module.exports = router => {
router.enter(
['posts'],
(ctx, next) => {
module.exports = (router) => {
router.enter(["posts"], (ctx, next) => {
ctx.controller = new PostListController(ctx);
});
};

View file

@ -1,16 +1,16 @@
'use strict';
"use strict";
const router = require('../router.js');
const api = require('../api.js');
const uri = require('../util/uri.js');
const misc = require('../util/misc.js');
const settings = require('../models/settings.js');
const Comment = require('../models/comment.js');
const Post = require('../models/post.js');
const PostList = require('../models/post_list.js');
const PostMainView = require('../views/post_main_view.js');
const BasePostController = require('./base_post_controller.js');
const EmptyView = require('../views/empty_view.js');
const router = require("../router.js");
const api = require("../api.js");
const uri = require("../util/uri.js");
const misc = require("../util/misc.js");
const settings = require("../models/settings.js");
const Comment = require("../models/comment.js");
const Post = require("../models/post.js");
const PostList = require("../models/post_list.js");
const PostMainView = require("../views/post_main_view.js");
const BasePostController = require("./base_post_controller.js");
const EmptyView = require("../views/empty_view.js");
class PostMainController extends BasePostController {
constructor(ctx, editMode) {
@ -21,17 +21,23 @@ class PostMainController extends BasePostController {
Post.get(ctx.parameters.id),
PostList.getAround(
ctx.parameters.id,
parameters ? parameters.query : null),
]).then(responses => {
parameters ? parameters.query : null
),
]).then(
(responses) => {
const [post, aroundResponse] = responses;
// remove junk from query, but save it into history so that it can
// be still accessed after history navigation / page refresh
if (parameters.query) {
ctx.state.parameters = parameters;
const url = editMode ?
uri.formatClientLink('post', ctx.parameters.id, 'edit') :
uri.formatClientLink('post', ctx.parameters.id);
const url = editMode
? uri.formatClientLink(
"post",
ctx.parameters.id,
"edit"
)
: uri.formatClientLink("post", ctx.parameters.id);
router.replace(url, ctx.state, false);
}
@ -39,56 +45,83 @@ class PostMainController extends BasePostController {
this._view = new PostMainView({
post: post,
editMode: editMode,
prevPostId: aroundResponse.prev ? aroundResponse.prev.id : null,
nextPostId: aroundResponse.next ? aroundResponse.next.id : null,
canEditPosts: api.hasPrivilege('posts:edit'),
canDeletePosts: api.hasPrivilege('posts:delete'),
canFeaturePosts: api.hasPrivilege('posts:feature'),
canListComments: api.hasPrivilege('comments:list'),
canCreateComments: api.hasPrivilege('comments:create'),
prevPostId: aroundResponse.prev
? aroundResponse.prev.id
: null,
nextPostId: aroundResponse.next
? aroundResponse.next.id
: null,
canEditPosts: api.hasPrivilege("posts:edit"),
canDeletePosts: api.hasPrivilege("posts:delete"),
canFeaturePosts: api.hasPrivilege("posts:feature"),
canListComments: api.hasPrivilege("comments:list"),
canCreateComments: api.hasPrivilege("comments:create"),
parameters: parameters,
});
if (this._view.sidebarControl) {
this._view.sidebarControl.addEventListener(
'favorite', e => this._evtFavoritePost(e));
"favorite",
(e) => this._evtFavoritePost(e)
);
this._view.sidebarControl.addEventListener(
'unfavorite', e => this._evtUnfavoritePost(e));
"unfavorite",
(e) => this._evtUnfavoritePost(e)
);
this._view.sidebarControl.addEventListener("score", (e) =>
this._evtScorePost(e)
);
this._view.sidebarControl.addEventListener(
'score', e => this._evtScorePost(e));
"fitModeChange",
(e) => this._evtFitModeChange(e)
);
this._view.sidebarControl.addEventListener("change", (e) =>
this._evtPostChange(e)
);
this._view.sidebarControl.addEventListener("submit", (e) =>
this._evtUpdatePost(e)
);
this._view.sidebarControl.addEventListener(
'fitModeChange', e => this._evtFitModeChange(e));
this._view.sidebarControl.addEventListener(
'change', e => this._evtPostChange(e));
this._view.sidebarControl.addEventListener(
'submit', e => this._evtUpdatePost(e));
this._view.sidebarControl.addEventListener(
'feature', e => this._evtFeaturePost(e));
this._view.sidebarControl.addEventListener(
'delete', e => this._evtDeletePost(e));
this._view.sidebarControl.addEventListener(
'merge', e => this._evtMergePost(e));
"feature",
(e) => this._evtFeaturePost(e)
);
this._view.sidebarControl.addEventListener("delete", (e) =>
this._evtDeletePost(e)
);
this._view.sidebarControl.addEventListener("merge", (e) =>
this._evtMergePost(e)
);
}
if (this._view.commentControl) {
this._view.commentControl.addEventListener(
'change', e => this._evtCommentChange(e));
this._view.commentControl.addEventListener(
'submit', e => this._evtCreateComment(e));
this._view.commentControl.addEventListener("change", (e) =>
this._evtCommentChange(e)
);
this._view.commentControl.addEventListener("submit", (e) =>
this._evtCreateComment(e)
);
}
if (this._view.commentListControl) {
this._view.commentListControl.addEventListener(
'submit', e => this._evtUpdateComment(e));
"submit",
(e) => this._evtUpdateComment(e)
);
this._view.commentListControl.addEventListener(
'score', e => this._evtScoreComment(e));
"score",
(e) => this._evtScoreComment(e)
);
this._view.commentListControl.addEventListener(
'delete', e => this._evtDeleteComment(e));
"delete",
(e) => this._evtDeleteComment(e)
);
}
}, error => {
},
(error) => {
this._view = new EmptyView();
this._view.showError(error.message);
});
}
);
}
_evtFitModeChange(e) {
@ -100,32 +133,36 @@ class PostMainController extends BasePostController {
_evtFeaturePost(e) {
this._view.sidebarControl.disableForm();
this._view.sidebarControl.clearMessages();
e.detail.post.feature()
.then(() => {
this._view.sidebarControl.showSuccess('Post featured.');
e.detail.post.feature().then(
() => {
this._view.sidebarControl.showSuccess("Post featured.");
this._view.sidebarControl.enableForm();
}, error => {
},
(error) => {
this._view.sidebarControl.showError(error.message);
this._view.sidebarControl.enableForm();
});
}
);
}
_evtMergePost(e) {
router.show(uri.formatClientLink('post', e.detail.post.id, 'merge'));
router.show(uri.formatClientLink("post", e.detail.post.id, "merge"));
}
_evtDeletePost(e) {
this._view.sidebarControl.disableForm();
this._view.sidebarControl.clearMessages();
e.detail.post.delete()
.then(() => {
e.detail.post.delete().then(
() => {
misc.disableExitConfirmation();
const ctx = router.show(uri.formatClientLink('posts'));
ctx.controller.showSuccess('Post deleted.');
}, error => {
const ctx = router.show(uri.formatClientLink("posts"));
ctx.controller.showSuccess("Post deleted.");
},
(error) => {
this._view.sidebarControl.showError(error.message);
this._view.sidebarControl.enableForm();
});
}
);
}
_evtUpdatePost(e) {
@ -150,15 +187,17 @@ class PostMainController extends BasePostController {
if (e.detail.source !== undefined) {
post.source = e.detail.source;
}
post.save()
.then(() => {
this._view.sidebarControl.showSuccess('Post saved.');
post.save().then(
() => {
this._view.sidebarControl.showSuccess("Post saved.");
this._view.sidebarControl.enableForm();
misc.disableExitConfirmation();
}, error => {
},
(error) => {
this._view.sidebarControl.showError(error.message);
this._view.sidebarControl.enableForm();
});
}
);
}
_evtPostChange(e) {
@ -173,75 +212,78 @@ class PostMainController extends BasePostController {
this._view.commentControl.disableForm();
const comment = Comment.create(this._post.id);
comment.text = e.detail.text;
comment.save()
.then(() => {
comment.save().then(
() => {
this._post.comments.add(comment);
this._view.commentControl.exitEditMode();
this._view.commentControl.enableForm();
misc.disableExitConfirmation();
}, error => {
},
(error) => {
this._view.commentControl.showError(error.message);
this._view.commentControl.enableForm();
});
}
);
}
_evtUpdateComment(e) {
// TODO: disable form
e.detail.comment.text = e.detail.text;
e.detail.comment.save()
.catch(error => {
e.detail.comment.save().catch((error) => {
e.detail.target.showError(error.message);
// TODO: enable form
});
}
_evtScoreComment(e) {
e.detail.comment.setScore(e.detail.score)
.catch(error => window.alert(error.message));
e.detail.comment
.setScore(e.detail.score)
.catch((error) => window.alert(error.message));
}
_evtDeleteComment(e) {
e.detail.comment.delete()
.catch(error => window.alert(error.message));
e.detail.comment
.delete()
.catch((error) => window.alert(error.message));
}
_evtScorePost(e) {
if (!api.hasPrivilege('posts:score')) {
if (!api.hasPrivilege("posts:score")) {
return;
}
e.detail.post.setScore(e.detail.score)
.catch(error => window.alert(error.message));
e.detail.post
.setScore(e.detail.score)
.catch((error) => window.alert(error.message));
}
_evtFavoritePost(e) {
if (!api.hasPrivilege('posts:favorite')) {
if (!api.hasPrivilege("posts:favorite")) {
return;
}
e.detail.post.addToFavorites()
.catch(error => window.alert(error.message));
e.detail.post
.addToFavorites()
.catch((error) => window.alert(error.message));
}
_evtUnfavoritePost(e) {
if (!api.hasPrivilege('posts:favorite')) {
if (!api.hasPrivilege("posts:favorite")) {
return;
}
e.detail.post.removeFromFavorites()
.catch(error => window.alert(error.message));
e.detail.post
.removeFromFavorites()
.catch((error) => window.alert(error.message));
}
}
module.exports = router => {
router.enter(['post', ':id', 'edit'],
(ctx, next) => {
module.exports = (router) => {
router.enter(["post", ":id", "edit"], (ctx, next) => {
// restore parameters from history state
if (ctx.state.parameters) {
Object.assign(ctx.parameters, ctx.state.parameters);
}
ctx.controller = new PostMainController(ctx, true);
});
router.enter(
['post', ':id'],
(ctx, next) => {
router.enter(["post", ":id"], (ctx, next) => {
// restore parameters from history state
if (ctx.state.parameters) {
Object.assign(ctx.parameters, ctx.state.parameters);

View file

@ -1,40 +1,40 @@
'use strict';
"use strict";
const api = require('../api.js');
const router = require('../router.js');
const uri = require('../util/uri.js');
const misc = require('../util/misc.js');
const progress = require('../util/progress.js');
const topNavigation = require('../models/top_navigation.js');
const Post = require('../models/post.js');
const Tag = require('../models/tag.js');
const PostUploadView = require('../views/post_upload_view.js');
const EmptyView = require('../views/empty_view.js');
const api = require("../api.js");
const router = require("../router.js");
const uri = require("../util/uri.js");
const misc = require("../util/misc.js");
const progress = require("../util/progress.js");
const topNavigation = require("../models/top_navigation.js");
const Post = require("../models/post.js");
const Tag = require("../models/tag.js");
const PostUploadView = require("../views/post_upload_view.js");
const EmptyView = require("../views/empty_view.js");
const genericErrorMessage =
'One of the posts needs your attention; ' +
"One of the posts needs your attention; " +
'click "resume upload" when you\'re ready.';
class PostUploadController {
constructor() {
this._lastCancellablePromise = null;
if (!api.hasPrivilege('posts:create')) {
if (!api.hasPrivilege("posts:create")) {
this._view = new EmptyView();
this._view.showError('You don\'t have privileges to upload posts.');
this._view.showError("You don't have privileges to upload posts.");
return;
}
topNavigation.activate('upload');
topNavigation.setTitle('Upload');
topNavigation.activate("upload");
topNavigation.setTitle("Upload");
this._view = new PostUploadView({
canUploadAnonymously: api.hasPrivilege('posts:create:anonymous'),
canViewPosts: api.hasPrivilege('posts:view'),
canUploadAnonymously: api.hasPrivilege("posts:create:anonymous"),
canViewPosts: api.hasPrivilege("posts:view"),
enableSafety: api.safetyEnabled(),
});
this._view.addEventListener('change', e => this._evtChange(e));
this._view.addEventListener('submit', e => this._evtSubmit(e));
this._view.addEventListener('cancel', e => this._evtCancel(e));
this._view.addEventListener("change", (e) => this._evtChange(e));
this._view.addEventListener("submit", (e) => this._evtSubmit(e));
this._view.addEventListener("cancel", (e) => this._evtCancel(e));
}
_evtChange(e) {
@ -56,45 +56,61 @@ class PostUploadController {
this._view.disableForm();
this._view.clearMessages();
e.detail.uploadables.reduce(
(promise, uploadable) => promise.then(() => this._uploadSinglePost(
uploadable, e.detail.skipDuplicates)),
Promise.resolve())
.then(() => {
e.detail.uploadables
.reduce(
(promise, uploadable) =>
promise.then(() =>
this._uploadSinglePost(
uploadable,
e.detail.skipDuplicates
)
),
Promise.resolve()
)
.then(
() => {
this._view.clearMessages();
misc.disableExitConfirmation();
const ctx = router.show(uri.formatClientLink('posts'));
ctx.controller.showSuccess('Posts uploaded.');
}, error => {
const ctx = router.show(uri.formatClientLink("posts"));
ctx.controller.showSuccess("Posts uploaded.");
},
(error) => {
if (error.uploadable) {
if (error.similarPosts) {
error.uploadable.lookalikes = error.similarPosts;
this._view.updateUploadable(error.uploadable);
this._view.showInfo(genericErrorMessage);
this._view.showInfo(
error.message, error.uploadable);
error.message,
error.uploadable
);
} else {
this._view.showError(genericErrorMessage);
this._view.showError(
error.message, error.uploadable);
error.message,
error.uploadable
);
}
} else {
this._view.showError(error.message);
}
this._view.enableForm();
});
}
);
}
_uploadSinglePost(uploadable, skipDuplicates) {
progress.start();
let reverseSearchPromise = Promise.resolve();
if (!uploadable.lookalikesConfirmed) {
reverseSearchPromise =
Post.reverseSearch(uploadable.url || uploadable.file);
reverseSearchPromise = Post.reverseSearch(
uploadable.url || uploadable.file
);
}
this._lastCancellablePromise = reverseSearchPromise;
return reverseSearchPromise.then(searchResult => {
return reverseSearchPromise
.then((searchResult) => {
if (searchResult) {
// notify about exact duplicate
if (searchResult.exactPost) {
@ -102,8 +118,10 @@ class PostUploadController {
this._view.removeUploadable(uploadable);
return Promise.resolve();
} else {
let error = new Error('Post already uploaded ' +
`(@${searchResult.exactPost.id})`);
let error = new Error(
"Post already uploaded " +
`(@${searchResult.exactPost.id})`
);
error.uploadable = uploadable;
return Promise.reject(error);
}
@ -113,7 +131,8 @@ class PostUploadController {
if (searchResult.similarPosts.length) {
let error = new Error(
`Found ${searchResult.similarPosts.length} similar ` +
'posts.\nYou can resume or discard this upload.');
"posts.\nYou can resume or discard this upload."
);
error.uploadable = uploadable;
error.similarPosts = searchResult.similarPosts;
return Promise.reject(error);
@ -122,21 +141,24 @@ class PostUploadController {
// no duplicates, proceed with saving
let post = this._uploadableToPost(uploadable);
let savePromise = post.save(uploadable.anonymous)
.then(() => {
let savePromise = post.save(uploadable.anonymous).then(() => {
this._view.removeUploadable(uploadable);
return Promise.resolve();
});
this._lastCancellablePromise = savePromise;
return savePromise;
}).then(result => {
})
.then(
(result) => {
progress.done();
return Promise.resolve(result);
}, error => {
},
(error) => {
error.uploadable = uploadable;
progress.done();
return Promise.reject(error);
});
}
);
}
_uploadableToPost(uploadable) {
@ -159,8 +181,8 @@ class PostUploadController {
}
}
module.exports = router => {
router.enter(['upload'], (ctx, next) => {
module.exports = (router) => {
router.enter(["upload"], (ctx, next) => {
ctx.controller = new PostUploadController();
});
};

View file

@ -1,28 +1,28 @@
'use strict';
"use strict";
const settings = require('../models/settings.js');
const topNavigation = require('../models/top_navigation.js');
const SettingsView = require('../views/settings_view.js');
const settings = require("../models/settings.js");
const topNavigation = require("../models/top_navigation.js");
const SettingsView = require("../views/settings_view.js");
class SettingsController {
constructor() {
topNavigation.activate('settings');
topNavigation.setTitle('Browsing settings');
topNavigation.activate("settings");
topNavigation.setTitle("Browsing settings");
this._view = new SettingsView({
settings: settings.get(),
});
this._view.addEventListener('submit', e => this._evtSubmit(e));
this._view.addEventListener("submit", (e) => this._evtSubmit(e));
}
_evtSubmit(e) {
this._view.clearMessages();
settings.save(e.detail);
this._view.showSuccess('Settings saved.');
this._view.showSuccess("Settings saved.");
}
}
module.exports = router => {
router.enter(['settings'], (ctx, next) => {
module.exports = (router) => {
router.enter(["settings"], (ctx, next) => {
ctx.controller = new SettingsController();
});
};

View file

@ -1,41 +1,43 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const SnapshotList = require('../models/snapshot_list.js');
const PageController = require('../controllers/page_controller.js');
const topNavigation = require('../models/top_navigation.js');
const SnapshotsPageView = require('../views/snapshots_page_view.js');
const EmptyView = require('../views/empty_view.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const SnapshotList = require("../models/snapshot_list.js");
const PageController = require("../controllers/page_controller.js");
const topNavigation = require("../models/top_navigation.js");
const SnapshotsPageView = require("../views/snapshots_page_view.js");
const EmptyView = require("../views/empty_view.js");
class SnapshotsController {
constructor(ctx) {
if (!api.hasPrivilege('snapshots:list')) {
if (!api.hasPrivilege("snapshots:list")) {
this._view = new EmptyView();
this._view.showError('You don\'t have privileges to view history.');
this._view.showError("You don't have privileges to view history.");
return;
}
topNavigation.activate('');
topNavigation.setTitle('History');
topNavigation.activate("");
topNavigation.setTitle("History");
this._pageController = new PageController();
this._pageController.run({
parameters: ctx.parameters,
defaultLimit: 25,
getClientUrlForPage: (offset, limit) => {
const parameters = Object.assign(
{}, ctx.parameters, {offset: offset, limit: limit});
return uri.formatClientLink('history', parameters);
const parameters = Object.assign({}, ctx.parameters, {
offset: offset,
limit: limit,
});
return uri.formatClientLink("history", parameters);
},
requestPage: (offset, limit) => {
return SnapshotList.search('', offset, limit);
return SnapshotList.search("", offset, limit);
},
pageRenderer: pageCtx => {
pageRenderer: (pageCtx) => {
Object.assign(pageCtx, {
canViewPosts: api.hasPrivilege('posts:view'),
canViewUsers: api.hasPrivilege('users:view'),
canViewTags: api.hasPrivilege('tags:view'),
canViewPosts: api.hasPrivilege("posts:view"),
canViewUsers: api.hasPrivilege("users:view"),
canViewTags: api.hasPrivilege("tags:view"),
});
return new SnapshotsPageView(pageCtx);
},
@ -43,9 +45,8 @@ class SnapshotsController {
}
}
module.exports = router => {
router.enter(['history'],
(ctx, next) => {
module.exports = (router) => {
router.enter(["history"], (ctx, next) => {
ctx.controller = new SnapshotsController(ctx);
});
};

View file

@ -1,57 +1,67 @@
'use strict';
"use strict";
const api = require('../api.js');
const tags = require('../tags.js');
const TagCategoryList = require('../models/tag_category_list.js');
const topNavigation = require('../models/top_navigation.js');
const TagCategoriesView = require('../views/tag_categories_view.js');
const EmptyView = require('../views/empty_view.js');
const api = require("../api.js");
const tags = require("../tags.js");
const TagCategoryList = require("../models/tag_category_list.js");
const topNavigation = require("../models/top_navigation.js");
const TagCategoriesView = require("../views/tag_categories_view.js");
const EmptyView = require("../views/empty_view.js");
class TagCategoriesController {
constructor() {
if (!api.hasPrivilege('tagCategories:list')) {
if (!api.hasPrivilege("tagCategories:list")) {
this._view = new EmptyView();
this._view.showError(
'You don\'t have privileges to view tag categories.');
"You don't have privileges to view tag categories."
);
return;
}
topNavigation.activate('tags');
topNavigation.setTitle('Listing tags');
TagCategoryList.get().then(response => {
topNavigation.activate("tags");
topNavigation.setTitle("Listing tags");
TagCategoryList.get().then(
(response) => {
this._tagCategories = response.results;
this._view = new TagCategoriesView({
tagCategories: this._tagCategories,
canEditName: api.hasPrivilege('tagCategories:edit:name'),
canEditColor: api.hasPrivilege('tagCategories:edit:color'),
canDelete: api.hasPrivilege('tagCategories:delete'),
canCreate: api.hasPrivilege('tagCategories:create'),
canSetDefault: api.hasPrivilege('tagCategories:setDefault'),
canEditName: api.hasPrivilege("tagCategories:edit:name"),
canEditColor: api.hasPrivilege("tagCategories:edit:color"),
canDelete: api.hasPrivilege("tagCategories:delete"),
canCreate: api.hasPrivilege("tagCategories:create"),
canSetDefault: api.hasPrivilege(
"tagCategories:setDefault"
),
});
this._view.addEventListener('submit', e => this._evtSubmit(e));
}, error => {
this._view.addEventListener("submit", (e) =>
this._evtSubmit(e)
);
},
(error) => {
this._view = new EmptyView();
this._view.showError(error.message);
});
}
);
}
_evtSubmit(e) {
this._view.clearMessages();
this._view.disableForm();
this._tagCategories.save()
.then(() => {
this._tagCategories.save().then(
() => {
tags.refreshCategoryColorMap();
this._view.enableForm();
this._view.showSuccess('Changes saved.');
}, error => {
this._view.showSuccess("Changes saved.");
},
(error) => {
this._view.enableForm();
this._view.showError(error.message);
});
}
);
}
}
module.exports = router => {
router.enter(['tag-categories'], (ctx, next) => {
module.exports = (router) => {
router.enter(["tag-categories"], (ctx, next) => {
ctx.controller = new TagCategoriesController(ctx, next);
});
};

View file

@ -1,34 +1,37 @@
'use strict';
"use strict";
const router = require('../router.js');
const api = require('../api.js');
const misc = require('../util/misc.js');
const uri = require('../util/uri.js');
const Tag = require('../models/tag.js');
const TagCategoryList = require('../models/tag_category_list.js');
const topNavigation = require('../models/top_navigation.js');
const TagView = require('../views/tag_view.js');
const EmptyView = require('../views/empty_view.js');
const router = require("../router.js");
const api = require("../api.js");
const misc = require("../util/misc.js");
const uri = require("../util/uri.js");
const Tag = require("../models/tag.js");
const TagCategoryList = require("../models/tag_category_list.js");
const topNavigation = require("../models/top_navigation.js");
const TagView = require("../views/tag_view.js");
const EmptyView = require("../views/empty_view.js");
class TagController {
constructor(ctx, section) {
if (!api.hasPrivilege('tags:view')) {
if (!api.hasPrivilege("tags:view")) {
this._view = new EmptyView();
this._view.showError('You don\'t have privileges to view tags.');
this._view.showError("You don't have privileges to view tags.");
return;
}
Promise.all([
TagCategoryList.get(),
Tag.get(ctx.parameters.name),
]).then(responses => {
]).then(
(responses) => {
const [tagCategoriesResponse, tag] = responses;
topNavigation.activate('tags');
topNavigation.setTitle('Tag #' + tag.names[0]);
topNavigation.activate("tags");
topNavigation.setTitle("Tag #" + tag.names[0]);
this._name = ctx.parameters.name;
tag.addEventListener('change', e => this._evtSaved(e, section));
tag.addEventListener("change", (e) =>
this._evtSaved(e, section)
);
const categories = {};
for (let category of tagCategoriesResponse.results) {
@ -38,26 +41,40 @@ class TagController {
this._view = new TagView({
tag: tag,
section: section,
canEditAnything: api.hasPrivilege('tags:edit'),
canEditNames: api.hasPrivilege('tags:edit:names'),
canEditCategory: api.hasPrivilege('tags:edit:category'),
canEditImplications: api.hasPrivilege('tags:edit:implications'),
canEditSuggestions: api.hasPrivilege('tags:edit:suggestions'),
canEditDescription: api.hasPrivilege('tags:edit:description'),
canMerge: api.hasPrivilege('tags:merge'),
canDelete: api.hasPrivilege('tags:delete'),
canEditAnything: api.hasPrivilege("tags:edit"),
canEditNames: api.hasPrivilege("tags:edit:names"),
canEditCategory: api.hasPrivilege("tags:edit:category"),
canEditImplications: api.hasPrivilege(
"tags:edit:implications"
),
canEditSuggestions: api.hasPrivilege(
"tags:edit:suggestions"
),
canEditDescription: api.hasPrivilege(
"tags:edit:description"
),
canMerge: api.hasPrivilege("tags:merge"),
canDelete: api.hasPrivilege("tags:delete"),
categories: categories,
escapeColons: uri.escapeColons,
});
this._view.addEventListener('change', e => this._evtChange(e));
this._view.addEventListener('submit', e => this._evtUpdate(e));
this._view.addEventListener('merge', e => this._evtMerge(e));
this._view.addEventListener('delete', e => this._evtDelete(e));
}, error => {
this._view.addEventListener("change", (e) =>
this._evtChange(e)
);
this._view.addEventListener("submit", (e) =>
this._evtUpdate(e)
);
this._view.addEventListener("merge", (e) => this._evtMerge(e));
this._view.addEventListener("delete", (e) =>
this._evtDelete(e)
);
},
(error) => {
this._view = new EmptyView();
this._view.showError(error.message);
});
}
);
}
_evtChange(e) {
@ -68,9 +85,10 @@ class TagController {
misc.disableExitConfirmation();
if (this._name !== e.detail.tag.names[0]) {
router.replace(
uri.formatClientLink('tag', e.detail.tag.names[0], section),
uri.formatClientLink("tag", e.detail.tag.names[0], section),
null,
false);
false
);
}
}
@ -86,59 +104,69 @@ class TagController {
if (e.detail.description !== undefined) {
e.detail.tag.description = e.detail.description;
}
e.detail.tag.save().then(() => {
this._view.showSuccess('Tag saved.');
e.detail.tag.save().then(
() => {
this._view.showSuccess("Tag saved.");
this._view.enableForm();
}, error => {
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
_evtMerge(e) {
this._view.clearMessages();
this._view.disableForm();
e.detail.tag
.merge(e.detail.targetTagName, e.detail.addAlias)
.then(() => {
this._view.showSuccess('Tag merged.');
e.detail.tag.merge(e.detail.targetTagName, e.detail.addAlias).then(
() => {
this._view.showSuccess("Tag merged.");
this._view.enableForm();
router.replace(
uri.formatClientLink(
'tag', e.detail.targetTagName, 'merge'),
"tag",
e.detail.targetTagName,
"merge"
),
null,
false);
}, error => {
false
);
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
_evtDelete(e) {
this._view.clearMessages();
this._view.disableForm();
e.detail.tag.delete()
.then(() => {
const ctx = router.show(uri.formatClientLink('tags'));
ctx.controller.showSuccess('Tag deleted.');
}, error => {
e.detail.tag.delete().then(
() => {
const ctx = router.show(uri.formatClientLink("tags"));
ctx.controller.showSuccess("Tag deleted.");
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
}
module.exports = router => {
router.enter(['tag', ':name', 'edit'], (ctx, next) => {
ctx.controller = new TagController(ctx, 'edit');
module.exports = (router) => {
router.enter(["tag", ":name", "edit"], (ctx, next) => {
ctx.controller = new TagController(ctx, "edit");
});
router.enter(['tag', ':name', 'merge'], (ctx, next) => {
ctx.controller = new TagController(ctx, 'merge');
router.enter(["tag", ":name", "merge"], (ctx, next) => {
ctx.controller = new TagController(ctx, "merge");
});
router.enter(['tag', ':name', 'delete'], (ctx, next) => {
ctx.controller = new TagController(ctx, 'delete');
router.enter(["tag", ":name", "delete"], (ctx, next) => {
ctx.controller = new TagController(ctx, "delete");
});
router.enter(['tag', ':name'], (ctx, next) => {
ctx.controller = new TagController(ctx, 'summary');
router.enter(["tag", ":name"], (ctx, next) => {
ctx.controller = new TagController(ctx, "summary");
});
};

View file

@ -1,46 +1,47 @@
'use strict';
"use strict";
const router = require('../router.js');
const api = require('../api.js');
const uri = require('../util/uri.js');
const TagList = require('../models/tag_list.js');
const topNavigation = require('../models/top_navigation.js');
const PageController = require('../controllers/page_controller.js');
const TagsHeaderView = require('../views/tags_header_view.js');
const TagsPageView = require('../views/tags_page_view.js');
const EmptyView = require('../views/empty_view.js');
const router = require("../router.js");
const api = require("../api.js");
const uri = require("../util/uri.js");
const TagList = require("../models/tag_list.js");
const topNavigation = require("../models/top_navigation.js");
const PageController = require("../controllers/page_controller.js");
const TagsHeaderView = require("../views/tags_header_view.js");
const TagsPageView = require("../views/tags_page_view.js");
const EmptyView = require("../views/empty_view.js");
const fields = [
'names',
'suggestions',
'implications',
'creationTime',
'usages',
'category'
"names",
"suggestions",
"implications",
"creationTime",
"usages",
"category",
];
class TagListController {
constructor(ctx) {
this._pageController = new PageController();
if (!api.hasPrivilege('tags:list')) {
if (!api.hasPrivilege("tags:list")) {
this._view = new EmptyView();
this._view.showError('You don\'t have privileges to view tags.');
this._view.showError("You don't have privileges to view tags.");
return;
}
this._ctx = ctx;
topNavigation.activate('tags');
topNavigation.setTitle('Listing tags');
topNavigation.activate("tags");
topNavigation.setTitle("Listing tags");
this._headerView = new TagsHeaderView({
hostNode: this._pageController.view.pageHeaderHolderNode,
parameters: ctx.parameters,
canEditTagCategories: api.hasPrivilege('tagCategories:edit'),
canEditTagCategories: api.hasPrivilege("tagCategories:edit"),
});
this._headerView.addEventListener(
'navigate', e => this._evtNavigate(e));
this._headerView.addEventListener("navigate", (e) =>
this._evtNavigate(e)
);
this._syncPageController();
}
@ -55,7 +56,8 @@ class TagListController {
_evtNavigate(e) {
router.showNoDispatch(
uri.formatClientLink('tags', e.detail.parameters));
uri.formatClientLink("tags", e.detail.parameters)
);
Object.assign(this._ctx.parameters, e.detail.parameters);
this._syncPageController();
}
@ -65,25 +67,29 @@ class TagListController {
parameters: this._ctx.parameters,
defaultLimit: 50,
getClientUrlForPage: (offset, limit) => {
const parameters = Object.assign(
{}, this._ctx.parameters, {offset: offset, limit: limit});
return uri.formatClientLink('tags', parameters);
const parameters = Object.assign({}, this._ctx.parameters, {
offset: offset,
limit: limit,
});
return uri.formatClientLink("tags", parameters);
},
requestPage: (offset, limit) => {
return TagList.search(
this._ctx.parameters.query, offset, limit, fields);
this._ctx.parameters.query,
offset,
limit,
fields
);
},
pageRenderer: pageCtx => {
pageRenderer: (pageCtx) => {
return new TagsPageView(pageCtx);
},
});
}
}
module.exports = router => {
router.enter(
['tags'],
(ctx, next) => {
module.exports = (router) => {
router.enter(["tags"], (ctx, next) => {
ctx.controller = new TagListController(ctx);
});
};

View file

@ -1,19 +1,20 @@
'use strict';
"use strict";
const api = require('../api.js');
const topNavigation = require('../models/top_navigation.js');
const TopNavigationView = require('../views/top_navigation_view.js');
const api = require("../api.js");
const topNavigation = require("../models/top_navigation.js");
const TopNavigationView = require("../views/top_navigation_view.js");
class TopNavigationController {
constructor() {
api.fetchConfig().then(() => {
this._topNavigationView = new TopNavigationView();
topNavigation.addEventListener(
'activate', e => this._evtActivate(e));
topNavigation.addEventListener("activate", (e) =>
this._evtActivate(e)
);
api.addEventListener('login', e => this._evtAuthChange(e));
api.addEventListener('logout', e => this._evtAuthChange(e));
api.addEventListener("login", (e) => this._evtAuthChange(e));
api.addEventListener("logout", (e) => this._evtAuthChange(e));
this._render();
});
@ -28,37 +29,38 @@ class TopNavigationController {
}
_updateNavigationFromPrivileges() {
topNavigation.get('account').url = 'user/' + api.userName;
topNavigation.get('account').imageUrl =
api.user ? api.user.avatarUrl : null;
topNavigation.get("account").url = "user/" + api.userName;
topNavigation.get("account").imageUrl = api.user
? api.user.avatarUrl
: null;
topNavigation.showAll();
if (!api.hasPrivilege('posts:list')) {
topNavigation.hide('posts');
if (!api.hasPrivilege("posts:list")) {
topNavigation.hide("posts");
}
if (!api.hasPrivilege('posts:create')) {
topNavigation.hide('upload');
if (!api.hasPrivilege("posts:create")) {
topNavigation.hide("upload");
}
if (!api.hasPrivilege('comments:list')) {
topNavigation.hide('comments');
if (!api.hasPrivilege("comments:list")) {
topNavigation.hide("comments");
}
if (!api.hasPrivilege('tags:list')) {
topNavigation.hide('tags');
if (!api.hasPrivilege("tags:list")) {
topNavigation.hide("tags");
}
if (!api.hasPrivilege('users:list')) {
topNavigation.hide('users');
if (!api.hasPrivilege("users:list")) {
topNavigation.hide("users");
}
if (api.isLoggedIn()) {
if (!api.hasPrivilege('users:create:any')) {
topNavigation.hide('register');
if (!api.hasPrivilege("users:create:any")) {
topNavigation.hide("register");
}
topNavigation.hide('login');
topNavigation.hide("login");
} else {
if (!api.hasPrivilege('users:create:self')) {
topNavigation.hide('register');
if (!api.hasPrivilege("users:create:self")) {
topNavigation.hide("register");
}
topNavigation.hide('account');
topNavigation.hide('logout');
topNavigation.hide("account");
topNavigation.hide("logout");
}
}
@ -66,10 +68,11 @@ class TopNavigationController {
this._updateNavigationFromPrivileges();
this._topNavigationView.render({
items: topNavigation.getAll(),
name: api.getName()
name: api.getName(),
});
this._topNavigationView.activate(
topNavigation.activeItem ? topNavigation.activeItem.key : '');
topNavigation.activeItem ? topNavigation.activeItem.key : ""
);
}
}

View file

@ -1,23 +1,25 @@
'use strict';
"use strict";
const router = require('../router.js');
const api = require('../api.js');
const uri = require('../util/uri.js');
const misc = require('../util/misc.js');
const views = require('../util/views.js');
const User = require('../models/user.js');
const UserToken = require('../models/user_token.js');
const topNavigation = require('../models/top_navigation.js');
const UserView = require('../views/user_view.js');
const EmptyView = require('../views/empty_view.js');
const router = require("../router.js");
const api = require("../api.js");
const uri = require("../util/uri.js");
const misc = require("../util/misc.js");
const views = require("../util/views.js");
const User = require("../models/user.js");
const UserToken = require("../models/user_token.js");
const topNavigation = require("../models/top_navigation.js");
const UserView = require("../views/user_view.js");
const EmptyView = require("../views/empty_view.js");
class UserController {
constructor(ctx, section) {
const userName = ctx.parameters.name;
if (!api.hasPrivilege('users:view') &&
!api.isLoggedIn({name: userName})) {
if (
!api.hasPrivilege("users:view") &&
!api.isLoggedIn({ name: userName })
) {
this._view = new EmptyView();
this._view.showError('You don\'t have privileges to view users.');
this._view.showError("You don't have privileges to view users.");
return;
}
@ -25,36 +27,40 @@ class UserController {
this._errorMessages = [];
let userTokenPromise = Promise.resolve([]);
if (section === 'list-tokens') {
userTokenPromise = UserToken.get(userName)
.then(userTokens => {
return userTokens.map(token => {
token.isCurrentAuthToken = api.isCurrentAuthToken(token);
if (section === "list-tokens") {
userTokenPromise = UserToken.get(userName).then(
(userTokens) => {
return userTokens.map((token) => {
token.isCurrentAuthToken = api.isCurrentAuthToken(
token
);
return token;
});
}, error => {
},
(error) => {
return [];
});
}
);
}
topNavigation.setTitle('User ' + userName);
Promise.all([
userTokenPromise,
User.get(userName)
]).then(responses => {
topNavigation.setTitle("User " + userName);
Promise.all([userTokenPromise, User.get(userName)]).then(
(responses) => {
const [userTokens, user] = responses;
const isLoggedIn = api.isLoggedIn(user);
const infix = isLoggedIn ? 'self' : 'any';
const infix = isLoggedIn ? "self" : "any";
this._name = userName;
user.addEventListener('change', e => this._evtSaved(e, section));
user.addEventListener("change", (e) =>
this._evtSaved(e, section)
);
const myRankIndex = api.user ?
api.allRanks.indexOf(api.user.rank) :
0;
const myRankIndex = api.user
? api.allRanks.indexOf(api.user.rank)
: 0;
let ranks = {};
for (let [rankIdx, rankIdentifier] of api.allRanks.entries()) {
if (rankIdentifier === 'anonymous') {
if (rankIdentifier === "anonymous") {
continue;
}
if (rankIdx > myRankIndex) {
@ -64,9 +70,9 @@ class UserController {
}
if (isLoggedIn) {
topNavigation.activate('account');
topNavigation.activate("account");
} else {
topNavigation.activate('users');
topNavigation.activate("users");
}
this._view = new UserView({
@ -74,25 +80,49 @@ class UserController {
section: section,
isLoggedIn: isLoggedIn,
canEditName: api.hasPrivilege(`users:edit:${infix}:name`),
canEditPassword: api.hasPrivilege(`users:edit:${infix}:pass`),
canEditEmail: api.hasPrivilege(`users:edit:${infix}:email`),
canEditPassword: api.hasPrivilege(
`users:edit:${infix}:pass`
),
canEditEmail: api.hasPrivilege(
`users:edit:${infix}:email`
),
canEditRank: api.hasPrivilege(`users:edit:${infix}:rank`),
canEditAvatar: api.hasPrivilege(`users:edit:${infix}:avatar`),
canEditAvatar: api.hasPrivilege(
`users:edit:${infix}:avatar`
),
canEditAnything: api.hasPrivilege(`users:edit:${infix}`),
canListTokens: api.hasPrivilege(`userTokens:list:${infix}`),
canCreateToken: api.hasPrivilege(`userTokens:create:${infix}`),
canListTokens: api.hasPrivilege(
`userTokens:list:${infix}`
),
canCreateToken: api.hasPrivilege(
`userTokens:create:${infix}`
),
canEditToken: api.hasPrivilege(`userTokens:edit:${infix}`),
canDeleteToken: api.hasPrivilege(`userTokens:delete:${infix}`),
canDeleteToken: api.hasPrivilege(
`userTokens:delete:${infix}`
),
canDelete: api.hasPrivilege(`users:delete:${infix}`),
ranks: ranks,
tokens: userTokens,
});
this._view.addEventListener('change', e => this._evtChange(e));
this._view.addEventListener('submit', e => this._evtUpdate(e));
this._view.addEventListener('delete', e => this._evtDelete(e));
this._view.addEventListener('create-token', e => this._evtCreateToken(e));
this._view.addEventListener('delete-token', e => this._evtDeleteToken(e));
this._view.addEventListener('update-token', e => this._evtUpdateToken(e));
this._view.addEventListener("change", (e) =>
this._evtChange(e)
);
this._view.addEventListener("submit", (e) =>
this._evtUpdate(e)
);
this._view.addEventListener("delete", (e) =>
this._evtDelete(e)
);
this._view.addEventListener("create-token", (e) =>
this._evtCreateToken(e)
);
this._view.addEventListener("delete-token", (e) =>
this._evtDeleteToken(e)
);
this._view.addEventListener("update-token", (e) =>
this._evtUpdateToken(e)
);
for (let message of this._successMessages) {
this.showSuccess(message);
@ -101,24 +131,25 @@ class UserController {
for (let message of this._errorMessages) {
this.showError(message);
}
}, error => {
},
(error) => {
this._view = new EmptyView();
this._view.showError(error.message);
});
}
);
}
showSuccess(message) {
if (typeof this._view === 'undefined') {
this._successMessages.push(message)
if (typeof this._view === "undefined") {
this._successMessages.push(message);
} else {
this._view.showSuccess(message);
}
}
showError(message) {
if (typeof this._view === 'undefined') {
this._errorMessages.push(message)
if (typeof this._view === "undefined") {
this._errorMessages.push(message);
} else {
this._view.showError(message);
}
@ -132,9 +163,10 @@ class UserController {
misc.disableExitConfirmation();
if (this._name !== e.detail.user.name) {
router.replace(
uri.formatClientLink('user', e.detail.user.name, section),
uri.formatClientLink("user", e.detail.user.name, section),
null,
false);
false
);
}
}
@ -142,7 +174,7 @@ class UserController {
this._view.clearMessages();
this._view.disableForm();
const isLoggedIn = api.isLoggedIn(e.detail.user);
const infix = isLoggedIn ? 'self' : 'any';
const infix = isLoggedIn ? "self" : "any";
if (e.detail.name !== undefined) {
e.detail.user.name = e.detail.name;
@ -165,72 +197,105 @@ class UserController {
}
}
e.detail.user.save().then(() => {
return isLoggedIn ?
api.login(
e.detail.user
.save()
.then(() => {
return isLoggedIn
? api.login(
e.detail.name || api.userName,
e.detail.password || api.userPassword,
false) :
Promise.resolve();
}).then(() => {
this._view.showSuccess('Settings updated.');
false
)
: Promise.resolve();
})
.then(
() => {
this._view.showSuccess("Settings updated.");
this._view.enableForm();
}, error => {
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
_evtDelete(e) {
this._view.clearMessages();
this._view.disableForm();
const isLoggedIn = api.isLoggedIn(e.detail.user);
e.detail.user.delete()
.then(() => {
e.detail.user.delete().then(
() => {
if (isLoggedIn) {
api.forget();
api.logout();
}
if (api.hasPrivilege('users:list')) {
const ctx = router.show(uri.formatClientLink('users'));
ctx.controller.showSuccess('Account deleted.');
if (api.hasPrivilege("users:list")) {
const ctx = router.show(uri.formatClientLink("users"));
ctx.controller.showSuccess("Account deleted.");
} else {
const ctx = router.show(uri.formatClientLink());
ctx.controller.showSuccess('Account deleted.');
ctx.controller.showSuccess("Account deleted.");
}
}, error => {
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
_evtCreateToken(e) {
this._view.clearMessages();
this._view.disableForm();
UserToken.create(e.detail.user.name, e.detail.note, e.detail.expirationTime)
.then(response => {
const ctx = router.show(uri.formatClientLink('user', e.detail.user.name, 'list-tokens'));
ctx.controller.showSuccess('Token ' + response.token + ' created.');
}, error => {
UserToken.create(
e.detail.user.name,
e.detail.note,
e.detail.expirationTime
).then(
(response) => {
const ctx = router.show(
uri.formatClientLink(
"user",
e.detail.user.name,
"list-tokens"
)
);
ctx.controller.showSuccess(
"Token " + response.token + " created."
);
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
_evtDeleteToken(e) {
this._view.clearMessages();
this._view.disableForm();
if (api.isCurrentAuthToken(e.detail.userToken)) {
router.show(uri.formatClientLink('logout'));
router.show(uri.formatClientLink("logout"));
} else {
e.detail.userToken.delete(e.detail.user.name)
.then(() => {
const ctx = router.show(uri.formatClientLink('user', e.detail.user.name, 'list-tokens'));
ctx.controller.showSuccess('Token ' + e.detail.userToken.token + ' deleted.');
}, error => {
e.detail.userToken.delete(e.detail.user.name).then(
() => {
const ctx = router.show(
uri.formatClientLink(
"user",
e.detail.user.name,
"list-tokens"
)
);
ctx.controller.showSuccess(
"Token " + e.detail.userToken.token + " deleted."
);
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
}
@ -242,27 +307,38 @@ class UserController {
e.detail.userToken.note = e.detail.note;
}
e.detail.userToken.save(e.detail.user.name).then(response => {
const ctx = router.show(uri.formatClientLink('user', e.detail.user.name, 'list-tokens'));
ctx.controller.showSuccess('Token ' + response.token + ' updated.');
}, error => {
e.detail.userToken.save(e.detail.user.name).then(
(response) => {
const ctx = router.show(
uri.formatClientLink(
"user",
e.detail.user.name,
"list-tokens"
)
);
ctx.controller.showSuccess(
"Token " + response.token + " updated."
);
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
}
module.exports = router => {
router.enter(['user', ':name'], (ctx, next) => {
ctx.controller = new UserController(ctx, 'summary');
module.exports = (router) => {
router.enter(["user", ":name"], (ctx, next) => {
ctx.controller = new UserController(ctx, "summary");
});
router.enter(['user', ':name', 'edit'], (ctx, next) => {
ctx.controller = new UserController(ctx, 'edit');
router.enter(["user", ":name", "edit"], (ctx, next) => {
ctx.controller = new UserController(ctx, "edit");
});
router.enter(['user', ':name', 'list-tokens'], (ctx, next) => {
ctx.controller = new UserController(ctx, 'list-tokens');
router.enter(["user", ":name", "list-tokens"], (ctx, next) => {
ctx.controller = new UserController(ctx, "list-tokens");
});
router.enter(['user', ':name', 'delete'], (ctx, next) => {
ctx.controller = new UserController(ctx, 'delete');
router.enter(["user", ":name", "delete"], (ctx, next) => {
ctx.controller = new UserController(ctx, "delete");
});
};

View file

@ -1,27 +1,27 @@
'use strict';
"use strict";
const api = require('../api.js');
const router = require('../router.js');
const uri = require('../util/uri.js');
const UserList = require('../models/user_list.js');
const topNavigation = require('../models/top_navigation.js');
const PageController = require('../controllers/page_controller.js');
const UsersHeaderView = require('../views/users_header_view.js');
const UsersPageView = require('../views/users_page_view.js');
const EmptyView = require('../views/empty_view.js');
const api = require("../api.js");
const router = require("../router.js");
const uri = require("../util/uri.js");
const UserList = require("../models/user_list.js");
const topNavigation = require("../models/top_navigation.js");
const PageController = require("../controllers/page_controller.js");
const UsersHeaderView = require("../views/users_header_view.js");
const UsersPageView = require("../views/users_page_view.js");
const EmptyView = require("../views/empty_view.js");
class UserListController {
constructor(ctx) {
this._pageController = new PageController();
if (!api.hasPrivilege('users:list')) {
if (!api.hasPrivilege("users:list")) {
this._view = new EmptyView();
this._view.showError('You don\'t have privileges to view users.');
this._view.showError("You don't have privileges to view users.");
return;
}
topNavigation.activate('users');
topNavigation.setTitle('Listing users');
topNavigation.activate("users");
topNavigation.setTitle("Listing users");
this._ctx = ctx;
@ -29,8 +29,9 @@ class UserListController {
hostNode: this._pageController.view.pageHeaderHolderNode,
parameters: ctx.parameters,
});
this._headerView.addEventListener(
'navigate', e => this._evtNavigate(e));
this._headerView.addEventListener("navigate", (e) =>
this._evtNavigate(e)
);
this._syncPageController();
}
@ -41,7 +42,8 @@ class UserListController {
_evtNavigate(e) {
router.showNoDispatch(
uri.formatClientLink('users', e.detail.parameters));
uri.formatClientLink("users", e.detail.parameters)
);
Object.assign(this._ctx.parameters, e.detail.parameters);
this._syncPageController();
}
@ -51,17 +53,22 @@ class UserListController {
parameters: this._ctx.parameters,
defaultLimit: 30,
getClientUrlForPage: (offset, limit) => {
const parameters = Object.assign(
{}, this._ctx.parameters, {offset: offset, limit: limit});
return uri.formatClientLink('users', parameters);
const parameters = Object.assign({}, this._ctx.parameters, {
offset: offset,
limit: limit,
});
return uri.formatClientLink("users", parameters);
},
requestPage: (offset, limit) => {
return UserList.search(
this._ctx.parameters.query, offset, limit);
this._ctx.parameters.query,
offset,
limit
);
},
pageRenderer: pageCtx => {
pageRenderer: (pageCtx) => {
Object.assign(pageCtx, {
canViewUsers: api.hasPrivilege('users:view'),
canViewUsers: api.hasPrivilege("users:view"),
});
return new UsersPageView(pageCtx);
},
@ -69,10 +76,8 @@ class UserListController {
}
}
module.exports = router => {
router.enter(
['users'],
(ctx, next) => {
module.exports = (router) => {
router.enter(["users"], (ctx, next) => {
ctx.controller = new UserListController(ctx);
});
};

View file

@ -1,25 +1,25 @@
'use strict';
"use strict";
const router = require('../router.js');
const api = require('../api.js');
const uri = require('../util/uri.js');
const User = require('../models/user.js');
const topNavigation = require('../models/top_navigation.js');
const RegistrationView = require('../views/registration_view.js');
const EmptyView = require('../views/empty_view.js');
const router = require("../router.js");
const api = require("../api.js");
const uri = require("../util/uri.js");
const User = require("../models/user.js");
const topNavigation = require("../models/top_navigation.js");
const RegistrationView = require("../views/registration_view.js");
const EmptyView = require("../views/empty_view.js");
class UserRegistrationController {
constructor() {
if (!api.hasPrivilege('users:create:self')) {
if (!api.hasPrivilege("users:create:self")) {
this._view = new EmptyView();
this._view.showError('Registration is closed.');
this._view.showError("Registration is closed.");
return;
}
topNavigation.activate('register');
topNavigation.setTitle('Registration');
topNavigation.activate("register");
topNavigation.setTitle("Registration");
this._view = new RegistrationView();
this._view.addEventListener('submit', e => this._evtRegister(e));
this._view.addEventListener("submit", (e) => this._evtRegister(e));
}
_evtRegister(e) {
@ -30,30 +30,35 @@ class UserRegistrationController {
user.email = e.detail.email;
user.password = e.detail.password;
const isLoggedIn = api.isLoggedIn();
user.save().then(() => {
user.save()
.then(() => {
if (isLoggedIn) {
return Promise.resolve();
} else {
api.forget();
return api.login(e.detail.name, e.detail.password, false);
}
}).then(() => {
})
.then(
() => {
if (isLoggedIn) {
const ctx = router.show(uri.formatClientLink('users'));
ctx.controller.showSuccess('User added!');
const ctx = router.show(uri.formatClientLink("users"));
ctx.controller.showSuccess("User added!");
} else {
const ctx = router.show(uri.formatClientLink());
ctx.controller.showSuccess('Welcome aboard!');
ctx.controller.showSuccess("Welcome aboard!");
}
}, error => {
},
(error) => {
this._view.showError(error.message);
this._view.enableForm();
});
}
);
}
}
module.exports = router => {
router.enter(['register'], (ctx, next) => {
module.exports = (router) => {
router.enter(["register"], (ctx, next) => {
new UserRegistrationController();
});
};

View file

@ -1,6 +1,6 @@
'use strict';
"use strict";
const views = require('../util/views.js');
const views = require("../util/views.js");
const KEY_TAB = 9;
const KEY_RETURN = 13;
@ -10,14 +10,14 @@ const KEY_UP = 38;
const KEY_DOWN = 40;
function _getSelectionStart(input) {
if ('selectionStart' in input) {
if ("selectionStart" in input) {
return input.selectionStart;
}
if (document.selection) {
input.focus();
const sel = document.selection.createRange();
const selLen = document.selection.createRange().text.length;
sel.moveStart('character', -input.value.length);
sel.moveStart("character", -input.value.length);
return sel.text.length - selLen;
}
return 0;
@ -27,18 +27,22 @@ class AutoCompleteControl {
constructor(sourceInputNode, options) {
this._sourceInputNode = sourceInputNode;
this._options = {};
Object.assign(this._options, {
Object.assign(
this._options,
{
verticalShift: 2,
maxResults: 15,
getTextToFind: () => {
const value = sourceInputNode.value;
const start = _getSelectionStart(sourceInputNode);
return value.substring(0, start).replace(/.*\s+/, '');
return value.substring(0, start).replace(/.*\s+/, "");
},
confirm: null,
delete: null,
getMatches: null,
}, options);
},
options
);
this._showTimeout = null;
this._results = [];
@ -49,22 +53,22 @@ class AutoCompleteControl {
hide() {
window.clearTimeout(this._showTimeout);
this._suggestionDiv.style.display = 'none';
this._suggestionDiv.style.display = "none";
this._isVisible = false;
}
replaceSelectedText(result, addSpace) {
const start = _getSelectionStart(this._sourceInputNode);
let prefix = '';
let prefix = "";
let suffix = this._sourceInputNode.value.substring(start);
let middle = this._sourceInputNode.value.substring(0, start);
const index = middle.lastIndexOf(' ');
const index = middle.lastIndexOf(" ");
if (index !== -1) {
prefix = this._sourceInputNode.value.substring(0, index + 1);
middle = this._sourceInputNode.value.substring(index + 1);
}
this._sourceInputNode.value = (
prefix + result.toString() + ' ' + suffix.trimLeft());
this._sourceInputNode.value =
prefix + result.toString() + " " + suffix.trimLeft();
if (!addSpace) {
this._sourceInputNode.value = this._sourceInputNode.value.trim();
}
@ -86,7 +90,7 @@ class AutoCompleteControl {
}
_show() {
this._suggestionDiv.style.display = 'block';
this._suggestionDiv.style.display = "block";
this._isVisible = true;
}
@ -101,27 +105,30 @@ class AutoCompleteControl {
_install() {
if (!this._sourceInputNode) {
throw new Error('Input element was not found');
throw new Error("Input element was not found");
}
if (this._sourceInputNode.getAttribute('data-autocomplete')) {
if (this._sourceInputNode.getAttribute("data-autocomplete")) {
throw new Error(
'Autocompletion was already added for this element');
"Autocompletion was already added for this element"
);
}
this._sourceInputNode.setAttribute('data-autocomplete', true);
this._sourceInputNode.setAttribute('autocomplete', 'off');
this._sourceInputNode.setAttribute("data-autocomplete", true);
this._sourceInputNode.setAttribute("autocomplete", "off");
this._sourceInputNode.addEventListener(
'keydown', e => this._evtKeyDown(e));
this._sourceInputNode.addEventListener(
'blur', e => this._evtBlur(e));
this._sourceInputNode.addEventListener("keydown", (e) =>
this._evtKeyDown(e)
);
this._sourceInputNode.addEventListener("blur", (e) =>
this._evtBlur(e)
);
this._suggestionDiv = views.htmlToDom(
'<div class="autocomplete"><ul></ul></div>');
this._suggestionList = this._suggestionDiv.querySelector('ul');
'<div class="autocomplete"><ul></ul></div>'
);
this._suggestionList = this._suggestionDiv.querySelector("ul");
document.body.appendChild(this._suggestionDiv);
views.monitorNodeRemoval(
this._sourceInputNode, () => {
views.monitorNodeRemoval(this._sourceInputNode, () => {
this._uninstall();
});
}
@ -174,8 +181,7 @@ class AutoCompleteControl {
func();
} else {
window.clearTimeout(this._showTimeout);
this._showTimeout = window.setTimeout(
() => {
this._showTimeout = window.setTimeout(() => {
this._showOrHide();
}, 250);
}
@ -196,9 +202,11 @@ class AutoCompleteControl {
}
_selectPrevious() {
this._select(this._activeResult === -1 ?
this._results.length - 1 :
this._activeResult - 1);
this._select(
this._activeResult === -1
? this._results.length - 1
: this._activeResult - 1
);
}
_selectNext() {
@ -206,15 +214,18 @@ class AutoCompleteControl {
}
_select(newActiveResult) {
this._activeResult =
newActiveResult.between(0, this._results.length - 1, true) ?
newActiveResult :
-1;
this._activeResult = newActiveResult.between(
0,
this._results.length - 1,
true
)
? newActiveResult
: -1;
this._refreshActiveResult();
}
_updateResults(textToFind) {
this._options.getMatches(textToFind).then(matches => {
this._options.getMatches(textToFind).then((matches) => {
const oldResults = this._results.slice();
this._results = matches.slice(0, this._options.maxResults);
const oldResultsHash = JSON.stringify(oldResults);
@ -237,21 +248,17 @@ class AutoCompleteControl {
}
for (let [resultIndex, resultItem] of this._results.entries()) {
let resultIndexWorkaround = resultIndex;
const listItem = document.createElement('li');
const link = document.createElement('a');
const listItem = document.createElement("li");
const link = document.createElement("a");
link.innerHTML = resultItem.caption;
link.setAttribute('href', '');
link.setAttribute('data-key', resultItem.value);
link.addEventListener(
'mouseenter',
e => {
link.setAttribute("href", "");
link.setAttribute("data-key", resultItem.value);
link.addEventListener("mouseenter", (e) => {
e.preventDefault();
this._activeResult = resultIndexWorkaround;
this._refreshActiveResult();
});
link.addEventListener(
'mousedown',
e => {
link.addEventListener("mousedown", (e) => {
e.preventDefault();
this._activeResult = resultIndexWorkaround;
this._confirm(this._getActiveSuggestion());
@ -263,8 +270,8 @@ class AutoCompleteControl {
this._refreshActiveResult();
// display the suggestions offscreen to get the height
this._suggestionDiv.style.left = '-9999px';
this._suggestionDiv.style.top = '-9999px';
this._suggestionDiv.style.left = "-9999px";
this._suggestionDiv.style.top = "-9999px";
this._show();
const verticalShift = this._options.verticalShift;
const inputRect = this._sourceInputNode.getBoundingClientRect();
@ -275,17 +282,23 @@ class AutoCompleteControl {
// choose where to view the suggestions: if there's more space above
// the input - draw the suggestions above it, otherwise below
const direction =
inputRect.top + (inputRect.height / 2) < viewPortHeight / 2 ? 1 : -1;
inputRect.top + inputRect.height / 2 < viewPortHeight / 2 ? 1 : -1;
let x = inputRect.left - bodyRect.left;
let y = direction === 1 ?
inputRect.bottom - bodyRect.top - verticalShift :
inputRect.top - bodyRect.top - listRect.height + verticalShift;
let y =
direction === 1
? inputRect.bottom - bodyRect.top - verticalShift
: inputRect.top -
bodyRect.top -
listRect.height +
verticalShift;
// remove offscreen items until whole suggestion list can fit on the
// screen
while ((y < 0 || y + listRect.height > viewPortHeight) &&
this._suggestionList.childNodes.length) {
while (
(y < 0 || y + listRect.height > viewPortHeight) &&
this._suggestionList.childNodes.length
) {
this._suggestionList.removeChild(this._suggestionList.lastChild);
const prevHeight = listRect.height;
listRect = this._suggestionDiv.getBoundingClientRect();
@ -295,19 +308,19 @@ class AutoCompleteControl {
}
}
this._suggestionDiv.style.left = x + 'px';
this._suggestionDiv.style.top = y + 'px';
this._suggestionDiv.style.left = x + "px";
this._suggestionDiv.style.top = y + "px";
}
_refreshActiveResult() {
let activeItem = this._suggestionList.querySelector('li.active');
let activeItem = this._suggestionList.querySelector("li.active");
if (activeItem) {
activeItem.classList.remove('active');
activeItem.classList.remove("active");
}
if (this._activeResult >= 0) {
const allItems = this._suggestionList.querySelectorAll('li');
const allItems = this._suggestionList.querySelectorAll("li");
activeItem = allItems[this._activeResult];
activeItem.classList.add('active');
activeItem.classList.add("active");
}
}
}

View file

@ -1,12 +1,12 @@
'use strict';
"use strict";
const api = require('../api.js');
const misc = require('../util/misc.js');
const events = require('../events.js');
const views = require('../util/views.js');
const api = require("../api.js");
const misc = require("../util/misc.js");
const events = require("../events.js");
const views = require("../util/views.js");
const template = views.getTemplate('comment');
const scoreTemplate = views.getTemplate('score');
const template = views.getTemplate("comment");
const scoreTemplate = views.getTemplate("score");
class CommentControl extends events.EventTarget {
constructor(hostNode, comment, onlyEditing) {
@ -16,104 +16,111 @@ class CommentControl extends events.EventTarget {
this._onlyEditing = onlyEditing;
if (comment) {
comment.addEventListener(
'change', e => this._evtChange(e));
comment.addEventListener(
'changeScore', e => this._evtChangeScore(e));
comment.addEventListener("change", (e) => this._evtChange(e));
comment.addEventListener("changeScore", (e) =>
this._evtChangeScore(e)
);
}
const isLoggedIn = comment && api.isLoggedIn(comment.user);
const infix = isLoggedIn ? 'own' : 'any';
views.replaceContent(this._hostNode, template({
const infix = isLoggedIn ? "own" : "any";
views.replaceContent(
this._hostNode,
template({
comment: comment,
user: comment ? comment.user : api.user,
canViewUsers: api.hasPrivilege('users:view'),
canViewUsers: api.hasPrivilege("users:view"),
canEditComment: api.hasPrivilege(`comments:edit:${infix}`),
canDeleteComment: api.hasPrivilege(`comments:delete:${infix}`),
onlyEditing: onlyEditing,
}));
})
);
if (this._editButtonNodes) {
for (let node of this._editButtonNodes) {
node.addEventListener('click', e => this._evtEditClick(e));
node.addEventListener("click", (e) => this._evtEditClick(e));
}
}
if (this._deleteButtonNode) {
this._deleteButtonNode.addEventListener(
'click', e => this._evtDeleteClick(e));
this._deleteButtonNode.addEventListener("click", (e) =>
this._evtDeleteClick(e)
);
}
if (this._previewEditingButtonNode) {
this._previewEditingButtonNode.addEventListener(
'click', e => this._evtPreviewEditingClick(e));
this._previewEditingButtonNode.addEventListener("click", (e) =>
this._evtPreviewEditingClick(e)
);
}
if (this._saveChangesButtonNode) {
this._saveChangesButtonNode.addEventListener(
'click', e => this._evtSaveChangesClick(e));
this._saveChangesButtonNode.addEventListener("click", (e) =>
this._evtSaveChangesClick(e)
);
}
if (this._cancelEditingButtonNode) {
this._cancelEditingButtonNode.addEventListener(
'click', e => this._evtCancelEditingClick(e));
this._cancelEditingButtonNode.addEventListener("click", (e) =>
this._evtCancelEditingClick(e)
);
}
this._installScore();
if (onlyEditing) {
this._selectNav('edit');
this._selectTab('edit');
this._selectNav("edit");
this._selectTab("edit");
} else {
this._selectNav('readonly');
this._selectTab('preview');
this._selectNav("readonly");
this._selectTab("preview");
}
}
get _formNode() {
return this._hostNode.querySelector('form');
return this._hostNode.querySelector("form");
}
get _scoreContainerNode() {
return this._hostNode.querySelector('.score-container');
return this._hostNode.querySelector(".score-container");
}
get _editButtonNodes() {
return this._hostNode.querySelectorAll('li.edit>a, a.edit');
return this._hostNode.querySelectorAll("li.edit>a, a.edit");
}
get _previewEditingButtonNode() {
return this._hostNode.querySelector('li.preview>a');
return this._hostNode.querySelector("li.preview>a");
}
get _deleteButtonNode() {
return this._hostNode.querySelector('.delete');
return this._hostNode.querySelector(".delete");
}
get _upvoteButtonNode() {
return this._hostNode.querySelector('.upvote');
return this._hostNode.querySelector(".upvote");
}
get _downvoteButtonNode() {
return this._hostNode.querySelector('.downvote');
return this._hostNode.querySelector(".downvote");
}
get _saveChangesButtonNode() {
return this._hostNode.querySelector('.save-changes');
return this._hostNode.querySelector(".save-changes");
}
get _cancelEditingButtonNode() {
return this._hostNode.querySelector('.cancel-editing');
return this._hostNode.querySelector(".cancel-editing");
}
get _textareaNode() {
return this._hostNode.querySelector('.tab.edit textarea');
return this._hostNode.querySelector(".tab.edit textarea");
}
get _contentNode() {
return this._hostNode.querySelector('.tab.preview .comment-content');
return this._hostNode.querySelector(".tab.preview .comment-content");
}
get _heightKeeperNode() {
return this._hostNode.querySelector('.keep-height');
return this._hostNode.querySelector(".keep-height");
}
_installScore() {
@ -122,32 +129,35 @@ class CommentControl extends events.EventTarget {
scoreTemplate({
score: this._comment ? this._comment.score : 0,
ownScore: this._comment ? this._comment.ownScore : 0,
canScore: api.hasPrivilege('comments:score'),
}));
canScore: api.hasPrivilege("comments:score"),
})
);
if (this._upvoteButtonNode) {
this._upvoteButtonNode.addEventListener(
'click', e => this._evtScoreClick(e, 1));
this._upvoteButtonNode.addEventListener("click", (e) =>
this._evtScoreClick(e, 1)
);
}
if (this._downvoteButtonNode) {
this._downvoteButtonNode.addEventListener(
'click', e => this._evtScoreClick(e, -1));
this._downvoteButtonNode.addEventListener("click", (e) =>
this._evtScoreClick(e, -1)
);
}
}
enterEditMode() {
this._selectNav('edit');
this._selectTab('edit');
this._selectNav("edit");
this._selectTab("edit");
}
exitEditMode() {
if (this._onlyEditing) {
this._selectNav('edit');
this._selectTab('edit');
this._setText('');
this._selectNav("edit");
this._selectTab("edit");
this._setText("");
} else {
this._selectNav('readonly');
this._selectTab('preview');
this._selectNav("readonly");
this._selectTab("preview");
this._setText(this._comment.text);
}
this._forgetHeight();
@ -173,27 +183,31 @@ class CommentControl extends events.EventTarget {
_evtScoreClick(e, score) {
e.preventDefault();
if (!api.hasPrivilege('comments:score')) {
if (!api.hasPrivilege("comments:score")) {
return;
}
this.dispatchEvent(new CustomEvent('score', {
this.dispatchEvent(
new CustomEvent("score", {
detail: {
comment: this._comment,
score: this._comment.ownScore === score ? 0 : score,
},
}));
})
);
}
_evtDeleteClick(e) {
e.preventDefault();
if (!window.confirm('Are you sure you want to delete this comment?')) {
if (!window.confirm("Are you sure you want to delete this comment?")) {
return;
}
this.dispatchEvent(new CustomEvent('delete', {
this.dispatchEvent(
new CustomEvent("delete", {
detail: {
comment: this._comment,
},
}));
})
);
}
_evtChange(e) {
@ -206,21 +220,24 @@ class CommentControl extends events.EventTarget {
_evtPreviewEditingClick(e) {
e.preventDefault();
this._contentNode.innerHTML =
misc.formatMarkdown(this._textareaNode.value);
this._selectTab('edit');
this._selectTab('preview');
this._contentNode.innerHTML = misc.formatMarkdown(
this._textareaNode.value
);
this._selectTab("edit");
this._selectTab("preview");
}
_evtSaveChangesClick(e) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', {
this.dispatchEvent(
new CustomEvent("submit", {
detail: {
target: this,
comment: this._comment,
text: this._textareaNode.value,
},
}));
})
);
}
_evtCancelEditingClick(e) {
@ -234,22 +251,22 @@ class CommentControl extends events.EventTarget {
}
_selectNav(modeName) {
for (let node of this._hostNode.querySelectorAll('nav')) {
node.classList.toggle('active', node.classList.contains(modeName));
for (let node of this._hostNode.querySelectorAll("nav")) {
node.classList.toggle("active", node.classList.contains(modeName));
}
}
_selectTab(tabName) {
this._ensureHeight();
for (let node of this._hostNode.querySelectorAll('.tab, .tabs li')) {
node.classList.toggle('active', node.classList.contains(tabName));
for (let node of this._hostNode.querySelectorAll(".tab, .tabs li")) {
node.classList.toggle("active", node.classList.contains(tabName));
}
}
_ensureHeight() {
this._heightKeeperNode.style.minHeight =
this._heightKeeperNode.getBoundingClientRect().height + 'px';
this._heightKeeperNode.getBoundingClientRect().height + "px";
}
_forgetHeight() {

View file

@ -1,10 +1,10 @@
'use strict';
"use strict";
const events = require('../events.js');
const views = require('../util/views.js');
const CommentControl = require('../controls/comment_control.js');
const events = require("../events.js");
const views = require("../util/views.js");
const CommentControl = require("../controls/comment_control.js");
const template = views.getTemplate('comment-list');
const template = views.getTemplate("comment-list");
class CommentListControl extends events.EventTarget {
constructor(hostNode, comments, reversed) {
@ -13,8 +13,8 @@ class CommentListControl extends events.EventTarget {
this._comments = comments;
this._commentIdToNode = {};
comments.addEventListener('add', e => this._evtAdd(e));
comments.addEventListener('remove', e => this._evtRemove(e));
comments.addEventListener("add", (e) => this._evtAdd(e));
comments.addEventListener("remove", (e) => this._evtRemove(e));
views.replaceContent(this._hostNode, template());
@ -28,16 +28,19 @@ class CommentListControl extends events.EventTarget {
}
get _commentListNode() {
return this._hostNode.querySelector('ul');
return this._hostNode.querySelector("ul");
}
_installCommentNode(comment) {
const commentListItemNode = document.createElement('li');
const commentListItemNode = document.createElement("li");
const commentControl = new CommentControl(
commentListItemNode, comment, false);
events.proxyEvent(commentControl, this, 'submit');
events.proxyEvent(commentControl, this, 'score');
events.proxyEvent(commentControl, this, 'delete');
commentListItemNode,
comment,
false
);
events.proxyEvent(commentControl, this, "submit");
events.proxyEvent(commentControl, this, "score");
events.proxyEvent(commentControl, this, "delete");
this._commentIdToNode[comment.id] = commentListItemNode;
this._commentListNode.appendChild(commentListItemNode);
}

View file

@ -1,26 +1,28 @@
'use strict';
"use strict";
const ICON_CLASS_OPENED = 'fa-chevron-down';
const ICON_CLASS_CLOSED = 'fa-chevron-up';
const ICON_CLASS_OPENED = "fa-chevron-down";
const ICON_CLASS_CLOSED = "fa-chevron-up";
const views = require('../util/views.js');
const views = require("../util/views.js");
const template = views.getTemplate('expander');
const template = views.getTemplate("expander");
class ExpanderControl {
constructor(name, title, nodes) {
this._name = name;
nodes = Array.from(nodes).filter(n => n);
nodes = Array.from(nodes).filter((n) => n);
if (!nodes.length) {
return;
}
const expanderNode = template({ title: title });
const toggleLinkNode = expanderNode.querySelector('a');
const toggleIconNode = expanderNode.querySelector('i');
const expanderContentNode = expanderNode.querySelector('div');
toggleLinkNode.addEventListener('click', e => this._evtToggleClick(e));
const toggleLinkNode = expanderNode.querySelector("a");
const toggleIconNode = expanderNode.querySelector("i");
const expanderContentNode = expanderNode.querySelector("div");
toggleLinkNode.addEventListener("click", (e) =>
this._evtToggleClick(e)
);
nodes[0].parentNode.insertBefore(expanderNode, nodes[0]);
@ -32,29 +34,30 @@ class ExpanderControl {
this._toggleIconNode = toggleIconNode;
expanderNode.classList.toggle(
'collapsed',
this._allStates[this._name] === undefined ?
false :
!this._allStates[this._name]);
"collapsed",
this._allStates[this._name] === undefined
? false
: !this._allStates[this._name]
);
this._syncIcon();
}
// eslint-disable-next-line accessor-pairs
set title(newTitle) {
if (this._expanderNode) {
this._expanderNode
.querySelector('header span')
.textContent = newTitle;
this._expanderNode.querySelector(
"header span"
).textContent = newTitle;
}
}
get _isOpened() {
return !this._expanderNode.classList.contains('collapsed');
return !this._expanderNode.classList.contains("collapsed");
}
get _allStates() {
try {
return JSON.parse(localStorage.getItem('expander')) || {};
return JSON.parse(localStorage.getItem("expander")) || {};
} catch (e) {
return {};
}
@ -63,12 +66,12 @@ class ExpanderControl {
_save() {
const newStates = Object.assign({}, this._allStates);
newStates[this._name] = this._isOpened;
localStorage.setItem('expander', JSON.stringify(newStates));
localStorage.setItem("expander", JSON.stringify(newStates));
}
_evtToggleClick(e) {
e.preventDefault();
this._expanderNode.classList.toggle('collapsed');
this._expanderNode.classList.toggle("collapsed");
this._save();
this._syncIcon();
}

View file

@ -1,9 +1,9 @@
'use strict';
"use strict";
const events = require('../events.js');
const views = require('../util/views.js');
const events = require("../events.js");
const views = require("../util/views.js");
const template = views.getTemplate('file-dropper');
const template = views.getTemplate("file-dropper");
const KEY_RETURN = 13;
@ -17,37 +17,42 @@ class FileDropperControl extends events.EventTarget {
allowMultiple: options.allowMultiple,
allowUrls: options.allowUrls,
lock: options.lock,
id: 'file-' + Math.random().toString(36).substring(7),
id: "file-" + Math.random().toString(36).substring(7),
urlPlaceholder:
options.urlPlaceholder || 'Alternatively, paste an URL here.',
options.urlPlaceholder || "Alternatively, paste an URL here.",
});
this._dropperNode = source.querySelector('.file-dropper');
this._urlInputNode = source.querySelector('input[type=text]');
this._urlConfirmButtonNode = source.querySelector('button');
this._fileInputNode = source.querySelector('input[type=file]');
this._fileInputNode.style.display = 'none';
this._dropperNode = source.querySelector(".file-dropper");
this._urlInputNode = source.querySelector("input[type=text]");
this._urlConfirmButtonNode = source.querySelector("button");
this._fileInputNode = source.querySelector("input[type=file]");
this._fileInputNode.style.display = "none";
this._fileInputNode.multiple = options.allowMultiple || false;
this._counter = 0;
this._dropperNode.addEventListener(
'dragenter', e => this._evtDragEnter(e));
this._dropperNode.addEventListener(
'dragleave', e => this._evtDragLeave(e));
this._dropperNode.addEventListener(
'dragover', e => this._evtDragOver(e));
this._dropperNode.addEventListener(
'drop', e => this._evtDrop(e));
this._fileInputNode.addEventListener(
'change', e => this._evtFileChange(e));
this._dropperNode.addEventListener("dragenter", (e) =>
this._evtDragEnter(e)
);
this._dropperNode.addEventListener("dragleave", (e) =>
this._evtDragLeave(e)
);
this._dropperNode.addEventListener("dragover", (e) =>
this._evtDragOver(e)
);
this._dropperNode.addEventListener("drop", (e) => this._evtDrop(e));
this._fileInputNode.addEventListener("change", (e) =>
this._evtFileChange(e)
);
if (this._urlInputNode) {
this._urlInputNode.addEventListener(
'keydown', e => this._evtUrlInputKeyDown(e));
this._urlInputNode.addEventListener("keydown", (e) =>
this._evtUrlInputKeyDown(e)
);
}
if (this._urlConfirmButtonNode) {
this._urlConfirmButtonNode.addEventListener(
'click', e => this._evtUrlConfirmButtonClick(e));
this._urlConfirmButtonNode.addEventListener("click", (e) =>
this._evtUrlConfirmButtonClick(e)
);
}
this._originalHtml = this._dropperNode.innerHTML;
@ -56,24 +61,27 @@ class FileDropperControl extends events.EventTarget {
reset() {
this._dropperNode.innerHTML = this._originalHtml;
this.dispatchEvent(new CustomEvent('reset'));
this.dispatchEvent(new CustomEvent("reset"));
}
_emitFiles(files) {
files = Array.from(files);
if (this._options.lock) {
this._dropperNode.innerText =
files.map(file => file.name).join(', ');
this._dropperNode.innerText = files
.map((file) => file.name)
.join(", ");
}
this.dispatchEvent(
new CustomEvent('fileadd', {detail: {files: files}}));
new CustomEvent("fileadd", { detail: { files: files } })
);
}
_emitUrls(urls) {
urls = Array.from(urls).map(url => url.trim());
urls = Array.from(urls).map((url) => url.trim());
if (this._options.lock) {
this._dropperNode.innerText =
urls.map(url => url.split(/\//).reverse()[0]).join(', ');
this._dropperNode.innerText = urls
.map((url) => url.split(/\//).reverse()[0])
.join(", ");
}
for (let url of urls) {
if (!url) {
@ -84,18 +92,20 @@ class FileDropperControl extends events.EventTarget {
return;
}
}
this.dispatchEvent(new CustomEvent('urladd', {detail: {urls: urls}}));
this.dispatchEvent(
new CustomEvent("urladd", { detail: { urls: urls } })
);
}
_evtDragEnter(e) {
this._dropperNode.classList.add('active');
this._dropperNode.classList.add("active");
this._counter++;
}
_evtDragLeave(e) {
this._counter--;
if (this._counter === 0) {
this._dropperNode.classList.remove('active');
this._dropperNode.classList.remove("active");
}
}
@ -109,12 +119,12 @@ class FileDropperControl extends events.EventTarget {
_evtDrop(e) {
e.preventDefault();
this._dropperNode.classList.remove('active');
this._dropperNode.classList.remove("active");
if (!e.dataTransfer.files.length) {
window.alert('Only files are supported.');
window.alert("Only files are supported.");
}
if (!this._options.allowMultiple && e.dataTransfer.files.length > 1) {
window.alert('Cannot select multiple files.');
window.alert("Cannot select multiple files.");
}
this._emitFiles(e.dataTransfer.files);
}
@ -124,16 +134,16 @@ class FileDropperControl extends events.EventTarget {
return;
}
e.preventDefault();
this._dropperNode.classList.remove('active');
this._dropperNode.classList.remove("active");
this._emitUrls(this._urlInputNode.value.split(/[\r\n]/));
this._urlInputNode.value = '';
this._urlInputNode.value = "";
}
_evtUrlConfirmButtonClick(e) {
e.preventDefault();
this._dropperNode.classList.remove('active');
this._dropperNode.classList.remove("active");
this._emitUrls(this._urlInputNode.value.split(/[\r\n]/));
this._urlInputNode.value = '';
this._urlInputNode.value = "";
}
}

View file

@ -1,18 +1,22 @@
'use strict';
"use strict";
const misc = require('../util/misc.js');
const PoolList = require('../models/pool_list.js');
const AutoCompleteControl = require('./auto_complete_control.js');
const misc = require("../util/misc.js");
const PoolList = require("../models/pool_list.js");
const AutoCompleteControl = require("./auto_complete_control.js");
function _poolListToMatches(pools, options) {
return [...pools].sort((pool1, pool2) => {
return [...pools]
.sort((pool1, pool2) => {
return pool2.postCount - pool1.postCount;
}).map(pool => {
let cssName = misc.makeCssName(pool.category, 'pool');
const caption = (
'<span class="' + cssName + '">'
+ misc.escapeHtml(pool.names[0] + ' (' + pool.postCount + ')')
+ '</span>');
})
.map((pool) => {
let cssName = misc.makeCssName(pool.category, "pool");
const caption =
'<span class="' +
cssName +
'">' +
misc.escapeHtml(pool.names[0] + " (" + pool.postCount + ")") +
"</span>";
return {
caption: caption,
value: pool,
@ -24,20 +28,27 @@ class PoolAutoCompleteControl extends AutoCompleteControl {
constructor(input, options) {
const minLengthForPartialSearch = 3;
options.getMatches = text => {
options.getMatches = (text) => {
const term = misc.escapeSearchTerm(text);
const query = (
text.length < minLengthForPartialSearch
? term + '*'
: '*' + term + '*') + ' sort:post-count';
const query =
(text.length < minLengthForPartialSearch
? term + "*"
: "*" + term + "*") + " sort:post-count";
return new Promise((resolve, reject) => {
PoolList.search(
query, 0, this._options.maxResults, ['id', 'names', 'category', 'postCount', 'version'])
.then(
response => resolve(
_poolListToMatches(response.results, this._options)),
reject);
PoolList.search(query, 0, this._options.maxResults, [
"id",
"names",
"category",
"postCount",
"version",
]).then(
(response) =>
resolve(
_poolListToMatches(response.results, this._options)
),
reject
);
});
};

View file

@ -1,24 +1,24 @@
'use strict';
"use strict";
const api = require('../api.js');
const pools = require('../pools.js');
const misc = require('../util/misc.js');
const uri = require('../util/uri.js');
const Pool = require('../models/pool.js');
const settings = require('../models/settings.js');
const events = require('../events.js');
const views = require('../util/views.js');
const PoolAutoCompleteControl = require('./pool_auto_complete_control.js');
const api = require("../api.js");
const pools = require("../pools.js");
const misc = require("../util/misc.js");
const uri = require("../util/uri.js");
const Pool = require("../models/pool.js");
const settings = require("../models/settings.js");
const events = require("../events.js");
const views = require("../util/views.js");
const PoolAutoCompleteControl = require("./pool_auto_complete_control.js");
const KEY_SPACE = 32;
const KEY_RETURN = 13;
const SOURCE_INIT = 'init';
const SOURCE_IMPLICATION = 'implication';
const SOURCE_USER_INPUT = 'user-input';
const SOURCE_CLIPBOARD = 'clipboard';
const SOURCE_INIT = "init";
const SOURCE_IMPLICATION = "implication";
const SOURCE_USER_INPUT = "user-input";
const SOURCE_CLIPBOARD = "clipboard";
const template = views.getTemplate('pool-input');
const template = views.getTemplate("pool-input");
function _fadeOutListItemNodeStatus(listItemNode) {
if (listItemNode.classList.length) {
@ -27,8 +27,7 @@ function _fadeOutListItemNodeStatus(listItemNode) {
}
listItemNode.fadeTimeout = window.setTimeout(() => {
while (listItemNode.classList.length) {
listItemNode.classList.remove(
listItemNode.classList.item(0));
listItemNode.classList.remove(listItemNode.classList.item(0));
}
listItemNode.fadeTimeout = null;
}, 2500);
@ -45,29 +44,33 @@ class PoolInputControl extends events.EventTarget {
// dom
const editAreaNode = template();
this._editAreaNode = editAreaNode;
this._poolInputNode = editAreaNode.querySelector('input');
this._poolListNode = editAreaNode.querySelector('ul.compact-pools');
this._poolInputNode = editAreaNode.querySelector("input");
this._poolListNode = editAreaNode.querySelector("ul.compact-pools");
this._autoCompleteControl = new PoolAutoCompleteControl(
this._poolInputNode, {
this._poolInputNode,
{
getTextToFind: () => {
return this._poolInputNode.value;
},
confirm: pool => {
this._poolInputNode.value = '';
confirm: (pool) => {
this._poolInputNode.value = "";
this.addPool(pool, SOURCE_USER_INPUT);
},
delete: pool => {
this._poolInputNode.value = '';
delete: (pool) => {
this._poolInputNode.value = "";
this.deletePool(pool);
},
verticalShift: -2
});
verticalShift: -2,
}
);
// show
this._hostNode.style.display = 'none';
this._hostNode.style.display = "none";
this._hostNode.parentNode.insertBefore(
this._editAreaNode, hostNode.nextSibling);
this._editAreaNode,
hostNode.nextSibling
);
// add existing pools
for (let pool of [...this.pools]) {
@ -81,19 +84,21 @@ class PoolInputControl extends events.EventTarget {
return Promise.resolve();
}
this.pools.add(pool, false)
this.pools.add(pool, false);
const listItemNode = this._createListItemNode(pool);
if (!pool.category) {
listItemNode.classList.add('new');
listItemNode.classList.add("new");
}
this._poolListNode.prependChild(listItemNode);
_fadeOutListItemNodeStatus(listItemNode);
this.dispatchEvent(new CustomEvent('add', {
this.dispatchEvent(
new CustomEvent("add", {
detail: { pool: pool, source: source },
}));
this.dispatchEvent(new CustomEvent('change'));
})
);
this.dispatchEvent(new CustomEvent("change"));
return Promise.resolve();
}
@ -107,52 +112,57 @@ class PoolInputControl extends events.EventTarget {
this._deleteListItemNode(pool);
this.dispatchEvent(new CustomEvent('remove', {
this.dispatchEvent(
new CustomEvent("remove", {
detail: { pool: pool },
}));
this.dispatchEvent(new CustomEvent('change'));
})
);
this.dispatchEvent(new CustomEvent("change"));
}
_createListItemNode(pool) {
const className = pool.category ?
misc.makeCssName(pool.category, 'pool') :
null;
const className = pool.category
? misc.makeCssName(pool.category, "pool")
: null;
const poolLinkNode = document.createElement('a');
const poolLinkNode = document.createElement("a");
if (className) {
poolLinkNode.classList.add(className);
}
poolLinkNode.setAttribute(
'href', uri.formatClientLink('pool', pool.names[0]));
"href",
uri.formatClientLink("pool", pool.names[0])
);
const poolIconNode = document.createElement('i');
poolIconNode.classList.add('fa');
poolIconNode.classList.add('fa-pool');
const poolIconNode = document.createElement("i");
poolIconNode.classList.add("fa");
poolIconNode.classList.add("fa-pool");
poolLinkNode.appendChild(poolIconNode);
const searchLinkNode = document.createElement('a');
const searchLinkNode = document.createElement("a");
if (className) {
searchLinkNode.classList.add(className);
}
searchLinkNode.setAttribute(
'href', uri.formatClientLink(
'posts', {query: "pool:" + pool.id}));
searchLinkNode.textContent = pool.names[0] + ' ';
"href",
uri.formatClientLink("posts", { query: "pool:" + pool.id })
);
searchLinkNode.textContent = pool.names[0] + " ";
const usagesNode = document.createElement('span');
usagesNode.classList.add('pool-usages');
usagesNode.setAttribute('data-pseudo-content', pool.postCount);
const usagesNode = document.createElement("span");
usagesNode.classList.add("pool-usages");
usagesNode.setAttribute("data-pseudo-content", pool.postCount);
const removalLinkNode = document.createElement('a');
removalLinkNode.classList.add('remove-pool');
removalLinkNode.setAttribute('href', '');
removalLinkNode.setAttribute('data-pseudo-content', '×');
removalLinkNode.addEventListener('click', e => {
const removalLinkNode = document.createElement("a");
removalLinkNode.classList.add("remove-pool");
removalLinkNode.setAttribute("href", "");
removalLinkNode.setAttribute("data-pseudo-content", "×");
removalLinkNode.addEventListener("click", (e) => {
e.preventDefault();
this.deletePool(pool);
});
const listItemNode = document.createElement('li');
const listItemNode = document.createElement("li");
listItemNode.appendChild(removalLinkNode);
listItemNode.appendChild(poolLinkNode);
listItemNode.appendChild(searchLinkNode);

View file

@ -1,36 +1,38 @@
'use strict';
"use strict";
const settings = require('../models/settings.js');
const views = require('../util/views.js');
const optimizedResize = require('../util/optimized_resize.js');
const settings = require("../models/settings.js");
const views = require("../util/views.js");
const optimizedResize = require("../util/optimized_resize.js");
class PostContentControl {
constructor(hostNode, post, viewportSizeCalculator, fitFunctionOverride) {
this._post = post;
this._viewportSizeCalculator = viewportSizeCalculator;
this._hostNode = hostNode;
this._template = views.getTemplate('post-content');
this._template = views.getTemplate("post-content");
let fitMode = settings.get().fitMode;
if (typeof fitFunctionOverride !== 'undefined') {
if (typeof fitFunctionOverride !== "undefined") {
fitMode = fitFunctionOverride;
}
this._currentFitFunction = {
'fit-both': this.fitBoth,
'fit-original': this.fitOriginal,
'fit-width': this.fitWidth,
'fit-height': this.fitHeight,
this._currentFitFunction =
{
"fit-both": this.fitBoth,
"fit-original": this.fitOriginal,
"fit-width": this.fitWidth,
"fit-height": this.fitHeight,
}[fitMode] || this.fitBoth;
this._install();
this._post.addEventListener(
'changeContent', e => this._evtPostContentChange(e));
this._post.addEventListener("changeContent", (e) =>
this._evtPostContentChange(e)
);
}
disableOverlay() {
this._hostNode.querySelector('.post-overlay').style.display = 'none';
this._hostNode.querySelector(".post-overlay").style.display = "none";
}
fitWidth() {
@ -92,10 +94,11 @@ class PostContentControl {
_resize(width, height) {
const resizeListenerNodes = [this._postContentNode].concat(
...this._postContentNode.querySelectorAll('.resize-listener'));
...this._postContentNode.querySelectorAll(".resize-listener")
);
for (let node of resizeListenerNodes) {
node.style.width = width + 'px';
node.style.height = height + 'px';
node.style.width = width + "px";
node.style.height = height + "px";
}
}
@ -106,8 +109,7 @@ class PostContentControl {
_install() {
this._reinstall();
optimizedResize.add(() => this._refreshSize());
views.monitorNodeRemoval(
this._hostNode, () => {
views.monitorNodeRemoval(this._hostNode, () => {
this._uninstall();
});
}
@ -118,7 +120,7 @@ class PostContentControl {
autoplay: settings.get().autoplayVideos,
});
if (settings.get().transparencyGrid) {
newNode.classList.add('transparency-grid');
newNode.classList.add("transparency-grid");
}
if (this._postContentNode) {
this._hostNode.replaceChild(newNode, this._postContentNode);

View file

@ -1,17 +1,17 @@
'use strict';
"use strict";
const api = require('../api.js');
const events = require('../events.js');
const misc = require('../util/misc.js');
const views = require('../util/views.js');
const Note = require('../models/note.js');
const Point = require('../models/point.js');
const TagInputControl = require('./tag_input_control.js');
const PoolInputControl = require('./pool_input_control.js');
const ExpanderControl = require('../controls/expander_control.js');
const FileDropperControl = require('../controls/file_dropper_control.js');
const api = require("../api.js");
const events = require("../events.js");
const misc = require("../util/misc.js");
const views = require("../util/views.js");
const Note = require("../models/note.js");
const Point = require("../models/point.js");
const TagInputControl = require("./tag_input_control.js");
const PoolInputControl = require("./pool_input_control.js");
const ExpanderControl = require("../controls/expander_control.js");
const FileDropperControl = require("../controls/file_dropper_control.js");
const template = views.getTemplate('post-edit-sidebar');
const template = views.getTemplate("post-edit-sidebar");
class PostEditSidebarControl extends events.EventTarget {
constructor(hostNode, post, postContentControl, postNotesOverlayControl) {
@ -24,180 +24,220 @@ class PostEditSidebarControl extends events.EventTarget {
this._postNotesOverlayControl.switchToPassiveEdit();
views.replaceContent(this._hostNode, template({
views.replaceContent(
this._hostNode,
template({
post: this._post,
enableSafety: api.safetyEnabled(),
hasClipboard: document.queryCommandSupported('copy'),
canEditPostSafety: api.hasPrivilege('posts:edit:safety'),
canEditPostSource: api.hasPrivilege('posts:edit:source'),
canEditPostTags: api.hasPrivilege('posts:edit:tags'),
canEditPostRelations: api.hasPrivilege('posts:edit:relations'),
canEditPostNotes: api.hasPrivilege('posts:edit:notes') &&
post.type !== 'video' &&
post.type !== 'flash',
canEditPostFlags: api.hasPrivilege('posts:edit:flags'),
canEditPostContent: api.hasPrivilege('posts:edit:content'),
canEditPostThumbnail: api.hasPrivilege('posts:edit:thumbnail'),
canEditPoolPosts: api.hasPrivilege('pools:edit:posts'),
canCreateAnonymousPosts: api.hasPrivilege('posts:create:anonymous'),
canDeletePosts: api.hasPrivilege('posts:delete'),
canFeaturePosts: api.hasPrivilege('posts:feature'),
canMergePosts: api.hasPrivilege('posts:merge'),
}));
hasClipboard: document.queryCommandSupported("copy"),
canEditPostSafety: api.hasPrivilege("posts:edit:safety"),
canEditPostSource: api.hasPrivilege("posts:edit:source"),
canEditPostTags: api.hasPrivilege("posts:edit:tags"),
canEditPostRelations: api.hasPrivilege("posts:edit:relations"),
canEditPostNotes:
api.hasPrivilege("posts:edit:notes") &&
post.type !== "video" &&
post.type !== "flash",
canEditPostFlags: api.hasPrivilege("posts:edit:flags"),
canEditPostContent: api.hasPrivilege("posts:edit:content"),
canEditPostThumbnail: api.hasPrivilege("posts:edit:thumbnail"),
canEditPoolPosts: api.hasPrivilege("pools:edit:posts"),
canCreateAnonymousPosts: api.hasPrivilege(
"posts:create:anonymous"
),
canDeletePosts: api.hasPrivilege("posts:delete"),
canFeaturePosts: api.hasPrivilege("posts:feature"),
canMergePosts: api.hasPrivilege("posts:merge"),
})
);
new ExpanderControl(
'post-info',
'Basic info',
this._hostNode.querySelectorAll('.safety, .relations, .flags, .post-source'));
"post-info",
"Basic info",
this._hostNode.querySelectorAll(
".safety, .relations, .flags, .post-source"
)
);
this._tagsExpander = new ExpanderControl(
'post-tags',
"post-tags",
`Tags (${this._post.tags.length})`,
this._hostNode.querySelectorAll('.tags'));
this._hostNode.querySelectorAll(".tags")
);
this._notesExpander = new ExpanderControl(
'post-notes',
'Notes',
this._hostNode.querySelectorAll('.notes'));
"post-notes",
"Notes",
this._hostNode.querySelectorAll(".notes")
);
this._poolsExpander = new ExpanderControl(
'post-pools',
"post-pools",
`Pools (${this._post.pools.length})`,
this._hostNode.querySelectorAll('.pools'));
this._hostNode.querySelectorAll(".pools")
);
new ExpanderControl(
'post-content',
'Content',
this._hostNode.querySelectorAll('.post-content, .post-thumbnail'));
"post-content",
"Content",
this._hostNode.querySelectorAll(".post-content, .post-thumbnail")
);
new ExpanderControl(
'post-management',
'Management',
this._hostNode.querySelectorAll('.management'));
"post-management",
"Management",
this._hostNode.querySelectorAll(".management")
);
this._syncExpanderTitles();
if (this._formNode) {
this._formNode.addEventListener('submit', e => this._evtSubmit(e));
this._formNode.addEventListener("submit", (e) =>
this._evtSubmit(e)
);
}
if (this._tagInputNode) {
this._tagControl = new TagInputControl(
this._tagInputNode, post.tags);
this._tagInputNode,
post.tags
);
}
if (this._poolInputNode) {
this._poolControl = new PoolInputControl(
this._poolInputNode, post.pools);
this._poolInputNode,
post.pools
);
}
if (this._contentInputNode) {
this._contentFileDropper = new FileDropperControl(
this._contentInputNode, {allowUrls: true,
this._contentInputNode,
{
allowUrls: true,
lock: true,
urlPlaceholder: '...or paste an URL here.'});
this._contentFileDropper.addEventListener('fileadd', e => {
urlPlaceholder: "...or paste an URL here.",
}
);
this._contentFileDropper.addEventListener("fileadd", (e) => {
this._newPostContent = e.detail.files[0];
});
this._contentFileDropper.addEventListener('urladd', e => {
this._contentFileDropper.addEventListener("urladd", (e) => {
this._newPostContent = e.detail.urls[0];
});
}
if (this._thumbnailInputNode) {
this._thumbnailFileDropper = new FileDropperControl(
this._thumbnailInputNode, {lock: true});
this._thumbnailFileDropper.addEventListener('fileadd', e => {
this._thumbnailInputNode,
{ lock: true }
);
this._thumbnailFileDropper.addEventListener("fileadd", (e) => {
this._newPostThumbnail = e.detail.files[0];
this._thumbnailRemovalLinkNode.style.display = 'block';
this._thumbnailRemovalLinkNode.style.display = "block";
});
}
if (this._thumbnailRemovalLinkNode) {
this._thumbnailRemovalLinkNode.addEventListener(
'click', e => this._evtRemoveThumbnailClick(e));
this._thumbnailRemovalLinkNode.style.display =
this._post.hasCustomThumbnail ? 'block' : 'none';
this._thumbnailRemovalLinkNode.addEventListener("click", (e) =>
this._evtRemoveThumbnailClick(e)
);
this._thumbnailRemovalLinkNode.style.display = this._post
.hasCustomThumbnail
? "block"
: "none";
}
if (this._addNoteLinkNode) {
this._addNoteLinkNode.addEventListener(
'click', e => this._evtAddNoteClick(e));
this._addNoteLinkNode.addEventListener("click", (e) =>
this._evtAddNoteClick(e)
);
}
if (this._copyNotesLinkNode) {
this._copyNotesLinkNode.addEventListener(
'click', e => this._evtCopyNotesClick(e));
this._copyNotesLinkNode.addEventListener("click", (e) =>
this._evtCopyNotesClick(e)
);
}
if (this._pasteNotesLinkNode) {
this._pasteNotesLinkNode.addEventListener(
'click', e => this._evtPasteNotesClick(e));
this._pasteNotesLinkNode.addEventListener("click", (e) =>
this._evtPasteNotesClick(e)
);
}
if (this._deleteNoteLinkNode) {
this._deleteNoteLinkNode.addEventListener(
'click', e => this._evtDeleteNoteClick(e));
this._deleteNoteLinkNode.addEventListener("click", (e) =>
this._evtDeleteNoteClick(e)
);
}
if (this._featureLinkNode) {
this._featureLinkNode.addEventListener(
'click', e => this._evtFeatureClick(e));
this._featureLinkNode.addEventListener("click", (e) =>
this._evtFeatureClick(e)
);
}
if (this._mergeLinkNode) {
this._mergeLinkNode.addEventListener(
'click', e => this._evtMergeClick(e));
this._mergeLinkNode.addEventListener("click", (e) =>
this._evtMergeClick(e)
);
}
if (this._deleteLinkNode) {
this._deleteLinkNode.addEventListener(
'click', e => this._evtDeleteClick(e));
this._deleteLinkNode.addEventListener("click", (e) =>
this._evtDeleteClick(e)
);
}
this._postNotesOverlayControl.addEventListener(
'blur', e => this._evtNoteBlur(e));
this._postNotesOverlayControl.addEventListener("blur", (e) =>
this._evtNoteBlur(e)
);
this._postNotesOverlayControl.addEventListener(
'focus', e => this._evtNoteFocus(e));
this._postNotesOverlayControl.addEventListener("focus", (e) =>
this._evtNoteFocus(e)
);
this._post.addEventListener(
'changeContent', e => this._evtPostContentChange(e));
this._post.addEventListener("changeContent", (e) =>
this._evtPostContentChange(e)
);
this._post.addEventListener(
'changeThumbnail', e => this._evtPostThumbnailChange(e));
this._post.addEventListener("changeThumbnail", (e) =>
this._evtPostThumbnailChange(e)
);
if (this._formNode) {
const inputNodes = this._formNode.querySelectorAll(
'input, textarea');
"input, textarea"
);
for (let node of inputNodes) {
node.addEventListener(
'change',
e => this.dispatchEvent(new CustomEvent('change')));
node.addEventListener("change", (e) =>
this.dispatchEvent(new CustomEvent("change"))
);
}
this._postNotesOverlayControl.addEventListener(
'change',
e => this.dispatchEvent(new CustomEvent('change')));
this._postNotesOverlayControl.addEventListener("change", (e) =>
this.dispatchEvent(new CustomEvent("change"))
);
}
for (let eventType of ['add', 'remove']) {
this._post.notes.addEventListener(eventType, e => {
for (let eventType of ["add", "remove"]) {
this._post.notes.addEventListener(eventType, (e) => {
this._syncExpanderTitles();
});
this._post.pools.addEventListener(eventType, e => {
this._post.pools.addEventListener(eventType, (e) => {
this._syncExpanderTitles();
});
}
this._tagControl.addEventListener(
'change', e => {
this.dispatchEvent(new CustomEvent('change'));
this._tagControl.addEventListener("change", (e) => {
this.dispatchEvent(new CustomEvent("change"));
this._syncExpanderTitles();
});
if (this._noteTextareaNode) {
this._noteTextareaNode.addEventListener(
'change', e => this._evtNoteTextChangeRequest(e));
this._noteTextareaNode.addEventListener("change", (e) =>
this._evtNoteTextChangeRequest(e)
);
}
this._poolControl.addEventListener(
'change', e => {
this.dispatchEvent(new CustomEvent('change'));
this._poolControl.addEventListener("change", (e) => {
this.dispatchEvent(new CustomEvent("change"));
this._syncExpanderTitles();
});
}
@ -220,37 +260,43 @@ class PostEditSidebarControl extends events.EventTarget {
e.preventDefault();
this._thumbnailFileDropper.reset();
this._newPostThumbnail = null;
this._thumbnailRemovalLinkNode.style.display = 'none';
this._thumbnailRemovalLinkNode.style.display = "none";
}
_evtFeatureClick(e) {
e.preventDefault();
if (confirm('Are you sure you want to feature this post?')) {
this.dispatchEvent(new CustomEvent('feature', {
if (confirm("Are you sure you want to feature this post?")) {
this.dispatchEvent(
new CustomEvent("feature", {
detail: {
post: this._post,
},
}));
})
);
}
}
_evtMergeClick(e) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('merge', {
this.dispatchEvent(
new CustomEvent("merge", {
detail: {
post: this._post,
},
}));
})
);
}
_evtDeleteClick(e) {
e.preventDefault();
if (confirm('Are you sure you want to delete this post?')) {
this.dispatchEvent(new CustomEvent('delete', {
if (confirm("Are you sure you want to delete this post?")) {
this.dispatchEvent(
new CustomEvent("delete", {
detail: {
post: this._post,
},
}));
})
);
}
}
@ -262,60 +308,64 @@ class PostEditSidebarControl extends events.EventTarget {
_evtNoteFocus(e) {
this._editedNote = e.detail.note;
this._addNoteLinkNode.classList.remove('inactive');
this._deleteNoteLinkNode.classList.remove('inactive');
this._noteTextareaNode.removeAttribute('disabled');
this._addNoteLinkNode.classList.remove("inactive");
this._deleteNoteLinkNode.classList.remove("inactive");
this._noteTextareaNode.removeAttribute("disabled");
this._noteTextareaNode.value = e.detail.note.text;
}
_evtNoteBlur(e) {
this._evtNoteTextChangeRequest(null);
this._addNoteLinkNode.classList.remove('inactive');
this._deleteNoteLinkNode.classList.add('inactive');
this._addNoteLinkNode.classList.remove("inactive");
this._deleteNoteLinkNode.classList.add("inactive");
this._noteTextareaNode.blur();
this._noteTextareaNode.setAttribute('disabled', 'disabled');
this._noteTextareaNode.value = '';
this._noteTextareaNode.setAttribute("disabled", "disabled");
this._noteTextareaNode.value = "";
}
_evtAddNoteClick(e) {
e.preventDefault();
if (e.target.classList.contains('inactive')) {
if (e.target.classList.contains("inactive")) {
return;
}
this._addNoteLinkNode.classList.add('inactive');
this._addNoteLinkNode.classList.add("inactive");
this._postNotesOverlayControl.switchToDrawing();
}
_evtCopyNotesClick(e) {
e.preventDefault();
let textarea = document.createElement('textarea');
textarea.style.position = 'fixed';
textarea.style.opacity = '0';
textarea.value = JSON.stringify([...this._post.notes].map(note => ({
polygon: [...note.polygon].map(
point => [point.x, point.y]),
let textarea = document.createElement("textarea");
textarea.style.position = "fixed";
textarea.style.opacity = "0";
textarea.value = JSON.stringify(
[...this._post.notes].map((note) => ({
polygon: [...note.polygon].map((point) => [point.x, point.y]),
text: note.text,
})));
}))
);
document.body.appendChild(textarea);
textarea.select();
let success = false;
try {
success = document.execCommand('copy');
success = document.execCommand("copy");
} catch (err) {
// continue regardless of error
}
textarea.blur();
document.body.removeChild(textarea);
alert(success
? 'Notes copied to clipboard.'
: 'Failed to copy the text to clipboard. Sorry.');
alert(
success
? "Notes copied to clipboard."
: "Failed to copy the text to clipboard. Sorry."
);
}
_evtPasteNotesClick(e) {
e.preventDefault();
const text = window.prompt(
'Please enter the exported notes snapshot:');
"Please enter the exported notes snapshot:"
);
if (!text) {
return;
}
@ -333,7 +383,7 @@ class PostEditSidebarControl extends events.EventTarget {
_evtDeleteNoteClick(e) {
e.preventDefault();
if (e.target.classList.contains('inactive')) {
if (e.target.classList.contains("inactive")) {
return;
}
this._post.notes.remove(this._editedNote);
@ -342,72 +392,78 @@ class PostEditSidebarControl extends events.EventTarget {
_evtSubmit(e) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', {
this.dispatchEvent(
new CustomEvent("submit", {
detail: {
post: this._post,
safety: this._safetyButtonNodes.length ?
Array.from(this._safetyButtonNodes)
.filter(node => node.checked)[0]
.value.toLowerCase() :
undefined,
safety: this._safetyButtonNodes.length
? Array.from(this._safetyButtonNodes)
.filter((node) => node.checked)[0]
.value.toLowerCase()
: undefined,
flags: this._videoFlags,
tags: this._tagInputNode ?
misc.splitByWhitespace(this._tagInputNode.value) :
undefined,
tags: this._tagInputNode
? misc.splitByWhitespace(this._tagInputNode.value)
: undefined,
pools: this._poolInputNode ?
misc.splitByWhitespace(this._poolInputNode.value) :
undefined,
pools: this._poolInputNode
? misc.splitByWhitespace(this._poolInputNode.value)
: undefined,
relations: this._relationsInputNode ?
misc.splitByWhitespace(this._relationsInputNode.value)
.map(x => parseInt(x)) :
undefined,
relations: this._relationsInputNode
? misc
.splitByWhitespace(
this._relationsInputNode.value
)
.map((x) => parseInt(x))
: undefined,
content: this._newPostContent ?
this._newPostContent :
undefined,
content: this._newPostContent
? this._newPostContent
: undefined,
thumbnail: this._newPostThumbnail !== undefined ?
this._newPostThumbnail :
undefined,
thumbnail:
this._newPostThumbnail !== undefined
? this._newPostThumbnail
: undefined,
source: this._sourceInputNode ?
this._sourceInputNode.value :
undefined,
source: this._sourceInputNode
? this._sourceInputNode.value
: undefined,
},
}));
})
);
}
get _formNode() {
return this._hostNode.querySelector('form');
return this._hostNode.querySelector("form");
}
get _submitButtonNode() {
return this._hostNode.querySelector('.submit');
return this._hostNode.querySelector(".submit");
}
get _safetyButtonNodes() {
return this._formNode.querySelectorAll('.safety input');
return this._formNode.querySelectorAll(".safety input");
}
get _tagInputNode() {
return this._formNode.querySelector('.tags input');
return this._formNode.querySelector(".tags input");
}
get _poolInputNode() {
return this._formNode.querySelector('.pools input');
return this._formNode.querySelector(".pools input");
}
get _loopVideoInputNode() {
return this._formNode.querySelector('.flags input[name=loop]');
return this._formNode.querySelector(".flags input[name=loop]");
}
get _soundVideoInputNode() {
return this._formNode.querySelector('.flags input[name=sound]');
return this._formNode.querySelector(".flags input[name=sound]");
}
get _videoFlags() {
@ -416,65 +472,68 @@ class PostEditSidebarControl extends events.EventTarget {
}
let ret = [];
if (this._loopVideoInputNode.checked) {
ret.push('loop');
ret.push("loop");
}
if (this._soundVideoInputNode.checked) {
ret.push('sound');
ret.push("sound");
}
return ret;
}
get _relationsInputNode() {
return this._formNode.querySelector('.relations input');
return this._formNode.querySelector(".relations input");
}
get _contentInputNode() {
return this._formNode.querySelector('.post-content .dropper-container');
return this._formNode.querySelector(
".post-content .dropper-container"
);
}
get _thumbnailInputNode() {
return this._formNode.querySelector(
'.post-thumbnail .dropper-container');
".post-thumbnail .dropper-container"
);
}
get _thumbnailRemovalLinkNode() {
return this._formNode.querySelector('.post-thumbnail a');
return this._formNode.querySelector(".post-thumbnail a");
}
get _sourceInputNode() {
return this._formNode.querySelector('.post-source textarea');
return this._formNode.querySelector(".post-source textarea");
}
get _featureLinkNode() {
return this._formNode.querySelector('.management .feature');
return this._formNode.querySelector(".management .feature");
}
get _mergeLinkNode() {
return this._formNode.querySelector('.management .merge');
return this._formNode.querySelector(".management .merge");
}
get _deleteLinkNode() {
return this._formNode.querySelector('.management .delete');
return this._formNode.querySelector(".management .delete");
}
get _addNoteLinkNode() {
return this._formNode.querySelector('.notes .add');
return this._formNode.querySelector(".notes .add");
}
get _copyNotesLinkNode() {
return this._formNode.querySelector('.notes .copy');
return this._formNode.querySelector(".notes .copy");
}
get _pasteNotesLinkNode() {
return this._formNode.querySelector('.notes .paste');
return this._formNode.querySelector(".notes .paste");
}
get _deleteNoteLinkNode() {
return this._formNode.querySelector('.notes .delete');
return this._formNode.querySelector(".notes .delete");
}
get _noteTextareaNode() {
return this._formNode.querySelector('.notes textarea');
return this._formNode.querySelector(".notes textarea");
}
enableForm() {

View file

@ -1,13 +1,13 @@
'use strict';
"use strict";
const keyboard = require('../util/keyboard.js');
const views = require('../util/views.js');
const events = require('../events.js');
const misc = require('../util/misc.js');
const Note = require('../models/note.js');
const Point = require('../models/point.js');
const keyboard = require("../util/keyboard.js");
const views = require("../util/views.js");
const events = require("../events.js");
const misc = require("../util/misc.js");
const Note = require("../models/note.js");
const Point = require("../models/point.js");
const svgNS = 'http://www.w3.org/2000/svg';
const svgNS = "http://www.w3.org/2000/svg";
const snapThreshold = 10;
const circleSize = 10;
@ -22,19 +22,19 @@ const KEY_RETURN = 13;
function _getDistance(point1, point2) {
return Math.sqrt(
Math.pow(point1.x - point2.x, 2) +
Math.pow(point1.y - point2.y, 2));
Math.pow(point1.x - point2.x, 2) + Math.pow(point1.y - point2.y, 2)
);
}
function _setNodeState(node, stateName) {
if (node === null) {
return;
}
node.setAttribute('data-state', stateName);
node.setAttribute("data-state", stateName);
}
function _clearEditedNote(hostNode) {
const node = hostNode.querySelector('[data-state=\'editing\']');
const node = hostNode.querySelector("[data-state='editing']");
_setNodeState(node, null);
return node !== null;
}
@ -48,7 +48,7 @@ function _getNoteCentroid(note) {
const y0 = note.polygon.at(i).y;
const x1 = note.polygon.at((i + 1) % vertexCount).x;
const y1 = note.polygon.at((i + 1) % vertexCount).y;
const a = (x0 * y1) - (x1 * y0);
const a = x0 * y1 - x1 * y0;
signedArea += a;
centroid.x += (x0 + x1) * a;
centroid.y += (y0 + y1) * a;
@ -82,32 +82,30 @@ class State {
return false;
}
evtCanvasKeyDown(e) {
}
evtCanvasKeyDown(e) {}
evtNoteMouseDown(e, hoveredNote) {
}
evtNoteMouseDown(e, hoveredNote) {}
evtCanvasMouseDown(e) {
}
evtCanvasMouseDown(e) {}
evtCanvasMouseMove(e) {
}
evtCanvasMouseMove(e) {}
evtCanvasMouseUp(e) {
}
evtCanvasMouseUp(e) {}
_getScreenPoint(point) {
return new Point(
point.x * this._control.boundingBox.width,
point.y * this._control.boundingBox.height);
point.y * this._control.boundingBox.height
);
}
_snapPoints(targetPoint, referencePoint) {
const targetScreenPoint = this._getScreenPoint(targetPoint);
const referenceScreenPoint = this._getScreenPoint(referencePoint);
if (_getDistance(targetScreenPoint, referenceScreenPoint) <
snapThreshold) {
if (
_getDistance(targetScreenPoint, referenceScreenPoint) <
snapThreshold
) {
targetPoint.x = referencePoint.x;
targetPoint.y = referencePoint.y;
}
@ -124,15 +122,16 @@ class State {
(e.clientX - this._control.boundingBox.left) /
this._control.boundingBox.width,
(e.clientY - this._control.boundingBox.top) /
this._control.boundingBox.height);
this._control.boundingBox.height
);
}
}
class ReadOnlyState extends State {
constructor(control) {
super(control, 'read-only');
super(control, "read-only");
if (_clearEditedNote(control._hostNode)) {
this._control.dispatchEvent(new CustomEvent('blur'));
this._control.dispatchEvent(new CustomEvent("blur"));
}
keyboard.unpause();
}
@ -144,9 +143,9 @@ class ReadOnlyState extends State {
class PassiveState extends State {
constructor(control) {
super(control, 'passive');
super(control, "passive");
if (_clearEditedNote(control._hostNode)) {
this._control.dispatchEvent(new CustomEvent('blur'));
this._control.dispatchEvent(new CustomEvent("blur"));
}
keyboard.unpause();
}
@ -164,23 +163,24 @@ class ActiveState extends State {
constructor(control, note, stateName) {
super(control, stateName);
if (_clearEditedNote(control._hostNode)) {
this._control.dispatchEvent(new CustomEvent('blur'));
this._control.dispatchEvent(new CustomEvent("blur"));
}
keyboard.pause();
if (note !== null) {
this._note = note;
this._control.dispatchEvent(
new CustomEvent('focus', {
new CustomEvent("focus", {
detail: { note: note },
}));
_setNodeState(this._note.groupNode, 'editing');
})
);
_setNodeState(this._note.groupNode, "editing");
}
}
}
class SelectedState extends ActiveState {
constructor(control, note) {
super(control, note, 'selected');
super(control, note, "selected");
this._clickTimeout = null;
this._control._hideNoteText();
}
@ -211,27 +211,40 @@ class SelectedState extends ActiveState {
const mouseScreenPoint = this._getScreenPoint(mousePoint);
if (e.shiftKey) {
this._control._state = new ScalingNoteState(
this._control, this._note, mousePoint);
this._control,
this._note,
mousePoint
);
return;
}
if (this._note !== hoveredNote) {
this._control._state =
new SelectedState(this._control, hoveredNote);
this._control._state = new SelectedState(
this._control,
hoveredNote
);
return;
}
this._clickTimeout = window.setTimeout(() => {
for (let polygonPoint of this._note.polygon) {
const distance = _getDistance(
mouseScreenPoint,
this._getScreenPoint(polygonPoint));
this._getScreenPoint(polygonPoint)
);
if (distance < circleSize) {
this._control._state = new MovingPointState(
this._control, this._note, polygonPoint, mousePoint);
this._control,
this._note,
polygonPoint,
mousePoint
);
return;
}
}
this._control._state = new MovingNoteState(
this._control, this._note, mousePoint);
this._control,
this._note,
mousePoint
);
}, 100);
}
@ -241,9 +254,12 @@ class SelectedState extends ActiveState {
for (let polygonPoint of this._note.polygon) {
const distance = _getDistance(
mouseScreenPoint,
this._getScreenPoint(polygonPoint));
this._getScreenPoint(polygonPoint)
);
polygonPoint.edgeNode.classList.toggle(
'nearby', distance < circleSize);
"nearby",
distance < circleSize
);
}
}
@ -252,16 +268,24 @@ class SelectedState extends ActiveState {
const mouseScreenPoint = this._getScreenPoint(mousePoint);
if (e.shiftKey) {
this._control._state = new ScalingNoteState(
this._control, this._note, mousePoint);
this._control,
this._note,
mousePoint
);
return;
}
for (let polygonPoint of this._note.polygon) {
const distance = _getDistance(
mouseScreenPoint,
this._getScreenPoint(polygonPoint));
this._getScreenPoint(polygonPoint)
);
if (distance < circleSize) {
this._control._state = new MovingPointState(
this._control, this._note, polygonPoint, mousePoint);
this._control,
this._note,
polygonPoint,
mousePoint
);
return;
}
}
@ -283,32 +307,37 @@ class SelectedState extends ActiveState {
const origin = _getNoteCentroid(this._note);
const originalSize = _getNoteSize(this._note);
const targetSize = new Point(
originalSize.x + (x / this._control.boundingBox.width),
originalSize.y + (y / this._control.boundingBox.height));
originalSize.x + x / this._control.boundingBox.width,
originalSize.y + y / this._control.boundingBox.height
);
const scale = new Point(
targetSize.x / originalSize.x,
targetSize.y / originalSize.y);
targetSize.y / originalSize.y
);
for (let point of this._note.polygon) {
point.x = origin.x + ((point.x - origin.x) * scale.x);
point.y = origin.y + ((point.y - origin.y) * scale.y);
point.x = origin.x + (point.x - origin.x) * scale.x;
point.y = origin.y + (point.y - origin.y) * scale.y;
}
}
}
class MovingPointState extends ActiveState {
constructor(control, note, notePoint, mousePoint) {
super(control, note, 'moving-point');
super(control, note, "moving-point");
this._notePoint = notePoint;
this._originalNotePoint = { x: notePoint.x, y: notePoint.y };
this._originalPosition = mousePoint;
_setNodeState(this._note.groupNode, 'editing');
_setNodeState(this._note.groupNode, "editing");
}
evtCanvasKeyDown(e) {
if (e.which === KEY_ESCAPE) {
this._notePoint.x = this._originalNotePoint.x;
this._notePoint.y = this._originalNotePoint.y;
this._control._state = new SelectedState(this._control, this._note);
this._control._state = new SelectedState(
this._control,
this._note
);
}
}
@ -326,9 +355,11 @@ class MovingPointState extends ActiveState {
class MovingNoteState extends ActiveState {
constructor(control, note, mousePoint) {
super(control, note, 'moving-note');
this._originalPolygon = [...note.polygon].map(
point => ({x: point.x, y: point.y}));
super(control, note, "moving-note");
this._originalPolygon = [...note.polygon].map((point) => ({
x: point.x,
y: point.y,
}));
this._originalPosition = mousePoint;
}
@ -338,7 +369,10 @@ class MovingNoteState extends ActiveState {
this._note.polygon.at(i).x = this._originalPolygon[i].x;
this._note.polygon.at(i).y = this._originalPolygon[i].y;
}
this._control._state = new SelectedState(this._control, this._note);
this._control._state = new SelectedState(
this._control,
this._note
);
}
}
@ -358,9 +392,11 @@ class MovingNoteState extends ActiveState {
class ScalingNoteState extends ActiveState {
constructor(control, note, mousePoint) {
super(control, note, 'scaling-note');
this._originalPolygon = [...note.polygon].map(
point => ({x: point.x, y: point.y}));
super(control, note, "scaling-note");
this._originalPolygon = [...note.polygon].map((point) => ({
x: point.x,
y: point.y,
}));
this._originalMousePoint = mousePoint;
this._originalSize = _getNoteSize(note);
}
@ -371,7 +407,10 @@ class ScalingNoteState extends ActiveState {
this._note.polygon.at(i).x = this._originalPolygon[i].x;
this._note.polygon.at(i).y = this._originalPolygon[i].y;
}
this._control._state = new SelectedState(this._control, this._note);
this._control._state = new SelectedState(
this._control,
this._note
);
}
}
@ -384,12 +423,16 @@ class ScalingNoteState extends ActiveState {
const originalPolygonPoint = this._originalPolygon[i];
polygonPoint.x =
originalMousePoint.x +
((originalPolygonPoint.x - originalMousePoint.x) *
(1 + ((mousePoint.x - originalMousePoint.x) / originalSize.x)));
(originalPolygonPoint.x - originalMousePoint.x) *
(1 +
(mousePoint.x - originalMousePoint.x) /
originalSize.x);
polygonPoint.y =
originalMousePoint.y +
((originalPolygonPoint.y - originalMousePoint.y) *
(1 + ((mousePoint.y - originalMousePoint.y) / originalSize.y)));
(originalPolygonPoint.y - originalMousePoint.y) *
(1 +
(mousePoint.y - originalMousePoint.y) /
originalSize.y);
}
}
@ -400,7 +443,7 @@ class ScalingNoteState extends ActiveState {
class ReadyToDrawState extends ActiveState {
constructor(control) {
super(control, null, 'ready-to-draw');
super(control, null, "ready-to-draw");
}
evtNoteMouseDown(e, hoveredNote) {
@ -411,23 +454,27 @@ class ReadyToDrawState extends ActiveState {
const mousePoint = this._getPointFromEvent(e);
if (e.shiftKey) {
this._control._state = new DrawingRectangleState(
this._control, mousePoint);
this._control,
mousePoint
);
} else {
this._control._state = new DrawingPolygonState(
this._control, mousePoint);
this._control,
mousePoint
);
}
}
}
class DrawingRectangleState extends ActiveState {
constructor(control, mousePoint) {
super(control, null, 'drawing-rectangle');
super(control, null, "drawing-rectangle");
this._note = this._createNote();
this._note.polygon.add(new Point(mousePoint.x, mousePoint.y));
this._note.polygon.add(new Point(mousePoint.x, mousePoint.y));
this._note.polygon.add(new Point(mousePoint.x, mousePoint.y));
this._note.polygon.add(new Point(mousePoint.x, mousePoint.y));
_setNodeState(this._note.groupNode, 'drawing');
_setNodeState(this._note.groupNode, "drawing");
}
evtCanvasMouseUp(e) {
@ -443,7 +490,10 @@ class DrawingRectangleState extends ActiveState {
this._control._state = new ReadyToDrawState(this._control);
} else {
this._control._post.notes.add(this._note);
this._control._state = new SelectedState(this._control, this._note);
this._control._state = new SelectedState(
this._control,
this._note
);
}
}
@ -458,11 +508,11 @@ class DrawingRectangleState extends ActiveState {
class DrawingPolygonState extends ActiveState {
constructor(control, mousePoint) {
super(control, null, 'drawing-polygon');
super(control, null, "drawing-polygon");
this._note = this._createNote();
this._note.polygon.add(new Point(mousePoint.x, mousePoint.y));
this._note.polygon.add(new Point(mousePoint.x, mousePoint.y));
_setNodeState(this._note.groupNode, 'drawing');
_setNodeState(this._note.groupNode, "drawing");
}
evtCanvasKeyDown(e) {
@ -502,11 +552,16 @@ class DrawingPolygonState extends ActiveState {
}
if (e.shiftKey && secondLastPoint) {
const direction = (Math.round(
const direction =
(Math.round(
Math.atan2(
secondLastPoint.y - mousePoint.y,
secondLastPoint.x - mousePoint.x) /
(2 * Math.PI / 4)) + 4) % 4;
secondLastPoint.x - mousePoint.x
) /
((2 * Math.PI) / 4)
) +
4) %
4;
if (direction === 0 || direction === 2) {
lastPoint.x = mousePoint.x;
lastPoint.y = secondLastPoint.y;
@ -533,7 +588,10 @@ class DrawingPolygonState extends ActiveState {
} else {
this._control._deleteDomNode(this._note);
this._control._post.notes.add(this._note);
this._control._state = new SelectedState(this._control, this._note);
this._control._state = new SelectedState(
this._control,
this._note
);
}
}
}
@ -544,45 +602,48 @@ class PostNotesOverlayControl extends events.EventTarget {
this._post = post;
this._hostNode = hostNode;
this._svgNode = document.createElementNS(svgNS, 'svg');
this._svgNode.classList.add('resize-listener');
this._svgNode.classList.add('notes-overlay');
this._svgNode.setAttribute('preserveAspectRatio', 'none');
this._svgNode.setAttribute('viewBox', '0 0 1 1');
this._svgNode = document.createElementNS(svgNS, "svg");
this._svgNode.classList.add("resize-listener");
this._svgNode.classList.add("notes-overlay");
this._svgNode.setAttribute("preserveAspectRatio", "none");
this._svgNode.setAttribute("viewBox", "0 0 1 1");
for (let note of this._post.notes) {
this._createPolygonNode(note);
}
this._hostNode.appendChild(this._svgNode);
this._post.addEventListener('change', e => this._evtPostChange(e));
this._post.notes.addEventListener('remove', e => {
this._post.addEventListener("change", (e) => this._evtPostChange(e));
this._post.notes.addEventListener("remove", (e) => {
this._deleteDomNode(e.detail.note);
});
this._post.notes.addEventListener('add', e => {
this._post.notes.addEventListener("add", (e) => {
this._createPolygonNode(e.detail.note);
});
const keyHandler = e => this._evtCanvasKeyDown(e);
document.addEventListener('keydown', keyHandler);
this._svgNode.addEventListener(
'mousedown', e => this._evtCanvasMouseDown(e));
this._svgNode.addEventListener(
'mouseup', e => this._evtCanvasMouseUp(e));
this._svgNode.addEventListener(
'mousemove', e => this._evtCanvasMouseMove(e));
const keyHandler = (e) => this._evtCanvasKeyDown(e);
document.addEventListener("keydown", keyHandler);
this._svgNode.addEventListener("mousedown", (e) =>
this._evtCanvasMouseDown(e)
);
this._svgNode.addEventListener("mouseup", (e) =>
this._evtCanvasMouseUp(e)
);
this._svgNode.addEventListener("mousemove", (e) =>
this._evtCanvasMouseMove(e)
);
const wrapperNode = document.createElement('div');
wrapperNode.classList.add('wrapper');
this._textNode = document.createElement('div');
this._textNode.classList.add('note-text');
const wrapperNode = document.createElement("div");
wrapperNode.classList.add("wrapper");
this._textNode = document.createElement("div");
this._textNode.classList.add("note-text");
this._textNode.appendChild(wrapperNode);
this._textNode.addEventListener(
'mouseleave', e => this._evtNoteMouseLeave(e));
this._textNode.addEventListener("mouseleave", (e) =>
this._evtNoteMouseLeave(e)
);
document.body.appendChild(this._textNode);
views.monitorNodeRemoval(
this._hostNode, () => {
views.monitorNodeRemoval(this._hostNode, () => {
this._hostNode.removeChild(this._svgNode);
document.removeEventListener('keydown', keyHandler);
document.removeEventListener("keydown", keyHandler);
document.body.removeChild(this._textNode);
this._state = new ReadOnlyState(this);
});
@ -613,7 +674,7 @@ class PostNotesOverlayControl extends events.EventTarget {
}
_evtCanvasKeyDown(e) {
const illegalNodeNames = ['textarea', 'input', 'select'];
const illegalNodeNames = ["textarea", "input", "select"];
if (illegalNodeNames.includes(e.target.nodeName.toLowerCase())) {
return;
}
@ -655,53 +716,58 @@ class PostNotesOverlayControl extends events.EventTarget {
_evtNoteMouseLeave(e) {
const newElement = e.relatedTarget;
if (newElement === this._svgNode ||
if (
newElement === this._svgNode ||
(!this._svgNode.contains(newElement) &&
!this._textNode.contains(newElement) &&
newElement !== this._textNode)) {
newElement !== this._textNode)
) {
this._hideNoteText();
}
}
_showNoteText(note) {
this._textNode.querySelector('.wrapper').innerHTML =
misc.formatMarkdown(note.text);
this._textNode.style.display = 'block';
this._textNode.querySelector(
".wrapper"
).innerHTML = misc.formatMarkdown(note.text);
this._textNode.style.display = "block";
const bodyRect = document.body.getBoundingClientRect();
const noteRect = this._textNode.getBoundingClientRect();
const svgRect = this.boundingBox;
const centroid = _getNoteCentroid(note);
const x = (
const x =
-bodyRect.left +
svgRect.left +
(svgRect.width * centroid.x) -
(noteRect.width / 2));
const y = (
svgRect.width * centroid.x -
noteRect.width / 2;
const y =
-bodyRect.top +
svgRect.top +
(svgRect.height * centroid.y) -
(noteRect.height / 2));
this._textNode.style.left = x + 'px';
this._textNode.style.top = y + 'px';
svgRect.height * centroid.y -
noteRect.height / 2;
this._textNode.style.left = x + "px";
this._textNode.style.top = y + "px";
}
_hideNoteText() {
this._textNode.style.display = 'none';
this._textNode.style.display = "none";
}
_updatePolygonNotePoints(note) {
note.polygonNode.setAttribute(
'points',
[...note.polygon].map(
point => [point.x, point.y].join(',')).join(' '));
"points",
[...note.polygon]
.map((point) => [point.x, point.y].join(","))
.join(" ")
);
}
_createEdgeNode(point, groupNode) {
const node = document.createElementNS(svgNS, 'ellipse');
node.setAttribute('cx', point.x);
node.setAttribute('cy', point.y);
node.setAttribute('rx', circleSize / 2 / this.boundingBox.width);
node.setAttribute('ry', circleSize / 2 / this.boundingBox.height);
const node = document.createElementNS(svgNS, "ellipse");
node.setAttribute("cx", point.x);
node.setAttribute("cy", point.y);
node.setAttribute("rx", circleSize / 2 / this.boundingBox.width);
node.setAttribute("ry", circleSize / 2 / this.boundingBox.height);
point.edgeNode = node;
groupNode.appendChild(node);
}
@ -713,8 +779,8 @@ class PostNotesOverlayControl extends events.EventTarget {
_updateEdgeNode(point, note) {
this._updatePolygonNotePoints(note);
point.edgeNode.setAttribute('cx', point.x);
point.edgeNode.setAttribute('cy', point.y);
point.edgeNode.setAttribute("cx", point.x);
point.edgeNode.setAttribute("cy", point.y);
}
_deleteDomNode(note) {
@ -722,17 +788,19 @@ class PostNotesOverlayControl extends events.EventTarget {
}
_createPolygonNode(note) {
const groupNode = document.createElementNS(svgNS, 'g');
const groupNode = document.createElementNS(svgNS, "g");
note.groupNode = groupNode;
{
const node = document.createElementNS(svgNS, 'polygon');
const node = document.createElementNS(svgNS, "polygon");
note.polygonNode = node;
node.setAttribute('vector-effect', 'non-scaling-stroke');
node.setAttribute('stroke-alignment', 'inside');
node.addEventListener(
'mouseenter', e => this._evtNoteMouseEnter(e, note));
node.addEventListener(
'mouseleave', e => this._evtNoteMouseLeave(e));
node.setAttribute("vector-effect", "non-scaling-stroke");
node.setAttribute("stroke-alignment", "inside");
node.addEventListener("mouseenter", (e) =>
this._evtNoteMouseEnter(e, note)
);
node.addEventListener("mouseleave", (e) =>
this._evtNoteMouseLeave(e)
);
this._updatePolygonNotePoints(note);
groupNode.appendChild(node);
}
@ -740,17 +808,17 @@ class PostNotesOverlayControl extends events.EventTarget {
this._createEdgeNode(point, groupNode);
}
note.polygon.addEventListener('change', e => {
note.polygon.addEventListener("change", (e) => {
this._updateEdgeNode(e.detail.point, note);
this.dispatchEvent(new CustomEvent('change'));
this.dispatchEvent(new CustomEvent("change"));
});
note.polygon.addEventListener('remove', e => {
note.polygon.addEventListener("remove", (e) => {
this._deleteEdgeNode(e.detail.point, note);
this.dispatchEvent(new CustomEvent('change'));
this.dispatchEvent(new CustomEvent("change"));
});
note.polygon.addEventListener('add', e => {
note.polygon.addEventListener("add", (e) => {
this._createEdgeNode(e.detail.point, groupNode);
this.dispatchEvent(new CustomEvent('change'));
this.dispatchEvent(new CustomEvent("change"));
});
this._svgNode.appendChild(groupNode);

View file

@ -1,14 +1,14 @@
'use strict';
"use strict";
const api = require('../api.js');
const events = require('../events.js');
const views = require('../util/views.js');
const uri = require('../util/uri.js');
const misc = require('../util/misc.js');
const api = require("../api.js");
const events = require("../events.js");
const views = require("../util/views.js");
const uri = require("../util/uri.js");
const misc = require("../util/misc.js");
const template = views.getTemplate('post-readonly-sidebar');
const scoreTemplate = views.getTemplate('score');
const favTemplate = views.getTemplate('fav');
const template = views.getTemplate("post-readonly-sidebar");
const scoreTemplate = views.getTemplate("score");
const favTemplate = views.getTemplate("fav");
class PostReadonlySidebarControl extends events.EventTarget {
constructor(hostNode, post, postContentControl) {
@ -17,19 +17,22 @@ class PostReadonlySidebarControl extends events.EventTarget {
this._post = post;
this._postContentControl = postContentControl;
post.addEventListener('changeFavorite', e => this._evtChangeFav(e));
post.addEventListener('changeScore', e => this._evtChangeScore(e));
post.addEventListener("changeFavorite", (e) => this._evtChangeFav(e));
post.addEventListener("changeScore", (e) => this._evtChangeScore(e));
views.replaceContent(this._hostNode, template({
views.replaceContent(
this._hostNode,
template({
post: this._post,
enableSafety: api.safetyEnabled(),
canListPosts: api.hasPrivilege('posts:list'),
canEditPosts: api.hasPrivilege('posts:edit'),
canViewTags: api.hasPrivilege('tags:view'),
canListPosts: api.hasPrivilege("posts:list"),
canEditPosts: api.hasPrivilege("posts:edit"),
canViewTags: api.hasPrivilege("tags:view"),
escapeColons: uri.escapeColons,
extractRootDomain: uri.extractRootDomain,
getPrettyTagName: misc.getPrettyTagName,
}));
})
);
this._installFav();
this._installScore();
@ -38,58 +41,62 @@ class PostReadonlySidebarControl extends events.EventTarget {
}
get _scoreContainerNode() {
return this._hostNode.querySelector('.score-container');
return this._hostNode.querySelector(".score-container");
}
get _favContainerNode() {
return this._hostNode.querySelector('.fav-container');
return this._hostNode.querySelector(".fav-container");
}
get _upvoteButtonNode() {
return this._hostNode.querySelector('.upvote');
return this._hostNode.querySelector(".upvote");
}
get _downvoteButtonNode() {
return this._hostNode.querySelector('.downvote');
return this._hostNode.querySelector(".downvote");
}
get _addFavButtonNode() {
return this._hostNode.querySelector('.add-favorite');
return this._hostNode.querySelector(".add-favorite");
}
get _remFavButtonNode() {
return this._hostNode.querySelector('.remove-favorite');
return this._hostNode.querySelector(".remove-favorite");
}
get _fitBothButtonNode() {
return this._hostNode.querySelector('.fit-both');
return this._hostNode.querySelector(".fit-both");
}
get _fitOriginalButtonNode() {
return this._hostNode.querySelector('.fit-original');
return this._hostNode.querySelector(".fit-original");
}
get _fitWidthButtonNode() {
return this._hostNode.querySelector('.fit-width');
return this._hostNode.querySelector(".fit-width");
}
get _fitHeightButtonNode() {
return this._hostNode.querySelector('.fit-height');
return this._hostNode.querySelector(".fit-height");
}
_installFitButtons() {
this._fitBothButtonNode.addEventListener(
'click', this._eventZoomProxy(
() => this._postContentControl.fitBoth()));
"click",
this._eventZoomProxy(() => this._postContentControl.fitBoth())
);
this._fitOriginalButtonNode.addEventListener(
'click', this._eventZoomProxy(
() => this._postContentControl.fitOriginal()));
"click",
this._eventZoomProxy(() => this._postContentControl.fitOriginal())
);
this._fitWidthButtonNode.addEventListener(
'click', this._eventZoomProxy(
() => this._postContentControl.fitWidth()));
"click",
this._eventZoomProxy(() => this._postContentControl.fitWidth())
);
this._fitHeightButtonNode.addEventListener(
'click', this._eventZoomProxy(
() => this._postContentControl.fitHeight()));
"click",
this._eventZoomProxy(() => this._postContentControl.fitHeight())
);
}
_installFav() {
@ -98,16 +105,19 @@ class PostReadonlySidebarControl extends events.EventTarget {
favTemplate({
favoriteCount: this._post.favoriteCount,
ownFavorite: this._post.ownFavorite,
canFavorite: api.hasPrivilege('posts:favorite'),
}));
canFavorite: api.hasPrivilege("posts:favorite"),
})
);
if (this._addFavButtonNode) {
this._addFavButtonNode.addEventListener(
'click', e => this._evtAddToFavoritesClick(e));
this._addFavButtonNode.addEventListener("click", (e) =>
this._evtAddToFavoritesClick(e)
);
}
if (this._remFavButtonNode) {
this._remFavButtonNode.addEventListener(
'click', e => this._evtRemoveFromFavoritesClick(e));
this._remFavButtonNode.addEventListener("click", (e) =>
this._evtRemoveFromFavoritesClick(e)
);
}
}
@ -117,77 +127,88 @@ class PostReadonlySidebarControl extends events.EventTarget {
scoreTemplate({
score: this._post.score,
ownScore: this._post.ownScore,
canScore: api.hasPrivilege('posts:score'),
}));
canScore: api.hasPrivilege("posts:score"),
})
);
if (this._upvoteButtonNode) {
this._upvoteButtonNode.addEventListener(
'click', e => this._evtScoreClick(e, 1));
this._upvoteButtonNode.addEventListener("click", (e) =>
this._evtScoreClick(e, 1)
);
}
if (this._downvoteButtonNode) {
this._downvoteButtonNode.addEventListener(
'click', e => this._evtScoreClick(e, -1));
this._downvoteButtonNode.addEventListener("click", (e) =>
this._evtScoreClick(e, -1)
);
}
}
_eventZoomProxy(func) {
return e => {
return (e) => {
e.preventDefault();
e.target.blur();
func();
this._syncFitButton();
this.dispatchEvent(new CustomEvent('fitModeChange', {
this.dispatchEvent(
new CustomEvent("fitModeChange", {
detail: {
mode: this._getFitMode(),
},
}));
})
);
};
}
_getFitMode() {
const funcToName = {};
funcToName[this._postContentControl.fitBoth] = 'fit-both';
funcToName[this._postContentControl.fitOriginal] = 'fit-original';
funcToName[this._postContentControl.fitWidth] = 'fit-width';
funcToName[this._postContentControl.fitHeight] = 'fit-height';
funcToName[this._postContentControl.fitBoth] = "fit-both";
funcToName[this._postContentControl.fitOriginal] = "fit-original";
funcToName[this._postContentControl.fitWidth] = "fit-width";
funcToName[this._postContentControl.fitHeight] = "fit-height";
return funcToName[this._postContentControl._currentFitFunction];
}
_syncFitButton() {
const className = this._getFitMode();
const oldNode = this._hostNode.querySelector('.zoom a.active');
const oldNode = this._hostNode.querySelector(".zoom a.active");
const newNode = this._hostNode.querySelector(`.zoom a.${className}`);
if (oldNode) {
oldNode.classList.remove('active');
oldNode.classList.remove("active");
}
newNode.classList.add('active');
newNode.classList.add("active");
}
_evtAddToFavoritesClick(e) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('favorite', {
this.dispatchEvent(
new CustomEvent("favorite", {
detail: {
post: this._post,
},
}));
})
);
}
_evtRemoveFromFavoritesClick(e) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('unfavorite', {
this.dispatchEvent(
new CustomEvent("unfavorite", {
detail: {
post: this._post,
},
}));
})
);
}
_evtScoreClick(e, score) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('score', {
this.dispatchEvent(
new CustomEvent("score", {
detail: {
post: this._post,
score: this._post.ownScore === score ? 0 : score,
},
}));
})
);
}
_evtChangeFav(e) {

View file

@ -1,22 +1,26 @@
'use strict';
"use strict";
const misc = require('../util/misc.js');
const views = require('../util/views.js');
const TagList = require('../models/tag_list.js');
const AutoCompleteControl = require('./auto_complete_control.js');
const misc = require("../util/misc.js");
const views = require("../util/views.js");
const TagList = require("../models/tag_list.js");
const AutoCompleteControl = require("./auto_complete_control.js");
function _tagListToMatches(tags, options) {
return [...tags].sort((tag1, tag2) => {
return [...tags]
.sort((tag1, tag2) => {
return tag2.usages - tag1.usages;
}).map(tag => {
let cssName = misc.makeCssName(tag.category, 'tag');
})
.map((tag) => {
let cssName = misc.makeCssName(tag.category, "tag");
if (options.isTaggedWith(tag.names[0])) {
cssName += ' disabled';
cssName += " disabled";
}
const caption = (
'<span class="' + cssName + '">'
+ misc.escapeHtml(tag.names[0] + ' (' + tag.postCount + ')')
+ '</span>');
const caption =
'<span class="' +
cssName +
'">' +
misc.escapeHtml(tag.names[0] + " (" + tag.postCount + ")") +
"</span>";
return {
caption: caption,
value: tag,
@ -28,24 +32,32 @@ class TagAutoCompleteControl extends AutoCompleteControl {
constructor(input, options) {
const minLengthForPartialSearch = 3;
options = Object.assign({
isTaggedWith: tag => false,
}, options);
options = Object.assign(
{
isTaggedWith: (tag) => false,
},
options
);
options.getMatches = text => {
options.getMatches = (text) => {
const term = misc.escapeSearchTerm(text);
const query = (
text.length < minLengthForPartialSearch
? term + '*'
: '*' + term + '*') + ' sort:usages';
const query =
(text.length < minLengthForPartialSearch
? term + "*"
: "*" + term + "*") + " sort:usages";
return new Promise((resolve, reject) => {
TagList.search(
query, 0, this._options.maxResults, ['names', 'category', 'usages'])
.then(
response => resolve(
_tagListToMatches(response.results, this._options)),
reject);
TagList.search(query, 0, this._options.maxResults, [
"names",
"category",
"usages",
]).then(
(response) =>
resolve(
_tagListToMatches(response.results, this._options)
),
reject
);
});
};

View file

@ -1,25 +1,25 @@
'use strict';
"use strict";
const api = require('../api.js');
const tags = require('../tags.js');
const misc = require('../util/misc.js');
const uri = require('../util/uri.js');
const Tag = require('../models/tag.js');
const settings = require('../models/settings.js');
const events = require('../events.js');
const views = require('../util/views.js');
const TagAutoCompleteControl = require('./tag_auto_complete_control.js');
const api = require("../api.js");
const tags = require("../tags.js");
const misc = require("../util/misc.js");
const uri = require("../util/uri.js");
const Tag = require("../models/tag.js");
const settings = require("../models/settings.js");
const events = require("../events.js");
const views = require("../util/views.js");
const TagAutoCompleteControl = require("./tag_auto_complete_control.js");
const KEY_SPACE = 32;
const KEY_RETURN = 13;
const SOURCE_INIT = 'init';
const SOURCE_IMPLICATION = 'implication';
const SOURCE_USER_INPUT = 'user-input';
const SOURCE_SUGGESTION = 'suggestions';
const SOURCE_CLIPBOARD = 'clipboard';
const SOURCE_INIT = "init";
const SOURCE_IMPLICATION = "implication";
const SOURCE_USER_INPUT = "user-input";
const SOURCE_SUGGESTION = "suggestions";
const SOURCE_CLIPBOARD = "clipboard";
const template = views.getTemplate('tag-input');
const template = views.getTemplate("tag-input");
function _fadeOutListItemNodeStatus(listItemNode) {
if (listItemNode.classList.length) {
@ -28,8 +28,7 @@ function _fadeOutListItemNodeStatus(listItemNode) {
}
listItemNode.fadeTimeout = window.setTimeout(() => {
while (listItemNode.classList.length) {
listItemNode.classList.remove(
listItemNode.classList.item(0));
listItemNode.classList.remove(listItemNode.classList.item(0));
}
listItemNode.fadeTimeout = null;
}, 2500);
@ -51,7 +50,9 @@ class SuggestionList {
}
set(suggestion, weight) {
if (Object.prototype.hasOwnProperty.call(this._suggestions, suggestion)) {
if (
Object.prototype.hasOwnProperty.call(this._suggestions, suggestion)
) {
weight = Math.max(weight, this._suggestions[suggestion]);
}
this._suggestions[suggestion] = weight;
@ -74,7 +75,7 @@ class SuggestionList {
let nameDiff = a[0].localeCompare(b[0]);
return weightDiff === 0 ? nameDiff : weightDiff;
});
return tuples.map(tuple => {
return tuples.map((tuple) => {
return { tagName: tuple[0], weight: tuple[1] };
});
}
@ -91,45 +92,58 @@ class TagInputControl extends events.EventTarget {
// dom
const editAreaNode = template();
this._editAreaNode = editAreaNode;
this._tagInputNode = editAreaNode.querySelector('input');
this._suggestionsNode = editAreaNode.querySelector('.tag-suggestions');
this._tagListNode = editAreaNode.querySelector('ul.compact-tags');
this._tagInputNode = editAreaNode.querySelector("input");
this._suggestionsNode = editAreaNode.querySelector(".tag-suggestions");
this._tagListNode = editAreaNode.querySelector("ul.compact-tags");
this._autoCompleteControl = new TagAutoCompleteControl(
this._tagInputNode, {
this._tagInputNode,
{
getTextToFind: () => {
return this._tagInputNode.value;
},
confirm: tag => {
this._tagInputNode.value = '';
confirm: (tag) => {
this._tagInputNode.value = "";
// note: tags from autocomplete don't contain implications
// so they need to be looked up in API
this.addTagByName(tag.names[0], SOURCE_USER_INPUT);
},
delete: tag => {
this._tagInputNode.value = '';
delete: (tag) => {
this._tagInputNode.value = "";
this.deleteTag(tag);
},
verticalShift: -2,
isTaggedWith: tagName => this.tags.isTaggedWith(tagName),
});
isTaggedWith: (tagName) => this.tags.isTaggedWith(tagName),
}
);
// dom events
this._tagInputNode.addEventListener(
'keydown', e => this._evtInputKeyDown(e));
this._tagInputNode.addEventListener(
'paste', e => this._evtInputPaste(e));
this._editAreaNode.querySelector('a.opacity').addEventListener(
'click', e => this._evtToggleSuggestionsPopupOpacityClick(e));
this._editAreaNode.querySelector('a.close').addEventListener(
'click', e => this._evtCloseSuggestionsPopupClick(e));
this._editAreaNode.querySelector('button').addEventListener(
'click', e => this._evtAddTagButtonClick(e));
this._tagInputNode.addEventListener("keydown", (e) =>
this._evtInputKeyDown(e)
);
this._tagInputNode.addEventListener("paste", (e) =>
this._evtInputPaste(e)
);
this._editAreaNode
.querySelector("a.opacity")
.addEventListener("click", (e) =>
this._evtToggleSuggestionsPopupOpacityClick(e)
);
this._editAreaNode
.querySelector("a.close")
.addEventListener("click", (e) =>
this._evtCloseSuggestionsPopupClick(e)
);
this._editAreaNode
.querySelector("button")
.addEventListener("click", (e) => this._evtAddTagButtonClick(e));
// show
this._hostNode.style.display = 'none';
this._hostNode.style.display = "none";
this._hostNode.parentNode.insertBefore(
this._editAreaNode, hostNode.nextSibling);
this._editAreaNode,
hostNode.nextSibling
);
// add existing tags
for (let tag of [...this.tags]) {
@ -139,7 +153,10 @@ class TagInputControl extends events.EventTarget {
}
addTagByText(text, source) {
for (let tagName of text.split(/\s+/).filter(word => word).reverse()) {
for (let tagName of text
.split(/\s+/)
.filter((word) => word)
.reverse()) {
this.addTagByName(tagName, source);
}
}
@ -149,46 +166,58 @@ class TagInputControl extends events.EventTarget {
if (!name) {
return;
}
return Tag.get(name).then(tag => {
return Tag.get(name).then(
(tag) => {
return this.addTag(tag, source);
}, () => {
},
() => {
const tag = new Tag();
tag.names = [name];
tag.category = null;
return this.addTag(tag, source);
});
}
);
}
addTag(tag, source) {
if (source !== SOURCE_INIT && this.tags.isTaggedWith(tag.names[0])) {
const listItemNode = this._getListItemNode(tag);
if (source !== SOURCE_IMPLICATION) {
listItemNode.classList.add('duplicate');
listItemNode.classList.add("duplicate");
_fadeOutListItemNodeStatus(listItemNode);
}
return Promise.resolve();
}
return this.tags.addByName(tag.names[0], false).then(() => {
return this.tags
.addByName(tag.names[0], false)
.then(() => {
const listItemNode = this._createListItemNode(tag);
if (!tag.category) {
listItemNode.classList.add('new');
listItemNode.classList.add("new");
}
if (source === SOURCE_IMPLICATION) {
listItemNode.classList.add('implication');
listItemNode.classList.add("implication");
}
this._tagListNode.prependChild(listItemNode);
_fadeOutListItemNodeStatus(listItemNode);
return Promise.all(
tag.implications.map(
implication => this.addTagByName(
implication.names[0], SOURCE_IMPLICATION)));
}).then(() => {
this.dispatchEvent(new CustomEvent('add', {
tag.implications.map((implication) =>
this.addTagByName(
implication.names[0],
SOURCE_IMPLICATION
)
)
);
})
.then(() => {
this.dispatchEvent(
new CustomEvent("add", {
detail: { tag: tag, source: source },
}));
this.dispatchEvent(new CustomEvent('change'));
})
);
this.dispatchEvent(new CustomEvent("change"));
return Promise.resolve();
});
}
@ -202,25 +231,27 @@ class TagInputControl extends events.EventTarget {
this._deleteListItemNode(tag);
this.dispatchEvent(new CustomEvent('remove', {
this.dispatchEvent(
new CustomEvent("remove", {
detail: { tag: tag },
}));
this.dispatchEvent(new CustomEvent('change'));
})
);
this.dispatchEvent(new CustomEvent("change"));
}
_evtInputPaste(e) {
e.preventDefault();
const pastedText = window.clipboardData ?
window.clipboardData.getData('Text') :
(e.originalEvent || e).clipboardData.getData('text/plain');
const pastedText = window.clipboardData
? window.clipboardData.getData("Text")
: (e.originalEvent || e).clipboardData.getData("text/plain");
if (pastedText.length > 2000) {
window.alert('Pasted text is too long.');
window.alert("Pasted text is too long.");
return;
}
this._hideAutoComplete();
this.addTagByText(pastedText, SOURCE_CLIPBOARD);
this._tagInputNode.value = '';
this._tagInputNode.value = "";
}
_evtCloseSuggestionsPopupClick(e) {
@ -231,7 +262,7 @@ class TagInputControl extends events.EventTarget {
_evtAddTagButtonClick(e) {
e.preventDefault();
this.addTagByName(this._tagInputNode.value, SOURCE_USER_INPUT);
this._tagInputNode.value = '';
this._tagInputNode.value = "";
}
_evtToggleSuggestionsPopupOpacityClick(e) {
@ -244,36 +275,41 @@ class TagInputControl extends events.EventTarget {
e.preventDefault();
this._hideAutoComplete();
this.addTagByText(this._tagInputNode.value, SOURCE_USER_INPUT);
this._tagInputNode.value = '';
this._tagInputNode.value = "";
}
}
_createListItemNode(tag) {
const className = tag.category ?
misc.makeCssName(tag.category, 'tag') :
null;
const className = tag.category
? misc.makeCssName(tag.category, "tag")
: null;
const tagLinkNode = document.createElement('a');
const tagLinkNode = document.createElement("a");
if (className) {
tagLinkNode.classList.add(className);
}
tagLinkNode.setAttribute(
'href', uri.formatClientLink('tag', tag.names[0]));
"href",
uri.formatClientLink("tag", tag.names[0])
);
const tagIconNode = document.createElement('i');
tagIconNode.classList.add('fa');
tagIconNode.classList.add('fa-tag');
const tagIconNode = document.createElement("i");
tagIconNode.classList.add("fa");
tagIconNode.classList.add("fa-tag");
tagLinkNode.appendChild(tagIconNode);
const searchLinkNode = document.createElement('a');
const searchLinkNode = document.createElement("a");
if (className) {
searchLinkNode.classList.add(className);
}
searchLinkNode.setAttribute(
'href', uri.formatClientLink(
'posts', {query: uri.escapeColons(tag.names[0])}));
searchLinkNode.textContent = tag.names[0] + ' ';
searchLinkNode.addEventListener('click', e => {
"href",
uri.formatClientLink("posts", {
query: uri.escapeColons(tag.names[0]),
})
);
searchLinkNode.textContent = tag.names[0] + " ";
searchLinkNode.addEventListener("click", (e) => {
e.preventDefault();
this._suggestions.clear();
if (tag.postCount > 0) {
@ -284,20 +320,20 @@ class TagInputControl extends events.EventTarget {
}
});
const usagesNode = document.createElement('span');
usagesNode.classList.add('tag-usages');
usagesNode.setAttribute('data-pseudo-content', tag.postCount);
const usagesNode = document.createElement("span");
usagesNode.classList.add("tag-usages");
usagesNode.setAttribute("data-pseudo-content", tag.postCount);
const removalLinkNode = document.createElement('a');
removalLinkNode.classList.add('remove-tag');
removalLinkNode.setAttribute('href', '');
removalLinkNode.setAttribute('data-pseudo-content', '×');
removalLinkNode.addEventListener('click', e => {
const removalLinkNode = document.createElement("a");
removalLinkNode.classList.add("remove-tag");
removalLinkNode.setAttribute("href", "");
removalLinkNode.setAttribute("data-pseudo-content", "×");
removalLinkNode.addEventListener("click", (e) => {
e.preventDefault();
this.deleteTag(tag);
});
const listItemNode = document.createElement('li');
const listItemNode = document.createElement("li");
listItemNode.appendChild(removalLinkNode);
listItemNode.appendChild(tagLinkNode);
listItemNode.appendChild(searchLinkNode);
@ -327,20 +363,25 @@ class TagInputControl extends events.EventTarget {
if (!browsingSettings.tagSuggestions) {
return;
}
api.get(
uri.formatApiLink('tag-siblings', tag.names[0]),
{noProgress: true})
.then(response => {
api.get(uri.formatApiLink("tag-siblings", tag.names[0]), {
noProgress: true,
})
.then(
(response) => {
return Promise.resolve(response.results);
}, response => {
},
(response) => {
return Promise.resolve([]);
}).then(siblings => {
const args = siblings.map(s => s.occurrences);
}
)
.then((siblings) => {
const args = siblings.map((s) => s.occurrences);
let maxSiblingOccurrences = Math.max(1, ...args);
for (let sibling of siblings) {
this._suggestions.set(
sibling.tag.names[0],
sibling.occurrences * 4.9 / maxSiblingOccurrences);
(sibling.occurrences * 4.9) / maxSiblingOccurrences
);
}
for (let suggestion of tag.suggestions || []) {
this._suggestions.set(suggestion, 5);
@ -354,10 +395,10 @@ class TagInputControl extends events.EventTarget {
}
_refreshSuggestionsPopup() {
if (!this._suggestionsNode.classList.contains('shown')) {
if (!this._suggestionsNode.classList.contains("shown")) {
return;
}
const listNode = this._suggestionsNode.querySelector('ul');
const listNode = this._suggestionsNode.querySelector("ul");
listNode.scrollTop = 0;
while (listNode.firstChild) {
listNode.removeChild(listNode.firstChild);
@ -369,35 +410,36 @@ class TagInputControl extends events.EventTarget {
continue;
}
const addLinkNode = document.createElement('a');
const addLinkNode = document.createElement("a");
addLinkNode.textContent = tagName;
addLinkNode.classList.add('add-tag');
addLinkNode.setAttribute('href', '');
Tag.get(tagName).then(tag => {
addLinkNode.classList.add("add-tag");
addLinkNode.setAttribute("href", "");
Tag.get(tagName).then((tag) => {
addLinkNode.classList.add(
misc.makeCssName(tag.category, 'tag'));
misc.makeCssName(tag.category, "tag")
);
});
addLinkNode.addEventListener('click', e => {
addLinkNode.addEventListener("click", (e) => {
e.preventDefault();
listNode.removeChild(listItemNode);
this.addTagByName(tagName, SOURCE_SUGGESTION);
});
const weightNode = document.createElement('span');
weightNode.classList.add('tag-weight');
weightNode.setAttribute('data-pseudo-content', weight);
const weightNode = document.createElement("span");
weightNode.classList.add("tag-weight");
weightNode.setAttribute("data-pseudo-content", weight);
const removalLinkNode = document.createElement('a');
removalLinkNode.classList.add('remove-tag');
removalLinkNode.setAttribute('href', '');
removalLinkNode.setAttribute('data-pseudo-content', '×');
removalLinkNode.addEventListener('click', e => {
const removalLinkNode = document.createElement("a");
removalLinkNode.classList.add("remove-tag");
removalLinkNode.setAttribute("href", "");
removalLinkNode.setAttribute("data-pseudo-content", "×");
removalLinkNode.addEventListener("click", (e) => {
e.preventDefault();
listNode.removeChild(listItemNode);
this._suggestions.ban(tagName);
});
const listItemNode = document.createElement('li');
const listItemNode = document.createElement("li");
listItemNode.appendChild(removalLinkNode);
listItemNode.appendChild(weightNode);
listItemNode.appendChild(addLinkNode);
@ -407,19 +449,19 @@ class TagInputControl extends events.EventTarget {
_closeSuggestionsPopup() {
this._suggestions.clear();
this._suggestionsNode.classList.remove('shown');
this._suggestionsNode.classList.remove("shown");
}
_removeSuggestionsPopupOpacity() {
this._suggestionsNode.classList.remove('translucent');
this._suggestionsNode.classList.remove("translucent");
}
_toggleSuggestionsPopupOpacity() {
this._suggestionsNode.classList.toggle('translucent');
this._suggestionsNode.classList.toggle("translucent");
}
_openSuggestionsPopup() {
this._suggestionsNode.classList.add('shown');
this._suggestionsNode.classList.add("shown");
this._refreshSuggestionsPopup();
}

View file

@ -1,12 +1,12 @@
'use strict';
"use strict";
class EventTarget {
constructor() {
this.eventTarget = document.createDocumentFragment();
for (let method of [
'addEventListener',
'dispatchEvent',
'removeEventListener'
"addEventListener",
"dispatchEvent",
"removeEventListener",
]) {
this[method] = this.eventTarget[method].bind(this.eventTarget);
}
@ -20,17 +20,19 @@ function proxyEvent(source, target, sourceEventType, targetEventType) {
if (!targetEventType) {
targetEventType = sourceEventType;
}
source.addEventListener(sourceEventType, e => {
target.dispatchEvent(new CustomEvent(targetEventType, {
source.addEventListener(sourceEventType, (e) => {
target.dispatchEvent(
new CustomEvent(targetEventType, {
detail: e.detail,
}));
})
);
});
}
module.exports = {
Success: 'success',
Error: 'error',
Info: 'info',
Success: "success",
Error: "error",
Info: "info",
proxyEvent: proxyEvent,
EventTarget: EventTarget,

View file

@ -1,15 +1,13 @@
'use strict';
"use strict";
require('./util/polyfill.js');
const misc = require('./util/misc.js');
const views = require('./util/views.js');
const router = require('./router.js');
require("./util/polyfill.js");
const misc = require("./util/misc.js");
const views = require("./util/views.js");
const router = require("./router.js");
history.scrollRestoration = 'manual';
history.scrollRestoration = "manual";
router.exit(
null,
(ctx, next) => {
router.exit(null, (ctx, next) => {
ctx.state.scrollX = window.scrollX;
ctx.state.scrollY = window.scrollY;
router.replace(router.url, ctx.state);
@ -18,65 +16,86 @@ router.exit(
}
});
const mousetrap = require('mousetrap');
router.enter(
null,
(ctx, next) => {
const mousetrap = require("mousetrap");
router.enter(null, (ctx, next) => {
mousetrap.reset();
next();
});
const tags = require('./tags.js');
const pools = require('./pools.js');
const api = require('./api.js');
const tags = require("./tags.js");
const pools = require("./pools.js");
const api = require("./api.js");
api.fetchConfig().then(() => {
api.fetchConfig()
.then(
() => {
// register controller routes
let controllers = [];
controllers.push(require('./controllers/home_controller.js'));
controllers.push(require('./controllers/help_controller.js'));
controllers.push(require('./controllers/auth_controller.js'));
controllers.push(require('./controllers/password_reset_controller.js'));
controllers.push(require('./controllers/comments_controller.js'));
controllers.push(require('./controllers/snapshots_controller.js'));
controllers.push(require('./controllers/post_detail_controller.js'));
controllers.push(require('./controllers/post_main_controller.js'));
controllers.push(require('./controllers/post_list_controller.js'));
controllers.push(require('./controllers/post_upload_controller.js'));
controllers.push(require('./controllers/tag_controller.js'));
controllers.push(require('./controllers/tag_list_controller.js'));
controllers.push(require('./controllers/tag_categories_controller.js'));
controllers.push(require('./controllers/pool_create_controller.js'));
controllers.push(require('./controllers/pool_controller.js'));
controllers.push(require('./controllers/pool_list_controller.js'));
controllers.push(require('./controllers/pool_categories_controller.js'));
controllers.push(require('./controllers/settings_controller.js'));
controllers.push(require('./controllers/user_controller.js'));
controllers.push(require('./controllers/user_list_controller.js'));
controllers.push(require('./controllers/user_registration_controller.js'));
controllers.push(require("./controllers/home_controller.js"));
controllers.push(require("./controllers/help_controller.js"));
controllers.push(require("./controllers/auth_controller.js"));
controllers.push(
require("./controllers/password_reset_controller.js")
);
controllers.push(require("./controllers/comments_controller.js"));
controllers.push(require("./controllers/snapshots_controller.js"));
controllers.push(
require("./controllers/post_detail_controller.js")
);
controllers.push(require("./controllers/post_main_controller.js"));
controllers.push(require("./controllers/post_list_controller.js"));
controllers.push(
require("./controllers/post_upload_controller.js")
);
controllers.push(require("./controllers/tag_controller.js"));
controllers.push(require("./controllers/tag_list_controller.js"));
controllers.push(
require("./controllers/tag_categories_controller.js")
);
controllers.push(
require("./controllers/pool_create_controller.js")
);
controllers.push(require("./controllers/pool_controller.js"));
controllers.push(require("./controllers/pool_list_controller.js"));
controllers.push(
require("./controllers/pool_categories_controller.js")
);
controllers.push(require("./controllers/settings_controller.js"));
controllers.push(require("./controllers/user_controller.js"));
controllers.push(require("./controllers/user_list_controller.js"));
controllers.push(
require("./controllers/user_registration_controller.js")
);
// 404 controller needs to be registered last
controllers.push(require('./controllers/not_found_controller.js'));
controllers.push(require("./controllers/not_found_controller.js"));
for (let controller of controllers) {
controller(router);
}
}, error => {
window.alert('Could not fetch basic configuration from server');
}).then(() => {
api.loginFromCookies().then(() => {
},
(error) => {
window.alert("Could not fetch basic configuration from server");
}
)
.then(() => {
api.loginFromCookies().then(
() => {
tags.refreshCategoryColorMap();
pools.refreshCategoryColorMap();
router.start();
}, error => {
if (window.location.href.indexOf('login') !== -1) {
},
(error) => {
if (window.location.href.indexOf("login") !== -1) {
api.forget();
router.start();
} else {
const ctx = router.start('/');
const ctx = router.start("/");
ctx.controller.showError(
'An error happened while trying to log you in: ' +
error.message);
"An error happened while trying to log you in: " +
error.message
);
}
});
}
);
});

View file

@ -1,6 +1,6 @@
'use strict';
"use strict";
const events = require('../events.js');
const events = require("../events.js");
class AbstractList extends events.EventTarget {
constructor() {
@ -13,13 +13,15 @@ class AbstractList extends events.EventTarget {
for (let item of response) {
const addedItem = this._itemClass.fromResponse(item);
if (addedItem.addEventListener) {
addedItem.addEventListener('delete', e => {
addedItem.addEventListener("delete", (e) => {
ret.remove(addedItem);
});
addedItem.addEventListener('change', e => {
ret.dispatchEvent(new CustomEvent('change', {
addedItem.addEventListener("change", (e) => {
ret.dispatchEvent(
new CustomEvent("change", {
detail: e.detail,
}));
})
);
});
}
ret._list.push(addedItem);
@ -29,28 +31,32 @@ class AbstractList extends events.EventTarget {
sync(plainList) {
this.clear();
for (let item of (plainList || [])) {
for (let item of plainList || []) {
this.add(this.constructor._itemClass.fromResponse(item));
}
}
add(item) {
if (item.addEventListener) {
item.addEventListener('delete', e => {
item.addEventListener("delete", (e) => {
this.remove(item);
});
item.addEventListener('change', e => {
this.dispatchEvent(new CustomEvent('change', {
item.addEventListener("change", (e) => {
this.dispatchEvent(
new CustomEvent("change", {
detail: e.detail,
}));
})
);
});
}
this._list.push(item);
const detail = {};
detail[this.constructor._itemName] = item;
this.dispatchEvent(new CustomEvent('add', {
this.dispatchEvent(
new CustomEvent("add", {
detail: detail,
}));
})
);
}
clear() {
@ -67,9 +73,11 @@ class AbstractList extends events.EventTarget {
this._list.splice(index, 1);
const detail = {};
detail[this.constructor._itemName] = itemToRemove;
this.dispatchEvent(new CustomEvent('remove', {
this.dispatchEvent(
new CustomEvent("remove", {
detail: detail,
}));
})
);
return;
}
}

View file

@ -1,8 +1,8 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const events = require('../events.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const events = require("../events.js");
class Comment extends events.EventTarget {
constructor() {
@ -31,7 +31,7 @@ class Comment extends events.EventTarget {
}
get text() {
return this._text || '';
return this._text || "";
}
get user() {
@ -63,47 +63,57 @@ class Comment extends events.EventTarget {
version: this._version,
text: this._text,
};
let promise = this._id ?
api.put(uri.formatApiLink('comment', this.id), detail) :
api.post(uri.formatApiLink('comments'),
Object.assign({postId: this._postId}, detail));
let promise = this._id
? api.put(uri.formatApiLink("comment", this.id), detail)
: api.post(
uri.formatApiLink("comments"),
Object.assign({ postId: this._postId }, detail)
);
return promise.then(response => {
return promise.then((response) => {
this._updateFromResponse(response);
this.dispatchEvent(new CustomEvent('change', {
this.dispatchEvent(
new CustomEvent("change", {
detail: {
comment: this,
},
}));
})
);
return Promise.resolve();
});
}
delete() {
return api.delete(
uri.formatApiLink('comment', this.id),
{version: this._version})
.then(response => {
this.dispatchEvent(new CustomEvent('delete', {
return api
.delete(uri.formatApiLink("comment", this.id), {
version: this._version,
})
.then((response) => {
this.dispatchEvent(
new CustomEvent("delete", {
detail: {
comment: this,
},
}));
})
);
return Promise.resolve();
});
}
setScore(score) {
return api.put(
uri.formatApiLink('comment', this.id, 'score'),
{score: score})
.then(response => {
return api
.put(uri.formatApiLink("comment", this.id, "score"), {
score: score,
})
.then((response) => {
this._updateFromResponse(response);
this.dispatchEvent(new CustomEvent('changeScore', {
this.dispatchEvent(
new CustomEvent("changeScore", {
detail: {
comment: this,
},
}));
})
);
return Promise.resolve();
});
}

View file

@ -1,12 +1,11 @@
'use strict';
"use strict";
const AbstractList = require('./abstract_list.js');
const Comment = require('./comment.js');
const AbstractList = require("./abstract_list.js");
const Comment = require("./comment.js");
class CommentList extends AbstractList {
}
class CommentList extends AbstractList {}
CommentList._itemClass = Comment;
CommentList._itemName = 'comment';
CommentList._itemName = "comment";
module.exports = CommentList;

View file

@ -1,21 +1,19 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const Post = require('./post.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const Post = require("./post.js");
class Info {
static get() {
return api.get(uri.formatApiLink('info'))
.then(response => {
return Promise.resolve(Object.assign(
{},
response,
{
featuredPost: response.featuredPost ?
Post.fromResponse(response.featuredPost) :
undefined
}));
return api.get(uri.formatApiLink("info")).then((response) => {
return Promise.resolve(
Object.assign({}, response, {
featuredPost: response.featuredPost
? Post.fromResponse(response.featuredPost)
: undefined,
})
);
});
}
}

View file

@ -1,13 +1,13 @@
'use strict';
"use strict";
const events = require('../events.js');
const Point = require('./point.js');
const PointList = require('./point_list.js');
const events = require("../events.js");
const Point = require("./point.js");
const PointList = require("./point_list.js");
class Note extends events.EventTarget {
constructor() {
super();
this._text = '…';
this._text = "…";
this._polygon = new PointList();
}

View file

@ -1,12 +1,11 @@
'use strict';
"use strict";
const AbstractList = require('./abstract_list.js');
const Note = require('./note.js');
const AbstractList = require("./abstract_list.js");
const Note = require("./note.js");
class NoteList extends AbstractList {
}
class NoteList extends AbstractList {}
NoteList._itemClass = Note;
NoteList._itemName = 'note';
NoteList._itemName = "note";
module.exports = NoteList;

View file

@ -1,6 +1,6 @@
'use strict';
"use strict";
const events = require('../events.js');
const events = require("../events.js");
class Point extends events.EventTarget {
constructor(x, y) {
@ -19,12 +19,16 @@ class Point extends events.EventTarget {
set x(value) {
this._x = value;
this.dispatchEvent(new CustomEvent('change', {detail: {point: this}}));
this.dispatchEvent(
new CustomEvent("change", { detail: { point: this } })
);
}
set y(value) {
this._y = value;
this.dispatchEvent(new CustomEvent('change', {detail: {point: this}}));
this.dispatchEvent(
new CustomEvent("change", { detail: { point: this } })
);
}
}

View file

@ -1,7 +1,7 @@
'use strict';
"use strict";
const AbstractList = require('./abstract_list.js');
const Point = require('./point.js');
const AbstractList = require("./abstract_list.js");
const Point = require("./point.js");
class PointList extends AbstractList {
get firstPoint() {
@ -18,6 +18,6 @@ class PointList extends AbstractList {
}
PointList._itemClass = Point;
PointList._itemName = 'point';
PointList._itemName = "point";
module.exports = PointList;

View file

@ -1,13 +1,13 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const events = require('../events.js');
const misc = require('../util/misc.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const events = require("../events.js");
const misc = require("../util/misc.js");
class Pool extends events.EventTarget {
constructor() {
const PostList = require('./post_list.js');
const PostList = require("./post_list.js");
super();
this._orig = {};
@ -70,8 +70,7 @@ class Pool extends events.EventTarget {
}
static get(id) {
return api.get(uri.formatApiLink('pool', id))
.then(response => {
return api.get(uri.formatApiLink("pool", id)).then((response) => {
return Promise.resolve(Pool.fromResponse(response));
});
}
@ -90,62 +89,71 @@ class Pool extends events.EventTarget {
detail.description = this._description;
}
if (misc.arraysDiffer(this._posts, this._orig._posts)) {
detail.posts = this._posts.map(post => post.id);
detail.posts = this._posts.map((post) => post.id);
}
let promise = this._id ?
api.put(uri.formatApiLink('pool', this._id), detail) :
api.post(uri.formatApiLink('pools'), detail);
return promise
.then(response => {
let promise = this._id
? api.put(uri.formatApiLink("pool", this._id), detail)
: api.post(uri.formatApiLink("pools"), detail);
return promise.then((response) => {
this._updateFromResponse(response);
this.dispatchEvent(new CustomEvent('change', {
this.dispatchEvent(
new CustomEvent("change", {
detail: {
pool: this,
},
}));
})
);
return Promise.resolve();
});
}
merge(targetId, addAlias) {
return api.get(uri.formatApiLink('pool', targetId))
.then(response => {
return api.post(uri.formatApiLink('pool-merge'), {
return api
.get(uri.formatApiLink("pool", targetId))
.then((response) => {
return api.post(uri.formatApiLink("pool-merge"), {
removeVersion: this._version,
remove: this._id,
mergeToVersion: response.version,
mergeTo: targetId,
});
}).then(response => {
})
.then((response) => {
if (!addAlias) {
return Promise.resolve(response);
}
return api.put(uri.formatApiLink('pool', targetId), {
return api.put(uri.formatApiLink("pool", targetId), {
version: response.version,
names: response.names.concat(this._names),
});
}).then(response => {
})
.then((response) => {
this._updateFromResponse(response);
this.dispatchEvent(new CustomEvent('change', {
this.dispatchEvent(
new CustomEvent("change", {
detail: {
pool: this,
},
}));
})
);
return Promise.resolve();
});
}
delete() {
return api.delete(
uri.formatApiLink('pool', this._id),
{version: this._version})
.then(response => {
this.dispatchEvent(new CustomEvent('delete', {
return api
.delete(uri.formatApiLink("pool", this._id), {
version: this._version,
})
.then((response) => {
this.dispatchEvent(
new CustomEvent("delete", {
detail: {
pool: this,
},
}));
})
);
return Promise.resolve();
});
}

View file

@ -1,14 +1,14 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const events = require('../events.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const events = require("../events.js");
class PoolCategory extends events.EventTarget {
constructor() {
super();
this._name = '';
this._color = '#000000';
this._name = "";
this._color = "#000000";
this._poolCount = 0;
this._isDefault = false;
this._origName = null;
@ -63,34 +63,39 @@ class PoolCategory extends events.EventTarget {
return Promise.resolve();
}
let promise = this._origName ?
api.put(
uri.formatApiLink('pool-category', this._origName),
detail) :
api.post(uri.formatApiLink('pool-categories'), detail);
let promise = this._origName
? api.put(
uri.formatApiLink("pool-category", this._origName),
detail
)
: api.post(uri.formatApiLink("pool-categories"), detail);
return promise
.then(response => {
return promise.then((response) => {
this._updateFromResponse(response);
this.dispatchEvent(new CustomEvent('change', {
this.dispatchEvent(
new CustomEvent("change", {
detail: {
poolCategory: this,
},
}));
})
);
return Promise.resolve();
});
}
delete() {
return api.delete(
uri.formatApiLink('pool-category', this._origName),
{version: this._version})
.then(response => {
this.dispatchEvent(new CustomEvent('delete', {
return api
.delete(uri.formatApiLink("pool-category", this._origName), {
version: this._version,
})
.then((response) => {
this.dispatchEvent(
new CustomEvent("delete", {
detail: {
poolCategory: this,
},
}));
})
);
return Promise.resolve();
});
}

View file

@ -1,9 +1,9 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const AbstractList = require('./abstract_list.js');
const PoolCategory = require('./pool_category.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const AbstractList = require("./abstract_list.js");
const PoolCategory = require("./pool_category.js");
class PoolCategoryList extends AbstractList {
constructor() {
@ -11,7 +11,7 @@ class PoolCategoryList extends AbstractList {
this._defaultCategory = null;
this._origDefaultCategory = null;
this._deletedCategories = [];
this.addEventListener('remove', e => this._evtCategoryDeleted(e));
this.addEventListener("remove", (e) => this._evtCategoryDeleted(e));
}
static fromResponse(response) {
@ -27,12 +27,16 @@ class PoolCategoryList extends AbstractList {
}
static get() {
return api.get(uri.formatApiLink('pool-categories'))
.then(response => {
return Promise.resolve(Object.assign(
{},
response,
{results: PoolCategoryList.fromResponse(response.results)}));
return api
.get(uri.formatApiLink("pool-categories"))
.then((response) => {
return Promise.resolve(
Object.assign({}, response, {
results: PoolCategoryList.fromResponse(
response.results
),
})
);
});
}
@ -57,13 +61,15 @@ class PoolCategoryList extends AbstractList {
promises.push(
api.put(
uri.formatApiLink(
'pool-category',
"pool-category",
this._defaultCategory.name,
'default')));
"default"
)
)
);
}
return Promise.all(promises)
.then(response => {
return Promise.all(promises).then((response) => {
this._deletedCategories = [];
return Promise.resolve();
});
@ -77,6 +83,6 @@ class PoolCategoryList extends AbstractList {
}
PoolCategoryList._itemClass = PoolCategory;
PoolCategoryList._itemName = 'poolCategory';
PoolCategoryList._itemName = "poolCategory";
module.exports = PoolCategoryList;

View file

@ -1,25 +1,27 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const AbstractList = require('./abstract_list.js');
const Pool = require('./pool.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const AbstractList = require("./abstract_list.js");
const Pool = require("./pool.js");
class PoolList extends AbstractList {
static search(text, offset, limit, fields) {
return api.get(
uri.formatApiLink(
'pools', {
return api
.get(
uri.formatApiLink("pools", {
query: text,
offset: offset,
limit: limit,
fields: fields.join(','),
}))
.then(response => {
return Promise.resolve(Object.assign(
{},
response,
{results: PoolList.fromResponse(response.results)}));
fields: fields.join(","),
})
)
.then((response) => {
return Promise.resolve(
Object.assign({}, response, {
results: PoolList.fromResponse(response.results),
})
);
});
}
@ -42,6 +44,6 @@ class PoolList extends AbstractList {
}
PoolList._itemClass = Pool;
PoolList._itemName = 'pool';
PoolList._itemName = "pool";
module.exports = PoolList;

View file

@ -1,15 +1,15 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const tags = require('../tags.js');
const events = require('../events.js');
const TagList = require('./tag_list.js');
const NoteList = require('./note_list.js');
const CommentList = require('./comment_list.js');
const PoolList = require('./pool_list.js');
const Pool = require('./pool.js');
const misc = require('../util/misc.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const tags = require("../tags.js");
const events = require("../events.js");
const TagList = require("./tag_list.js");
const NoteList = require("./note_list.js");
const CommentList = require("./comment_list.js");
const PoolList = require("./pool_list.js");
const Pool = require("./pool.js");
const misc = require("../util/misc.js");
class Post extends events.EventTarget {
constructor() {
@ -67,7 +67,7 @@ class Post extends events.EventTarget {
}
get sourceSplit() {
return this._source.split('\n');
return this._source.split("\n");
}
get canvasWidth() {
@ -83,11 +83,11 @@ class Post extends events.EventTarget {
}
get newContent() {
throw 'Invalid operation';
throw "Invalid operation";
}
get newThumbnail() {
throw 'Invalid operation';
throw "Invalid operation";
}
get flags() {
@ -99,7 +99,7 @@ class Post extends events.EventTarget {
}
get tagNames() {
return this._tags.map(tag => tag.names[0]);
return this._tags.map((tag) => tag.names[0]);
}
get notes() {
@ -174,11 +174,11 @@ class Post extends events.EventTarget {
static reverseSearch(content) {
let apiPromise = api.post(
uri.formatApiLink('posts', 'reverse-search'),
uri.formatApiLink("posts", "reverse-search"),
{},
{content: content});
let returnedPromise = apiPromise
.then(response => {
{ content: content }
);
let returnedPromise = apiPromise.then((response) => {
if (response.exactPost) {
response.exactPost = Post.fromResponse(response.exactPost);
}
@ -192,14 +192,13 @@ class Post extends events.EventTarget {
}
static get(id) {
return api.get(uri.formatApiLink('post', id))
.then(response => {
return api.get(uri.formatApiLink("post", id)).then((response) => {
return Promise.resolve(Post.fromResponse(response));
});
}
_savePoolPosts() {
const difference = (a, b) => a.filter(post => !b.hasPoolId(post.id));
const difference = (a, b) => a.filter((post) => !b.hasPoolId(post.id));
// find the pools where the post was added or removed
const added = difference(this.pools, this._orig._pools);
@ -209,7 +208,7 @@ class Post extends events.EventTarget {
// update each pool's list of posts
for (let pool of added) {
let op = Pool.get(pool.id).then(response => {
let op = Pool.get(pool.id).then((response) => {
if (!response.posts.hasPostId(this._id)) {
response.posts.addById(this._id);
return response.save();
@ -221,7 +220,7 @@ class Post extends events.EventTarget {
}
for (let pool of removed) {
let op = Pool.get(pool.id).then(response => {
let op = Pool.get(pool.id).then((response) => {
if (response.posts.hasPostId(this._id)) {
response.posts.removeById(this._id);
return response.save();
@ -250,14 +249,14 @@ class Post extends events.EventTarget {
detail.flags = this._flags;
}
if (misc.arraysDiffer(this._tags, this._orig._tags)) {
detail.tags = this._tags.map(tag => tag.names[0]);
detail.tags = this._tags.map((tag) => tag.names[0]);
}
if (misc.arraysDiffer(this._relations, this._orig._relations)) {
detail.relations = this._relations;
}
if (misc.arraysDiffer(this._notes, this._orig._notes)) {
detail.notes = this._notes.map(note => ({
polygon: note.polygon.map(point => [point.x, point.y]),
detail.notes = this._notes.map((note) => ({
polygon: note.polygon.map((point) => [point.x, point.y]),
text: note.text,
}));
}
@ -271,145 +270,178 @@ class Post extends events.EventTarget {
detail.source = this._source;
}
let apiPromise = this._id ?
api.put(uri.formatApiLink('post', this.id), detail, files) :
api.post(uri.formatApiLink('posts'), detail, files);
let apiPromise = this._id
? api.put(uri.formatApiLink("post", this.id), detail, files)
: api.post(uri.formatApiLink("posts"), detail, files);
return apiPromise.then(response => {
return apiPromise
.then((response) => {
if (misc.arraysDiffer(this._pools, this._orig._pools)) {
return this._savePoolPosts()
.then(() => Promise.resolve(response));
return this._savePoolPosts().then(() =>
Promise.resolve(response)
);
}
return Promise.resolve(response);
}).then(response => {
})
.then(
(response) => {
this._updateFromResponse(response);
this.dispatchEvent(
new CustomEvent('change', {detail: {post: this}}));
new CustomEvent("change", { detail: { post: this } })
);
if (this._newContent) {
this.dispatchEvent(
new CustomEvent('changeContent', {detail: {post: this}}));
new CustomEvent("changeContent", {
detail: { post: this },
})
);
}
if (this._newThumbnail) {
this.dispatchEvent(
new CustomEvent('changeThumbnail', {detail: {post: this}}));
new CustomEvent("changeThumbnail", {
detail: { post: this },
})
);
}
return Promise.resolve();
}, error => {
if (error.response &&
error.response.name === 'PostAlreadyUploadedError') {
error.message =
`Post already uploaded (@${error.response.otherPostId})`;
},
(error) => {
if (
error.response &&
error.response.name === "PostAlreadyUploadedError"
) {
error.message = `Post already uploaded (@${error.response.otherPostId})`;
}
return Promise.reject(error);
});
}
);
}
feature() {
return api.post(
uri.formatApiLink('featured-post'),
{id: this._id})
.then(response => {
return api
.post(uri.formatApiLink("featured-post"), { id: this._id })
.then((response) => {
return Promise.resolve();
});
}
delete() {
return api.delete(
uri.formatApiLink('post', this.id),
{version: this._version})
.then(response => {
this.dispatchEvent(new CustomEvent('delete', {
return api
.delete(uri.formatApiLink("post", this.id), {
version: this._version,
})
.then((response) => {
this.dispatchEvent(
new CustomEvent("delete", {
detail: {
post: this,
},
}));
})
);
return Promise.resolve();
});
}
merge(targetId, useOldContent) {
return api.get(uri.formatApiLink('post', targetId))
.then(response => {
return api.post(uri.formatApiLink('post-merge'), {
return api
.get(uri.formatApiLink("post", targetId))
.then((response) => {
return api.post(uri.formatApiLink("post-merge"), {
removeVersion: this._version,
remove: this._id,
mergeToVersion: response.version,
mergeTo: targetId,
replaceContent: useOldContent,
});
}).then(response => {
})
.then((response) => {
this._updateFromResponse(response);
this.dispatchEvent(new CustomEvent('change', {
this.dispatchEvent(
new CustomEvent("change", {
detail: {
post: this,
},
}));
})
);
return Promise.resolve();
});
}
setScore(score) {
return api.put(
uri.formatApiLink('post', this.id, 'score'),
{score: score})
.then(response => {
return api
.put(uri.formatApiLink("post", this.id, "score"), { score: score })
.then((response) => {
const prevFavorite = this._ownFavorite;
this._updateFromResponse(response);
if (this._ownFavorite !== prevFavorite) {
this.dispatchEvent(new CustomEvent('changeFavorite', {
this.dispatchEvent(
new CustomEvent("changeFavorite", {
detail: {
post: this,
},
}));
})
);
}
this.dispatchEvent(new CustomEvent('changeScore', {
this.dispatchEvent(
new CustomEvent("changeScore", {
detail: {
post: this,
},
}));
})
);
return Promise.resolve();
});
}
addToFavorites() {
return api.post(uri.formatApiLink('post', this.id, 'favorite'))
.then(response => {
return api
.post(uri.formatApiLink("post", this.id, "favorite"))
.then((response) => {
const prevScore = this._ownScore;
this._updateFromResponse(response);
if (this._ownScore !== prevScore) {
this.dispatchEvent(new CustomEvent('changeScore', {
this.dispatchEvent(
new CustomEvent("changeScore", {
detail: {
post: this,
},
}));
})
);
}
this.dispatchEvent(new CustomEvent('changeFavorite', {
this.dispatchEvent(
new CustomEvent("changeFavorite", {
detail: {
post: this,
},
}));
})
);
return Promise.resolve();
});
}
removeFromFavorites() {
return api.delete(uri.formatApiLink('post', this.id, 'favorite'))
.then(response => {
return api
.delete(uri.formatApiLink("post", this.id, "favorite"))
.then((response) => {
const prevScore = this._ownScore;
this._updateFromResponse(response);
if (this._ownScore !== prevScore) {
this.dispatchEvent(new CustomEvent('changeScore', {
this.dispatchEvent(
new CustomEvent("changeScore", {
detail: {
post: this,
},
}));
})
);
}
this.dispatchEvent(new CustomEvent('changeFavorite', {
this.dispatchEvent(
new CustomEvent("changeFavorite", {
detail: {
post: this,
},
}));
})
);
return Promise.resolve();
});
}
@ -417,7 +449,7 @@ class Post extends events.EventTarget {
mutateContentUrl() {
this._contentUrl =
this._orig._contentUrl +
'?bypass-cache=' +
"?bypass-cache=" +
Math.round(Math.random() * 1000);
}
@ -431,15 +463,18 @@ class Post extends events.EventTarget {
_user: response.user,
_safety: response.safety,
_contentUrl: response.contentUrl,
_fullContentUrl: new URL(response.contentUrl, document.getElementsByTagName('base')[0].href).href,
_fullContentUrl: new URL(
response.contentUrl,
document.getElementsByTagName("base")[0].href
).href,
_thumbnailUrl: response.thumbnailUrl,
_source: response.source,
_canvasWidth: response.canvasWidth,
_canvasHeight: response.canvasHeight,
_fileSize: response.fileSize,
_flags: [...response.flags || []],
_relations: [...response.relations || []],
_flags: [...(response.flags || [])],
_relations: [...(response.relations || [])],
_score: response.score,
_commentCount: response.commentCount,

View file

@ -1,35 +1,37 @@
'use strict';
"use strict";
const settings = require('../models/settings.js');
const api = require('../api.js');
const uri = require('../util/uri.js');
const AbstractList = require('./abstract_list.js');
const Post = require('./post.js');
const settings = require("../models/settings.js");
const api = require("../api.js");
const uri = require("../util/uri.js");
const AbstractList = require("./abstract_list.js");
const Post = require("./post.js");
class PostList extends AbstractList {
static getAround(id, searchQuery) {
return api.get(
uri.formatApiLink(
'post', id, 'around', {
query: PostList._decorateSearchQuery(searchQuery || ''),
fields: 'id',
}));
uri.formatApiLink("post", id, "around", {
query: PostList._decorateSearchQuery(searchQuery || ""),
fields: "id",
})
);
}
static search(text, offset, limit, fields) {
return api.get(
uri.formatApiLink(
'posts', {
query: PostList._decorateSearchQuery(text || ''),
return api
.get(
uri.formatApiLink("posts", {
query: PostList._decorateSearchQuery(text || ""),
offset: offset,
limit: limit,
fields: fields.join(','),
}))
.then(response => {
return Promise.resolve(Object.assign(
{},
response,
{results: PostList.fromResponse(response.results)}));
fields: fields.join(","),
})
)
.then((response) => {
return Promise.resolve(
Object.assign({}, response, {
results: PostList.fromResponse(response.results),
})
);
});
}
@ -43,7 +45,7 @@ class PostList extends AbstractList {
}
}
if (disabledSafety.length) {
text = `-rating:${disabledSafety.join(',')} ${text}`;
text = `-rating:${disabledSafety.join(",")} ${text}`;
}
}
return text.trim();
@ -77,6 +79,6 @@ class PostList extends AbstractList {
}
PostList._itemClass = Post;
PostList._itemName = 'post';
PostList._itemName = "post";
module.exports = PostList;

View file

@ -1,6 +1,6 @@
'use strict';
"use strict";
const events = require('../events.js');
const events = require("../events.js");
const defaultSettings = {
listPosts: {
@ -12,7 +12,7 @@ const defaultSettings = {
endlessScroll: false,
keyboardShortcuts: true,
transparencyGrid: true,
fitMode: 'fit-both',
fitMode: "fit-both",
tagSuggestions: true,
autoplayVideos: false,
postsPerPage: 42,
@ -28,7 +28,7 @@ class Settings extends events.EventTarget {
_getFromLocalStorage() {
let ret = Object.assign({}, defaultSettings);
try {
Object.assign(ret, JSON.parse(localStorage.getItem('settings')));
Object.assign(ret, JSON.parse(localStorage.getItem("settings")));
} catch (e) {
// continue regardless of error
}
@ -37,14 +37,16 @@ class Settings extends events.EventTarget {
save(newSettings, silent) {
newSettings = Object.assign(this.cache, newSettings);
localStorage.setItem('settings', JSON.stringify(newSettings));
localStorage.setItem("settings", JSON.stringify(newSettings));
this.cache = this._getFromLocalStorage();
if (silent !== true) {
this.dispatchEvent(new CustomEvent('change', {
this.dispatchEvent(
new CustomEvent("change", {
detail: {
settings: this.cache,
},
}));
})
);
}
}

View file

@ -1,7 +1,7 @@
'use strict';
"use strict";
const api = require('../api.js');
const events = require('../events.js');
const api = require("../api.js");
const events = require("../events.js");
class Snapshot extends events.EventTarget {
constructor() {

View file

@ -1,24 +1,31 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const AbstractList = require('./abstract_list.js');
const Snapshot = require('./snapshot.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const AbstractList = require("./abstract_list.js");
const Snapshot = require("./snapshot.js");
class SnapshotList extends AbstractList {
static search(text, offset, limit) {
return api.get(uri.formatApiLink(
'snapshots', {query: text, offset: offset, limit: limit}))
.then(response => {
return Promise.resolve(Object.assign(
{},
response,
{results: SnapshotList.fromResponse(response.results)}));
return api
.get(
uri.formatApiLink("snapshots", {
query: text,
offset: offset,
limit: limit,
})
)
.then((response) => {
return Promise.resolve(
Object.assign({}, response, {
results: SnapshotList.fromResponse(response.results),
})
);
});
}
}
SnapshotList._itemClass = Snapshot;
SnapshotList._itemName = 'snapshot';
SnapshotList._itemName = "snapshot";
module.exports = SnapshotList;

View file

@ -1,13 +1,13 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const events = require('../events.js');
const misc = require('../util/misc.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const events = require("../events.js");
const misc = require("../util/misc.js");
class Tag extends events.EventTarget {
constructor() {
const TagList = require('./tag_list.js');
const TagList = require("./tag_list.js");
super();
this._orig = {};
@ -71,8 +71,7 @@ class Tag extends events.EventTarget {
}
static get(name) {
return api.get(uri.formatApiLink('tag', name))
.then(response => {
return api.get(uri.formatApiLink("tag", name)).then((response) => {
return Promise.resolve(Tag.fromResponse(response));
});
}
@ -92,66 +91,77 @@ class Tag extends events.EventTarget {
}
if (misc.arraysDiffer(this._implications, this._orig._implications)) {
detail.implications = this._implications.map(
relation => relation.names[0]);
(relation) => relation.names[0]
);
}
if (misc.arraysDiffer(this._suggestions, this._orig._suggestions)) {
detail.suggestions = this._suggestions.map(
relation => relation.names[0]);
(relation) => relation.names[0]
);
}
let promise = this._origName ?
api.put(uri.formatApiLink('tag', this._origName), detail) :
api.post(uri.formatApiLink('tags'), detail);
return promise
.then(response => {
let promise = this._origName
? api.put(uri.formatApiLink("tag", this._origName), detail)
: api.post(uri.formatApiLink("tags"), detail);
return promise.then((response) => {
this._updateFromResponse(response);
this.dispatchEvent(new CustomEvent('change', {
this.dispatchEvent(
new CustomEvent("change", {
detail: {
tag: this,
},
}));
})
);
return Promise.resolve();
});
}
merge(targetName, addAlias) {
return api.get(uri.formatApiLink('tag', targetName))
.then(response => {
return api.post(uri.formatApiLink('tag-merge'), {
return api
.get(uri.formatApiLink("tag", targetName))
.then((response) => {
return api.post(uri.formatApiLink("tag-merge"), {
removeVersion: this._version,
remove: this._origName,
mergeToVersion: response.version,
mergeTo: targetName,
});
}).then(response => {
})
.then((response) => {
if (!addAlias) {
return Promise.resolve(response);
}
return api.put(uri.formatApiLink('tag', targetName), {
return api.put(uri.formatApiLink("tag", targetName), {
version: response.version,
names: response.names.concat(this._names),
});
}).then(response => {
})
.then((response) => {
this._updateFromResponse(response);
this.dispatchEvent(new CustomEvent('change', {
this.dispatchEvent(
new CustomEvent("change", {
detail: {
tag: this,
},
}));
})
);
return Promise.resolve();
});
}
delete() {
return api.delete(
uri.formatApiLink('tag', this._origName),
{version: this._version})
.then(response => {
this.dispatchEvent(new CustomEvent('delete', {
return api
.delete(uri.formatApiLink("tag", this._origName), {
version: this._version,
})
.then((response) => {
this.dispatchEvent(
new CustomEvent("delete", {
detail: {
tag: this,
},
}));
})
);
return Promise.resolve();
});
}

View file

@ -1,14 +1,14 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const events = require('../events.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const events = require("../events.js");
class TagCategory extends events.EventTarget {
constructor() {
super();
this._name = '';
this._color = '#000000';
this._name = "";
this._color = "#000000";
this._tagCount = 0;
this._isDefault = false;
this._origName = null;
@ -63,34 +63,39 @@ class TagCategory extends events.EventTarget {
return Promise.resolve();
}
let promise = this._origName ?
api.put(
uri.formatApiLink('tag-category', this._origName),
detail) :
api.post(uri.formatApiLink('tag-categories'), detail);
let promise = this._origName
? api.put(
uri.formatApiLink("tag-category", this._origName),
detail
)
: api.post(uri.formatApiLink("tag-categories"), detail);
return promise
.then(response => {
return promise.then((response) => {
this._updateFromResponse(response);
this.dispatchEvent(new CustomEvent('change', {
this.dispatchEvent(
new CustomEvent("change", {
detail: {
tagCategory: this,
},
}));
})
);
return Promise.resolve();
});
}
delete() {
return api.delete(
uri.formatApiLink('tag-category', this._origName),
{version: this._version})
.then(response => {
this.dispatchEvent(new CustomEvent('delete', {
return api
.delete(uri.formatApiLink("tag-category", this._origName), {
version: this._version,
})
.then((response) => {
this.dispatchEvent(
new CustomEvent("delete", {
detail: {
tagCategory: this,
},
}));
})
);
return Promise.resolve();
});
}

View file

@ -1,9 +1,9 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const AbstractList = require('./abstract_list.js');
const TagCategory = require('./tag_category.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const AbstractList = require("./abstract_list.js");
const TagCategory = require("./tag_category.js");
class TagCategoryList extends AbstractList {
constructor() {
@ -11,7 +11,7 @@ class TagCategoryList extends AbstractList {
this._defaultCategory = null;
this._origDefaultCategory = null;
this._deletedCategories = [];
this.addEventListener('remove', e => this._evtCategoryDeleted(e));
this.addEventListener("remove", (e) => this._evtCategoryDeleted(e));
}
static fromResponse(response) {
@ -27,12 +27,16 @@ class TagCategoryList extends AbstractList {
}
static get() {
return api.get(uri.formatApiLink('tag-categories'))
.then(response => {
return Promise.resolve(Object.assign(
{},
response,
{results: TagCategoryList.fromResponse(response.results)}));
return api
.get(uri.formatApiLink("tag-categories"))
.then((response) => {
return Promise.resolve(
Object.assign({}, response, {
results: TagCategoryList.fromResponse(
response.results
),
})
);
});
}
@ -57,13 +61,15 @@ class TagCategoryList extends AbstractList {
promises.push(
api.put(
uri.formatApiLink(
'tag-category',
"tag-category",
this._defaultCategory.name,
'default')));
"default"
)
)
);
}
return Promise.all(promises)
.then(response => {
return Promise.all(promises).then((response) => {
this._deletedCategories = [];
return Promise.resolve();
});
@ -77,6 +83,6 @@ class TagCategoryList extends AbstractList {
}
TagCategoryList._itemClass = TagCategory;
TagCategoryList._itemName = 'tagCategory';
TagCategoryList._itemName = "tagCategory";
module.exports = TagCategoryList;

View file

@ -1,25 +1,27 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const AbstractList = require('./abstract_list.js');
const Tag = require('./tag.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const AbstractList = require("./abstract_list.js");
const Tag = require("./tag.js");
class TagList extends AbstractList {
static search(text, offset, limit, fields) {
return api.get(
uri.formatApiLink(
'tags', {
return api
.get(
uri.formatApiLink("tags", {
query: text,
offset: offset,
limit: limit,
fields: fields.join(','),
}))
.then(response => {
return Promise.resolve(Object.assign(
{},
response,
{results: TagList.fromResponse(response.results)}));
fields: fields.join(","),
})
)
.then((response) => {
return Promise.resolve(
Object.assign({}, response, {
results: TagList.fromResponse(response.results),
})
);
});
}
@ -45,10 +47,12 @@ class TagList extends AbstractList {
this.add(tag);
if (addImplications !== false) {
return Tag.get(tagName).then(actualTag => {
return Tag.get(tagName).then((actualTag) => {
return Promise.all(
actualTag.implications.map(
relation => this.addByName(relation.names[0], true)));
actualTag.implications.map((relation) =>
this.addByName(relation.names[0], true)
)
);
});
}
@ -67,6 +71,6 @@ class TagList extends AbstractList {
}
TagList._itemClass = Tag;
TagList._itemName = 'tag';
TagList._itemName = "tag";
module.exports = TagList;

View file

@ -1,7 +1,7 @@
'use strict';
"use strict";
const events = require('../events.js');
const api = require('../api.js');
const events = require("../events.js");
const api = require("../api.js");
class TopNavigationItem {
constructor(accessKey, title, url, available, imageUrl) {
@ -44,18 +44,20 @@ class TopNavigation extends events.EventTarget {
activate(key) {
this.activeItem = null;
this.dispatchEvent(new CustomEvent('activate', {
this.dispatchEvent(
new CustomEvent("activate", {
detail: {
key: key,
item: key ? this.get(key) : null,
},
}));
})
);
}
setTitle(title) {
api.fetchConfig().then(() => {
document.oldTitle = null;
document.title = api.getName() + (title ? (' ' + title) : '');
document.title = api.getName() + (title ? " " + title : "");
});
}
@ -76,24 +78,22 @@ class TopNavigation extends events.EventTarget {
function _makeTopNavigation() {
const ret = new TopNavigation();
ret.add('home', new TopNavigationItem('H', 'Home', ''));
ret.add('posts', new TopNavigationItem('P', 'Posts', 'posts'));
ret.add('upload', new TopNavigationItem('U', 'Upload', 'upload'));
ret.add('comments', new TopNavigationItem('C', 'Comments', 'comments'));
ret.add('tags', new TopNavigationItem('T', 'Tags', 'tags'));
ret.add('pools', new TopNavigationItem('O', 'Pools', 'pools'));
ret.add('users', new TopNavigationItem('S', 'Users', 'users'));
ret.add('account', new TopNavigationItem('A', 'Account', 'user/{me}'));
ret.add('register', new TopNavigationItem('R', 'Register', 'register'));
ret.add('login', new TopNavigationItem('L', 'Log in', 'login'));
ret.add('logout', new TopNavigationItem('O', 'Logout', 'logout'));
ret.add('help', new TopNavigationItem('E', 'Help', 'help'));
ret.add("home", new TopNavigationItem("H", "Home", ""));
ret.add("posts", new TopNavigationItem("P", "Posts", "posts"));
ret.add("upload", new TopNavigationItem("U", "Upload", "upload"));
ret.add("comments", new TopNavigationItem("C", "Comments", "comments"));
ret.add("tags", new TopNavigationItem("T", "Tags", "tags"));
ret.add("pools", new TopNavigationItem("O", "Pools", "pools"));
ret.add("users", new TopNavigationItem("S", "Users", "users"));
ret.add("account", new TopNavigationItem("A", "Account", "user/{me}"));
ret.add("register", new TopNavigationItem("R", "Register", "register"));
ret.add("login", new TopNavigationItem("L", "Log in", "login"));
ret.add("logout", new TopNavigationItem("O", "Logout", "logout"));
ret.add("help", new TopNavigationItem("E", "Help", "help"));
ret.add(
'settings',
new TopNavigationItem(
null,
'<i class=\'fa fa-cog\'></i>',
'settings'));
"settings",
new TopNavigationItem(null, "<i class='fa fa-cog'></i>", "settings")
);
return ret;
}

View file

@ -1,8 +1,8 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const events = require('../events.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const events = require("../events.js");
class User extends events.EventTarget {
constructor() {
@ -64,11 +64,11 @@ class User extends events.EventTarget {
}
get avatarContent() {
throw 'Invalid operation';
throw "Invalid operation";
}
get password() {
throw 'Invalid operation';
throw "Invalid operation";
}
set name(value) {
@ -102,8 +102,7 @@ class User extends events.EventTarget {
}
static get(name) {
return api.get(uri.formatApiLink('user', name))
.then(response => {
return api.get(uri.formatApiLink("user", name)).then((response) => {
return Promise.resolve(User.fromResponse(response));
});
}
@ -133,33 +132,40 @@ class User extends events.EventTarget {
detail.password = this._password;
}
let promise = this._orig._name ?
api.put(
uri.formatApiLink('user', this._orig._name), detail, files) :
api.post(uri.formatApiLink('users'), detail, files);
let promise = this._orig._name
? api.put(
uri.formatApiLink("user", this._orig._name),
detail,
files
)
: api.post(uri.formatApiLink("users"), detail, files);
return promise
.then(response => {
return promise.then((response) => {
this._updateFromResponse(response);
this.dispatchEvent(new CustomEvent('change', {
this.dispatchEvent(
new CustomEvent("change", {
detail: {
user: this,
},
}));
})
);
return Promise.resolve();
});
}
delete() {
return api.delete(
uri.formatApiLink('user', this._orig._name),
{version: this._version})
.then(response => {
this.dispatchEvent(new CustomEvent('delete', {
return api
.delete(uri.formatApiLink("user", this._orig._name), {
version: this._version,
})
.then((response) => {
this.dispatchEvent(
new CustomEvent("delete", {
detail: {
user: this,
},
}));
})
);
return Promise.resolve();
});
}

View file

@ -1,25 +1,31 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const AbstractList = require('./abstract_list.js');
const User = require('./user.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const AbstractList = require("./abstract_list.js");
const User = require("./user.js");
class UserList extends AbstractList {
static search(text, offset, limit) {
return api.get(
uri.formatApiLink(
'users', {query: text, offset: offset, limit: limit}))
.then(response => {
return Promise.resolve(Object.assign(
{},
response,
{results: UserList.fromResponse(response.results)}));
return api
.get(
uri.formatApiLink("users", {
query: text,
offset: offset,
limit: limit,
})
)
.then((response) => {
return Promise.resolve(
Object.assign({}, response, {
results: UserList.fromResponse(response.results),
})
);
});
}
}
UserList._itemClass = User;
UserList._itemName = 'user';
UserList._itemName = "user";
module.exports = UserList;

View file

@ -1,8 +1,8 @@
'use strict';
"use strict";
const api = require('../api.js');
const uri = require('../util/uri.js');
const events = require('../events.js');
const api = require("../api.js");
const uri = require("../util/uri.js");
const events = require("../events.js");
class UserToken extends events.EventTarget {
constructor() {
@ -48,12 +48,12 @@ class UserToken extends events.EventTarget {
}
static fromResponse(response) {
if (typeof response.results !== 'undefined') {
if (typeof response.results !== "undefined") {
let tokenList = [];
for (let responseToken of response.results) {
const token = new UserToken();
token._updateFromResponse(responseToken);
tokenList.push(token)
tokenList.push(token);
}
return tokenList;
} else {
@ -64,15 +64,16 @@ class UserToken extends events.EventTarget {
}
static get(userName) {
return api.get(uri.formatApiLink('user-tokens', userName))
.then(response => {
return api
.get(uri.formatApiLink("user-tokens", userName))
.then((response) => {
return Promise.resolve(UserToken.fromResponse(response));
});
}
static create(userName, note, expirationTime) {
let userTokenRequest = {
enabled: true
enabled: true,
};
if (note) {
userTokenRequest.note = note;
@ -80,9 +81,10 @@ class UserToken extends events.EventTarget {
if (expirationTime) {
userTokenRequest.expirationTime = expirationTime;
}
return api.post(uri.formatApiLink('user-token', userName), userTokenRequest)
.then(response => {
return Promise.resolve(UserToken.fromResponse(response))
return api
.post(uri.formatApiLink("user-token", userName), userTokenRequest)
.then((response) => {
return Promise.resolve(UserToken.fromResponse(response));
});
}
@ -93,30 +95,40 @@ class UserToken extends events.EventTarget {
detail.note = this._note;
}
return api.put(
uri.formatApiLink('user-token', userName, this._orig._token),
detail)
.then(response => {
return api
.put(
uri.formatApiLink("user-token", userName, this._orig._token),
detail
)
.then((response) => {
this._updateFromResponse(response);
this.dispatchEvent(new CustomEvent('change', {
this.dispatchEvent(
new CustomEvent("change", {
detail: {
userToken: this,
},
}));
})
);
return Promise.resolve(this);
});
}
delete(userName) {
return api.delete(
uri.formatApiLink('user-token', userName, this._orig._token),
{version: this._version})
.then(response => {
this.dispatchEvent(new CustomEvent('delete', {
return api
.delete(
uri.formatApiLink("user-token", userName, this._orig._token),
{
version: this._version,
}
)
.then((response) => {
this.dispatchEvent(
new CustomEvent("delete", {
detail: {
userToken: this,
},
}));
})
);
return Promise.resolve();
});
}

View file

@ -1,22 +1,23 @@
'use strict';
"use strict";
const misc = require('./util/misc.js');
const PoolCategoryList = require('./models/pool_category_list.js');
const misc = require("./util/misc.js");
const PoolCategoryList = require("./models/pool_category_list.js");
let _stylesheet = null;
function refreshCategoryColorMap() {
return PoolCategoryList.get().then(response => {
return PoolCategoryList.get().then((response) => {
if (_stylesheet) {
document.head.removeChild(_stylesheet);
}
_stylesheet = document.createElement('style');
_stylesheet = document.createElement("style");
document.head.appendChild(_stylesheet);
for (let category of response.results) {
const ruleName = misc.makeCssName(category.name, 'pool');
const ruleName = misc.makeCssName(category.name, "pool");
_stylesheet.sheet.insertRule(
`.${ruleName} { color: ${category.color} }`,
_stylesheet.sheet.cssRules.length);
_stylesheet.sheet.cssRules.length
);
}
});
}

View file

@ -1,4 +1,4 @@
'use strict';
"use strict";
// modified page.js by visionmedia
// - changed regexes to components
@ -10,13 +10,17 @@
// - rename .save() to .replaceState()
// - offer .url
const clickEvent = document.ontouchstart ? 'touchstart' : 'click';
const uri = require('./util/uri.js');
const clickEvent = document.ontouchstart ? "touchstart" : "click";
const uri = require("./util/uri.js");
let location = window.history.location || window.location;
function _getOrigin() {
return location.protocol + '//' + location.hostname
+ (location.port ? (':' + location.port) : '');
return (
location.protocol +
"//" +
location.hostname +
(location.port ? ":" + location.port : "")
);
}
function _isSameOrigin(href) {
@ -24,15 +28,16 @@ function _isSameOrigin(href) {
}
function _getBaseHref() {
const bases = document.getElementsByTagName('base');
return bases.length > 0 ?
bases[0].href.replace(_getOrigin(), '').replace(/\/+$/, '') : '';
const bases = document.getElementsByTagName("base");
return bases.length > 0
? bases[0].href.replace(_getOrigin(), "").replace(/\/+$/, "")
: "";
}
class Context {
constructor(path, state) {
const base = _getBaseHref();
path = path.indexOf('/') !== 0 ? '/' + path : path;
path = path.indexOf("/") !== 0 ? "/" + path : path;
path = path.indexOf(base) !== 0 ? base + path : path;
this.canonicalPath = path;
@ -55,7 +60,7 @@ class Context {
class Route {
constructor(path) {
this.method = 'GET';
this.method = "GET";
this.path = path;
this.parameterNames = [];
@ -64,16 +69,17 @@ class Route {
} else {
let parts = [];
for (let component of this.path) {
if (component[0] === ':') {
parts.push('([^/]+)');
if (component[0] === ":") {
parts.push("([^/]+)");
this.parameterNames.push(component.substr(1));
} else { // assert [a-z]+
} else {
// assert [a-z]+
parts.push(component);
}
}
let regexString = '^/' + parts.join('/');
regexString += '(?:/*|/((?:(?:[a-z]+=[^/]+);)*(?:[a-z]+=[^/]+)))$';
this.parameterNames.push('variable');
let regexString = "^/" + parts.join("/");
regexString += "(?:/*|/((?:(?:[a-z]+=[^/]+);)*(?:[a-z]+=[^/]+)))$";
this.parameterNames.push("variable");
this.regex = new RegExp(regexString);
}
}
@ -88,7 +94,7 @@ class Route {
}
match(path, parameters) {
const qsIndex = path.indexOf('?');
const qsIndex = path.indexOf("?");
const pathname = ~qsIndex ? path.slice(0, qsIndex) : path;
const match = this.regex.exec(pathname);
@ -104,8 +110,8 @@ class Route {
continue;
}
if (name === 'variable') {
for (let word of (value || '').split(/;/)) {
if (name === "variable") {
for (let word of (value || "").split(/;/)) {
const [key, subvalue] = word.split(/=/, 2);
parameters[key] = uri.unescapeParam(subvalue);
}
@ -148,7 +154,7 @@ class Router {
this._running = true;
this._onPopState = _onPopState(this);
this._onClick = _onClick(this);
window.addEventListener('popstate', this._onPopState, false);
window.addEventListener("popstate", this._onPopState, false);
document.addEventListener(clickEvent, this._onClick, false);
const url = location.pathname + location.search + location.hash;
return this.replace(url, history.state, true);
@ -160,7 +166,7 @@ class Router {
}
this._running = false;
document.removeEventListener(clickEvent, this._onClick, false);
window.removeEventListener('popstate', this._onPopState, false);
window.removeEventListener("popstate", this._onPopState, false);
}
showNoDispatch(path, state) {
@ -199,11 +205,11 @@ class Router {
middle();
next();
};
const callChain = (this.ctx ? this._exits : [])
.concat(
const callChain = (this.ctx ? this._exits : []).concat(
[swap],
this._callbacks,
[this._unhandled, (ctx, next) => {}]);
[this._unhandled, (ctx, next) => {}]
);
let i = 0;
let fn = () => {
@ -226,20 +232,18 @@ class Router {
}
}
const _onPopState = router => {
const _onPopState = (router) => {
let loaded = false;
if (document.readyState === 'complete') {
if (document.readyState === "complete") {
loaded = true;
} else {
window.addEventListener(
'load',
() => {
window.addEventListener("load", () => {
setTimeout(() => {
loaded = true;
}, 0);
});
}
return e => {
return (e) => {
if (!loaded) {
return;
}
@ -247,16 +251,13 @@ const _onPopState = router => {
const path = e.state.path;
router.replace(path, e.state, true);
} else {
router.show(
location.pathname + location.hash,
undefined,
false);
router.show(location.pathname + location.hash, undefined, false);
}
};
};
const _onClick = router => {
return e => {
const _onClick = (router) => {
return (e) => {
if (1 !== _which(e)) {
return;
}
@ -268,23 +269,25 @@ const _onClick = router => {
}
let el = e.path ? e.path[0] : e.target;
while (el && el.nodeName !== 'A') {
while (el && el.nodeName !== "A") {
el = el.parentNode;
}
if (!el || el.nodeName !== 'A') {
if (!el || el.nodeName !== "A") {
return;
}
if (el.hasAttribute('download') ||
el.getAttribute('rel') === 'external') {
if (
el.hasAttribute("download") ||
el.getAttribute("rel") === "external"
) {
return;
}
const link = el.getAttribute('href');
if (el.pathname === location.pathname && (el.hash || '#' === link)) {
const link = el.getAttribute("href");
if (el.pathname === location.pathname && (el.hash || "#" === link)) {
return;
}
if (link && link.indexOf('mailto:') > -1) {
if (link && link.indexOf("mailto:") > -1) {
return;
}
if (el.target) {
@ -295,7 +298,7 @@ const _onClick = router => {
}
const base = _getBaseHref();
const orig = el.pathname + el.search + (el.hash || '');
const orig = el.pathname + el.search + (el.hash || "");
const path = !orig.indexOf(base) ? orig.slice(base.length) : orig;
if (base && orig === path) {

View file

@ -1,22 +1,23 @@
'use strict';
"use strict";
const misc = require('./util/misc.js');
const TagCategoryList = require('./models/tag_category_list.js');
const misc = require("./util/misc.js");
const TagCategoryList = require("./models/tag_category_list.js");
let _stylesheet = null;
function refreshCategoryColorMap() {
return TagCategoryList.get().then(response => {
return TagCategoryList.get().then((response) => {
if (_stylesheet) {
document.head.removeChild(_stylesheet);
}
_stylesheet = document.createElement('style');
_stylesheet = document.createElement("style");
document.head.appendChild(_stylesheet);
for (let category of response.results) {
const ruleName = misc.makeCssName(category.name, 'tag');
const ruleName = misc.makeCssName(category.name, "tag");
_stylesheet.sheet.insertRule(
`.${ruleName} { color: ${category.color} }`,
_stylesheet.sheet.cssRules.length);
_stylesheet.sheet.cssRules.length
);
}
});
}

View file

@ -1,3 +1,3 @@
'use strict';
"use strict";
module.exports = require('./.templates.autogen.js');
module.exports = require("./.templates.autogen.js");

View file

@ -1,7 +1,7 @@
'use strict';
"use strict";
const mousetrap = require('mousetrap');
const settings = require('../models/settings.js');
const mousetrap = require("mousetrap");
const settings = require("../models/settings.js");
let paused = false;
const _originalStopCallback = mousetrap.prototype.stopCallback;

View file

@ -1,6 +1,6 @@
'use strict';
"use strict";
const marked = require('marked');
const marked = require("marked");
class BaseMarkdownWrapper {
preprocess(text) {
@ -20,42 +20,44 @@ class SjisWrapper extends BaseMarkdownWrapper {
preprocess(text) {
return text.replace(
/\[sjis\]((?:[^\[]|\[(?!\/?sjis\]))+)\[\/sjis\]/ig,
/\[sjis\]((?:[^\[]|\[(?!\/?sjis\]))+)\[\/sjis\]/gi,
(match, capture) => {
var ret = '%%%SJIS' + this.buf.length;
var ret = "%%%SJIS" + this.buf.length;
this.buf.push(capture);
return ret;
});
}
);
}
postprocess(text) {
return text.replace(
/(?:<p>)?%%%SJIS(\d+)(?:<\/p>)?/,
(match, capture) => {
return '<div class="sjis">' + this.buf[capture] + '</div>';
});
return '<div class="sjis">' + this.buf[capture] + "</div>";
}
);
}
}
// fix \ before ~ being stripped away
class TildeWrapper extends BaseMarkdownWrapper {
preprocess(text) {
return text.replace(/\\~/g, '%%%T');
return text.replace(/\\~/g, "%%%T");
}
postprocess(text) {
return text.replace(/%%%T/g, '\\~');
return text.replace(/%%%T/g, "\\~");
}
}
// prevent ^#... from being treated as headers, due to tag permalinks
class TagPermalinkFixWrapper extends BaseMarkdownWrapper {
preprocess(text) {
return text.replace(/^#/g, '%%%#');
return text.replace(/^#/g, "%%%#");
}
postprocess(text) {
return text.replace(/%%%#/g, '#');
return text.replace(/%%%#/g, "#");
}
}
@ -63,19 +65,23 @@ class TagPermalinkFixWrapper extends BaseMarkdownWrapper {
class EntityPermalinkWrapper extends BaseMarkdownWrapper {
preprocess(text) {
// URL-based permalinks
text = text.replace(new RegExp("\\b/post/(\\d+)/?\\b", "g"), "@$1");
text = text.replace(
new RegExp('\\b/post/(\\d+)/?\\b', 'g'), '@$1');
new RegExp("\\b/tag/([a-zA-Z0-9_-]+?)/?", "g"),
"#$1"
);
text = text.replace(
new RegExp('\\b/tag/([a-zA-Z0-9_-]+?)/?', 'g'), '#$1');
text = text.replace(
new RegExp('\\b/user/([a-zA-Z0-9_-]+?)/?', 'g'), '+$1');
new RegExp("\\b/user/([a-zA-Z0-9_-]+?)/?", "g"),
"+$1"
);
text = text.replace(
/(^|^\(|(?:[^\]])\(|[\s<>\[\]\)])([+#@][a-zA-Z0-9_-]+)/g,
'$1[$2]($2)');
text = text.replace(/\]\(@(\d+)\)/g, '](/post/$1)');
text = text.replace(/\]\(\+([a-zA-Z0-9_-]+)\)/g, '](/user/$1)');
text = text.replace(/\]\(#([a-zA-Z0-9_-]+)\)/g, '](/posts/query=$1)');
"$1[$2]($2)"
);
text = text.replace(/\]\(@(\d+)\)/g, "](/post/$1)");
text = text.replace(/\]\(\+([a-zA-Z0-9_-]+)\)/g, "](/user/$1)");
text = text.replace(/\]\(#([a-zA-Z0-9_-]+)\)/g, "](/posts/query=$1)");
return text;
}
}
@ -83,51 +89,58 @@ class EntityPermalinkWrapper extends BaseMarkdownWrapper {
class SearchPermalinkWrapper extends BaseMarkdownWrapper {
postprocess(text) {
return text.replace(
/\[search\]((?:[^\[]|\[(?!\/?search\]))+)\[\/search\]/ig,
'<a href="/posts/query=$1"><code>$1</code></a>');
/\[search\]((?:[^\[]|\[(?!\/?search\]))+)\[\/search\]/gi,
'<a href="/posts/query=$1"><code>$1</code></a>'
);
}
}
class SpoilersWrapper extends BaseMarkdownWrapper {
postprocess(text) {
return text.replace(
/\[spoiler\]((?:[^\[]|\[(?!\/?spoiler\]))+)\[\/spoiler\]/ig,
'<span class="spoiler">$1</span>');
/\[spoiler\]((?:[^\[]|\[(?!\/?spoiler\]))+)\[\/spoiler\]/gi,
'<span class="spoiler">$1</span>'
);
}
}
class SmallWrapper extends BaseMarkdownWrapper {
postprocess(text) {
return text.replace(
/\[small\]((?:[^\[]|\[(?!\/?small\]))+)\[\/small\]/ig,
'<small>$1</small>');
/\[small\]((?:[^\[]|\[(?!\/?small\]))+)\[\/small\]/gi,
"<small>$1</small>"
);
}
}
class StrikeThroughWrapper extends BaseMarkdownWrapper {
postprocess(text) {
text = text.replace(/(^|[^\\])(~~|~)([^~]+)\2/g, '$1<del>$3</del>');
return text.replace(/\\~/g, '~');
text = text.replace(/(^|[^\\])(~~|~)([^~]+)\2/g, "$1<del>$3</del>");
return text.replace(/\\~/g, "~");
}
}
function createRenderer() {
function sanitize(str) {
return str.replace(/&<"/g, m => {
if (m === '&') {
return '&amp;';
return str.replace(/&<"/g, (m) => {
if (m === "&") {
return "&amp;";
}
if (m === '<') {
return '&lt;';
if (m === "<") {
return "&lt;";
}
return '&quot;';
return "&quot;";
});
}
const renderer = new marked.Renderer();
renderer.image = (href, title, alt) => {
let [_, url, width, height] =
(/^(.+?)(?:\s=\s*(\d*)\s*x\s*(\d*)\s*)?$/).exec(href);
let [
_,
url,
width,
height,
] = /^(.+?)(?:\s=\s*(\d*)\s*x\s*(\d*)\s*)?$/.exec(href);
let res = '<img src="' + sanitize(url) + '" alt="' + sanitize(alt);
if (width) {
res += '" width="' + width;

View file

@ -1,14 +1,14 @@
'use strict';
"use strict";
const markdown = require('./markdown.js');
const uri = require('./uri.js');
const settings = require('../models/settings.js');
const markdown = require("./markdown.js");
const uri = require("./uri.js");
const settings = require("../models/settings.js");
function decamelize(str, sep) {
sep = sep === undefined ? '-' : sep;
sep = sep === undefined ? "-" : sep;
return str
.replace(/([a-z\d])([A-Z])/g, '$1' + sep + '$2')
.replace(/([A-Z]+)([A-Z][a-z\d]+)/g, '$1' + sep + '$2')
.replace(/([a-z\d])([A-Z])/g, "$1" + sep + "$2")
.replace(/([A-Z]+)([A-Z][a-z\d]+)/g, "$1" + sep + "$2")
.toLowerCase();
}
@ -45,16 +45,17 @@ function formatFileSize(fileSize) {
return _formatUnits(
fileSize,
1024,
['B', 'K', 'M', 'G'],
["B", "K", "M", "G"],
(number, suffix) => {
const decimalPlaces = number < 20 && suffix !== 'B' ? 1 : 0;
const decimalPlaces = number < 20 && suffix !== "B" ? 1 : 0;
return number.toFixed(decimalPlaces) + suffix;
});
}
);
}
function formatRelativeTime(timeString) {
if (!timeString) {
return 'never';
return "never";
}
const then = Date.parse(timeString);
@ -63,17 +64,17 @@ function formatRelativeTime(timeString) {
const future = now < then;
const descriptions = [
[60, 'a few seconds', null],
[60 * 2, 'a minute', null],
[60 * 60, '% minutes', 60],
[60 * 60 * 2, 'an hour', null],
[60 * 60 * 24, '% hours', 60 * 60],
[60 * 60 * 24 * 2, 'a day', null],
[60 * 60 * 24 * 30.42, '% days', 60 * 60 * 24],
[60 * 60 * 24 * 30.42 * 2, 'a month', null],
[60 * 60 * 24 * 30.42 * 12, '% months', 60 * 60 * 24 * 30.42],
[60 * 60 * 24 * 30.42 * 12 * 2, 'a year', null],
[8640000000000000 /* max*/, '% years', 60 * 60 * 24 * 30.42 * 12],
[60, "a few seconds", null],
[60 * 2, "a minute", null],
[60 * 60, "% minutes", 60],
[60 * 60 * 2, "an hour", null],
[60 * 60 * 24, "% hours", 60 * 60],
[60 * 60 * 24 * 2, "a day", null],
[60 * 60 * 24 * 30.42, "% days", 60 * 60 * 24],
[60 * 60 * 24 * 30.42 * 2, "a month", null],
[60 * 60 * 24 * 30.42 * 12, "% months", 60 * 60 * 24 * 30.42],
[60 * 60 * 24 * 30.42 * 12 * 2, "a year", null],
[8640000000000000 /* max*/, "% years", 60 * 60 * 24 * 30.42 * 12],
];
let text = null;
@ -87,10 +88,10 @@ function formatRelativeTime(timeString) {
}
}
if (text === 'a day') {
return future ? 'tomorrow' : 'yesterday';
if (text === "a day") {
return future ? "tomorrow" : "yesterday";
}
return future ? 'in ' + text : text + ' ago';
return future ? "in " + text : text + " ago";
}
function formatMarkdown(text) {
@ -102,7 +103,7 @@ function formatInlineMarkdown(text) {
}
function splitByWhitespace(str) {
return str.split(/\s+/).filter(s => s);
return str.split(/\s+/).filter((s) => s);
}
function unindent(callSite, ...args) {
@ -110,28 +111,30 @@ function unindent(callSite, ...args) {
let size = -1;
return str.replace(/\n(\s+)/g, (m, m1) => {
if (size < 0) {
size = m1.replace(/\t/g, ' ').length;
size = m1.replace(/\t/g, " ").length;
}
return '\n' + m1.slice(Math.min(m1.length, size));
return "\n" + m1.slice(Math.min(m1.length, size));
});
}
if (typeof callSite === 'string') {
if (typeof callSite === "string") {
return format(callSite);
}
if (typeof callSite === 'function') {
if (typeof callSite === "function") {
return (...args) => format(callSite(...args));
}
let output = callSite
.slice(0, args.length + 1)
.map((text, i) => (i === 0 ? '' : args[i - 1]) + text)
.join('');
.map((text, i) => (i === 0 ? "" : args[i - 1]) + text)
.join("");
return format(output);
}
function enableExitConfirmation() {
window.onbeforeunload = e => {
return 'Are you sure you want to leave? ' +
'Data you have entered may not be saved.';
window.onbeforeunload = (e) => {
return (
"Are you sure you want to leave? " +
"Data you have entered may not be saved."
);
};
}
@ -150,16 +153,17 @@ function confirmPageExit() {
}
function makeCssName(text, suffix) {
return suffix + '-' + text.replace(/[^a-z0-9]/g, '_');
return suffix + "-" + text.replace(/[^a-z0-9]/g, "_");
}
function escapeHtml(unsafe) {
return unsafe.toString()
.replace(/&/g, '&amp;')
.replace(/</g, '&lt;')
.replace(/>/g, '&gt;')
.replace(/"/g, '&quot;')
.replace(/'/g, '&apos;');
return unsafe
.toString()
.replace(/&/g, "&amp;")
.replace(/</g, "&lt;")
.replace(/>/g, "&gt;")
.replace(/"/g, "&quot;")
.replace(/'/g, "&apos;");
}
function arraysDiffer(source1, source2, orderImportant) {
@ -177,20 +181,22 @@ function arraysDiffer(source1, source2, orderImportant) {
return false;
}
return (
source1.filter(value => !source2.includes(value)).length > 0 ||
source2.filter(value => !source1.includes(value)).length > 0);
source1.filter((value) => !source2.includes(value)).length > 0 ||
source2.filter((value) => !source1.includes(value)).length > 0
);
}
function escapeSearchTerm(text) {
return text.replace(/([a-z_-]):/g, '$1\\:');
return text.replace(/([a-z_-]):/g, "$1\\:");
}
function dataURItoBlob(dataURI) {
const chunks = dataURI.split(',');
const byteString = chunks[0].indexOf('base64') >= 0 ?
window.atob(chunks[1]) :
unescape(chunks[1]);
const mimeString = chunks[0].split(':')[1].split(';')[0];
const chunks = dataURI.split(",");
const byteString =
chunks[0].indexOf("base64") >= 0
? window.atob(chunks[1])
: unescape(chunks[1]);
const mimeString = chunks[0].split(":")[1].split(";")[0];
const data = new Uint8Array(byteString.length);
for (let i = 0; i < byteString.length; i++) {
data[i] = byteString.charCodeAt(i);

View file

@ -1,4 +1,4 @@
'use strict';
"use strict";
let callbacks = [];
let running = false;
@ -15,7 +15,7 @@ function resize() {
}
function runCallbacks() {
callbacks.forEach(callback => {
callbacks.forEach((callback) => {
callback();
});
running = false;
@ -26,8 +26,8 @@ function add(callback) {
}
function remove(callback) {
callbacks = callbacks.filter(c => c !== callback);
callbacks = callbacks.filter((c) => c !== callback);
}
window.addEventListener('resize', resize);
window.addEventListener("resize", resize);
module.exports = { add: add, remove: remove };

View file

@ -1,6 +1,6 @@
/* eslint-disable func-names, no-extend-native */
'use strict';
"use strict";
// fix iterating over NodeList in Chrome and Opera
NodeList.prototype[Symbol.iterator] = Array.prototype[Symbol.iterator];
@ -42,9 +42,7 @@ Node.prototype.prependChild = function(child) {
// non standard
Promise.prototype.always = function (onResolveOrReject) {
return this.then(
onResolveOrReject,
reason => {
return this.then(onResolveOrReject, (reason) => {
onResolveOrReject(reason);
throw reason;
});
@ -54,9 +52,7 @@ Promise.prototype.always = function(onResolveOrReject) {
Number.prototype.between = function (a, b, inclusive) {
const min = Math.min(a, b);
const max = Math.max(a, b);
return inclusive ?
this >= min && this <= max :
this > min && this < max;
return inclusive ? this >= min && this <= max : this > min && this < max;
};
// non standard

View file

@ -1,6 +1,6 @@
'use strict';
"use strict";
const nprogress = require('nprogress');
const nprogress = require("nprogress");
let nesting = 0;

View file

@ -1,14 +1,16 @@
'use strict';
"use strict";
const misc = require('./misc.js');
const keyboard = require('../util/keyboard.js');
const views = require('./views.js');
const misc = require("./misc.js");
const keyboard = require("../util/keyboard.js");
const views = require("./views.js");
function searchInputNodeFocusHelper(inputNode) {
keyboard.bind('q', () => {
keyboard.bind("q", () => {
inputNode.focus();
inputNode.setSelectionRange(
inputNode.value.length, inputNode.value.length);
inputNode.value.length,
inputNode.value.length
);
});
}

View file

@ -1,11 +1,11 @@
'use strict';
"use strict";
const direction = {
NONE: null,
LEFT: 'left',
RIGHT: 'right',
DOWN: 'down',
UP: 'up'
LEFT: "left",
RIGHT: "right",
DOWN: "down",
UP: "up",
};
function handleTouchStart(handler, evt) {
@ -58,11 +58,13 @@ function handleTouchEnd(handler) {
}
class Touch {
constructor(target,
constructor(
target,
swipeLeft = () => {},
swipeRight = () => {},
swipeUp = () => {},
swipeDown = () => {}) {
swipeDown = () => {}
) {
this._target = target;
this._swipeLeftTask = swipeLeft;
@ -74,16 +76,13 @@ class Touch {
this._yStart = null;
this._direction = direction.NONE;
this._target.addEventListener('touchstart',
evt => {
this._target.addEventListener("touchstart", (evt) => {
handleTouchStart(this, evt);
});
this._target.addEventListener('touchmove',
evt => {
this._target.addEventListener("touchmove", (evt) => {
handleTouchMove(this, evt);
});
this._target.addEventListener('touchend',
() => {
this._target.addEventListener("touchend", () => {
handleTouchEnd(this);
});
}

View file

@ -1,4 +1,4 @@
'use strict';
"use strict";
function formatApiLink(...values) {
let parts = [];
@ -9,18 +9,19 @@ function formatApiLink(...values) {
for (let key of Object.keys(value)) {
if (value[key]) {
variableParts.push(
key + '=' + encodeURIComponent(value[key].toString()));
key + "=" + encodeURIComponent(value[key].toString())
);
}
}
if (variableParts.length) {
parts.push('?' + variableParts.join('&'));
parts.push("?" + variableParts.join("&"));
}
break;
} else {
parts.push(encodeURIComponent(value.toString()));
}
}
return '/' + parts.join('/');
return "/" + parts.join("/");
}
function escapeParam(text) {
@ -40,48 +41,52 @@ function formatClientLink(...values) {
for (let key of Object.keys(value)) {
if (value[key]) {
variableParts.push(
key + '=' + escapeParam(value[key].toString()));
key + "=" + escapeParam(value[key].toString())
);
}
}
if (variableParts.length) {
parts.push(variableParts.join(';'));
parts.push(variableParts.join(";"));
}
break;
} else {
parts.push(escapeParam(value.toString()));
}
}
return parts.join('/');
return parts.join("/");
}
function extractHostname(url) {
// https://stackoverflow.com/a/23945027
return url
.split('/')[url.indexOf("//") > -1 ? 2 : 0]
.split(':')[0]
.split('?')[0];
.split("/")
[url.indexOf("//") > -1 ? 2 : 0].split(":")[0]
.split("?")[0];
}
function extractRootDomain(url) {
// https://stackoverflow.com/a/23945027
let domain = extractHostname(url);
let splitArr = domain.split('.');
let splitArr = domain.split(".");
let arrLen = splitArr.length;
// if there is a subdomain
if (arrLen > 2) {
domain = splitArr[arrLen - 2] + '.' + splitArr[arrLen - 1];
domain = splitArr[arrLen - 2] + "." + splitArr[arrLen - 1];
// check to see if it's using a Country Code Top Level Domain (ccTLD) (i.e. ".me.uk")
if (splitArr[arrLen - 2].length === 2 && splitArr[arrLen - 1].length === 2) {
if (
splitArr[arrLen - 2].length === 2 &&
splitArr[arrLen - 1].length === 2
) {
// this is using a ccTLD
domain = splitArr[arrLen - 3] + '.' + domain;
domain = splitArr[arrLen - 3] + "." + domain;
}
}
return domain;
}
function escapeColons(text) {
return text.replace(new RegExp(':', 'g'), '\\:');
return text.replace(new RegExp(":", "g"), "\\:");
}
module.exports = {

View file

@ -1,27 +1,27 @@
'use strict';
"use strict";
require('../util/polyfill.js');
const api = require('../api.js');
const templates = require('../templates.js');
require("../util/polyfill.js");
const api = require("../api.js");
const templates = require("../templates.js");
const domParser = new DOMParser();
const misc = require('./misc.js');
const uri = require('./uri.js');
const misc = require("./misc.js");
const uri = require("./uri.js");
function _imbueId(options) {
if (!options.id) {
options.id = 'gen-' + Math.random().toString(36).substring(7);
options.id = "gen-" + Math.random().toString(36).substring(7);
}
}
function _makeLabel(options, attrs) {
if (!options.text) {
return '';
return "";
}
if (!attrs) {
attrs = {};
}
attrs.for = options.id;
return makeElement('label', attrs, options.text);
return makeElement("label", attrs, options.text);
}
function makeFileSize(fileSize) {
@ -34,251 +34,282 @@ function makeMarkdown(text) {
function makeRelativeTime(time) {
return makeElement(
'time', {datetime: time, title: time}, misc.formatRelativeTime(time));
"time",
{ datetime: time, title: time },
misc.formatRelativeTime(time)
);
}
function makeThumbnail(url) {
return makeElement(
'span',
url ?
{class: 'thumbnail', style: `background-image: url(\'${url}\')`} :
{class: 'thumbnail empty'},
makeElement('img', {alt: 'thumbnail', src: url}));
"span",
url
? {
class: "thumbnail",
style: `background-image: url(\'${url}\')`,
}
: { class: "thumbnail empty" },
makeElement("img", { alt: "thumbnail", src: url })
);
}
function makeRadio(options) {
_imbueId(options);
return makeElement(
'label',
"label",
{ for: options.id },
makeElement(
'input',
{
makeElement("input", {
id: options.id,
name: options.name,
value: options.value,
type: 'radio',
type: "radio",
checked: options.selectedValue === options.value,
disabled: options.readonly,
required: options.required,
}),
makeElement('span', {class: 'radio'}, options.text));
makeElement("span", { class: "radio" }, options.text)
);
}
function makeCheckbox(options) {
_imbueId(options);
return makeElement(
'label',
"label",
{ for: options.id },
makeElement(
'input',
{
makeElement("input", {
id: options.id,
name: options.name,
value: options.value,
type: 'checkbox',
checked: options.checked !== undefined ?
options.checked : false,
type: "checkbox",
checked: options.checked !== undefined ? options.checked : false,
disabled: options.readonly,
required: options.required,
}),
makeElement('span', {class: 'checkbox'}, options.text));
makeElement("span", { class: "checkbox" }, options.text)
);
}
function makeSelect(options) {
return _makeLabel(options) +
return (
_makeLabel(options) +
makeElement(
'select',
"select",
{
id: options.id,
name: options.name,
disabled: options.readonly,
},
...Object.keys(options.keyValues).map(key => makeElement(
'option',
...Object.keys(options.keyValues).map((key) =>
makeElement(
"option",
{ value: key, selected: key === options.selectedKey },
options.keyValues[key])));
options.keyValues[key]
)
)
)
);
}
function makeInput(options) {
options.value = options.value || '';
return _makeLabel(options) + makeElement('input', options);
options.value = options.value || "";
return _makeLabel(options) + makeElement("input", options);
}
function makeButton(options) {
options.type = 'button';
options.type = "button";
return makeInput(options);
}
function makeTextInput(options) {
options.type = 'text';
options.type = "text";
return makeInput(options);
}
function makeTextarea(options) {
const value = options.value || '';
const value = options.value || "";
delete options.value;
return _makeLabel(options) + makeElement('textarea', options, value);
return _makeLabel(options) + makeElement("textarea", options, value);
}
function makePasswordInput(options) {
options.type = 'password';
options.type = "password";
return makeInput(options);
}
function makeEmailInput(options) {
options.type = 'email';
options.type = "email";
return makeInput(options);
}
function makeColorInput(options) {
const textInput = makeElement(
'input', {
type: 'text',
value: options.value || '',
const textInput = makeElement("input", {
type: "text",
value: options.value || "",
required: options.required,
class: 'color',
class: "color",
});
const backgroundPreviewNode = makeElement(
'div',
{
class: 'preview background-preview',
style:
`border-color: ${options.value};
const backgroundPreviewNode = makeElement("div", {
class: "preview background-preview",
style: `border-color: ${options.value};
background-color: ${options.value}`,
});
const textPreviewNode = makeElement(
'div',
{
class: 'preview text-preview',
style:
`border-color: ${options.value};
const textPreviewNode = makeElement("div", {
class: "preview text-preview",
style: `border-color: ${options.value};
color: ${options.value}`,
});
return makeElement(
'label', {class: 'color'}, textInput, backgroundPreviewNode, textPreviewNode);
"label",
{ class: "color" },
textInput,
backgroundPreviewNode,
textPreviewNode
);
}
function makeNumericInput(options) {
options.type = 'number';
options.type = "number";
return makeInput(options);
}
function makeDateInput(options) {
options.type = 'date';
return makeInput(options)
options.type = "date";
return makeInput(options);
}
function getPostUrl(id, parameters) {
return uri.formatClientLink(
'post', id, parameters ? {query: parameters.query} : {});
"post",
id,
parameters ? { query: parameters.query } : {}
);
}
function getPostEditUrl(id, parameters) {
return uri.formatClientLink(
'post', id, 'edit', parameters ? {query: parameters.query} : {});
"post",
id,
"edit",
parameters ? { query: parameters.query } : {}
);
}
function makePostLink(id, includeHash) {
let text = id;
if (includeHash) {
text = '@' + id;
text = "@" + id;
}
return api.hasPrivilege('posts:view') ?
makeElement(
'a',
{href: uri.formatClientLink('post', id)},
misc.escapeHtml(text)) :
misc.escapeHtml(text);
return api.hasPrivilege("posts:view")
? makeElement(
"a",
{ href: uri.formatClientLink("post", id) },
misc.escapeHtml(text)
)
: misc.escapeHtml(text);
}
function makeTagLink(name, includeHash, includeCount, tag) {
const category = tag ? tag.category : 'unknown';
const category = tag ? tag.category : "unknown";
let text = misc.getPrettyTagName(name);
if (includeHash === true) {
text = '#' + text;
text = "#" + text;
}
if (includeCount === true) {
text += ' (' + (tag ? tag.postCount : 0) + ')';
text += " (" + (tag ? tag.postCount : 0) + ")";
}
return api.hasPrivilege('tags:view') ?
makeElement(
'a',
return api.hasPrivilege("tags:view")
? makeElement(
"a",
{
href: uri.formatClientLink('tag', name),
class: misc.makeCssName(category, 'tag'),
href: uri.formatClientLink("tag", name),
class: misc.makeCssName(category, "tag"),
},
misc.escapeHtml(text)) :
makeElement(
'span',
{class: misc.makeCssName(category, 'tag')},
misc.escapeHtml(text));
misc.escapeHtml(text)
)
: makeElement(
"span",
{ class: misc.makeCssName(category, "tag") },
misc.escapeHtml(text)
);
}
function makePoolLink(id, includeHash, includeCount, pool, name) {
const category = pool ? pool.category : 'unknown';
const category = pool ? pool.category : "unknown";
let text = name ? name : pool.names[0];
if (includeHash === true) {
text = '#' + text;
text = "#" + text;
}
if (includeCount === true) {
text += ' (' + (pool ? pool.postCount : 0) + ')';
text += " (" + (pool ? pool.postCount : 0) + ")";
}
return api.hasPrivilege('pools:view') ?
makeElement(
'a',
return api.hasPrivilege("pools:view")
? makeElement(
"a",
{
href: uri.formatClientLink('pool', id),
class: misc.makeCssName(category, 'pool'),
href: uri.formatClientLink("pool", id),
class: misc.makeCssName(category, "pool"),
},
misc.escapeHtml(text)) :
makeElement(
'span',
{class: misc.makeCssName(category, 'pool')},
misc.escapeHtml(text));
misc.escapeHtml(text)
)
: makeElement(
"span",
{ class: misc.makeCssName(category, "pool") },
misc.escapeHtml(text)
);
}
function makeUserLink(user) {
let text = makeThumbnail(user ? user.avatarUrl : null);
text += user && user.name ? misc.escapeHtml(user.name) : 'Anonymous';
const link = user && api.hasPrivilege('users:view') ?
makeElement(
'a', {href: uri.formatClientLink('user', user.name)}, text) :
text;
return makeElement('span', {class: 'user'}, link);
text += user && user.name ? misc.escapeHtml(user.name) : "Anonymous";
const link =
user && api.hasPrivilege("users:view")
? makeElement(
"a",
{ href: uri.formatClientLink("user", user.name) },
text
)
: text;
return makeElement("span", { class: "user" }, link);
}
function makeFlexboxAlign(options) {
return [...misc.range(20)]
.map(() => '<li class="flexbox-dummy"></li>').join('');
.map(() => '<li class="flexbox-dummy"></li>')
.join("");
}
function makeAccessKey(html, key) {
const regex = new RegExp('(' + key + ')', 'i');
const regex = new RegExp("(" + key + ")", "i");
html = html.replace(
regex, '<span class="access-key" data-accesskey="$1">$1</span>');
regex,
'<span class="access-key" data-accesskey="$1">$1</span>'
);
return html;
}
function _serializeElement(name, attributes) {
return [name]
.concat(Object.keys(attributes).map(key => {
.concat(
Object.keys(attributes).map((key) => {
if (attributes[key] === true) {
return key;
} else if (attributes[key] === false ||
attributes[key] === undefined) {
return '';
} else if (
attributes[key] === false ||
attributes[key] === undefined
) {
return "";
}
const attribute = misc.escapeHtml(attributes[key] || '');
const attribute = misc.escapeHtml(attributes[key] || "");
return `${key}="${attribute}"`;
}))
.join(' ');
})
)
.join(" ");
}
function makeElement(name, attrs, ...content) {
return content.length !== undefined ?
`<${_serializeElement(name, attrs)}>${content.join('')}</${name}>` :
`<${_serializeElement(name, attrs)}/>`;
return content.length !== undefined
? `<${_serializeElement(name, attrs)}>${content.join("")}</${name}>`
: `<${_serializeElement(name, attrs)}/>`;
}
function emptyContent(target) {
@ -302,25 +333,25 @@ function replaceContent(target, source) {
function showMessage(target, message, className) {
if (!message) {
message = 'Unknown message';
message = "Unknown message";
}
const messagesHolderNode = target.querySelector('.messages');
const messagesHolderNode = target.querySelector(".messages");
if (!messagesHolderNode) {
return false;
}
const textNode = document.createElement('div');
textNode.innerHTML = message.replace(/\n/g, '<br/>');
textNode.classList.add('message');
const textNode = document.createElement("div");
textNode.innerHTML = message.replace(/\n/g, "<br/>");
textNode.classList.add("message");
textNode.classList.add(className);
const wrapperNode = document.createElement('div');
wrapperNode.classList.add('message-wrapper');
const wrapperNode = document.createElement("div");
wrapperNode.classList.add("message-wrapper");
wrapperNode.appendChild(textNode);
messagesHolderNode.appendChild(wrapperNode);
return true;
}
function appendExclamationMark() {
if (!document.title.startsWith('!')) {
if (!document.title.startsWith("!")) {
document.oldTitle = document.title;
document.title = `! ${document.title}`;
}
@ -328,15 +359,15 @@ function appendExclamationMark() {
function showError(target, message) {
appendExclamationMark();
return showMessage(target, misc.formatInlineMarkdown(message), 'error');
return showMessage(target, misc.formatInlineMarkdown(message), "error");
}
function showSuccess(target, message) {
return showMessage(target, misc.formatInlineMarkdown(message), 'success');
return showMessage(target, misc.formatInlineMarkdown(message), "success");
}
function showInfo(target, message) {
return showMessage(target, misc.formatInlineMarkdown(message), 'info');
return showMessage(target, misc.formatInlineMarkdown(message), "info");
}
function clearMessages(target) {
@ -344,7 +375,7 @@ function clearMessages(target) {
document.title = document.oldTitle;
document.oldTitle = null;
}
for (let messagesHolderNode of target.querySelectorAll('.messages')) {
for (let messagesHolderNode of target.querySelectorAll(".messages")) {
emptyContent(messagesHolderNode);
}
}
@ -352,15 +383,15 @@ function clearMessages(target) {
function htmlToDom(html) {
// code taken from jQuery + Krasimir Tsonev's blog
const wrapMap = {
_: [1, '<div>', '</div>'],
option: [1, '<select multiple>', '</select>'],
legend: [1, '<fieldset>', '</fieldset>'],
area: [1, '<map>', '</map>'],
param: [1, '<object>', '</object>'],
thead: [1, '<table>', '</table>'],
tr: [2, '<table><tbody>', '</tbody></table>'],
td: [3, '<table><tbody><tr>', '</tr></tbody></table>'],
col: [2, '<table><tbody></tbody><colgroup>', '</colgroup></table>'],
_: [1, "<div>", "</div>"],
option: [1, "<select multiple>", "</select>"],
legend: [1, "<fieldset>", "</fieldset>"],
area: [1, "<map>", "</map>"],
param: [1, "<object>", "</object>"],
thead: [1, "<table>", "</table>"],
tr: [2, "<table><tbody>", "</tbody></table>"],
td: [3, "<table><tbody><tr>", "</tr></tbody></table>"],
col: [2, "<table><tbody></tbody><colgroup>", "</colgroup></table>"],
};
wrapMap.optgroup = wrapMap.option;
wrapMap.tbody = wrapMap.thead;
@ -369,8 +400,8 @@ function htmlToDom(html) {
wrapMap.caption = wrapMap.thead;
wrapMap.th = wrapMap.td;
let element = document.createElement('div');
const match = (/<\s*(\w+)[^>]*?>/g).exec(html);
let element = document.createElement("div");
const match = /<\s*(\w+)[^>]*?>/g.exec(html);
if (match) {
const tag = match[1];
@ -382,9 +413,9 @@ function htmlToDom(html) {
} else {
element.innerHTML = html;
}
return element.childNodes.length > 1 ?
element.childNodes :
element.firstChild;
return element.childNodes.length > 1
? element.childNodes
: element.firstChild;
}
function getTemplate(templatePath) {
@ -392,7 +423,7 @@ function getTemplate(templatePath) {
throw `Missing template: ${templatePath}`;
}
const templateFactory = templates[templatePath];
return ctx => {
return (ctx) => {
if (!ctx) {
ctx = {};
}
@ -423,7 +454,7 @@ function getTemplate(templatePath) {
makeElement: makeElement,
makeCssName: misc.makeCssName,
makeNumericInput: makeNumericInput,
formatClientLink: uri.formatClientLink
formatClientLink: uri.formatClientLink,
});
return htmlToDom(templateFactory(ctx));
};
@ -432,36 +463,38 @@ function getTemplate(templatePath) {
function decorateValidator(form) {
// postpone showing form fields validity until user actually tries
// to submit it (seeing red/green form w/o doing anything breaks POLA)
let submitButton = form.querySelector('.buttons input');
let submitButton = form.querySelector(".buttons input");
if (!submitButton) {
submitButton = form.querySelector('input[type=submit]');
submitButton = form.querySelector("input[type=submit]");
}
if (submitButton) {
submitButton.addEventListener('click', e => {
form.classList.add('show-validation');
submitButton.addEventListener("click", (e) => {
form.classList.add("show-validation");
});
}
form.addEventListener('submit', e => {
form.classList.remove('show-validation');
form.addEventListener("submit", (e) => {
form.classList.remove("show-validation");
});
}
function disableForm(form) {
for (let input of form.querySelectorAll('input')) {
for (let input of form.querySelectorAll("input")) {
input.disabled = true;
}
}
function enableForm(form) {
for (let input of form.querySelectorAll('input')) {
for (let input of form.querySelectorAll("input")) {
input.disabled = false;
}
}
function syncScrollPosition() {
window.requestAnimationFrame(
() => {
if (history.state && Object.prototype.hasOwnProperty.call(history.state, 'scrollX')) {
window.requestAnimationFrame(() => {
if (
history.state &&
Object.prototype.hasOwnProperty.call(history.state, "scrollX")
) {
window.scrollTo(history.state.scrollX, history.state.scrollY);
} else {
window.scrollTo(0, 0);
@ -473,8 +506,8 @@ function slideDown(element) {
const duration = 500;
return new Promise((resolve, reject) => {
const height = element.getBoundingClientRect().height;
element.style.maxHeight = '0';
element.style.overflow = 'hidden';
element.style.maxHeight = "0";
element.style.overflow = "hidden";
window.setTimeout(() => {
element.style.transition = `all ${duration}ms ease`;
element.style.maxHeight = `${height}px`;
@ -489,7 +522,7 @@ function slideUp(element) {
const duration = 500;
return new Promise((resolve, reject) => {
const height = element.getBoundingClientRect().height;
element.style.overflow = 'hidden';
element.style.overflow = "hidden";
element.style.maxHeight = `${height}px`;
element.style.transition = `all ${duration}ms ease`;
window.setTimeout(() => {
@ -502,8 +535,7 @@ function slideUp(element) {
}
function monitorNodeRemoval(monitoredNode, callback) {
const mutationObserver = new MutationObserver(
mutations => {
const mutationObserver = new MutationObserver((mutations) => {
for (let mutation of mutations) {
for (let node of mutation.removedNodes) {
if (node.contains(monitoredNode)) {
@ -514,14 +546,16 @@ function monitorNodeRemoval(monitoredNode, callback) {
}
}
});
mutationObserver.observe(
document.body, {childList: true, subtree: true});
mutationObserver.observe(document.body, {
childList: true,
subtree: true,
});
}
document.addEventListener('input', e => {
if (e.target.classList.contains('color')) {
let bkNode = e.target.parentNode.querySelector('.background-preview');
let textNode = e.target.parentNode.querySelector('.text-preview');
document.addEventListener("input", (e) => {
if (e.target.classList.contains("color")) {
let bkNode = e.target.parentNode.querySelector(".background-preview");
let textNode = e.target.parentNode.querySelector(".text-preview");
bkNode.style.backgroundColor = e.target.value;
bkNode.style.borderColor = e.target.value;
textNode.style.color = e.target.value;
@ -530,8 +564,8 @@ document.addEventListener('input', e => {
});
// prevent opening buttons in new tabs
document.addEventListener('click', e => {
if (e.target.getAttribute('href') === '' && e.which === 2) {
document.addEventListener("click", (e) => {
if (e.target.getAttribute("href") === "" && e.which === 2) {
e.preventDefault();
}
});

View file

@ -1,10 +1,10 @@
'use strict';
"use strict";
const events = require('../events.js');
const views = require('../util/views.js');
const CommentListControl = require('../controls/comment_list_control.js');
const events = require("../events.js");
const views = require("../util/views.js");
const CommentListControl = require("../controls/comment_list_control.js");
const template = views.getTemplate('comments-page');
const template = views.getTemplate("comments-page");
class CommentsPageView extends events.EventTarget {
constructor(ctx) {
@ -16,12 +16,14 @@ class CommentsPageView extends events.EventTarget {
for (let post of ctx.response.results) {
const commentListControl = new CommentListControl(
sourceNode.querySelector(
`.comments-container[data-for="${post.id}"]`),
`.comments-container[data-for="${post.id}"]`
),
post.comments,
true);
events.proxyEvent(commentListControl, this, 'submit');
events.proxyEvent(commentListControl, this, 'score');
events.proxyEvent(commentListControl, this, 'delete');
true
);
events.proxyEvent(commentListControl, this, "submit");
events.proxyEvent(commentListControl, this, "score");
events.proxyEvent(commentListControl, this, "delete");
}
views.replaceContent(this._hostNode, sourceNode);

View file

@ -1,15 +1,16 @@
'use strict';
"use strict";
const views = require('../util/views.js');
const views = require("../util/views.js");
const template = () => {
return views.htmlToDom(
'<div class="wrapper"><div class="messages"></div></div>');
'<div class="wrapper"><div class="messages"></div></div>'
);
};
class EmptyView {
constructor() {
this._hostNode = document.getElementById('content-holder');
this._hostNode = document.getElementById("content-holder");
views.replaceContent(this._hostNode, template());
views.syncScrollPosition();
}

View file

@ -1,10 +1,10 @@
'use strict';
"use strict";
const router = require('../router.js');
const views = require('../util/views.js');
const router = require("../router.js");
const views = require("../util/views.js");
const holderTemplate = views.getTemplate('endless-pager');
const pageTemplate = views.getTemplate('endless-pager-page');
const holderTemplate = views.getTemplate("endless-pager");
const pageTemplate = views.getTemplate("endless-pager-page");
function isScrolledIntoView(element) {
let top = 0;
@ -12,14 +12,12 @@ function isScrolledIntoView(element) {
top += element.offsetTop || 0;
element = element.offsetParent;
} while (element);
return (
(top >= window.scrollY) &&
(top <= window.scrollY + window.innerHeight));
return top >= window.scrollY && top <= window.scrollY + window.innerHeight;
}
class EndlessPageView {
constructor(ctx) {
this._hostNode = document.getElementById('content-holder');
this._hostNode = document.getElementById("content-holder");
views.replaceContent(this._hostNode, holderTemplate());
}
@ -40,12 +38,13 @@ class EndlessPageView {
this.defaultLimit = parseInt(ctx.parameters.limit || ctx.defaultLimit);
const initialOffset = parseInt(ctx.parameters.offset || 0);
this._loadPage(ctx, initialOffset, this.defaultLimit, true)
.then(pageNode => {
this._loadPage(ctx, initialOffset, this.defaultLimit, true).then(
(pageNode) => {
if (initialOffset !== 0) {
pageNode.scrollIntoView();
}
});
}
);
this._timeout = window.setInterval(() => {
window.requestAnimationFrame(() => {
@ -58,19 +57,19 @@ class EndlessPageView {
}
get pageHeaderHolderNode() {
return this._hostNode.querySelector('.page-header-holder');
return this._hostNode.querySelector(".page-header-holder");
}
get topPageGuardNode() {
return this._hostNode.querySelector('.page-guard.top');
return this._hostNode.querySelector(".page-guard.top");
}
get bottomPageGuardNode() {
return this._hostNode.querySelector('.page-guard.bottom');
return this._hostNode.querySelector(".page-guard.bottom");
}
get _pagesHolderNode() {
return this._hostNode.querySelector('.pages-holder');
return this._hostNode.querySelector(".pages-holder");
}
_destroy() {
@ -82,9 +81,10 @@ class EndlessPageView {
let topPageNode = null;
let element = document.elementFromPoint(
window.innerWidth / 2,
window.innerHeight / 2);
window.innerHeight / 2
);
while (element.parentNode !== null) {
if (element.classList.contains('page')) {
if (element.classList.contains("page")) {
topPageNode = element;
break;
}
@ -93,15 +93,17 @@ class EndlessPageView {
if (!topPageNode) {
return;
}
let topOffset = parseInt(topPageNode.getAttribute('data-offset'));
let topLimit = parseInt(topPageNode.getAttribute('data-limit'));
let topOffset = parseInt(topPageNode.getAttribute("data-offset"));
let topLimit = parseInt(topPageNode.getAttribute("data-limit"));
if (topOffset !== this.currentOffset) {
router.replace(
ctx.getClientUrlForPage(
topOffset,
topLimit === ctx.defaultLimit ? null : topLimit),
topLimit === ctx.defaultLimit ? null : topLimit
),
ctx.state,
false);
false
);
this.currentOffset = topOffset;
}
}
@ -115,29 +117,31 @@ class EndlessPageView {
return;
}
if (this.minOffsetShown > 0 &&
isScrolledIntoView(this.topPageGuardNode)) {
if (
this.minOffsetShown > 0 &&
isScrolledIntoView(this.topPageGuardNode)
) {
this._loadPage(
ctx,
this.minOffsetShown - this.defaultLimit,
this.defaultLimit,
false);
false
);
}
if (this.maxOffsetShown < this.totalRecords &&
isScrolledIntoView(this.bottomPageGuardNode)) {
this._loadPage(
ctx,
this.maxOffsetShown,
this.defaultLimit,
true);
if (
this.maxOffsetShown < this.totalRecords &&
isScrolledIntoView(this.bottomPageGuardNode)
) {
this._loadPage(ctx, this.maxOffsetShown, this.defaultLimit, true);
}
}
_loadPage(ctx, offset, limit, append) {
this._runningRequests++;
return new Promise((resolve, reject) => {
ctx.requestPage(offset, limit).then(response => {
ctx.requestPage(offset, limit).then(
(response) => {
if (!this._active) {
this._runningRequests--;
return Promise.reject();
@ -147,11 +151,13 @@ class EndlessPageView {
this._runningRequests--;
resolve(pageNode);
});
}, error => {
},
(error) => {
this.showError(error.message);
this._runningRequests--;
reject();
});
}
);
});
}
@ -162,30 +168,35 @@ class EndlessPageView {
pageNode = pageTemplate({
totalPages: Math.ceil(response.total / response.limit),
page: Math.ceil(
(response.offset + response.limit) / response.limit),
(response.offset + response.limit) / response.limit
),
});
pageNode.setAttribute('data-offset', response.offset);
pageNode.setAttribute('data-limit', response.limit);
pageNode.setAttribute("data-offset", response.offset);
pageNode.setAttribute("data-limit", response.limit);
ctx.pageRenderer({
parameters: ctx.parameters,
response: response,
hostNode: pageNode.querySelector('.page-content-holder'),
hostNode: pageNode.querySelector(".page-content-holder"),
});
this.totalRecords = response.total;
if (response.offset < this.minOffsetShown ||
this.minOffsetShown === null) {
if (
response.offset < this.minOffsetShown ||
this.minOffsetShown === null
) {
this.minOffsetShown = response.offset;
}
if (response.offset + response.results.length
> this.maxOffsetShown ||
this.maxOffsetShown === null) {
if (
response.offset + response.results.length >
this.maxOffsetShown ||
this.maxOffsetShown === null
) {
this.maxOffsetShown =
response.offset + response.results.length;
}
response.results.addEventListener('remove', e => {
response.results.addEventListener("remove", (e) => {
this.maxOffsetShown--;
this.totalRecords--;
});
@ -200,10 +211,11 @@ class EndlessPageView {
window.scroll(
window.scrollX,
window.scrollY + pageNode.offsetHeight);
window.scrollY + pageNode.offsetHeight
);
}
} else if (!response.results.length) {
this.showInfo('No data to show');
this.showInfo("No data to show");
}
this._initialPageLoad = false;

View file

@ -1,73 +1,81 @@
'use strict';
"use strict";
const api = require('../api.js');
const views = require('../util/views.js');
const api = require("../api.js");
const views = require("../util/views.js");
const template = views.getTemplate('help');
const template = views.getTemplate("help");
const sectionTemplates = {
'about': views.getTemplate('help-about'),
'keyboard': views.getTemplate('help-keyboard'),
'search': views.getTemplate('help-search'),
'comments': views.getTemplate('help-comments'),
'tos': views.getTemplate('help-tos'),
about: views.getTemplate("help-about"),
keyboard: views.getTemplate("help-keyboard"),
search: views.getTemplate("help-search"),
comments: views.getTemplate("help-comments"),
tos: views.getTemplate("help-tos"),
};
const subsectionTemplates = {
'search': {
'default': views.getTemplate('help-search-general'),
'posts': views.getTemplate('help-search-posts'),
'users': views.getTemplate('help-search-users'),
'tags': views.getTemplate('help-search-tags'),
'pools': views.getTemplate('help-search-pools'),
search: {
default: views.getTemplate("help-search-general"),
posts: views.getTemplate("help-search-posts"),
users: views.getTemplate("help-search-users"),
tags: views.getTemplate("help-search-tags"),
pools: views.getTemplate("help-search-pools"),
},
};
class HelpView {
constructor(section, subsection) {
this._hostNode = document.getElementById('content-holder');
this._hostNode = document.getElementById("content-holder");
const sourceNode = template();
const ctx = {
name: api.getName(),
};
section = section || 'about';
section = section || "about";
if (section in sectionTemplates) {
views.replaceContent(
sourceNode.querySelector('.content'),
sectionTemplates[section](ctx));
sourceNode.querySelector(".content"),
sectionTemplates[section](ctx)
);
}
subsection = subsection || 'default';
if (section in subsectionTemplates &&
subsection in subsectionTemplates[section]) {
subsection = subsection || "default";
if (
section in subsectionTemplates &&
subsection in subsectionTemplates[section]
) {
views.replaceContent(
sourceNode.querySelector('.subcontent'),
subsectionTemplates[section][subsection](ctx));
sourceNode.querySelector(".subcontent"),
subsectionTemplates[section][subsection](ctx)
);
}
views.replaceContent(this._hostNode, sourceNode);
for (let itemNode of
sourceNode.querySelectorAll('.primary [data-name]')) {
for (let itemNode of sourceNode.querySelectorAll(
".primary [data-name]"
)) {
itemNode.classList.toggle(
'active',
itemNode.getAttribute('data-name') === section);
if (itemNode.getAttribute('data-name') === section) {
"active",
itemNode.getAttribute("data-name") === section
);
if (itemNode.getAttribute("data-name") === section) {
itemNode.parentNode.scrollLeft =
itemNode.getBoundingClientRect().left -
itemNode.parentNode.getBoundingClientRect().left
itemNode.parentNode.getBoundingClientRect().left;
}
}
for (let itemNode of
sourceNode.querySelectorAll('.secondary [data-name]')) {
for (let itemNode of sourceNode.querySelectorAll(
".secondary [data-name]"
)) {
itemNode.classList.toggle(
'active',
itemNode.getAttribute('data-name') === subsection);
if (itemNode.getAttribute('data-name') === subsection) {
"active",
itemNode.getAttribute("data-name") === subsection
);
if (itemNode.getAttribute("data-name") === subsection) {
itemNode.parentNode.scrollLeft =
itemNode.getBoundingClientRect().left -
itemNode.parentNode.getBoundingClientRect().left
itemNode.parentNode.getBoundingClientRect().left;
}
}

View file

@ -1,22 +1,20 @@
'use strict';
"use strict";
const router = require('../router.js');
const uri = require('../util/uri.js');
const misc = require('../util/misc.js');
const views = require('../util/views.js');
const PostContentControl = require('../controls/post_content_control.js');
const PostNotesOverlayControl
= require('../controls/post_notes_overlay_control.js');
const TagAutoCompleteControl =
require('../controls/tag_auto_complete_control.js');
const router = require("../router.js");
const uri = require("../util/uri.js");
const misc = require("../util/misc.js");
const views = require("../util/views.js");
const PostContentControl = require("../controls/post_content_control.js");
const PostNotesOverlayControl = require("../controls/post_notes_overlay_control.js");
const TagAutoCompleteControl = require("../controls/tag_auto_complete_control.js");
const template = views.getTemplate('home');
const footerTemplate = views.getTemplate('home-footer');
const featuredPostTemplate = views.getTemplate('home-featured-post');
const template = views.getTemplate("home");
const footerTemplate = views.getTemplate("home-footer");
const featuredPostTemplate = views.getTemplate("home-featured-post");
class HomeView {
constructor(ctx) {
this._hostNode = document.getElementById('content-holder');
this._hostNode = document.getElementById("content-holder");
this._ctx = ctx;
const sourceNode = template(ctx);
@ -27,11 +25,16 @@ class HomeView {
this._autoCompleteControl = new TagAutoCompleteControl(
this._searchInputNode,
{
confirm: tag => this._autoCompleteControl.replaceSelectedText(
misc.escapeSearchTerm(tag.names[0]), true),
});
this._formNode.addEventListener(
'submit', e => this._evtFormSubmit(e));
confirm: (tag) =>
this._autoCompleteControl.replaceSelectedText(
misc.escapeSearchTerm(tag.names[0]),
true
),
}
);
this._formNode.addEventListener("submit", (e) =>
this._evtFormSubmit(e)
);
}
}
@ -46,59 +49,67 @@ class HomeView {
setStats(stats) {
views.replaceContent(
this._footerContainerNode,
footerTemplate(Object.assign({}, stats, this._ctx)));
footerTemplate(Object.assign({}, stats, this._ctx))
);
}
setFeaturedPost(postInfo) {
views.replaceContent(
this._postInfoContainerNode, featuredPostTemplate(postInfo));
this._postInfoContainerNode,
featuredPostTemplate(postInfo)
);
if (this._postContainerNode && postInfo.featuredPost) {
this._postContentControl = new PostContentControl(
this._postContainerNode,
postInfo.featuredPost,
() => {
return [
window.innerWidth * 0.8,
window.innerHeight * 0.7,
];
return [window.innerWidth * 0.8, window.innerHeight * 0.7];
},
'fit-both');
"fit-both"
);
this._postNotesOverlay = new PostNotesOverlayControl(
this._postContainerNode.querySelector('.post-overlay'),
postInfo.featuredPost);
this._postContainerNode.querySelector(".post-overlay"),
postInfo.featuredPost
);
if (postInfo.featuredPost.type === 'video'
|| postInfo.featuredPost.type === 'flash') {
if (
postInfo.featuredPost.type === "video" ||
postInfo.featuredPost.type === "flash"
) {
this._postContentControl.disableOverlay();
}
}
}
get _footerContainerNode() {
return this._hostNode.querySelector('.footer-container');
return this._hostNode.querySelector(".footer-container");
}
get _postInfoContainerNode() {
return this._hostNode.querySelector('.post-info-container');
return this._hostNode.querySelector(".post-info-container");
}
get _postContainerNode() {
return this._hostNode.querySelector('.post-container');
return this._hostNode.querySelector(".post-container");
}
get _formNode() {
return this._hostNode.querySelector('form');
return this._hostNode.querySelector("form");
}
get _searchInputNode() {
return this._formNode.querySelector('input[name=search-text]');
return this._formNode.querySelector("input[name=search-text]");
}
_evtFormSubmit(e) {
e.preventDefault();
this._searchInputNode.blur();
router.show(uri.formatClientLink('posts', {query: this._searchInputNode.value}));
router.show(
uri.formatClientLink("posts", {
query: this._searchInputNode.value,
})
);
}
}

View file

@ -1,52 +1,63 @@
'use strict';
"use strict";
const events = require('../events.js');
const api = require('../api.js');
const views = require('../util/views.js');
const events = require("../events.js");
const api = require("../api.js");
const views = require("../util/views.js");
const template = views.getTemplate('login');
const template = views.getTemplate("login");
class LoginView extends events.EventTarget {
constructor() {
super();
this._hostNode = document.getElementById('content-holder');
this._hostNode = document.getElementById("content-holder");
views.replaceContent(this._hostNode, template({
views.replaceContent(
this._hostNode,
template({
userNamePattern: api.getUserNameRegex(),
passwordPattern: api.getPasswordRegex(),
canSendMails: api.canSendMails(),
}));
})
);
views.syncScrollPosition();
views.decorateValidator(this._formNode);
this._userNameInputNode.setAttribute('pattern', api.getUserNameRegex());
this._passwordInputNode.setAttribute('pattern', api.getPasswordRegex());
this._formNode.addEventListener('submit', e => {
this._userNameInputNode.setAttribute(
"pattern",
api.getUserNameRegex()
);
this._passwordInputNode.setAttribute(
"pattern",
api.getPasswordRegex()
);
this._formNode.addEventListener("submit", (e) => {
e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', {
this.dispatchEvent(
new CustomEvent("submit", {
detail: {
name: this._userNameInputNode.value,
password: this._passwordInputNode.value,
remember: this._rememberInputNode.checked,
},
}));
})
);
});
}
get _formNode() {
return this._hostNode.querySelector('form');
return this._hostNode.querySelector("form");
}
get _userNameInputNode() {
return this._formNode.querySelector('[name=name]');
return this._formNode.querySelector("[name=name]");
}
get _passwordInputNode() {
return this._formNode.querySelector('[name=password]');
return this._formNode.querySelector("[name=password]");
}
get _rememberInputNode() {
return this._formNode.querySelector('[name=remember-user]');
return this._formNode.querySelector("[name=remember-user]");
}
disableForm() {

View file

@ -1,11 +1,11 @@
'use strict';
"use strict";
const router = require('../router.js');
const keyboard = require('../util/keyboard.js');
const views = require('../util/views.js');
const router = require("../router.js");
const keyboard = require("../util/keyboard.js");
const views = require("../util/views.js");
const holderTemplate = views.getTemplate('manual-pager');
const navTemplate = views.getTemplate('manual-pager-nav');
const holderTemplate = views.getTemplate("manual-pager");
const navTemplate = views.getTemplate("manual-pager-nav");
function _removeConsecutiveDuplicates(a) {
return a.filter((item, pos, ary) => {
@ -22,9 +22,7 @@ function _getVisiblePageNumbers(currentPage, totalPages) {
for (let i = totalPages - threshold; i <= totalPages; i++) {
pagesVisible.push(i);
}
for (let i = currentPage - threshold;
i <= currentPage + threshold;
i++) {
for (let i = currentPage - threshold; i <= currentPage + threshold; i++) {
pagesVisible.push(i);
}
pagesVisible = pagesVisible.filter((item, pos, ary) => {
@ -38,7 +36,12 @@ function _getVisiblePageNumbers(currentPage, totalPages) {
}
function _getPages(
currentPage, pageNumbers, limit, defaultLimit, removedItems) {
currentPage,
pageNumbers,
limit,
defaultLimit,
removedItems
) {
const pages = new Map();
let prevPage = 0;
for (let page of pageNumbers) {
@ -48,8 +51,7 @@ function _getPages(
pages.set(page, {
number: page,
offset:
((page - 1) * limit) -
(page > currentPage ? removedItems : 0),
(page - 1) * limit - (page > currentPage ? removedItems : 0),
limit: limit === defaultLimit ? null : limit,
active: currentPage === page,
});
@ -60,7 +62,7 @@ function _getPages(
class ManualPageView {
constructor(ctx) {
this._hostNode = document.getElementById('content-holder');
this._hostNode = document.getElementById("content-holder");
views.replaceContent(this._hostNode, holderTemplate());
}
@ -70,52 +72,65 @@ class ManualPageView {
this.clearMessages();
views.emptyContent(this._pageNavNode);
ctx.requestPage(offset, limit).then(response => {
ctx.requestPage(offset, limit).then(
(response) => {
ctx.pageRenderer({
parameters: ctx.parameters,
response: response,
hostNode: this._pageContentHolderNode,
});
keyboard.bind(['a', 'left'], () => {
this._navigateToPrevNextPage('prev');
keyboard.bind(["a", "left"], () => {
this._navigateToPrevNextPage("prev");
});
keyboard.bind(['d', 'right'], () => {
this._navigateToPrevNextPage('next');
keyboard.bind(["d", "right"], () => {
this._navigateToPrevNextPage("next");
});
let removedItems = 0;
if (response.total) {
this._refreshNav(
offset, limit, response.total, removedItems, ctx);
offset,
limit,
response.total,
removedItems,
ctx
);
}
if (!response.results.length) {
this.showInfo('No data to show');
this.showInfo("No data to show");
}
response.results.addEventListener('remove', e => {
response.results.addEventListener("remove", (e) => {
removedItems++;
this._refreshNav(
offset, limit, response.total, removedItems, ctx);
offset,
limit,
response.total,
removedItems,
ctx
);
});
views.syncScrollPosition();
}, response => {
},
(response) => {
this.showError(response.message);
});
}
);
}
get pageHeaderHolderNode() {
return this._hostNode.querySelector('.page-header-holder');
return this._hostNode.querySelector(".page-header-holder");
}
get _pageContentHolderNode() {
return this._hostNode.querySelector('.page-content-holder');
return this._hostNode.querySelector(".page-content-holder");
}
get _pageNavNode() {
return this._hostNode.querySelector('.page-nav');
return this._hostNode.querySelector(".page-nav");
}
clearMessages() {
@ -135,11 +150,11 @@ class ManualPageView {
}
_navigateToPrevNextPage(className) {
const linkNode = this._hostNode.querySelector('a.' + className);
if (linkNode.classList.contains('disabled')) {
const linkNode = this._hostNode.querySelector("a." + className);
if (linkNode.classList.contains("disabled")) {
return;
}
router.show(linkNode.getAttribute('href'));
router.show(linkNode.getAttribute("href"));
}
_refreshNav(offset, limit, total, removedItems, ctx) {
@ -147,7 +162,12 @@ class ManualPageView {
const totalPages = Math.ceil((total - removedItems) / limit);
const pageNumbers = _getVisiblePageNumbers(currentPage, totalPages);
const pages = _getPages(
currentPage, pageNumbers, limit, ctx.defaultLimit, removedItems);
currentPage,
pageNumbers,
limit,
ctx.defaultLimit,
removedItems
);
views.replaceContent(
this._pageNavNode,
@ -158,7 +178,8 @@ class ManualPageView {
currentPage: currentPage,
totalPages: totalPages,
pages: pages,
}));
})
);
}
}

View file

@ -1,12 +1,12 @@
'use strict';
"use strict";
const views = require('../util/views.js');
const views = require("../util/views.js");
const template = views.getTemplate('not-found');
const template = views.getTemplate("not-found");
class NotFoundView {
constructor(path) {
this._hostNode = document.getElementById('content-holder');
this._hostNode = document.getElementById("content-holder");
const sourceNode = template({ path: path });
views.replaceContent(this._hostNode, sourceNode);

View file

@ -1,30 +1,35 @@
'use strict';
"use strict";
const events = require('../events.js');
const api = require('../api.js');
const views = require('../util/views.js');
const events = require("../events.js");
const api = require("../api.js");
const views = require("../util/views.js");
const template = views.getTemplate('password-reset');
const template = views.getTemplate("password-reset");
class PasswordResetView extends events.EventTarget {
constructor() {
super();
this._hostNode = document.getElementById('content-holder');
this._hostNode = document.getElementById("content-holder");
views.replaceContent(this._hostNode, template({
views.replaceContent(
this._hostNode,
template({
canSendMails: api.canSendMails(),
contactEmail: api.getContactEmail(),
}));
})
);
views.syncScrollPosition();
views.decorateValidator(this._formNode);
this._formNode.addEventListener('submit', e => {
this._formNode.addEventListener("submit", (e) => {
e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', {
this.dispatchEvent(
new CustomEvent("submit", {
detail: {
userNameOrEmail: this._userNameOrEmailFieldNode.value,
},
}));
})
);
});
}
@ -49,11 +54,11 @@ class PasswordResetView extends events.EventTarget {
}
get _formNode() {
return this._hostNode.querySelector('form');
return this._hostNode.querySelector("form");
}
get _userNameOrEmailFieldNode() {
return this._formNode.querySelector('[name=user-name]');
return this._formNode.querySelector("[name=user-name]");
}
}

View file

@ -1,17 +1,17 @@
'use strict';
"use strict";
const events = require('../events.js');
const views = require('../util/views.js');
const PoolCategory = require('../models/pool_category.js');
const events = require("../events.js");
const views = require("../util/views.js");
const PoolCategory = require("../models/pool_category.js");
const template = views.getTemplate('pool-categories');
const rowTemplate = views.getTemplate('pool-category-row');
const template = views.getTemplate("pool-categories");
const rowTemplate = views.getTemplate("pool-category-row");
class PoolCategoriesView extends events.EventTarget {
constructor(ctx) {
super();
this._ctx = ctx;
this._hostNode = document.getElementById('content-holder');
this._hostNode = document.getElementById("content-holder");
views.replaceContent(this._hostNode, template(ctx));
views.syncScrollPosition();
@ -31,18 +31,22 @@ class PoolCategoriesView extends events.EventTarget {
}
if (this._addLinkNode) {
this._addLinkNode.addEventListener(
'click', e => this._evtAddButtonClick(e));
this._addLinkNode.addEventListener("click", (e) =>
this._evtAddButtonClick(e)
);
}
ctx.poolCategories.addEventListener(
'add', e => this._evtPoolCategoryAdded(e));
ctx.poolCategories.addEventListener("add", (e) =>
this._evtPoolCategoryAdded(e)
);
ctx.poolCategories.addEventListener(
'remove', e => this._evtPoolCategoryDeleted(e));
ctx.poolCategories.addEventListener("remove", (e) =>
this._evtPoolCategoryDeleted(e)
);
this._formNode.addEventListener(
'submit', e => this._evtSaveButtonClick(e, ctx));
this._formNode.addEventListener("submit", (e) =>
this._evtSaveButtonClick(e, ctx)
);
}
enableForm() {
@ -66,44 +70,48 @@ class PoolCategoriesView extends events.EventTarget {
}
get _formNode() {
return this._hostNode.querySelector('form');
return this._hostNode.querySelector("form");
}
get _tableBodyNode() {
return this._hostNode.querySelector('tbody');
return this._hostNode.querySelector("tbody");
}
get _addLinkNode() {
return this._hostNode.querySelector('a.add');
return this._hostNode.querySelector("a.add");
}
_addPoolCategoryRowNode(poolCategory) {
const rowNode = rowTemplate(
Object.assign(
{}, this._ctx, {poolCategory: poolCategory}));
Object.assign({}, this._ctx, { poolCategory: poolCategory })
);
const nameInput = rowNode.querySelector('.name input');
const nameInput = rowNode.querySelector(".name input");
if (nameInput) {
nameInput.addEventListener(
'change', e => this._evtNameChange(e, rowNode));
nameInput.addEventListener("change", (e) =>
this._evtNameChange(e, rowNode)
);
}
const colorInput = rowNode.querySelector('.color input');
const colorInput = rowNode.querySelector(".color input");
if (colorInput) {
colorInput.addEventListener(
'change', e => this._evtColorChange(e, rowNode));
colorInput.addEventListener("change", (e) =>
this._evtColorChange(e, rowNode)
);
}
const removeLinkNode = rowNode.querySelector('.remove a');
const removeLinkNode = rowNode.querySelector(".remove a");
if (removeLinkNode) {
removeLinkNode.addEventListener(
'click', e => this._evtDeleteButtonClick(e, rowNode));
removeLinkNode.addEventListener("click", (e) =>
this._evtDeleteButtonClick(e, rowNode)
);
}
const defaultLinkNode = rowNode.querySelector('.set-default a');
const defaultLinkNode = rowNode.querySelector(".set-default a");
if (defaultLinkNode) {
defaultLinkNode.addEventListener(
'click', e => this._evtSetDefaultButtonClick(e, rowNode));
defaultLinkNode.addEventListener("click", (e) =>
this._evtSetDefaultButtonClick(e, rowNode)
);
}
this._tableBodyNode.appendChild(rowNode);
@ -141,7 +149,7 @@ class PoolCategoriesView extends events.EventTarget {
_evtDeleteButtonClick(e, rowNode, link) {
e.preventDefault();
if (e.target.classList.contains('inactive')) {
if (e.target.classList.contains("inactive")) {
return;
}
this._ctx.poolCategories.remove(rowNode._poolCategory);
@ -150,16 +158,16 @@ class PoolCategoriesView extends events.EventTarget {
_evtSetDefaultButtonClick(e, rowNode) {
e.preventDefault();
this._ctx.poolCategories.defaultCategory = rowNode._poolCategory;
const oldRowNode = rowNode.parentNode.querySelector('tr.default');
const oldRowNode = rowNode.parentNode.querySelector("tr.default");
if (oldRowNode) {
oldRowNode.classList.remove('default');
oldRowNode.classList.remove("default");
}
rowNode.classList.add('default');
rowNode.classList.add("default");
}
_evtSaveButtonClick(e, ctx) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('submit'));
this.dispatchEvent(new CustomEvent("submit"));
}
}

View file

@ -1,41 +1,43 @@
'use strict';
"use strict";
const events = require('../events.js');
const api = require('../api.js');
const misc = require('../util/misc.js');
const views = require('../util/views.js');
const Pool = require('../models/pool.js')
const events = require("../events.js");
const api = require("../api.js");
const misc = require("../util/misc.js");
const views = require("../util/views.js");
const Pool = require("../models/pool.js");
const template = views.getTemplate('pool-create');
const template = views.getTemplate("pool-create");
class PoolCreateView extends events.EventTarget {
constructor(ctx) {
super();
this._hostNode = document.getElementById('content-holder');
this._hostNode = document.getElementById("content-holder");
views.replaceContent(this._hostNode, template(ctx));
views.decorateValidator(this._formNode);
if (this._namesFieldNode) {
this._namesFieldNode.addEventListener(
'input', e => this._evtNameInput(e));
this._namesFieldNode.addEventListener("input", (e) =>
this._evtNameInput(e)
);
}
if (this._postsFieldNode) {
this._postsFieldNode.addEventListener(
'input', e => this._evtPostsInput(e));
this._postsFieldNode.addEventListener("input", (e) =>
this._evtPostsInput(e)
);
}
for (let node of this._formNode.querySelectorAll(
'input, select, textarea, posts')) {
node.addEventListener(
'change', e => {
this.dispatchEvent(new CustomEvent('change'));
"input, select, textarea, posts"
)) {
node.addEventListener("change", (e) => {
this.dispatchEvent(new CustomEvent("change"));
});
}
this._formNode.addEventListener('submit', e => this._evtSubmit(e));
this._formNode.addEventListener("submit", (e) => this._evtSubmit(e));
}
clearMessages() {
@ -64,19 +66,21 @@ class PoolCreateView extends events.EventTarget {
if (!list.length) {
this._namesFieldNode.setCustomValidity(
'Pools must have at least one name.');
"Pools must have at least one name."
);
return;
}
for (let item of list) {
if (!regex.test(item)) {
this._namesFieldNode.setCustomValidity(
`Pool name "${item}" contains invalid symbols.`);
`Pool name "${item}" contains invalid symbols.`
);
return;
}
}
this._namesFieldNode.setCustomValidity('');
this._namesFieldNode.setCustomValidity("");
}
_evtPostsInput(e) {
@ -86,46 +90,50 @@ class PoolCreateView extends events.EventTarget {
for (let item of list) {
if (!regex.test(item)) {
this._postsFieldNode.setCustomValidity(
`Pool ID "${item}" is not an integer.`);
`Pool ID "${item}" is not an integer.`
);
return;
}
}
this._postsFieldNode.setCustomValidity('');
this._postsFieldNode.setCustomValidity("");
}
_evtSubmit(e) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', {
this.dispatchEvent(
new CustomEvent("submit", {
detail: {
names: misc.splitByWhitespace(this._namesFieldNode.value),
category: this._categoryFieldNode.value,
description: this._descriptionFieldNode.value,
posts: misc.splitByWhitespace(this._postsFieldNode.value)
.map(i => parseInt(i))
posts: misc
.splitByWhitespace(this._postsFieldNode.value)
.map((i) => parseInt(i)),
},
}));
})
);
}
get _formNode() {
return this._hostNode.querySelector('form');
return this._hostNode.querySelector("form");
}
get _namesFieldNode() {
return this._formNode.querySelector('.names input');
return this._formNode.querySelector(".names input");
}
get _categoryFieldNode() {
return this._formNode.querySelector('.category select');
return this._formNode.querySelector(".category select");
}
get _descriptionFieldNode() {
return this._formNode.querySelector('.description textarea');
return this._formNode.querySelector(".description textarea");
}
get _postsFieldNode() {
return this._formNode.querySelector('.posts input');
return this._formNode.querySelector(".posts input");
}
}

View file

@ -1,9 +1,9 @@
'use strict';
"use strict";
const events = require('../events.js');
const views = require('../util/views.js');
const events = require("../events.js");
const views = require("../util/views.js");
const template = views.getTemplate('pool-delete');
const template = views.getTemplate("pool-delete");
class PoolDeleteView extends events.EventTarget {
constructor(ctx) {
@ -13,7 +13,7 @@ class PoolDeleteView extends events.EventTarget {
this._pool = ctx.pool;
views.replaceContent(this._hostNode, template(ctx));
views.decorateValidator(this._formNode);
this._formNode.addEventListener('submit', e => this._evtSubmit(e));
this._formNode.addEventListener("submit", (e) => this._evtSubmit(e));
}
clearMessages() {
@ -38,15 +38,17 @@ class PoolDeleteView extends events.EventTarget {
_evtSubmit(e) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', {
this.dispatchEvent(
new CustomEvent("submit", {
detail: {
pool: this._pool,
},
}));
})
);
}
get _formNode() {
return this._hostNode.querySelector('form');
return this._hostNode.querySelector("form");
}
}

View file

@ -1,12 +1,12 @@
'use strict';
"use strict";
const events = require('../events.js');
const api = require('../api.js');
const misc = require('../util/misc.js');
const views = require('../util/views.js');
const Post = require('../models/post.js');
const events = require("../events.js");
const api = require("../api.js");
const misc = require("../util/misc.js");
const views = require("../util/views.js");
const Post = require("../models/post.js");
const template = views.getTemplate('pool-edit');
const template = views.getTemplate("pool-edit");
class PoolEditView extends events.EventTarget {
constructor(ctx) {
@ -19,24 +19,26 @@ class PoolEditView extends events.EventTarget {
views.decorateValidator(this._formNode);
if (this._namesFieldNode) {
this._namesFieldNode.addEventListener(
'input', e => this._evtNameInput(e));
this._namesFieldNode.addEventListener("input", (e) =>
this._evtNameInput(e)
);
}
if (this._postsFieldNode) {
this._postsFieldNode.addEventListener(
'input', e => this._evtPostsInput(e));
this._postsFieldNode.addEventListener("input", (e) =>
this._evtPostsInput(e)
);
}
for (let node of this._formNode.querySelectorAll(
'input, select, textarea, posts')) {
node.addEventListener(
'change', e => {
this.dispatchEvent(new CustomEvent('change'));
"input, select, textarea, posts"
)) {
node.addEventListener("change", (e) => {
this.dispatchEvent(new CustomEvent("change"));
});
}
this._formNode.addEventListener('submit', e => this._evtSubmit(e));
this._formNode.addEventListener("submit", (e) => this._evtSubmit(e));
}
clearMessages() {
@ -65,19 +67,21 @@ class PoolEditView extends events.EventTarget {
if (!list.length) {
this._namesFieldNode.setCustomValidity(
'Pools must have at least one name.');
"Pools must have at least one name."
);
return;
}
for (let item of list) {
if (!regex.test(item)) {
this._namesFieldNode.setCustomValidity(
`Pool name "${item}" contains invalid symbols.`);
`Pool name "${item}" contains invalid symbols.`
);
return;
}
}
this._namesFieldNode.setCustomValidity('');
this._namesFieldNode.setCustomValidity("");
}
_evtPostsInput(e) {
@ -87,57 +91,60 @@ class PoolEditView extends events.EventTarget {
for (let item of list) {
if (!regex.test(item)) {
this._postsFieldNode.setCustomValidity(
`Pool ID "${item}" is not an integer.`);
`Pool ID "${item}" is not an integer.`
);
return;
}
}
this._postsFieldNode.setCustomValidity('');
this._postsFieldNode.setCustomValidity("");
}
_evtSubmit(e) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', {
this.dispatchEvent(
new CustomEvent("submit", {
detail: {
pool: this._pool,
names: this._namesFieldNode ?
misc.splitByWhitespace(this._namesFieldNode.value) :
undefined,
names: this._namesFieldNode
? misc.splitByWhitespace(this._namesFieldNode.value)
: undefined,
category: this._categoryFieldNode ?
this._categoryFieldNode.value :
undefined,
category: this._categoryFieldNode
? this._categoryFieldNode.value
: undefined,
description: this._descriptionFieldNode ?
this._descriptionFieldNode.value :
undefined,
description: this._descriptionFieldNode
? this._descriptionFieldNode.value
: undefined,
posts: this._postsFieldNode ?
misc.splitByWhitespace(this._postsFieldNode.value) :
undefined,
posts: this._postsFieldNode
? misc.splitByWhitespace(this._postsFieldNode.value)
: undefined,
},
}));
})
);
}
get _formNode() {
return this._hostNode.querySelector('form');
return this._hostNode.querySelector("form");
}
get _namesFieldNode() {
return this._formNode.querySelector('.names input');
return this._formNode.querySelector(".names input");
}
get _categoryFieldNode() {
return this._formNode.querySelector('.category select');
return this._formNode.querySelector(".category select");
}
get _descriptionFieldNode() {
return this._formNode.querySelector('.description textarea');
return this._formNode.querySelector(".description textarea");
}
get _postsFieldNode() {
return this._formNode.querySelector('.posts input');
return this._formNode.querySelector(".posts input");
}
}

View file

@ -1,12 +1,11 @@
'use strict';
"use strict";
const events = require('../events.js');
const api = require('../api.js');
const views = require('../util/views.js');
const PoolAutoCompleteControl =
require('../controls/pool_auto_complete_control.js');
const events = require("../events.js");
const api = require("../api.js");
const views = require("../util/views.js");
const PoolAutoCompleteControl = require("../controls/pool_auto_complete_control.js");
const template = views.getTemplate('pool-merge');
const template = views.getTemplate("pool-merge");
class PoolMergeView extends events.EventTarget {
constructor(ctx) {
@ -23,15 +22,18 @@ class PoolMergeView extends events.EventTarget {
this._autoCompleteControl = new PoolAutoCompleteControl(
this._targetPoolFieldNode,
{
confirm: pool => {
confirm: (pool) => {
this._targetPoolId = pool.id;
this._autoCompleteControl.replaceSelectedText(
pool.names[0], false);
pool.names[0],
false
);
},
}
});
);
}
this._formNode.addEventListener('submit', e => this._evtSubmit(e));
this._formNode.addEventListener("submit", (e) => this._evtSubmit(e));
}
clearMessages() {
@ -56,24 +58,26 @@ class PoolMergeView extends events.EventTarget {
_evtSubmit(e) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', {
this.dispatchEvent(
new CustomEvent("submit", {
detail: {
pool: this._pool,
targetPoolId: this._targetPoolId
targetPoolId: this._targetPoolId,
},
}));
})
);
}
get _formNode() {
return this._hostNode.querySelector('form');
return this._hostNode.querySelector("form");
}
get _targetPoolFieldNode() {
return this._formNode.querySelector('input[name=target-pool]');
return this._formNode.querySelector("input[name=target-pool]");
}
get _addAliasCheckboxNode() {
return this._formNode.querySelector('input[name=alias]');
return this._formNode.querySelector("input[name=alias]");
}
}

View file

@ -1,8 +1,8 @@
'use strict';
"use strict";
const views = require('../util/views.js');
const views = require("../util/views.js");
const template = views.getTemplate('pool-summary');
const template = views.getTemplate("pool-summary");
class PoolSummaryView {
constructor(ctx) {

View file

@ -1,26 +1,26 @@
'use strict';
"use strict";
const events = require('../events.js');
const views = require('../util/views.js');
const misc = require('../util/misc.js');
const PoolSummaryView = require('./pool_summary_view.js');
const PoolEditView = require('./pool_edit_view.js');
const PoolMergeView = require('./pool_merge_view.js');
const PoolDeleteView = require('./pool_delete_view.js');
const EmptyView = require('../views/empty_view.js');
const events = require("../events.js");
const views = require("../util/views.js");
const misc = require("../util/misc.js");
const PoolSummaryView = require("./pool_summary_view.js");
const PoolEditView = require("./pool_edit_view.js");
const PoolMergeView = require("./pool_merge_view.js");
const PoolDeleteView = require("./pool_delete_view.js");
const EmptyView = require("../views/empty_view.js");
const template = views.getTemplate('pool');
const template = views.getTemplate("pool");
class PoolView extends events.EventTarget {
constructor(ctx) {
super();
this._ctx = ctx;
ctx.pool.addEventListener('change', e => this._evtChange(e));
ctx.section = ctx.section || 'summary';
ctx.pool.addEventListener("change", (e) => this._evtChange(e));
ctx.section = ctx.section || "summary";
ctx.getPrettyPoolName = misc.getPrettyPoolName;
this._hostNode = document.getElementById('content-holder');
this._hostNode = document.getElementById("content-holder");
this._install();
}
@ -28,52 +28,54 @@ class PoolView extends events.EventTarget {
const ctx = this._ctx;
views.replaceContent(this._hostNode, template(ctx));
for (let item of this._hostNode.querySelectorAll('[data-name]')) {
for (let item of this._hostNode.querySelectorAll("[data-name]")) {
item.classList.toggle(
'active', item.getAttribute('data-name') === ctx.section);
if (item.getAttribute('data-name') === ctx.section) {
"active",
item.getAttribute("data-name") === ctx.section
);
if (item.getAttribute("data-name") === ctx.section) {
item.parentNode.scrollLeft =
item.getBoundingClientRect().left -
item.parentNode.getBoundingClientRect().left
item.parentNode.getBoundingClientRect().left;
}
}
ctx.hostNode = this._hostNode.querySelector('.pool-content-holder');
if (ctx.section === 'edit') {
ctx.hostNode = this._hostNode.querySelector(".pool-content-holder");
if (ctx.section === "edit") {
if (!this._ctx.canEditAnything) {
this._view = new EmptyView();
this._view.showError(
'You don\'t have privileges to edit pools.');
"You don't have privileges to edit pools."
);
} else {
this._view = new PoolEditView(ctx);
events.proxyEvent(this._view, this, 'submit');
events.proxyEvent(this._view, this, "submit");
}
} else if (ctx.section === 'merge') {
} else if (ctx.section === "merge") {
if (!this._ctx.canMerge) {
this._view = new EmptyView();
this._view.showError(
'You don\'t have privileges to merge pools.');
"You don't have privileges to merge pools."
);
} else {
this._view = new PoolMergeView(ctx);
events.proxyEvent(this._view, this, 'submit', 'merge');
events.proxyEvent(this._view, this, "submit", "merge");
}
} else if (ctx.section === 'delete') {
} else if (ctx.section === "delete") {
if (!this._ctx.canDelete) {
this._view = new EmptyView();
this._view.showError(
'You don\'t have privileges to delete pools.');
"You don't have privileges to delete pools."
);
} else {
this._view = new PoolDeleteView(ctx);
events.proxyEvent(this._view, this, 'submit', 'delete');
events.proxyEvent(this._view, this, "submit", "delete");
}
} else {
this._view = new PoolSummaryView(ctx);
}
events.proxyEvent(this._view, this, 'change');
events.proxyEvent(this._view, this, "change");
views.syncScrollPosition();
}

Some files were not shown because too many files have changed in this diff Show more