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: repos:
- repo: https://github.com/pre-commit/pre-commit-hooks - repo: https://github.com/pre-commit/pre-commit-hooks
rev: v2.4.0 rev: v2.4.0
hooks: hooks:
- id: trailing-whitespace - id: trailing-whitespace
- id: end-of-file-fixer - id: end-of-file-fixer
- id: check-yaml - 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 - repo: https://github.com/pre-commit/mirrors-eslint
rev: v7.1.0 rev: v7.1.0
hooks: hooks:
- id: eslint - id: eslint
files: client/js/ files: client/js/
args: ['--fix'] args: ['--fix']
additional_dependencies:
- eslint-config-prettier
- repo: https://gitlab.com/pycqa/flake8 - repo: https://gitlab.com/pycqa/flake8
rev: '3.8.2' rev: '3.8.2'
hooks: hooks:
@ -19,32 +47,45 @@ repos:
files: server/szurubooru/ files: server/szurubooru/
additional_dependencies: additional_dependencies:
- flake8-print - flake8-print
args: ['--config=server/setup.cfg'] args: ['--config=server/.flake8']
- repo: local - repo: local
hooks: 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 - id: docker-build-client
name: Test building the client in Docker name: Docker - build client
entry: bash -c 'docker build -t szurubooru-client:$(git rev-parse --short HEAD) client/' entry: bash -c 'docker build client/'
language: system language: system
types: [file] types: [file]
files: client/ files: client/
pass_filenames: false pass_filenames: false
- id: docker-build-server - id: docker-build-server
name: Test building the server in Docker name: Docker - build server
entry: bash -c 'docker build -t szurubooru-server:$(git rev-parse --short HEAD) server/' entry: bash -c 'docker build server/'
language: system language: system
types: [file] types: [file]
files: server/ files: server/
pass_filenames: false 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 exclude: LICENSE.md

View file

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

View file

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

View file

@ -1,19 +1,19 @@
'use strict'; "use strict";
const api = require('../api.js'); const api = require("../api.js");
const topNavigation = require('../models/top_navigation.js'); const topNavigation = require("../models/top_navigation.js");
const EmptyView = require('../views/empty_view.js'); const EmptyView = require("../views/empty_view.js");
class BasePostController { class BasePostController {
constructor(ctx) { constructor(ctx) {
if (!api.hasPrivilege('posts:view')) { if (!api.hasPrivilege("posts:view")) {
this._view = new EmptyView(); 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; return;
} }
topNavigation.activate('posts'); topNavigation.activate("posts");
topNavigation.setTitle('Post #' + ctx.parameters.id.toString()); topNavigation.setTitle("Post #" + ctx.parameters.id.toString());
} }
} }

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

