szurubooru/client/js/controls/post_edit_sidebar_control.js

378 lines
12 KiB
JavaScript
Raw Normal View History

'use strict';
2016-07-03 13:46:49 +02:00
const api = require('../api.js');
const events = require('../events.js');
2016-07-03 13:46:49 +02:00
const misc = require('../util/misc.js');
const views = require('../util/views.js');
2016-07-03 13:46:49 +02:00
const TagInputControl = require('./tag_input_control.js');
const ExpanderControl = require('../controls/expander_control.js');
2016-07-27 22:27:33 +02:00
const FileDropperControl = require('../controls/file_dropper_control.js');
const template = views.getTemplate('post-edit-sidebar');
class PostEditSidebarControl extends events.EventTarget {
2016-07-22 13:27:52 +02:00
constructor(hostNode, post, postContentControl, postNotesOverlayControl) {
super();
this._hostNode = hostNode;
this._post = post;
this._postContentControl = postContentControl;
2016-07-22 13:27:52 +02:00
this._postNotesOverlayControl = postNotesOverlayControl;
2016-07-27 22:27:33 +02:00
this._newPostContent = null;
2016-07-22 13:27:52 +02:00
this._postNotesOverlayControl.switchToPassiveEdit();
views.replaceContent(this._hostNode, template({
post: this._post,
2016-07-03 13:46:49 +02:00
canEditPostSafety: api.hasPrivilege('posts:edit:safety'),
canEditPostSource: api.hasPrivilege('posts:edit:source'),
canEditPostTags: api.hasPrivilege('posts:edit:tags'),
canEditPostRelations: api.hasPrivilege('posts:edit:relations'),
canEditPostNotes: api.hasPrivilege('posts:edit:notes') &&
post.type !== 'video' &&
post.type !== 'flash',
canEditPostFlags: api.hasPrivilege('posts:edit:flags'),
canEditPostContent: api.hasPrivilege('posts:edit:content'),
2016-07-03 13:46:49 +02:00
canEditPostThumbnail: api.hasPrivilege('posts:edit:thumbnail'),
canCreateAnonymousPosts: api.hasPrivilege('posts:create:anonymous'),
canDeletePosts: api.hasPrivilege('posts:delete'),
canFeaturePosts: api.hasPrivilege('posts:feature'),
2016-10-22 10:03:38 +02:00
canMergePosts: api.hasPrivilege('posts:merge'),
}));
2016-07-03 13:46:49 +02:00
new ExpanderControl(
'post-info',
'Basic info',
this._hostNode.querySelectorAll('.safety, .relations, .flags'));
this._tagsExpander = new ExpanderControl(
'post-tags',
`Tags (${this._post.tags.length})`,
this._hostNode.querySelectorAll('.tags'));
this._notesExpander = new ExpanderControl(
'post-notes',
2016-07-22 13:27:52 +02:00
'Notes',
this._hostNode.querySelectorAll('.notes'));
new ExpanderControl(
'post-content',
'Content',
this._hostNode.querySelectorAll('.post-content, .post-thumbnail'));
2016-08-02 10:37:56 +02:00
new ExpanderControl(
'post-management',
2016-08-02 10:37:56 +02:00
'Management',
this._hostNode.querySelectorAll('.management'));
this._syncExpanderTitles();
2016-07-03 13:46:49 +02:00
if (this._formNode) {
this._formNode.addEventListener('submit', e => this._evtSubmit(e));
}
if (this._tagInputNode) {
this._tagControl = new TagInputControl(this._tagInputNode);
}
2016-07-27 22:27:33 +02:00
if (this._contentInputNode) {
this._contentFileDropper = new FileDropperControl(
this._contentInputNode, {allowUrls: true, lock: true});
this._contentFileDropper.addEventListener('fileadd', e => {
this._newPostContent = e.detail.files[0];
});
this._contentFileDropper.addEventListener('urladd', e => {
this._newPostContent = e.detail.urls[0];
});
2016-07-27 22:27:33 +02:00
}
if (this._thumbnailInputNode) {
this._thumbnailFileDropper = new FileDropperControl(
this._thumbnailInputNode, {lock: true});
this._thumbnailFileDropper.addEventListener('fileadd', e => {
this._newPostThumbnail = e.detail.files[0];
this._thumbnailRemovalLinkNode.style.display = 'block';
});
}
if (this._thumbnailRemovalLinkNode) {
this._thumbnailRemovalLinkNode.addEventListener(
'click', e => this._evtRemoveThumbnailClick(e));
this._thumbnailRemovalLinkNode.style.display =
this._post.hasCustomThumbnail ? 'block' : 'none';
}
2016-07-22 13:27:52 +02:00
if (this._addNoteLinkNode) {
this._addNoteLinkNode.addEventListener(
'click', e => this._evtAddNoteClick(e));
}
if (this._deleteNoteLinkNode) {
this._deleteNoteLinkNode.addEventListener(
'click', e => this._evtDeleteNoteClick(e));
}
2016-08-02 10:37:56 +02:00
if (this._featureLinkNode) {
this._featureLinkNode.addEventListener(
'click', e => this._evtFeatureClick(e));
}
2016-10-22 10:03:38 +02:00
if (this._mergeLinkNode) {
this._mergeLinkNode.addEventListener(
'click', e => this._evtMergeClick(e));
}
2016-08-02 11:05:40 +02:00
if (this._deleteLinkNode) {
this._deleteLinkNode.addEventListener(
'click', e => this._evtDeleteClick(e));
}
2016-07-22 13:27:52 +02:00
this._postNotesOverlayControl.addEventListener(
'blur', e => this._evtNoteBlur(e));
this._postNotesOverlayControl.addEventListener(
'focus', e => this._evtNoteFocus(e));
2016-07-27 22:27:33 +02:00
this._post.addEventListener(
'changeContent', e => this._evtPostContentChange(e));
this._post.addEventListener(
'changeThumbnail', e => this._evtPostThumbnailChange(e));
if (this._formNode) {
const inputNodes = this._formNode.querySelectorAll(
'input, textarea');
for (let node of inputNodes) {
node.addEventListener(
'change',
e => this.dispatchEvent(new CustomEvent('change')));
}
this._postNotesOverlayControl.addEventListener(
'change',
e => this.dispatchEvent(new CustomEvent('change')));
}
2016-07-22 13:27:52 +02:00
for (let eventType of ['add', 'remove']) {
this._post.notes.addEventListener(eventType, e => {
this._syncExpanderTitles();
});
}
this._tagControl.addEventListener('change', e => {
this._post.tags = this._tagControl.tags;
this._syncExpanderTitles();
});
2016-07-22 13:27:52 +02:00
if (this._noteTextareaNode) {
this._noteTextareaNode.addEventListener(
'change', e => this._evtNoteTextChangeRequest(e));
}
2016-07-27 22:27:33 +02:00
}
_syncExpanderTitles() {
this._notesExpander.title = `Notes (${this._post.notes.length})`;
this._tagsExpander.title = `Tags (${this._post.tags.length})`;
}
2016-07-27 22:27:33 +02:00
_evtPostContentChange(e) {
this._contentFileDropper.reset();
2016-07-03 13:46:49 +02:00
}
_evtPostThumbnailChange(e) {
this._thumbnailFileDropper.reset();
}
_evtRemoveThumbnailClick(e) {
e.preventDefault();
this._thumbnailFileDropper.reset();
this._newPostThumbnail = null;
this._thumbnailRemovalLinkNode.style.display = 'none';
}
2016-08-02 10:37:56 +02:00
_evtFeatureClick(e) {
e.preventDefault();
2016-08-02 10:37:56 +02:00
if (confirm('Are you sure you want to feature this post?')) {
this.dispatchEvent(new CustomEvent('feature', {
detail: {
post: this._post,
},
}));
}
}
2016-10-22 10:03:38 +02:00
_evtMergeClick(e) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('merge', {
detail: {
post: this._post,
},
}));
}
2016-08-02 11:05:40 +02:00
_evtDeleteClick(e) {
e.preventDefault();
2016-08-02 11:05:40 +02:00
if (confirm('Are you sure you want to delete this post?')) {
this.dispatchEvent(new CustomEvent('delete', {
detail: {
post: this._post,
},
}));
}
}
2016-07-22 13:27:52 +02:00
_evtNoteTextChangeRequest(e) {
if (this._editedNote) {
this._editedNote.text = this._noteTextareaNode.value;
}
}
_evtNoteFocus(e) {
this._editedNote = e.detail.note;
this._addNoteLinkNode.classList.remove('inactive');
this._deleteNoteLinkNode.classList.remove('inactive');
this._noteTextareaNode.removeAttribute('disabled');
this._noteTextareaNode.value = e.detail.note.text;
}
_evtNoteBlur(e) {
this._evtNoteTextChangeRequest(null);
this._addNoteLinkNode.classList.remove('inactive');
this._deleteNoteLinkNode.classList.add('inactive');
this._noteTextareaNode.blur();
this._noteTextareaNode.setAttribute('disabled', 'disabled');
this._noteTextareaNode.value = '';
}
_evtAddNoteClick(e) {
e.preventDefault();
2016-07-22 13:27:52 +02:00
if (e.target.classList.contains('inactive')) {
return;
}
this._addNoteLinkNode.classList.add('inactive');
this._postNotesOverlayControl.switchToDrawing();
}
_evtDeleteNoteClick(e) {
e.preventDefault();
2016-07-22 13:27:52 +02:00
if (e.target.classList.contains('inactive')) {
return;
}
this._post.notes.remove(this._editedNote);
this._postNotesOverlayControl.switchToPassiveEdit();
}
2016-07-03 13:46:49 +02:00
_evtSubmit(e) {
e.preventDefault();
this.dispatchEvent(new CustomEvent('submit', {
detail: {
post: this._post,
safety: this._safetyButtonNodes.length ?
2016-07-03 13:46:49 +02:00
Array.from(this._safetyButtonNodes)
.filter(node => node.checked)[0]
.value.toLowerCase() :
undefined,
flags: this._loopVideoInputNode ?
(this._loopVideoInputNode.checked ? ['loop'] : []) :
undefined,
tags: this._tagInputNode ?
misc.splitByWhitespace(this._tagInputNode.value) :
undefined,
relations: this._relationsInputNode ?
misc.splitByWhitespace(this._relationsInputNode.value)
.map(x => parseInt(x)) :
undefined,
2016-07-27 22:27:33 +02:00
content: this._newPostContent ?
this._newPostContent :
undefined,
thumbnail: this._newPostThumbnail !== undefined ?
this._newPostThumbnail :
undefined,
2016-07-03 13:46:49 +02:00
},
}));
}
get _formNode() {
return this._hostNode.querySelector('form');
}
get _submitButtonNode() {
return this._hostNode.querySelector('.submit');
}
get _safetyButtonNodes() {
return this._formNode.querySelectorAll('.safety input');
}
get _tagInputNode() {
return this._formNode.querySelector('.tags input');
}
get _loopVideoInputNode() {
return this._formNode.querySelector('.flags input[name=loop]');
}
2016-07-03 13:46:49 +02:00
get _relationsInputNode() {
return this._formNode.querySelector('.relations input');
}
2016-07-27 22:27:33 +02:00
get _contentInputNode() {
return this._formNode.querySelector('.post-content .dropper-container');
}
get _thumbnailInputNode() {
return this._formNode.querySelector(
'.post-thumbnail .dropper-container');
}
get _thumbnailRemovalLinkNode() {
return this._formNode.querySelector('.post-thumbnail a');
}
2016-08-02 10:37:56 +02:00
get _featureLinkNode() {
return this._formNode.querySelector('.management .feature');
}
2016-10-22 10:03:38 +02:00
get _mergeLinkNode() {
return this._formNode.querySelector('.management .merge');
}
2016-08-02 11:05:40 +02:00
get _deleteLinkNode() {
return this._formNode.querySelector('.management .delete');
}
2016-07-22 13:27:52 +02:00
get _addNoteLinkNode() {
return this._formNode.querySelector('.notes .add');
}
get _deleteNoteLinkNode() {
return this._formNode.querySelector('.notes .delete');
}
get _noteTextareaNode() {
return this._formNode.querySelector('.notes textarea');
}
enableForm() {
views.enableForm(this._formNode);
}
disableForm() {
views.disableForm(this._formNode);
}
clearMessages() {
views.clearMessages(this._hostNode);
}
showSuccess(message) {
views.showSuccess(this._hostNode, message);
}
showError(message) {
views.showError(this._hostNode, message);
}
};
module.exports = PostEditSidebarControl;