iceshrimp/packages/client/src/components/MkRenoteButton.vue

205 lines
4.9 KiB
Vue
Raw Normal View History

<template>
2022-07-24 15:45:16 +09:00
<button
v-if="canRenote"
2021-11-19 19:36:12 +09:00
ref="buttonRef"
2022-11-25 20:46:10 +09:00
v-tooltip.noDelay.bottom="i18n.ts.renote"
2022-11-27 14:37:23 +09:00
class="eddddedb _button canRenote"
2022-10-26 10:45:24 +09:00
@click="renote(false, $event)"
>
2023-03-12 06:01:04 +09:00
<i class="ph-repeat ph-bold ph-lg"></i>
2021-11-19 19:36:12 +09:00
<p v-if="count > 0" class="count">{{ count }}</p>
</button>
2021-11-12 23:54:02 +09:00
<button v-else class="eddddedb _button">
2023-03-12 06:01:04 +09:00
<i class="ph-prohibit ph-bold ph-lg"></i>
</button>
</template>
2022-07-24 15:45:16 +09:00
<script lang="ts" setup>
import { computed, ref } from 'vue';
2022-12-12 12:24:12 +09:00
import type * as misskey from 'calckey-js';
2022-10-26 03:23:59 +09:00
import Ripple from '@/components/MkRipple.vue';
import XDetails from '@/components/MkUsersTooltip.vue';
import { pleaseLogin } from '@/scripts/please-login';
import * as os from '@/os';
import { $i } from '@/account';
import { useTooltip } from '@/scripts/use-tooltip';
import { i18n } from '@/i18n';
2022-11-27 14:37:23 +09:00
import { defaultStore } from '@/store';
2022-07-24 15:45:16 +09:00
const props = defineProps<{
note: misskey.entities.Note;
count: number;
}>();
2022-07-24 15:45:16 +09:00
const buttonRef = ref<HTMLElement>();
2022-07-24 15:45:16 +09:00
const canRenote = computed(() => ['public', 'home'].includes(props.note.visibility) || props.note.userId === $i.id);
2022-07-24 15:45:16 +09:00
useTooltip(buttonRef, async (showing) => {
const renotes = await os.api('notes/renotes', {
noteId: props.note.id,
limit: 11,
});
2022-07-24 15:45:16 +09:00
const users = renotes.map(x => x.user);
2022-07-24 15:45:16 +09:00
if (users.length < 1) return;
2022-07-24 15:45:16 +09:00
os.popup(XDetails, {
showing,
users,
count: props.count,
targetElement: buttonRef.value,
}, {}, 'closed');
});
2022-12-02 16:19:37 +09:00
const renote = async (viaKeyboard = false, ev?: MouseEvent) => {
2022-07-24 15:45:16 +09:00
pleaseLogin();
const renotes = await os.api('notes/renotes', {
noteId: props.note.id,
limit: 11,
});
const users = renotes.map(x => x.user.id);
const hasRenotedBefore = users.includes($i.id);
let buttonActions = [];
if (props.note.visibility === 'public') {
buttonActions.push({
text: i18n.ts.renote,
textStyle: 'font-weight: bold',
icon: 'ph-repeat ph-bold ph-lg',
danger: false,
action: () => {
os.api('notes/create', {
renoteId: props.note.id,
visibility: 'public',
});
const el = ev && (ev.currentTarget ?? ev.target) as HTMLElement | null | undefined;
if (el) {
const rect = el.getBoundingClientRect();
const x = rect.left + (el.offsetWidth / 2);
const y = rect.top + (el.offsetHeight / 2);
os.popup(Ripple, { x, y }, {}, 'end');
}
},
});
}
if (['public', 'home'].includes(props.note.visibility)) {
buttonActions.push({
text: i18n.ts.renoteAsUnlisted,
icons: ['ph-repeat ph-bold ph-lg', 'ph-house ph-bold ph-lg'],
danger: false,
action: () => {
os.api('notes/create', {
renoteId: props.note.id,
visibility: 'home',
});
const el = ev && (ev.currentTarget ?? ev.target) as HTMLElement | null | undefined;
if (el) {
const rect = el.getBoundingClientRect();
const x = rect.left + (el.offsetWidth / 2);
const y = rect.top + (el.offsetHeight / 2);
os.popup(Ripple, { x, y }, {}, 'end');
}
},
});
}
if (props.note.visibility === 'specified') {
buttonActions.push({
text: i18n.ts.renoteToRecipients,
icons: ['ph-repeat ph-bold ph-lg', 'ph-envelope-simple-open ph-bold ph-lg'],
danger: false,
action: () => {
os.api('notes/create', {
renoteId: props.note.id,
visibility: 'specified',
visibleUserIds: props.note.visibleUserIds,
});
const el = ev && (ev.currentTarget ?? ev.target) as HTMLElement | null | undefined;
if (el) {
const rect = el.getBoundingClientRect();
const x = rect.left + (el.offsetWidth / 2);
const y = rect.top + (el.offsetHeight / 2);
os.popup(Ripple, { x, y }, {}, 'end');
}
},
});
} else {
buttonActions.push({
text: i18n.ts.renoteToFollowers,
icons: ['ph-repeat ph-bold ph-lg', 'ph-lock-simple-open ph-bold ph-lg'],
danger: false,
action: () => {
os.api('notes/create', {
renoteId: props.note.id,
visibility: 'followers',
});
const el = ev && (ev.currentTarget ?? ev.target) as HTMLElement | null | undefined;
if (el) {
const rect = el.getBoundingClientRect();
const x = rect.left + (el.offsetWidth / 2);
const y = rect.top + (el.offsetHeight / 2);
os.popup(Ripple, { x, y }, {}, 'end');
}
},
});
}
2022-12-02 16:19:37 +09:00
if (!defaultStore.state.seperateRenoteQuote) {
buttonActions.push({
2022-09-29 01:22:03 +09:00
text: i18n.ts.quote,
2023-03-12 06:01:04 +09:00
icon: 'ph-quotes ph-bold ph-lg',
2022-12-02 16:28:16 +09:00
danger: false,
2022-09-29 01:22:03 +09:00
action: () => {
os.post({
renote: props.note,
});
},
});
}
if (hasRenotedBefore) {
buttonActions.push({
text: i18n.ts.unrenote,
2023-03-12 06:01:04 +09:00
icon: 'ph-trash ph-bold ph-lg',
danger: true,
action: () => {
os.api('notes/unrenote', {
noteId: props.note.id,
});
},
});
}
os.popupMenu(buttonActions, buttonRef.value, { viaKeyboard });
2022-07-24 15:45:16 +09:00
};
</script>
<style lang="scss" scoped>
.eddddedb {
display: inline-block;
height: 32px;
margin: 2px;
padding: 0 6px;
border-radius: 4px;
&:not(.canRenote) {
cursor: default;
}
&.renoted {
background: var(--accent);
}
> .count {
display: inline;
margin-left: 8px;
opacity: 0.7;
}
}
</style>