feature: rss improvements (#1810)

This commit is contained in:
Manuel
2024-01-11 19:19:59 +01:00
committed by GitHub
parent 60bca7412c
commit 6717bcf8b4
3 changed files with 92 additions and 66 deletions

View File

@@ -17,6 +17,9 @@
}, },
"textLinesClamp": { "textLinesClamp": {
"label": "Text lines clamp" "label": "Text lines clamp"
},
"sortByPublishDateAscending": {
"label": "Sort by publish date (ascending)"
} }
}, },
"card": { "card": {

View File

@@ -15,6 +15,11 @@ type CustomItem = {
enclosure: { enclosure: {
url: string; url: string;
}; };
'media:group'?: {
'media:description'?: string;
'media:thumbnail'?: string;
},
pubDate?: string;
}; };
const rssFeedResultObjectSchema = z const rssFeedResultObjectSchema = z
@@ -38,7 +43,7 @@ const rssFeedResultObjectSchema = z
categories: z.array(z.string()).or(z.undefined()), categories: z.array(z.string()).or(z.undefined()),
title: z.string(), title: z.string(),
content: z.string(), content: z.string(),
pubDate: z.string().optional(), pubDate: z.date().optional(),
}), }),
), ),
}), }),
@@ -73,13 +78,11 @@ export const rssRouter = createTRPCRouter({
return []; return [];
} }
const result = await Promise.all( return await Promise.all(
input.feedUrls.map(async (feedUrl) => input.feedUrls.map(async (feedUrl) =>
getFeedUrl(feedUrl, rssWidget.properties.dangerousAllowSanitizedItemContent), getFeedUrl(feedUrl, rssWidget.properties.dangerousAllowSanitizedItemContent),
), ),
); );
return result;
}), }),
}); });
@@ -97,7 +100,12 @@ const getFeedUrl = async (feedUrl: string, dangerousAllowSanitizedItemContent: b
title: string; title: string;
content: string; content: string;
'content:encoded': string; 'content:encoded': string;
'media:group'?: {
'media:description'?: string;
'media:thumbnail'?: string;
}
categories: string[] | { _: string }[]; categories: string[] | { _: string }[];
pubDate?: string;
}) => ({ }) => ({
...item, ...item,
categories: item.categories categories: item.categories
@@ -105,11 +113,12 @@ const getFeedUrl = async (feedUrl: string, dangerousAllowSanitizedItemContent: b
.filter((category: unknown): category is string => typeof category === 'string'), .filter((category: unknown): category is string => typeof category === 'string'),
title: item.title ? decode(item.title) : undefined, title: item.title ? decode(item.title) : undefined,
content: processItemContent( content: processItemContent(
item['content:encoded'] ?? item.content, item['content:encoded'] ?? item.content ?? item['media:group']?.['media:description'],
dangerousAllowSanitizedItemContent, dangerousAllowSanitizedItemContent,
), ),
enclosure: createEnclosure(item), enclosure: createEnclosure(item),
link: createLink(item), link: createLink(item),
pubDate: item.pubDate ? new Date(item.pubDate) : null,
}), }),
) )
.sort((a: { pubDate: number }, b: { pubDate: number }) => { .sort((a: { pubDate: number }, b: { pubDate: number }) => {
@@ -183,11 +192,18 @@ const createEnclosure = (item: any) => {
}; };
} }
if (item['media:group'] && item['media:group']['media:thumbnail']) {
// no clue why this janky parse is needed
return {
url: item['media:group']['media:thumbnail'][0].$.url
};
}
return undefined; return undefined;
}; };
const parser: RssParser<any, CustomItem> = new RssParser({ const parser: RssParser<any, CustomItem> = new RssParser({
customFields: { customFields: {
item: ['media:content', 'enclosure'], item: ['media:content', 'enclosure', 'media:group'],
}, },
}); });

View File

