misskey/src/client/app/common/scripts/note-mixin.ts

197 lines
4.2 KiB
TypeScript
Raw Normal View History

2019-01-30 08:56:27 +01:00
import { parse } from '../../../../mfm/parse';
import { sum, unique } from '../../../../prelude/array';
import shouldMuteNote from './should-mute-note';
2018-10-13 05:48:33 +02:00
import MkNoteMenu from '../views/components/note-menu.vue';
2018-10-12 07:28:48 +02:00
import MkReactionPicker from '../views/components/reaction-picker.vue';
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 = {}) => ({
data() {
return {
showContent: false,
hideThisNote: false
2018-10-12 07:28:48 +02:00
};
},
computed: {
keymap(): any {
return {
'r': () => this.reply(true),
2018-10-12 07:28:48 +02:00
'e|a|plus': () => this.react(true),
'q': () => this.renote(true),
2018-10-13 05:48:33 +02:00
'f|b': this.favorite,
'delete|ctrl+d': this.del,
'ctrl+q': this.renoteDirectly,
2018-10-12 07:28:48 +02: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-12 07:28:48 +02: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 14:50:36 +02:00
return this.appearNote.reactions
? sum(Object.values(this.appearNote.reactions))
2018-10-12 07:28:48 +02:00
: 0;
},
title(): string {
return '';
2018-10-12 07:28:48 +02:00
},
urls(): string[] {
if (this.appearNote.text) {
const ast = parse(this.appearNote.text);
2018-11-24 20:30:32 +01:00
// TODO: 再帰的にURL要素がないか調べる
return 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));
2018-10-12 07:28:48 +02:00
} else {
return null;
}
}
},
created() {
this.hideThisNote = shouldMuteNote(this.$store.state.i, this.$store.state.settings, this.appearNote);
},
2018-10-12 07:28:48 +02:00
methods: {
2018-10-12 07:34:54 +02:00
reply(viaKeyboard = false) {
this.$root.$post({
2018-10-12 07:34:54 +02:00
reply: this.appearNote,
animation: !viaKeyboard,
cb: () => {
this.focus();
}
});
},
renote(viaKeyboard = false) {
this.$root.$post({
2018-10-12 07:34:54 +02:00
renote: this.appearNote,
animation: !viaKeyboard,
cb: () => {
this.focus();
}
});
},
2018-10-12 07:28:48 +02:00
renoteDirectly() {
(this as any).api('notes/create', {
renoteId: this.appearNote.id
});
},
react(viaKeyboard = false) {
this.blur();
2018-11-09 00:13:34 +01:00
this.$root.new(MkReactionPicker, {
2018-10-12 07:28:48 +02:00
source: this.$refs.reactButton,
note: this.appearNote,
showFocus: viaKeyboard,
2018-12-29 17:32:58 +01:00
animation: !viaKeyboard
2018-10-12 07:28:48 +02:00
}).$once('closed', this.focus);
},
reactDirectly(reaction) {
2018-11-09 00:13:34 +01:00
(this.$root.api('notes/reactions/create', {
2018-10-12 07:28:48 +02: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-13 05:48:33 +02:00
favorite() {
2018-11-09 00:13:34 +01:00
this.$root.api('notes/favorites/create', {
2018-10-13 05:48:33 +02:00
noteId: this.appearNote.id
}).then(() => {
2018-12-02 07:28:52 +01:00
this.$root.dialog({
2018-11-14 16:01:49 +01:00
type: 'success',
splash: true
});
2018-10-13 05:48:33 +02:00
});
},
del() {
2018-11-09 00:13:34 +01:00
this.$root.api('notes/delete', {
2018-10-13 05:48:33 +02:00
noteId: this.appearNote.id
});
},
2018-10-12 07:28:48 +02:00
menu(viaKeyboard = false) {
2018-11-09 00:13:34 +01:00
this.$root.new(MkNoteMenu, {
2018-10-12 07:28:48 +02:00
source: this.$refs.menuButton,
note: this.appearNote,
2018-12-29 17:32:58 +01:00
animation: !viaKeyboard
2018-10-12 07:28:48 +02: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);
}
}
});