50 lines
1.4 KiB
TypeScript
50 lines
1.4 KiB
TypeScript
|
import escapeTextContentForBrowser from 'escape-html';
|
||
|
import { z } from 'zod';
|
||
|
|
||
|
import emojify from 'soapbox/features/emoji';
|
||
|
|
||
|
import { customEmojiSchema } from './custom-emoji';
|
||
|
import { filteredArray, makeCustomEmojiMap } from './utils';
|
||
|
|
||
|
const pollOptionSchema = z.object({
|
||
|
title: z.string().catch(''),
|
||
|
votes_count: z.number().catch(0),
|
||
|
});
|
||
|
|
||
|
const pollSchema = z.object({
|
||
|
emojis: filteredArray(customEmojiSchema),
|
||
|
expired: z.boolean().catch(false),
|
||
|
expires_at: z.string().datetime().catch(new Date().toUTCString()),
|
||
|
id: z.string(),
|
||
|
multiple: z.boolean().catch(false),
|
||
|
options: z.array(pollOptionSchema).min(2),
|
||
|
voters_count: z.number().catch(0),
|
||
|
votes_count: z.number().catch(0),
|
||
|
own_votes: z.array(z.number()).nonempty().nullable().catch(null),
|
||
|
voted: z.boolean().catch(false),
|
||
|
pleroma: z.object({
|
||
|
non_anonymous: z.boolean().catch(false),
|
||
|
}).optional().catch(undefined),
|
||
|
}).transform((poll) => {
|
||
|
const emojiMap = makeCustomEmojiMap(poll.emojis);
|
||
|
|
||
|
const emojifiedOptions = poll.options.map((option) => ({
|
||
|
...option,
|
||
|
title_emojified: emojify(escapeTextContentForBrowser(option.title), emojiMap),
|
||
|
}));
|
||
|
|
||
|
// If the user has votes, they have certainly voted.
|
||
|
if (poll.own_votes?.length) {
|
||
|
poll.voted = true;
|
||
|
}
|
||
|
|
||
|
return {
|
||
|
...poll,
|
||
|
options: emojifiedOptions,
|
||
|
};
|
||
|
});
|
||
|
|
||
|
type Poll = z.infer<typeof pollSchema>;
|
||
|
type PollOption = Poll['options'][number];
|
||
|
|
||
|
export { pollSchema, type Poll, type PollOption };
|