mirror of
https://github.com/paricafe/misskey.git
synced 2025-03-23 11:39:25 -05:00
frontend: reset reactions viewer
This commit is contained in:
parent
0199f38379
commit
dab6cf09d5
3 changed files with 32 additions and 99 deletions
packages/frontend/src/components
|
@ -41,11 +41,9 @@ const emit = defineEmits<{
|
|||
}>();
|
||||
|
||||
function getReactionName(reaction: string): string {
|
||||
if (reaction.startsWith(':')) {
|
||||
const match = reaction.match(/^:([^@]+)(?:@[^:]+)?:$/);
|
||||
if (match) {
|
||||
return `:${match[1]}:`;
|
||||
}
|
||||
const trimLocal = reaction.replace('@.', '');
|
||||
if (trimLocal.startsWith(':')) {
|
||||
return trimLocal;
|
||||
}
|
||||
return getEmojiName(reaction);
|
||||
}
|
||||
|
|
|
@ -8,7 +8,7 @@ SPDX-License-Identifier: AGPL-3.0-only
|
|||
ref="buttonEl"
|
||||
v-ripple="canToggle"
|
||||
class="_button"
|
||||
:class="[$style.root, { [$style.reacted]: isReacted, [$style.canToggle]: canToggle, [$style.small]: prefer.s.reactionsDisplaySize === 'small', [$style.large]: prefer.s.reactionsDisplaySize === 'large' }]"
|
||||
:class="[$style.root, { [$style.reacted]: note.myReaction == reaction, [$style.canToggle]: canToggle, [$style.small]: prefer.s.reactionsDisplaySize === 'small', [$style.large]: prefer.s.reactionsDisplaySize === 'large' }]"
|
||||
@click.stop="toggleReaction()"
|
||||
@contextmenu.prevent.stop="menu"
|
||||
>
|
||||
|
@ -26,7 +26,7 @@ import XDetails from '@/components/MkReactionsViewer.details.vue';
|
|||
import MkReactionIcon from '@/components/MkReactionIcon.vue';
|
||||
import * as os from '@/os.js';
|
||||
import { misskeyApi, misskeyApiGet } from '@/utility/misskey-api.js';
|
||||
import { useTooltip } from '@/utility/use-tooltip.js';
|
||||
import { useTooltip } from '@/use/use-tooltip.js';
|
||||
import { $i } from '@/account.js';
|
||||
import MkReactionEffect from '@/components/MkReactionEffect.vue';
|
||||
import { claimAchievement } from '@/utility/achievements.js';
|
||||
|
@ -59,27 +59,18 @@ const canToggle = computed(() => {
|
|||
});
|
||||
const canGetInfo = computed(() => !props.reaction.match(/@\w/) && props.reaction.includes(':'));
|
||||
|
||||
const isReacted = computed(() => {
|
||||
if (!props.note.myReaction) return false;
|
||||
|
||||
return normalizeReaction(props.note.myReaction) === normalizeReaction(props.reaction);
|
||||
});
|
||||
|
||||
async function toggleReaction() {
|
||||
if (!canToggle.value) return;
|
||||
|
||||
const oldReaction = props.note.myReaction;
|
||||
if (oldReaction) {
|
||||
const normalizedOldReaction = normalizeReaction(oldReaction);
|
||||
const normalizedNewReaction = normalizeReaction(props.reaction);
|
||||
|
||||
const confirm = await os.confirm({
|
||||
type: 'warning',
|
||||
text: normalizedOldReaction !== normalizedNewReaction ? i18n.ts.changeReactionConfirm : i18n.ts.cancelReactionConfirm,
|
||||
text: oldReaction !== props.reaction ? i18n.ts.changeReactionConfirm : i18n.ts.cancelReactionConfirm,
|
||||
});
|
||||
if (confirm.canceled) return;
|
||||
|
||||
if (normalizedOldReaction !== normalizedNewReaction) {
|
||||
if (oldReaction !== props.reaction) {
|
||||
sound.playMisskeySfx('reaction');
|
||||
}
|
||||
|
||||
|
@ -91,7 +82,7 @@ async function toggleReaction() {
|
|||
misskeyApi('notes/reactions/delete', {
|
||||
noteId: props.note.id,
|
||||
}).then(() => {
|
||||
if (normalizedOldReaction !== normalizedNewReaction) {
|
||||
if (oldReaction !== props.reaction) {
|
||||
misskeyApi('notes/reactions/create', {
|
||||
noteId: props.note.id,
|
||||
reaction: props.reaction,
|
||||
|
@ -184,16 +175,6 @@ if (!mock) {
|
|||
});
|
||||
}, 100);
|
||||
}
|
||||
|
||||
function normalizeReaction(reaction) {
|
||||
if (reaction.startsWith(':') && reaction.endsWith(':')) {
|
||||
const match = reaction.match(/^:([^@]+)(?:@[^:]+)?:$/);
|
||||
if (match) {
|
||||
return `:${match[1]}:`;
|
||||
}
|
||||
}
|
||||
return reaction;
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" module>
|
||||
|
|
|
@ -11,7 +11,6 @@ SPDX-License-Identifier: AGPL-3.0-only
|
|||
:leaveToClass="prefer.s.animation ? $style.transition_x_leaveTo : ''"
|
||||
:moveClass="prefer.s.animation ? $style.transition_x_move : ''"
|
||||
tag="div" :class="$style.root"
|
||||
@click.stop
|
||||
>
|
||||
<XReaction v-for="[reaction, count] in reactions" :key="reaction" :reaction="reaction" :count="count" :isInitial="initialReactions.has(reaction)" :note="note" @reactionToggled="onMockToggleReaction"/>
|
||||
<slot v-if="hasMoreReactions" name="more"/>
|
||||
|
@ -37,96 +36,51 @@ const emit = defineEmits<{
|
|||
(ev: 'mockUpdateMyReaction', emoji: string, delta: number): void;
|
||||
}>();
|
||||
|
||||
const initialReactions = ref(new Set<string>());
|
||||
const initialReactions = new Set(Object.keys(props.note.reactions));
|
||||
|
||||
const reactions = ref<[string, number][]>([]);
|
||||
const hasMoreReactions = ref(false);
|
||||
|
||||
function normalizeReaction(reaction) {
|
||||
if (reaction.startsWith(':') && reaction.endsWith(':')) {
|
||||
const match = reaction.match(/^:([^@]+)(?:@[^:]+)?:$/);
|
||||
if (match) {
|
||||
return `:${match[1]}:`;
|
||||
}
|
||||
}
|
||||
return reaction;
|
||||
if (props.note.myReaction && !Object.keys(reactions.value).includes(props.note.myReaction)) {
|
||||
reactions.value[props.note.myReaction] = props.note.reactions[props.note.myReaction];
|
||||
}
|
||||
|
||||
watch(() => props.note.myReaction, (newMyReaction) => {
|
||||
if (newMyReaction && !Object.keys(reactions.value).includes(newMyReaction)) {
|
||||
reactions.value[newMyReaction] = props.note.reactions[newMyReaction];
|
||||
}
|
||||
}, { immediate: true });
|
||||
|
||||
function onMockToggleReaction(emoji: string, count: number) {
|
||||
if (!mock) return;
|
||||
|
||||
const i = reactions.value.findIndex((item) => {
|
||||
return normalizeReaction(item[0]) === normalizeReaction(emoji);
|
||||
});
|
||||
const i = reactions.value.findIndex((item) => item[0] === emoji);
|
||||
if (i < 0) return;
|
||||
|
||||
emit('mockUpdateMyReaction', emoji, (count - reactions.value[i][1]));
|
||||
}
|
||||
|
||||
watch([() => props.note.reactions, () => props.maxNumber], ([newSource, maxNumber]) => {
|
||||
initialReactions.value = new Set(Object.keys(newSource));
|
||||
|
||||
const normalizedCounts = new Map<string, number>();
|
||||
const normalizedOriginals = new Map<string, string>();
|
||||
|
||||
for (const [reaction, count] of Object.entries(newSource)) {
|
||||
const normalized = normalizeReaction(reaction);
|
||||
const currentCount = normalizedCounts.get(normalized) || 0;
|
||||
normalizedCounts.set(normalized, currentCount + count);
|
||||
|
||||
if (!normalizedOriginals.has(normalized) || !reaction.includes('@')) {
|
||||
normalizedOriginals.set(normalized, reaction);
|
||||
}
|
||||
}
|
||||
|
||||
let newReactions: [string, number][] = [];
|
||||
|
||||
for (let i = 0; i < reactions.value.length; i++) {
|
||||
const [reaction] = reactions.value[i];
|
||||
const normalized = normalizeReaction(reaction);
|
||||
|
||||
if (normalizedCounts.has(normalized) && normalizedCounts.get(normalized)! > 0) {
|
||||
newReactions.push([
|
||||
normalizedOriginals.get(normalized)!,
|
||||
normalizedCounts.get(normalized)!,
|
||||
]);
|
||||
normalizedCounts.delete(normalized);
|
||||
}
|
||||
}
|
||||
|
||||
const remainingEntries = Array.from(normalizedCounts.entries())
|
||||
.filter(([, count]) => count > 0)
|
||||
.sort(([, a], [, b]) => b - a);
|
||||
|
||||
for (const [normalized, count] of remainingEntries) {
|
||||
if (newReactions.length < maxNumber) {
|
||||
newReactions.push([normalizedOriginals.get(normalized)!, count]);
|
||||
}
|
||||
}
|
||||
|
||||
hasMoreReactions.value = Object.keys(newSource).length > maxNumber;
|
||||
|
||||
if (props.note.myReaction) {
|
||||
const normalizedMyReaction = normalizeReaction(props.note.myReaction);
|
||||
const alreadyIncluded = newReactions.some(([x]) =>
|
||||
normalizeReaction(x) === normalizedMyReaction,
|
||||
);
|
||||
|
||||
if (!alreadyIncluded && newSource[props.note.myReaction]) {
|
||||
newReactions.push([
|
||||
props.note.myReaction,
|
||||
newSource[props.note.myReaction],
|
||||
]);
|
||||
for (let i = 0; i < reactions.value.length; i++) {
|
||||
const reaction = reactions.value[i][0];
|
||||
if (reaction in newSource && newSource[reaction] !== 0) {
|
||||
reactions.value[i][1] = newSource[reaction];
|
||||
newReactions.push(reactions.value[i]);
|
||||
}
|
||||
}
|
||||
|
||||
reactions.value = newReactions.slice(0, props.maxNumber);
|
||||
const newReactionsNames = newReactions.map(([x]) => x);
|
||||
newReactions = [
|
||||
...newReactions,
|
||||
...Object.entries(newSource)
|
||||
.sort(([, a], [, b]) => b - a)
|
||||
.filter(([y], i) => i < maxNumber && !newReactionsNames.includes(y)),
|
||||
];
|
||||
|
||||
newReactions = newReactions.slice(0, props.maxNumber);
|
||||
|
||||
if (props.note.myReaction && !newReactions.map(([x]) => x).includes(props.note.myReaction)) {
|
||||
newReactions.push([props.note.myReaction, newSource[props.note.myReaction]]);
|
||||
}
|
||||
|
||||
reactions.value = newReactions;
|
||||
}, { immediate: true, deep: true });
|
||||
</script>
|
||||
|
||||
|
|
Loading…
Add table
Reference in a new issue