2020-01-29 20:37:25 +01:00
|
|
|
|
<template>
|
|
|
|
|
<div
|
2020-12-05 04:57:15 +01:00
|
|
|
|
class="note _panel"
|
2020-07-27 06:34:20 +02:00
|
|
|
|
v-if="!muted"
|
2020-07-27 01:46:21 +02:00
|
|
|
|
v-show="!isDeleted"
|
2020-02-06 01:09:24 +01:00
|
|
|
|
:tabindex="!isDeleted ? '-1' : null"
|
2020-01-29 20:37:25 +01:00
|
|
|
|
:class="{ renote: isRenote }"
|
|
|
|
|
v-hotkey="keymap"
|
2020-08-09 08:51:02 +02:00
|
|
|
|
v-size="{ max: [500, 450, 350, 300] }"
|
2020-01-29 20:37:25 +01:00
|
|
|
|
>
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<XSub v-for="note in conversation" class="reply-to-more" :key="note.id" :note="note"/>
|
|
|
|
|
<XSub :note="appearNote.reply" class="reply-to" v-if="appearNote.reply"/>
|
|
|
|
|
<div class="info" v-if="pinned"><Fa :icon="faThumbtack"/> {{ $t('pinnedNote') }}</div>
|
|
|
|
|
<div class="info" v-if="appearNote._prId_"><Fa :icon="faBullhorn"/> {{ $t('promotion') }}<button class="_textButton hide" @click="readPromo()">{{ $t('hideThisNote') }} <Fa :icon="faTimes"/></button></div>
|
|
|
|
|
<div class="info" v-if="appearNote._featuredId_"><Fa :icon="faBolt"/> {{ $t('featured') }}</div>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
<div class="renote" v-if="isRenote">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<MkAvatar class="avatar" :user="note.user"/>
|
|
|
|
|
<Fa :icon="faRetweet"/>
|
|
|
|
|
<i18n-t keypath="renotedBy" tag="span">
|
|
|
|
|
<template #user>
|
2020-10-24 18:21:41 +02:00
|
|
|
|
<MkA class="name" :to="userPage(note.user)" v-user-preview="note.userId">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<MkUserName :user="note.user"/>
|
2020-10-24 18:21:41 +02:00
|
|
|
|
</MkA>
|
2020-10-17 13:12:00 +02:00
|
|
|
|
</template>
|
|
|
|
|
</i18n-t>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
<div class="info">
|
2020-05-15 13:39:11 +02:00
|
|
|
|
<button class="_button time" @click="showRenoteMenu()" ref="renoteTime">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<Fa class="dropdownIcon" v-if="isMyRenote" :icon="faEllipsisH"/>
|
|
|
|
|
<MkTime :time="note.createdAt"/>
|
2020-05-15 13:39:11 +02:00
|
|
|
|
</button>
|
2020-05-10 08:36:42 +02:00
|
|
|
|
<span class="visibility" v-if="note.visibility !== 'public'">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<Fa v-if="note.visibility === 'home'" :icon="faHome"/>
|
|
|
|
|
<Fa v-if="note.visibility === 'followers'" :icon="faUnlock"/>
|
|
|
|
|
<Fa v-if="note.visibility === 'specified'" :icon="faEnvelope"/>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</span>
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<span class="localOnly" v-if="note.localOnly"><Fa :icon="faBiohazard"/></span>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</div>
|
|
|
|
|
</div>
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<article class="article" @contextmenu="onContextmenu">
|
|
|
|
|
<MkAvatar class="avatar" :user="appearNote.user"/>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
<div class="main">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<XNoteHeader class="header" :note="appearNote" :mini="true"/>
|
2020-10-27 08:16:59 +01:00
|
|
|
|
<MkInstanceTicker v-if="showTicker" class="ticker" :instance="appearNote.user.instance"/>
|
2020-11-01 07:04:46 +01:00
|
|
|
|
<div class="body">
|
2020-01-29 20:37:25 +01:00
|
|
|
|
<p v-if="appearNote.cw != null" class="cw">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<Mfm v-if="appearNote.cw != ''" class="text" :text="appearNote.cw" :author="appearNote.user" :i="$store.state.i" :custom-emojis="appearNote.emojis"/>
|
|
|
|
|
<XCwButton v-model:value="showContent" :note="appearNote"/>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</p>
|
|
|
|
|
<div class="content" v-show="appearNote.cw == null || showContent">
|
|
|
|
|
<div class="text">
|
|
|
|
|
<span v-if="appearNote.isHidden" style="opacity: 0.5">({{ $t('private') }})</span>
|
2020-10-24 18:21:41 +02:00
|
|
|
|
<MkA class="reply" v-if="appearNote.replyId" :to="`/notes/${appearNote.replyId}`"><Fa :icon="faReply"/></MkA>
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<Mfm v-if="appearNote.text" :text="appearNote.text" :author="appearNote.user" :i="$store.state.i" :custom-emojis="appearNote.emojis"/>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
<a class="rp" v-if="appearNote.renote != null">RN:</a>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="files" v-if="appearNote.files.length > 0">
|
2020-11-01 07:04:46 +01:00
|
|
|
|
<XMediaList :media-list="appearNote.files"/>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</div>
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<XPoll v-if="appearNote.poll" :note="appearNote" ref="pollViewer" class="poll"/>
|
|
|
|
|
<MkUrlPreview v-for="url in urls" :url="url" :key="url" :compact="true" :detail="detail" class="url-preview"/>
|
|
|
|
|
<div class="renote" v-if="appearNote.renote"><XNotePreview :note="appearNote.renote"/></div>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</div>
|
2020-10-24 18:21:41 +02:00
|
|
|
|
<MkA v-if="appearNote.channel && !inChannel" class="channel" :to="`/channels/${appearNote.channel.id}`"><Fa :icon="faSatelliteDish"/> {{ appearNote.channel.name }}</MkA>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</div>
|
2020-07-27 01:46:21 +02:00
|
|
|
|
<footer class="footer">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<XReactionsViewer :note="appearNote" ref="reactionsViewer"/>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
<button @click="reply()" class="button _button">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<template v-if="appearNote.reply"><Fa :icon="faReplyAll"/></template>
|
|
|
|
|
<template v-else><Fa :icon="faReply"/></template>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
<p class="count" v-if="appearNote.repliesCount > 0">{{ appearNote.repliesCount }}</p>
|
|
|
|
|
</button>
|
2020-02-16 13:10:52 +01:00
|
|
|
|
<button v-if="canRenote" @click="renote()" class="button _button" ref="renoteButton">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<Fa :icon="faRetweet"/><p class="count" v-if="appearNote.renoteCount > 0">{{ appearNote.renoteCount }}</p>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</button>
|
|
|
|
|
<button v-else class="button _button">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<Fa :icon="faBan"/>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</button>
|
2020-07-30 13:28:35 +02:00
|
|
|
|
<button v-if="appearNote.myReaction == null" class="button _button" @click="react()" ref="reactButton">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<Fa :icon="faPlus"/>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</button>
|
2020-07-30 13:28:35 +02:00
|
|
|
|
<button v-if="appearNote.myReaction != null" class="button _button reacted" @click="undoReact(appearNote)" ref="reactButton">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<Fa :icon="faMinus"/>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</button>
|
|
|
|
|
<button class="button _button" @click="menu()" ref="menuButton">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<Fa :icon="faEllipsisH"/>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</button>
|
|
|
|
|
</footer>
|
|
|
|
|
</div>
|
|
|
|
|
</article>
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<XSub v-for="note in replies" :key="note.id" :note="note" class="reply" :detail="true"/>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</div>
|
2020-07-27 06:34:20 +02:00
|
|
|
|
<div v-else class="_panel muted" @click="muted = false">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<i18n-t keypath="userSaysSomething" tag="small">
|
|
|
|
|
<template #name>
|
2020-10-24 18:21:41 +02:00
|
|
|
|
<MkA class="name" :to="userPage(appearNote.user)" v-user-preview="appearNote.userId">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
<MkUserName :user="appearNote.user"/>
|
2020-10-24 18:21:41 +02:00
|
|
|
|
</MkA>
|
2020-10-17 13:12:00 +02:00
|
|
|
|
</template>
|
|
|
|
|
</i18n-t>
|
2020-07-27 06:34:20 +02:00
|
|
|
|
</div>
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<script lang="ts">
|
2020-10-17 13:12:00 +02:00
|
|
|
|
import { computed, defineAsyncComponent, defineComponent, markRaw, ref } from 'vue';
|
2020-11-15 04:04:54 +01:00
|
|
|
|
import { faSatelliteDish, faBolt, faTimes, faBullhorn, faStar, faLink, faExternalLinkSquareAlt, faPlus, faMinus, faRetweet, faReply, faReplyAll, faEllipsisH, faHome, faUnlock, faEnvelope, faThumbtack, faBan, faQuoteRight, faInfoCircle, faBiohazard, faPlug, faExclamationCircle, faPaperclip } from '@fortawesome/free-solid-svg-icons';
|
2020-03-21 10:58:05 +01:00
|
|
|
|
import { faCopy, faTrashAlt, faEdit, faEye, faEyeSlash } from '@fortawesome/free-regular-svg-icons';
|
2020-01-29 20:37:25 +01:00
|
|
|
|
import { parse } from '../../mfm/parse';
|
|
|
|
|
import { sum, unique } from '../../prelude/array';
|
|
|
|
|
import XSub from './note.sub.vue';
|
|
|
|
|
import XNoteHeader from './note-header.vue';
|
|
|
|
|
import XNotePreview from './note-preview.vue';
|
|
|
|
|
import XReactionsViewer from './reactions-viewer.vue';
|
|
|
|
|
import XMediaList from './media-list.vue';
|
|
|
|
|
import XCwButton from './cw-button.vue';
|
|
|
|
|
import XPoll from './poll.vue';
|
2020-10-17 13:12:00 +02:00
|
|
|
|
import { pleaseLogin } from '@/scripts/please-login';
|
|
|
|
|
import { focusPrev, focusNext } from '@/scripts/focus';
|
|
|
|
|
import { url } from '@/config';
|
|
|
|
|
import copyToClipboard from '@/scripts/copy-to-clipboard';
|
|
|
|
|
import { checkWordMute } from '@/scripts/check-word-mute';
|
|
|
|
|
import { userPage } from '@/filters/user';
|
|
|
|
|
import * as os from '@/os';
|
|
|
|
|
import { noteActions, noteViewInterruptors } from '@/store';
|
|
|
|
|
|
|
|
|
|
function markRawAll(...xs) {
|
|
|
|
|
for (const x of xs) {
|
|
|
|
|
markRaw(x);
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-07-27 16:25:37 +02:00
|
|
|
|
|
2020-10-17 13:12:00 +02:00
|
|
|
|
markRawAll(faEdit, faBolt, faTimes, faBullhorn, faPlus, faMinus, faRetweet, faReply, faReplyAll, faEllipsisH, faHome, faUnlock, faEnvelope, faThumbtack, faBan, faBiohazard, faPlug, faSatelliteDish);
|
|
|
|
|
|
|
|
|
|
export default defineComponent({
|
2020-01-29 20:37:25 +01:00
|
|
|
|
components: {
|
|
|
|
|
XSub,
|
|
|
|
|
XNoteHeader,
|
|
|
|
|
XNotePreview,
|
|
|
|
|
XReactionsViewer,
|
|
|
|
|
XMediaList,
|
|
|
|
|
XCwButton,
|
|
|
|
|
XPoll,
|
2020-10-17 13:12:00 +02:00
|
|
|
|
MkUrlPreview: defineAsyncComponent(() => import('@/components/url-preview.vue')),
|
2020-10-27 08:16:59 +01:00
|
|
|
|
MkInstanceTicker: defineAsyncComponent(() => import('@/components/instance-ticker.vue')),
|
2020-01-29 20:37:25 +01:00
|
|
|
|
},
|
|
|
|
|
|
2020-08-18 15:44:21 +02:00
|
|
|
|
inject: {
|
|
|
|
|
inChannel: {
|
|
|
|
|
default: null
|
2020-10-24 18:21:41 +02:00
|
|
|
|
},
|
2020-08-18 15:44:21 +02:00
|
|
|
|
},
|
|
|
|
|
|
2020-01-29 20:37:25 +01:00
|
|
|
|
props: {
|
|
|
|
|
note: {
|
|
|
|
|
type: Object,
|
|
|
|
|
required: true
|
|
|
|
|
},
|
|
|
|
|
detail: {
|
|
|
|
|
type: Boolean,
|
|
|
|
|
required: false,
|
|
|
|
|
default: false
|
|
|
|
|
},
|
|
|
|
|
pinned: {
|
|
|
|
|
type: Boolean,
|
|
|
|
|
required: false,
|
|
|
|
|
default: false
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
|
2020-10-17 13:12:00 +02:00
|
|
|
|
emits: ['update:note'],
|
|
|
|
|
|
2020-01-29 20:37:25 +01:00
|
|
|
|
data() {
|
|
|
|
|
return {
|
|
|
|
|
connection: null,
|
|
|
|
|
conversation: [],
|
|
|
|
|
replies: [],
|
|
|
|
|
showContent: false,
|
2020-07-27 01:46:21 +02:00
|
|
|
|
isDeleted: false,
|
2020-07-27 06:34:20 +02:00
|
|
|
|
muted: false,
|
2020-08-18 15:44:21 +02:00
|
|
|
|
faEdit, faBolt, faTimes, faBullhorn, faPlus, faMinus, faRetweet, faReply, faReplyAll, faEllipsisH, faHome, faUnlock, faEnvelope, faThumbtack, faBan, faBiohazard, faPlug, faSatelliteDish
|
2020-01-29 20:37:25 +01:00
|
|
|
|
};
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
computed: {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
rs() {
|
|
|
|
|
return this.$store.state.settings.reactions;
|
|
|
|
|
},
|
2020-01-29 20:37:25 +01:00
|
|
|
|
keymap(): any {
|
|
|
|
|
return {
|
|
|
|
|
'r': () => this.reply(true),
|
|
|
|
|
'e|a|plus': () => this.react(true),
|
|
|
|
|
'q': () => this.renote(true),
|
|
|
|
|
'f|b': this.favorite,
|
|
|
|
|
'delete|ctrl+d': this.del,
|
|
|
|
|
'ctrl+q': this.renoteDirectly,
|
|
|
|
|
'up|k|shift+tab': this.focusBefore,
|
|
|
|
|
'down|j|tab': this.focusAfter,
|
|
|
|
|
'esc': this.blur,
|
|
|
|
|
'm|o': () => this.menu(true),
|
|
|
|
|
's': this.toggleShowContent,
|
2020-10-17 13:12:00 +02:00
|
|
|
|
'1': () => this.reactDirectly(this.rs[0]),
|
|
|
|
|
'2': () => this.reactDirectly(this.rs[1]),
|
|
|
|
|
'3': () => this.reactDirectly(this.rs[2]),
|
|
|
|
|
'4': () => this.reactDirectly(this.rs[3]),
|
|
|
|
|
'5': () => this.reactDirectly(this.rs[4]),
|
|
|
|
|
'6': () => this.reactDirectly(this.rs[5]),
|
|
|
|
|
'7': () => this.reactDirectly(this.rs[6]),
|
|
|
|
|
'8': () => this.reactDirectly(this.rs[7]),
|
|
|
|
|
'9': () => this.reactDirectly(this.rs[8]),
|
|
|
|
|
'0': () => this.reactDirectly(this.rs[9]),
|
2020-01-29 20:37:25 +01:00
|
|
|
|
};
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
},
|
|
|
|
|
|
2020-05-16 05:15:59 +02:00
|
|
|
|
isMyRenote(): boolean {
|
|
|
|
|
return this.$store.getters.isSignedIn && (this.$store.state.i.id === this.note.userId);
|
|
|
|
|
},
|
|
|
|
|
|
2020-02-16 13:10:52 +01:00
|
|
|
|
canRenote(): boolean {
|
|
|
|
|
return ['public', 'home'].includes(this.appearNote.visibility) || this.isMyNote;
|
|
|
|
|
},
|
|
|
|
|
|
2020-01-29 20:37:25 +01:00
|
|
|
|
reactionsCount(): number {
|
2020-07-27 16:25:37 +02:00
|
|
|
|
return this.appearNote.reactions
|
|
|
|
|
? sum(Object.values(this.appearNote.reactions))
|
|
|
|
|
: 0;
|
2020-01-29 20:37:25 +01:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
urls(): string[] {
|
|
|
|
|
if (this.appearNote.text) {
|
|
|
|
|
const ast = parse(this.appearNote.text);
|
|
|
|
|
// 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;
|
|
|
|
|
}, []);
|
|
|
|
|
} else {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
2020-10-27 08:16:59 +01:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
showTicker() {
|
|
|
|
|
if (this.$store.state.device.instanceTicker === 'always') return true;
|
|
|
|
|
if (this.$store.state.device.instanceTicker === 'remote' && this.appearNote.user.instance) return true;
|
|
|
|
|
return false;
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
2020-07-27 06:34:20 +02:00
|
|
|
|
async created() {
|
2020-01-29 20:37:25 +01:00
|
|
|
|
if (this.$store.getters.isSignedIn) {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
this.connection = os.stream;
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-07-28 18:15:02 +02:00
|
|
|
|
// plugin
|
2020-10-17 13:12:00 +02:00
|
|
|
|
if (noteViewInterruptors.length > 0) {
|
2020-07-28 18:15:02 +02:00
|
|
|
|
let result = this.note;
|
2020-10-17 13:12:00 +02:00
|
|
|
|
for (const interruptor of noteViewInterruptors) {
|
|
|
|
|
result = await interruptor.handler(JSON.parse(JSON.stringify(result)));
|
2020-07-28 18:15:02 +02:00
|
|
|
|
}
|
2020-10-17 13:12:00 +02:00
|
|
|
|
this.$emit('update:note', Object.freeze(result));
|
2020-07-28 18:15:02 +02:00
|
|
|
|
}
|
|
|
|
|
|
2020-07-27 06:34:20 +02:00
|
|
|
|
this.muted = await checkWordMute(this.appearNote, this.$store.state.i, this.$store.state.settings.mutedWords);
|
|
|
|
|
|
2020-01-29 20:37:25 +01:00
|
|
|
|
if (this.detail) {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.api('notes/children', {
|
2020-01-29 20:37:25 +01:00
|
|
|
|
noteId: this.appearNote.id,
|
|
|
|
|
limit: 30
|
|
|
|
|
}).then(replies => {
|
|
|
|
|
this.replies = replies;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (this.appearNote.replyId) {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.api('notes/conversation', {
|
2020-01-29 20:37:25 +01:00
|
|
|
|
noteId: this.appearNote.replyId
|
|
|
|
|
}).then(conversation => {
|
|
|
|
|
this.conversation = conversation.reverse();
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
mounted() {
|
|
|
|
|
this.capture(true);
|
|
|
|
|
|
|
|
|
|
if (this.$store.getters.isSignedIn) {
|
|
|
|
|
this.connection.on('_connected_', this.onStreamConnected);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
2020-10-17 13:12:00 +02:00
|
|
|
|
beforeUnmount() {
|
2020-01-29 20:37:25 +01:00
|
|
|
|
this.decapture(true);
|
|
|
|
|
|
|
|
|
|
if (this.$store.getters.isSignedIn) {
|
|
|
|
|
this.connection.off('_connected_', this.onStreamConnected);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
methods: {
|
2020-07-27 16:25:37 +02:00
|
|
|
|
updateAppearNote(v) {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
this.$emit('update:note', Object.freeze(this.isRenote ? {
|
2020-07-27 16:25:37 +02:00
|
|
|
|
...this.note,
|
|
|
|
|
renote: {
|
|
|
|
|
...this.note.renote,
|
|
|
|
|
...v
|
|
|
|
|
}
|
|
|
|
|
} : {
|
|
|
|
|
...this.note,
|
|
|
|
|
...v
|
|
|
|
|
}));
|
|
|
|
|
},
|
|
|
|
|
|
2020-02-18 10:14:38 +01:00
|
|
|
|
readPromo() {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.api('promo/read', {
|
2020-02-18 10:14:38 +01:00
|
|
|
|
noteId: this.appearNote.id
|
|
|
|
|
});
|
2020-07-27 01:46:21 +02:00
|
|
|
|
this.isDeleted = true;
|
2020-02-18 10:14:38 +01:00
|
|
|
|
},
|
|
|
|
|
|
2020-01-29 20:37:25 +01:00
|
|
|
|
capture(withHandler = false) {
|
|
|
|
|
if (this.$store.getters.isSignedIn) {
|
2020-02-08 19:49:18 +01:00
|
|
|
|
this.connection.send(document.body.contains(this.$el) ? 'sn' : 's', { id: this.appearNote.id });
|
|
|
|
|
if (withHandler) this.connection.on('noteUpdated', this.onStreamNoteUpdated);
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
decapture(withHandler = false) {
|
|
|
|
|
if (this.$store.getters.isSignedIn) {
|
|
|
|
|
this.connection.send('un', {
|
|
|
|
|
id: this.appearNote.id
|
|
|
|
|
});
|
|
|
|
|
if (withHandler) this.connection.off('noteUpdated', this.onStreamNoteUpdated);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
onStreamConnected() {
|
|
|
|
|
this.capture();
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
onStreamNoteUpdated(data) {
|
|
|
|
|
const { type, id, body } = data;
|
|
|
|
|
|
|
|
|
|
if (id !== this.appearNote.id) return;
|
|
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
|
case 'reacted': {
|
|
|
|
|
const reaction = body.reaction;
|
|
|
|
|
|
2020-07-27 16:25:37 +02:00
|
|
|
|
// DeepではなくShallowコピーであることに注意。n.reactions[reaction] = hogeとかしないように(親からもらったデータをミューテートすることになるので)
|
|
|
|
|
let n = {
|
|
|
|
|
...this.appearNote,
|
|
|
|
|
};
|
|
|
|
|
|
2020-04-13 17:42:59 +02:00
|
|
|
|
if (body.emoji) {
|
2020-07-27 16:25:37 +02:00
|
|
|
|
const emojis = this.appearNote.emojis || [];
|
|
|
|
|
if (!emojis.includes(body.emoji)) {
|
|
|
|
|
n.emojis = [...emojis, body.emoji];
|
2020-04-13 17:42:59 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-27 16:25:37 +02:00
|
|
|
|
// TODO: reactionsプロパティがない場合ってあったっけ? なければ || {} は消せる
|
|
|
|
|
const currentCount = (this.appearNote.reactions || {})[reaction] || 0;
|
2020-01-29 20:37:25 +01:00
|
|
|
|
|
|
|
|
|
// Increment the count
|
2020-07-27 16:25:37 +02:00
|
|
|
|
n.reactions = {
|
|
|
|
|
...this.appearNote.reactions,
|
|
|
|
|
[reaction]: currentCount + 1
|
|
|
|
|
};
|
2020-01-29 20:37:25 +01:00
|
|
|
|
|
2020-07-27 01:46:21 +02:00
|
|
|
|
if (body.userId === this.$store.state.i.id) {
|
2020-07-27 16:25:37 +02:00
|
|
|
|
n.myReaction = reaction;
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
2020-07-27 16:25:37 +02:00
|
|
|
|
|
|
|
|
|
this.updateAppearNote(n);
|
2020-01-29 20:37:25 +01:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case 'unreacted': {
|
|
|
|
|
const reaction = body.reaction;
|
|
|
|
|
|
2020-07-27 16:25:37 +02:00
|
|
|
|
// DeepではなくShallowコピーであることに注意。n.reactions[reaction] = hogeとかしないように(親からもらったデータをミューテートすることになるので)
|
|
|
|
|
let n = {
|
|
|
|
|
...this.appearNote,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// TODO: reactionsプロパティがない場合ってあったっけ? なければ || {} は消せる
|
|
|
|
|
const currentCount = (this.appearNote.reactions || {})[reaction] || 0;
|
2020-01-29 20:37:25 +01:00
|
|
|
|
|
|
|
|
|
// Decrement the count
|
2020-07-27 16:25:37 +02:00
|
|
|
|
n.reactions = {
|
|
|
|
|
...this.appearNote.reactions,
|
|
|
|
|
[reaction]: Math.max(0, currentCount - 1)
|
|
|
|
|
};
|
2020-01-29 20:37:25 +01:00
|
|
|
|
|
2020-07-27 01:46:21 +02:00
|
|
|
|
if (body.userId === this.$store.state.i.id) {
|
2020-07-27 16:25:37 +02:00
|
|
|
|
n.myReaction = null;
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
2020-07-27 16:25:37 +02:00
|
|
|
|
|
|
|
|
|
this.updateAppearNote(n);
|
2020-01-29 20:37:25 +01:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case 'pollVoted': {
|
|
|
|
|
const choice = body.choice;
|
2020-07-27 16:25:37 +02:00
|
|
|
|
|
|
|
|
|
// DeepではなくShallowコピーであることに注意。n.reactions[reaction] = hogeとかしないように(親からもらったデータをミューテートすることになるので)
|
|
|
|
|
let n = {
|
|
|
|
|
...this.appearNote,
|
|
|
|
|
};
|
|
|
|
|
|
2020-07-31 12:09:38 +02:00
|
|
|
|
const choices = [...this.appearNote.poll.choices];
|
|
|
|
|
choices[choice] = {
|
|
|
|
|
...choices[choice],
|
|
|
|
|
votes: choices[choice].votes + 1,
|
|
|
|
|
...(body.userId === this.$store.state.i.id ? {
|
|
|
|
|
isVoted: true
|
|
|
|
|
} : {})
|
|
|
|
|
};
|
|
|
|
|
|
2020-07-27 16:25:37 +02:00
|
|
|
|
n.poll = {
|
|
|
|
|
...this.appearNote.poll,
|
2020-07-31 12:09:38 +02:00
|
|
|
|
choices: choices
|
2020-07-27 16:25:37 +02:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
this.updateAppearNote(n);
|
2020-01-29 20:37:25 +01:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case 'deleted': {
|
2020-07-27 01:46:21 +02:00
|
|
|
|
this.isDeleted = true;
|
2020-01-29 20:37:25 +01:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
reply(viaKeyboard = false) {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
pleaseLogin();
|
|
|
|
|
os.post({
|
2020-01-29 20:37:25 +01:00
|
|
|
|
reply: this.appearNote,
|
|
|
|
|
animation: !viaKeyboard,
|
|
|
|
|
}, () => {
|
|
|
|
|
this.focus();
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
2020-02-08 15:52:40 +01:00
|
|
|
|
renote(viaKeyboard = false) {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
pleaseLogin();
|
2020-01-29 20:37:25 +01:00
|
|
|
|
this.blur();
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.modalMenu([{
|
|
|
|
|
text: this.$t('renote'),
|
|
|
|
|
icon: faRetweet,
|
|
|
|
|
action: () => {
|
|
|
|
|
os.api('notes/create', {
|
|
|
|
|
renoteId: this.appearNote.id
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}, {
|
|
|
|
|
text: this.$t('quote'),
|
|
|
|
|
icon: faQuoteRight,
|
|
|
|
|
action: () => {
|
|
|
|
|
os.post({
|
|
|
|
|
renote: this.appearNote,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}], this.$refs.renoteButton, {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
viaKeyboard
|
2020-02-22 18:34:54 +01:00
|
|
|
|
});
|
2020-01-29 20:37:25 +01:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
renoteDirectly() {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.api('notes/create', {
|
2020-01-29 20:37:25 +01:00
|
|
|
|
renoteId: this.appearNote.id
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
react(viaKeyboard = false) {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
pleaseLogin();
|
2020-01-29 20:37:25 +01:00
|
|
|
|
this.blur();
|
2020-11-14 03:47:30 +01:00
|
|
|
|
os.popup(import('@/components/emoji-picker.vue'), {
|
|
|
|
|
src: this.$refs.reactButton,
|
2020-11-18 04:09:14 +01:00
|
|
|
|
asReactionPicker: true
|
2020-11-14 03:47:30 +01:00
|
|
|
|
}, {
|
|
|
|
|
done: reaction => {
|
|
|
|
|
if (reaction) {
|
|
|
|
|
os.api('notes/reactions/create', {
|
|
|
|
|
noteId: this.appearNote.id,
|
|
|
|
|
reaction: reaction
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
this.focus();
|
|
|
|
|
},
|
|
|
|
|
}, 'closed');
|
2020-01-29 20:37:25 +01:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
reactDirectly(reaction) {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.api('notes/reactions/create', {
|
2020-01-29 20:37:25 +01:00
|
|
|
|
noteId: this.appearNote.id,
|
|
|
|
|
reaction: reaction
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
2020-07-27 16:25:37 +02:00
|
|
|
|
undoReact(note) {
|
|
|
|
|
const oldReaction = note.myReaction;
|
2020-01-29 20:37:25 +01:00
|
|
|
|
if (!oldReaction) return;
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.api('notes/reactions/delete', {
|
2020-07-27 16:25:37 +02:00
|
|
|
|
noteId: note.id
|
2020-01-29 20:37:25 +01:00
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
favorite() {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
pleaseLogin();
|
|
|
|
|
os.apiWithDialog('notes/favorites/create', {
|
2020-01-29 20:37:25 +01:00
|
|
|
|
noteId: this.appearNote.id
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
del() {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.dialog({
|
2020-01-29 20:37:25 +01:00
|
|
|
|
type: 'warning',
|
|
|
|
|
text: this.$t('noteDeleteConfirm'),
|
|
|
|
|
showCancelButton: true
|
|
|
|
|
}).then(({ canceled }) => {
|
|
|
|
|
if (canceled) return;
|
|
|
|
|
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.api('notes/delete', {
|
2020-01-29 20:37:25 +01:00
|
|
|
|
noteId: this.appearNote.id
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
2020-03-21 10:58:05 +01:00
|
|
|
|
delEdit() {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.dialog({
|
2020-03-21 10:58:05 +01:00
|
|
|
|
type: 'warning',
|
|
|
|
|
text: this.$t('deleteAndEditConfirm'),
|
|
|
|
|
showCancelButton: true
|
|
|
|
|
}).then(({ canceled }) => {
|
|
|
|
|
if (canceled) return;
|
|
|
|
|
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.api('notes/delete', {
|
2020-03-21 10:58:05 +01:00
|
|
|
|
noteId: this.appearNote.id
|
|
|
|
|
});
|
|
|
|
|
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.post({ initialNote: this.appearNote, renote: this.appearNote.renote, reply: this.appearNote.reply, channel: this.appearNote.channel });
|
2020-03-21 10:58:05 +01:00
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
2020-02-08 15:52:40 +01:00
|
|
|
|
toggleFavorite(favorite: boolean) {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.apiWithDialog(favorite ? 'notes/favorites/create' : 'notes/favorites/delete', {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
noteId: this.appearNote.id
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
toggleWatch(watch: boolean) {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.apiWithDialog(watch ? 'notes/watching/create' : 'notes/watching/delete', {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
noteId: this.appearNote.id
|
2020-01-29 20:37:25 +01:00
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
2020-10-17 13:12:00 +02:00
|
|
|
|
getMenu() {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
let menu;
|
|
|
|
|
if (this.$store.getters.isSignedIn) {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
const statePromise = os.api('notes/state', {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
noteId: this.appearNote.id
|
|
|
|
|
});
|
2020-10-17 13:12:00 +02:00
|
|
|
|
|
2020-02-08 15:52:40 +01:00
|
|
|
|
menu = [{
|
|
|
|
|
icon: faCopy,
|
|
|
|
|
text: this.$t('copyContent'),
|
|
|
|
|
action: this.copyContent
|
|
|
|
|
}, {
|
|
|
|
|
icon: faLink,
|
|
|
|
|
text: this.$t('copyLink'),
|
|
|
|
|
action: this.copyLink
|
2020-04-02 14:59:14 +02:00
|
|
|
|
}, (this.appearNote.url || this.appearNote.uri) ? {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
icon: faExternalLinkSquareAlt,
|
|
|
|
|
text: this.$t('showOnRemote'),
|
|
|
|
|
action: () => {
|
2020-04-02 14:59:14 +02:00
|
|
|
|
window.open(this.appearNote.url || this.appearNote.uri, '_blank');
|
2020-02-08 15:52:40 +01:00
|
|
|
|
}
|
|
|
|
|
} : undefined,
|
|
|
|
|
null,
|
2020-10-17 13:12:00 +02:00
|
|
|
|
statePromise.then(state => state.isFavorited ? {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
icon: faStar,
|
|
|
|
|
text: this.$t('unfavorite'),
|
|
|
|
|
action: () => this.toggleFavorite(false)
|
|
|
|
|
} : {
|
|
|
|
|
icon: faStar,
|
|
|
|
|
text: this.$t('favorite'),
|
|
|
|
|
action: () => this.toggleFavorite(true)
|
2020-10-17 13:12:00 +02:00
|
|
|
|
}),
|
2020-11-15 04:04:54 +01:00
|
|
|
|
{
|
|
|
|
|
icon: faPaperclip,
|
|
|
|
|
text: this.$t('clip'),
|
|
|
|
|
action: () => this.clip()
|
|
|
|
|
},
|
2020-10-17 13:12:00 +02:00
|
|
|
|
(this.appearNote.userId != this.$store.state.i.id) ? statePromise.then(state => state.isWatching ? {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
icon: faEyeSlash,
|
|
|
|
|
text: this.$t('unwatch'),
|
|
|
|
|
action: () => this.toggleWatch(false)
|
|
|
|
|
} : {
|
|
|
|
|
icon: faEye,
|
|
|
|
|
text: this.$t('watch'),
|
|
|
|
|
action: () => this.toggleWatch(true)
|
2020-10-17 13:12:00 +02:00
|
|
|
|
}) : undefined,
|
2020-02-08 15:52:40 +01:00
|
|
|
|
this.appearNote.userId == this.$store.state.i.id ? (this.$store.state.i.pinnedNoteIds || []).includes(this.appearNote.id) ? {
|
|
|
|
|
icon: faThumbtack,
|
|
|
|
|
text: this.$t('unpin'),
|
|
|
|
|
action: () => this.togglePin(false)
|
|
|
|
|
} : {
|
|
|
|
|
icon: faThumbtack,
|
|
|
|
|
text: this.$t('pin'),
|
|
|
|
|
action: () => this.togglePin(true)
|
|
|
|
|
} : undefined,
|
2020-02-18 00:41:32 +01:00
|
|
|
|
...(this.$store.state.i.isModerator || this.$store.state.i.isAdmin ? [
|
|
|
|
|
null,
|
|
|
|
|
{
|
|
|
|
|
icon: faBullhorn,
|
|
|
|
|
text: this.$t('promote'),
|
|
|
|
|
action: this.promote
|
|
|
|
|
}]
|
|
|
|
|
: []
|
|
|
|
|
),
|
2020-10-19 12:29:04 +02:00
|
|
|
|
...(this.appearNote.userId != this.$store.state.i.id ? [
|
|
|
|
|
null,
|
|
|
|
|
{
|
|
|
|
|
icon: faExclamationCircle,
|
|
|
|
|
text: this.$t('reportAbuse'),
|
|
|
|
|
action: () => {
|
|
|
|
|
const u = `${url}/notes/${this.appearNote.id}`;
|
2020-11-03 07:22:55 +01:00
|
|
|
|
os.popup(import('@/components/abuse-report-window.vue'), {
|
2020-10-19 12:29:04 +02:00
|
|
|
|
user: this.appearNote.user,
|
|
|
|
|
initialComment: `Note: ${u}\n-----\n`
|
|
|
|
|
}, {}, 'closed');
|
|
|
|
|
}
|
|
|
|
|
}]
|
|
|
|
|
: []
|
|
|
|
|
),
|
2020-04-13 16:27:12 +02:00
|
|
|
|
...(this.appearNote.userId == this.$store.state.i.id || this.$store.state.i.isModerator || this.$store.state.i.isAdmin ? [
|
2020-02-08 15:52:40 +01:00
|
|
|
|
null,
|
2020-04-13 16:27:12 +02:00
|
|
|
|
this.appearNote.userId == this.$store.state.i.id ? {
|
2020-03-21 10:58:05 +01:00
|
|
|
|
icon: faEdit,
|
|
|
|
|
text: this.$t('deleteAndEdit'),
|
|
|
|
|
action: this.delEdit
|
2020-04-13 16:27:12 +02:00
|
|
|
|
} : undefined,
|
2020-02-08 15:52:40 +01:00
|
|
|
|
{
|
|
|
|
|
icon: faTrashAlt,
|
|
|
|
|
text: this.$t('delete'),
|
2020-10-17 13:12:00 +02:00
|
|
|
|
danger: true,
|
2020-02-08 15:52:40 +01:00
|
|
|
|
action: this.del
|
|
|
|
|
}]
|
|
|
|
|
: []
|
|
|
|
|
)]
|
|
|
|
|
.filter(x => x !== undefined);
|
|
|
|
|
} else {
|
|
|
|
|
menu = [{
|
|
|
|
|
icon: faCopy,
|
|
|
|
|
text: this.$t('copyContent'),
|
|
|
|
|
action: this.copyContent
|
|
|
|
|
}, {
|
|
|
|
|
icon: faLink,
|
|
|
|
|
text: this.$t('copyLink'),
|
|
|
|
|
action: this.copyLink
|
2020-04-02 14:59:14 +02:00
|
|
|
|
}, (this.appearNote.url || this.appearNote.uri) ? {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
icon: faExternalLinkSquareAlt,
|
|
|
|
|
text: this.$t('showOnRemote'),
|
|
|
|
|
action: () => {
|
2020-04-02 14:59:14 +02:00
|
|
|
|
window.open(this.appearNote.url || this.appearNote.uri, '_blank');
|
2020-02-08 15:52:40 +01:00
|
|
|
|
}
|
|
|
|
|
} : undefined]
|
|
|
|
|
.filter(x => x !== undefined);
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-17 13:12:00 +02:00
|
|
|
|
if (noteActions.length > 0) {
|
|
|
|
|
menu = menu.concat([null, ...noteActions.map(action => ({
|
2020-07-11 17:38:55 +02:00
|
|
|
|
icon: faPlug,
|
|
|
|
|
text: action.title,
|
|
|
|
|
action: () => {
|
|
|
|
|
action.handler(this.appearNote);
|
|
|
|
|
}
|
|
|
|
|
}))]);
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-17 13:12:00 +02:00
|
|
|
|
return menu;
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
onContextmenu(e) {
|
|
|
|
|
const isLink = (el: HTMLElement) => {
|
|
|
|
|
if (el.tagName === 'A') return true;
|
|
|
|
|
if (el.parentElement) {
|
|
|
|
|
return isLink(el.parentElement);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
if (isLink(e.target)) return;
|
|
|
|
|
if (window.getSelection().toString() !== '') return;
|
|
|
|
|
os.contextMenu(this.getMenu(), e).then(this.focus);
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
menu(viaKeyboard = false) {
|
|
|
|
|
os.modalMenu(this.getMenu(), this.$refs.menuButton, {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
viaKeyboard
|
|
|
|
|
}).then(this.focus);
|
|
|
|
|
},
|
|
|
|
|
|
2020-02-09 10:52:53 +01:00
|
|
|
|
showRenoteMenu(viaKeyboard = false) {
|
2020-05-16 05:15:59 +02:00
|
|
|
|
if (!this.isMyRenote) return;
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.modalMenu([{
|
|
|
|
|
text: this.$t('unrenote'),
|
|
|
|
|
icon: faTrashAlt,
|
2020-10-18 03:38:35 +02:00
|
|
|
|
danger: true,
|
2020-10-17 13:12:00 +02:00
|
|
|
|
action: () => {
|
|
|
|
|
os.api('notes/delete', {
|
|
|
|
|
noteId: this.note.id
|
|
|
|
|
});
|
|
|
|
|
this.isDeleted = true;
|
|
|
|
|
}
|
|
|
|
|
}], this.$refs.renoteTime, {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
viaKeyboard: viaKeyboard
|
2020-02-06 01:09:24 +01:00
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
2020-01-29 20:37:25 +01:00
|
|
|
|
toggleShowContent() {
|
|
|
|
|
this.showContent = !this.showContent;
|
|
|
|
|
},
|
|
|
|
|
|
2020-02-08 15:52:40 +01:00
|
|
|
|
copyContent() {
|
|
|
|
|
copyToClipboard(this.appearNote.text);
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.success();
|
2020-02-08 15:52:40 +01:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
copyLink() {
|
|
|
|
|
copyToClipboard(`${url}/notes/${this.appearNote.id}`);
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.success();
|
2020-02-08 15:52:40 +01:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
togglePin(pin: boolean) {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.apiWithDialog(pin ? 'i/pin' : 'i/unpin', {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
noteId: this.appearNote.id
|
2020-10-17 13:12:00 +02:00
|
|
|
|
}, undefined, null, e => {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
if (e.id === '72dab508-c64d-498f-8740-a8eec1ba385a') {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.dialog({
|
2020-02-08 15:52:40 +01:00
|
|
|
|
type: 'error',
|
|
|
|
|
text: this.$t('pinLimitExceeded')
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
2020-11-15 04:04:54 +01:00
|
|
|
|
async clip() {
|
|
|
|
|
const clips = await os.api('clips/list');
|
|
|
|
|
os.modalMenu([{
|
|
|
|
|
icon: faPlus,
|
|
|
|
|
text: this.$t('createNew'),
|
|
|
|
|
action: async () => {
|
|
|
|
|
const { canceled, result } = await os.form(this.$t('createNewClip'), {
|
|
|
|
|
name: {
|
|
|
|
|
type: 'string',
|
|
|
|
|
label: this.$t('name')
|
|
|
|
|
},
|
|
|
|
|
description: {
|
|
|
|
|
type: 'string',
|
|
|
|
|
required: false,
|
|
|
|
|
multiline: true,
|
|
|
|
|
label: this.$t('description')
|
|
|
|
|
},
|
|
|
|
|
isPublic: {
|
|
|
|
|
type: 'boolean',
|
2020-11-15 06:18:25 +01:00
|
|
|
|
label: this.$t('public'),
|
|
|
|
|
default: false
|
2020-11-15 04:04:54 +01:00
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
if (canceled) return;
|
|
|
|
|
|
|
|
|
|
const clip = await os.apiWithDialog('clips/create', result);
|
|
|
|
|
|
|
|
|
|
os.apiWithDialog('clips/add-note', { clipId: clip.id, noteId: this.appearNote.id });
|
|
|
|
|
}
|
|
|
|
|
}, null, ...clips.map(clip => ({
|
|
|
|
|
text: clip.name,
|
|
|
|
|
action: () => {
|
|
|
|
|
os.apiWithDialog('clips/add-note', { clipId: clip.id, noteId: this.appearNote.id });
|
|
|
|
|
}
|
|
|
|
|
}))], this.$refs.menuButton, {
|
|
|
|
|
}).then(this.focus);
|
|
|
|
|
},
|
|
|
|
|
|
2020-02-18 00:41:32 +01:00
|
|
|
|
async promote() {
|
2020-10-17 13:12:00 +02:00
|
|
|
|
const { canceled, result: days } = await os.dialog({
|
2020-02-18 00:41:32 +01:00
|
|
|
|
title: this.$t('numberOfDays'),
|
|
|
|
|
input: { type: 'number' }
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (canceled) return;
|
|
|
|
|
|
2020-10-17 13:12:00 +02:00
|
|
|
|
os.apiWithDialog('admin/promo/create', {
|
2020-02-18 00:41:32 +01:00
|
|
|
|
noteId: this.appearNote.id,
|
|
|
|
|
expiresAt: Date.now() + (86400000 * days)
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
2020-01-29 20:37:25 +01:00
|
|
|
|
focus() {
|
|
|
|
|
this.$el.focus();
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
blur() {
|
|
|
|
|
this.$el.blur();
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
focusBefore() {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
focusPrev(this.$el);
|
2020-01-29 20:37:25 +01:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
focusAfter() {
|
2020-02-08 15:52:40 +01:00
|
|
|
|
focusNext(this.$el);
|
2020-10-17 13:12:00 +02:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
userPage
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
|
.note {
|
|
|
|
|
position: relative;
|
|
|
|
|
transition: box-shadow 0.1s ease;
|
2020-03-06 16:31:48 +01:00
|
|
|
|
overflow: hidden;
|
2020-10-17 13:12:00 +02:00
|
|
|
|
contain: content;
|
2020-01-29 20:37:25 +01:00
|
|
|
|
|
|
|
|
|
&:focus {
|
|
|
|
|
outline: none;
|
2020-10-17 13:12:00 +02:00
|
|
|
|
|
|
|
|
|
&:after {
|
|
|
|
|
content: "";
|
|
|
|
|
pointer-events: none;
|
|
|
|
|
display: block;
|
|
|
|
|
position: absolute;
|
|
|
|
|
z-index: 10;
|
|
|
|
|
top: 0;
|
|
|
|
|
left: 0;
|
|
|
|
|
right: 0;
|
|
|
|
|
bottom: 0;
|
|
|
|
|
margin: auto;
|
|
|
|
|
width: calc(100% - 8px);
|
|
|
|
|
height: calc(100% - 8px);
|
|
|
|
|
border: dashed 1px var(--focus);
|
|
|
|
|
border-radius: var(--radius);
|
|
|
|
|
box-sizing: border-box;
|
|
|
|
|
}
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
&:hover > .article > .main > .footer > .button {
|
|
|
|
|
opacity: 1;
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-18 10:14:38 +01:00
|
|
|
|
> .info {
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
2020-01-29 20:37:25 +01:00
|
|
|
|
padding: 16px 32px 8px 32px;
|
|
|
|
|
line-height: 24px;
|
|
|
|
|
font-size: 90%;
|
|
|
|
|
white-space: pre;
|
|
|
|
|
color: #d28a3f;
|
|
|
|
|
|
|
|
|
|
> [data-icon] {
|
|
|
|
|
margin-right: 4px;
|
|
|
|
|
}
|
2020-02-18 10:14:38 +01:00
|
|
|
|
|
|
|
|
|
> .hide {
|
|
|
|
|
margin-left: auto;
|
|
|
|
|
color: inherit;
|
|
|
|
|
}
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-02-18 10:14:38 +01:00
|
|
|
|
> .info + .article {
|
2020-01-29 20:37:25 +01:00
|
|
|
|
padding-top: 8px;
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-20 11:19:28 +01:00
|
|
|
|
> .reply-to {
|
|
|
|
|
opacity: 0.7;
|
|
|
|
|
padding-bottom: 0;
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-10 10:37:11 +02:00
|
|
|
|
> .reply-to-more {
|
|
|
|
|
opacity: 0.7;
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-29 20:37:25 +01:00
|
|
|
|
> .renote {
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
padding: 16px 32px 8px 32px;
|
|
|
|
|
line-height: 28px;
|
|
|
|
|
white-space: pre;
|
|
|
|
|
color: var(--renote);
|
|
|
|
|
|
|
|
|
|
> .avatar {
|
|
|
|
|
flex-shrink: 0;
|
|
|
|
|
display: inline-block;
|
|
|
|
|
width: 28px;
|
|
|
|
|
height: 28px;
|
|
|
|
|
margin: 0 8px 0 0;
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> [data-icon] {
|
|
|
|
|
margin-right: 4px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> span {
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
flex-shrink: 1;
|
|
|
|
|
text-overflow: ellipsis;
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
|
|
|
|
|
> .name {
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .info {
|
|
|
|
|
margin-left: auto;
|
|
|
|
|
font-size: 0.9em;
|
|
|
|
|
|
2020-02-06 01:09:24 +01:00
|
|
|
|
> .time {
|
2020-01-29 20:37:25 +01:00
|
|
|
|
flex-shrink: 0;
|
2020-02-06 01:09:24 +01:00
|
|
|
|
color: inherit;
|
2020-05-15 13:39:11 +02:00
|
|
|
|
|
|
|
|
|
> .dropdownIcon {
|
|
|
|
|
margin-right: 4px;
|
|
|
|
|
}
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .visibility {
|
|
|
|
|
margin-left: 8px;
|
2020-05-16 15:06:39 +02:00
|
|
|
|
}
|
2020-01-29 20:37:25 +01:00
|
|
|
|
|
2020-05-16 15:06:39 +02:00
|
|
|
|
> .localOnly {
|
|
|
|
|
margin-left: 8px;
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .renote + .article {
|
|
|
|
|
padding-top: 8px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .article {
|
|
|
|
|
display: flex;
|
|
|
|
|
padding: 28px 32px 18px;
|
|
|
|
|
|
|
|
|
|
> .avatar {
|
|
|
|
|
flex-shrink: 0;
|
|
|
|
|
display: block;
|
|
|
|
|
//position: sticky;
|
|
|
|
|
//top: 72px;
|
|
|
|
|
margin: 0 14px 8px 0;
|
|
|
|
|
width: 58px;
|
|
|
|
|
height: 58px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .main {
|
|
|
|
|
flex: 1;
|
|
|
|
|
min-width: 0;
|
|
|
|
|
|
|
|
|
|
> .body {
|
|
|
|
|
> .cw {
|
|
|
|
|
cursor: default;
|
|
|
|
|
display: block;
|
|
|
|
|
margin: 0;
|
|
|
|
|
padding: 0;
|
|
|
|
|
overflow-wrap: break-word;
|
|
|
|
|
|
|
|
|
|
> .text {
|
|
|
|
|
margin-right: 8px;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .content {
|
|
|
|
|
> .text {
|
|
|
|
|
overflow-wrap: break-word;
|
|
|
|
|
|
|
|
|
|
> .reply {
|
|
|
|
|
color: var(--accent);
|
|
|
|
|
margin-right: 0.5em;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .rp {
|
|
|
|
|
margin-left: 4px;
|
|
|
|
|
font-style: oblique;
|
|
|
|
|
color: var(--renote);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .url-preview {
|
|
|
|
|
margin-top: 8px;
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-10 10:06:20 +02:00
|
|
|
|
> .poll {
|
2020-01-29 20:37:25 +01:00
|
|
|
|
font-size: 80%;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .renote {
|
|
|
|
|
padding: 8px 0;
|
|
|
|
|
|
|
|
|
|
> * {
|
|
|
|
|
padding: 16px;
|
|
|
|
|
border: dashed 1px var(--renote);
|
|
|
|
|
border-radius: 8px;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-08-18 15:44:21 +02:00
|
|
|
|
|
|
|
|
|
> .channel {
|
|
|
|
|
opacity: 0.7;
|
|
|
|
|
font-size: 80%;
|
|
|
|
|
}
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .footer {
|
|
|
|
|
> .button {
|
|
|
|
|
margin: 0;
|
|
|
|
|
padding: 8px;
|
|
|
|
|
opacity: 0.7;
|
|
|
|
|
|
|
|
|
|
&:not(:last-child) {
|
|
|
|
|
margin-right: 28px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
&:hover {
|
2020-02-14 17:18:48 +01:00
|
|
|
|
color: var(--fgHighlighted);
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .count {
|
|
|
|
|
display: inline;
|
|
|
|
|
margin: 0 0 0 8px;
|
|
|
|
|
opacity: 0.7;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
&.reacted {
|
|
|
|
|
color: var(--accent);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-03-21 04:32:40 +01:00
|
|
|
|
|
|
|
|
|
> .reply {
|
|
|
|
|
border-top: solid 1px var(--divider);
|
|
|
|
|
}
|
2020-07-11 03:13:11 +02:00
|
|
|
|
|
|
|
|
|
&.max-width_500px {
|
|
|
|
|
font-size: 0.9em;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
&.max-width_450px {
|
|
|
|
|
> .renote {
|
|
|
|
|
padding: 8px 16px 0 16px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .info {
|
|
|
|
|
padding: 8px 16px 0 16px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .article {
|
|
|
|
|
padding: 14px 16px 9px;
|
|
|
|
|
|
|
|
|
|
> .avatar {
|
|
|
|
|
margin: 0 10px 8px 0;
|
|
|
|
|
width: 50px;
|
|
|
|
|
height: 50px;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
&.max-width_350px {
|
|
|
|
|
> .article {
|
|
|
|
|
> .main {
|
|
|
|
|
> .footer {
|
|
|
|
|
> .button {
|
|
|
|
|
&:not(:last-child) {
|
|
|
|
|
margin-right: 18px;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
&.max-width_300px {
|
|
|
|
|
font-size: 0.825em;
|
|
|
|
|
|
|
|
|
|
> .article {
|
|
|
|
|
> .avatar {
|
|
|
|
|
width: 44px;
|
|
|
|
|
height: 44px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> .main {
|
|
|
|
|
> .footer {
|
|
|
|
|
> .button {
|
|
|
|
|
&:not(:last-child) {
|
|
|
|
|
margin-right: 12px;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-01-29 20:37:25 +01:00
|
|
|
|
}
|
2020-07-27 06:34:20 +02:00
|
|
|
|
|
|
|
|
|
.muted {
|
|
|
|
|
padding: 8px;
|
|
|
|
|
text-align: center;
|
|
|
|
|
opacity: 0.7;
|
|
|
|
|
}
|
2020-01-29 20:37:25 +01:00
|
|
|
|
</style>
|