Skip to content
Snippets Groups Projects
reactions-viewer.reaction.vue 3.38 KiB
Newer Older
<template>
syuilo's avatar
syuilo committed
<button
	class="hkzvhatu _button"
	:class="{ reacted: note.myReaction == reaction, canToggle }"
	@click="toggleReaction(reaction)"
	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">
syuilo's avatar
syuilo committed
import { defineComponent, ref } 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';
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,
		},
	},
syuilo's avatar
syuilo committed
			close: null,
Aya Morisawa's avatar
Aya Morisawa committed
			detailsTimeoutId: null,
			isHovering: false
	computed: {
		canToggle(): boolean {
syuilo's avatar
syuilo committed
			return !this.reaction.match(/@\w/) && this.$i;
	},
	watch: {
		count(newCount, oldCount) {
			if (oldCount < newCount) this.anime();
syuilo's avatar
syuilo committed
			if (this.close != null) this.openDetails();
	methods: {
		toggleReaction() {
			if (!this.canToggle) return;

			const oldReaction = this.note.myReaction;
			if (oldReaction) {
syuilo's avatar
syuilo committed
				os.api('notes/reactions/delete', {
					noteId: this.note.id
				}).then(() => {
					if (oldReaction !== this.reaction) {
syuilo's avatar
syuilo committed
						os.api('notes/reactions/create', {
							noteId: this.note.id,
							reaction: this.reaction
						});
					}
				});
			} else {
syuilo's avatar
syuilo committed
				os.api('notes/reactions/create', {
					noteId: this.note.id,
					reaction: this.reaction
				});
			}
		},
Aya Morisawa's avatar
Aya Morisawa committed
			this.isHovering = true;
			this.detailsTimeoutId = setTimeout(this.openDetails, 300);
		},
		onMouseleave() {
Aya Morisawa's avatar
Aya Morisawa committed
			this.isHovering = false;
			clearTimeout(this.detailsTimeoutId);
			this.closeDetails();
		},
		openDetails() {
syuilo's avatar
syuilo committed
			os.api('notes/reactions', {
syuilo's avatar
syuilo committed
				noteId: this.note.id,
syuilo's avatar
syuilo committed
				type: this.reaction,
				limit: 11
			}).then((reactions: any[]) => {
syuilo's avatar
syuilo committed
				const users = reactions
					.sort((a, b) => new Date(a.createdAt).getTime() - new Date(b.createdAt).getTime())
syuilo's avatar
syuilo committed
					.map(x => x.user);
Aya Morisawa's avatar
Aya Morisawa committed
				if (!this.isHovering) return;
syuilo's avatar
syuilo committed

				const showing = ref(true);
				os.popup(XDetails, {
					showing,
					reaction: this.reaction,
syuilo's avatar
syuilo committed
					emojis: this.note.emojis,
syuilo's avatar
syuilo committed
					count: this.count,
					source: this.$refs.reaction
syuilo's avatar
syuilo committed
				}, {}, 'closed');

				this.close = () => {
					showing.value = false;
				};
syuilo's avatar
syuilo committed
			if (this.close != null) {
				this.close();
				this.close = null;
		anime() {
			if (document.hidden) return;

syuilo's avatar
syuilo committed
			// TODO
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>