feat: add missing integrations and rich notification previews

Add TickTick, Google Calendar, Google Drive and API (WebPage) notification
types, which the backend already supported but the extension ignored.

Fill the previously empty notification previews with content modeled on the
web app: a metadata sidebar (status, priority, assignee, labels, dates,
channel, etc.) plus a markdown body and comment/message threads. Add shared
helpers: PreviewDetail wrapper, TaskMetadata, Slack mrkdwn renderer, GitHub
check/review emoji, and date/HTML utils (cleanHtml strips raw HTML from
GitHub bodies).

The preview metadata "Type" row shows the source item type (Linear Issue,
GitHub Pull Request, Slack Thread, etc.).

Swap list-screen shortcuts: Enter shows details, Cmd+Enter opens in browser.
This commit is contained in:
2026-06-06 19:46:02 +02:00
parent fc5b290c5e
commit 9e51e0df6c
32 changed files with 1287 additions and 162 deletions
+14
View File
@@ -2,6 +2,20 @@
## [Unreleased]
### Added
- Add support for TickTick, Google Calendar, Google Drive and API (web page) notifications
- Fill notification previews with rich content (metadata sidebar, body and comment/message threads) modeled on the web app
- Show the notification source type (e.g. Linear Issue, GitHub Pull Request) in the preview metadata
### Changed
- Swap notification list shortcuts: `Enter` shows details, `Cmd+Enter` opens in browser
### Fixed
- Fix "could not open app" error when opening Google Calendar notifications
## [0.2.0] - 2024-12-16
### Added
+1 -1
View File
@@ -26,8 +26,8 @@ export function NotificationActions({ notification, detailsTarget, mutate }: Not
return (
<ActionPanel>
<Action.OpenInBrowser url={notificationHtmlUrl} />
<Action.Push title="Show Details" icon={Icon.AppWindowSidebarRight} target={detailsTarget} />
<Action.OpenInBrowser url={notificationHtmlUrl} shortcut={{ modifiers: ["cmd"], key: "enter" }} />
<Action
title="Delete Notification"
icon={Icon.Trash}
+1 -1
View File
@@ -22,8 +22,8 @@ export function NotificationTaskActions({ notification, detailsTarget, mutate }:
return (
<ActionPanel>
<Action.OpenInBrowser url={notificationHtmlUrl} />
<Action.Push title="Show Details" icon={Icon.AppWindowSidebarRight} target={detailsTarget} />
<Action.OpenInBrowser url={notificationHtmlUrl} shortcut={{ modifiers: ["cmd"], key: "enter" }} />
<Action
title="Delete Notification"
icon={Icon.Trash}
+16
View File
@@ -1,10 +1,14 @@
import { GoogleCalendarNotificationListItem } from "./integrations/google-calendar/listitem/GoogleCalendarNotificationListItem";
import { GoogleDriveNotificationListItem } from "./integrations/google-drive/listitem/GoogleDriveNotificationListItem";
import { Action, ActionPanel, Detail, Icon, List, getPreferenceValues, openExtensionPreferences } from "@raycast/api";
import { GoogleMailNotificationListItem } from "./integrations/google-mail/listitem/GoogleMailNotificationListItem";
import { TickTickNotificationListItem } from "./integrations/ticktick/listitem/TickTickNotificationListItem";
import { TodoistNotificationListItem } from "./integrations/todoist/listitem/TodoistNotificationListItem";
import { LinearNotificationListItem } from "./integrations/linear/listitem/LinearNotificationListItem";
import { GithubNotificationListItem } from "./integrations/github/listitem/GithubNotificationListItem";
import { SlackNotificationListItem } from "./integrations/slack/listitem/SlackNotificationListItem";
import { Notification, NotificationListItemProps, NotificationSourceKind } from "./notification";
import { APINotificationListItem } from "./integrations/api/listitem/APINotificationListItem";
import { NotificationActions } from "./action/NotificationActions";
import { Page, UniversalInboxPreferences } from "./types";
import { useFetch } from "@raycast/utils";
@@ -78,6 +82,14 @@ function NotificationListItem({ notification, mutate }: NotificationListItemProp
return <SlackNotificationListItem notification={notification} mutate={mutate} />;
case NotificationSourceKind.Todoist:
return <TodoistNotificationListItem notification={notification} mutate={mutate} />;
case NotificationSourceKind.TickTick:
return <TickTickNotificationListItem notification={notification} mutate={mutate} />;
case NotificationSourceKind.GoogleCalendar:
return <GoogleCalendarNotificationListItem notification={notification} mutate={mutate} />;
case NotificationSourceKind.GoogleDrive:
return <GoogleDriveNotificationListItem notification={notification} mutate={mutate} />;
case NotificationSourceKind.API:
return <APINotificationListItem notification={notification} mutate={mutate} />;
default:
return <DefaultNotificationListItem notification={notification} mutate={mutate} />;
}
@@ -115,6 +127,10 @@ function NotificationKindDropdown({ value, onNotificationKindChange }: Notificat
<List.Dropdown.Item key="GoogleMail" title="Google Mail" value="GoogleMail" />
<List.Dropdown.Item key="Slack" title="Slack" value="Slack" />
<List.Dropdown.Item key="Todoist" title="Todoist" value="Todoist" />
<List.Dropdown.Item key="TickTick" title="TickTick" value="TickTick" />
<List.Dropdown.Item key="GoogleCalendar" title="Google Calendar" value="GoogleCalendar" />
<List.Dropdown.Item key="GoogleDrive" title="Google Drive" value="GoogleDrive" />
<List.Dropdown.Item key="API" title="API" value="API" />
</List.Dropdown.Section>
</List.Dropdown>
);
@@ -0,0 +1,35 @@
import { APINotificationPreview } from "../preview/APINotificationPreview";
import { NotificationActions } from "../../../action/NotificationActions";
import { NotificationListItemProps } from "../../../notification";
import { Icon, List } from "@raycast/api";
import { WebPage } from "../types";
export function APINotificationListItem({ notification, mutate }: NotificationListItemProps) {
if (notification.source_item.data.type !== "WebPage") return null;
const webPage: WebPage = notification.source_item.data.content;
const accessories: List.Item.Accessory[] = [
{
date: new Date(notification.updated_at),
tooltip: `Updated at ${notification.updated_at}`,
},
];
return (
<List.Item
key={notification.id}
title={notification.title}
icon={Icon.Globe}
subtitle={webPage.url}
accessories={accessories}
actions={
<NotificationActions
notification={notification}
detailsTarget={<APINotificationPreview notification={notification} webPage={webPage} />}
mutate={mutate}
/>
}
/>
);
}
@@ -0,0 +1,38 @@
import { formatDateTime, formatElapsedTime } from "../../../utils";
import { PreviewDetail } from "../../../preview/PreviewDetail";
import { Notification } from "../../../notification";
import { Detail } from "@raycast/api";
import { WebPage } from "../types";
interface APINotificationPreviewProps {
notification: Notification;
webPage: WebPage;
}
function getHost(url: string): string {
try {
return new URL(url).host;
} catch {
return url;
}
}
export function APINotificationPreview({ notification, webPage }: APINotificationPreviewProps) {
const sourceLabel = webPage.source === "universalinboxextension" ? "Universal Inbox extension" : webPage.source;
const markdown = `# ${webPage.title || notification.title}\n\n[${webPage.url}](${webPage.url})`;
const metadata = (
<>
<Detail.Metadata.Link title="Host" target={webPage.url} text={getHost(webPage.url)} />
<Detail.Metadata.Label title="Source" text={sourceLabel} />
{webPage.timestamp ? (
<Detail.Metadata.Label
title="Captured"
text={`${formatDateTime(webPage.timestamp)} (${formatElapsedTime(webPage.timestamp)})`}
/>
) : null}
</>
);
return <PreviewDetail notification={notification} markdown={markdown} metadata={metadata} />;
}
+13
View File
@@ -0,0 +1,13 @@
export type APISource = "universalinboxextension" | string;
export interface WebPage {
url: string;
title: string;
timestamp: string;
source: APISource;
favicon?: string;
}
export function getWebPageHtmlUrl(webPage: WebPage): string {
return webPage.url;
}
+42
View File
@@ -0,0 +1,42 @@
import { GithubCheckConclusionState, GithubCheckStatusState, GithubPullRequestReviewState } from "./types";
/** Emoji for a GitHub check run based on its conclusion / status. */
export function checkRunEmoji(conclusion?: GithubCheckConclusionState, status?: GithubCheckStatusState): string {
if (status && status !== GithubCheckStatusState.Completed) {
return "⏳";
}
switch (conclusion) {
case GithubCheckConclusionState.Success:
return "✅";
case GithubCheckConclusionState.Failure:
case GithubCheckConclusionState.StartupFailure:
case GithubCheckConclusionState.TimedOut:
return "❌";
case GithubCheckConclusionState.ActionRequired:
return "⚠️";
case GithubCheckConclusionState.Cancelled:
case GithubCheckConclusionState.Skipped:
case GithubCheckConclusionState.Stale:
return "⚪";
case GithubCheckConclusionState.Neutral:
return "";
default:
return "🟡";
}
}
/** Emoji for a GitHub pull request review state. */
export function reviewStateEmoji(state: GithubPullRequestReviewState): string {
switch (state) {
case GithubPullRequestReviewState.Approved:
return "✅";
case GithubPullRequestReviewState.ChangesRequested:
return "🔴";
case GithubPullRequestReviewState.Commented:
return "💬";
case GithubPullRequestReviewState.Dismissed:
return "🚫";
case GithubPullRequestReviewState.Pending:
return "⏳";
}
}
@@ -1,26 +1,45 @@
import { getNotificationHtmlUrl, Notification } from "../../../notification";
import { Detail, ActionPanel, Action } from "@raycast/api";
import { GithubDiscussion } from "../types";
import { useMemo } from "react";
import { getGithubActorName, GithubDiscussion } from "../types";
import { PreviewDetail } from "../../../preview/PreviewDetail";
import { cleanHtml, formatElapsedTime } from "../../../utils";
import { Notification } from "../../../notification";
import { Detail } from "@raycast/api";
interface GithubDiscussionPreviewProps {
notification: Notification;
githubDiscussion: GithubDiscussion;
}
export function GithubDiscussionPreview({ notification, githubDiscussion }: GithubDiscussionPreviewProps) {
const notificationHtmlUrl = useMemo(() => {
return getNotificationHtmlUrl(notification);
}, [notification]);
export function GithubDiscussionPreview({ notification, githubDiscussion: discussion }: GithubDiscussionPreviewProps) {
let markdown = `# ${discussion.title} #${discussion.number}`;
if (discussion.author) {
markdown += `\n\n_Opened by ${getGithubActorName(discussion.author)} · ${formatElapsedTime(discussion.created_at)}_`;
}
if (discussion.body) {
markdown += `\n\n${cleanHtml(discussion.body)}`;
}
if (discussion.answer) {
markdown += `\n\n---\n\n## ✅ Answer`;
markdown += `\n\n**${getGithubActorName(discussion.answer.author)}**\n\n${cleanHtml(discussion.answer.body)}`;
}
return (
<Detail
markdown={`# ${githubDiscussion.title}`}
actions={
<ActionPanel>
<Action.OpenInBrowser url={notificationHtmlUrl} />
</ActionPanel>
}
/>
const metadata = (
<>
<Detail.Metadata.Label title="Repository" text={discussion.repository.name_with_owner} />
{discussion.state_reason ? <Detail.Metadata.Label title="State" text={discussion.state_reason} /> : null}
<Detail.Metadata.Label title="Comments" text={`${discussion.comments_count}`} />
<Detail.Metadata.Label title="Updated" text={formatElapsedTime(discussion.updated_at)} />
{discussion.answer_chosen_by ? (
<Detail.Metadata.Label title="Answered by" text={getGithubActorName(discussion.answer_chosen_by)} />
) : null}
{discussion.labels.length > 0 ? (
<Detail.Metadata.TagList title="Labels">
{discussion.labels.map((label) => (
<Detail.Metadata.TagList.Item key={label.name} text={label.name} color={`#${label.color}`} />
))}
</Detail.Metadata.TagList>
) : null}
</>
);
return <PreviewDetail notification={notification} markdown={markdown} metadata={metadata} />;
}
@@ -1,26 +1,103 @@
import { getNotificationHtmlUrl, Notification } from "../../../notification";
import { Detail, ActionPanel, Action } from "@raycast/api";
import { GithubPullRequest } from "../types";
import { useMemo } from "react";
import { getGithubActorName, GithubPullRequest, GithubPullRequestState, GithubReviewer } from "../types";
import { PreviewDetail } from "../../../preview/PreviewDetail";
import { cleanHtml, formatElapsedTime } from "../../../utils";
import { checkRunEmoji, reviewStateEmoji } from "../markdown";
import { Notification } from "../../../notification";
import { Color, Detail } from "@raycast/api";
interface GithubPullRequestPreviewProps {
notification: Notification;
githubPullRequest: GithubPullRequest;
}
export function GithubPullRequestPreview({ notification, githubPullRequest }: GithubPullRequestPreviewProps) {
const notificationHtmlUrl = useMemo(() => {
return getNotificationHtmlUrl(notification);
}, [notification]);
return (
<Detail
markdown={`# ${githubPullRequest.title}`}
actions={
<ActionPanel>
<Action.OpenInBrowser url={notificationHtmlUrl} />
</ActionPanel>
}
/>
);
function stateTag(pr: GithubPullRequest): { text: string; color: Color } {
if (pr.state === GithubPullRequestState.Merged) return { text: "Merged", color: Color.Purple };
if (pr.state === GithubPullRequestState.Closed) return { text: "Closed", color: Color.Red };
if (pr.is_draft) return { text: "Draft", color: Color.SecondaryText };
return { text: "Open", color: Color.Green };
}
function reviewerName(reviewer: GithubReviewer): string {
switch (reviewer.type) {
case "GithubUserSummary":
return reviewer.content.name || reviewer.content.login;
case "GithubBotSummary":
case "GithubMannequinSummary":
return reviewer.content.login;
case "GithubTeamSummary":
return reviewer.content.name;
}
}
export function GithubPullRequestPreview({ notification, githubPullRequest: pr }: GithubPullRequestPreviewProps) {
let markdown = `# ${pr.title} #${pr.number}`;
if (pr.body) {
markdown += `\n\n${cleanHtml(pr.body)}`;
}
const checkRuns = pr.latest_commit.check_suites?.flatMap((suite) => suite.check_runs) ?? [];
if (checkRuns.length > 0) {
markdown += `\n\n---\n\n## Checks\n\n`;
markdown += checkRuns.map((run) => `${checkRunEmoji(run.conclusion, run.status)} ${run.name}`).join("\n");
}
if (pr.reviews.length > 0) {
markdown += `\n\n---\n\n## Reviews\n\n`;
markdown += pr.reviews
.map((review) => {
const head = `${reviewStateEmoji(review.state)} **${getGithubActorName(review.author)}**`;
return review.body ? `${head}\n\n${cleanHtml(review.body)}` : head;
})
.join("\n\n");
}
if (pr.comments.length > 0) {
markdown += `\n\n---\n\n## Comments\n\n`;
markdown += pr.comments
.map(
(comment) =>
`**${getGithubActorName(comment.author)}** · ${formatElapsedTime(comment.created_at)}\n\n${cleanHtml(comment.body)}`,
)
.join("\n\n---\n\n");
}
const state = stateTag(pr);
const metadata = (
<>
<Detail.Metadata.TagList title="Status">
<Detail.Metadata.TagList.Item text={state.text} color={state.color} />
</Detail.Metadata.TagList>
{pr.state === GithubPullRequestState.Open ? (
<Detail.Metadata.Label title="Mergeable" text={`${pr.mergeable_state} · ${pr.merge_state_status}`} />
) : null}
<Detail.Metadata.Label title="Changes" text={`+${pr.additions} ${pr.deletions} in ${pr.changed_files} files`} />
<Detail.Metadata.Label title="Branch" text={`${pr.head_ref_name}${pr.base_ref_name}`} />
{pr.review_decision ? <Detail.Metadata.Label title="Review" text={pr.review_decision} /> : null}
{pr.author ? <Detail.Metadata.Label title="Author" text={getGithubActorName(pr.author)} /> : null}
{pr.assignees.length > 0 ? (
<Detail.Metadata.TagList title="Assignees">
{pr.assignees.map((assignee, index) => (
<Detail.Metadata.TagList.Item key={index} text={getGithubActorName(assignee)} />
))}
</Detail.Metadata.TagList>
) : null}
{pr.review_requests.length > 0 ? (
<Detail.Metadata.TagList title="Review requests">
{pr.review_requests.map((reviewer, index) => (
<Detail.Metadata.TagList.Item key={index} text={reviewerName(reviewer)} />
))}
</Detail.Metadata.TagList>
) : null}
{pr.labels.length > 0 ? (
<Detail.Metadata.TagList title="Labels">
{pr.labels.map((label) => (
<Detail.Metadata.TagList.Item key={label.name} text={label.name} color={`#${label.color}`} />
))}
</Detail.Metadata.TagList>
) : null}
</>
);
return <PreviewDetail notification={notification} markdown={markdown} metadata={metadata} />;
}
@@ -0,0 +1,59 @@
import { GoogleCalendarEventPreview } from "../preview/GoogleCalendarEventPreview";
import { GoogleCalendarEvent, getEventStartDisplay, getMeetLink } from "../types";
import { NotificationActions } from "../../../action/NotificationActions";
import { NotificationListItemProps } from "../../../notification";
import { Icon, Color, List } from "@raycast/api";
export function GoogleCalendarNotificationListItem({ notification, mutate }: NotificationListItemProps) {
if (notification.source_item.data.type !== "GoogleCalendarEvent") return null;
const event: GoogleCalendarEvent = notification.source_item.data.content;
const subtitle = getEventStartDisplay(event);
const meetLink = getMeetLink(event);
const accessories: List.Item.Accessory[] = [
{
date: new Date(notification.updated_at),
tooltip: `Updated at ${notification.updated_at}`,
},
];
if (event.attendees.length > 0) {
accessories.unshift({
text: `${event.attendees.length}`,
icon: Icon.Person,
tooltip: `${event.attendees.length} attendee(s)`,
});
}
if (meetLink) {
accessories.unshift({
icon: { source: Icon.Video, tintColor: Color.Green },
tooltip: "Google Meet link available",
});
}
if (event.status === "cancelled") {
accessories.unshift({
icon: { source: Icon.XMarkCircle, tintColor: Color.Red },
tooltip: "Cancelled",
});
}
return (
<List.Item
key={notification.id}
title={notification.title}
icon={Icon.Calendar}
subtitle={subtitle}
accessories={accessories}
actions={
<NotificationActions
notification={notification}
detailsTarget={<GoogleCalendarEventPreview notification={notification} event={event} />}
mutate={mutate}
/>
}
/>
);
}
@@ -0,0 +1,58 @@
import { GoogleCalendarEvent, getEventStartDisplay, getMeetLink } from "../types";
import { getNotificationHtmlUrl, Notification } from "../../../notification";
import { getThirdPartyItemSourceLabel } from "../../../third_party_item";
import { Detail, ActionPanel, Action, Color } from "@raycast/api";
import { useMemo } from "react";
interface GoogleCalendarEventPreviewProps {
notification: Notification;
event: GoogleCalendarEvent;
}
export function GoogleCalendarEventPreview({ notification, event }: GoogleCalendarEventPreviewProps) {
const notificationHtmlUrl = useMemo(() => getNotificationHtmlUrl(notification), [notification]);
const meetLink = getMeetLink(event);
const endDisplay = event.end.dateTime ? new Date(event.end.dateTime).toLocaleString() : (event.end.date ?? "");
const attendeeLines = event.attendees
.map((a) => {
const name = a.displayName ?? a.email;
const status = a.responseStatus === "accepted" ? "✓" : a.responseStatus === "declined" ? "✗" : "?";
return `- ${status} ${name}`;
})
.join("\n");
const markdown = [
`# ${event.summary}`,
event.description ? `\n${event.description}` : null,
attendeeLines ? `\n## Attendees\n\n${attendeeLines}` : null,
]
.filter(Boolean)
.join("\n");
const statusColor =
event.status === "cancelled" ? Color.Red : event.status === "tentative" ? Color.Yellow : Color.Green;
const metadata = (
<Detail.Metadata>
<Detail.Metadata.Label title="Type" text={getThirdPartyItemSourceLabel(notification.source_item)} />
<Detail.Metadata.Separator />
<Detail.Metadata.TagList title="Status">
<Detail.Metadata.TagList.Item text={event.status} color={statusColor} />
</Detail.Metadata.TagList>
<Detail.Metadata.Label title="Start" text={getEventStartDisplay(event)} />
<Detail.Metadata.Label title="End" text={endDisplay} />
{event.location ? <Detail.Metadata.Label title="Location" text={event.location} /> : null}
{meetLink ? <Detail.Metadata.Link title="Meet" target={meetLink} text="Join" /> : null}
<Detail.Metadata.Label title="Attendees" text={`${event.attendees.length}`} />
</Detail.Metadata>
);
const actions = [<Action.OpenInBrowser key="open" url={notificationHtmlUrl} />];
if (meetLink) {
actions.push(<Action.OpenInBrowser key="meet" title="Join Meeting" url={meetLink} />);
}
return <Detail markdown={markdown} metadata={metadata} actions={<ActionPanel>{actions}</ActionPanel>} />;
}
+64
View File
@@ -0,0 +1,64 @@
export interface EventDateTime {
dateTime?: string;
date?: string;
timeZone?: string;
}
export interface EventAttendee {
email: string;
displayName?: string;
responseStatus: "needsAction" | "declined" | "tentative" | "accepted";
self?: boolean;
organizer?: boolean;
}
export interface EntryPoint {
entryPointType: "video" | "phone" | "sip" | "more";
uri: string;
label?: string;
}
export interface ConferenceSolution {
name: string;
iconUri?: string;
}
export interface ConferenceData {
entryPoints?: EntryPoint[];
conferenceSolution?: ConferenceSolution;
}
export interface GoogleCalendarEvent {
id: string;
summary: string;
htmlLink: string;
hangoutLink?: string;
start: EventDateTime;
end: EventDateTime;
attendees: EventAttendee[];
status: "confirmed" | "tentative" | "cancelled";
eventType: string;
conferenceData?: ConferenceData;
description?: string;
location?: string;
}
export function getGoogleCalendarEventHtmlUrl(event: GoogleCalendarEvent): string {
return event.htmlLink;
}
export function getEventStartDisplay(event: GoogleCalendarEvent): string {
if (event.start.dateTime) {
return new Date(event.start.dateTime).toLocaleString();
}
if (event.start.date) {
return event.start.date;
}
return "";
}
export function getMeetLink(event: GoogleCalendarEvent): string | undefined {
if (event.hangoutLink) return event.hangoutLink;
const meetEntry = event.conferenceData?.entryPoints?.find((ep) => ep.entryPointType === "video");
return meetEntry?.uri;
}
@@ -0,0 +1,50 @@
import { GoogleDriveCommentPreview } from "../preview/GoogleDriveCommentPreview";
import { NotificationActions } from "../../../action/NotificationActions";
import { NotificationListItemProps } from "../../../notification";
import { Icon, Color, List } from "@raycast/api";
import { GoogleDriveComment } from "../types";
export function GoogleDriveNotificationListItem({ notification, mutate }: NotificationListItemProps) {
if (notification.source_item.data.type !== "GoogleDriveComment") return null;
const comment: GoogleDriveComment = notification.source_item.data.content;
const accessories: List.Item.Accessory[] = [
{
date: new Date(notification.updated_at),
tooltip: `Updated at ${notification.updated_at}`,
},
];
if (comment.resolved) {
accessories.unshift({
icon: { source: Icon.CheckCircle, tintColor: Color.Green },
tooltip: "Resolved",
});
}
if (comment.replies.length > 0) {
accessories.unshift({
text: `${comment.replies.length}`,
icon: Icon.Message,
tooltip: `${comment.replies.length} repl${comment.replies.length === 1 ? "y" : "ies"}`,
});
}
return (
<List.Item
key={notification.id}
title={notification.title}
icon={Icon.Document}
subtitle={comment.file_name}
accessories={accessories}
actions={
<NotificationActions
notification={notification}
detailsTarget={<GoogleDriveCommentPreview notification={notification} comment={comment} />}
mutate={mutate}
/>
}
/>
);
}
@@ -0,0 +1,44 @@
import { PreviewDetail } from "../../../preview/PreviewDetail";
import { Notification } from "../../../notification";
import { GoogleDriveComment } from "../types";
import { Color, Detail } from "@raycast/api";
interface GoogleDriveCommentPreviewProps {
notification: Notification;
comment: GoogleDriveComment;
}
export function GoogleDriveCommentPreview({ notification, comment }: GoogleDriveCommentPreviewProps) {
const quotedSection = comment.quoted_file_content
? `> ${comment.quoted_file_content.split("\n").join("\n> ")}\n\n`
: "";
const replyLines = comment.replies.map((r) => `**${r.author.display_name}:** ${r.content}`).join("\n\n");
const markdown = [
`# ${comment.file_name}`,
``,
`**Comment by ${comment.author.display_name}:**`,
``,
quotedSection + comment.content,
replyLines ? `\n---\n\n### Replies\n\n${replyLines}` : null,
]
.filter((line) => line !== null)
.join("\n");
const metadata = (
<>
<Detail.Metadata.Label title="File" text={comment.file_name} />
<Detail.Metadata.Label title="Author" text={comment.author.display_name} />
<Detail.Metadata.Label title="Replies" text={`${comment.replies.length}`} />
<Detail.Metadata.TagList title="Status">
<Detail.Metadata.TagList.Item
text={comment.resolved ? "Resolved" : "Open"}
color={comment.resolved ? Color.Green : Color.Yellow}
/>
</Detail.Metadata.TagList>
</>
);
return <PreviewDetail notification={notification} markdown={markdown} metadata={metadata} />;
}
+34
View File
@@ -0,0 +1,34 @@
export interface GoogleDriveCommentAuthor {
display_name: string;
email_address?: string;
photo_link?: string;
}
export interface GoogleDriveCommentReply {
id: string;
content: string;
html_content?: string;
author: GoogleDriveCommentAuthor;
created_time: string;
modified_time: string;
action?: string;
}
export interface GoogleDriveComment {
id: string;
file_id: string;
file_name: string;
file_mime_type: string;
content: string;
html_content?: string;
quoted_file_content?: string;
author: GoogleDriveCommentAuthor;
created_time: string;
modified_time: string;
resolved?: boolean;
replies: GoogleDriveCommentReply[];
}
export function getGoogleDriveCommentHtmlUrl(comment: GoogleDriveComment): string {
return `https://drive.google.com/file/d/${comment.file_id}/view`;
}
@@ -1,26 +1,49 @@
import { getNotificationHtmlUrl, Notification } from "../../../notification";
import { Detail, ActionPanel, Action } from "@raycast/api";
import { GoogleMailThread } from "../types";
import { useMemo } from "react";
import { PreviewDetail } from "../../../preview/PreviewDetail";
import { GoogleMailMessage, GoogleMailThread } from "../types";
import { Notification } from "../../../notification";
import { formatDateTime } from "../../../utils";
import { Color, Detail } from "@raycast/api";
interface GoogleMailThreadPreviewProps {
notification: Notification;
googleMailThread: GoogleMailThread;
}
export function GoogleMailThreadPreview({ notification }: GoogleMailThreadPreviewProps) {
const notificationHtmlUrl = useMemo(() => {
return getNotificationHtmlUrl(notification);
}, [notification]);
return (
<Detail
markdown={`# ${notification.title}`}
actions={
<ActionPanel>
<Action.OpenInBrowser url={notificationHtmlUrl} />
</ActionPanel>
}
/>
);
function header(message: GoogleMailMessage, name: string): string | undefined {
return message.payload.headers.find((h) => h.name.toLowerCase() === name.toLowerCase())?.value;
}
export function GoogleMailThreadPreview({ notification, googleMailThread }: GoogleMailThreadPreviewProps) {
const messages = googleMailThread.messages;
const firstMessage = messages[0];
const subject = (firstMessage && header(firstMessage, "Subject")) || notification.title;
const isStarred = messages.some((m) => m.labelIds?.includes("STARRED"));
const isImportant = messages.some((m) => m.labelIds?.includes("IMPORTANT"));
const body = messages
.map((message) => {
const from = header(message, "From") ?? "Unknown sender";
const date = formatDateTime(message.internalDate);
return `**${from}** · ${date}\n\n${message.snippet}`;
})
.join("\n\n---\n\n");
const markdown = `# ${subject}\n\n${body}`;
const metadata = (
<>
{firstMessage ? <Detail.Metadata.Label title="From" text={header(firstMessage, "From") ?? "Unknown"} /> : null}
<Detail.Metadata.Label title="Subject" text={subject} />
<Detail.Metadata.Label title="Messages" text={`${messages.length}`} />
{isStarred || isImportant ? (
<Detail.Metadata.TagList title="Labels">
{isStarred ? <Detail.Metadata.TagList.Item text="Starred" color={Color.Yellow} /> : null}
{isImportant ? <Detail.Metadata.TagList.Item text="Important" color={Color.Red} /> : null}
</Detail.Metadata.TagList>
) : null}
</>
);
return <PreviewDetail notification={notification} markdown={markdown} metadata={metadata} />;
}
@@ -1,26 +1,85 @@
import { getNotificationHtmlUrl, Notification } from "../../../notification";
import { Detail, ActionPanel, Action } from "@raycast/api";
import { LinearIssue } from "../types";
import { useMemo } from "react";
import { PreviewDetail } from "../../../preview/PreviewDetail";
import { formatDate, formatElapsedTime } from "../../../utils";
import { LinearIssue, LinearIssuePriority } from "../types";
import { Notification } from "../../../notification";
import { Color, Detail, Image } from "@raycast/api";
interface LinearIssuePreviewProps {
notification: Notification;
linearIssue: LinearIssue;
}
export function LinearIssuePreview({ notification, linearIssue }: LinearIssuePreviewProps) {
const notificationHtmlUrl = useMemo(() => {
return getNotificationHtmlUrl(notification);
}, [notification]);
const PRIORITY_LABEL: Record<LinearIssuePriority, string> = {
[LinearIssuePriority.NoPriority]: "No priority",
[LinearIssuePriority.Urgent]: "Urgent",
[LinearIssuePriority.High]: "High",
[LinearIssuePriority.Normal]: "Normal",
[LinearIssuePriority.Low]: "Low",
};
return (
<Detail
markdown={`# ${linearIssue.title}`}
actions={
<ActionPanel>
<Action.OpenInBrowser url={notificationHtmlUrl} />
</ActionPanel>
}
/>
);
const PRIORITY_COLOR: Record<LinearIssuePriority, Color> = {
[LinearIssuePriority.NoPriority]: Color.SecondaryText,
[LinearIssuePriority.Urgent]: Color.Red,
[LinearIssuePriority.High]: Color.Orange,
[LinearIssuePriority.Normal]: Color.Yellow,
[LinearIssuePriority.Low]: Color.Blue,
};
function userIcon(avatarUrl?: string): Image.ImageLike | undefined {
return avatarUrl ? { source: avatarUrl, mask: Image.Mask.Circle } : undefined;
}
export function LinearIssuePreview({ notification, linearIssue }: LinearIssuePreviewProps) {
let markdown = `# ${linearIssue.identifier} ${linearIssue.title}`;
if (linearIssue.creator) {
markdown += `\n\n_Opened by ${linearIssue.creator.name} · ${formatElapsedTime(linearIssue.created_at)}_`;
}
if (linearIssue.description) {
markdown += `\n\n${linearIssue.description}`;
}
const metadata = (
<>
<Detail.Metadata.TagList title="Status">
<Detail.Metadata.TagList.Item text={linearIssue.state.name} color={linearIssue.state.color} />
</Detail.Metadata.TagList>
<Detail.Metadata.TagList title="Priority">
<Detail.Metadata.TagList.Item
text={PRIORITY_LABEL[linearIssue.priority]}
color={PRIORITY_COLOR[linearIssue.priority]}
/>
</Detail.Metadata.TagList>
{linearIssue.assignee ? (
<Detail.Metadata.Label
title="Assignee"
text={linearIssue.assignee.name}
icon={userIcon(linearIssue.assignee.avatar_url)}
/>
) : null}
{linearIssue.creator ? (
<Detail.Metadata.Label
title="Creator"
text={linearIssue.creator.name}
icon={userIcon(linearIssue.creator.avatar_url)}
/>
) : null}
{linearIssue.project ? (
<Detail.Metadata.Link title="Project" target={linearIssue.project.url} text={linearIssue.project.name} />
) : null}
{linearIssue.project_milestone ? (
<Detail.Metadata.Label title="Milestone" text={linearIssue.project_milestone.name} />
) : null}
<Detail.Metadata.Label title="Team" text={linearIssue.team.name} />
{linearIssue.due_date ? <Detail.Metadata.Label title="Due" text={formatDate(linearIssue.due_date)} /> : null}
{linearIssue.labels.length > 0 ? (
<Detail.Metadata.TagList title="Labels">
{linearIssue.labels.map((label) => (
<Detail.Metadata.TagList.Item key={label.name} text={label.name} color={label.color} />
))}
</Detail.Metadata.TagList>
) : null}
</>
);
return <PreviewDetail notification={notification} markdown={markdown} metadata={metadata} />;
}
@@ -1,26 +1,60 @@
import { getNotificationHtmlUrl, Notification } from "../../../notification";
import { Detail, ActionPanel, Action } from "@raycast/api";
import { LinearProject } from "../types";
import { useMemo } from "react";
import { PreviewDetail } from "../../../preview/PreviewDetail";
import { LinearProject, LinearProjectState } from "../types";
import { Notification } from "../../../notification";
import { Color, Detail, Image } from "@raycast/api";
import { formatDate } from "../../../utils";
interface LinearProjectPreviewProps {
notification: Notification;
linearProject: LinearProject;
}
export function LinearProjectPreview({ notification, linearProject }: LinearProjectPreviewProps) {
const notificationHtmlUrl = useMemo(() => {
return getNotificationHtmlUrl(notification);
}, [notification]);
const STATE_COLOR: Record<LinearProjectState, Color> = {
[LinearProjectState.Planned]: Color.Blue,
[LinearProjectState.Backlog]: Color.SecondaryText,
[LinearProjectState.Started]: Color.Yellow,
[LinearProjectState.Paused]: Color.Orange,
[LinearProjectState.Completed]: Color.Green,
[LinearProjectState.Canceled]: Color.Red,
};
return (
<Detail
markdown={`# ${linearProject.name}`}
actions={
<ActionPanel>
<Action.OpenInBrowser url={notificationHtmlUrl} />
</ActionPanel>
}
/>
);
function progressBar(progress: number): string {
const filled = Math.round(progress / 10);
return `${"▰".repeat(filled)}${"▱".repeat(10 - filled)} ${progress}%`;
}
export function LinearProjectPreview({ notification, linearProject }: LinearProjectPreviewProps) {
let markdown = `# ${linearProject.name}`;
markdown += `\n\n${progressBar(linearProject.progress)}`;
if (linearProject.description) {
markdown += `\n\n${linearProject.description}`;
}
const metadata = (
<>
<Detail.Metadata.TagList title="State">
<Detail.Metadata.TagList.Item text={linearProject.state} color={STATE_COLOR[linearProject.state]} />
</Detail.Metadata.TagList>
<Detail.Metadata.Label title="Progress" text={`${linearProject.progress}%`} />
{linearProject.lead ? (
<Detail.Metadata.Label
title="Lead"
text={linearProject.lead.name}
icon={
linearProject.lead.avatar_url
? { source: linearProject.lead.avatar_url, mask: Image.Mask.Circle }
: undefined
}
/>
) : null}
{linearProject.start_date ? (
<Detail.Metadata.Label title="Start" text={formatDate(linearProject.start_date)} />
) : null}
{linearProject.target_date ? (
<Detail.Metadata.Label title="Target" text={formatDate(linearProject.target_date)} />
) : null}
</>
);
return <PreviewDetail notification={notification} markdown={markdown} metadata={metadata} />;
}
+100
View File
@@ -0,0 +1,100 @@
import { SlackBlock, SlackBlockText, SlackHistoryMessage, SlackMessageSenderDetails, SlackReferences } from "./types";
function blockTextToString(text?: SlackBlockText): string {
if (!text) return "";
return text.type === "plain_text" ? text.value : text.text;
}
/** Resolve the display name of a message's sender from the thread's sender_profiles map. */
export function slackSenderName(
message: SlackHistoryMessage,
senderProfiles: Record<string, SlackMessageSenderDetails>,
): string {
const key = message.user ?? message.bot_id ?? "";
const profile = senderProfiles[key];
if (!profile) return "Unknown";
if (profile.type === "User") {
return (
profile.content.profile?.display_name || profile.content.real_name || profile.content.name || profile.content.id
);
}
return profile.content.name;
}
/** Replace Slack reference tokens (<@U…>, <#C…>, <url|label>, <!here>) with markdown. */
function resolveReferences(text: string, references?: SlackReferences): string {
let out = text;
out = out.replace(/<@([A-Z0-9]+)(\|[^>]*)?>/g, (_match, id: string) => {
const name = references?.users?.[id];
return name ? `@${name}` : `@${id}`;
});
out = out.replace(/<#([A-Z0-9]+)(\|([^>]*))?>/g, (_match, id: string, _full: string, name: string) => {
const refName = name || references?.channels?.[id];
return refName ? `#${refName}` : `#${id}`;
});
out = out.replace(/<!subteam\^([A-Z0-9]+)(\|([^>]*))?>/g, (_match, id: string, _full: string, name: string) =>
name ? `@${name}` : `@${id}`,
);
out = out.replace(/<!(here|channel|everyone)>/g, (_match, keyword: string) => `@${keyword}`);
out = out.replace(/<(https?:[^|>]+)\|([^>]+)>/g, (_match, url: string, label: string) => `[${label}](${url})`);
out = out.replace(/<(https?:[^|>]+)>/g, (_match, url: string) => url);
return out;
}
/** Convert Slack mrkdwn emphasis to GitHub-flavoured markdown. */
function slackMrkdwnToMarkdown(text: string): string {
let out = text;
// *bold* -> **bold** (Slack uses single asterisks)
out = out.replace(/(^|[\s(])\*(?!\s)([^*\n]+?)\*(?=[\s).,!?:]|$)/g, "$1**$2**");
// ~strike~ -> ~~strike~~
out = out.replace(/(^|[\s(])~(?!\s)([^~\n]+?)~(?=[\s).,!?:]|$)/g, "$1~~$2~~");
return out;
}
function blockToMarkdown(block: SlackBlock): string {
switch (block.type) {
case "section": {
const main = blockTextToString(block.text);
const fields = block.fields?.map(blockTextToString).filter(Boolean).join("\n") ?? "";
return [main, fields].filter(Boolean).join("\n");
}
case "header":
return `### ${blockTextToString(block.text)}`;
case "divider":
return "---";
case "image":
return `![${block.alt_text}](${block.image_url})`;
default:
return "";
}
}
/** Render a Slack message (text or blocks, files, reactions) as markdown. */
export function slackMessageToMarkdown(message: SlackHistoryMessage, references?: SlackReferences): string {
const parts: string[] = [];
let body = message.text ?? "";
if (!body && message.blocks) {
body = message.blocks.map(blockToMarkdown).filter(Boolean).join("\n\n");
}
if (body) {
parts.push(slackMrkdwnToMarkdown(resolveReferences(body, references)));
}
if (message.files?.length) {
parts.push(
message.files
.map((file) => {
const name = file.title || file.name || "file";
return file.permalink ? `📎 [${name}](${file.permalink})` : `📎 ${name}`;
})
.join("\n"),
);
}
if (message.reactions?.length) {
parts.push(`_${message.reactions.map((reaction) => `:${reaction.name}: ${reaction.count}`).join(" ")}_`);
}
return parts.join("\n\n") || "_No message content_";
}
@@ -1,26 +1,41 @@
import { getNotificationHtmlUrl, Notification } from "../../../notification";
import { Detail, ActionPanel, Action } from "@raycast/api";
import { SlackReaction } from "../types";
import { useMemo } from "react";
import { PreviewDetail } from "../../../preview/PreviewDetail";
import { SlackReaction, SlackReactionState } from "../types";
import { Notification } from "../../../notification";
import { slackMessageToMarkdown } from "../markdown";
import { Color, Detail } from "@raycast/api";
import { match } from "ts-pattern";
interface SlackReactionPreviewProps {
notification: Notification;
slack_reaction: SlackReaction;
}
export function SlackReactionPreview({ notification }: SlackReactionPreviewProps) {
const notificationHtmlUrl = useMemo(() => {
return getNotificationHtmlUrl(notification);
}, [notification]);
return (
<Detail
markdown={`# ${notification.title}`}
actions={
<ActionPanel>
<Action.OpenInBrowser url={notificationHtmlUrl} />
</ActionPanel>
}
/>
);
function reactionContent(reaction: SlackReaction): { body: string; channel?: string } {
return match(reaction.item)
.with({ type: "Message" }, (item) => ({
body: slackMessageToMarkdown(item.content.message),
channel: item.content.channel.name,
}))
.with({ type: "File" }, (item) => ({ body: "_Reacted file_", channel: item.content.channel.name }))
.exhaustive();
}
export function SlackReactionPreview({ notification, slack_reaction }: SlackReactionPreviewProps) {
const { body, channel } = reactionContent(slack_reaction);
const markdown = `# ${notification.title}\n\n:${slack_reaction.name}:\n\n${body}`;
const metadata = (
<>
<Detail.Metadata.TagList title="State">
<Detail.Metadata.TagList.Item
text={slack_reaction.state === SlackReactionState.ReactionAdded ? "Added" : "Removed"}
color={slack_reaction.state === SlackReactionState.ReactionAdded ? Color.Green : Color.SecondaryText}
/>
</Detail.Metadata.TagList>
<Detail.Metadata.Label title="Reaction" text={`:${slack_reaction.name}:`} />
{channel ? <Detail.Metadata.Label title="Channel" text={`#${channel}`} /> : null}
</>
);
return <PreviewDetail notification={notification} markdown={markdown} metadata={metadata} />;
}
@@ -1,26 +1,47 @@
import { getNotificationHtmlUrl, Notification } from "../../../notification";
import { Detail, ActionPanel, Action } from "@raycast/api";
import { SlackStar } from "../types";
import { useMemo } from "react";
import { PreviewDetail } from "../../../preview/PreviewDetail";
import { Notification } from "../../../notification";
import { slackMessageToMarkdown } from "../markdown";
import { SlackStar, SlackStarState } from "../types";
import { Color, Detail } from "@raycast/api";
import { match } from "ts-pattern";
interface SlackStarPreviewProps {
notification: Notification;
slack_star: SlackStar;
}
export function SlackStarPreview({ notification }: SlackStarPreviewProps) {
const notificationHtmlUrl = useMemo(() => {
return getNotificationHtmlUrl(notification);
}, [notification]);
return (
<Detail
markdown={`# ${notification.title}`}
actions={
<ActionPanel>
<Action.OpenInBrowser url={notificationHtmlUrl} />
</ActionPanel>
}
/>
);
function starContent(star: SlackStar): { body: string; channel?: string } {
return match(star.item)
.with({ type: "Message" }, (item) => ({
body: slackMessageToMarkdown(item.content.message),
channel: item.content.channel.name,
}))
.with({ type: "File" }, (item) => ({ body: "_Starred file_", channel: item.content.channel.name }))
.with({ type: "FileComment" }, (item) => ({ body: "_File comment_", channel: item.content.channel.name }))
.with({ type: "Channel" }, (item) => ({
body: `_Channel_ #${item.content.channel.name ?? item.content.channel.id}`,
channel: item.content.channel.name,
}))
.with({ type: "Im" }, (item) => ({ body: "_Direct message_", channel: item.content.channel.name }))
.with({ type: "Group" }, (item) => ({ body: "_Group message_", channel: item.content.channel.name }))
.exhaustive();
}
export function SlackStarPreview({ notification, slack_star }: SlackStarPreviewProps) {
const { body, channel } = starContent(slack_star);
const markdown = `# ${notification.title}\n\n${body}`;
const metadata = (
<>
<Detail.Metadata.TagList title="State">
<Detail.Metadata.TagList.Item
text={slack_star.state === SlackStarState.StarAdded ? "Starred" : "Unstarred"}
color={slack_star.state === SlackStarState.StarAdded ? Color.Yellow : Color.SecondaryText}
/>
</Detail.Metadata.TagList>
{channel ? <Detail.Metadata.Label title="Channel" text={`#${channel}`} /> : null}
</>
);
return <PreviewDetail notification={notification} markdown={markdown} metadata={metadata} />;
}
@@ -1,26 +1,36 @@
import { getNotificationHtmlUrl, Notification } from "../../../notification";
import { Detail, ActionPanel, Action } from "@raycast/api";
import { slackMessageToMarkdown, slackSenderName } from "../markdown";
import { PreviewDetail } from "../../../preview/PreviewDetail";
import { Notification } from "../../../notification";
import { formatElapsedTime } from "../../../utils";
import { SlackThread } from "../types";
import { useMemo } from "react";
import { Detail } from "@raycast/api";
interface SlackThreadPreviewProps {
notification: Notification;
slack_thread: SlackThread;
}
export function SlackThreadPreview({ notification }: SlackThreadPreviewProps) {
const notificationHtmlUrl = useMemo(() => {
return getNotificationHtmlUrl(notification);
}, [notification]);
export function SlackThreadPreview({ notification, slack_thread }: SlackThreadPreviewProps) {
const channelName = slack_thread.channel.name ?? slack_thread.channel.id;
return (
<Detail
markdown={`# ${notification.title}`}
actions={
<ActionPanel>
<Action.OpenInBrowser url={notificationHtmlUrl} />
</ActionPanel>
}
/>
const body = slack_thread.messages
.map((message) => {
const sender = slackSenderName(message, slack_thread.sender_profiles);
const when = message.ts ? ` · ${formatElapsedTime(new Date(parseFloat(message.ts) * 1000))}` : "";
return `**${sender}**${when}\n\n${slackMessageToMarkdown(message, slack_thread.references)}`;
})
.join("\n\n---\n\n");
const markdown = `# ${notification.title}\n\n${body}`;
const metadata = (
<>
<Detail.Metadata.Link title="Channel" target={slack_thread.url} text={`#${channelName}`} />
{slack_thread.team.name ? <Detail.Metadata.Label title="Team" text={slack_thread.team.name} /> : null}
<Detail.Metadata.Label title="Messages" text={`${slack_thread.messages.length}`} />
<Detail.Metadata.Label title="Subscribed" text={slack_thread.subscribed ? "Yes" : "No"} />
</>
);
return <PreviewDetail notification={notification} markdown={markdown} metadata={metadata} />;
}
@@ -0,0 +1,52 @@
import { NotificationTaskActions } from "../../../action/NotificationTaskActions";
import { TickTickTaskPreview } from "../preview/TickTickTaskPreview";
import { NotificationListItemProps } from "../../../notification";
import { Icon, List, Color } from "@raycast/api";
import { TaskPriority } from "../../../task";
import { match } from "ts-pattern";
import dayjs from "dayjs";
export function TickTickNotificationListItem({ notification, mutate }: NotificationListItemProps) {
const dueAt = notification.task?.due_at?.content;
const subtitle = dueAt ? dayjs(dueAt).format("YYYY-MM-DD") : undefined;
const color = match(notification)
.with({ task: { priority: TaskPriority.P1 } }, () => Color.Red)
.with({ task: { priority: TaskPriority.P2 } }, () => Color.Orange)
.with({ task: { priority: TaskPriority.P3 } }, () => Color.Yellow)
.otherwise(() => null);
const accessories: List.Item.Accessory[] = [
{
icon: { source: Icon.Circle, tintColor: color },
},
{
date: new Date(notification.updated_at),
tooltip: `Updated at ${notification.updated_at}`,
},
];
const task = notification.task;
if (task) {
for (const tag of task.tags) {
accessories.unshift({ tag: { value: tag } });
}
}
return (
<List.Item
key={notification.id}
title={notification.title}
icon={Icon.CheckCircle}
subtitle={subtitle}
accessories={accessories}
actions={
<NotificationTaskActions
notification={notification}
detailsTarget={<TickTickTaskPreview notification={notification} />}
mutate={mutate}
/>
}
/>
);
}
@@ -0,0 +1,34 @@
import { PreviewDetail } from "../../../preview/PreviewDetail";
import { TaskMetadata } from "../../../preview/TaskMetadata";
import { Notification } from "../../../notification";
import { TickTickTaskStatus } from "../types";
interface TickTickTaskPreviewProps {
notification: Notification;
}
export function TickTickTaskPreview({ notification }: TickTickTaskPreviewProps) {
const task = notification.task;
const item =
notification.source_item.data.type === "TickTickItem" ? notification.source_item.data.content : undefined;
const title = task?.title ?? notification.title;
let markdown = `# ${title}`;
if (task?.body) {
markdown += `\n\n${task.body}`;
}
if (item?.items?.length) {
const checklist = item.items
.map((entry) => `- [${entry.status === TickTickTaskStatus.Completed ? "x" : " "}] ${entry.title}`)
.join("\n");
markdown += `\n\n## Checklist\n\n${checklist}`;
}
return (
<PreviewDetail
notification={notification}
markdown={markdown}
metadata={task ? <TaskMetadata task={task} /> : undefined}
/>
);
}
+41
View File
@@ -0,0 +1,41 @@
export enum TickTickItemPriority {
None = 0,
Low = 1,
Medium = 3,
High = 5,
}
export enum TickTickTaskStatus {
Normal = 0,
Completed = 2,
}
export interface TickTickTag {
name: string;
color?: string;
}
export interface TickTickChecklistItem {
id: string;
title: string;
status: TickTickTaskStatus;
sortOrder?: number;
}
export interface TickTickItem {
id: string;
projectId: string;
title: string;
content?: string;
allDay?: boolean;
startDate?: string;
dueDate?: string;
priority: TickTickItemPriority;
status: TickTickTaskStatus;
tags?: TickTickTag[];
items?: TickTickChecklistItem[];
}
export function getTickTickItemHtmlUrl(item: TickTickItem): string {
return `https://ticktick.com/webapp/#p/${item.projectId}/tasks/${item.id}`;
}
@@ -1,24 +1,25 @@
import { getNotificationHtmlUrl, Notification } from "../../../notification";
import { Detail, ActionPanel, Action } from "@raycast/api";
import { useMemo } from "react";
import { PreviewDetail } from "../../../preview/PreviewDetail";
import { TaskMetadata } from "../../../preview/TaskMetadata";
import { Notification } from "../../../notification";
interface TodoistTaskPreviewProps {
notification: Notification;
}
export function TodoistTaskPreview({ notification }: TodoistTaskPreviewProps) {
const notificationHtmlUrl = useMemo(() => {
return getNotificationHtmlUrl(notification);
}, [notification]);
const task = notification.task;
const title = task?.title ?? notification.title;
let markdown = `# ${title}`;
if (task?.body) {
markdown += `\n\n${task.body}`;
}
return (
<Detail
markdown={`# ${notification.title}`}
actions={
<ActionPanel>
<Action.OpenInBrowser url={notificationHtmlUrl} />
</ActionPanel>
}
<PreviewDetail
notification={notification}
markdown={markdown}
metadata={task ? <TaskMetadata task={task} /> : undefined}
/>
);
}
+5 -1
View File
@@ -20,9 +20,13 @@ export interface Notification {
export enum NotificationSourceKind {
Github = "Github",
Todoist = "Todoist",
TickTick = "TickTick",
Linear = "Linear",
GoogleMail = "GoogleMail",
GoogleCalendar = "GoogleCalendar",
GoogleDrive = "GoogleDrive",
Slack = "Slack",
API = "API",
}
export enum NotificationStatus {
@@ -42,5 +46,5 @@ export function getNotificationHtmlUrl(notification: Notification): string {
}
export function isNotificationBuiltFromTask(notification: Notification) {
return notification.kind === NotificationSourceKind.Todoist;
return notification.kind === NotificationSourceKind.Todoist || notification.kind === NotificationSourceKind.TickTick;
}
+38
View File
@@ -0,0 +1,38 @@
import { getNotificationHtmlUrl, Notification } from "../notification";
import { getThirdPartyItemSourceLabel } from "../third_party_item";
import { Detail, ActionPanel, Action } from "@raycast/api";
import { ReactNode, useMemo } from "react";
interface PreviewDetailProps {
notification: Notification;
markdown: string;
/** Inner `Detail.Metadata.*` items — rendered below the notification Type row. */
metadata?: ReactNode;
}
/**
* Shared wrapper for every notification preview: renders a `Detail` with the
* standard "Open in Browser" action, and a metadata sidebar whose first row is
* always the notification type, followed by the preview-specific `metadata`.
*/
export function PreviewDetail({ notification, markdown, metadata }: PreviewDetailProps) {
const notificationHtmlUrl = useMemo(() => getNotificationHtmlUrl(notification), [notification]);
return (
<Detail
markdown={markdown}
metadata={
<Detail.Metadata>
<Detail.Metadata.Label title="Type" text={getThirdPartyItemSourceLabel(notification.source_item)} />
{metadata ? <Detail.Metadata.Separator /> : null}
{metadata}
</Detail.Metadata>
}
actions={
<ActionPanel>
<Action.OpenInBrowser url={notificationHtmlUrl} />
</ActionPanel>
}
/>
);
}
+38
View File
@@ -0,0 +1,38 @@
import { Task, TaskPriority, TaskStatus } from "../task";
import { Color, Detail } from "@raycast/api";
import { formatDate } from "../utils";
const PRIORITY_COLOR: Record<TaskPriority, Color> = {
[TaskPriority.P1]: Color.Red,
[TaskPriority.P2]: Color.Orange,
[TaskPriority.P3]: Color.Yellow,
[TaskPriority.P4]: Color.SecondaryText,
};
/** Metadata sidebar shared by the Todoist and TickTick task previews. */
export function TaskMetadata({ task }: { task: Task }) {
const due = task.due_at?.content ? formatDate(task.due_at.content) : undefined;
return (
<>
<Detail.Metadata.TagList title="Status">
<Detail.Metadata.TagList.Item
text={task.status}
color={task.status === TaskStatus.Done ? Color.Green : Color.Blue}
/>
</Detail.Metadata.TagList>
<Detail.Metadata.TagList title="Priority">
<Detail.Metadata.TagList.Item text={`P${task.priority}`} color={PRIORITY_COLOR[task.priority]} />
</Detail.Metadata.TagList>
{due ? <Detail.Metadata.Label title="Due" text={due} /> : null}
{task.project ? <Detail.Metadata.Label title="Project" text={task.project} /> : null}
{task.tags.length > 0 ? (
<Detail.Metadata.TagList title="Tags">
{task.tags.map((tag) => (
<Detail.Metadata.TagList.Item key={tag} text={tag} />
))}
</Detail.Metadata.TagList>
) : null}
</>
);
}
+55 -1
View File
@@ -12,7 +12,11 @@ import {
LinearIssue,
LinearNotification,
} from "./integrations/linear/types";
import { getGoogleCalendarEventHtmlUrl, GoogleCalendarEvent } from "./integrations/google-calendar/types";
import { getGoogleDriveCommentHtmlUrl, GoogleDriveComment } from "./integrations/google-drive/types";
import { getGithubNotificationHtmlUrl, GithubNotification } from "./integrations/github/types";
import { getTickTickItemHtmlUrl, TickTickItem } from "./integrations/ticktick/types";
import { getWebPageHtmlUrl, WebPage } from "./integrations/api/types";
import { GoogleMailThread } from "./integrations/google-mail/types";
import { TodoistItem } from "./integrations/todoist/types";
@@ -28,13 +32,55 @@ export interface ThirdPartyItem {
export type ThirdPartyItemData =
| { type: "TodoistItem"; content: TodoistItem }
| { type: "TickTickItem"; content: TickTickItem }
| { type: "SlackStar"; content: SlackStar }
| { type: "SlackReaction"; content: SlackReaction }
| { type: "SlackThread"; content: SlackThread }
| { type: "LinearIssue"; content: LinearIssue }
| { type: "LinearNotification"; content: LinearNotification }
| { type: "GithubNotification"; content: GithubNotification }
| { type: "GoogleMailThread"; content: GoogleMailThread };
| { type: "GoogleMailThread"; content: GoogleMailThread }
| { type: "GoogleCalendarEvent"; content: GoogleCalendarEvent }
| { type: "GoogleDriveComment"; content: GoogleDriveComment }
| { type: "WebPage"; content: WebPage };
/** Human-readable label for the notification's source item type (e.g. "Linear Issue", "GitHub Pull Request"). */
export function getThirdPartyItemSourceLabel(thirdPartyItem: ThirdPartyItem): string {
switch (thirdPartyItem.data.type) {
case "TodoistItem":
return "Todoist Task";
case "TickTickItem":
return "TickTick Task";
case "SlackStar":
return "Slack Star";
case "SlackReaction":
return "Slack Reaction";
case "SlackThread":
return "Slack Thread";
case "LinearIssue":
return "Linear Issue";
case "LinearNotification":
return thirdPartyItem.data.content.type === "ProjectNotification" ? "Linear Project" : "Linear Issue";
case "GithubNotification": {
switch (thirdPartyItem.data.content.item?.type) {
case "GithubPullRequest":
return "GitHub Pull Request";
case "GithubDiscussion":
return "GitHub Discussion";
default:
return "GitHub";
}
}
case "GoogleMailThread":
return "Google Mail";
case "GoogleCalendarEvent":
return "Google Calendar Event";
case "GoogleDriveComment":
return "Google Drive Comment";
case "WebPage":
return "Web Page";
}
}
export function getThirdPartyItemHtmlUrl(thirdPartyItem: ThirdPartyItem): string {
switch (thirdPartyItem.data.type) {
@@ -54,5 +100,13 @@ export function getThirdPartyItemHtmlUrl(thirdPartyItem: ThirdPartyItem): string
return getGithubNotificationHtmlUrl(thirdPartyItem.data.content);
case "GoogleMailThread":
return `https://mail.google.com/mail/u/0/#inbox/${thirdPartyItem.data.content.id}`;
case "TickTickItem":
return getTickTickItemHtmlUrl(thirdPartyItem.data.content);
case "GoogleCalendarEvent":
return getGoogleCalendarEventHtmlUrl(thirdPartyItem.data.content);
case "GoogleDriveComment":
return getGoogleDriveCommentHtmlUrl(thirdPartyItem.data.content);
case "WebPage":
return getWebPageHtmlUrl(thirdPartyItem.data.content);
}
}
+38
View File
@@ -0,0 +1,38 @@
import relativeTime from "dayjs/plugin/relativeTime";
import dayjs from "dayjs";
dayjs.extend(relativeTime);
export function formatElapsedTime(date: Date | string): string {
return dayjs(date).fromNow();
}
export function formatDate(date: Date | string): string {
return dayjs(date).format("YYYY-MM-DD");
}
export function formatDateTime(date: Date | string): string {
return dayjs(date).format("YYYY-MM-DD HH:mm");
}
/**
* Convert an HTML-flavoured body (as GitHub sometimes returns) into readable
* markdown: strips comments and tags, turns block elements into line breaks,
* and decodes common entities. Preserves markdown autolinks like <https://…>.
*/
export function cleanHtml(text: string): string {
return text
.replace(/<!--[\s\S]*?-->/g, "")
.replace(/<br\s*\/?>/gi, "\n")
.replace(/<\/(?:p|div|li|h[1-6]|tr|ul|ol|blockquote)>/gi, "\n\n")
.replace(/<li[^>]*>/gi, "- ")
.replace(/<\/?[a-zA-Z][a-zA-Z0-9-]*(?:\s[^>]*)?>/g, "")
.replace(/&nbsp;/g, " ")
.replace(/&amp;/g, "&")
.replace(/&lt;/g, "<")
.replace(/&gt;/g, ">")
.replace(/&quot;/g, '"')
.replace(/&#39;/g, "'")
.replace(/\n{3,}/g, "\n\n")
.trim();
}