Skip to content
Snippets Groups Projects
reactions-viewer.reaction.vue 3.11 KiB
Newer Older
<template>
syuilo's avatar
syuilo committed
<button
	class="hkzvhatu _button"
	:class="{ reacted: note.myReaction == reaction, canToggle }"
	@click="toggleReaction()"
	v-if="count > 0"
syuilo's avatar
syuilo committed
	@touchstart.passive="onMouseover"
	@mouseover="onMouseover"
	@mouseleave="onMouseleave"
syuilo's avatar
syuilo committed
	<XReactionIcon :reaction="reaction" :custom-emojis="note.emojis"/>
	<span>{{ count }}</span>
syuilo's avatar
syuilo committed
</button>
</template>

<script lang="ts">
import { computed, defineComponent, onMounted, ref, watch } from 'vue';
syuilo's avatar
syuilo committed
import XDetails from '@/components/reactions-viewer.details.vue';
import XReactionIcon from '@/components/reaction-icon.vue';
import * as os from '@/os';
import { useTooltip } from '@/scripts/use-tooltip';
import { $i } from '@/account';
syuilo's avatar
syuilo committed
export default defineComponent({
syuilo's avatar
syuilo committed
	components: {
		XReactionIcon
	},
	props: {
		reaction: {
			type: String,
			required: true,
		},
		count: {
			type: Number,
			required: true,
		},
		isInitial: {
			type: Boolean,
			required: true,
		},
		note: {
			type: Object,
			required: true,
		},
	},

	setup(props) {
		const buttonRef = ref<HTMLElement>();

		const canToggle = computed(() => !props.reaction.match(/@\w/) && $i);

		const toggleReaction = () => {
			if (!canToggle.value) return;

			const oldReaction = props.note.myReaction;
			if (oldReaction) {
syuilo's avatar
syuilo committed
				os.api('notes/reactions/delete', {
					noteId: props.note.id
				}).then(() => {
					if (oldReaction !== props.reaction) {
syuilo's avatar
syuilo committed
						os.api('notes/reactions/create', {
							noteId: props.note.id,
							reaction: props.reaction
						});
					}
				});
			} else {
syuilo's avatar
syuilo committed
				os.api('notes/reactions/create', {
					noteId: props.note.id,
					reaction: props.reaction
		};

		const anime = () => {
			if (document.hidden) return;

			// TODO: 新しくリアクションが付いたことが視覚的に分かりやすいアニメーション
		};

		watch(() => props.count, (newCount, oldCount) => {
			if (oldCount < newCount) anime();
		});

		onMounted(() => {
			if (!props.isInitial) anime();
		});

		const { onMouseover, onMouseleave } = useTooltip(async (showing) => {
			const reactions = await os.api('notes/reactions', {
				noteId: props.note.id,
				type: props.reaction,
syuilo's avatar
syuilo committed
				limit: 11
			const users = reactions
				.sort((a, b) => new Date(a.createdAt).getTime() - new Date(b.createdAt).getTime())
				.map(x => x.user);

			os.popup(XDetails, {
				showing,
				reaction: props.reaction,
				emojis: props.note.emojis,
				users,
				count: props.count,
				source: buttonRef.value
			}, {}, 'closed');
		});

		return {
			buttonRef,
			canToggle,
			toggleReaction,
			onMouseover,
			onMouseleave,
		};
	},
});
</script>

syuilo's avatar
syuilo committed
<style lang="scss" scoped>
syuilo's avatar
syuilo committed
.hkzvhatu {
syuilo's avatar
syuilo committed
	display: inline-block;
	height: 32px;
	margin: 2px;
	padding: 0 6px;
	border-radius: 4px;
	&.canToggle {
		background: rgba(0, 0, 0, 0.05);
		&:hover {
			background: rgba(0, 0, 0, 0.1);
syuilo's avatar
syuilo committed
		}
	}
	&:not(.canToggle) {
		cursor: default;
	}
	&.reacted {
		background: var(--accent);

syuilo's avatar
syuilo committed
		&:hover {
			background: var(--accent);
		}

		> span {
syuilo's avatar
syuilo committed
			color: var(--fgOnAccent);
syuilo's avatar
syuilo committed
		}
	}
syuilo's avatar
syuilo committed
	> span {
		font-size: 0.9em;
		line-height: 32px;
syuilo's avatar
syuilo committed
		margin: 0 0 0 4px;
syuilo's avatar
syuilo committed
	}
}
</style>