2022-11-29 21:29:02 +01:00
|
|
|
<script setup lang="ts">
|
2023-01-08 07:21:09 +01:00
|
|
|
import type { mastodon } from 'masto'
|
2022-11-29 21:29:02 +01:00
|
|
|
|
2022-12-28 12:49:47 +01:00
|
|
|
const { status } = defineProps<{
|
2023-01-08 07:21:09 +01:00
|
|
|
status: mastodon.v1.Status
|
2022-11-29 21:29:02 +01:00
|
|
|
}>()
|
2022-12-28 12:49:47 +01:00
|
|
|
const poll = reactive({ ...status.poll! })
|
2022-11-29 21:29:02 +01:00
|
|
|
|
|
|
|
function toPercentage(num: number) {
|
|
|
|
const percentage = 100 * num
|
|
|
|
return `${percentage.toFixed(1).replace(/\.?0+$/, '')}%`
|
|
|
|
}
|
2022-12-02 03:18:36 +01:00
|
|
|
const timeAgoOptions = useTimeAgoOptions()
|
|
|
|
const expiredTimeAgo = useTimeAgo(poll.expiresAt!, timeAgoOptions)
|
2022-12-08 11:07:54 +01:00
|
|
|
const expiredTimeFormatted = useFormattedDateTime(poll.expiresAt!)
|
2023-01-09 12:24:26 +01:00
|
|
|
const { formatPercentage } = useHumanReadableNumber()
|
2022-11-29 21:29:02 +01:00
|
|
|
|
|
|
|
const masto = useMasto()
|
|
|
|
async function vote(e: Event) {
|
|
|
|
const formData = new FormData(e.target as HTMLFormElement)
|
|
|
|
const choices = formData.getAll('choices') as string[]
|
|
|
|
|
|
|
|
// Update the poll optimistically
|
|
|
|
for (const [index, option] of poll.options.entries()) {
|
|
|
|
if (choices.includes(String(index)))
|
|
|
|
option.votesCount = (option.votesCount || 0) + 1
|
|
|
|
}
|
|
|
|
poll.voted = true
|
|
|
|
poll.votesCount++
|
|
|
|
poll.votersCount = (poll.votersCount || 0) + 1
|
2022-12-28 12:49:47 +01:00
|
|
|
cacheStatus({ ...status, poll }, undefined, true)
|
|
|
|
|
2023-01-08 07:21:09 +01:00
|
|
|
await masto.v1.polls.vote(poll.id, { choices })
|
2022-11-29 21:29:02 +01:00
|
|
|
}
|
2023-01-01 15:29:11 +01:00
|
|
|
|
|
|
|
const votersCount = $computed(() => poll.votersCount ?? 0)
|
2022-11-29 21:29:02 +01:00
|
|
|
</script>
|
|
|
|
|
|
|
|
<template>
|
2023-01-01 15:29:11 +01:00
|
|
|
<div flex flex-col w-full items-stretch gap-3 dir="auto">
|
2022-11-30 07:35:12 +01:00
|
|
|
<form v-if="!poll.voted && !poll.expired" flex flex-col gap-4 accent-primary @click.stop="noop" @submit.prevent="vote">
|
2022-11-29 21:29:02 +01:00
|
|
|
<label v-for="(option, index) of poll.options" :key="index" flex items-center gap-2 px-2>
|
|
|
|
<input name="choices" :value="index" :type="poll.multiple ? 'checkbox' : 'radio'">
|
|
|
|
{{ option.title }}
|
|
|
|
</label>
|
|
|
|
<button btn-solid>
|
2023-01-09 21:20:26 +01:00
|
|
|
{{ $t('action_vote') }}
|
2022-11-29 21:29:02 +01:00
|
|
|
</button>
|
|
|
|
</form>
|
|
|
|
<template v-else>
|
2022-12-04 19:04:40 +01:00
|
|
|
<div v-for="(option, index) of poll.options" :key="index" py-1 relative :style="{ '--bar-width': toPercentage((option.votesCount || 0) / poll.votesCount) }">
|
|
|
|
<div flex justify-between pb-2 w-full>
|
|
|
|
<span inline-flex align-items>
|
|
|
|
{{ option.title }}
|
2023-01-01 15:29:11 +01:00
|
|
|
<span v-if="poll.voted && poll.ownVotes?.includes(index)" ms-2 mt-1 inline-block i-ri:checkbox-circle-line />
|
2022-12-04 19:04:40 +01:00
|
|
|
</span>
|
2023-01-01 15:29:11 +01:00
|
|
|
<span text-primary-active> {{ formatPercentage(votersCount > 0 ? (option.votesCount || 0) / votersCount : 0) }}</span>
|
2022-11-29 21:29:02 +01:00
|
|
|
</div>
|
2022-12-04 19:04:40 +01:00
|
|
|
<div class="bg-gray/40" rounded-l-sm rounded-r-lg h-5px w-full>
|
|
|
|
<div bg-primary-active h-full class="w-[var(--bar-width)]" />
|
2022-11-29 21:29:02 +01:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</template>
|
2023-01-01 15:29:11 +01:00
|
|
|
<div text-sm flex="~ inline" gap-x-1>
|
2023-01-09 12:24:26 +01:00
|
|
|
<CommonLocalizedNumber
|
2023-01-09 21:20:26 +01:00
|
|
|
keypath="status_poll.count"
|
2023-01-09 12:24:26 +01:00
|
|
|
:count="poll.votesCount"
|
|
|
|
/>
|
2022-12-04 19:58:21 +01:00
|
|
|
·
|
2022-12-08 11:07:54 +01:00
|
|
|
<CommonTooltip :content="expiredTimeFormatted" class="inline-block" placement="right">
|
2023-01-09 21:20:26 +01:00
|
|
|
<time :datetime="poll.expiresAt!">{{ $t(poll.expired ? 'status_poll_finished' : 'status_poll_ends', { date: expiredTimeAgo }) }}</time>
|
2022-12-08 11:07:54 +01:00
|
|
|
</CommonTooltip>
|
2022-11-29 21:29:02 +01:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</template>
|