remove tags.json
This commit is contained in:
parent
253e28c1b5
commit
1c4c5c5f91
50 changed files with 448 additions and 633 deletions
37
API.md
37
API.md
|
@ -72,6 +72,7 @@
|
|||
- [Micro user](#micro-user)
|
||||
- [Tag category](#tag-category)
|
||||
- [Tag](#tag)
|
||||
- [Micro tag](#micro-tag)
|
||||
- [Post](#post)
|
||||
- [Micro post](#micro-post)
|
||||
- [Note](#note)
|
||||
|
@ -254,12 +255,6 @@ data.
|
|||
|
||||
Lists all tag categories. Doesn't use paging.
|
||||
|
||||
**Note**: independently, the server exports current tag category list
|
||||
snapshots to the data directory under `tags.json` name. Its purpose is to
|
||||
reduce the trips frontend needs to make when doing autocompletion, and ease
|
||||
caching. The data directory and its URL are controlled with `data_dir` and
|
||||
`data_url` variables in server's configuration.
|
||||
|
||||
## Creating tag category
|
||||
- **Request**
|
||||
|
||||
|
@ -419,12 +414,6 @@ data.
|
|||
|
||||
Searches for tags.
|
||||
|
||||
**Note**: independently, the server exports current tag list snapshots to
|
||||
the data directory under `tags.json` name. Its purpose is to reduce the
|
||||
trips frontend needs to make when doing autocompletion, and ease caching.
|
||||
The data directory and its URL are controlled with `data_dir` and
|
||||
`data_url` variables in server's configuration.
|
||||
|
||||
**Anonymous tokens**
|
||||
|
||||
Same as `name` token.
|
||||
|
@ -1766,16 +1755,23 @@ A single tag. Tags are used to let users search for posts.
|
|||
- `<names>`: a list of tag names (aliases). Tagging a post with any name will
|
||||
automatically assign the first name from this list.
|
||||
- `<category>`: the name of the category the given tag belongs to.
|
||||
- `<implications>`: a list of implied tag names. Implied tags are automatically
|
||||
appended by the web client on usage.
|
||||
- `<suggestions>`: a list of suggested tag names. Suggested tags are shown to
|
||||
the user by the web client on usage.
|
||||
- `<implications>`: a list of implied tags, serialized as [micro
|
||||
tag resource](#micro-tag). Implied tags are automatically appended by the web
|
||||
client on usage.
|
||||
- `<suggestions>`: a list of suggested tags, serialized as [micro
|
||||
tag resource](#micro-tag). Suggested tags are shown to the user by the web
|
||||
client on usage.
|
||||
- `<creation-time>`: time the tag was created, formatted as per RFC 3339.
|
||||
- `<last-edit-time>`: time the tag was edited, formatted as per RFC 3339.
|
||||
- `<usage-count>`: the number of posts the tag was used in.
|
||||
- `<description>`: the tag description (instructions how to use, history etc.)
|
||||
The client should render is as Markdown.
|
||||
|
||||
## Micro tag
|
||||
**Description**
|
||||
|
||||
A [tag resource](#tag) stripped down to `names`, `category` and `usages` fields.
|
||||
|
||||
## Post
|
||||
**Description**
|
||||
|
||||
|
@ -1814,12 +1810,12 @@ One file together with its metadata posted to the site.
|
|||
"lastFeatureTime": <last-feature-time>,
|
||||
"favoritedBy": <favorited-by>,
|
||||
"hasCustomThumbnail": <has-custom-thumbnail>,
|
||||
"mimeType": <mime-type>
|
||||
"comments": {
|
||||
"mimeType": <mime-type>,
|
||||
"comments": [
|
||||
<comment>,
|
||||
<comment>,
|
||||
<comment>
|
||||
}
|
||||
]
|
||||
}
|
||||
```
|
||||
|
||||
|
@ -1856,7 +1852,8 @@ One file together with its metadata posted to the site.
|
|||
- `<thumbnail-url>`: where the post thumbnail is located.
|
||||
- `<flags>`: various flags such as whether the post is looped, represented as
|
||||
array of plain strings.
|
||||
- `<tags>`: list of tag names the post is tagged with.
|
||||
- `<tags>`: list of tags the post is tagged with, serialized as [micro
|
||||
tag resource](#micro-tag).
|
||||
- `<relations>`: a list of related posts, serialized as [micro post
|
||||
resources](#micro-post). Links to related posts are shown
|
||||
to the user by the web client.
|
||||
|
|
|
@ -55,9 +55,7 @@
|
|||
|
||||
<% if (ctx.canEditPostTags) { %>
|
||||
<section class='tags'>
|
||||
<%= ctx.makeTextInput({
|
||||
value: ctx.post.tags.join(' '),
|
||||
}) %>
|
||||
<%= ctx.makeTextInput({}) %>
|
||||
</section>
|
||||
<% } %>
|
||||
|
||||
|
|
|
@ -69,20 +69,20 @@
|
|||
--><% for (let tag of ctx.post.tags) { %><!--
|
||||
--><li><!--
|
||||
--><% if (ctx.canViewTags) { %><!--
|
||||
--><a href='<%- ctx.formatClientLink('tag', tag) %>' class='<%= ctx.makeCssName(ctx.getTagCategory(tag), 'tag') %>'><!--
|
||||
--><a href='<%- ctx.formatClientLink('tag', tag.names[0]) %>' class='<%= ctx.makeCssName(tag.category, 'tag') %>'><!--
|
||||
--><i class='fa fa-tag'></i><!--
|
||||
--><% } %><!--
|
||||
--><% if (ctx.canViewTags) { %><!--
|
||||
--></a><!--
|
||||
--><% } %><!--
|
||||
--><% if (ctx.canListPosts) { %><!--
|
||||
--><a href='<%- ctx.formatClientLink('posts', {query: tag}) %>' class='<%= ctx.makeCssName(ctx.getTagCategory(tag), 'tag') %>'><!--
|
||||
--><a href='<%- ctx.formatClientLink('posts', {query: tag.names[0]}) %>' class='<%= ctx.makeCssName(tag.category, 'tag') %>'><!--
|
||||
--><% } %><!--
|
||||
--><%- tag %> <!--
|
||||
--><%- tag.names[0] %> <!--
|
||||
--><% if (ctx.canListPosts) { %><!--
|
||||
--></a><!--
|
||||
--><% } %><!--
|
||||
--><span class='tag-usages' data-pseudo-content='<%- ctx.getTagUsages(tag) %>'></span><!--
|
||||
--><span class='tag-usages' data-pseudo-content='<%- tag.postCount %>'></span><!--
|
||||
--></li><!--
|
||||
--><% } %><!--
|
||||
--></ul>
|
||||
|
|
|
@ -4,7 +4,7 @@
|
|||
<% for (let post of ctx.response.results) { %>
|
||||
<li data-post-id='<%= post.id %>'>
|
||||
<a class='thumbnail-wrapper <%= post.tags.length > 0 ? "tags" : "no-tags" %>'
|
||||
title='@<%- post.id %> (<%- post.type %>) Tags: <%- post.tags.map(tag => '#' + tag).join(' ') || 'none' %>'
|
||||
title='@<%- post.id %> (<%- post.type %>) Tags: <%- post.tags.map(tag => '#' + tag.names[0]).join(' ') || 'none' %>'
|
||||
href='<%= ctx.canViewPosts ? ctx.getPostUrl(post.id, ctx.parameters) : '' %>'>
|
||||
<%= ctx.makeThumbnail(post.thumbnailUrl) %>
|
||||
<span class='type' data-type='<%- post.type %>'>
|
||||
|
|
|
@ -22,18 +22,12 @@
|
|||
</li>
|
||||
<li class='implications'>
|
||||
<% if (ctx.canEditImplications) { %>
|
||||
<%= ctx.makeTextInput({
|
||||
text: 'Implications',
|
||||
value: ctx.tag.implications.join(' '),
|
||||
}) %>
|
||||
<%= ctx.makeTextInput({text: 'Implications'}) %>
|
||||
<% } %>
|
||||
</li>
|
||||
<li class='suggestions'>
|
||||
<% if (ctx.canEditSuggestions) { %>
|
||||
<%= ctx.makeTextInput({
|
||||
text: 'Suggestions',
|
||||
value: ctx.tag.suggestions.join(' '),
|
||||
}) %>
|
||||
<%= ctx.makeTextInput({text: 'Suggestions'}) %>
|
||||
<% } %>
|
||||
</li>
|
||||
<li class='description'>
|
||||
|
|
|
@ -9,7 +9,7 @@
|
|||
Aliases:<br/>
|
||||
<ul><!--
|
||||
--><% for (let name of ctx.tag.names.slice(1)) { %><!--
|
||||
--><li><%= ctx.makeTagLink(name) %></li><!--
|
||||
--><li><%= ctx.makeTagLink(name, false, false, ctx.tag) %></li><!--
|
||||
--><% } %><!--
|
||||
--></ul>
|
||||
</section>
|
||||
|
@ -18,7 +18,7 @@
|
|||
Implications:<br/>
|
||||
<ul><!--
|
||||
--><% for (let tag of ctx.tag.implications) { %><!--
|
||||
--><li><%= ctx.makeTagLink(tag) %></li><!--
|
||||
--><li><%= ctx.makeTagLink(tag.names[0], false, false, tag) %></li><!--
|
||||
--><% } %><!--
|
||||
--></ul>
|
||||
</section>
|
||||
|
@ -27,7 +27,7 @@
|
|||
Suggestions:<br/>
|
||||
<ul><!--
|
||||
--><% for (let tag of ctx.tag.suggestions) { %><!--
|
||||
--><li><%= ctx.makeTagLink(tag) %></li><!--
|
||||
--><li><%= ctx.makeTagLink(tag.names[0], false, false, tag) %></li><!--
|
||||
--><% } %><!--
|
||||
--></ul>
|
||||
</section>
|
||||
|
|
|
@ -44,15 +44,15 @@
|
|||
<td class='names'>
|
||||
<ul>
|
||||
<% for (let name of tag.names) { %>
|
||||
<li><%= ctx.makeTagLink(name) %></li>
|
||||
<li><%= ctx.makeTagLink(name, false, false, tag) %></li>
|
||||
<% } %>
|
||||
</ul>
|
||||
</td>
|
||||
<td class='implications'>
|
||||
<% if (tag.implications.length) { %>
|
||||
<ul>
|
||||
<% for (let name of tag.implications) { %>
|
||||
<li><%= ctx.makeTagLink(name) %></li>
|
||||
<% for (let relation of tag.implications) { %>
|
||||
<li><%= ctx.makeTagLink(relation.names[0], false, false, relation) %></li>
|
||||
<% } %>
|
||||
</ul>
|
||||
<% } else { %>
|
||||
|
@ -62,8 +62,8 @@
|
|||
<td class='suggestions'>
|
||||
<% if (tag.suggestions.length) { %>
|
||||
<ul>
|
||||
<% for (let name of tag.suggestions) { %>
|
||||
<li><%= ctx.makeTagLink(name) %></li>
|
||||
<% for (let relation of tag.suggestions) { %>
|
||||
<li><%= ctx.makeTagLink(relation.names[0], false, false, relation) %></li>
|
||||
<% } %>
|
||||
</ul>
|
||||
<% } else { %>
|
||||
|
|
|
@ -62,15 +62,16 @@ class PostListController {
|
|||
}
|
||||
|
||||
_evtTag(e) {
|
||||
for (let tag of this._bulkEditTags) {
|
||||
e.detail.post.addTag(tag);
|
||||
}
|
||||
e.detail.post.save().catch(error => window.alert(error.message));
|
||||
Promise.all(
|
||||
this._bulkEditTags.map(tag =>
|
||||
e.detail.post.tags.addByName(tag)))
|
||||
.then(() => { e.detail.post.save(); })
|
||||
.catch(error => window.alert(error.message));
|
||||
}
|
||||
|
||||
_evtUntag(e) {
|
||||
for (let tag of this._bulkEditTags) {
|
||||
e.detail.post.removeTag(tag);
|
||||
e.detail.post.tags.removeByName(tag);
|
||||
}
|
||||
e.detail.post.save().catch(error => window.alert(error.message));
|
||||
}
|
||||
|
|
|
@ -132,9 +132,6 @@ class PostMainController extends BasePostController {
|
|||
this._view.sidebarControl.disableForm();
|
||||
this._view.sidebarControl.clearMessages();
|
||||
const post = e.detail.post;
|
||||
if (e.detail.tags !== undefined) {
|
||||
post.tags = e.detail.tags;
|
||||
}
|
||||
if (e.detail.safety !== undefined) {
|
||||
post.safety = e.detail.safety;
|
||||
}
|
||||
|
|
|
@ -8,6 +8,7 @@ const misc = require('../util/misc.js');
|
|||
const progress = require('../util/progress.js');
|
||||
const topNavigation = require('../models/top_navigation.js');
|
||||
const Post = require('../models/post.js');
|
||||
const Tag = require('../models/tag.js');
|
||||
const PostUploadView = require('../views/post_upload_view.js');
|
||||
const EmptyView = require('../views/empty_view.js');
|
||||
|
||||
|
@ -144,7 +145,11 @@ class PostUploadController {
|
|||
let post = new Post();
|
||||
post.safety = uploadable.safety;
|
||||
post.flags = uploadable.flags;
|
||||
post.tags = uploadable.tags;
|
||||
for (let tagName of uploadable.tags) {
|
||||
const tag = new Tag();
|
||||
tag.names = [tagName];
|
||||
post.tags.add(tag);
|
||||
}
|
||||
post.relations = uploadable.relations;
|
||||
post.newContent = uploadable.url || uploadable.file;
|
||||
return post;
|
||||
|
|
|
@ -40,7 +40,7 @@ class TagCategoriesController {
|
|||
this._view.disableForm();
|
||||
this._tagCategories.save()
|
||||
.then(() => {
|
||||
tags.refreshExport();
|
||||
tags.refreshCategoryColorMap();
|
||||
this._view.enableForm();
|
||||
this._view.showSuccess('Changes saved.');
|
||||
}, error => {
|
||||
|
|
|
@ -4,8 +4,8 @@ const router = require('../router.js');
|
|||
const api = require('../api.js');
|
||||
const misc = require('../util/misc.js');
|
||||
const uri = require('../util/uri.js');
|
||||
const tags = require('../tags.js');
|
||||
const Tag = require('../models/tag.js');
|
||||
const TagCategoryList = require('../models/tag_category_list.js');
|
||||
const topNavigation = require('../models/top_navigation.js');
|
||||
const TagView = require('../views/tag_view.js');
|
||||
const EmptyView = require('../views/empty_view.js');
|
||||
|
@ -18,7 +18,12 @@ class TagController {
|
|||
return;
|
||||
}
|
||||
|
||||
Tag.get(ctx.parameters.name).then(tag => {
|
||||
Promise.all([
|
||||
TagCategoryList.get(),
|
||||
Tag.get(ctx.parameters.name),
|
||||
]).then(responses => {
|
||||
const [tagCategoriesResponse, tag] = responses;
|
||||
|
||||
topNavigation.activate('tags');
|
||||
topNavigation.setTitle('Tag #' + tag.names[0]);
|
||||
|
||||
|
@ -26,7 +31,7 @@ class TagController {
|
|||
tag.addEventListener('change', e => this._evtSaved(e, section));
|
||||
|
||||
const categories = {};
|
||||
for (let category of tags.getAllCategories()) {
|
||||
for (let category of tagCategoriesResponse.results) {
|
||||
categories[category.name] = category.name;
|
||||
}
|
||||
|
||||
|
@ -76,12 +81,6 @@ class TagController {
|
|||
if (e.detail.category !== undefined) {
|
||||
e.detail.tag.category = e.detail.category;
|
||||
}
|
||||
if (e.detail.implications !== undefined) {
|
||||
e.detail.tag.implications = e.detail.implications;
|
||||
}
|
||||
if (e.detail.suggestions !== undefined) {
|
||||
e.detail.tag.suggestions = e.detail.suggestions;
|
||||
}
|
||||
if (e.detail.description !== undefined) {
|
||||
e.detail.tag.description = e.detail.description;
|
||||
}
|
||||
|
|
|
@ -11,7 +11,12 @@ const TagsPageView = require('../views/tags_page_view.js');
|
|||
const EmptyView = require('../views/empty_view.js');
|
||||
|
||||
const fields = [
|
||||
'names', 'suggestions', 'implications', 'creationTime', 'usages'];
|
||||
'names',
|
||||
'suggestions',
|
||||
'implications',
|
||||
'creationTime',
|
||||
'usages',
|
||||
'category'];
|
||||
|
||||
class TagListController {
|
||||
constructor(ctx) {
|
||||
|
|
|
@ -28,10 +28,7 @@ class AutoCompleteControl {
|
|||
this._sourceInputNode = sourceInputNode;
|
||||
this._options = {};
|
||||
Object.assign(this._options, {
|
||||
transform: null,
|
||||
verticalShift: 2,
|
||||
source: null,
|
||||
addSpace: false,
|
||||
maxResults: 15,
|
||||
getTextToFind: () => {
|
||||
const value = sourceInputNode.value;
|
||||
|
@ -56,7 +53,7 @@ class AutoCompleteControl {
|
|||
this._isVisible = false;
|
||||
}
|
||||
|
||||
defaultConfirmStrategy(text) {
|
||||
replaceSelectedText(result, addSpace) {
|
||||
const start = _getSelectionStart(this._sourceInputNode);
|
||||
let prefix = '';
|
||||
let suffix = this._sourceInputNode.value.substring(start);
|
||||
|
@ -66,30 +63,25 @@ class AutoCompleteControl {
|
|||
prefix = this._sourceInputNode.value.substring(0, index + 1);
|
||||
middle = this._sourceInputNode.value.substring(index + 1);
|
||||
}
|
||||
this._sourceInputNode.value = prefix + text + ' ' + suffix.trimLeft();
|
||||
if (!this._options.addSpace) {
|
||||
this._sourceInputNode.value = (
|
||||
prefix + result.toString() + ' ' + suffix.trimLeft());
|
||||
if (!addSpace) {
|
||||
this._sourceInputNode.value = this._sourceInputNode.value.trim();
|
||||
}
|
||||
this._sourceInputNode.focus();
|
||||
}
|
||||
|
||||
_delete(text) {
|
||||
if (this._options.transform) {
|
||||
text = this._options.transform(text);
|
||||
}
|
||||
_delete(result) {
|
||||
if (this._options.delete) {
|
||||
this._options.delete(text);
|
||||
this._options.delete(result);
|
||||
}
|
||||
}
|
||||
|
||||
_confirm(text) {
|
||||
if (this._options.transform) {
|
||||
text = this._options.transform(text);
|
||||
}
|
||||
_confirm(result) {
|
||||
if (this._options.confirm) {
|
||||
this._options.confirm(text);
|
||||
this._options.confirm(result);
|
||||
} else {
|
||||
this.defaultConfirmStrategy(text);
|
||||
this.defaultConfirmStrategy(result);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -104,7 +96,6 @@ class AutoCompleteControl {
|
|||
this.hide();
|
||||
} else {
|
||||
this._updateResults(textToFind);
|
||||
this._refreshList();
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -209,15 +200,16 @@ class AutoCompleteControl {
|
|||
}
|
||||
|
||||
_updateResults(textToFind) {
|
||||
this._options.getMatches(textToFind).then(matches => {
|
||||
const oldResults = this._results.slice();
|
||||
this._results =
|
||||
this._options.getMatches(textToFind)
|
||||
.slice(0, this._options.maxResults);
|
||||
this._results = matches.slice(0, this._options.maxResults);
|
||||
const oldResultsHash = JSON.stringify(oldResults);
|
||||
const newResultsHash = JSON.stringify(this._results);
|
||||
if (oldResultsHash !== newResultsHash) {
|
||||
this._activeResult = -1;
|
||||
}
|
||||
this._refreshList();
|
||||
});
|
||||
}
|
||||
|
||||
_refreshList() {
|
||||
|
|
|
@ -72,7 +72,8 @@ class PostEditSidebarControl extends events.EventTarget {
|
|||
}
|
||||
|
||||
if (this._tagInputNode) {
|
||||
this._tagControl = new TagInputControl(this._tagInputNode);
|
||||
this._tagControl = new TagInputControl(
|
||||
this._tagInputNode, post.tags);
|
||||
}
|
||||
|
||||
if (this._contentInputNode) {
|
||||
|
@ -171,8 +172,9 @@ class PostEditSidebarControl extends events.EventTarget {
|
|||
});
|
||||
}
|
||||
|
||||
this._tagControl.addEventListener('change', e => {
|
||||
this._post.tags = this._tagControl.tags;
|
||||
this._tagControl.addEventListener(
|
||||
'change', e => {
|
||||
this.dispatchEvent(new CustomEvent('change'));
|
||||
this._syncExpanderTitles();
|
||||
});
|
||||
|
||||
|
|
|
@ -3,7 +3,6 @@
|
|||
const api = require('../api.js');
|
||||
const config = require('../config.js');
|
||||
const events = require('../events.js');
|
||||
const tags = require('../tags.js');
|
||||
const views = require('../util/views.js');
|
||||
|
||||
const template = views.getTemplate('post-readonly-sidebar');
|
||||
|
@ -22,8 +21,6 @@ class PostReadonlySidebarControl extends events.EventTarget {
|
|||
|
||||
views.replaceContent(this._hostNode, template({
|
||||
post: this._post,
|
||||
getTagCategory: this._getTagCategory,
|
||||
getTagUsages: this._getTagUsages,
|
||||
enableSafety: config.enableSafety,
|
||||
canListPosts: api.hasPrivilege('posts:list'),
|
||||
canEditPosts: api.hasPrivilege('posts:edit'),
|
||||
|
@ -161,16 +158,6 @@ class PostReadonlySidebarControl extends events.EventTarget {
|
|||
newNode.classList.add('active');
|
||||
}
|
||||
|
||||
_getTagUsages(name) {
|
||||
const tag = tags.getTagByName(name);
|
||||
return tag ? tag.usages : 0;
|
||||
}
|
||||
|
||||
_getTagCategory(name) {
|
||||
const tag = tags.getTagByName(name);
|
||||
return tag ? tag.category : 'unknown';
|
||||
}
|
||||
|
||||
_evtAddToFavoritesClick(e) {
|
||||
e.preventDefault();
|
||||
this.dispatchEvent(new CustomEvent('favorite', {
|
||||
|
|
|
@ -1,9 +1,33 @@
|
|||
'use strict';
|
||||
|
||||
const tags = require('../tags.js');
|
||||
const misc = require('../util/misc.js');
|
||||
const views = require('../util/views.js');
|
||||
const TagList = require('../models/tag_list.js');
|
||||
const AutoCompleteControl = require('./auto_complete_control.js');
|
||||
|
||||
function _escapeSearch(text) {
|
||||
return text.replace('\\', '\\\\').replace(':', '\\:');
|
||||
}
|
||||
|
||||
function _tagListToMatches(tags, options) {
|
||||
return [...tags].sort((tag1, tag2) => {
|
||||
return tag2.usages - tag1.usages;
|
||||
}).map(tag => {
|
||||
let cssName = misc.makeCssName(tag.category, 'tag');
|
||||
if (options.isTaggedWith(tag.names[0])) {
|
||||
cssName += ' disabled';
|
||||
}
|
||||
const caption = (
|
||||
'<span class="' + cssName + '">'
|
||||
+ misc.escapeHtml(tag.names[0] + ' (' + tag.postCount + ')')
|
||||
+ '</span>');
|
||||
return {
|
||||
caption: caption,
|
||||
value: tag,
|
||||
};
|
||||
});
|
||||
}
|
||||
|
||||
class TagAutoCompleteControl extends AutoCompleteControl {
|
||||
constructor(input, options) {
|
||||
const minLengthForPartialSearch = 3;
|
||||
|
@ -13,31 +37,20 @@ class TagAutoCompleteControl extends AutoCompleteControl {
|
|||
}, options);
|
||||
|
||||
options.getMatches = text => {
|
||||
const transform = x => x.toLowerCase();
|
||||
const match = text.length < minLengthForPartialSearch ?
|
||||
(a, b) => a.startsWith(b) :
|
||||
(a, b) => a.includes(b);
|
||||
text = transform(text);
|
||||
return Array.from(tags.getNameToTagMap().entries())
|
||||
.filter(kv => match(transform(kv[0]), text))
|
||||
.sort((kv1, kv2) => {
|
||||
return kv2[1].usages - kv1[1].usages;
|
||||
})
|
||||
.map(kv => {
|
||||
const origName = tags.getOriginalTagName(kv[0]);
|
||||
const category = kv[1].category;
|
||||
const usages = kv[1].usages;
|
||||
let cssName = misc.makeCssName(category, 'tag');
|
||||
if (options.isTaggedWith(kv[0])) {
|
||||
cssName += ' disabled';
|
||||
}
|
||||
return {
|
||||
caption: misc.unindent`
|
||||
<span class="${cssName}">
|
||||
${misc.escapeHtml(origName)} (${usages})
|
||||
</span>`,
|
||||
value: origName,
|
||||
};
|
||||
const term = misc.escapeSearchTerm(text);
|
||||
const query = (
|
||||
text.length < minLengthForPartialSearch
|
||||
? term + '*'
|
||||
: '*' + term + '*') + ' sort:usages';
|
||||
|
||||
return new Promise((resolve, reject) => {
|
||||
TagList.search(
|
||||
query, 0, this._options.maxResults,
|
||||
['names', 'category', 'usages'])
|
||||
.then(
|
||||
response => resolve(
|
||||
_tagListToMatches(response.results, this._options)),
|
||||
reject);
|
||||
});
|
||||
};
|
||||
|
||||
|
|
|
@ -4,6 +4,7 @@ const api = require('../api.js');
|
|||
const tags = require('../tags.js');
|
||||
const misc = require('../util/misc.js');
|
||||
const uri = require('../util/uri.js');
|
||||
const Tag = require('../models/tag.js');
|
||||
const settings = require('../models/settings.js');
|
||||
const events = require('../events.js');
|
||||
const views = require('../util/views.js');
|
||||
|
@ -80,11 +81,12 @@ class SuggestionList {
|
|||
}
|
||||
|
||||
class TagInputControl extends events.EventTarget {
|
||||
constructor(hostNode) {
|
||||
constructor(hostNode, tagList) {
|
||||
super();
|
||||
this.tags = [];
|
||||
this.tags = tagList;
|
||||
this._hostNode = hostNode;
|
||||
this._suggestions = new SuggestionList();
|
||||
this._tagToListItemNode = new Map();
|
||||
|
||||
// dom
|
||||
const editAreaNode = template();
|
||||
|
@ -98,16 +100,18 @@ class TagInputControl extends events.EventTarget {
|
|||
getTextToFind: () => {
|
||||
return this._tagInputNode.value;
|
||||
},
|
||||
confirm: text => {
|
||||
confirm: tag => {
|
||||
this._tagInputNode.value = '';
|
||||
this.addTag(text, SOURCE_USER_INPUT);
|
||||
// XXX: tags from autocomplete don't contain implications
|
||||
// so they need to be looked up in API
|
||||
this.addTagByName(tag.names[0], SOURCE_USER_INPUT);
|
||||
},
|
||||
delete: text => {
|
||||
delete: tag => {
|
||||
this._tagInputNode.value = '';
|
||||
this.deleteTag(text);
|
||||
this.deleteTag(tag);
|
||||
},
|
||||
verticalShift: -2,
|
||||
isTaggedWith: tagName => this.isTaggedWith(tagName),
|
||||
isTaggedWith: tagName => this.tags.isTaggedWith(tagName),
|
||||
});
|
||||
|
||||
// dom events
|
||||
|
@ -127,114 +131,81 @@ class TagInputControl extends events.EventTarget {
|
|||
this._hostNode.parentNode.insertBefore(
|
||||
this._editAreaNode, hostNode.nextSibling);
|
||||
|
||||
this.addEventListener('change', e => this._evtTagsChanged(e));
|
||||
this.addEventListener('add', e => this._evtTagAdded(e));
|
||||
this.addEventListener('remove', e => this._evtTagRemoved(e));
|
||||
|
||||
// add existing tags
|
||||
this.addMultipleTags(this._hostNode.value, SOURCE_INIT);
|
||||
for (let tag of [...this.tags]) {
|
||||
const listItemNode = this._createListItemNode(tag);
|
||||
this._tagListNode.appendChild(listItemNode);
|
||||
}
|
||||
}
|
||||
|
||||
isTaggedWith(tagName) {
|
||||
let actualTag = null;
|
||||
[tagName, actualTag] = this._transformTagName(tagName);
|
||||
return this.tags
|
||||
.map(t => t.toLowerCase())
|
||||
.includes(tagName.toLowerCase());
|
||||
}
|
||||
|
||||
addMultipleTags(text, source) {
|
||||
addTagByText(text, source) {
|
||||
for (let tagName of text.split(/\s+/).filter(word => word).reverse()) {
|
||||
this.addTag(tagName, source);
|
||||
this.addTagByName(tagName, source);
|
||||
}
|
||||
}
|
||||
|
||||
addTag(tagName, source) {
|
||||
tagName = tags.getOriginalTagName(tagName);
|
||||
|
||||
if (!tagName) {
|
||||
addTagByName(name, source) {
|
||||
name = name.trim();
|
||||
if (!name) {
|
||||
return;
|
||||
}
|
||||
|
||||
let actualTag = null;
|
||||
[tagName, actualTag] = this._transformTagName(tagName);
|
||||
if (!this.isTaggedWith(tagName)) {
|
||||
this.tags.push(tagName);
|
||||
}
|
||||
this.dispatchEvent(new CustomEvent('add', {
|
||||
detail: {
|
||||
tagName: tagName,
|
||||
source: source,
|
||||
},
|
||||
}));
|
||||
this.dispatchEvent(new CustomEvent('change'));
|
||||
|
||||
// XXX: perhaps we should aggregate suggestions from all implications
|
||||
// for call to the _suggestRelations
|
||||
if (source !== SOURCE_INIT && source !== SOURCE_CLIPBOARD) {
|
||||
for (let otherTagName of tags.getAllImplications(tagName)) {
|
||||
this.addTag(otherTagName, SOURCE_IMPLICATION);
|
||||
}
|
||||
}
|
||||
return Tag.get(name).then(tag => {
|
||||
return this.addTag(tag, source);
|
||||
}, () => {
|
||||
const tag = new Tag();
|
||||
tag.names = [name];
|
||||
tag.category = null;
|
||||
return this.addTag(tag, source);
|
||||
});
|
||||
}
|
||||
|
||||
deleteTag(tagName) {
|
||||
if (!tagName) {
|
||||
return;
|
||||
}
|
||||
let actualTag = null;
|
||||
[tagName, actualTag] = this._transformTagName(tagName);
|
||||
if (!this.isTaggedWith(tagName)) {
|
||||
return;
|
||||
}
|
||||
this._hideAutoComplete();
|
||||
this.tags = this.tags.filter(
|
||||
t => t.toLowerCase() != tagName.toLowerCase());
|
||||
this.dispatchEvent(new CustomEvent('remove', {
|
||||
detail: {
|
||||
tagName: tagName,
|
||||
},
|
||||
}));
|
||||
this.dispatchEvent(new CustomEvent('change'));
|
||||
}
|
||||
|
||||
_evtTagsChanged(e) {
|
||||
this._hostNode.value = this.tags.join(' ');
|
||||
this._hostNode.dispatchEvent(new CustomEvent('change'));
|
||||
}
|
||||
|
||||
_evtTagAdded(e) {
|
||||
const tagName = e.detail.tagName;
|
||||
const actualTag = tags.getTagByName(tagName);
|
||||
let listItemNode = this._getListItemNodeFromTagName(tagName);
|
||||
const alreadyAdded = !!listItemNode;
|
||||
if (alreadyAdded) {
|
||||
if (e.detail.source !== SOURCE_IMPLICATION) {
|
||||
addTag(tag, source) {
|
||||
if (source != SOURCE_INIT && this.tags.isTaggedWith(tag.names[0])) {
|
||||
const listItemNode = this._getListItemNode(tag);
|
||||
if (source !== SOURCE_IMPLICATION) {
|
||||
listItemNode.classList.add('duplicate');
|
||||
_fadeOutListItemNodeStatus(listItemNode);
|
||||
}
|
||||
} else {
|
||||
listItemNode = this._createListItemNode(tagName);
|
||||
if (!actualTag) {
|
||||
return Promise.resolve();
|
||||
}
|
||||
|
||||
return this.tags.addByName(tag.names[0], false).then(() => {
|
||||
const listItemNode = this._createListItemNode(tag);
|
||||
if (!tag.category) {
|
||||
listItemNode.classList.add('new');
|
||||
}
|
||||
if (e.detail.source === SOURCE_IMPLICATION) {
|
||||
if (source === SOURCE_IMPLICATION) {
|
||||
listItemNode.classList.add('implication');
|
||||
}
|
||||
this._tagListNode.prependChild(listItemNode);
|
||||
}
|
||||
_fadeOutListItemNodeStatus(listItemNode);
|
||||
|
||||
if ([SOURCE_USER_INPUT, SOURCE_SUGGESTION].includes(e.detail.source) &&
|
||||
actualTag) {
|
||||
this._loadSuggestions(actualTag);
|
||||
}
|
||||
return Promise.all(
|
||||
tag.implications.map(
|
||||
implication => this.addTagByName(
|
||||
implication.names[0], SOURCE_IMPLICATION)));
|
||||
}).then(() => {
|
||||
this.dispatchEvent(new CustomEvent('add', {
|
||||
detail: {tag: tag, source: source},
|
||||
}));
|
||||
this.dispatchEvent(new CustomEvent('change'));
|
||||
return Promise.resolve();
|
||||
});
|
||||
}
|
||||
|
||||
_evtTagRemoved(e) {
|
||||
const listItemNode = this._getListItemNodeFromTagName(e.detail.tagName);
|
||||
if (listItemNode) {
|
||||
listItemNode.parentNode.removeChild(listItemNode);
|
||||
deleteTag(tag) {
|
||||
if (!this.tags.isTaggedWith(tag.names[0])) {
|
||||
return;
|
||||
}
|
||||
this.tags.removeByName(tag.names[0]);
|
||||
this._hideAutoComplete();
|
||||
|
||||
this._deleteListItemNode(tag);
|
||||
|
||||
this.dispatchEvent(new CustomEvent('remove', {
|
||||
detail: {tag: tag},
|
||||
}));
|
||||
this.dispatchEvent(new CustomEvent('change'));
|
||||
}
|
||||
|
||||
_evtInputPaste(e) {
|
||||
|
@ -248,7 +219,7 @@ class TagInputControl extends events.EventTarget {
|
|||
return;
|
||||
}
|
||||
this._hideAutoComplete();
|
||||
this.addMultipleTags(pastedText, SOURCE_CLIPBOARD);
|
||||
this.addTagByText(pastedText, SOURCE_CLIPBOARD);
|
||||
this._tagInputNode.value = '';
|
||||
}
|
||||
|
||||
|
@ -259,7 +230,7 @@ class TagInputControl extends events.EventTarget {
|
|||
|
||||
_evtAddTagButtonClick(e) {
|
||||
e.preventDefault();
|
||||
this.addTag(this._tagInputNode.value, SOURCE_USER_INPUT);
|
||||
this.addTagByName(this._tagInputNode.value, SOURCE_USER_INPUT);
|
||||
this._tagInputNode.value = '';
|
||||
}
|
||||
|
||||
|
@ -272,36 +243,14 @@ class TagInputControl extends events.EventTarget {
|
|||
if (e.which == KEY_RETURN || e.which == KEY_SPACE) {
|
||||
e.preventDefault();
|
||||
this._hideAutoComplete();
|
||||
this.addMultipleTags(this._tagInputNode.value, SOURCE_USER_INPUT);
|
||||
this.addTagByText(this._tagInputNode.value, SOURCE_USER_INPUT);
|
||||
this._tagInputNode.value = '';
|
||||
}
|
||||
}
|
||||
|
||||
_transformTagName(tagName) {
|
||||
const actualTag = tags.getTagByName(tagName);
|
||||
if (actualTag) {
|
||||
tagName = actualTag.names[0];
|
||||
}
|
||||
return [tagName, actualTag];
|
||||
}
|
||||
|
||||
_getListItemNodeFromTagName(tagName) {
|
||||
let actualTag = null;
|
||||
[tagName, actualTag] = this._transformTagName(tagName);
|
||||
for (let listItemNode of this._tagListNode.querySelectorAll('li')) {
|
||||
if (listItemNode.getAttribute('data-tag').toLowerCase() ===
|
||||
tagName.toLowerCase()) {
|
||||
return listItemNode;
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
_createListItemNode(tagName) {
|
||||
let actualTag = null;
|
||||
[tagName, actualTag] = this._transformTagName(tagName);
|
||||
const className = actualTag ?
|
||||
misc.makeCssName(actualTag.category, 'tag') :
|
||||
_createListItemNode(tag) {
|
||||
const className = tag.category ?
|
||||
misc.makeCssName(tag.category, 'tag') :
|
||||
null;
|
||||
|
||||
const tagLinkNode = document.createElement('a');
|
||||
|
@ -309,7 +258,8 @@ class TagInputControl extends events.EventTarget {
|
|||
tagLinkNode.classList.add(className);
|
||||
}
|
||||
tagLinkNode.setAttribute(
|
||||
'href', uri.formatClientLink('tag', tagName));
|
||||
'href', uri.formatClientLink('tag', tag.names[0]));
|
||||
|
||||
const tagIconNode = document.createElement('i');
|
||||
tagIconNode.classList.add('fa');
|
||||
tagIconNode.classList.add('fa-tag');
|
||||
|
@ -320,13 +270,13 @@ class TagInputControl extends events.EventTarget {
|
|||
searchLinkNode.classList.add(className);
|
||||
}
|
||||
searchLinkNode.setAttribute(
|
||||
'href', uri.formatClientLink('posts', {query: tagName}));
|
||||
searchLinkNode.textContent = tagName + ' ';
|
||||
'href', uri.formatClientLink('posts', {query: tag.names[0]}));
|
||||
searchLinkNode.textContent = tag.names[0] + ' ';
|
||||
searchLinkNode.addEventListener('click', e => {
|
||||
e.preventDefault();
|
||||
if (actualTag) {
|
||||
this._suggestions.clear();
|
||||
this._loadSuggestions(actualTag);
|
||||
if (tag.postCount > 0) {
|
||||
this._loadSuggestions(tag);
|
||||
this._removeSuggestionsPopupOpacity();
|
||||
} else {
|
||||
this._closeSuggestionsPopup();
|
||||
|
@ -335,8 +285,7 @@ class TagInputControl extends events.EventTarget {
|
|||
|
||||
const usagesNode = document.createElement('span');
|
||||
usagesNode.classList.add('tag-usages');
|
||||
usagesNode.setAttribute(
|
||||
'data-pseudo-content', actualTag ? actualTag.usages : 0);
|
||||
usagesNode.setAttribute('data-pseudo-content', tag.postCount);
|
||||
|
||||
const removalLinkNode = document.createElement('a');
|
||||
removalLinkNode.classList.add('remove-tag');
|
||||
|
@ -344,18 +293,34 @@ class TagInputControl extends events.EventTarget {
|
|||
removalLinkNode.setAttribute('data-pseudo-content', '×');
|
||||
removalLinkNode.addEventListener('click', e => {
|
||||
e.preventDefault();
|
||||
this.deleteTag(tagName);
|
||||
this.deleteTag(tag);
|
||||
});
|
||||
|
||||
const listItemNode = document.createElement('li');
|
||||
listItemNode.setAttribute('data-tag', tagName);
|
||||
listItemNode.appendChild(removalLinkNode);
|
||||
listItemNode.appendChild(tagLinkNode);
|
||||
listItemNode.appendChild(searchLinkNode);
|
||||
listItemNode.appendChild(usagesNode);
|
||||
for (let name of tag.names) {
|
||||
this._tagToListItemNode.set(name, listItemNode);
|
||||
}
|
||||
return listItemNode;
|
||||
}
|
||||
|
||||
_deleteListItemNode(tag) {
|
||||
const listItemNode = this._getListItemNode(tag);
|
||||
if (listItemNode) {
|
||||
listItemNode.parentNode.removeChild(listItemNode);
|
||||
}
|
||||
for (let name of tag.names) {
|
||||
this._tagToListItemNode.delete(name);
|
||||
}
|
||||
}
|
||||
|
||||
_getListItemNode(tag) {
|
||||
return this._tagToListItemNode.get(tag.names[0]);
|
||||
}
|
||||
|
||||
_loadSuggestions(tag) {
|
||||
const browsingSettings = settings.get();
|
||||
if (!browsingSettings.tagSuggestions) {
|
||||
|
@ -399,23 +364,22 @@ class TagInputControl extends events.EventTarget {
|
|||
for (let tuple of this._suggestions.getAll()) {
|
||||
const tagName = tuple.tagName;
|
||||
const weight = tuple.weight;
|
||||
if (this.isTaggedWith(tagName)) {
|
||||
if (this.tags.isTaggedWith(tagName)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
const actualTag = tags.getTagByName(tagName);
|
||||
const addLinkNode = document.createElement('a');
|
||||
addLinkNode.textContent = tagName;
|
||||
addLinkNode.classList.add('add-tag');
|
||||
addLinkNode.setAttribute('href', '');
|
||||
if (actualTag) {
|
||||
Tag.get(tagName).then(tag => {
|
||||
addLinkNode.classList.add(
|
||||
misc.makeCssName(actualTag.category, 'tag'));
|
||||
}
|
||||
misc.makeCssName(tag.category, 'tag'));
|
||||
});
|
||||
addLinkNode.addEventListener('click', e => {
|
||||
e.preventDefault();
|
||||
listNode.removeChild(listItemNode);
|
||||
this.addTag(tagName, SOURCE_SUGGESTION);
|
||||
this.addTagByName(tagName, SOURCE_SUGGESTION);
|
||||
});
|
||||
|
||||
const weightNode = document.createElement('span');
|
||||
|
|
|
@ -55,7 +55,7 @@ for (let controller of controllers) {
|
|||
|
||||
const tags = require('./tags.js');
|
||||
const api = require('./api.js');
|
||||
tags.refreshExport(); // we don't care about errors
|
||||
tags.refreshCategoryColorMap(); // we don't care about errors
|
||||
api.loginFromCookies().then(() => {
|
||||
router.start();
|
||||
}, error => {
|
||||
|
|
|
@ -27,6 +27,13 @@ class AbstractList extends events.EventTarget {
|
|||
return ret;
|
||||
}
|
||||
|
||||
sync(plainList) {
|
||||
this.clear();
|
||||
for (let item of (plainList || [])) {
|
||||
this.add(this.constructor._itemClass.fromResponse(item));
|
||||
}
|
||||
}
|
||||
|
||||
add(item) {
|
||||
if (item.addEventListener) {
|
||||
item.addEventListener('delete', e => {
|
||||
|
@ -75,6 +82,10 @@ class AbstractList extends events.EventTarget {
|
|||
return this._list[index];
|
||||
}
|
||||
|
||||
map(...args) {
|
||||
return this._list.map(...args);
|
||||
}
|
||||
|
||||
[Symbol.iterator]() {
|
||||
return this._list[Symbol.iterator]();
|
||||
}
|
||||
|
|
|
@ -4,23 +4,18 @@ const api = require('../api.js');
|
|||
const uri = require('../util/uri.js');
|
||||
const tags = require('../tags.js');
|
||||
const events = require('../events.js');
|
||||
const TagList = require('./tag_list.js');
|
||||
const NoteList = require('./note_list.js');
|
||||
const CommentList = require('./comment_list.js');
|
||||
const misc = require('../util/misc.js');
|
||||
|
||||
function _syncObservableCollection(target, plainList) {
|
||||
target.clear();
|
||||
for (let item of (plainList || [])) {
|
||||
target.add(target.constructor._itemClass.fromResponse(item));
|
||||
}
|
||||
}
|
||||
|
||||
class Post extends events.EventTarget {
|
||||
constructor() {
|
||||
super();
|
||||
this._orig = {};
|
||||
|
||||
for (let obj of [this, this._orig]) {
|
||||
obj._tags = new TagList();
|
||||
obj._notes = new NoteList();
|
||||
obj._comments = new CommentList();
|
||||
}
|
||||
|
@ -56,7 +51,6 @@ class Post extends events.EventTarget {
|
|||
get hasCustomThumbnail() { return this._hasCustomThumbnail; }
|
||||
|
||||
set flags(value) { this._flags = value; }
|
||||
set tags(value) { this._tags = value; }
|
||||
set safety(value) { this._safety = value; }
|
||||
set relations(value) { this._relations = value; }
|
||||
set newContent(value) { this._newContent = value; }
|
||||
|
@ -94,29 +88,6 @@ class Post extends events.EventTarget {
|
|||
});
|
||||
}
|
||||
|
||||
isTaggedWith(tagName) {
|
||||
return this._tags
|
||||
.map(s => s.toLowerCase())
|
||||
.includes(tagName.toLowerCase());
|
||||
}
|
||||
|
||||
addTag(tagName, addImplications) {
|
||||
if (this.isTaggedWith(tagName)) {
|
||||
return;
|
||||
}
|
||||
this._tags.push(tagName);
|
||||
if (addImplications !== false) {
|
||||
for (let otherTag of tags.getAllImplications(tagName)) {
|
||||
this.addTag(otherTag, addImplications);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
removeTag(tagName) {
|
||||
this._tags = this._tags.filter(
|
||||
s => s.toLowerCase() != tagName.toLowerCase());
|
||||
}
|
||||
|
||||
save(anonymous) {
|
||||
const files = {};
|
||||
const detail = {version: this._version};
|
||||
|
@ -132,15 +103,14 @@ class Post extends events.EventTarget {
|
|||
detail.flags = this._flags;
|
||||
}
|
||||
if (misc.arraysDiffer(this._tags, this._orig._tags)) {
|
||||
detail.tags = this._tags;
|
||||
detail.tags = this._tags.map(tag => tag.names[0]);
|
||||
}
|
||||
if (misc.arraysDiffer(this._relations, this._orig._relations)) {
|
||||
detail.relations = this._relations;
|
||||
}
|
||||
if (misc.arraysDiffer(this._notes, this._orig._notes)) {
|
||||
detail.notes = [...this._notes].map(note => ({
|
||||
polygon: [...note.polygon].map(
|
||||
point => [point.x, point.y]),
|
||||
detail.notes = this._notes.map(note => ({
|
||||
polygon: note.polygon.map(point => [point.x, point.y]),
|
||||
text: note.text,
|
||||
}));
|
||||
}
|
||||
|
@ -310,7 +280,6 @@ class Post extends events.EventTarget {
|
|||
_fileSize: response.fileSize,
|
||||
|
||||
_flags: [...response.flags || []],
|
||||
_tags: [...response.tags || []],
|
||||
_relations: [...response.relations || []],
|
||||
|
||||
_score: response.score,
|
||||
|
@ -322,8 +291,9 @@ class Post extends events.EventTarget {
|
|||
});
|
||||
|
||||
for (let obj of [this, this._orig]) {
|
||||
_syncObservableCollection(obj._notes, response.notes);
|
||||
_syncObservableCollection(obj._comments, response.comments);
|
||||
obj._tags.sync(response.tags);
|
||||
obj._notes.sync(response.notes);
|
||||
obj._comments.sync(response.comments);
|
||||
}
|
||||
|
||||
Object.assign(this, map());
|
||||
|
|
|
@ -7,8 +7,16 @@ const misc = require('../util/misc.js');
|
|||
|
||||
class Tag extends events.EventTarget {
|
||||
constructor() {
|
||||
const TagList = require('./tag_list.js');
|
||||
|
||||
super();
|
||||
this._orig = {};
|
||||
|
||||
for (let obj of [this, this._orig]) {
|
||||
obj._suggestions = new TagList();
|
||||
obj._implications = new TagList();
|
||||
}
|
||||
|
||||
this._updateFromResponse({});
|
||||
}
|
||||
|
||||
|
@ -24,8 +32,6 @@ class Tag extends events.EventTarget {
|
|||
set names(value) { this._names = value; }
|
||||
set category(value) { this._category = value; }
|
||||
set description(value) { this._description = value; }
|
||||
set implications(value) { this._implications = value; }
|
||||
set suggestions(value) { this._suggestions = value; }
|
||||
|
||||
static fromResponse(response) {
|
||||
const ret = new Tag();
|
||||
|
@ -54,10 +60,12 @@ class Tag extends events.EventTarget {
|
|||
detail.description = this._description;
|
||||
}
|
||||
if (misc.arraysDiffer(this._implications, this._orig._implications)) {
|
||||
detail.implications = this._implications;
|
||||
detail.implications = this._implications.map(
|
||||
relation => relation.names[0]);
|
||||
}
|
||||
if (misc.arraysDiffer(this._suggestions, this._orig._suggestions)) {
|
||||
detail.suggestions = this._suggestions;
|
||||
detail.suggestions = this._suggestions.map(
|
||||
relation => relation.names[0]);
|
||||
}
|
||||
|
||||
let promise = this._origName ?
|
||||
|
@ -124,13 +132,16 @@ class Tag extends events.EventTarget {
|
|||
_names: response.names,
|
||||
_category: response.category,
|
||||
_description: response.description,
|
||||
_implications: response.implications,
|
||||
_suggestions: response.suggestions,
|
||||
_creationTime: response.creationTime,
|
||||
_lastEditTime: response.lastEditTime,
|
||||
_postCount: response.usages,
|
||||
_postCount: response.usages || 0,
|
||||
};
|
||||
|
||||
for (let obj of [this, this._orig]) {
|
||||
obj._suggestions.sync(response.suggestions);
|
||||
obj._implications.sync(response.implications);
|
||||
}
|
||||
|
||||
Object.assign(this, map);
|
||||
Object.assign(this._orig, map);
|
||||
}
|
||||
|
|
|
@ -22,6 +22,48 @@ class TagList extends AbstractList {
|
|||
{results: TagList.fromResponse(response.results)}));
|
||||
});
|
||||
}
|
||||
|
||||
isTaggedWith(testName) {
|
||||
for (let tag of this._list) {
|
||||
for (let tagName of tag.names) {
|
||||
if (tagName.toLowerCase() === testName.toLowerCase()) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
addByName(tagName, addImplications) {
|
||||
if (this.isTaggedWith(tagName)) {
|
||||
return Promise.resolve();
|
||||
}
|
||||
|
||||
const tag = new Tag();
|
||||
tag.names = [tagName];
|
||||
|
||||
this.add(tag);
|
||||
|
||||
if (addImplications !== false) {
|
||||
return Tag.get(tagName).then(actualTag => {
|
||||
return Promise.all(
|
||||
actualTag.implications.map(
|
||||
relation => this.addByName(relation.names[0], true)));
|
||||
});
|
||||
}
|
||||
|
||||
return Promise.resolve();
|
||||
}
|
||||
|
||||
removeByName(testName) {
|
||||
for (let tag of this._list) {
|
||||
for (let tagName of tag.names) {
|
||||
if (tagName.toLowerCase() === testName.toLowerCase()) {
|
||||
this.remove(tag);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
TagList._itemClass = Tag;
|
||||
|
|
|
@ -1,92 +1,23 @@
|
|||
'use strict';
|
||||
|
||||
const misc = require('./util/misc.js');
|
||||
const request = require('superagent');
|
||||
const TagCategoryList = require('./models/tag_category_list.js');
|
||||
|
||||
let _tags = new Map();
|
||||
let _categories = new Map();
|
||||
let _stylesheet = null;
|
||||
|
||||
function getTagByName(name) {
|
||||
return _tags.get(name.toLowerCase());
|
||||
}
|
||||
|
||||
function getCategoryByName(name) {
|
||||
return _categories.get(name.toLowerCase());
|
||||
}
|
||||
|
||||
function getNameToTagMap() {
|
||||
return _tags;
|
||||
}
|
||||
|
||||
function getAllTags() {
|
||||
return _tags.values();
|
||||
}
|
||||
|
||||
function getAllCategories() {
|
||||
return _categories.values();
|
||||
}
|
||||
|
||||
function getOriginalTagName(name) {
|
||||
const actualTag = getTagByName(name);
|
||||
if (actualTag) {
|
||||
for (let originalName of actualTag.names) {
|
||||
if (originalName.toLowerCase() == name.toLowerCase()) {
|
||||
return originalName;
|
||||
}
|
||||
}
|
||||
}
|
||||
return name;
|
||||
}
|
||||
|
||||
function _tagsToMap(tags) {
|
||||
let map = new Map();
|
||||
for (let tag of tags) {
|
||||
for (let name of tag.names) {
|
||||
map.set(name.toLowerCase(), tag);
|
||||
}
|
||||
}
|
||||
return map;
|
||||
}
|
||||
|
||||
function _tagCategoriesToMap(categories) {
|
||||
let map = new Map();
|
||||
for (let category of categories) {
|
||||
map.set(category.name.toLowerCase(), category);
|
||||
}
|
||||
return map;
|
||||
}
|
||||
|
||||
function _refreshStylesheet() {
|
||||
function refreshCategoryColorMap() {
|
||||
return TagCategoryList.get().then(response => {
|
||||
if (_stylesheet) {
|
||||
document.head.removeChild(_stylesheet);
|
||||
}
|
||||
_stylesheet = document.createElement('style');
|
||||
document.head.appendChild(_stylesheet);
|
||||
for (let category of getAllCategories()) {
|
||||
for (let category of response.results) {
|
||||
const ruleName = misc.makeCssName(category.name, 'tag');
|
||||
_stylesheet.sheet.insertRule(
|
||||
`.${ruleName} { color: ${category.color} }`,
|
||||
_stylesheet.sheet.cssRules.length);
|
||||
}
|
||||
}
|
||||
|
||||
function refreshExport() {
|
||||
return new Promise((resolve, reject) => {
|
||||
request.get('/data/tags.json').end((error, response) => {
|
||||
if (error) {
|
||||
_tags = new Map();
|
||||
_categories = new Map();
|
||||
reject(error);
|
||||
return;
|
||||
}
|
||||
_tags = _tagsToMap(
|
||||
response.body ? response.body.tags : []);
|
||||
_categories = _tagCategoriesToMap(
|
||||
response.body ? response.body.categories : []);
|
||||
_refreshStylesheet();
|
||||
resolve();
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -107,19 +38,7 @@ function getAllImplications(tagName) {
|
|||
return Array.from(implications);
|
||||
}
|
||||
|
||||
function getSuggestions(tagName) {
|
||||
const actualTag = getTagByName(tagName) || {};
|
||||
return actualTag.suggestions || [];
|
||||
}
|
||||
|
||||
module.exports = {
|
||||
getAllCategories: getAllCategories,
|
||||
getAllTags: getAllTags,
|
||||
getTagByName: getTagByName,
|
||||
getCategoryByName: getCategoryByName,
|
||||
getNameToTagMap: getNameToTagMap,
|
||||
getOriginalTagName: getOriginalTagName,
|
||||
refreshExport: refreshExport,
|
||||
refreshCategoryColorMap: refreshCategoryColorMap,
|
||||
getAllImplications: getAllImplications,
|
||||
getSuggestions: getSuggestions,
|
||||
};
|
||||
|
|
|
@ -3,7 +3,6 @@
|
|||
require('../util/polyfill.js');
|
||||
const api = require('../api.js');
|
||||
const templates = require('../templates.js');
|
||||
const tags = require('../tags.js');
|
||||
const domParser = new DOMParser();
|
||||
const misc = require('./misc.js');
|
||||
const uri = require('./uri.js');
|
||||
|
@ -194,13 +193,15 @@ function makePostLink(id, includeHash) {
|
|||
misc.escapeHtml(text);
|
||||
}
|
||||
|
||||
function makeTagLink(name, includeHash) {
|
||||
const tag = tags.getTagByName(name);
|
||||
function makeTagLink(name, includeHash, includeCount, tag) {
|
||||
const category = tag ? tag.category : 'unknown';
|
||||
let text = name;
|
||||
if (includeHash === true) {
|
||||
text = '#' + text;
|
||||
}
|
||||
if (includeCount === true) {
|
||||
text += ' (' + (tag ? tag.postCount : 0) + ')';
|
||||
}
|
||||
return api.hasPrivilege('tags:view') ?
|
||||
makeElement(
|
||||
'a',
|
||||
|
|
|
@ -2,6 +2,7 @@
|
|||
|
||||
const router = require('../router.js');
|
||||
const uri = require('../util/uri.js');
|
||||
const misc = require('../util/misc.js');
|
||||
const views = require('../util/views.js');
|
||||
const PostContentControl = require('../controls/post_content_control.js');
|
||||
const PostNotesOverlayControl
|
||||
|
@ -23,12 +24,16 @@ class HomeView {
|
|||
views.syncScrollPosition();
|
||||
|
||||
if (this._formNode) {
|
||||
this._tagAutoCompleteControl = new TagAutoCompleteControl(
|
||||
this._searchInputNode);
|
||||
this._autoCompleteControl = new TagAutoCompleteControl(
|
||||
this._searchInputNode,
|
||||
{
|
||||
confirm: tag =>
|
||||
this._autoCompleteControl.replaceSelectedText(
|
||||
misc.escapeSearchTerm(tag.names[0]), true),
|
||||
});
|
||||
this._formNode.addEventListener(
|
||||
'submit', e => this._evtFormSubmit(e));
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
showSuccess(text) {
|
||||
|
|
|
@ -73,7 +73,12 @@ class BulkTagEditor extends BulkEditor {
|
|||
constructor(hostNode) {
|
||||
super(hostNode);
|
||||
this._autoCompleteControl = new TagAutoCompleteControl(
|
||||
this._inputNode, {addSpace: false});
|
||||
this._inputNode,
|
||||
{
|
||||
confirm: tag =>
|
||||
this._autoCompleteControl.replaceSelectedText(
|
||||
tag.names[0], false),
|
||||
});
|
||||
this._hostNode.addEventListener('submit', e => this._evtFormSubmit(e));
|
||||
}
|
||||
|
||||
|
@ -124,9 +129,13 @@ class PostsHeaderView extends events.EventTarget {
|
|||
this._hostNode = ctx.hostNode;
|
||||
views.replaceContent(this._hostNode, template(ctx));
|
||||
|
||||
this._queryAutoCompleteControl = new TagAutoCompleteControl(
|
||||
this._autoCompleteControl = new TagAutoCompleteControl(
|
||||
this._queryInputNode,
|
||||
{addSpace: true, transform: misc.escapeSearchTerm});
|
||||
{
|
||||
confirm: tag =>
|
||||
this._autoCompleteControl.replaceSelectedText(
|
||||
misc.escapeSearchTerm(tag.names[0]), true),
|
||||
});
|
||||
|
||||
keyboard.bind('p', () => this._focusFirstPostNode());
|
||||
search.searchInputNodeFocusHelper(this._queryInputNode);
|
||||
|
@ -235,7 +244,7 @@ class PostsHeaderView extends events.EventTarget {
|
|||
}
|
||||
|
||||
_navigate() {
|
||||
this._queryAutoCompleteControl.hide();
|
||||
this._autoCompleteControl.hide();
|
||||
let parameters = {query: this._queryInputNode.value};
|
||||
parameters.offset = parameters.query === this._ctx.parameters.query ?
|
||||
this._ctx.parameters.offset : 0;
|
||||
|
|
|
@ -100,7 +100,7 @@ class PostsPageView extends events.EventTarget {
|
|||
if (tagFlipperNode) {
|
||||
let tagged = true;
|
||||
for (let tag of this._ctx.bulkEdit.tags) {
|
||||
tagged = tagged & post.isTaggedWith(tag);
|
||||
tagged = tagged & post.tags.isTaggedWith(tag);
|
||||
}
|
||||
tagFlipperNode.classList.toggle('tagged', tagged);
|
||||
}
|
||||
|
|
|
@ -24,10 +24,12 @@ class TagEditView extends events.EventTarget {
|
|||
}
|
||||
|
||||
if (this._implicationsFieldNode) {
|
||||
new TagInputControl(this._implicationsFieldNode);
|
||||
new TagInputControl(
|
||||
this._implicationsFieldNode, this._tag.implications);
|
||||
}
|
||||
if (this._suggestionsFieldNode) {
|
||||
new TagInputControl(this._suggestionsFieldNode);
|
||||
new TagInputControl(
|
||||
this._suggestionsFieldNode, this._tag.suggestions);
|
||||
}
|
||||
|
||||
for (let node of this._formNode.querySelectorAll(
|
||||
|
|
|
@ -2,6 +2,7 @@
|
|||
|
||||
const config = require('../config.js');
|
||||
const events = require('../events.js');
|
||||
const misc = require('../util/misc.js');
|
||||
const views = require('../util/views.js');
|
||||
const TagAutoCompleteControl =
|
||||
require('../controls/tag_auto_complete_control.js');
|
||||
|
@ -19,7 +20,13 @@ class TagMergeView extends events.EventTarget {
|
|||
|
||||
views.decorateValidator(this._formNode);
|
||||
if (this._targetTagFieldNode) {
|
||||
new TagAutoCompleteControl(this._targetTagFieldNode);
|
||||
this._autoCompleteControl = new TagAutoCompleteControl(
|
||||
this._targetTagFieldNode,
|
||||
{
|
||||
confirm: tag =>
|
||||
this._autoCompleteControl.replaceSelectedText(
|
||||
tag.names[0], false),
|
||||
});
|
||||
}
|
||||
|
||||
this._formNode.addEventListener('submit', e => this._evtSubmit(e));
|
||||
|
|
|
@ -17,8 +17,13 @@ class TagsHeaderView extends events.EventTarget {
|
|||
views.replaceContent(this._hostNode, template(ctx));
|
||||
|
||||
if (this._queryInputNode) {
|
||||
new TagAutoCompleteControl(
|
||||
this._queryInputNode, {transform: misc.escapeSearchTerm});
|
||||
this._autoCompleteControl = new TagAutoCompleteControl(
|
||||
this._queryInputNode,
|
||||
{
|
||||
confirm: tag =>
|
||||
this._autoCompleteControl.replaceSelectedText(
|
||||
misc.escapeSearchTerm(tag.names[0]), true),
|
||||
});
|
||||
}
|
||||
|
||||
search.searchInputNodeFocusHelper(this._queryInputNode);
|
||||
|
|
|
@ -106,7 +106,7 @@ privileges:
|
|||
'tags:edit:description': power
|
||||
'tags:edit:implications': power
|
||||
'tags:edit:suggestions': power
|
||||
'tags:list': regular # note: will be available as data_url/tags.json anyway
|
||||
'tags:list': regular
|
||||
'tags:view': anonymous
|
||||
'tags:merge': moderator
|
||||
'tags:delete': moderator
|
||||
|
@ -114,7 +114,7 @@ privileges:
|
|||
'tag_categories:create': moderator
|
||||
'tag_categories:edit:name': moderator
|
||||
'tag_categories:edit:color': moderator
|
||||
'tag_categories:list': anonymous # note: will be available as data_url/tags.json anyway
|
||||
'tag_categories:list': anonymous
|
||||
'tag_categories:view': anonymous
|
||||
'tag_categories:delete': moderator
|
||||
'tag_categories:set_default': moderator
|
||||
|
|
|
@ -73,7 +73,6 @@ def create_post(
|
|||
for tag in new_tags:
|
||||
snapshots.create(tag, None if anonymous else ctx.user)
|
||||
ctx.session.commit()
|
||||
tags.export_to_json()
|
||||
return _serialize_post(ctx, post)
|
||||
|
||||
|
||||
|
@ -126,7 +125,6 @@ def update_post(ctx: rest.Context, params: Dict[str, str]) -> rest.Response:
|
|||
ctx.session.flush()
|
||||
snapshots.modify(post, ctx.user)
|
||||
ctx.session.commit()
|
||||
tags.export_to_json()
|
||||
return _serialize_post(ctx, post)
|
||||
|
||||
|
||||
|
@ -138,7 +136,6 @@ def delete_post(ctx: rest.Context, params: Dict[str, str]) -> rest.Response:
|
|||
snapshots.delete(post, ctx.user)
|
||||
posts.delete(post)
|
||||
ctx.session.commit()
|
||||
tags.export_to_json()
|
||||
return {}
|
||||
|
||||
|
||||
|
|
|
@ -54,7 +54,6 @@ def create_tag(
|
|||
ctx.session.flush()
|
||||
snapshots.create(tag, ctx.user)
|
||||
ctx.session.commit()
|
||||
tags.export_to_json()
|
||||
return _serialize(ctx, tag)
|
||||
|
||||
|
||||
|
@ -95,7 +94,6 @@ def update_tag(ctx: rest.Context, params: Dict[str, str]) -> rest.Response:
|
|||
ctx.session.flush()
|
||||
snapshots.modify(tag, ctx.user)
|
||||
ctx.session.commit()
|
||||
tags.export_to_json()
|
||||
return _serialize(ctx, tag)
|
||||
|
||||
|
||||
|
@ -107,7 +105,6 @@ def delete_tag(ctx: rest.Context, params: Dict[str, str]) -> rest.Response:
|
|||
snapshots.delete(tag, ctx.user)
|
||||
tags.delete(tag)
|
||||
ctx.session.commit()
|
||||
tags.export_to_json()
|
||||
return {}
|
||||
|
||||
|
||||
|
@ -125,7 +122,6 @@ def merge_tags(
|
|||
tags.merge_tags(source_tag, target_tag)
|
||||
snapshots.merge(source_tag, target_tag, ctx.user)
|
||||
ctx.session.commit()
|
||||
tags.export_to_json()
|
||||
return _serialize(ctx, target_tag)
|
||||
|
||||
|
||||
|
|
|
@ -31,7 +31,6 @@ def create_tag_category(
|
|||
ctx.session.flush()
|
||||
snapshots.create(category, ctx.user)
|
||||
ctx.session.commit()
|
||||
tags.export_to_json()
|
||||
return _serialize(ctx, category)
|
||||
|
||||
|
||||
|
@ -61,7 +60,6 @@ def update_tag_category(
|
|||
ctx.session.flush()
|
||||
snapshots.modify(category, ctx.user)
|
||||
ctx.session.commit()
|
||||
tags.export_to_json()
|
||||
return _serialize(ctx, category)
|
||||
|
||||
|
||||
|
@ -75,7 +73,6 @@ def delete_tag_category(
|
|||
tag_categories.delete_category(category)
|
||||
snapshots.delete(category, ctx.user)
|
||||
ctx.session.commit()
|
||||
tags.export_to_json()
|
||||
return {}
|
||||
|
||||
|
||||
|
@ -89,5 +86,4 @@ def set_tag_category_as_default(
|
|||
ctx.session.flush()
|
||||
snapshots.modify(category, ctx.user)
|
||||
ctx.session.commit()
|
||||
tags.export_to_json()
|
||||
return _serialize(ctx, category)
|
||||
|
|
|
@ -224,7 +224,13 @@ class PostSerializer(serialization.BaseSerializer):
|
|||
return self.post.flags
|
||||
|
||||
def serialize_tags(self) -> Any:
|
||||
return [tag.names[0].name for tag in tags.sort_tags(self.post.tags)]
|
||||
return [
|
||||
{
|
||||
'names': [name.name for name in tag.names],
|
||||
'category': tag.category.name,
|
||||
'usages': tag.post_count,
|
||||
}
|
||||
for tag in tags.sort_tags(self.post.tags)]
|
||||
|
||||
def serialize_relations(self) -> Any:
|
||||
return sorted(
|
||||
|
|
|
@ -72,6 +72,14 @@ def sort_tags(tags: List[model.Tag]) -> List[model.Tag]:
|
|||
)
|
||||
|
||||
|
||||
def serialize_relation(tag):
|
||||
return {
|
||||
'names': [tag_name.name for tag_name in tag.names],
|
||||
'category': tag.category.name,
|
||||
'usages': tag.post_count,
|
||||
}
|
||||
|
||||
|
||||
class TagSerializer(serialization.BaseSerializer):
|
||||
def __init__(self, tag: model.Tag) -> None:
|
||||
self.tag = tag
|
||||
|
@ -112,12 +120,12 @@ class TagSerializer(serialization.BaseSerializer):
|
|||
|
||||
def serialize_suggestions(self) -> Any:
|
||||
return [
|
||||
relation.names[0].name
|
||||
serialize_relation(relation)
|
||||
for relation in sort_tags(self.tag.suggestions)]
|
||||
|
||||
def serialize_implications(self) -> Any:
|
||||
return [
|
||||
relation.names[0].name
|
||||
serialize_relation(relation)
|
||||
for relation in sort_tags(self.tag.implications)]
|
||||
|
||||
|
||||
|
@ -128,67 +136,6 @@ def serialize_tag(
|
|||
return TagSerializer(tag).serialize(options)
|
||||
|
||||
|
||||
def export_to_json() -> None:
|
||||
tags = {} # type: Dict[int, Any]
|
||||
categories = {} # type: Dict[int, Any]
|
||||
|
||||
for result in db.session.query(
|
||||
model.TagCategory.tag_category_id,
|
||||
model.TagCategory.name,
|
||||
model.TagCategory.color).all():
|
||||
categories[result[0]] = {
|
||||
'name': result[1],
|
||||
'color': result[2],
|
||||
}
|
||||
|
||||
for result in (
|
||||
db.session
|
||||
.query(model.TagName.tag_id, model.TagName.name)
|
||||
.order_by(model.TagName.order)
|
||||
.all()):
|
||||
if not result[0] in tags:
|
||||
tags[result[0]] = {'names': []}
|
||||
tags[result[0]]['names'].append(result[1])
|
||||
|
||||
for result in (
|
||||
db.session
|
||||
.query(model.TagSuggestion.parent_id, model.TagName.name)
|
||||
.join(
|
||||
model.TagName,
|
||||
model.TagName.tag_id == model.TagSuggestion.child_id)
|
||||
.all()):
|
||||
if 'suggestions' not in tags[result[0]]:
|
||||
tags[result[0]]['suggestions'] = []
|
||||
tags[result[0]]['suggestions'].append(result[1])
|
||||
|
||||
for result in (
|
||||
db.session
|
||||
.query(model.TagImplication.parent_id, model.TagName.name)
|
||||
.join(
|
||||
model.TagName,
|
||||
model.TagName.tag_id == model.TagImplication.child_id)
|
||||
.all()):
|
||||
if 'implications' not in tags[result[0]]:
|
||||
tags[result[0]]['implications'] = []
|
||||
tags[result[0]]['implications'].append(result[1])
|
||||
|
||||
for result in db.session.query(
|
||||
model.Tag.tag_id,
|
||||
model.Tag.category_id,
|
||||
model.Tag.post_count).all():
|
||||
tags[result[0]]['category'] = categories[result[1]]['name']
|
||||
tags[result[0]]['usages'] = result[2]
|
||||
|
||||
output = {
|
||||
'categories': list(categories.values()),
|
||||
'tags': list(tags.values()),
|
||||
}
|
||||
|
||||
export_path = os.path.join(config.config['data_dir'], 'tags.json')
|
||||
with open(export_path, 'w') as handle:
|
||||
handle.write(json.dumps(output, separators=(',', ':')))
|
||||
|
||||
|
||||
def try_get_tag_by_name(name: str) -> Optional[model.Tag]:
|
||||
return (
|
||||
db.session
|
||||
|
|
|
@ -134,7 +134,7 @@ class Executor:
|
|||
'offset': offset,
|
||||
'limit': limit,
|
||||
'total': count,
|
||||
'results': [serializer(entity) for entity in entities],
|
||||
'results': list([serializer(entity) for entity in entities]),
|
||||
}
|
||||
|
||||
def _prepare_db_query(
|
||||
|
|
|
@ -30,7 +30,6 @@ def test_creating_minimal_posts(
|
|||
patch('szurubooru.func.posts.update_post_flags'), \
|
||||
patch('szurubooru.func.posts.update_post_thumbnail'), \
|
||||
patch('szurubooru.func.posts.serialize_post'), \
|
||||
patch('szurubooru.func.tags.export_to_json'), \
|
||||
patch('szurubooru.func.snapshots.create'):
|
||||
posts.create_post.return_value = (post, [])
|
||||
posts.serialize_post.return_value = 'serialized post'
|
||||
|
@ -62,7 +61,6 @@ def test_creating_minimal_posts(
|
|||
posts.serialize_post.assert_called_once_with(
|
||||
post, auth_user, options=[])
|
||||
snapshots.create.assert_called_once_with(post, auth_user)
|
||||
tags.export_to_json.assert_called_once_with()
|
||||
|
||||
|
||||
def test_creating_full_posts(context_factory, post_factory, user_factory):
|
||||
|
@ -78,7 +76,6 @@ def test_creating_full_posts(context_factory, post_factory, user_factory):
|
|||
patch('szurubooru.func.posts.update_post_notes'), \
|
||||
patch('szurubooru.func.posts.update_post_flags'), \
|
||||
patch('szurubooru.func.posts.serialize_post'), \
|
||||
patch('szurubooru.func.tags.export_to_json'), \
|
||||
patch('szurubooru.func.snapshots.create'):
|
||||
posts.create_post.return_value = (post, [])
|
||||
posts.serialize_post.return_value = 'serialized post'
|
||||
|
@ -111,7 +108,6 @@ def test_creating_full_posts(context_factory, post_factory, user_factory):
|
|||
posts.serialize_post.assert_called_once_with(
|
||||
post, auth_user, options=[])
|
||||
snapshots.create.assert_called_once_with(post, auth_user)
|
||||
tags.export_to_json.assert_called_once_with()
|
||||
|
||||
|
||||
def test_anonymous_uploads(
|
||||
|
@ -121,8 +117,7 @@ def test_anonymous_uploads(
|
|||
db.session.add(post)
|
||||
db.session.flush()
|
||||
|
||||
with patch('szurubooru.func.tags.export_to_json'), \
|
||||
patch('szurubooru.func.posts.serialize_post'), \
|
||||
with patch('szurubooru.func.posts.serialize_post'), \
|
||||
patch('szurubooru.func.posts.create_post'), \
|
||||
patch('szurubooru.func.posts.update_post_source'):
|
||||
config_injector({
|
||||
|
@ -152,7 +147,6 @@ def test_creating_from_url_saves_source(
|
|||
db.session.flush()
|
||||
|
||||
with patch('szurubooru.func.net.download'), \
|
||||
patch('szurubooru.func.tags.export_to_json'), \
|
||||
patch('szurubooru.func.posts.serialize_post'), \
|
||||
patch('szurubooru.func.posts.create_post'), \
|
||||
patch('szurubooru.func.posts.update_post_source'):
|
||||
|
@ -183,7 +177,6 @@ def test_creating_from_url_with_source_specified(
|
|||
db.session.flush()
|
||||
|
||||
with patch('szurubooru.func.net.download'), \
|
||||
patch('szurubooru.func.tags.export_to_json'), \
|
||||
patch('szurubooru.func.posts.serialize_post'), \
|
||||
patch('szurubooru.func.posts.create_post'), \
|
||||
patch('szurubooru.func.posts.update_post_source'):
|
||||
|
@ -245,7 +238,6 @@ def test_omitting_optional_field(
|
|||
patch('szurubooru.func.posts.update_post_notes'), \
|
||||
patch('szurubooru.func.posts.update_post_flags'), \
|
||||
patch('szurubooru.func.posts.serialize_post'), \
|
||||
patch('szurubooru.func.tags.export_to_json'), \
|
||||
patch('szurubooru.func.snapshots.create'):
|
||||
posts.create_post.return_value = (post, [])
|
||||
posts.serialize_post.return_value = 'serialized post'
|
||||
|
|
|
@ -14,15 +14,13 @@ def test_deleting(user_factory, post_factory, context_factory):
|
|||
post = post_factory(id=1)
|
||||
db.session.add(post)
|
||||
db.session.flush()
|
||||
with patch('szurubooru.func.tags.export_to_json'), \
|
||||
patch('szurubooru.func.snapshots.delete'):
|
||||
with patch('szurubooru.func.snapshots.delete'):
|
||||
result = api.post_api.delete_post(
|
||||
context_factory(params={'version': 1}, user=auth_user),
|
||||
{'post_id': 1})
|
||||
assert result == {}
|
||||
assert db.session.query(model.Post).count() == 0
|
||||
snapshots.delete.assert_called_once_with(post, auth_user)
|
||||
tags.export_to_json.assert_called_once_with()
|
||||
|
||||
|
||||
def test_trying_to_delete_non_existing(user_factory, context_factory):
|
||||
|
|
|
@ -39,7 +39,6 @@ def test_post_updating(
|
|||
patch('szurubooru.func.posts.update_post_notes'), \
|
||||
patch('szurubooru.func.posts.update_post_flags'), \
|
||||
patch('szurubooru.func.posts.serialize_post'), \
|
||||
patch('szurubooru.func.tags.export_to_json'), \
|
||||
patch('szurubooru.func.snapshots.modify'), \
|
||||
fake_datetime('1997-01-01'):
|
||||
posts.serialize_post.return_value = 'serialized post'
|
||||
|
@ -78,7 +77,6 @@ def test_post_updating(
|
|||
posts.serialize_post.assert_called_once_with(
|
||||
post, auth_user, options=[])
|
||||
snapshots.modify.assert_called_once_with(post, auth_user)
|
||||
tags.export_to_json.assert_called_once_with()
|
||||
assert post.last_edit_time == datetime(1997, 1, 1)
|
||||
|
||||
|
||||
|
@ -88,7 +86,6 @@ def test_uploading_from_url_saves_source(
|
|||
db.session.add(post)
|
||||
db.session.flush()
|
||||
with patch('szurubooru.func.net.download'), \
|
||||
patch('szurubooru.func.tags.export_to_json'), \
|
||||
patch('szurubooru.func.posts.serialize_post'), \
|
||||
patch('szurubooru.func.posts.update_post_content'), \
|
||||
patch('szurubooru.func.posts.update_post_source'), \
|
||||
|
@ -110,7 +107,6 @@ def test_uploading_from_url_with_source_specified(
|
|||
db.session.add(post)
|
||||
db.session.flush()
|
||||
with patch('szurubooru.func.net.download'), \
|
||||
patch('szurubooru.func.tags.export_to_json'), \
|
||||
patch('szurubooru.func.posts.serialize_post'), \
|
||||
patch('szurubooru.func.posts.update_post_content'), \
|
||||
patch('szurubooru.func.posts.update_post_source'), \
|
||||
|
|
|
@ -24,8 +24,7 @@ def test_creating_category(
|
|||
with patch('szurubooru.func.tag_categories.create_category'), \
|
||||
patch('szurubooru.func.tag_categories.serialize_category'), \
|
||||
patch('szurubooru.func.tag_categories.update_category_name'), \
|
||||
patch('szurubooru.func.snapshots.create'), \
|
||||
patch('szurubooru.func.tags.export_to_json'):
|
||||
patch('szurubooru.func.snapshots.create'):
|
||||
tag_categories.create_category.return_value = category
|
||||
tag_categories.update_category_name.side_effect = _update_category_name
|
||||
tag_categories.serialize_category.return_value = 'serialized category'
|
||||
|
@ -35,7 +34,6 @@ def test_creating_category(
|
|||
assert result == 'serialized category'
|
||||
tag_categories.create_category.assert_called_once_with('meta', 'black')
|
||||
snapshots.create.assert_called_once_with(category, auth_user)
|
||||
tags.export_to_json.assert_called_once_with()
|
||||
|
||||
|
||||
@pytest.mark.parametrize('field', ['name', 'color'])
|
||||
|
|
|
@ -17,8 +17,7 @@ def test_deleting(user_factory, tag_category_factory, context_factory):
|
|||
db.session.add(tag_category_factory(name='root'))
|
||||
db.session.add(category)
|
||||
db.session.flush()
|
||||
with patch('szurubooru.func.snapshots.delete'), \
|
||||
patch('szurubooru.func.tags.export_to_json'):
|
||||
with patch('szurubooru.func.snapshots.delete'):
|
||||
result = api.tag_category_api.delete_tag_category(
|
||||
context_factory(params={'version': 1}, user=auth_user),
|
||||
{'category_name': 'category'})
|
||||
|
@ -26,7 +25,6 @@ def test_deleting(user_factory, tag_category_factory, context_factory):
|
|||
assert db.session.query(model.TagCategory).count() == 1
|
||||
assert db.session.query(model.TagCategory).one().name == 'root'
|
||||
snapshots.delete.assert_called_once_with(category, auth_user)
|
||||
tags.export_to_json.assert_called_once_with()
|
||||
|
||||
|
||||
def test_trying_to_delete_used(
|
||||
|
|
|
@ -27,8 +27,7 @@ def test_simple_updating(user_factory, tag_category_factory, context_factory):
|
|||
with patch('szurubooru.func.tag_categories.serialize_category'), \
|
||||
patch('szurubooru.func.tag_categories.update_category_name'), \
|
||||
patch('szurubooru.func.tag_categories.update_category_color'), \
|
||||
patch('szurubooru.func.snapshots.modify'), \
|
||||
patch('szurubooru.func.tags.export_to_json'):
|
||||
patch('szurubooru.func.snapshots.modify'):
|
||||
tag_categories.update_category_name.side_effect = _update_category_name
|
||||
tag_categories.serialize_category.return_value = 'serialized category'
|
||||
result = api.tag_category_api.update_tag_category(
|
||||
|
@ -42,7 +41,6 @@ def test_simple_updating(user_factory, tag_category_factory, context_factory):
|
|||
tag_categories.update_category_color.assert_called_once_with(
|
||||
category, 'white')
|
||||
snapshots.modify.assert_called_once_with(category, auth_user)
|
||||
tags.export_to_json.assert_called_once_with()
|
||||
|
||||
|
||||
@pytest.mark.parametrize('field', ['name', 'color'])
|
||||
|
@ -56,8 +54,7 @@ def test_omitting_optional_field(
|
|||
}
|
||||
del params[field]
|
||||
with patch('szurubooru.func.tag_categories.serialize_category'), \
|
||||
patch('szurubooru.func.tag_categories.update_category_name'), \
|
||||
patch('szurubooru.func.tags.export_to_json'):
|
||||
patch('szurubooru.func.tag_categories.update_category_name'):
|
||||
api.tag_category_api.update_tag_category(
|
||||
context_factory(
|
||||
params={**params, **{'version': 1}},
|
||||
|
@ -95,8 +92,7 @@ def test_set_as_default(user_factory, tag_category_factory, context_factory):
|
|||
db.session.add(category)
|
||||
db.session.commit()
|
||||
with patch('szurubooru.func.tag_categories.serialize_category'), \
|
||||
patch('szurubooru.func.tag_categories.set_default_category'), \
|
||||
patch('szurubooru.func.tags.export_to_json'):
|
||||
patch('szurubooru.func.tag_categories.set_default_category'):
|
||||
tag_categories.update_category_name.side_effect = _update_category_name
|
||||
tag_categories.serialize_category.return_value = 'serialized category'
|
||||
result = api.tag_category_api.set_tag_category_as_default(
|
||||
|
|
|
@ -15,8 +15,7 @@ def test_creating_simple_tags(tag_factory, user_factory, context_factory):
|
|||
with patch('szurubooru.func.tags.create_tag'), \
|
||||
patch('szurubooru.func.tags.get_or_create_tags_by_names'), \
|
||||
patch('szurubooru.func.tags.serialize_tag'), \
|
||||
patch('szurubooru.func.snapshots.create'), \
|
||||
patch('szurubooru.func.tags.export_to_json'):
|
||||
patch('szurubooru.func.snapshots.create'):
|
||||
tags.get_or_create_tags_by_names.return_value = ([], [])
|
||||
tags.create_tag.return_value = tag
|
||||
tags.serialize_tag.return_value = 'serialized tag'
|
||||
|
@ -34,7 +33,6 @@ def test_creating_simple_tags(tag_factory, user_factory, context_factory):
|
|||
tags.create_tag.assert_called_once_with(
|
||||
['tag1', 'tag2'], 'meta', ['sug1', 'sug2'], ['imp1', 'imp2'])
|
||||
snapshots.create.assert_called_once_with(tag, auth_user)
|
||||
tags.export_to_json.assert_called_once_with()
|
||||
|
||||
|
||||
@pytest.mark.parametrize('field', ['names', 'category'])
|
||||
|
@ -64,8 +62,7 @@ def test_omitting_optional_field(
|
|||
}
|
||||
del params[field]
|
||||
with patch('szurubooru.func.tags.create_tag'), \
|
||||
patch('szurubooru.func.tags.serialize_tag'), \
|
||||
patch('szurubooru.func.tags.export_to_json'):
|
||||
patch('szurubooru.func.tags.serialize_tag'):
|
||||
tags.create_tag.return_value = tag_factory()
|
||||
api.tag_api.create_tag(
|
||||
context_factory(
|
||||
|
|
|
@ -14,15 +14,13 @@ def test_deleting(user_factory, tag_factory, context_factory):
|
|||
tag = tag_factory(names=['tag'])
|
||||
db.session.add(tag)
|
||||
db.session.commit()
|
||||
with patch('szurubooru.func.tags.export_to_json'), \
|
||||
patch('szurubooru.func.snapshots.delete'):
|
||||
with patch('szurubooru.func.snapshots.delete'):
|
||||
result = api.tag_api.delete_tag(
|
||||
context_factory(params={'version': 1}, user=auth_user),
|
||||
{'tag_name': 'tag'})
|
||||
assert result == {}
|
||||
assert db.session.query(model.Tag).count() == 0
|
||||
snapshots.delete.assert_called_once_with(tag, auth_user)
|
||||
tags.export_to_json.assert_called_once_with()
|
||||
|
||||
|
||||
def test_deleting_used(
|
||||
|
@ -32,7 +30,6 @@ def test_deleting_used(
|
|||
post.tags.append(tag)
|
||||
db.session.add_all([tag, post])
|
||||
db.session.commit()
|
||||
with patch('szurubooru.func.tags.export_to_json'):
|
||||
api.tag_api.delete_tag(
|
||||
context_factory(
|
||||
params={'version': 1},
|
||||
|
|
|
@ -25,8 +25,7 @@ def test_merging(user_factory, tag_factory, context_factory, post_factory):
|
|||
assert target_tag.post_count == 0
|
||||
with patch('szurubooru.func.tags.serialize_tag'), \
|
||||
patch('szurubooru.func.tags.merge_tags'), \
|
||||
patch('szurubooru.func.snapshots.merge'), \
|
||||
patch('szurubooru.func.tags.export_to_json'):
|
||||
patch('szurubooru.func.snapshots.merge'):
|
||||
api.tag_api.merge_tags(
|
||||
context_factory(
|
||||
params={
|
||||
|
@ -39,7 +38,6 @@ def test_merging(user_factory, tag_factory, context_factory, post_factory):
|
|||
tags.merge_tags.called_once_with(source_tag, target_tag)
|
||||
snapshots.merge.assert_called_once_with(
|
||||
source_tag, target_tag, auth_user)
|
||||
tags.export_to_json.assert_called_once_with()
|
||||
|
||||
|
||||
@pytest.mark.parametrize(
|
||||
|
|
|
@ -31,8 +31,7 @@ def test_simple_updating(user_factory, tag_factory, context_factory):
|
|||
patch('szurubooru.func.tags.update_tag_suggestions'), \
|
||||
patch('szurubooru.func.tags.update_tag_implications'), \
|
||||
patch('szurubooru.func.tags.serialize_tag'), \
|
||||
patch('szurubooru.func.snapshots.modify'), \
|
||||
patch('szurubooru.func.tags.export_to_json'):
|
||||
patch('szurubooru.func.snapshots.modify'):
|
||||
tags.get_or_create_tags_by_names.return_value = ([], [])
|
||||
tags.serialize_tag.return_value = 'serialized tag'
|
||||
result = api.tag_api.update_tag(
|
||||
|
@ -58,7 +57,6 @@ def test_simple_updating(user_factory, tag_factory, context_factory):
|
|||
tag, ['imp1', 'imp2'])
|
||||
tags.serialize_tag.assert_called_once_with(tag, options=[])
|
||||
snapshots.modify.assert_called_once_with(tag, auth_user)
|
||||
tags.export_to_json.assert_called_once_with()
|
||||
|
||||
|
||||
@pytest.mark.parametrize(
|
||||
|
@ -84,8 +82,7 @@ def test_omitting_optional_field(
|
|||
with patch('szurubooru.func.tags.create_tag'), \
|
||||
patch('szurubooru.func.tags.update_tag_names'), \
|
||||
patch('szurubooru.func.tags.update_tag_category_name'), \
|
||||
patch('szurubooru.func.tags.serialize_tag'), \
|
||||
patch('szurubooru.func.tags.export_to_json'):
|
||||
patch('szurubooru.func.tags.serialize_tag'):
|
||||
api.tag_api.update_tag(
|
||||
context_factory(
|
||||
params={**params, **{'version': 1}},
|
||||
|
|
|
@ -75,7 +75,11 @@ def test_serialize_post_when_empty():
|
|||
|
||||
|
||||
def test_serialize_post(
|
||||
user_factory, comment_factory, tag_factory, config_injector):
|
||||
user_factory,
|
||||
comment_factory,
|
||||
tag_factory,
|
||||
tag_category_factory,
|
||||
config_injector):
|
||||
config_injector({'data_url': 'http://example.com/', 'secret': 'test'})
|
||||
with patch('szurubooru.func.comments.serialize_comment'), \
|
||||
patch('szurubooru.func.users.serialize_micro_user'), \
|
||||
|
@ -92,8 +96,12 @@ def test_serialize_post(
|
|||
post.creation_time = datetime(1997, 1, 1)
|
||||
post.last_edit_time = datetime(1998, 1, 1)
|
||||
post.tags = [
|
||||
tag_factory(names=['tag1', 'tag2']),
|
||||
tag_factory(names=['tag3'])
|
||||
tag_factory(
|
||||
names=['tag1', 'tag2'],
|
||||
category=tag_category_factory('test-cat1')),
|
||||
tag_factory(
|
||||
names=['tag3'],
|
||||
category=tag_category_factory('test-cat2'))
|
||||
]
|
||||
post.safety = model.Post.SAFETY_SAFE
|
||||
post.source = '4gag'
|
||||
|
@ -143,7 +151,7 @@ def test_serialize_post(
|
|||
db.session.flush()
|
||||
|
||||
result = posts.serialize_post(post, auth_user)
|
||||
result['tags'].sort()
|
||||
result['tags'].sort(key=lambda tag: tag['names'][0])
|
||||
|
||||
assert result == {
|
||||
'id': 1,
|
||||
|
@ -162,7 +170,17 @@ def test_serialize_post(
|
|||
'http://example.com/'
|
||||
'generated-thumbnails/1_244c8840887984c4.jpg',
|
||||
'flags': ['loop'],
|
||||
'tags': ['tag1', 'tag3'],
|
||||
'tags': [
|
||||
{
|
||||
'names': ['tag1', 'tag2'],
|
||||
'category': 'test-cat1', 'usages': 1,
|
||||
},
|
||||
{
|
||||
'names': ['tag3'],
|
||||
'category': 'test-cat2',
|
||||
'usages': 1,
|
||||
},
|
||||
],
|
||||
'relations': [],
|
||||
'notes': [],
|
||||
'user': 'post author',
|
||||
|
|
|
@ -45,15 +45,18 @@ def test_serialize_tag_when_empty():
|
|||
|
||||
|
||||
def test_serialize_tag(post_factory, tag_factory, tag_category_factory):
|
||||
tag = tag_factory(
|
||||
names=['tag1', 'tag2'],
|
||||
category=tag_category_factory(name='cat'))
|
||||
cat = tag_category_factory(name='cat')
|
||||
tag = tag_factory(names=['tag1', 'tag2'], category=cat)
|
||||
tag.tag_id = 1
|
||||
tag.description = 'description'
|
||||
tag.suggestions = [
|
||||
tag_factory(names=['sug1']), tag_factory(names=['sug2'])]
|
||||
tag_factory(names=['sug1'], category=cat),
|
||||
tag_factory(names=['sug2'], category=cat),
|
||||
]
|
||||
tag.implications = [
|
||||
tag_factory(names=['impl1']), tag_factory(names=['impl2'])]
|
||||
tag_factory(names=['impl1'], category=cat),
|
||||
tag_factory(names=['impl2'], category=cat),
|
||||
]
|
||||
tag.last_edit_time = datetime(1998, 1, 1)
|
||||
post1 = post_factory()
|
||||
post2 = post_factory()
|
||||
|
@ -62,8 +65,8 @@ def test_serialize_tag(post_factory, tag_factory, tag_category_factory):
|
|||
db.session.add_all([tag, post1, post2])
|
||||
db.session.flush()
|
||||
result = tags.serialize_tag(tag)
|
||||
result['suggestions'].sort()
|
||||
result['implications'].sort()
|
||||
result['suggestions'].sort(key=lambda relation: relation['names'][0])
|
||||
result['implications'].sort(key=lambda relation: relation['names'][0])
|
||||
assert result == {
|
||||
'names': ['tag1', 'tag2'],
|
||||
'version': 1,
|
||||
|
@ -71,66 +74,15 @@ def test_serialize_tag(post_factory, tag_factory, tag_category_factory):
|
|||
'creationTime': datetime(1996, 1, 1, 0, 0),
|
||||
'lastEditTime': datetime(1998, 1, 1, 0, 0),
|
||||
'description': 'description',
|
||||
'suggestions': ['sug1', 'sug2'],
|
||||
'implications': ['impl1', 'impl2'],
|
||||
'usages': 2,
|
||||
}
|
||||
|
||||
|
||||
def test_export_to_json(
|
||||
tmpdir,
|
||||
query_counter,
|
||||
config_injector,
|
||||
post_factory,
|
||||
tag_factory,
|
||||
tag_category_factory):
|
||||
config_injector({'data_dir': str(tmpdir)})
|
||||
cat1 = tag_category_factory(name='cat1', color='black')
|
||||
cat2 = tag_category_factory(name='cat2', color='white')
|
||||
tag = tag_factory(names=['alias1', 'alias2'], category=cat2)
|
||||
tag.suggestions = [
|
||||
tag_factory(names=['sug1'], category=cat1),
|
||||
tag_factory(names=['sug2'], category=cat1),
|
||||
]
|
||||
tag.implications = [
|
||||
tag_factory(names=['imp1'], category=cat1),
|
||||
tag_factory(names=['imp2'], category=cat1),
|
||||
]
|
||||
post = post_factory()
|
||||
post.tags = [tag]
|
||||
db.session.add_all([post, tag])
|
||||
db.session.flush()
|
||||
|
||||
with query_counter:
|
||||
tags.export_to_json()
|
||||
assert len(query_counter.statements) == 5
|
||||
|
||||
export_path = os.path.join(str(tmpdir), 'tags.json')
|
||||
assert os.path.exists(export_path)
|
||||
with open(export_path, 'r') as handle:
|
||||
actual_json = json.loads(handle.read())
|
||||
assert actual_json['tags']
|
||||
assert actual_json['categories']
|
||||
actual_json['tags'].sort(key=lambda tag: tag['names'][0])
|
||||
actual_json['categories'].sort(key=lambda category: category['name'])
|
||||
assert actual_json == {
|
||||
'tags': [
|
||||
{
|
||||
'names': ['alias1', 'alias2'],
|
||||
'usages': 1,
|
||||
'category': 'cat2',
|
||||
'suggestions': ['sug1', 'sug2'],
|
||||
'implications': ['imp1', 'imp2'],
|
||||
},
|
||||
{'names': ['imp1'], 'usages': 0, 'category': 'cat1'},
|
||||
{'names': ['imp2'], 'usages': 0, 'category': 'cat1'},
|
||||
{'names': ['sug1'], 'usages': 0, 'category': 'cat1'},
|
||||
{'names': ['sug2'], 'usages': 0, 'category': 'cat1'},
|
||||
'suggestions': [
|
||||
{'names': ['sug1'], 'category': 'cat', 'usages': 0},
|
||||
{'names': ['sug2'], 'category': 'cat', 'usages': 0},
|
||||
],
|
||||
'categories': [
|
||||
{'name': 'cat1', 'color': 'black'},
|
||||
{'name': 'cat2', 'color': 'white'},
|
||||
]
|
||||
'implications': [
|
||||
{'names': ['impl1'], 'category': 'cat', 'usages': 0},
|
||||
{'names': ['impl2'], 'category': 'cat', 'usages': 0},
|
||||
],
|
||||
'usages': 2,
|
||||
}
|
||||
|
||||
|
||||
|
|
Loading…
Reference in a new issue