@ -1,28 +1,33 @@
'use strict'; "use strict";
const router = require('../router.js'); const router = require("../router.js");
const api = require('../api.js'); const api = require("../api.js");
const misc = require('../util/misc.js'); const misc = require("../util/misc.js");
const uri = require('../util/uri.js'); const uri = require("../util/uri.js");
const settings = require('../models/settings.js'); const settings = require("../models/settings.js");
const Post = require('../models/post.js'); const Post = require("../models/post.js");
const PostList = require('../models/post_list.js'); const PostList = require("../models/post_list.js");
const PostDetailView = require('../views/post_detail_view.js'); const PostDetailView = require("../views/post_detail_view.js");
const BasePostController = require('./base_post_controller.js'); const BasePostController = require("./base_post_controller.js");
const EmptyView = require('../views/empty_view.js'); const EmptyView = require("../views/empty_view.js");
class PostDetailController extends BasePostController { class PostDetailController extends BasePostController {
constructor(ctx, section) { constructor(ctx, section) {
super(ctx); super(ctx);
Post.get(ctx.parameters.id).then(post => { Post.get(ctx.parameters.id).then(
(post) => {
this._id = ctx.parameters.id; 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); this._installView(post, section);
}, error => { },
(error) => {
this._view = new EmptyView(); this._view = new EmptyView();
this._view.showError(error.message); this._view.showError(error.message);
}); }
);
} }
showSuccess(message) { showSuccess(message) {
@ -33,58 +38,68 @@ class PostDetailController extends BasePostController {
this._view = new PostDetailView({ this._view = new PostDetailView({
post: post, post: post,
section: section, section: section,
canMerge: api.hasPrivilege('posts:merge'), canMerge: api.hasPrivilege("posts:merge"),
}); });
this._view.addEventListener('select', e => this._evtSelect(e)); this._view.addEventListener("select", (e) => this._evtSelect(e));
this._view.addEventListener('merge', e => this._evtMerge(e)); this._view.addEventListener("merge", (e) => this._evtMerge(e));
} }
_evtSelect(e) { _evtSelect(e) {
this._view.clearMessages(); this._view.clearMessages();
this._view.disableForm(); this._view.disableForm();
Post.get(e.detail.postId).then(post => { Post.get(e.detail.postId).then(
(post) => {
this._view.selectPost(post); this._view.selectPost(post);
this._view.enableForm(); this._view.enableForm();
}, error => { },
(error) => {
this._view.showError(error.message); this._view.showError(error.message);
this._view.enableForm(); this._view.enableForm();
}); }
);
} }
_evtSaved(e, section) { _evtSaved(e, section) {
misc.disableExitConfirmation(); misc.disableExitConfirmation();
if (this._id !== e.detail.post.id) { if (this._id !== e.detail.post.id) {
router.replace( router.replace(
uri.formatClientLink('post', e.detail.post.id, section), uri.formatClientLink("post", e.detail.post.id, section),
null, null,
false); false
);
} }
} }
_evtMerge(e) { _evtMerge(e) {
this._view.clearMessages(); this._view.clearMessages();
this._view.disableForm(); this._view.disableForm();
e.detail.post.merge(e.detail.targetPost.id, e.detail.useOldContent) e.detail.post
.then(() => { .merge(e.detail.targetPost.id, e.detail.useOldContent)
this._installView(e.detail.post, 'merge'); .then(
this._view.showSuccess('Post merged.'); () => {
this._installView(e.detail.post, "merge");
this._view.showSuccess("Post merged.");
router.replace( router.replace(
uri.formatClientLink( uri.formatClientLink(
'post', e.detail.targetPost.id, 'merge'), "post",
e.detail.targetPost.id,
"merge"
),
null, null,
false); false
}, error => { );
},
(error) => {
this._view.showError(error.message); this._view.showError(error.message);
this._view.enableForm(); this._view.enableForm();
}); }
);
} }
} }
module.exports = router => { module.exports = (router) => {
router.enter( router.enter(["post", ":id", "merge"], (ctx, next) => {
['post', ':id', 'merge'], ctx.controller = new PostDetailController(ctx, "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 router = require("../router.js");
const api = require('../api.js'); const api = require("../api.js");
const settings = require('../models/settings.js'); const settings = require("../models/settings.js");
const uri = require('../util/uri.js'); const uri = require("../util/uri.js");
const PostList = require('../models/post_list.js'); const PostList = require("../models/post_list.js");
const topNavigation = require('../models/top_navigation.js'); const topNavigation = require("../models/top_navigation.js");
const PageController = require('../controllers/page_controller.js'); const PageController = require("../controllers/page_controller.js");
const PostsHeaderView = require('../views/posts_header_view.js'); const PostsHeaderView = require("../views/posts_header_view.js");
const PostsPageView = require('../views/posts_page_view.js'); const PostsPageView = require("../views/posts_page_view.js");
const EmptyView = require('../views/empty_view.js'); const EmptyView = require("../views/empty_view.js");
const fields = [ const fields = [
'id', 'thumbnailUrl', 'type', 'safety', "id",
'score', 'favoriteCount', 'commentCount', 'tags', 'version' "thumbnailUrl",
"type",
"safety",
"score",
"favoriteCount",
"commentCount",
"tags",
"version",
]; ];
class PostListController { class PostListController {
constructor(ctx) { constructor(ctx) {
this._pageController = new PageController(); this._pageController = new PageController();
if (!api.hasPrivilege('posts:list')) { if (!api.hasPrivilege("posts:list")) {
this._view = new EmptyView(); 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; return;
} }
this._ctx = ctx; this._ctx = ctx;
topNavigation.activate('posts'); topNavigation.activate("posts");
topNavigation.setTitle('Listing posts'); topNavigation.setTitle("Listing posts");
this._headerView = new PostsHeaderView({ this._headerView = new PostsHeaderView({
hostNode: this._pageController.view.pageHeaderHolderNode, hostNode: this._pageController.view.pageHeaderHolderNode,
parameters: ctx.parameters, parameters: ctx.parameters,
enableSafety: api.safetyEnabled(), enableSafety: api.safetyEnabled(),
canBulkEditTags: api.hasPrivilege('posts:bulk-edit:tags'), canBulkEditTags: api.hasPrivilege("posts:bulk-edit:tags"),
canBulkEditSafety: api.hasPrivilege('posts:bulk-edit:safety'), canBulkEditSafety: api.hasPrivilege("posts:bulk-edit:safety"),
bulkEdit: { bulkEdit: {
tags: this._bulkEditTags tags: this._bulkEditTags,
}, },
}); });
this._headerView.addEventListener( this._headerView.addEventListener("navigate", (e) =>
'navigate', e => this._evtNavigate(e)); this._evtNavigate(e)
);
this._syncPageController(); this._syncPageController();
} }
@ -52,33 +60,35 @@ class PostListController {
} }
get _bulkEditTags() { get _bulkEditTags() {
return (this._ctx.parameters.tag || '').split(/\s+/).filter(s => s); return (this._ctx.parameters.tag || "").split(/\s+/).filter((s) => s);
} }
_evtNavigate(e) { _evtNavigate(e) {
router.showNoDispatch( router.showNoDispatch(
uri.formatClientLink('posts', e.detail.parameters)); uri.formatClientLink("posts", e.detail.parameters)
);
Object.assign(this._ctx.parameters, e.detail.parameters); Object.assign(this._ctx.parameters, e.detail.parameters);
this._syncPageController(); this._syncPageController();
} }
_evtTag(e) { _evtTag(e) {
Promise.all( 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()) .then(e.detail.post.save())
.catch(error => window.alert(error.message)); .catch((error) => window.alert(error.message));
} }
_evtUntag(e) { _evtUntag(e) {
for (let tag of this._bulkEditTags) { for (let tag of this._bulkEditTags) {
e.detail.post.tags.removeByName(tag); 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) { _evtChangeSafety(e) {
e.detail.post.safety = e.detail.safety; 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() { _syncPageController() {
@ -86,39 +96,45 @@ class PostListController {
parameters: this._ctx.parameters, parameters: this._ctx.parameters,
defaultLimit: parseInt(settings.get().postsPerPage), defaultLimit: parseInt(settings.get().postsPerPage),
getClientUrlForPage: (offset, limit) => { getClientUrlForPage: (offset, limit) => {
const parameters = Object.assign( const parameters = Object.assign({}, this._ctx.parameters, {
{}, this._ctx.parameters, {offset: offset, limit: limit}); offset: offset,
return uri.formatClientLink('posts', parameters); limit: limit,
});
return uri.formatClientLink("posts", parameters);
}, },
requestPage: (offset, limit) => { requestPage: (offset, limit) => {
return PostList.search( return PostList.search(
this._ctx.parameters.query, offset, limit, fields); this._ctx.parameters.query,
offset,
limit,
fields
);
}, },
pageRenderer: pageCtx => { pageRenderer: (pageCtx) => {
Object.assign(pageCtx, { Object.assign(pageCtx, {
canViewPosts: api.hasPrivilege('posts:view'), canViewPosts: api.hasPrivilege("posts:view"),
canBulkEditTags: api.hasPrivilege('posts:bulk-edit:tags'), canBulkEditTags: api.hasPrivilege("posts:bulk-edit:tags"),
canBulkEditSafety: canBulkEditSafety: api.hasPrivilege(
api.hasPrivilege('posts:bulk-edit:safety'), "posts:bulk-edit:safety"
),
bulkEdit: { bulkEdit: {
tags: this._bulkEditTags, tags: this._bulkEditTags,
}, },
}); });
const view = new PostsPageView(pageCtx); const view = new PostsPageView(pageCtx);
view.addEventListener('tag', e => this._evtTag(e)); view.addEventListener("tag", (e) => this._evtTag(e));
view.addEventListener('untag', e => this._evtUntag(e)); view.addEventListener("untag", (e) => this._evtUntag(e));
view.addEventListener( view.addEventListener("changeSafety", (e) =>
'changeSafety', e => this._evtChangeSafety(e)); this._evtChangeSafety(e)
);
return view; return view;
}, },
}); });
} }
} }
module.exports = router => { module.exports = (router) => {
router.enter( router.enter(["posts"], (ctx, next) => {
['posts'],
(ctx, next) => {
ctx.controller = new PostListController(ctx); ctx.controller = new PostListController(ctx);
}); });
}; };

View file

@ -1,16 +1,16 @@
'use strict'; "use strict";
const router = require('../router.js'); const router = require("../router.js");
const api = require('../api.js'); const api = require("../api.js");
const uri = require('../util/uri.js'); const uri = require("../util/uri.js");
const misc = require('../util/misc.js'); const misc = require("../util/misc.js");
const settings = require('../models/settings.js'); const settings = require("../models/settings.js");
const Comment = require('../models/comment.js'); const Comment = require("../models/comment.js");
const Post = require('../models/post.js'); const Post = require("../models/post.js");
const PostList = require('../models/post_list.js'); const PostList = require("../models/post_list.js");
const PostMainView = require('../views/post_main_view.js'); const PostMainView = require("../views/post_main_view.js");
const BasePostController = require('./base_post_controller.js'); const BasePostController = require("./base_post_controller.js");
const EmptyView = require('../views/empty_view.js'); const EmptyView = require("../views/empty_view.js");
class PostMainController extends BasePostController { class PostMainController extends BasePostController {
constructor(ctx, editMode) { constructor(ctx, editMode) {
@ -21,17 +21,23 @@ class PostMainController extends BasePostController {
Post.get(ctx.parameters.id), Post.get(ctx.parameters.id),
PostList.getAround( PostList.getAround(
ctx.parameters.id, ctx.parameters.id,
parameters ? parameters.query : null), parameters ? parameters.query : null
]).then(responses => { ),
]).then(
(responses) => {
const [post, aroundResponse] = responses; const [post, aroundResponse] = responses;
// remove junk from query, but save it into history so that it can // remove junk from query, but save it into history so that it can
// be still accessed after history navigation / page refresh // be still accessed after history navigation / page refresh
if (parameters.query) { if (parameters.query) {
ctx.state.parameters = parameters; ctx.state.parameters = parameters;
const url = editMode ? const url = editMode
uri.formatClientLink('post', ctx.parameters.id, 'edit') : ? uri.formatClientLink(
uri.formatClientLink('post', ctx.parameters.id); "post",
ctx.parameters.id,
"edit"
)
: uri.formatClientLink("post", ctx.parameters.id);
router.replace(url, ctx.state, false); router.replace(url, ctx.state, false);
} }
@ -39,56 +45,83 @@ class PostMainController extends BasePostController {
this._view = new PostMainView({ this._view = new PostMainView({
post: post, post: post,
editMode: editMode, editMode: editMode,
prevPostId: aroundResponse.prev ? aroundResponse.prev.id : null, prevPostId: aroundResponse.prev
nextPostId: aroundResponse.next ? aroundResponse.next.id : null, ? aroundResponse.prev.id
canEditPosts: api.hasPrivilege('posts:edit'), : null,
canDeletePosts: api.hasPrivilege('posts:delete'), nextPostId: aroundResponse.next
canFeaturePosts: api.hasPrivilege('posts:feature'), ? aroundResponse.next.id
canListComments: api.hasPrivilege('comments:list'), : null,
canCreateComments: api.hasPrivilege('comments:create'), 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, parameters: parameters,
}); });
if (this._view.sidebarControl) { if (this._view.sidebarControl) {
this._view.sidebarControl.addEventListener( this._view.sidebarControl.addEventListener(
'favorite', e => this._evtFavoritePost(e)); "favorite",
(e) => this._evtFavoritePost(e)
);
this._view.sidebarControl.addEventListener( 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( 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( this._view.sidebarControl.addEventListener(
'fitModeChange', e => this._evtFitModeChange(e)); "feature",
this._view.sidebarControl.addEventListener( (e) => this._evtFeaturePost(e)
'change', e => this._evtPostChange(e)); );
this._view.sidebarControl.addEventListener( this._view.sidebarControl.addEventListener("delete", (e) =>
'submit', e => this._evtUpdatePost(e)); this._evtDeletePost(e)
this._view.sidebarControl.addEventListener( );
'feature', e => this._evtFeaturePost(e)); this._view.sidebarControl.addEventListener("merge", (e) =>
this._view.sidebarControl.addEventListener( this._evtMergePost(e)
'delete', e => this._evtDeletePost(e)); );
this._view.sidebarControl.addEventListener(
'merge', e => this._evtMergePost(e));
} }
if (this._view.commentControl) { if (this._view.commentControl) {
this._view.commentControl.addEventListener( this._view.commentControl.addEventListener("change", (e) =>
'change', e => this._evtCommentChange(e)); this._evtCommentChange(e)
this._view.commentControl.addEventListener( );
'submit', e => this._evtCreateComment(e)); this._view.commentControl.addEventListener("submit", (e) =>
this._evtCreateComment(e)
);
} }
if (this._view.commentListControl) { if (this._view.commentListControl) {
this._view.commentListControl.addEventListener( this._view.commentListControl.addEventListener(
'submit', e => this._evtUpdateComment(e)); "submit",
(e) => this._evtUpdateComment(e)
);
this._view.commentListControl.addEventListener( this._view.commentListControl.addEventListener(
'score', e => this._evtScoreComment(e)); "score",
(e) => this._evtScoreComment(e)
);
this._view.commentListControl.addEventListener( this._view.commentListControl.addEventListener(
'delete', e => this._evtDeleteComment(e)); "delete",
(e) => this._evtDeleteComment(e)
);
} }
}, error => { },
(error) => {
this._view = new EmptyView(); this._view = new EmptyView();
this._view.showError(error.message); this._view.showError(error.message);
}); }
);
} }
_evtFitModeChange(e) { _evtFitModeChange(e) {
@ -100,32 +133,36 @@ class PostMainController extends BasePostController {
_evtFeaturePost(e) { _evtFeaturePost(e) {
this._view.sidebarControl.disableForm(); this._view.sidebarControl.disableForm();
this._view.sidebarControl.clearMessages(); this._view.sidebarControl.clearMessages();
e.detail.post.feature() e.detail.post.feature().then(
.then(() => { () => {
this._view.sidebarControl.showSuccess('Post featured.'); this._view.sidebarControl.showSuccess("Post featured.");
this._view.sidebarControl.enableForm(); this._view.sidebarControl.enableForm();
}, error => { },
(error) => {
this._view.sidebarControl.showError(error.message); this._view.sidebarControl.showError(error.message);
this._view.sidebarControl.enableForm(); this._view.sidebarControl.enableForm();
}); }
);
} }
_evtMergePost(e) { _evtMergePost(e) {
router.show(uri.formatClientLink('post', e.detail.post.id, 'merge')); router.show(uri.formatClientLink("post", e.detail.post.id, "merge"));
} }
_evtDeletePost(e) { _evtDeletePost(e) {
this._view.sidebarControl.disableForm(); this._view.sidebarControl.disableForm();
this._view.sidebarControl.clearMessages(); this._view.sidebarControl.clearMessages();
e.detail.post.delete() e.detail.post.delete().then(
.then(() => { () => {
misc.disableExitConfirmation(); misc.disableExitConfirmation();
const ctx = router.show(uri.formatClientLink('posts')); const ctx = router.show(uri.formatClientLink("posts"));
ctx.controller.showSuccess('Post deleted.'); ctx.controller.showSuccess("Post deleted.");
}, error => { },
(error) => {
this._view.sidebarControl.showError(error.message); this._view.sidebarControl.showError(error.message);
this._view.sidebarControl.enableForm(); this._view.sidebarControl.enableForm();
}); }
);
} }
_evtUpdatePost(e) { _evtUpdatePost(e) {
@ -150,15 +187,17 @@ class PostMainController extends BasePostController {
if (e.detail.source !== undefined) { if (e.detail.source !== undefined) {
post.source = e.detail.source; post.source = e.detail.source;
} }
post.save() post.save().then(
.then(() => { () => {
this._view.sidebarControl.showSuccess('Post saved.'); this._view.sidebarControl.showSuccess("Post saved.");
this._view.sidebarControl.enableForm(); this._view.sidebarControl.enableForm();
misc.disableExitConfirmation(); misc.disableExitConfirmation();
}, error => { },
(error) => {
this._view.sidebarControl.showError(error.message); this._view.sidebarControl.showError(error.message);
this._view.sidebarControl.enableForm(); this._view.sidebarControl.enableForm();
}); }
);
} }
_evtPostChange(e) { _evtPostChange(e) {
@ -173,75 +212,78 @@ class PostMainController extends BasePostController {
this._view.commentControl.disableForm(); this._view.commentControl.disableForm();
const comment = Comment.create(this._post.id); const comment = Comment.create(this._post.id);
comment.text = e.detail.text; comment.text = e.detail.text;
comment.save() comment.save().then(
.then(() => { () => {
this._post.comments.add(comment); this._post.comments.add(comment);
this._view.commentControl.exitEditMode(); this._view.commentControl.exitEditMode();
this._view.commentControl.enableForm(); this._view.commentControl.enableForm();
misc.disableExitConfirmation(); misc.disableExitConfirmation();
}, error => { },
(error) => {
this._view.commentControl.showError(error.message); this._view.commentControl.showError(error.message);
this._view.commentControl.enableForm(); this._view.commentControl.enableForm();
}); }
);
} }
_evtUpdateComment(e) { _evtUpdateComment(e) {
// TODO: disable form // TODO: disable form
e.detail.comment.text = e.detail.text; e.detail.comment.text = e.detail.text;
e.detail.comment.save() e.detail.comment.save().catch((error) => {
.catch(error => {
e.detail.target.showError(error.message); e.detail.target.showError(error.message);
// TODO: enable form // TODO: enable form
}); });
} }
_evtScoreComment(e) { _evtScoreComment(e) {
e.detail.comment.setScore(e.detail.score) e.detail.comment
.catch(error => window.alert(error.message)); .setScore(e.detail.score)
.catch((error) => window.alert(error.message));
} }
_evtDeleteComment(e) { _evtDeleteComment(e) {
e.detail.comment.delete() e.detail.comment
.catch(error => window.alert(error.message)); .delete()
.catch((error) => window.alert(error.message));
} }
_evtScorePost(e) { _evtScorePost(e) {
if (!api.hasPrivilege('posts:score')) { if (!api.hasPrivilege("posts:score")) {
return; return;
} }
e.detail.post.setScore(e.detail.score) e.detail.post
.catch(error => window.alert(error.message)); .setScore(e.detail.score)
.catch((error) => window.alert(error.message));
} }
_evtFavoritePost(e) { _evtFavoritePost(e) {
if (!api.hasPrivilege('posts:favorite')) { if (!api.hasPrivilege("posts:favorite")) {
return; return;
} }
e.detail.post.addToFavorites() e.detail.post
.catch(error => window.alert(error.message)); .addToFavorites()
.catch((error) => window.alert(error.message));
} }
_evtUnfavoritePost(e) { _evtUnfavoritePost(e) {
if (!api.hasPrivilege('posts:favorite')) { if (!api.hasPrivilege("posts:favorite")) {
return; return;
} }
e.detail.post.removeFromFavorites() e.detail.post
.catch(error => window.alert(error.message)); .removeFromFavorites()
.catch((error) => window.alert(error.message));
} }
} }
module.exports = router => { module.exports = (router) => {
router.enter(['post', ':id', 'edit'], router.enter(["post", ":id", "edit"], (ctx, next) => {
(ctx, next) => {
// restore parameters from history state // restore parameters from history state
if (ctx.state.parameters) { if (ctx.state.parameters) {
Object.assign(ctx.parameters, ctx.state.parameters); Object.assign(ctx.parameters, ctx.state.parameters);
} }
ctx.controller = new PostMainController(ctx, true); ctx.controller = new PostMainController(ctx, true);
}); });
router.enter( router.enter(["post", ":id"], (ctx, next) => {
['post', ':id'],
(ctx, next) => {
// restore parameters from history state // restore parameters from history state
if (ctx.state.parameters) { if (ctx.state.parameters) {
Object.assign(ctx.parameters, 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 api = require("../api.js");
const router = require('../router.js'); const router = require("../router.js");
const uri = require('../util/uri.js'); const uri = require("../util/uri.js");
const misc = require('../util/misc.js'); const misc = require("../util/misc.js");
const progress = require('../util/progress.js'); const progress = require("../util/progress.js");
const topNavigation = require('../models/top_navigation.js'); const topNavigation = require("../models/top_navigation.js");
const Post = require('../models/post.js'); const Post = require("../models/post.js");
const Tag = require('../models/tag.js'); const Tag = require("../models/tag.js");
const PostUploadView = require('../views/post_upload_view.js'); const PostUploadView = require("../views/post_upload_view.js");
const EmptyView = require('../views/empty_view.js'); const EmptyView = require("../views/empty_view.js");
const genericErrorMessage = const genericErrorMessage =
'One of the posts needs your attention; ' + "One of the posts needs your attention; " +
'click "resume upload" when you\'re ready.'; 'click "resume upload" when you\'re ready.';
class PostUploadController { class PostUploadController {
constructor() { constructor() {
this._lastCancellablePromise = null; this._lastCancellablePromise = null;
if (!api.hasPrivilege('posts:create')) { if (!api.hasPrivilege("posts:create")) {
this._view = new EmptyView(); 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; return;
} }
topNavigation.activate('upload'); topNavigation.activate("upload");
topNavigation.setTitle('Upload'); topNavigation.setTitle("Upload");
this._view = new PostUploadView({ this._view = new PostUploadView({
canUploadAnonymously: api.hasPrivilege('posts:create:anonymous'), canUploadAnonymously: api.hasPrivilege("posts:create:anonymous"),
canViewPosts: api.hasPrivilege('posts:view'), canViewPosts: api.hasPrivilege("posts:view"),
enableSafety: api.safetyEnabled(), enableSafety: api.safetyEnabled(),
}); });
this._view.addEventListener('change', e => this._evtChange(e)); this._view.addEventListener("change", (e) => this._evtChange(e));
this._view.addEventListener('submit', e => this._evtSubmit(e)); this._view.addEventListener("submit", (e) => this._evtSubmit(e));
this._view.addEventListener('cancel', e => this._evtCancel(e)); this._view.addEventListener("cancel", (e) => this._evtCancel(e));
} }
_evtChange(e) { _evtChange(e) {
@ -56,45 +56,61 @@ class PostUploadController {
this._view.disableForm(); this._view.disableForm();
this._view.clearMessages(); this._view.clearMessages();
e.detail.uploadables.reduce( e.detail.uploadables
(promise, uploadable) => promise.then(() => this._uploadSinglePost( .reduce(
uploadable, e.detail.skipDuplicates)), (promise, uploadable) =>
Promise.resolve()) promise.then(() =>
.then(() => { this._uploadSinglePost(
uploadable,
e.detail.skipDuplicates
)
),
Promise.resolve()
)
.then(
() => {
this._view.clearMessages(); this._view.clearMessages();
misc.disableExitConfirmation(); misc.disableExitConfirmation();
const ctx = router.show(uri.formatClientLink('posts')); const ctx = router.show(uri.formatClientLink("posts"));
ctx.controller.showSuccess('Posts uploaded.'); ctx.controller.showSuccess("Posts uploaded.");
}, error => { },
(error) => {
if (error.uploadable) { if (error.uploadable) {
if (error.similarPosts) { if (error.similarPosts) {
error.uploadable.lookalikes = error.similarPosts; error.uploadable.lookalikes = error.similarPosts;
this._view.updateUploadable(error.uploadable); this._view.updateUploadable(error.uploadable);
this._view.showInfo(genericErrorMessage); this._view.showInfo(genericErrorMessage);
this._view.showInfo( this._view.showInfo(
error.message, error.uploadable); error.message,
error.uploadable
);
} else { } else {
this._view.showError(genericErrorMessage); this._view.showError(genericErrorMessage);
this._view.showError( this._view.showError(
error.message, error.uploadable); error.message,
error.uploadable
);
} }
} else { } else {
this._view.showError(error.message); this._view.showError(error.message);
} }
this._view.enableForm(); this._view.enableForm();
}); }
);
} }
_uploadSinglePost(uploadable, skipDuplicates) { _uploadSinglePost(uploadable, skipDuplicates) {
progress.start(); progress.start();
let reverseSearchPromise = Promise.resolve(); let reverseSearchPromise = Promise.resolve();
if (!uploadable.lookalikesConfirmed) { if (!uploadable.lookalikesConfirmed) {
reverseSearchPromise = reverseSearchPromise = Post.reverseSearch(
Post.reverseSearch(uploadable.url || uploadable.file); uploadable.url || uploadable.file
);
} }
this._lastCancellablePromise = reverseSearchPromise; this._lastCancellablePromise = reverseSearchPromise;
return reverseSearchPromise.then(searchResult => { return reverseSearchPromise
.then((searchResult) => {
if (searchResult) { if (searchResult) {
// notify about exact duplicate // notify about exact duplicate
if (searchResult.exactPost) { if (searchResult.exactPost) {
@ -102,8 +118,10 @@ class PostUploadController {
this._view.removeUploadable(uploadable); this._view.removeUploadable(uploadable);
return Promise.resolve(); return Promise.resolve();
} else { } else {
let error = new Error('Post already uploaded ' + let error = new Error(
`(@${searchResult.exactPost.id})`); "Post already uploaded " +
`(@${searchResult.exactPost.id})`
);
error.uploadable = uploadable; error.uploadable = uploadable;
return Promise.reject(error); return Promise.reject(error);
} }
@ -113,7 +131,8 @@ class PostUploadController {
if (searchResult.similarPosts.length) { if (searchResult.similarPosts.length) {
let error = new Error( let error = new Error(
`Found ${searchResult.similarPosts.length} similar ` + `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.uploadable = uploadable;
error.similarPosts = searchResult.similarPosts; error.similarPosts = searchResult.similarPosts;
return Promise.reject(error); return Promise.reject(error);
@ -122,21 +141,24 @@ class PostUploadController {
// no duplicates, proceed with saving // no duplicates, proceed with saving
let post = this._uploadableToPost(uploadable); let post = this._uploadableToPost(uploadable);
let savePromise = post.save(uploadable.anonymous) let savePromise = post.save(uploadable.anonymous).then(() => {
.then(() => {
this._view.removeUploadable(uploadable); this._view.removeUploadable(uploadable);
return Promise.resolve(); return Promise.resolve();
}); });
this._lastCancellablePromise = savePromise; this._lastCancellablePromise = savePromise;
return savePromise; return savePromise;
}).then(result => { })
.then(
(result) => {
progress.done(); progress.done();
return Promise.resolve(result); return Promise.resolve(result);
}, error => { },
(error) => {
error.uploadable = uploadable; error.uploadable = uploadable;
progress.done(); progress.done();
return Promise.reject(error); return Promise.reject(error);
}); }
);
} }
_uploadableToPost(uploadable) { _uploadableToPost(uploadable) {
@ -159,8 +181,8 @@ class PostUploadController {
} }
} }
module.exports = router => { module.exports = (router) => {
router.enter(['upload'], (ctx, next) => { router.enter(["upload"], (ctx, next) => {
ctx.controller = new PostUploadController(); ctx.controller = new PostUploadController();
}); });
}; };

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

@ -1,25 +1,25 @@
'use strict'; "use strict";
const router = require('../router.js'); const router = require("../router.js");
const api = require('../api.js'); const api = require("../api.js");
const uri = require('../util/uri.js'); const uri = require("../util/uri.js");
const User = require('../models/user.js'); const User = require("../models/user.js");
const topNavigation = require('../models/top_navigation.js'); const topNavigation = require("../models/top_navigation.js");
const RegistrationView = require('../views/registration_view.js'); const RegistrationView = require("../views/registration_view.js");
const EmptyView = require('../views/empty_view.js'); const EmptyView = require("../views/empty_view.js");
class UserRegistrationController { class UserRegistrationController {
constructor() { constructor() {
if (!api.hasPrivilege('users:create:self')) { if (!api.hasPrivilege("users:create:self")) {
this._view = new EmptyView(); this._view = new EmptyView();
this._view.showError('Registration is closed.'); this._view.showError("Registration is closed.");
return; return;
} }
topNavigation.activate('register'); topNavigation.activate("register");
topNavigation.setTitle('Registration'); topNavigation.setTitle("Registration");
this._view = new RegistrationView(); this._view = new RegistrationView();
this._view.addEventListener('submit', e => this._evtRegister(e)); this._view.addEventListener("submit", (e) => this._evtRegister(e));
} }
_evtRegister(e) { _evtRegister(e) {
@ -30,30 +30,35 @@ class UserRegistrationController {
user.email = e.detail.email; user.email = e.detail.email;
user.password = e.detail.password; user.password = e.detail.password;
const isLoggedIn = api.isLoggedIn(); const isLoggedIn = api.isLoggedIn();
user.save().then(() => { user.save()
.then(() => {
if (isLoggedIn) { if (isLoggedIn) {
return Promise.resolve(); return Promise.resolve();
} else { } else {
api.forget(); api.forget();
return api.login(e.detail.name, e.detail.password, false); return api.login(e.detail.name, e.detail.password, false);
} }
}).then(() => { })
.then(
() => {
if (isLoggedIn) { if (isLoggedIn) {
const ctx = router.show(uri.formatClientLink('users')); const ctx = router.show(uri.formatClientLink("users"));
ctx.controller.showSuccess('User added!'); ctx.controller.showSuccess("User added!");
} else { } else {
const ctx = router.show(uri.formatClientLink()); 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.showError(error.message);
this._view.enableForm(); this._view.enableForm();
}); }
);
} }
} }
module.exports = router => { module.exports = (router) => {
router.enter(['register'], (ctx, next) => { router.enter(["register"], (ctx, next) => {
new UserRegistrationController(); 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_TAB = 9;
const KEY_RETURN = 13; const KEY_RETURN = 13;
@ -10,14 +10,14 @@ const KEY_UP = 38;
const KEY_DOWN = 40; const KEY_DOWN = 40;
function _getSelectionStart(input) { function _getSelectionStart(input) {
if ('selectionStart' in input) { if ("selectionStart" in input) {
return input.selectionStart; return input.selectionStart;
} }
if (document.selection) { if (document.selection) {
input.focus(); input.focus();
const sel = document.selection.createRange(); const sel = document.selection.createRange();
const selLen = document.selection.createRange().text.length; 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 sel.text.length - selLen;
} }
return 0; return 0;
@ -27,18 +27,22 @@ class AutoCompleteControl {
constructor(sourceInputNode, options) { constructor(sourceInputNode, options) {
this._sourceInputNode = sourceInputNode; this._sourceInputNode = sourceInputNode;
this._options = {}; this._options = {};
Object.assign(this._options, { Object.assign(
this._options,
{
verticalShift: 2, verticalShift: 2,
maxResults: 15, maxResults: 15,
getTextToFind: () => { getTextToFind: () => {
const value = sourceInputNode.value; const value = sourceInputNode.value;
const start = _getSelectionStart(sourceInputNode); const start = _getSelectionStart(sourceInputNode);
return value.substring(0, start).replace(/.*\s+/, ''); return value.substring(0, start).replace(/.*\s+/, "");
}, },
confirm: null, confirm: null,
delete: null, delete: null,
getMatches: null, getMatches: null,
}, options); },
options
);
this._showTimeout = null; this._showTimeout = null;
this._results = []; this._results = [];
@ -49,22 +53,22 @@ class AutoCompleteControl {
hide() { hide() {
window.clearTimeout(this._showTimeout); window.clearTimeout(this._showTimeout);
this._suggestionDiv.style.display = 'none'; this._suggestionDiv.style.display = "none";
this._isVisible = false; this._isVisible = false;
} }
replaceSelectedText(result, addSpace) { replaceSelectedText(result, addSpace) {
const start = _getSelectionStart(this._sourceInputNode); const start = _getSelectionStart(this._sourceInputNode);
let prefix = ''; let prefix = "";
let suffix = this._sourceInputNode.value.substring(start); let suffix = this._sourceInputNode.value.substring(start);
let middle = this._sourceInputNode.value.substring(0, start); let middle = this._sourceInputNode.value.substring(0, start);
const index = middle.lastIndexOf(' '); const index = middle.lastIndexOf(" ");
if (index !== -1) { if (index !== -1) {
prefix = this._sourceInputNode.value.substring(0, index + 1); prefix = this._sourceInputNode.value.substring(0, index + 1);
middle = this._sourceInputNode.value.substring(index + 1); middle = this._sourceInputNode.value.substring(index + 1);
} }
this._sourceInputNode.value = ( this._sourceInputNode.value =
prefix + result.toString() + ' ' + suffix.trimLeft()); prefix + result.toString() + " " + suffix.trimLeft();
if (!addSpace) { if (!addSpace) {
this._sourceInputNode.value = this._sourceInputNode.value.trim(); this._sourceInputNode.value = this._sourceInputNode.value.trim();
} }
@ -86,7 +90,7 @@ class AutoCompleteControl {
} }
_show() { _show() {
this._suggestionDiv.style.display = 'block'; this._suggestionDiv.style.display = "block";
this._isVisible = true; this._isVisible = true;
} }
@ -101,27 +105,30 @@ class AutoCompleteControl {
_install() { _install() {
if (!this._sourceInputNode) { 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( 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("data-autocomplete", true);
this._sourceInputNode.setAttribute('autocomplete', 'off'); this._sourceInputNode.setAttribute("autocomplete", "off");
this._sourceInputNode.addEventListener( this._sourceInputNode.addEventListener("keydown", (e) =>
'keydown', e => this._evtKeyDown(e)); this._evtKeyDown(e)
this._sourceInputNode.addEventListener( );
'blur', e => this._evtBlur(e)); this._sourceInputNode.addEventListener("blur", (e) =>
this._evtBlur(e)
);
this._suggestionDiv = views.htmlToDom( this._suggestionDiv = views.htmlToDom(
'<div class="autocomplete"><ul></ul></div>'); '<div class="autocomplete"><ul></ul></div>'
this._suggestionList = this._suggestionDiv.querySelector('ul'); );
this._suggestionList = this._suggestionDiv.querySelector("ul");
document.body.appendChild(this._suggestionDiv); document.body.appendChild(this._suggestionDiv);
views.monitorNodeRemoval( views.monitorNodeRemoval(this._sourceInputNode, () => {
this._sourceInputNode, () => {
this._uninstall(); this._uninstall();
}); });
} }
@ -174,8 +181,7 @@ class AutoCompleteControl {
func(); func();
} else { } else {
window.clearTimeout(this._showTimeout); window.clearTimeout(this._showTimeout);
this._showTimeout = window.setTimeout( this._showTimeout = window.setTimeout(() => {
() => {
this._showOrHide(); this._showOrHide();
}, 250); }, 250);
} }
@ -196,9 +202,11 @@ class AutoCompleteControl {
} }
_selectPrevious() { _selectPrevious() {
this._select(this._activeResult === -1 ? this._select(
this._results.length - 1 : this._activeResult === -1
this._activeResult - 1); ? this._results.length - 1
: this._activeResult - 1
);
} }
_selectNext() { _selectNext() {
@ -206,15 +214,18 @@ class AutoCompleteControl {
} }
_select(newActiveResult) { _select(newActiveResult) {
this._activeResult = this._activeResult = newActiveResult.between(
newActiveResult.between(0, this._results.length - 1, true) ? 0,
newActiveResult : this._results.length - 1,
-1; true
)
? newActiveResult
: -1;
this._refreshActiveResult(); this._refreshActiveResult();
} }
_updateResults(textToFind) { _updateResults(textToFind) {
this._options.getMatches(textToFind).then(matches => { this._options.getMatches(textToFind).then((matches) => {
const oldResults = this._results.slice(); const oldResults = this._results.slice();
this._results = matches.slice(0, this._options.maxResults); this._results = matches.slice(0, this._options.maxResults);
const oldResultsHash = JSON.stringify(oldResults); const oldResultsHash = JSON.stringify(oldResults);
@ -237,21 +248,17 @@ class AutoCompleteControl {
} }
for (let [resultIndex, resultItem] of this._results.entries()) { for (let [resultIndex, resultItem] of this._results.entries()) {
let resultIndexWorkaround = resultIndex; let resultIndexWorkaround = resultIndex;
const listItem = document.createElement('li'); const listItem = document.createElement("li");
const link = document.createElement('a'); const link = document.createElement("a");
link.innerHTML = resultItem.caption; link.innerHTML = resultItem.caption;
link.setAttribute('href', ''); link.setAttribute("href", "");
link.setAttribute('data-key', resultItem.value); link.setAttribute("data-key", resultItem.value);
link.addEventListener( link.addEventListener("mouseenter", (e) => {
'mouseenter',
e => {
e.preventDefault(); e.preventDefault();
this._activeResult = resultIndexWorkaround; this._activeResult = resultIndexWorkaround;
this._refreshActiveResult(); this._refreshActiveResult();
}); });
link.addEventListener( link.addEventListener("mousedown", (e) => {
'mousedown',
e => {
e.preventDefault(); e.preventDefault();
this._activeResult = resultIndexWorkaround; this._activeResult = resultIndexWorkaround;
this._confirm(this._getActiveSuggestion()); this._confirm(this._getActiveSuggestion());
@ -263,8 +270,8 @@ class AutoCompleteControl {
this._refreshActiveResult(); this._refreshActiveResult();
// display the suggestions offscreen to get the height // display the suggestions offscreen to get the height
this._suggestionDiv.style.left = '-9999px'; this._suggestionDiv.style.left = "-9999px";
this._suggestionDiv.style.top = '-9999px'; this._suggestionDiv.style.top = "-9999px";
this._show(); this._show();
const verticalShift = this._options.verticalShift; const verticalShift = this._options.verticalShift;
const inputRect = this._sourceInputNode.getBoundingClientRect(); const inputRect = this._sourceInputNode.getBoundingClientRect();
@ -275,17 +282,23 @@ class AutoCompleteControl {
// choose where to view the suggestions: if there's more space above // choose where to view the suggestions: if there's more space above
// the input - draw the suggestions above it, otherwise below // the input - draw the suggestions above it, otherwise below
const direction = 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 x = inputRect.left - bodyRect.left;
let y = direction === 1 ? let y =
inputRect.bottom - bodyRect.top - verticalShift : direction === 1
inputRect.top - bodyRect.top - listRect.height + verticalShift; ? inputRect.bottom - bodyRect.top - verticalShift
: inputRect.top -
bodyRect.top -
listRect.height +
verticalShift;
// remove offscreen items until whole suggestion list can fit on the // remove offscreen items until whole suggestion list can fit on the
// screen // screen
while ((y < 0 || y + listRect.height > viewPortHeight) && while (
this._suggestionList.childNodes.length) { (y < 0 || y + listRect.height > viewPortHeight) &&
this._suggestionList.childNodes.length
) {
this._suggestionList.removeChild(this._suggestionList.lastChild); this._suggestionList.removeChild(this._suggestionList.lastChild);
const prevHeight = listRect.height; const prevHeight = listRect.height;
listRect = this._suggestionDiv.getBoundingClientRect(); listRect = this._suggestionDiv.getBoundingClientRect();
@ -295,19 +308,19 @@ class AutoCompleteControl {
} }
} }
this._suggestionDiv.style.left = x + 'px'; this._suggestionDiv.style.left = x + "px";
this._suggestionDiv.style.top = y + 'px'; this._suggestionDiv.style.top = y + "px";
} }
_refreshActiveResult() { _refreshActiveResult() {
let activeItem = this._suggestionList.querySelector('li.active'); let activeItem = this._suggestionList.querySelector("li.active");
if (activeItem) { if (activeItem) {
activeItem.classList.remove('active'); activeItem.classList.remove("active");
} }
if (this._activeResult >= 0) { if (this._activeResult >= 0) {
const allItems = this._suggestionList.querySelectorAll('li'); const allItems = this._suggestionList.querySelectorAll("li");
activeItem = allItems[this._activeResult]; 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 api = require("../api.js");
const misc = require('../util/misc.js'); const misc = require("../util/misc.js");
const events = require('../events.js'); const events = require("../events.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const template = views.getTemplate('comment'); const template = views.getTemplate("comment");
const scoreTemplate = views.getTemplate('score'); const scoreTemplate = views.getTemplate("score");
class CommentControl extends events.EventTarget { class CommentControl extends events.EventTarget {
constructor(hostNode, comment, onlyEditing) { constructor(hostNode, comment, onlyEditing) {
@ -16,104 +16,111 @@ class CommentControl extends events.EventTarget {
this._onlyEditing = onlyEditing; this._onlyEditing = onlyEditing;
if (comment) { if (comment) {
comment.addEventListener( comment.addEventListener("change", (e) => this._evtChange(e));
'change', e => this._evtChange(e)); comment.addEventListener("changeScore", (e) =>
comment.addEventListener( this._evtChangeScore(e)
'changeScore', e => this._evtChangeScore(e)); );
} }
const isLoggedIn = comment && api.isLoggedIn(comment.user); const isLoggedIn = comment && api.isLoggedIn(comment.user);
const infix = isLoggedIn ? 'own' : 'any'; const infix = isLoggedIn ? "own" : "any";
views.replaceContent(this._hostNode, template({ views.replaceContent(
this._hostNode,
template({
comment: comment, comment: comment,
user: comment ? comment.user : api.user, user: comment ? comment.user : api.user,
canViewUsers: api.hasPrivilege('users:view'), canViewUsers: api.hasPrivilege("users:view"),
canEditComment: api.hasPrivilege(`comments:edit:${infix}`), canEditComment: api.hasPrivilege(`comments:edit:${infix}`),
canDeleteComment: api.hasPrivilege(`comments:delete:${infix}`), canDeleteComment: api.hasPrivilege(`comments:delete:${infix}`),
onlyEditing: onlyEditing, onlyEditing: onlyEditing,
})); })
);
if (this._editButtonNodes) { if (this._editButtonNodes) {
for (let node of 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) { if (this._deleteButtonNode) {
this._deleteButtonNode.addEventListener( this._deleteButtonNode.addEventListener("click", (e) =>
'click', e => this._evtDeleteClick(e)); this._evtDeleteClick(e)
);
} }
if (this._previewEditingButtonNode) { if (this._previewEditingButtonNode) {
this._previewEditingButtonNode.addEventListener( this._previewEditingButtonNode.addEventListener("click", (e) =>
'click', e => this._evtPreviewEditingClick(e)); this._evtPreviewEditingClick(e)
);
} }
if (this._saveChangesButtonNode) { if (this._saveChangesButtonNode) {
this._saveChangesButtonNode.addEventListener( this._saveChangesButtonNode.addEventListener("click", (e) =>
'click', e => this._evtSaveChangesClick(e)); this._evtSaveChangesClick(e)
);
} }
if (this._cancelEditingButtonNode) { if (this._cancelEditingButtonNode) {
this._cancelEditingButtonNode.addEventListener( this._cancelEditingButtonNode.addEventListener("click", (e) =>
'click', e => this._evtCancelEditingClick(e)); this._evtCancelEditingClick(e)
);
} }
this._installScore(); this._installScore();
if (onlyEditing) { if (onlyEditing) {
this._selectNav('edit'); this._selectNav("edit");
this._selectTab('edit'); this._selectTab("edit");
} else { } else {
this._selectNav('readonly'); this._selectNav("readonly");
this._selectTab('preview'); this._selectTab("preview");
} }
} }
get _formNode() { get _formNode() {
return this._hostNode.querySelector('form'); return this._hostNode.querySelector("form");
} }
get _scoreContainerNode() { get _scoreContainerNode() {
return this._hostNode.querySelector('.score-container'); return this._hostNode.querySelector(".score-container");
} }
get _editButtonNodes() { get _editButtonNodes() {
return this._hostNode.querySelectorAll('li.edit>a, a.edit'); return this._hostNode.querySelectorAll("li.edit>a, a.edit");
} }
get _previewEditingButtonNode() { get _previewEditingButtonNode() {
return this._hostNode.querySelector('li.preview>a'); return this._hostNode.querySelector("li.preview>a");
} }
get _deleteButtonNode() { get _deleteButtonNode() {
return this._hostNode.querySelector('.delete'); return this._hostNode.querySelector(".delete");
} }
get _upvoteButtonNode() { get _upvoteButtonNode() {
return this._hostNode.querySelector('.upvote'); return this._hostNode.querySelector(".upvote");
} }
get _downvoteButtonNode() { get _downvoteButtonNode() {
return this._hostNode.querySelector('.downvote'); return this._hostNode.querySelector(".downvote");
} }
get _saveChangesButtonNode() { get _saveChangesButtonNode() {
return this._hostNode.querySelector('.save-changes'); return this._hostNode.querySelector(".save-changes");
} }
get _cancelEditingButtonNode() { get _cancelEditingButtonNode() {
return this._hostNode.querySelector('.cancel-editing'); return this._hostNode.querySelector(".cancel-editing");
} }
get _textareaNode() { get _textareaNode() {
return this._hostNode.querySelector('.tab.edit textarea'); return this._hostNode.querySelector(".tab.edit textarea");
} }
get _contentNode() { get _contentNode() {
return this._hostNode.querySelector('.tab.preview .comment-content'); return this._hostNode.querySelector(".tab.preview .comment-content");
} }
get _heightKeeperNode() { get _heightKeeperNode() {
return this._hostNode.querySelector('.keep-height'); return this._hostNode.querySelector(".keep-height");
} }
_installScore() { _installScore() {
@ -122,32 +129,35 @@ class CommentControl extends events.EventTarget {
scoreTemplate({ scoreTemplate({
score: this._comment ? this._comment.score : 0, score: this._comment ? this._comment.score : 0,
ownScore: this._comment ? this._comment.ownScore : 0, ownScore: this._comment ? this._comment.ownScore : 0,
canScore: api.hasPrivilege('comments:score'), canScore: api.hasPrivilege("comments:score"),
})); })
);
if (this._upvoteButtonNode) { if (this._upvoteButtonNode) {
this._upvoteButtonNode.addEventListener( this._upvoteButtonNode.addEventListener("click", (e) =>
'click', e => this._evtScoreClick(e, 1)); this._evtScoreClick(e, 1)
);
} }
if (this._downvoteButtonNode) { if (this._downvoteButtonNode) {
this._downvoteButtonNode.addEventListener( this._downvoteButtonNode.addEventListener("click", (e) =>
'click', e => this._evtScoreClick(e, -1)); this._evtScoreClick(e, -1)
);
} }
} }
enterEditMode() { enterEditMode() {
this._selectNav('edit'); this._selectNav("edit");
this._selectTab('edit'); this._selectTab("edit");
} }
exitEditMode() { exitEditMode() {
if (this._onlyEditing) { if (this._onlyEditing) {
this._selectNav('edit'); this._selectNav("edit");
this._selectTab('edit'); this._selectTab("edit");
this._setText(''); this._setText("");
} else { } else {
this._selectNav('readonly'); this._selectNav("readonly");
this._selectTab('preview'); this._selectTab("preview");
this._setText(this._comment.text); this._setText(this._comment.text);
} }
this._forgetHeight(); this._forgetHeight();
@ -173,27 +183,31 @@ class CommentControl extends events.EventTarget {
_evtScoreClick(e, score) { _evtScoreClick(e, score) {
e.preventDefault(); e.preventDefault();
if (!api.hasPrivilege('comments:score')) { if (!api.hasPrivilege("comments:score")) {
return; return;
} }
this.dispatchEvent(new CustomEvent('score', { this.dispatchEvent(
new CustomEvent("score", {
detail: { detail: {
comment: this._comment, comment: this._comment,
score: this._comment.ownScore === score ? 0 : score, score: this._comment.ownScore === score ? 0 : score,
}, },
})); })
);
} }
_evtDeleteClick(e) { _evtDeleteClick(e) {
e.preventDefault(); 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; return;
} }
this.dispatchEvent(new CustomEvent('delete', { this.dispatchEvent(
new CustomEvent("delete", {
detail: { detail: {
comment: this._comment, comment: this._comment,
}, },
})); })
);
} }
_evtChange(e) { _evtChange(e) {
@ -206,21 +220,24 @@ class CommentControl extends events.EventTarget {
_evtPreviewEditingClick(e) { _evtPreviewEditingClick(e) {
e.preventDefault(); e.preventDefault();
this._contentNode.innerHTML = this._contentNode.innerHTML = misc.formatMarkdown(
misc.formatMarkdown(this._textareaNode.value); this._textareaNode.value
this._selectTab('edit'); );
this._selectTab('preview'); this._selectTab("edit");
this._selectTab("preview");
} }
_evtSaveChangesClick(e) { _evtSaveChangesClick(e) {
e.preventDefault(); e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', { this.dispatchEvent(
new CustomEvent("submit", {
detail: { detail: {
target: this, target: this,
comment: this._comment, comment: this._comment,
text: this._textareaNode.value, text: this._textareaNode.value,
}, },
})); })
);
} }
_evtCancelEditingClick(e) { _evtCancelEditingClick(e) {
@ -234,22 +251,22 @@ class CommentControl extends events.EventTarget {
} }
_selectNav(modeName) { _selectNav(modeName) {
for (let node of this._hostNode.querySelectorAll('nav')) { for (let node of this._hostNode.querySelectorAll("nav")) {
node.classList.toggle('active', node.classList.contains(modeName)); node.classList.toggle("active", node.classList.contains(modeName));
} }
} }
_selectTab(tabName) { _selectTab(tabName) {
this._ensureHeight(); this._ensureHeight();
for (let node of this._hostNode.querySelectorAll('.tab, .tabs li')) { for (let node of this._hostNode.querySelectorAll(".tab, .tabs li")) {
node.classList.toggle('active', node.classList.contains(tabName)); node.classList.toggle("active", node.classList.contains(tabName));
} }
} }
_ensureHeight() { _ensureHeight() {
this._heightKeeperNode.style.minHeight = this._heightKeeperNode.style.minHeight =
this._heightKeeperNode.getBoundingClientRect().height + 'px'; this._heightKeeperNode.getBoundingClientRect().height + "px";
} }
_forgetHeight() { _forgetHeight() {

View file

@ -1,10 +1,10 @@
'use strict'; "use strict";
const events = require('../events.js'); const events = require("../events.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const CommentControl = require('../controls/comment_control.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 { class CommentListControl extends events.EventTarget {
constructor(hostNode, comments, reversed) { constructor(hostNode, comments, reversed) {
@ -13,8 +13,8 @@ class CommentListControl extends events.EventTarget {
this._comments = comments; this._comments = comments;
this._commentIdToNode = {}; this._commentIdToNode = {};
comments.addEventListener('add', e => this._evtAdd(e)); comments.addEventListener("add", (e) => this._evtAdd(e));
comments.addEventListener('remove', e => this._evtRemove(e)); comments.addEventListener("remove", (e) => this._evtRemove(e));
views.replaceContent(this._hostNode, template()); views.replaceContent(this._hostNode, template());
@ -28,16 +28,19 @@ class CommentListControl extends events.EventTarget {
} }
get _commentListNode() { get _commentListNode() {
return this._hostNode.querySelector('ul'); return this._hostNode.querySelector("ul");
} }
_installCommentNode(comment) { _installCommentNode(comment) {
const commentListItemNode = document.createElement('li'); const commentListItemNode = document.createElement("li");
const commentControl = new CommentControl( const commentControl = new CommentControl(
commentListItemNode, comment, false); commentListItemNode,
events.proxyEvent(commentControl, this, 'submit'); comment,
events.proxyEvent(commentControl, this, 'score'); false
events.proxyEvent(commentControl, this, 'delete'); );
events.proxyEvent(commentControl, this, "submit");
events.proxyEvent(commentControl, this, "score");
events.proxyEvent(commentControl, this, "delete");
this._commentIdToNode[comment.id] = commentListItemNode; this._commentIdToNode[comment.id] = commentListItemNode;
this._commentListNode.appendChild(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_OPENED = "fa-chevron-down";
const ICON_CLASS_CLOSED = 'fa-chevron-up'; 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 { class ExpanderControl {
constructor(name, title, nodes) { constructor(name, title, nodes) {
this._name = name; this._name = name;
nodes = Array.from(nodes).filter(n => n); nodes = Array.from(nodes).filter((n) => n);
if (!nodes.length) { if (!nodes.length) {
return; return;
} }
const expanderNode = template({title: title}); const expanderNode = template({ title: title });
const toggleLinkNode = expanderNode.querySelector('a'); const toggleLinkNode = expanderNode.querySelector("a");
const toggleIconNode = expanderNode.querySelector('i'); const toggleIconNode = expanderNode.querySelector("i");
const expanderContentNode = expanderNode.querySelector('div'); const expanderContentNode = expanderNode.querySelector("div");
toggleLinkNode.addEventListener('click', e => this._evtToggleClick(e)); toggleLinkNode.addEventListener("click", (e) =>
this._evtToggleClick(e)
);
nodes[0].parentNode.insertBefore(expanderNode, nodes[0]); nodes[0].parentNode.insertBefore(expanderNode, nodes[0]);
@ -32,29 +34,30 @@ class ExpanderControl {
this._toggleIconNode = toggleIconNode; this._toggleIconNode = toggleIconNode;
expanderNode.classList.toggle( expanderNode.classList.toggle(
'collapsed', "collapsed",
this._allStates[this._name] === undefined ? this._allStates[this._name] === undefined
false : ? false
!this._allStates[this._name]); : !this._allStates[this._name]
);
this._syncIcon(); this._syncIcon();
} }
// eslint-disable-next-line accessor-pairs // eslint-disable-next-line accessor-pairs
set title(newTitle) { set title(newTitle) {
if (this._expanderNode) { if (this._expanderNode) {
this._expanderNode this._expanderNode.querySelector(
.querySelector('header span') "header span"
.textContent = newTitle; ).textContent = newTitle;
} }
} }
get _isOpened() { get _isOpened() {
return !this._expanderNode.classList.contains('collapsed'); return !this._expanderNode.classList.contains("collapsed");
} }
get _allStates() { get _allStates() {
try { try {
return JSON.parse(localStorage.getItem('expander')) || {}; return JSON.parse(localStorage.getItem("expander")) || {};
} catch (e) { } catch (e) {
return {}; return {};
} }
@ -63,12 +66,12 @@ class ExpanderControl {
_save() { _save() {
const newStates = Object.assign({}, this._allStates); const newStates = Object.assign({}, this._allStates);
newStates[this._name] = this._isOpened; newStates[this._name] = this._isOpened;
localStorage.setItem('expander', JSON.stringify(newStates)); localStorage.setItem("expander", JSON.stringify(newStates));
} }
_evtToggleClick(e) { _evtToggleClick(e) {
e.preventDefault(); e.preventDefault();
this._expanderNode.classList.toggle('collapsed'); this._expanderNode.classList.toggle("collapsed");
this._save(); this._save();
this._syncIcon(); this._syncIcon();
} }

View file

@ -1,9 +1,9 @@
'use strict'; "use strict";
const events = require('../events.js'); const events = require("../events.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const template = views.getTemplate('file-dropper'); const template = views.getTemplate("file-dropper");
const KEY_RETURN = 13; const KEY_RETURN = 13;
@ -17,37 +17,42 @@ class FileDropperControl extends events.EventTarget {
allowMultiple: options.allowMultiple, allowMultiple: options.allowMultiple,
allowUrls: options.allowUrls, allowUrls: options.allowUrls,
lock: options.lock, lock: options.lock,
id: 'file-' + Math.random().toString(36).substring(7), id: "file-" + Math.random().toString(36).substring(7),
urlPlaceholder: urlPlaceholder:
options.urlPlaceholder || 'Alternatively, paste an URL here.', options.urlPlaceholder || "Alternatively, paste an URL here.",
}); });
this._dropperNode = source.querySelector('.file-dropper'); this._dropperNode = source.querySelector(".file-dropper");
this._urlInputNode = source.querySelector('input[type=text]'); this._urlInputNode = source.querySelector("input[type=text]");
this._urlConfirmButtonNode = source.querySelector('button'); this._urlConfirmButtonNode = source.querySelector("button");
this._fileInputNode = source.querySelector('input[type=file]'); this._fileInputNode = source.querySelector("input[type=file]");
this._fileInputNode.style.display = 'none'; this._fileInputNode.style.display = "none";
this._fileInputNode.multiple = options.allowMultiple || false; this._fileInputNode.multiple = options.allowMultiple || false;
this._counter = 0; this._counter = 0;
this._dropperNode.addEventListener( this._dropperNode.addEventListener("dragenter", (e) =>
'dragenter', e => this._evtDragEnter(e)); this._evtDragEnter(e)
this._dropperNode.addEventListener( );
'dragleave', e => this._evtDragLeave(e)); this._dropperNode.addEventListener("dragleave", (e) =>
this._dropperNode.addEventListener( this._evtDragLeave(e)
'dragover', e => this._evtDragOver(e)); );
this._dropperNode.addEventListener( this._dropperNode.addEventListener("dragover", (e) =>
'drop', e => this._evtDrop(e)); this._evtDragOver(e)
this._fileInputNode.addEventListener( );
'change', e => this._evtFileChange(e)); this._dropperNode.addEventListener("drop", (e) => this._evtDrop(e));
this._fileInputNode.addEventListener("change", (e) =>
this._evtFileChange(e)
);
if (this._urlInputNode) { if (this._urlInputNode) {
this._urlInputNode.addEventListener( this._urlInputNode.addEventListener("keydown", (e) =>
'keydown', e => this._evtUrlInputKeyDown(e)); this._evtUrlInputKeyDown(e)
);
} }
if (this._urlConfirmButtonNode) { if (this._urlConfirmButtonNode) {
this._urlConfirmButtonNode.addEventListener( this._urlConfirmButtonNode.addEventListener("click", (e) =>
'click', e => this._evtUrlConfirmButtonClick(e)); this._evtUrlConfirmButtonClick(e)
);
} }
this._originalHtml = this._dropperNode.innerHTML; this._originalHtml = this._dropperNode.innerHTML;
@ -56,24 +61,27 @@ class FileDropperControl extends events.EventTarget {
reset() { reset() {
this._dropperNode.innerHTML = this._originalHtml; this._dropperNode.innerHTML = this._originalHtml;
this.dispatchEvent(new CustomEvent('reset')); this.dispatchEvent(new CustomEvent("reset"));
} }
_emitFiles(files) { _emitFiles(files) {
files = Array.from(files); files = Array.from(files);
if (this._options.lock) { if (this._options.lock) {
this._dropperNode.innerText = this._dropperNode.innerText = files
files.map(file => file.name).join(', '); .map((file) => file.name)
.join(", ");
} }
this.dispatchEvent( this.dispatchEvent(
new CustomEvent('fileadd', {detail: {files: files}})); new CustomEvent("fileadd", { detail: { files: files } })
);
} }
_emitUrls(urls) { _emitUrls(urls) {
urls = Array.from(urls).map(url => url.trim()); urls = Array.from(urls).map((url) => url.trim());
if (this._options.lock) { if (this._options.lock) {
this._dropperNode.innerText = this._dropperNode.innerText = urls
urls.map(url => url.split(/\//).reverse()[0]).join(', '); .map((url) => url.split(/\//).reverse()[0])
.join(", ");
} }
for (let url of urls) { for (let url of urls) {
if (!url) { if (!url) {
@ -84,18 +92,20 @@ class FileDropperControl extends events.EventTarget {
return; return;
} }
} }
this.dispatchEvent(new CustomEvent('urladd', {detail: {urls: urls}})); this.dispatchEvent(
new CustomEvent("urladd", { detail: { urls: urls } })
);
} }
_evtDragEnter(e) { _evtDragEnter(e) {
this._dropperNode.classList.add('active'); this._dropperNode.classList.add("active");
this._counter++; this._counter++;
} }
_evtDragLeave(e) { _evtDragLeave(e) {
this._counter--; this._counter--;
if (this._counter === 0) { 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) { _evtDrop(e) {
e.preventDefault(); e.preventDefault();
this._dropperNode.classList.remove('active'); this._dropperNode.classList.remove("active");
if (!e.dataTransfer.files.length) { 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) { 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); this._emitFiles(e.dataTransfer.files);
} }
@ -124,16 +134,16 @@ class FileDropperControl extends events.EventTarget {
return; return;
} }
e.preventDefault(); e.preventDefault();
this._dropperNode.classList.remove('active'); this._dropperNode.classList.remove("active");
this._emitUrls(this._urlInputNode.value.split(/[\r\n]/)); this._emitUrls(this._urlInputNode.value.split(/[\r\n]/));
this._urlInputNode.value = ''; this._urlInputNode.value = "";
} }
_evtUrlConfirmButtonClick(e) { _evtUrlConfirmButtonClick(e) {
e.preventDefault(); e.preventDefault();
this._dropperNode.classList.remove('active'); this._dropperNode.classList.remove("active");
this._emitUrls(this._urlInputNode.value.split(/[\r\n]/)); 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 misc = require("../util/misc.js");
const PoolList = require('../models/pool_list.js'); const PoolList = require("../models/pool_list.js");
const AutoCompleteControl = require('./auto_complete_control.js'); const AutoCompleteControl = require("./auto_complete_control.js");
function _poolListToMatches(pools, options) { function _poolListToMatches(pools, options) {
return [...pools].sort((pool1, pool2) => { return [...pools]
.sort((pool1, pool2) => {
return pool2.postCount - pool1.postCount; return pool2.postCount - pool1.postCount;
}).map(pool => { })
let cssName = misc.makeCssName(pool.category, 'pool'); .map((pool) => {
const caption = ( let cssName = misc.makeCssName(pool.category, "pool");
'<span class="' + cssName + '">' const caption =
+ misc.escapeHtml(pool.names[0] + ' (' + pool.postCount + ')') '<span class="' +
+ '</span>'); cssName +
'">' +
misc.escapeHtml(pool.names[0] + " (" + pool.postCount + ")") +
"</span>";
return { return {
caption: caption, caption: caption,
value: pool, value: pool,
@ -24,20 +28,27 @@ class PoolAutoCompleteControl extends AutoCompleteControl {
constructor(input, options) { constructor(input, options) {
const minLengthForPartialSearch = 3; const minLengthForPartialSearch = 3;
options.getMatches = text => { options.getMatches = (text) => {
const term = misc.escapeSearchTerm(text); const term = misc.escapeSearchTerm(text);
const query = ( const query =
text.length < minLengthForPartialSearch (text.length < minLengthForPartialSearch
? term + '*' ? term + "*"
: '*' + term + '*') + ' sort:post-count'; : "*" + term + "*") + " sort:post-count";
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
PoolList.search( PoolList.search(query, 0, this._options.maxResults, [
query, 0, this._options.maxResults, ['id', 'names', 'category', 'postCount', 'version']) "id",
.then( "names",
response => resolve( "category",
_poolListToMatches(response.results, this._options)), "postCount",
reject); "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 api = require("../api.js");
const pools = require('../pools.js'); const pools = require("../pools.js");
const misc = require('../util/misc.js'); const misc = require("../util/misc.js");
const uri = require('../util/uri.js'); const uri = require("../util/uri.js");
const Pool = require('../models/pool.js'); const Pool = require("../models/pool.js");
const settings = require('../models/settings.js'); const settings = require("../models/settings.js");
const events = require('../events.js'); const events = require("../events.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const PoolAutoCompleteControl = require('./pool_auto_complete_control.js'); const PoolAutoCompleteControl = require("./pool_auto_complete_control.js");
const KEY_SPACE = 32; const KEY_SPACE = 32;
const KEY_RETURN = 13; const KEY_RETURN = 13;
const SOURCE_INIT = 'init'; const SOURCE_INIT = "init";
const SOURCE_IMPLICATION = 'implication'; const SOURCE_IMPLICATION = "implication";
const SOURCE_USER_INPUT = 'user-input'; const SOURCE_USER_INPUT = "user-input";
const SOURCE_CLIPBOARD = 'clipboard'; const SOURCE_CLIPBOARD = "clipboard";
const template = views.getTemplate('pool-input'); const template = views.getTemplate("pool-input");
function _fadeOutListItemNodeStatus(listItemNode) { function _fadeOutListItemNodeStatus(listItemNode) {
if (listItemNode.classList.length) { if (listItemNode.classList.length) {
@ -27,8 +27,7 @@ function _fadeOutListItemNodeStatus(listItemNode) {
} }
listItemNode.fadeTimeout = window.setTimeout(() => { listItemNode.fadeTimeout = window.setTimeout(() => {
while (listItemNode.classList.length) { while (listItemNode.classList.length) {
listItemNode.classList.remove( listItemNode.classList.remove(listItemNode.classList.item(0));
listItemNode.classList.item(0));
} }
listItemNode.fadeTimeout = null; listItemNode.fadeTimeout = null;
}, 2500); }, 2500);
@ -45,29 +44,33 @@ class PoolInputControl extends events.EventTarget {
// dom // dom
const editAreaNode = template(); const editAreaNode = template();
this._editAreaNode = editAreaNode; this._editAreaNode = editAreaNode;
this._poolInputNode = editAreaNode.querySelector('input'); this._poolInputNode = editAreaNode.querySelector("input");
this._poolListNode = editAreaNode.querySelector('ul.compact-pools'); this._poolListNode = editAreaNode.querySelector("ul.compact-pools");
this._autoCompleteControl = new PoolAutoCompleteControl( this._autoCompleteControl = new PoolAutoCompleteControl(
this._poolInputNode, { this._poolInputNode,
{
getTextToFind: () => { getTextToFind: () => {
return this._poolInputNode.value; return this._poolInputNode.value;
}, },
confirm: pool => { confirm: (pool) => {
this._poolInputNode.value = ''; this._poolInputNode.value = "";
this.addPool(pool, SOURCE_USER_INPUT); this.addPool(pool, SOURCE_USER_INPUT);
}, },
delete: pool => { delete: (pool) => {
this._poolInputNode.value = ''; this._poolInputNode.value = "";
this.deletePool(pool); this.deletePool(pool);
}, },
verticalShift: -2 verticalShift: -2,
}); }
);
// show // show
this._hostNode.style.display = 'none'; this._hostNode.style.display = "none";
this._hostNode.parentNode.insertBefore( this._hostNode.parentNode.insertBefore(
this._editAreaNode, hostNode.nextSibling); this._editAreaNode,
hostNode.nextSibling
);
// add existing pools // add existing pools
for (let pool of [...this.pools]) { for (let pool of [...this.pools]) {
@ -81,19 +84,21 @@ class PoolInputControl extends events.EventTarget {
return Promise.resolve(); return Promise.resolve();
} }
this.pools.add(pool, false) this.pools.add(pool, false);
const listItemNode = this._createListItemNode(pool); const listItemNode = this._createListItemNode(pool);
if (!pool.category) { if (!pool.category) {
listItemNode.classList.add('new'); listItemNode.classList.add("new");
} }
this._poolListNode.prependChild(listItemNode); this._poolListNode.prependChild(listItemNode);
_fadeOutListItemNodeStatus(listItemNode); _fadeOutListItemNodeStatus(listItemNode);
this.dispatchEvent(new CustomEvent('add', { this.dispatchEvent(
detail: {pool: pool, source: source}, new CustomEvent("add", {
})); detail: { pool: pool, source: source },
this.dispatchEvent(new CustomEvent('change')); })
);
this.dispatchEvent(new CustomEvent("change"));
return Promise.resolve(); return Promise.resolve();
} }
@ -107,52 +112,57 @@ class PoolInputControl extends events.EventTarget {
this._deleteListItemNode(pool); this._deleteListItemNode(pool);
this.dispatchEvent(new CustomEvent('remove', { this.dispatchEvent(
detail: {pool: pool}, new CustomEvent("remove", {
})); detail: { pool: pool },
this.dispatchEvent(new CustomEvent('change')); })
);
this.dispatchEvent(new CustomEvent("change"));
} }
_createListItemNode(pool) { _createListItemNode(pool) {
const className = pool.category ? const className = pool.category
misc.makeCssName(pool.category, 'pool') : ? misc.makeCssName(pool.category, "pool")
null; : null;
const poolLinkNode = document.createElement('a'); const poolLinkNode = document.createElement("a");
if (className) { if (className) {
poolLinkNode.classList.add(className); poolLinkNode.classList.add(className);
} }
poolLinkNode.setAttribute( poolLinkNode.setAttribute(
'href', uri.formatClientLink('pool', pool.names[0])); "href",
uri.formatClientLink("pool", pool.names[0])
);
const poolIconNode = document.createElement('i'); const poolIconNode = document.createElement("i");
poolIconNode.classList.add('fa'); poolIconNode.classList.add("fa");
poolIconNode.classList.add('fa-pool'); poolIconNode.classList.add("fa-pool");
poolLinkNode.appendChild(poolIconNode); poolLinkNode.appendChild(poolIconNode);
const searchLinkNode = document.createElement('a'); const searchLinkNode = document.createElement("a");
if (className) { if (className) {
searchLinkNode.classList.add(className); searchLinkNode.classList.add(className);
} }
searchLinkNode.setAttribute( searchLinkNode.setAttribute(
'href', uri.formatClientLink( "href",
'posts', {query: "pool:" + pool.id})); uri.formatClientLink("posts", { query: "pool:" + pool.id })
searchLinkNode.textContent = pool.names[0] + ' '; );
searchLinkNode.textContent = pool.names[0] + " ";
const usagesNode = document.createElement('span'); const usagesNode = document.createElement("span");
usagesNode.classList.add('pool-usages'); usagesNode.classList.add("pool-usages");
usagesNode.setAttribute('data-pseudo-content', pool.postCount); usagesNode.setAttribute("data-pseudo-content", pool.postCount);
const removalLinkNode = document.createElement('a'); const removalLinkNode = document.createElement("a");
removalLinkNode.classList.add('remove-pool'); removalLinkNode.classList.add("remove-pool");
removalLinkNode.setAttribute('href', ''); removalLinkNode.setAttribute("href", "");
removalLinkNode.setAttribute('data-pseudo-content', '×'); removalLinkNode.setAttribute("data-pseudo-content", "×");
removalLinkNode.addEventListener('click', e => { removalLinkNode.addEventListener("click", (e) => {
e.preventDefault(); e.preventDefault();
this.deletePool(pool); this.deletePool(pool);
}); });
const listItemNode = document.createElement('li'); const listItemNode = document.createElement("li");
listItemNode.appendChild(removalLinkNode); listItemNode.appendChild(removalLinkNode);
listItemNode.appendChild(poolLinkNode); listItemNode.appendChild(poolLinkNode);
listItemNode.appendChild(searchLinkNode); listItemNode.appendChild(searchLinkNode);

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

@ -1,82 +1,101 @@
'use strict'; "use strict";
require('./util/polyfill.js'); require("./util/polyfill.js");
const misc = require('./util/misc.js'); const misc = require("./util/misc.js");
const views = require('./util/views.js'); const views = require("./util/views.js");
const router = require('./router.js'); const router = require("./router.js");
history.scrollRestoration = 'manual'; history.scrollRestoration = "manual";
router.exit( router.exit(null, (ctx, next) => {
null,
(ctx, next) => {
ctx.state.scrollX = window.scrollX; ctx.state.scrollX = window.scrollX;
ctx.state.scrollY = window.scrollY; ctx.state.scrollY = window.scrollY;
router.replace(router.url, ctx.state); router.replace(router.url, ctx.state);
if (misc.confirmPageExit()) { if (misc.confirmPageExit()) {
next(); next();
} }
}); });
const mousetrap = require('mousetrap'); const mousetrap = require("mousetrap");
router.enter( router.enter(null, (ctx, next) => {
null,
(ctx, next) => {
mousetrap.reset(); mousetrap.reset();
next(); next();
}); });
const tags = require('./tags.js'); const tags = require("./tags.js");
const pools = require('./pools.js'); const pools = require("./pools.js");
const api = require('./api.js'); const api = require("./api.js");
api.fetchConfig().then(() => { api.fetchConfig()
.then(
() => {
// register controller routes // register controller routes
let controllers = []; let controllers = [];
controllers.push(require('./controllers/home_controller.js')); controllers.push(require("./controllers/home_controller.js"));
controllers.push(require('./controllers/help_controller.js')); controllers.push(require("./controllers/help_controller.js"));
controllers.push(require('./controllers/auth_controller.js')); controllers.push(require("./controllers/auth_controller.js"));
controllers.push(require('./controllers/password_reset_controller.js')); controllers.push(
controllers.push(require('./controllers/comments_controller.js')); require("./controllers/password_reset_controller.js")
controllers.push(require('./controllers/snapshots_controller.js')); );
controllers.push(require('./controllers/post_detail_controller.js')); controllers.push(require("./controllers/comments_controller.js"));
controllers.push(require('./controllers/post_main_controller.js')); controllers.push(require("./controllers/snapshots_controller.js"));
controllers.push(require('./controllers/post_list_controller.js')); controllers.push(
controllers.push(require('./controllers/post_upload_controller.js')); require("./controllers/post_detail_controller.js")
controllers.push(require('./controllers/tag_controller.js')); );
controllers.push(require('./controllers/tag_list_controller.js')); controllers.push(require("./controllers/post_main_controller.js"));
controllers.push(require('./controllers/tag_categories_controller.js')); controllers.push(require("./controllers/post_list_controller.js"));
controllers.push(require('./controllers/pool_create_controller.js')); controllers.push(
controllers.push(require('./controllers/pool_controller.js')); require("./controllers/post_upload_controller.js")
controllers.push(require('./controllers/pool_list_controller.js')); );
controllers.push(require('./controllers/pool_categories_controller.js')); controllers.push(require("./controllers/tag_controller.js"));
controllers.push(require('./controllers/settings_controller.js')); controllers.push(require("./controllers/tag_list_controller.js"));
controllers.push(require('./controllers/user_controller.js')); controllers.push(
controllers.push(require('./controllers/user_list_controller.js')); require("./controllers/tag_categories_controller.js")
controllers.push(require('./controllers/user_registration_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 // 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) { for (let controller of controllers) {
controller(router); controller(router);
} }
}, error => { },
window.alert('Could not fetch basic configuration from server'); (error) => {
}).then(() => { window.alert("Could not fetch basic configuration from server");
api.loginFromCookies().then(() => { }
)
.then(() => {
api.loginFromCookies().then(
() => {
tags.refreshCategoryColorMap(); tags.refreshCategoryColorMap();
pools.refreshCategoryColorMap(); pools.refreshCategoryColorMap();
router.start(); router.start();
}, error => { },
if (window.location.href.indexOf('login') !== -1) { (error) => {
if (window.location.href.indexOf("login") !== -1) {
api.forget(); api.forget();
router.start(); router.start();
} else { } else {
const ctx = router.start('/'); const ctx = router.start("/");
ctx.controller.showError( ctx.controller.showError(
'An error happened while trying to log you in: ' + "An error happened while trying to log you in: " +
error.message); 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 { class AbstractList extends events.EventTarget {
constructor() { constructor() {
@ -13,13 +13,15 @@ class AbstractList extends events.EventTarget {
for (let item of response) { for (let item of response) {
const addedItem = this._itemClass.fromResponse(item); const addedItem = this._itemClass.fromResponse(item);
if (addedItem.addEventListener) { if (addedItem.addEventListener) {
addedItem.addEventListener('delete', e => { addedItem.addEventListener("delete", (e) => {
ret.remove(addedItem); ret.remove(addedItem);
}); });
addedItem.addEventListener('change', e => { addedItem.addEventListener("change", (e) => {
ret.dispatchEvent(new CustomEvent('change', { ret.dispatchEvent(
new CustomEvent("change", {
detail: e.detail, detail: e.detail,
})); })
);
}); });
} }
ret._list.push(addedItem); ret._list.push(addedItem);
@ -29,28 +31,32 @@ class AbstractList extends events.EventTarget {
sync(plainList) { sync(plainList) {
this.clear(); this.clear();
for (let item of (plainList || [])) { for (let item of plainList || []) {
this.add(this.constructor._itemClass.fromResponse(item)); this.add(this.constructor._itemClass.fromResponse(item));
} }
} }
add(item) { add(item) {
if (item.addEventListener) { if (item.addEventListener) {
item.addEventListener('delete', e => { item.addEventListener("delete", (e) => {
this.remove(item); this.remove(item);
}); });
item.addEventListener('change', e => { item.addEventListener("change", (e) => {
this.dispatchEvent(new CustomEvent('change', { this.dispatchEvent(
new CustomEvent("change", {
detail: e.detail, detail: e.detail,
})); })
);
}); });
} }
this._list.push(item); this._list.push(item);
const detail = {}; const detail = {};
detail[this.constructor._itemName] = item; detail[this.constructor._itemName] = item;
this.dispatchEvent(new CustomEvent('add', { this.dispatchEvent(
new CustomEvent("add", {
detail: detail, detail: detail,
})); })
);
} }
clear() { clear() {
@ -67,9 +73,11 @@ class AbstractList extends events.EventTarget {
this._list.splice(index, 1); this._list.splice(index, 1);
const detail = {}; const detail = {};
detail[this.constructor._itemName] = itemToRemove; detail[this.constructor._itemName] = itemToRemove;
this.dispatchEvent(new CustomEvent('remove', { this.dispatchEvent(
new CustomEvent("remove", {
detail: detail, detail: detail,
})); })
);
return; return;
} }
} }

View file

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

View file

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

View file

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

View file

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

View file

@ -1,12 +1,11 @@
'use strict'; "use strict";
const AbstractList = require('./abstract_list.js'); const AbstractList = require("./abstract_list.js");
const Note = require('./note.js'); const Note = require("./note.js");
class NoteList extends AbstractList { class NoteList extends AbstractList {}
}
NoteList._itemClass = Note; NoteList._itemClass = Note;
NoteList._itemName = 'note'; NoteList._itemName = "note";
module.exports = NoteList; 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 { class Point extends events.EventTarget {
constructor(x, y) { constructor(x, y) {
@ -19,12 +19,16 @@ class Point extends events.EventTarget {
set x(value) { set x(value) {
this._x = value; this._x = value;
this.dispatchEvent(new CustomEvent('change', {detail: {point: this}})); this.dispatchEvent(
new CustomEvent("change", { detail: { point: this } })
);
} }
set y(value) { set y(value) {
this._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 AbstractList = require("./abstract_list.js");
const Point = require('./point.js'); const Point = require("./point.js");
class PointList extends AbstractList { class PointList extends AbstractList {
get firstPoint() { get firstPoint() {
@ -18,6 +18,6 @@ class PointList extends AbstractList {
} }
PointList._itemClass = Point; PointList._itemClass = Point;
PointList._itemName = 'point'; PointList._itemName = "point";
module.exports = PointList; module.exports = PointList;

View file

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

View file

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

View file

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

View file

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

View file

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

View file

@ -1,35 +1,37 @@
'use strict'; "use strict";
const settings = require('../models/settings.js'); const settings = require("../models/settings.js");
const api = require('../api.js'); const api = require("../api.js");
const uri = require('../util/uri.js'); const uri = require("../util/uri.js");
const AbstractList = require('./abstract_list.js'); const AbstractList = require("./abstract_list.js");
const Post = require('./post.js'); const Post = require("./post.js");
class PostList extends AbstractList { class PostList extends AbstractList {
static getAround(id, searchQuery) { static getAround(id, searchQuery) {
return api.get( return api.get(
uri.formatApiLink( uri.formatApiLink("post", id, "around", {
'post', id, 'around', { query: PostList._decorateSearchQuery(searchQuery || ""),
query: PostList._decorateSearchQuery(searchQuery || ''), fields: "id",
fields: 'id', })
})); );
} }
static search(text, offset, limit, fields) { static search(text, offset, limit, fields) {
return api.get( return api
uri.formatApiLink( .get(
'posts', { uri.formatApiLink("posts", {
query: PostList._decorateSearchQuery(text || ''), query: PostList._decorateSearchQuery(text || ""),
offset: offset, offset: offset,
limit: limit, limit: limit,
fields: fields.join(','), fields: fields.join(","),
})) })
.then(response => { )
return Promise.resolve(Object.assign( .then((response) => {
{}, return Promise.resolve(
response, Object.assign({}, response, {
{results: PostList.fromResponse(response.results)})); results: PostList.fromResponse(response.results),
})
);
}); });
} }
@ -43,7 +45,7 @@ class PostList extends AbstractList {
} }
} }
if (disabledSafety.length) { if (disabledSafety.length) {
text = `-rating:${disabledSafety.join(',')} ${text}`; text = `-rating:${disabledSafety.join(",")} ${text}`;
} }
} }
return text.trim(); return text.trim();
@ -63,7 +65,7 @@ class PostList extends AbstractList {
return; return;
} }
let post = Post.fromResponse({id: id}); let post = Post.fromResponse({ id: id });
this.add(post); this.add(post);
} }
@ -77,6 +79,6 @@ class PostList extends AbstractList {
} }
PostList._itemClass = Post; PostList._itemClass = Post;
PostList._itemName = 'post'; PostList._itemName = "post";
module.exports = PostList; 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 = { const defaultSettings = {
listPosts: { listPosts: {
@ -12,7 +12,7 @@ const defaultSettings = {
endlessScroll: false, endlessScroll: false,
keyboardShortcuts: true, keyboardShortcuts: true,
transparencyGrid: true, transparencyGrid: true,
fitMode: 'fit-both', fitMode: "fit-both",
tagSuggestions: true, tagSuggestions: true,
autoplayVideos: false, autoplayVideos: false,
postsPerPage: 42, postsPerPage: 42,
@ -28,7 +28,7 @@ class Settings extends events.EventTarget {
_getFromLocalStorage() { _getFromLocalStorage() {
let ret = Object.assign({}, defaultSettings); let ret = Object.assign({}, defaultSettings);
try { try {
Object.assign(ret, JSON.parse(localStorage.getItem('settings'))); Object.assign(ret, JSON.parse(localStorage.getItem("settings")));
} catch (e) { } catch (e) {
// continue regardless of error // continue regardless of error
} }
@ -37,14 +37,16 @@ class Settings extends events.EventTarget {
save(newSettings, silent) { save(newSettings, silent) {
newSettings = Object.assign(this.cache, newSettings); newSettings = Object.assign(this.cache, newSettings);
localStorage.setItem('settings', JSON.stringify(newSettings)); localStorage.setItem("settings", JSON.stringify(newSettings));
this.cache = this._getFromLocalStorage(); this.cache = this._getFromLocalStorage();
if (silent !== true) { if (silent !== true) {
this.dispatchEvent(new CustomEvent('change', { this.dispatchEvent(
new CustomEvent("change", {
detail: { detail: {
settings: this.cache, settings: this.cache,
}, },
})); })
);
} }
} }

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

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

View file

@ -1,22 +1,23 @@
'use strict'; "use strict";
const misc = require('./util/misc.js'); const misc = require("./util/misc.js");
const TagCategoryList = require('./models/tag_category_list.js'); const TagCategoryList = require("./models/tag_category_list.js");
let _stylesheet = null; let _stylesheet = null;
function refreshCategoryColorMap() { function refreshCategoryColorMap() {
return TagCategoryList.get().then(response => { return TagCategoryList.get().then((response) => {
if (_stylesheet) { if (_stylesheet) {
document.head.removeChild(_stylesheet); document.head.removeChild(_stylesheet);
} }
_stylesheet = document.createElement('style'); _stylesheet = document.createElement("style");
document.head.appendChild(_stylesheet); document.head.appendChild(_stylesheet);
for (let category of response.results) { for (let category of response.results) {
const ruleName = misc.makeCssName(category.name, 'tag'); const ruleName = misc.makeCssName(category.name, "tag");
_stylesheet.sheet.insertRule( _stylesheet.sheet.insertRule(
`.${ruleName} { color: ${category.color} }`, `.${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,12 +1,12 @@
'use strict'; "use strict";
const mousetrap = require('mousetrap'); const mousetrap = require("mousetrap");
const settings = require('../models/settings.js'); const settings = require("../models/settings.js");
let paused = false; let paused = false;
const _originalStopCallback = mousetrap.prototype.stopCallback; const _originalStopCallback = mousetrap.prototype.stopCallback;
// eslint-disable-next-line func-names // eslint-disable-next-line func-names
mousetrap.prototype.stopCallback = function(...args) { mousetrap.prototype.stopCallback = function (...args) {
var self = this; var self = this;
if (paused) { if (paused) {
return true; return true;

View file

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

View file

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

View file

@ -1,4 +1,4 @@
'use strict'; "use strict";
let callbacks = []; let callbacks = [];
let running = false; let running = false;
@ -15,7 +15,7 @@ function resize() {
} }
function runCallbacks() { function runCallbacks() {
callbacks.forEach(callback => { callbacks.forEach((callback) => {
callback(); callback();
}); });
running = false; running = false;
@ -26,8 +26,8 @@ function add(callback) {
} }
function remove(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}; module.exports = { add: add, remove: remove };

View file

@ -1,11 +1,11 @@
/* eslint-disable func-names, no-extend-native */ /* eslint-disable func-names, no-extend-native */
'use strict'; "use strict";
// fix iterating over NodeList in Chrome and Opera // fix iterating over NodeList in Chrome and Opera
NodeList.prototype[Symbol.iterator] = Array.prototype[Symbol.iterator]; NodeList.prototype[Symbol.iterator] = Array.prototype[Symbol.iterator];
NodeList.prototype.querySelector = function(...args) { NodeList.prototype.querySelector = function (...args) {
for (let node of this) { for (let node of this) {
if (node.nodeType === 3) { if (node.nodeType === 3) {
continue; continue;
@ -18,7 +18,7 @@ NodeList.prototype.querySelector = function(...args) {
return null; return null;
}; };
NodeList.prototype.querySelectorAll = function(...args) { NodeList.prototype.querySelectorAll = function (...args) {
let result = []; let result = [];
for (let node of this) { for (let node of this) {
if (node.nodeType === 3) { if (node.nodeType === 3) {
@ -32,7 +32,7 @@ NodeList.prototype.querySelectorAll = function(...args) {
}; };
// non standard // non standard
Node.prototype.prependChild = function(child) { Node.prototype.prependChild = function (child) {
if (this.firstChild) { if (this.firstChild) {
this.insertBefore(child, this.firstChild); this.insertBefore(child, this.firstChild);
} else { } else {
@ -41,29 +41,25 @@ Node.prototype.prependChild = function(child) {
}; };
// non standard // non standard
Promise.prototype.always = function(onResolveOrReject) { Promise.prototype.always = function (onResolveOrReject) {
return this.then( return this.then(onResolveOrReject, (reason) => {
onResolveOrReject,
reason => {
onResolveOrReject(reason); onResolveOrReject(reason);
throw reason; throw reason;
}); });
}; };
// non standard // non standard
Number.prototype.between = function(a, b, inclusive) { Number.prototype.between = function (a, b, inclusive) {
const min = Math.min(a, b); const min = Math.min(a, b);
const max = Math.max(a, b); const max = Math.max(a, b);
return inclusive ? return inclusive ? this >= min && this <= max : this > min && this < max;
this >= min && this <= max :
this > min && this < max;
}; };
// non standard // non standard
Promise.prototype.abort = () => {}; Promise.prototype.abort = () => {};
// non standard // non standard
Date.prototype.addDays = function(days) { Date.prototype.addDays = function (days) {
let dat = new Date(this.valueOf()); let dat = new Date(this.valueOf());
dat.setDate(dat.getDate() + days); dat.setDate(dat.getDate() + days);
return dat; return dat;

View file

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

View file

@ -1,14 +1,16 @@
'use strict'; "use strict";
const misc = require('./misc.js'); const misc = require("./misc.js");
const keyboard = require('../util/keyboard.js'); const keyboard = require("../util/keyboard.js");
const views = require('./views.js'); const views = require("./views.js");
function searchInputNodeFocusHelper(inputNode) { function searchInputNodeFocusHelper(inputNode) {
keyboard.bind('q', () => { keyboard.bind("q", () => {
inputNode.focus(); inputNode.focus();
inputNode.setSelectionRange( 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 = { const direction = {
NONE: null, NONE: null,
LEFT: 'left', LEFT: "left",
RIGHT: 'right', RIGHT: "right",
DOWN: 'down', DOWN: "down",
UP: 'up' UP: "up",
}; };
function handleTouchStart(handler, evt) { function handleTouchStart(handler, evt) {
@ -58,11 +58,13 @@ function handleTouchEnd(handler) {
} }
class Touch { class Touch {
constructor(target, constructor(
target,
swipeLeft = () => {}, swipeLeft = () => {},
swipeRight = () => {}, swipeRight = () => {},
swipeUp = () => {}, swipeUp = () => {},
swipeDown = () => {}) { swipeDown = () => {}
) {
this._target = target; this._target = target;
this._swipeLeftTask = swipeLeft; this._swipeLeftTask = swipeLeft;
@ -74,16 +76,13 @@ class Touch {
this._yStart = null; this._yStart = null;
this._direction = direction.NONE; this._direction = direction.NONE;
this._target.addEventListener('touchstart', this._target.addEventListener("touchstart", (evt) => {
evt => {
handleTouchStart(this, evt); handleTouchStart(this, evt);
}); });
this._target.addEventListener('touchmove', this._target.addEventListener("touchmove", (evt) => {
evt => {
handleTouchMove(this, evt); handleTouchMove(this, evt);
}); });
this._target.addEventListener('touchend', this._target.addEventListener("touchend", () => {
() => {
handleTouchEnd(this); handleTouchEnd(this);
}); });
} }

View file

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

View file

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

View file

@ -1,10 +1,10 @@
'use strict'; "use strict";
const events = require('../events.js'); const events = require("../events.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const CommentListControl = require('../controls/comment_list_control.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 { class CommentsPageView extends events.EventTarget {
constructor(ctx) { constructor(ctx) {
@ -16,12 +16,14 @@ class CommentsPageView extends events.EventTarget {
for (let post of ctx.response.results) { for (let post of ctx.response.results) {
const commentListControl = new CommentListControl( const commentListControl = new CommentListControl(
sourceNode.querySelector( sourceNode.querySelector(
`.comments-container[data-for="${post.id}"]`), `.comments-container[data-for="${post.id}"]`
),
post.comments, post.comments,
true); true
events.proxyEvent(commentListControl, this, 'submit'); );
events.proxyEvent(commentListControl, this, 'score'); events.proxyEvent(commentListControl, this, "submit");
events.proxyEvent(commentListControl, this, 'delete'); events.proxyEvent(commentListControl, this, "score");
events.proxyEvent(commentListControl, this, "delete");
} }
views.replaceContent(this._hostNode, sourceNode); 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 = () => { const template = () => {
return views.htmlToDom( return views.htmlToDom(
'<div class="wrapper"><div class="messages"></div></div>'); '<div class="wrapper"><div class="messages"></div></div>'
);
}; };
class EmptyView { class EmptyView {
constructor() { constructor() {
this._hostNode = document.getElementById('content-holder'); this._hostNode = document.getElementById("content-holder");
views.replaceContent(this._hostNode, template()); views.replaceContent(this._hostNode, template());
views.syncScrollPosition(); views.syncScrollPosition();
} }

View file

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

View file

@ -1,73 +1,81 @@
'use strict'; "use strict";
const api = require('../api.js'); const api = require("../api.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const template = views.getTemplate('help'); const template = views.getTemplate("help");
const sectionTemplates = { const sectionTemplates = {
'about': views.getTemplate('help-about'), about: views.getTemplate("help-about"),
'keyboard': views.getTemplate('help-keyboard'), keyboard: views.getTemplate("help-keyboard"),
'search': views.getTemplate('help-search'), search: views.getTemplate("help-search"),
'comments': views.getTemplate('help-comments'), comments: views.getTemplate("help-comments"),
'tos': views.getTemplate('help-tos'), tos: views.getTemplate("help-tos"),
}; };
const subsectionTemplates = { const subsectionTemplates = {
'search': { search: {
'default': views.getTemplate('help-search-general'), default: views.getTemplate("help-search-general"),
'posts': views.getTemplate('help-search-posts'), posts: views.getTemplate("help-search-posts"),
'users': views.getTemplate('help-search-users'), users: views.getTemplate("help-search-users"),
'tags': views.getTemplate('help-search-tags'), tags: views.getTemplate("help-search-tags"),
'pools': views.getTemplate('help-search-pools'), pools: views.getTemplate("help-search-pools"),
}, },
}; };
class HelpView { class HelpView {
constructor(section, subsection) { constructor(section, subsection) {
this._hostNode = document.getElementById('content-holder'); this._hostNode = document.getElementById("content-holder");
const sourceNode = template(); const sourceNode = template();
const ctx = { const ctx = {
name: api.getName(), name: api.getName(),
}; };
section = section || 'about'; section = section || "about";
if (section in sectionTemplates) { if (section in sectionTemplates) {
views.replaceContent( views.replaceContent(
sourceNode.querySelector('.content'), sourceNode.querySelector(".content"),
sectionTemplates[section](ctx)); sectionTemplates[section](ctx)
);
} }
subsection = subsection || 'default'; subsection = subsection || "default";
if (section in subsectionTemplates && if (
subsection in subsectionTemplates[section]) { section in subsectionTemplates &&
subsection in subsectionTemplates[section]
) {
views.replaceContent( views.replaceContent(
sourceNode.querySelector('.subcontent'), sourceNode.querySelector(".subcontent"),
subsectionTemplates[section][subsection](ctx)); subsectionTemplates[section][subsection](ctx)
);
} }
views.replaceContent(this._hostNode, sourceNode); views.replaceContent(this._hostNode, sourceNode);
for (let itemNode of for (let itemNode of sourceNode.querySelectorAll(
sourceNode.querySelectorAll('.primary [data-name]')) { ".primary [data-name]"
)) {
itemNode.classList.toggle( itemNode.classList.toggle(
'active', "active",
itemNode.getAttribute('data-name') === section); itemNode.getAttribute("data-name") === section
if (itemNode.getAttribute('data-name') === section) { );
if (itemNode.getAttribute("data-name") === section) {
itemNode.parentNode.scrollLeft = itemNode.parentNode.scrollLeft =
itemNode.getBoundingClientRect().left - itemNode.getBoundingClientRect().left -
itemNode.parentNode.getBoundingClientRect().left itemNode.parentNode.getBoundingClientRect().left;
} }
} }
for (let itemNode of for (let itemNode of sourceNode.querySelectorAll(
sourceNode.querySelectorAll('.secondary [data-name]')) { ".secondary [data-name]"
)) {
itemNode.classList.toggle( itemNode.classList.toggle(
'active', "active",
itemNode.getAttribute('data-name') === subsection); itemNode.getAttribute("data-name") === subsection
if (itemNode.getAttribute('data-name') === subsection) { );
if (itemNode.getAttribute("data-name") === subsection) {
itemNode.parentNode.scrollLeft = itemNode.parentNode.scrollLeft =
itemNode.getBoundingClientRect().left - 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 router = require("../router.js");
const uri = require('../util/uri.js'); const uri = require("../util/uri.js");
const misc = require('../util/misc.js'); const misc = require("../util/misc.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const PostContentControl = require('../controls/post_content_control.js'); const PostContentControl = require("../controls/post_content_control.js");
const PostNotesOverlayControl const PostNotesOverlayControl = require("../controls/post_notes_overlay_control.js");
= require('../controls/post_notes_overlay_control.js'); const TagAutoCompleteControl = require("../controls/tag_auto_complete_control.js");
const TagAutoCompleteControl =
require('../controls/tag_auto_complete_control.js');
const template = views.getTemplate('home'); const template = views.getTemplate("home");
const footerTemplate = views.getTemplate('home-footer'); const footerTemplate = views.getTemplate("home-footer");
const featuredPostTemplate = views.getTemplate('home-featured-post'); const featuredPostTemplate = views.getTemplate("home-featured-post");
class HomeView { class HomeView {
constructor(ctx) { constructor(ctx) {
this._hostNode = document.getElementById('content-holder'); this._hostNode = document.getElementById("content-holder");
this._ctx = ctx; this._ctx = ctx;
const sourceNode = template(ctx); const sourceNode = template(ctx);
@ -27,11 +25,16 @@ class HomeView {
this._autoCompleteControl = new TagAutoCompleteControl( this._autoCompleteControl = new TagAutoCompleteControl(
this._searchInputNode, this._searchInputNode,
{ {
confirm: tag => this._autoCompleteControl.replaceSelectedText( confirm: (tag) =>
misc.escapeSearchTerm(tag.names[0]), true), this._autoCompleteControl.replaceSelectedText(
}); misc.escapeSearchTerm(tag.names[0]),
this._formNode.addEventListener( true
'submit', e => this._evtFormSubmit(e)); ),
}
);
this._formNode.addEventListener("submit", (e) =>
this._evtFormSubmit(e)
);
} }
} }
@ -46,59 +49,67 @@ class HomeView {
setStats(stats) { setStats(stats) {
views.replaceContent( views.replaceContent(
this._footerContainerNode, this._footerContainerNode,
footerTemplate(Object.assign({}, stats, this._ctx))); footerTemplate(Object.assign({}, stats, this._ctx))
);
} }
setFeaturedPost(postInfo) { setFeaturedPost(postInfo) {
views.replaceContent( views.replaceContent(
this._postInfoContainerNode, featuredPostTemplate(postInfo)); this._postInfoContainerNode,
featuredPostTemplate(postInfo)
);
if (this._postContainerNode && postInfo.featuredPost) { if (this._postContainerNode && postInfo.featuredPost) {
this._postContentControl = new PostContentControl( this._postContentControl = new PostContentControl(
this._postContainerNode, this._postContainerNode,
postInfo.featuredPost, postInfo.featuredPost,
() => { () => {
return [ return [window.innerWidth * 0.8, window.innerHeight * 0.7];
window.innerWidth * 0.8,
window.innerHeight * 0.7,
];
}, },
'fit-both'); "fit-both"
);
this._postNotesOverlay = new PostNotesOverlayControl( this._postNotesOverlay = new PostNotesOverlayControl(
this._postContainerNode.querySelector('.post-overlay'), this._postContainerNode.querySelector(".post-overlay"),
postInfo.featuredPost); postInfo.featuredPost
);
if (postInfo.featuredPost.type === 'video' if (
|| postInfo.featuredPost.type === 'flash') { postInfo.featuredPost.type === "video" ||
postInfo.featuredPost.type === "flash"
) {
this._postContentControl.disableOverlay(); this._postContentControl.disableOverlay();
} }
} }
} }
get _footerContainerNode() { get _footerContainerNode() {
return this._hostNode.querySelector('.footer-container'); return this._hostNode.querySelector(".footer-container");
} }
get _postInfoContainerNode() { get _postInfoContainerNode() {
return this._hostNode.querySelector('.post-info-container'); return this._hostNode.querySelector(".post-info-container");
} }
get _postContainerNode() { get _postContainerNode() {
return this._hostNode.querySelector('.post-container'); return this._hostNode.querySelector(".post-container");
} }
get _formNode() { get _formNode() {
return this._hostNode.querySelector('form'); return this._hostNode.querySelector("form");
} }
get _searchInputNode() { get _searchInputNode() {
return this._formNode.querySelector('input[name=search-text]'); return this._formNode.querySelector("input[name=search-text]");
} }
_evtFormSubmit(e) { _evtFormSubmit(e) {
e.preventDefault(); e.preventDefault();
this._searchInputNode.blur(); 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 events = require("../events.js");
const api = require('../api.js'); const api = require("../api.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const template = views.getTemplate('login'); const template = views.getTemplate("login");
class LoginView extends events.EventTarget { class LoginView extends events.EventTarget {
constructor() { constructor() {
super(); 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(), userNamePattern: api.getUserNameRegex(),
passwordPattern: api.getPasswordRegex(), passwordPattern: api.getPasswordRegex(),
canSendMails: api.canSendMails(), canSendMails: api.canSendMails(),
})); })
);
views.syncScrollPosition(); views.syncScrollPosition();
views.decorateValidator(this._formNode); views.decorateValidator(this._formNode);
this._userNameInputNode.setAttribute('pattern', api.getUserNameRegex()); this._userNameInputNode.setAttribute(
this._passwordInputNode.setAttribute('pattern', api.getPasswordRegex()); "pattern",
this._formNode.addEventListener('submit', e => { api.getUserNameRegex()
);
this._passwordInputNode.setAttribute(
"pattern",
api.getPasswordRegex()
);
this._formNode.addEventListener("submit", (e) => {
e.preventDefault(); e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', { this.dispatchEvent(
new CustomEvent("submit", {
detail: { detail: {
name: this._userNameInputNode.value, name: this._userNameInputNode.value,
password: this._passwordInputNode.value, password: this._passwordInputNode.value,
remember: this._rememberInputNode.checked, remember: this._rememberInputNode.checked,
}, },
})); })
);
}); });
} }
get _formNode() { get _formNode() {
return this._hostNode.querySelector('form'); return this._hostNode.querySelector("form");
} }
get _userNameInputNode() { get _userNameInputNode() {
return this._formNode.querySelector('[name=name]'); return this._formNode.querySelector("[name=name]");
} }
get _passwordInputNode() { get _passwordInputNode() {
return this._formNode.querySelector('[name=password]'); return this._formNode.querySelector("[name=password]");
} }
get _rememberInputNode() { get _rememberInputNode() {
return this._formNode.querySelector('[name=remember-user]'); return this._formNode.querySelector("[name=remember-user]");
} }
disableForm() { disableForm() {

View file

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

View file

@ -1,14 +1,14 @@
'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 { class NotFoundView {
constructor(path) { constructor(path) {
this._hostNode = document.getElementById('content-holder'); this._hostNode = document.getElementById("content-holder");
const sourceNode = template({path: path}); const sourceNode = template({ path: path });
views.replaceContent(this._hostNode, sourceNode); views.replaceContent(this._hostNode, sourceNode);
views.syncScrollPosition(); views.syncScrollPosition();
} }

View file

@ -1,30 +1,35 @@
'use strict'; "use strict";
const events = require('../events.js'); const events = require("../events.js");
const api = require('../api.js'); const api = require("../api.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const template = views.getTemplate('password-reset'); const template = views.getTemplate("password-reset");
class PasswordResetView extends events.EventTarget { class PasswordResetView extends events.EventTarget {
constructor() { constructor() {
super(); 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(), canSendMails: api.canSendMails(),
contactEmail: api.getContactEmail(), contactEmail: api.getContactEmail(),
})); })
);
views.syncScrollPosition(); views.syncScrollPosition();
views.decorateValidator(this._formNode); views.decorateValidator(this._formNode);
this._formNode.addEventListener('submit', e => { this._formNode.addEventListener("submit", (e) => {
e.preventDefault(); e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', { this.dispatchEvent(
new CustomEvent("submit", {
detail: { detail: {
userNameOrEmail: this._userNameOrEmailFieldNode.value, userNameOrEmail: this._userNameOrEmailFieldNode.value,
}, },
})); })
);
}); });
} }
@ -49,11 +54,11 @@ class PasswordResetView extends events.EventTarget {
} }
get _formNode() { get _formNode() {
return this._hostNode.querySelector('form'); return this._hostNode.querySelector("form");
} }
get _userNameOrEmailFieldNode() { 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 events = require("../events.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const PoolCategory = require('../models/pool_category.js'); const PoolCategory = require("../models/pool_category.js");
const template = views.getTemplate('pool-categories'); const template = views.getTemplate("pool-categories");
const rowTemplate = views.getTemplate('pool-category-row'); const rowTemplate = views.getTemplate("pool-category-row");
class PoolCategoriesView extends events.EventTarget { class PoolCategoriesView extends events.EventTarget {
constructor(ctx) { constructor(ctx) {
super(); super();
this._ctx = ctx; this._ctx = ctx;
this._hostNode = document.getElementById('content-holder'); this._hostNode = document.getElementById("content-holder");
views.replaceContent(this._hostNode, template(ctx)); views.replaceContent(this._hostNode, template(ctx));
views.syncScrollPosition(); views.syncScrollPosition();
@ -31,18 +31,22 @@ class PoolCategoriesView extends events.EventTarget {
} }
if (this._addLinkNode) { if (this._addLinkNode) {
this._addLinkNode.addEventListener( this._addLinkNode.addEventListener("click", (e) =>
'click', e => this._evtAddButtonClick(e)); this._evtAddButtonClick(e)
);
} }
ctx.poolCategories.addEventListener( ctx.poolCategories.addEventListener("add", (e) =>
'add', e => this._evtPoolCategoryAdded(e)); this._evtPoolCategoryAdded(e)
);
ctx.poolCategories.addEventListener( ctx.poolCategories.addEventListener("remove", (e) =>
'remove', e => this._evtPoolCategoryDeleted(e)); this._evtPoolCategoryDeleted(e)
);
this._formNode.addEventListener( this._formNode.addEventListener("submit", (e) =>
'submit', e => this._evtSaveButtonClick(e, ctx)); this._evtSaveButtonClick(e, ctx)
);
} }
enableForm() { enableForm() {
@ -66,44 +70,48 @@ class PoolCategoriesView extends events.EventTarget {
} }
get _formNode() { get _formNode() {
return this._hostNode.querySelector('form'); return this._hostNode.querySelector("form");
} }
get _tableBodyNode() { get _tableBodyNode() {
return this._hostNode.querySelector('tbody'); return this._hostNode.querySelector("tbody");
} }
get _addLinkNode() { get _addLinkNode() {
return this._hostNode.querySelector('a.add'); return this._hostNode.querySelector("a.add");
} }
_addPoolCategoryRowNode(poolCategory) { _addPoolCategoryRowNode(poolCategory) {
const rowNode = rowTemplate( const rowNode = rowTemplate(
Object.assign( Object.assign({}, this._ctx, { poolCategory: poolCategory })
{}, this._ctx, {poolCategory: poolCategory})); );
const nameInput = rowNode.querySelector('.name input'); const nameInput = rowNode.querySelector(".name input");
if (nameInput) { if (nameInput) {
nameInput.addEventListener( nameInput.addEventListener("change", (e) =>
'change', e => this._evtNameChange(e, rowNode)); this._evtNameChange(e, rowNode)
);
} }
const colorInput = rowNode.querySelector('.color input'); const colorInput = rowNode.querySelector(".color input");
if (colorInput) { if (colorInput) {
colorInput.addEventListener( colorInput.addEventListener("change", (e) =>
'change', e => this._evtColorChange(e, rowNode)); this._evtColorChange(e, rowNode)
);
} }
const removeLinkNode = rowNode.querySelector('.remove a'); const removeLinkNode = rowNode.querySelector(".remove a");
if (removeLinkNode) { if (removeLinkNode) {
removeLinkNode.addEventListener( removeLinkNode.addEventListener("click", (e) =>
'click', e => this._evtDeleteButtonClick(e, rowNode)); this._evtDeleteButtonClick(e, rowNode)
);
} }
const defaultLinkNode = rowNode.querySelector('.set-default a'); const defaultLinkNode = rowNode.querySelector(".set-default a");
if (defaultLinkNode) { if (defaultLinkNode) {
defaultLinkNode.addEventListener( defaultLinkNode.addEventListener("click", (e) =>
'click', e => this._evtSetDefaultButtonClick(e, rowNode)); this._evtSetDefaultButtonClick(e, rowNode)
);
} }
this._tableBodyNode.appendChild(rowNode); this._tableBodyNode.appendChild(rowNode);
@ -141,7 +149,7 @@ class PoolCategoriesView extends events.EventTarget {
_evtDeleteButtonClick(e, rowNode, link) { _evtDeleteButtonClick(e, rowNode, link) {
e.preventDefault(); e.preventDefault();
if (e.target.classList.contains('inactive')) { if (e.target.classList.contains("inactive")) {
return; return;
} }
this._ctx.poolCategories.remove(rowNode._poolCategory); this._ctx.poolCategories.remove(rowNode._poolCategory);
@ -150,16 +158,16 @@ class PoolCategoriesView extends events.EventTarget {
_evtSetDefaultButtonClick(e, rowNode) { _evtSetDefaultButtonClick(e, rowNode) {
e.preventDefault(); e.preventDefault();
this._ctx.poolCategories.defaultCategory = rowNode._poolCategory; this._ctx.poolCategories.defaultCategory = rowNode._poolCategory;
const oldRowNode = rowNode.parentNode.querySelector('tr.default'); const oldRowNode = rowNode.parentNode.querySelector("tr.default");
if (oldRowNode) { if (oldRowNode) {
oldRowNode.classList.remove('default'); oldRowNode.classList.remove("default");
} }
rowNode.classList.add('default'); rowNode.classList.add("default");
} }
_evtSaveButtonClick(e, ctx) { _evtSaveButtonClick(e, ctx) {
e.preventDefault(); 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 events = require("../events.js");
const api = require('../api.js'); const api = require("../api.js");
const misc = require('../util/misc.js'); const misc = require("../util/misc.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const Pool = require('../models/pool.js') const Pool = require("../models/pool.js");
const template = views.getTemplate('pool-create'); const template = views.getTemplate("pool-create");
class PoolCreateView extends events.EventTarget { class PoolCreateView extends events.EventTarget {
constructor(ctx) { constructor(ctx) {
super(); super();
this._hostNode = document.getElementById('content-holder'); this._hostNode = document.getElementById("content-holder");
views.replaceContent(this._hostNode, template(ctx)); views.replaceContent(this._hostNode, template(ctx));
views.decorateValidator(this._formNode); views.decorateValidator(this._formNode);
if (this._namesFieldNode) { if (this._namesFieldNode) {
this._namesFieldNode.addEventListener( this._namesFieldNode.addEventListener("input", (e) =>
'input', e => this._evtNameInput(e)); this._evtNameInput(e)
);
} }
if (this._postsFieldNode) { if (this._postsFieldNode) {
this._postsFieldNode.addEventListener( this._postsFieldNode.addEventListener("input", (e) =>
'input', e => this._evtPostsInput(e)); this._evtPostsInput(e)
);
} }
for (let node of this._formNode.querySelectorAll( for (let node of this._formNode.querySelectorAll(
'input, select, textarea, posts')) { "input, select, textarea, posts"
node.addEventListener( )) {
'change', e => { node.addEventListener("change", (e) => {
this.dispatchEvent(new CustomEvent('change')); this.dispatchEvent(new CustomEvent("change"));
}); });
} }
this._formNode.addEventListener('submit', e => this._evtSubmit(e)); this._formNode.addEventListener("submit", (e) => this._evtSubmit(e));
} }
clearMessages() { clearMessages() {
@ -64,19 +66,21 @@ class PoolCreateView extends events.EventTarget {
if (!list.length) { if (!list.length) {
this._namesFieldNode.setCustomValidity( this._namesFieldNode.setCustomValidity(
'Pools must have at least one name.'); "Pools must have at least one name."
);
return; return;
} }
for (let item of list) { for (let item of list) {
if (!regex.test(item)) { if (!regex.test(item)) {
this._namesFieldNode.setCustomValidity( this._namesFieldNode.setCustomValidity(
`Pool name "${item}" contains invalid symbols.`); `Pool name "${item}" contains invalid symbols.`
);
return; return;
} }
} }
this._namesFieldNode.setCustomValidity(''); this._namesFieldNode.setCustomValidity("");
} }
_evtPostsInput(e) { _evtPostsInput(e) {
@ -86,46 +90,50 @@ class PoolCreateView extends events.EventTarget {
for (let item of list) { for (let item of list) {
if (!regex.test(item)) { if (!regex.test(item)) {
this._postsFieldNode.setCustomValidity( this._postsFieldNode.setCustomValidity(
`Pool ID "${item}" is not an integer.`); `Pool ID "${item}" is not an integer.`
);
return; return;
} }
} }
this._postsFieldNode.setCustomValidity(''); this._postsFieldNode.setCustomValidity("");
} }
_evtSubmit(e) { _evtSubmit(e) {
e.preventDefault(); e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', { this.dispatchEvent(
new CustomEvent("submit", {
detail: { detail: {
names: misc.splitByWhitespace(this._namesFieldNode.value), names: misc.splitByWhitespace(this._namesFieldNode.value),
category: this._categoryFieldNode.value, category: this._categoryFieldNode.value,
description: this._descriptionFieldNode.value, description: this._descriptionFieldNode.value,
posts: misc.splitByWhitespace(this._postsFieldNode.value) posts: misc
.map(i => parseInt(i)) .splitByWhitespace(this._postsFieldNode.value)
.map((i) => parseInt(i)),
}, },
})); })
);
} }
get _formNode() { get _formNode() {
return this._hostNode.querySelector('form'); return this._hostNode.querySelector("form");
} }
get _namesFieldNode() { get _namesFieldNode() {
return this._formNode.querySelector('.names input'); return this._formNode.querySelector(".names input");
} }
get _categoryFieldNode() { get _categoryFieldNode() {
return this._formNode.querySelector('.category select'); return this._formNode.querySelector(".category select");
} }
get _descriptionFieldNode() { get _descriptionFieldNode() {
return this._formNode.querySelector('.description textarea'); return this._formNode.querySelector(".description textarea");
} }
get _postsFieldNode() { 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 events = require("../events.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const template = views.getTemplate('pool-delete'); const template = views.getTemplate("pool-delete");
class PoolDeleteView extends events.EventTarget { class PoolDeleteView extends events.EventTarget {
constructor(ctx) { constructor(ctx) {
@ -13,7 +13,7 @@ class PoolDeleteView extends events.EventTarget {
this._pool = ctx.pool; this._pool = ctx.pool;
views.replaceContent(this._hostNode, template(ctx)); views.replaceContent(this._hostNode, template(ctx));
views.decorateValidator(this._formNode); views.decorateValidator(this._formNode);
this._formNode.addEventListener('submit', e => this._evtSubmit(e)); this._formNode.addEventListener("submit", (e) => this._evtSubmit(e));
} }
clearMessages() { clearMessages() {
@ -38,15 +38,17 @@ class PoolDeleteView extends events.EventTarget {
_evtSubmit(e) { _evtSubmit(e) {
e.preventDefault(); e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', { this.dispatchEvent(
new CustomEvent("submit", {
detail: { detail: {
pool: this._pool, pool: this._pool,
}, },
})); })
);
} }
get _formNode() { 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 events = require("../events.js");
const api = require('../api.js'); const api = require("../api.js");
const misc = require('../util/misc.js'); const misc = require("../util/misc.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const Post = require('../models/post.js'); const Post = require("../models/post.js");
const template = views.getTemplate('pool-edit'); const template = views.getTemplate("pool-edit");
class PoolEditView extends events.EventTarget { class PoolEditView extends events.EventTarget {
constructor(ctx) { constructor(ctx) {
@ -19,24 +19,26 @@ class PoolEditView extends events.EventTarget {
views.decorateValidator(this._formNode); views.decorateValidator(this._formNode);
if (this._namesFieldNode) { if (this._namesFieldNode) {
this._namesFieldNode.addEventListener( this._namesFieldNode.addEventListener("input", (e) =>
'input', e => this._evtNameInput(e)); this._evtNameInput(e)
);
} }
if (this._postsFieldNode) { if (this._postsFieldNode) {
this._postsFieldNode.addEventListener( this._postsFieldNode.addEventListener("input", (e) =>
'input', e => this._evtPostsInput(e)); this._evtPostsInput(e)
);
} }
for (let node of this._formNode.querySelectorAll( for (let node of this._formNode.querySelectorAll(
'input, select, textarea, posts')) { "input, select, textarea, posts"
node.addEventListener( )) {
'change', e => { node.addEventListener("change", (e) => {
this.dispatchEvent(new CustomEvent('change')); this.dispatchEvent(new CustomEvent("change"));
}); });
} }
this._formNode.addEventListener('submit', e => this._evtSubmit(e)); this._formNode.addEventListener("submit", (e) => this._evtSubmit(e));
} }
clearMessages() { clearMessages() {
@ -65,19 +67,21 @@ class PoolEditView extends events.EventTarget {
if (!list.length) { if (!list.length) {
this._namesFieldNode.setCustomValidity( this._namesFieldNode.setCustomValidity(
'Pools must have at least one name.'); "Pools must have at least one name."
);
return; return;
} }
for (let item of list) { for (let item of list) {
if (!regex.test(item)) { if (!regex.test(item)) {
this._namesFieldNode.setCustomValidity( this._namesFieldNode.setCustomValidity(
`Pool name "${item}" contains invalid symbols.`); `Pool name "${item}" contains invalid symbols.`
);
return; return;
} }
} }
this._namesFieldNode.setCustomValidity(''); this._namesFieldNode.setCustomValidity("");
} }
_evtPostsInput(e) { _evtPostsInput(e) {
@ -87,57 +91,60 @@ class PoolEditView extends events.EventTarget {
for (let item of list) { for (let item of list) {
if (!regex.test(item)) { if (!regex.test(item)) {
this._postsFieldNode.setCustomValidity( this._postsFieldNode.setCustomValidity(
`Pool ID "${item}" is not an integer.`); `Pool ID "${item}" is not an integer.`
);
return; return;
} }
} }
this._postsFieldNode.setCustomValidity(''); this._postsFieldNode.setCustomValidity("");
} }
_evtSubmit(e) { _evtSubmit(e) {
e.preventDefault(); e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', { this.dispatchEvent(
new CustomEvent("submit", {
detail: { detail: {
pool: this._pool, pool: this._pool,
names: this._namesFieldNode ? names: this._namesFieldNode
misc.splitByWhitespace(this._namesFieldNode.value) : ? misc.splitByWhitespace(this._namesFieldNode.value)
undefined, : undefined,
category: this._categoryFieldNode ? category: this._categoryFieldNode
this._categoryFieldNode.value : ? this._categoryFieldNode.value
undefined, : undefined,
description: this._descriptionFieldNode ? description: this._descriptionFieldNode
this._descriptionFieldNode.value : ? this._descriptionFieldNode.value
undefined, : undefined,
posts: this._postsFieldNode ? posts: this._postsFieldNode
misc.splitByWhitespace(this._postsFieldNode.value) : ? misc.splitByWhitespace(this._postsFieldNode.value)
undefined, : undefined,
}, },
})); })
);
} }
get _formNode() { get _formNode() {
return this._hostNode.querySelector('form'); return this._hostNode.querySelector("form");
} }
get _namesFieldNode() { get _namesFieldNode() {
return this._formNode.querySelector('.names input'); return this._formNode.querySelector(".names input");
} }
get _categoryFieldNode() { get _categoryFieldNode() {
return this._formNode.querySelector('.category select'); return this._formNode.querySelector(".category select");
} }
get _descriptionFieldNode() { get _descriptionFieldNode() {
return this._formNode.querySelector('.description textarea'); return this._formNode.querySelector(".description textarea");
} }
get _postsFieldNode() { 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 events = require("../events.js");
const api = require('../api.js'); const api = require("../api.js");
const views = require('../util/views.js'); const views = require("../util/views.js");
const PoolAutoCompleteControl = const PoolAutoCompleteControl = require("../controls/pool_auto_complete_control.js");
require('../controls/pool_auto_complete_control.js');
const template = views.getTemplate('pool-merge'); const template = views.getTemplate("pool-merge");
class PoolMergeView extends events.EventTarget { class PoolMergeView extends events.EventTarget {
constructor(ctx) { constructor(ctx) {
@ -23,15 +22,18 @@ class PoolMergeView extends events.EventTarget {
this._autoCompleteControl = new PoolAutoCompleteControl( this._autoCompleteControl = new PoolAutoCompleteControl(
this._targetPoolFieldNode, this._targetPoolFieldNode,
{ {
confirm: pool => { confirm: (pool) => {
this._targetPoolId = pool.id; this._targetPoolId = pool.id;
this._autoCompleteControl.replaceSelectedText( 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() { clearMessages() {
@ -56,24 +58,26 @@ class PoolMergeView extends events.EventTarget {
_evtSubmit(e) { _evtSubmit(e) {
e.preventDefault(); e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', { this.dispatchEvent(
new CustomEvent("submit", {
detail: { detail: {
pool: this._pool, pool: this._pool,
targetPoolId: this._targetPoolId targetPoolId: this._targetPoolId,
}, },
})); })
);
} }
get _formNode() { get _formNode() {
return this._hostNode.querySelector('form'); return this._hostNode.querySelector("form");
} }
get _targetPoolFieldNode() { get _targetPoolFieldNode() {
return this._formNode.querySelector('input[name=target-pool]'); return this._formNode.querySelector("input[name=target-pool]");
} }
get _addAliasCheckboxNode() { 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 { class PoolSummaryView {
constructor(ctx) { constructor(ctx) {

View file

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

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