1
0
Fork 0
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:
FLY_MC 2025-03-13 20:42:06 +08:00
parent 0199f38379
commit dab6cf09d5
3 changed files with 32 additions and 99 deletions

View file

@ -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);
}

View file

@ -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>

View file

@ -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>