jormungandr-bite/src/client/app/common/scripts/note-mixin.ts

223 lines
4.9 KiB
TypeScript
Raw Normal View History

2019-01-30 00:56:27 -07:00
import { parse } from '../../../../mfm/parse';
import { sum, unique } from '../../../../prelude/array';
import shouldMuteNote from './should-mute-note';
2018-10-12 21:48:33 -06:00
import MkNoteMenu from '../views/components/note-menu.vue';
2018-10-11 23:28:48 -06:00
import MkReactionPicker from '../views/components/reaction-picker.vue';
import pleaseLogin from './please-login';
2019-08-04 19:30:31 -06:00
import i18n from '../../i18n';
2018-10-11 23:28:48 -06:00
function focus(el, fn) {
const target = fn(el);
if (target) {
if (target.hasAttribute('tabindex')) {
target.focus();
} else {
focus(target, fn);
}
}
}
type Opts = {
mobile?: boolean;
};
export default (opts: Opts = {}) => ({
2019-08-04 19:30:31 -06:00
i18n: i18n(),
2018-10-11 23:28:48 -06:00
data() {
return {
showContent: false,
hideThisNote: false
2018-10-11 23:28:48 -06:00
};
},
computed: {
keymap(): any {
return {
'r': () => this.reply(true),
2018-10-11 23:28:48 -06:00
'e|a|plus': () => this.react(true),
'q': () => this.renote(true),
2018-10-12 21:48:33 -06:00
'f|b': this.favorite,
'delete|ctrl+d': this.del,
'ctrl+q': this.renoteDirectly,
2018-10-11 23:28:48 -06:00
'up|k|shift+tab': this.focusBefore,
'down|j|tab': this.focusAfter,
'esc': this.blur,
'm|o': () => this.menu(true),
's': this.toggleShowContent,
'1': () => this.reactDirectly('like'),
'2': () => this.reactDirectly('love'),
'3': () => this.reactDirectly('laugh'),
'4': () => this.reactDirectly('hmm'),
'5': () => this.reactDirectly('surprise'),
'6': () => this.reactDirectly('congrats'),
'7': () => this.reactDirectly('angry'),
'8': () => this.reactDirectly('confused'),
'9': () => this.reactDirectly('rip'),
'0': () => this.reactDirectly('pudding'),
};
},
isRenote(): boolean {
return (this.note.renote &&
this.note.text == null &&
this.note.fileIds.length == 0 &&
this.note.poll == null);
},
appearNote(): any {
return this.isRenote ? this.note.renote : this.note;
},
isMyNote(): boolean {
return this.$store.getters.isSignedIn && (this.$store.state.i.id === this.appearNote.userId);
},
2018-10-11 23:28:48 -06:00
reactionsCount(): number {
Use PostgreSQL instead of MongoDB (#4572) * wip * Update note.ts * Update timeline.ts * Update core.ts * wip * Update generate-visibility-query.ts * wip * wip * wip * wip * wip * Update global-timeline.ts * wip * wip * wip * Update vote.ts * wip * wip * Update create.ts * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * Update files.ts * wip * wip * Update CONTRIBUTING.md * wip * wip * wip * wip * wip * wip * wip * wip * Update read-notification.ts * wip * wip * wip * wip * wip * wip * wip * Update cancel.ts * wip * wip * wip * Update show.ts * wip * wip * Update gen-id.ts * Update create.ts * Update id.ts * wip * wip * wip * wip * wip * wip * wip * Docker: Update files about Docker (#4599) * Docker: Use cache if files used by `yarn install` was not updated This patch reduces the number of times to installing node_modules. For example, `yarn install` step will be skipped when only ".config/default.yml" is updated. * Docker: Migrate MongoDB to Postgresql Misskey uses Postgresql as a database instead of Mongodb since version 11. * Docker: Uncomment about data persistence This patch will save a lot of databases. * wip * wip * wip * Update activitypub.ts * wip * wip * wip * Update logs.ts * wip * Update drive-file.ts * Update register.ts * wip * wip * Update mentions.ts * wip * wip * wip * Update recommendation.ts * wip * Update index.ts * wip * Update recommendation.ts * Doc: Update docker.ja.md and docker.en.md (#1) (#4608) Update how to set up misskey. * wip * :v: * wip * Update note.ts * Update postgre.ts * wip * wip * wip * wip * Update add-file.ts * wip * wip * wip * Clean up * Update logs.ts * wip * :pizza: * wip * Ad notes * wip * Update api-visibility.ts * Update note.ts * Update add-file.ts * tests * tests * Update postgre.ts * Update utils.ts * wip * wip * Refactor * wip * Refactor * wip * wip * Update show-users.ts * Update update-instance.ts * wip * Update feed.ts * Update outbox.ts * Update outbox.ts * Update user.ts * wip * Update list.ts * Update update-hashtag.ts * wip * Update update-hashtag.ts * Refactor * Update update.ts * wip * wip * :v: * clean up * docs * Update push.ts * wip * Update api.ts * wip * :v: * Update make-pagination-query.ts * :v: * Delete hashtags.ts * Update instances.ts * Update instances.ts * Update create.ts * Update search.ts * Update reversi-game.ts * Update signup.ts * Update user.ts * id * Update example.yml * :art: * objectid * fix * reversi * reversi * Fix bug of chart engine * Add test of chart engine * Improve test * Better testing * Improve chart engine * Refactor * Add test of chart engine * Refactor * Add chart test * Fix bug * コミットし忘れ * Refactoring * :v: * Add tests * Add test * Extarct note tests * Refactor * 存在しないユーザーにメンションできなくなっていた問題を修正 * Fix bug * Update update-meta.ts * Fix bug * Update mention.vue * Fix bug * Update meta.ts * Update CONTRIBUTING.md * Fix bug * Fix bug * Fix bug * Clean up * Clean up * Update notification.ts * Clean up * Add mute tests * Add test * Refactor * Add test * Fix test * Refactor * Refactor * Add tests * Update utils.ts * Update utils.ts * Fix test * Update package.json * Update update.ts * Update manifest.ts * Fix bug * Fix bug * Add test * :art: * Update endpoint permissions * Updaye permisison * Update person.ts #4299 * データベースと同期しないように * Fix bug * Fix bug * Update reversi-game.ts * Use a feature of Node v11.7.0 to extract a public key (#4644) * wip * wip * :v: * Refactoring #1540 * test * test * test * test * test * test * test * Fix bug * Fix test * :sushi: * wip * #4471 * Add test for #4335 * Refactor * Fix test * Add tests * :clock4: * Fix bug * Add test * Add test * rename * Fix bug
2019-04-07 06:50:36 -06:00
return this.appearNote.reactions
? sum(Object.values(this.appearNote.reactions))
2018-10-11 23:28:48 -06:00
: 0;
},
title(): string {
return '';
2018-10-11 23:28:48 -06:00
},
urls(): string[] {
if (this.appearNote.text) {
const ast = parse(this.appearNote.text);
2018-11-24 12:30:32 -07:00
// TODO: 再帰的にURL要素がないか調べる
const urls = unique(ast
.filter(t => ((t.node.type == 'url' || t.node.type == 'link') && t.node.props.url && !t.node.props.silent))
.map(t => t.node.props.url));
// unique without hash
// [ http://a/#1, http://a/#2, http://b/#3 ] => [ http://a/#1, http://b/#3 ]
const removeHash = x => x.replace(/#[^#]*$/, '');
return urls.reduce((array, url) => {
const removed = removeHash(url);
if (!array.map(x => removeHash(x)).includes(removed)) array.push(url);
return array;
}, []);
2018-10-11 23:28:48 -06:00
} else {
return null;
}
}
},
created() {
this.hideThisNote = shouldMuteNote(this.$store.state.i, this.$store.state.settings, this.appearNote);
},
2018-10-11 23:28:48 -06:00
methods: {
2018-10-11 23:34:54 -06:00
reply(viaKeyboard = false) {
pleaseLogin(this.$root);
this.$root.$post({
2018-10-11 23:34:54 -06:00
reply: this.appearNote,
animation: !viaKeyboard,
cb: () => {
this.focus();
}
});
},
renote(viaKeyboard = false) {
pleaseLogin(this.$root);
this.$root.$post({
2018-10-11 23:34:54 -06:00
renote: this.appearNote,
animation: !viaKeyboard,
cb: () => {
this.focus();
}
});
},
2018-10-11 23:28:48 -06:00
renoteDirectly() {
(this as any).api('notes/create', {
renoteId: this.appearNote.id
});
},
react(viaKeyboard = false) {
pleaseLogin(this.$root);
2018-10-11 23:28:48 -06:00
this.blur();
2018-11-08 16:13:34 -07:00
this.$root.new(MkReactionPicker, {
2018-10-11 23:28:48 -06:00
source: this.$refs.reactButton,
note: this.appearNote,
showFocus: viaKeyboard,
2018-12-29 09:32:58 -07:00
animation: !viaKeyboard
2018-10-11 23:28:48 -06:00
}).$once('closed', this.focus);
},
reactDirectly(reaction) {
2019-04-16 09:30:34 -06:00
this.$root.api('notes/reactions/create', {
2018-10-11 23:28:48 -06:00
noteId: this.appearNote.id,
reaction: reaction
});
},
undoReact(note) {
const oldReaction = note.myReaction;
if (!oldReaction) return;
this.$root.api('notes/reactions/delete', {
noteId: note.id
});
},
2018-10-12 21:48:33 -06:00
favorite() {
pleaseLogin(this.$root);
2018-11-08 16:13:34 -07:00
this.$root.api('notes/favorites/create', {
2018-10-12 21:48:33 -06:00
noteId: this.appearNote.id
}).then(() => {
2018-12-01 23:28:52 -07:00
this.$root.dialog({
2018-11-14 08:01:49 -07:00
type: 'success',
splash: true
});
2018-10-12 21:48:33 -06:00
});
},
del() {
2019-08-04 19:30:31 -06:00
this.$root.dialog({
type: 'warning',
text: this.$t('@.delete-confirm'),
showCancelButton: true
}).then(({ canceled }) => {
if (canceled) return;
this.$root.api('notes/delete', {
noteId: this.appearNote.id
});
2018-10-12 21:48:33 -06:00
});
},
2018-10-11 23:28:48 -06:00
menu(viaKeyboard = false) {
2018-11-08 16:13:34 -07:00
this.$root.new(MkNoteMenu, {
2018-10-11 23:28:48 -06:00
source: this.$refs.menuButton,
note: this.appearNote,
2018-12-29 09:32:58 -07:00
animation: !viaKeyboard
2018-10-11 23:28:48 -06:00
}).$once('closed', this.focus);
},
toggleShowContent() {
this.showContent = !this.showContent;
},
focus() {
this.$el.focus();
},
blur() {
this.$el.blur();
},
focusBefore() {
focus(this.$el, e => e.previousElementSibling);
},
focusAfter() {
focus(this.$el, e => e.nextElementSibling);
}
}
});