@@ -51,6 +51,10 @@ const definition = defineWidget({
max: 50, max: 50,
step: 1, step: 1,
}, },
sortByPublishDateAscending: {
type: 'switch',
defaultValue: true,
},
}, },
gridstack: { gridstack: {
minWidth: 2, minWidth: 2,
@@ -74,7 +78,7 @@ function RssTile({ widget }: RssTileProps) {
configName, configName,
widget.properties.rssFeedUrl, widget.properties.rssFeedUrl,
widget.properties.refreshInterval, widget.properties.refreshInterval,
widget.id widget.id,
); );
const { classes } = useStyles(); const { classes } = useStyles();
@@ -100,7 +104,7 @@ function RssTile({ widget }: RssTileProps) {
); );
} }
if (!data || data.length < 1 || !data[0].feed || isError) { if (!data || data.length < 1 || isError) {
return ( return (
<Center h="100%"> <Center h="100%">
<Stack align="center"> <Stack align="center">
@@ -113,70 +117,73 @@ function RssTile({ widget }: RssTileProps) {
); );
} }
const flatFeeds = data.filter(feed => feed.success).flatMap(feed => feed.feed);
const flatFeedItems = flatFeeds.flatMap(feed => feed!.items);
const orderedFeedItems = widget.properties.sortByPublishDateAscending ?
flatFeedItems.sort((item1, item2) =>
(item2.pubDate?.getTime() as number) - (item1.pubDate?.getTime() as number)) : flatFeedItems;
return ( return (
<Stack h="100%"> <Stack h="100%">
<ScrollArea className="scroll-area-w100" w="100%" mt="sm" mb="sm"> <ScrollArea className="scroll-area-w100" w="100%" mt="sm" mb="sm">
{data.map((feed, index) => ( <Stack w="100%" spacing="xs">
<Stack w="100%" spacing="xs"> {orderedFeedItems.map((item: any, index: number) => (
{feed.feed && <Card
feed.feed.items.map((item: any, index: number) => ( key={index}
<Card withBorder
key={index} component={Link ?? 'div'}
withBorder href={item.link}
component={Link ?? 'div'} radius="md"
href={item.link} target="_blank"
radius="md" w="100%"
target="_blank" >
w="100%" {item.enclosure && (
> // eslint-disable-next-line @next/next/no-img-element
{item.enclosure && ( <img
// eslint-disable-next-line @next/next/no-img-element className={classes.backgroundImage}
<img src={item.enclosure.url ?? undefined}
className={classes.backgroundImage} alt="backdrop"
/>
)}
<Flex gap="xs">
{item.enclosure && item.enclosure.url && (
<MediaQuery query="(max-width: 1200px)" styles={{ display: 'none' }}>
<Image
src={item.enclosure.url ?? undefined} src={item.enclosure.url ?? undefined}
alt="backdrop" width={140}
height={140}
radius="md"
withPlaceholder
/> />
</MediaQuery>
)}
<Flex gap={2} direction="column" w="100%">
{item.categories && (
<Flex gap="xs" wrap="wrap" h={20} style={{ overflow: 'hidden' }}>
{item.categories.map((category: any, categoryIndex: number) => (
<Badge key={categoryIndex}>{category}</Badge>
))}
</Flex>
)} )}
<Flex gap="xs"> <Text lineClamp={2}>{item.title}</Text>
{item.enclosure && item.enclosure.url && ( <Text
<MediaQuery query="(max-width: 1200px)" styles={{ display: 'none' }}> className={classes.itemContent}
<Image color="dimmed"
src={item.enclosure.url ?? undefined} size="xs"
width={140} lineClamp={widget.properties.textLinesClamp}
height={140} dangerouslySetInnerHTML={{ __html: item.content }}
radius="md" />
withPlaceholder
/>
</MediaQuery>
)}
<Flex gap={2} direction="column" w="100%">
{item.categories && (
<Flex gap="xs" wrap="wrap" h={20} style={{ overflow: 'hidden' }}>
{item.categories.map((category: any, categoryIndex: number) => (
<Badge key={categoryIndex}>{category}</Badge>
))}
</Flex>
)}
<Text lineClamp={2}>{item.title}</Text> {item.pubDate && (
<Text <InfoDisplay title={item.title} date={formatDate(item.pubDate)} />
className={classes.itemContent} )}
color="dimmed" </Flex>
size="xs" </Flex>
lineClamp={widget.properties.textLinesClamp} </Card>
dangerouslySetInnerHTML={{ __html: item.content }} ))}
/> </Stack>
{item.pubDate && (
<InfoDisplay title={feed.feed.title} date={formatDate(item.pubDate)} />
)}
</Flex>
</Flex>
</Card>
))}
</Stack>
))}
</ScrollArea> </ScrollArea>
<RefetchButton refetch={refetch} isFetching={isFetching} /> <RefetchButton refetch={refetch} isFetching={isFetching} />
@@ -188,7 +195,7 @@ export const useGetRssFeeds = (
configName: string | undefined, configName: string | undefined,
feedUrls: string[], feedUrls: string[],
refreshInterval: number, refreshInterval: number,
widgetId: string widgetId: string,
) => ) =>
api.rss.all.useQuery( api.rss.all.useQuery(
{ {
@@ -201,7 +208,7 @@ export const useGetRssFeeds = (
cacheTime: 1000 * 60 * 60 * 24, cacheTime: 1000 * 60 * 60 * 24,
staleTime: 1000 * 60 * refreshInterval, staleTime: 1000 * 60 * refreshInterval,
enabled: !!configName, enabled: !!configName,
} },
); );
interface RefetchButtonProps { interface RefetchButtonProps {