Add support for multiple RSS feeds

This commit is contained in:
ajnart
2023-04-05 14:16:28 +09:00
parent 7cf6fe53fc
commit 6811388991
3 changed files with 123 additions and 135 deletions

View File

@@ -243,6 +243,25 @@ const WidgetOptionTypeSwitch: FC<{
</DraggableList> </DraggableList>
</Stack> </Stack>
); );
case 'multiple-text':
return (
<MultiSelect
data={value.map((v: any) => ({ value: v, label: v }))}
label={t(`descriptor.settings.${key}.label`)}
description={t(`descriptor.settings.${key}.description`)}
defaultValue={value as string[]}
withinPortal
searchable
creatable
getCreateLabel={(query) => `+ Add ${query}`}
onChange={(values) =>
handleChange(
key,
values.map((item: string) => item)
)
}
/>
);
/* eslint-enable no-case-declarations */ /* eslint-enable no-case-declarations */
default: default:
return null; return null;

View File

@@ -1,9 +1,9 @@
import Link from 'next/link';
import { import {
ActionIcon, ActionIcon,
Badge, Badge,
Card, Card,
Center, Center,
createStyles,
Flex, Flex,
Group, Group,
Image, Image,
@@ -14,31 +14,27 @@ import {
Stack, Stack,
Text, Text,
Title, Title,
createStyles,
} from '@mantine/core'; } from '@mantine/core';
import { import {
IconBulldozer,
IconCalendarTime,
IconClock, IconClock,
IconCopyright,
IconRefresh, IconRefresh,
IconRss, IconRss,
IconSpeakerphone,
} from '@tabler/icons'; } from '@tabler/icons';
import { useQuery } from '@tanstack/react-query'; import { useQuery } from '@tanstack/react-query';
import dayjs from 'dayjs'; import dayjs from 'dayjs';
import { useTranslation } from 'next-i18next'; import { useTranslation } from 'next-i18next';
import Link from 'next/link';
import { useState } from 'react';
import { IWidget } from '../widgets';
import { defineWidget } from '../helper'; import { defineWidget } from '../helper';
import { IWidget } from '../widgets';
const definition = defineWidget({ const definition = defineWidget({
id: 'rss', id: 'rss',
icon: IconRss, icon: IconRss,
options: { options: {
rssFeedUrl: { rssFeedUrl: {
type: 'text', type: 'multiple-text',
defaultValue: '', defaultValue: ['https://japantimes.co.jp/feed'],
}, },
}, },
gridstack: { gridstack: {
@@ -56,34 +52,42 @@ interface RssTileProps {
widget: IRssWidget; widget: IRssWidget;
} }
export const useGetRssFeed = (feedUrl: string, widgetId: string) => export const useGetRssFeeds = (feedUrls: string[], widgetId: string) =>
useQuery({ useQuery({
queryKey: ['rss-feed', feedUrl], queryKey: ['rss-feeds', feedUrls],
queryFn: async () => { queryFn: async () => {
const response = await fetch(`/api/modules/rss?widgetId=${widgetId}`); const responses = await Promise.all(
return response.json(); feedUrls.map((feedUrl) =>
fetch(
`/api/modules/rss?widgetId=${widgetId}&feedUrl=${encodeURIComponent(feedUrl)}`
).then((response) => response.json())
)
);
return responses;
}, },
}); });
function RssTile({ widget }: RssTileProps) { function RssTile({ widget }: RssTileProps) {
const { t } = useTranslation('modules/rss'); const { t } = useTranslation('modules/rss');
const { data, isLoading, isFetching, isError, refetch } = useGetRssFeed( const { data, isLoading, isFetching, isError, refetch } = useGetRssFeeds(
widget.properties.rssFeedUrl, widget.properties.rssFeedUrl,
widget.id widget.id
); );
const { classes } = useStyles(); const { classes } = useStyles();
const [loadingOverlayVisible, setLoadingOverlayVisible] = useState(false);
function formatDate(input: string): string { function formatDate(input: string): string {
// Parse the input date as a local date // Parse the input date as a local date
try {
const inputDate = dayjs(new Date(input)); const inputDate = dayjs(new Date(input));
const now = dayjs(); // Current date and time const now = dayjs(); // Current date and time
// The difference between the input date and now // The difference between the input date and now
const difference = now.diff(inputDate, 'ms'); const difference = now.diff(inputDate, 'ms');
const duration = dayjs.duration(difference, 'ms'); const duration = dayjs.duration(difference, 'ms');
const humanizedDuration = duration.humanize(); const humanizedDuration = duration.humanize();
return `${humanizedDuration} ago`; return `${humanizedDuration} ago`;
} catch (e) {
return 'Error';
}
} }
if (!data || isLoading) { if (!data || isLoading) {
@@ -94,7 +98,7 @@ function RssTile({ widget }: RssTileProps) {
); );
} }
if (!data.success || isError) { if (data.length < 1 || isError) {
return ( return (
<Center h="100%"> <Center h="100%">
<Stack align="center"> <Stack align="center">
@@ -109,10 +113,10 @@ function RssTile({ widget }: RssTileProps) {
return ( return (
<Stack h="100%"> <Stack h="100%">
<LoadingOverlay visible={isFetching} /> <LoadingOverlay visible={isFetching} />
<Flex align="end">{data.feed.title && <Title order={5}>{data.feed.title}</Title>}</Flex> <ScrollArea className="scroll-area-w100" w="100%" mt="sm" mb="sm">
<ScrollArea className="scroll-area-w100" w="100%"> {data.map((item, index) => (
<Stack w="100%" spacing="xs"> <Stack w="100%" spacing="xs">
{data.feed.items.map((item: any, index: number) => ( {item.feed.items.map((item: any, index: number) => (
<Card <Card
key={index} key={index}
withBorder withBorder
@@ -163,48 +167,9 @@ function RssTile({ widget }: RssTileProps) {
</Card> </Card>
))} ))}
</Stack> </Stack>
))}
</ScrollArea> </ScrollArea>
<Flex wrap="wrap" columnGap="md">
{data.feed.copyright && (
<Group spacing="sm">
<IconCopyright size={14} />
<Text color="dimmed" size="sm">
{data.feed.copyright}
</Text>
</Group>
)}
{data.feed.pubDate && (
<Group>
<IconCalendarTime size={14} />
<Text color="dimmed" size="sm">
{data.feed.pubDate}
</Text>
</Group>
)}
{data.feed.lastBuildDate && (
<Group>
<IconBulldozer size={14} />
<Text color="dimmed" size="sm">
{formatDate(data.feed.lastBuildDate)}
</Text>
</Group>
)}
{data.feed.feedUrl && (
<Group spacing="sm">
<IconSpeakerphone size={14} />
<Text
color="dimmed"
size="sm"
variant="link"
target="_blank"
component={Link}
href={data.feed.feedUrl}
>
Feed URL
</Text>
</Group>
)}
<ActionIcon <ActionIcon
size="sm" size="sm"
radius="xl" radius="xl"
@@ -218,13 +183,8 @@ function RssTile({ widget }: RssTileProps) {
}, },
}} }}
> >
{data.feed.image ? (
<Image src={data.feed.image.url} alt={data.feed.image.title} mx="auto" />
) : (
<IconRefresh /> <IconRefresh />
)}
</ActionIcon> </ActionIcon>
</Flex>
</Stack> </Stack>
); );
} }

View File

@@ -1,3 +1,4 @@
import React from 'react';
import { import {
MultiSelectProps, MultiSelectProps,
NumberInputProps, NumberInputProps,
@@ -7,7 +8,7 @@ import {
TextInputProps, TextInputProps,
} from '@mantine/core'; } from '@mantine/core';
import { TablerIcon } from '@tabler/icons'; import { TablerIcon } from '@tabler/icons';
import React from 'react';
import { AreaType } from '../types/area'; import { AreaType } from '../types/area';
import { ShapeType } from '../types/shape'; import { ShapeType } from '../types/shape';
@@ -37,7 +38,8 @@ export type IWidgetOptionValue =
| ISliderInputOptionValue | ISliderInputOptionValue
| ISelectOptionValue | ISelectOptionValue
| INumberInputOptionValue | INumberInputOptionValue
| IDraggableListInputValue; | IDraggableListInputValue
| IMultipleTextInputOptionValue;
// Interface for data type // Interface for data type
interface DataType { interface DataType {
@@ -105,6 +107,13 @@ export type IDraggableListInputValue = {
>; >;
}; };
// will show a text-input with a button to add a new line
export type IMultipleTextInputOptionValue = {
type: 'multiple-text';
defaultValue: string[];
inputProps?: Partial<TextInputProps>;
};
// is used to type the widget definitions which will be used to display all widgets // is used to type the widget definitions which will be used to display all widgets
export type IWidgetDefinition<TKey extends string = string> = { export type IWidgetDefinition<TKey extends string = string> = {
id: TKey; id: TKey;