github-issue-tower-defence-management 1.90.0 → 1.91.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (155) hide show
  1. package/CHANGELOG.md +7 -0
  2. package/README.md +14 -1
  3. package/bin/adapter/entry-points/cli/index.js +16 -12
  4. package/bin/adapter/entry-points/cli/index.js.map +1 -1
  5. package/bin/adapter/entry-points/cli/projectConfig.js +2 -0
  6. package/bin/adapter/entry-points/cli/projectConfig.js.map +1 -1
  7. package/bin/adapter/entry-points/console/consoleOperationApi.js +54 -27
  8. package/bin/adapter/entry-points/console/consoleOperationApi.js.map +1 -1
  9. package/bin/adapter/entry-points/console/consoleProjectResolver.js +38 -0
  10. package/bin/adapter/entry-points/console/consoleProjectResolver.js.map +1 -0
  11. package/bin/adapter/entry-points/console/consoleServer.js +3 -4
  12. package/bin/adapter/entry-points/console/consoleServer.js.map +1 -1
  13. package/bin/adapter/entry-points/console/ui-dist/assets/index-BU6p3cGU.css +1 -0
  14. package/bin/adapter/entry-points/console/ui-dist/assets/index-BvuSQN9s.js +100 -0
  15. package/bin/adapter/entry-points/console/ui-dist/index.html +2 -2
  16. package/bin/adapter/repositories/issue/ApiV3CheerioRestIssueRepository.js +16 -0
  17. package/bin/adapter/repositories/issue/ApiV3CheerioRestIssueRepository.js.map +1 -1
  18. package/jest.config.js +57 -9
  19. package/package.json +17 -13
  20. package/src/adapter/entry-points/cli/index.test.ts +12 -2
  21. package/src/adapter/entry-points/cli/index.ts +30 -12
  22. package/src/adapter/entry-points/cli/projectConfig.ts +3 -0
  23. package/src/adapter/entry-points/console/consoleOperationApi.test.ts +129 -15
  24. package/src/adapter/entry-points/console/consoleOperationApi.ts +83 -28
  25. package/src/adapter/entry-points/console/consoleProjectResolver.test.ts +96 -0
  26. package/src/adapter/entry-points/console/consoleProjectResolver.ts +50 -0
  27. package/src/adapter/entry-points/console/consoleServer.test.ts +5 -4
  28. package/src/adapter/entry-points/console/consoleServer.ts +5 -7
  29. package/src/adapter/entry-points/console/ui/jest.setup.ts +1 -0
  30. package/src/adapter/entry-points/console/ui/jest.styleMock.js +1 -0
  31. package/src/adapter/entry-points/console/ui/src/features/console/colors.test.ts +34 -0
  32. package/src/adapter/entry-points/console/ui/src/features/console/colors.ts +73 -0
  33. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleChangedFileList.stories.tsx +28 -0
  34. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleChangedFileList.test.tsx +42 -0
  35. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleChangedFileList.tsx +55 -0
  36. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleCloseButtonGroup.stories.tsx +14 -0
  37. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleCloseButtonGroup.test.tsx +23 -0
  38. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleCloseButtonGroup.tsx +26 -0
  39. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleCommentList.stories.tsx +29 -0
  40. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleCommentList.test.tsx +55 -0
  41. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleCommentList.tsx +66 -0
  42. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleCommitList.stories.tsx +25 -0
  43. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleCommitList.test.tsx +53 -0
  44. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleCommitList.tsx +53 -0
  45. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleItemDetail.stories.tsx +79 -0
  46. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleItemDetail.test.tsx +81 -0
  47. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleItemDetail.tsx +226 -0
  48. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleItemIcon.stories.tsx +82 -0
  49. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleItemIcon.test.tsx +52 -0
  50. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleItemIcon.tsx +32 -0
  51. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleListItemRow.stories.tsx +25 -0
  52. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleListItemRow.test.tsx +43 -0
  53. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleListItemRow.tsx +34 -0
  54. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleListView.stories.tsx +22 -6
  55. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleListView.test.tsx +87 -0
  56. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleListView.tsx +36 -32
  57. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleMarkdownView.stories.tsx +27 -0
  58. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleMarkdownView.test.tsx +36 -0
  59. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleMarkdownView.tsx +50 -0
  60. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleMermaidDiagram.stories.tsx +22 -0
  61. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleMermaidDiagram.test.tsx +38 -0
  62. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleMermaidDiagram.tsx +65 -0
  63. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleNextActionDateGroup.stories.tsx +20 -0
  64. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleNextActionDateGroup.test.tsx +42 -0
  65. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleNextActionDateGroup.tsx +28 -0
  66. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleOperationBar.stories.tsx +55 -0
  67. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleOperationBar.test.tsx +85 -0
  68. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleOperationBar.tsx +55 -0
  69. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsolePanel.stories.tsx +26 -0
  70. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsolePanel.test.tsx +32 -0
  71. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsolePanel.tsx +36 -0
  72. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleProjectHeader.test.tsx +14 -0
  73. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsolePullRequestReviewGroup.stories.tsx +14 -0
  74. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsolePullRequestReviewGroup.test.tsx +33 -0
  75. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsolePullRequestReviewGroup.tsx +34 -0
  76. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsolePullRequestSection.stories.tsx +31 -0
  77. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsolePullRequestSection.test.tsx +40 -0
  78. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsolePullRequestSection.tsx +88 -0
  79. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleStatusButtonGroup.stories.tsx +17 -0
  80. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleStatusButtonGroup.test.tsx +49 -0
  81. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleStatusButtonGroup.tsx +63 -0
  82. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleStoryButtonGroup.stories.tsx +17 -0
  83. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleStoryButtonGroup.test.tsx +45 -0
  84. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleStoryButtonGroup.tsx +42 -0
  85. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleStoryGroupHeader.stories.tsx +27 -0
  86. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleStoryGroupHeader.test.tsx +24 -0
  87. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleStoryGroupHeader.tsx +28 -0
  88. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleTabBar.stories.tsx +41 -5
  89. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleTabBar.test.tsx +59 -0
  90. package/src/adapter/entry-points/console/ui/src/features/console/components/ConsoleTabBar.tsx +28 -19
  91. package/src/adapter/entry-points/console/ui/src/features/console/fileStatus.test.ts +35 -0
  92. package/src/adapter/entry-points/console/ui/src/features/console/fileStatus.ts +21 -0
  93. package/src/adapter/entry-points/console/ui/src/features/console/fixtures.ts +206 -9
  94. package/src/adapter/entry-points/console/ui/src/features/console/grouping.test.ts +91 -0
  95. package/src/adapter/entry-points/console/ui/src/features/console/grouping.ts +79 -0
  96. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleCaches.test.ts +22 -0
  97. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleCaches.ts +42 -0
  98. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleItemDetailData.test.ts +126 -0
  99. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleItemDetailData.ts +167 -0
  100. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleOperations.test.ts +198 -0
  101. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleOperations.ts +243 -0
  102. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleOverlay.test.ts +40 -0
  103. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleOverlay.ts +71 -0
  104. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleResource.test.ts +41 -0
  105. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleResource.ts +57 -0
  106. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleTabData.test.ts +63 -0
  107. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleTabData.ts +129 -0
  108. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleToken.test.ts +41 -0
  109. package/src/adapter/entry-points/console/ui/src/features/console/itemIcons.test.ts +97 -0
  110. package/src/adapter/entry-points/console/ui/src/features/console/itemIcons.ts +95 -0
  111. package/src/adapter/entry-points/console/ui/src/features/console/lib/consoleApi.test.ts +155 -0
  112. package/src/adapter/entry-points/console/ui/src/features/console/lib/consoleApi.ts +187 -0
  113. package/src/adapter/entry-points/console/ui/src/features/console/lib/markdown.test.ts +76 -0
  114. package/src/adapter/entry-points/console/ui/src/features/console/lib/markdown.ts +73 -0
  115. package/src/adapter/entry-points/console/ui/src/features/console/lib/mermaidLoader.test.ts +27 -0
  116. package/src/adapter/entry-points/console/ui/src/features/console/lib/mermaidLoader.ts +71 -0
  117. package/src/adapter/entry-points/console/ui/src/features/console/lib/resourceCache.test.ts +56 -0
  118. package/src/adapter/entry-points/console/ui/src/features/console/lib/resourceCache.ts +51 -0
  119. package/src/adapter/entry-points/console/ui/src/features/console/operations.test.ts +37 -0
  120. package/src/adapter/entry-points/console/ui/src/features/console/operations.ts +35 -0
  121. package/src/adapter/entry-points/console/ui/src/features/console/overlay.test.ts +124 -0
  122. package/src/adapter/entry-points/console/ui/src/features/console/overlay.ts +101 -0
  123. package/src/adapter/entry-points/console/ui/src/features/console/pages/ConsoleItemDetailContainer.test.tsx +79 -0
  124. package/src/adapter/entry-points/console/ui/src/features/console/pages/ConsoleItemDetailContainer.tsx +109 -0
  125. package/src/adapter/entry-points/console/ui/src/features/console/pages/ConsolePage.test.tsx +74 -0
  126. package/src/adapter/entry-points/console/ui/src/features/console/pages/ConsolePage.tsx +133 -7
  127. package/src/adapter/entry-points/console/ui/src/features/console/relativeTime.test.ts +52 -0
  128. package/src/adapter/entry-points/console/ui/src/features/console/relativeTime.ts +51 -0
  129. package/src/adapter/entry-points/console/ui/src/features/console/types.ts +73 -1
  130. package/src/adapter/entry-points/console/ui/src/index.css +352 -2
  131. package/src/adapter/entry-points/console/ui/tsconfig.json +1 -0
  132. package/src/adapter/entry-points/console/ui/vite.config.ts +5 -0
  133. package/src/adapter/entry-points/console/ui-dist/assets/index-BU6p3cGU.css +1 -0
  134. package/src/adapter/entry-points/console/ui-dist/assets/index-BvuSQN9s.js +100 -0
  135. package/src/adapter/entry-points/console/ui-dist/index.html +2 -2
  136. package/src/adapter/repositories/issue/ApiV3CheerioRestIssueRepository.ts +25 -0
  137. package/src/domain/usecases/adapter-interfaces/IssueRepository.ts +4 -0
  138. package/types/adapter/entry-points/cli/index.d.ts.map +1 -1
  139. package/types/adapter/entry-points/cli/projectConfig.d.ts +1 -0
  140. package/types/adapter/entry-points/cli/projectConfig.d.ts.map +1 -1
  141. package/types/adapter/entry-points/console/consoleOperationApi.d.ts +6 -2
  142. package/types/adapter/entry-points/console/consoleOperationApi.d.ts.map +1 -1
  143. package/types/adapter/entry-points/console/consoleProjectResolver.d.ts +6 -0
  144. package/types/adapter/entry-points/console/consoleProjectResolver.d.ts.map +1 -0
  145. package/types/adapter/entry-points/console/consoleServer.d.ts +2 -3
  146. package/types/adapter/entry-points/console/consoleServer.d.ts.map +1 -1
  147. package/types/adapter/repositories/issue/ApiV3CheerioRestIssueRepository.d.ts +1 -0
  148. package/types/adapter/repositories/issue/ApiV3CheerioRestIssueRepository.d.ts.map +1 -1
  149. package/types/domain/usecases/adapter-interfaces/IssueRepository.d.ts +1 -0
  150. package/types/domain/usecases/adapter-interfaces/IssueRepository.d.ts.map +1 -1
  151. package/bin/adapter/entry-points/console/ui-dist/assets/index-DDjYPXRT.js +0 -49
  152. package/bin/adapter/entry-points/console/ui-dist/assets/index-DHlBLm7d.css +0 -1
  153. package/src/adapter/entry-points/console/ui/src/features/console/hooks/useConsoleList.ts +0 -78
  154. package/src/adapter/entry-points/console/ui-dist/assets/index-DDjYPXRT.js +0 -49
  155. package/src/adapter/entry-points/console/ui-dist/assets/index-DHlBLm7d.css +0 -1
@@ -1,23 +1,149 @@
1
- import { useState } from 'react';
1
+ import { useMemo, useState } from 'react';
2
2
  import { ConsoleListView } from '../components/ConsoleListView';
3
3
  import { ConsoleProjectHeader } from '../components/ConsoleProjectHeader';
4
4
  import { ConsoleTabBar } from '../components/ConsoleTabBar';
5
- import { useConsoleList } from '../hooks/useConsoleList';
5
+ import { buildConsoleListRows, resolveItemStory } from '../grouping';
6
+ import { useConsoleCaches } from '../hooks/useConsoleCaches';
7
+ import { useConsoleOperations } from '../hooks/useConsoleOperations';
8
+ import { useConsoleOverlay } from '../hooks/useConsoleOverlay';
6
9
  import { useConsolePjcode } from '../hooks/useConsolePjcode';
7
- import { CONSOLE_TABS, type ConsoleTabName } from '../types';
10
+ import { useConsoleTabData } from '../hooks/useConsoleTabData';
11
+ import {
12
+ countPendingItems,
13
+ filterPendingItems,
14
+ overlayKeyForItem,
15
+ parseGeneratedAtMs,
16
+ } from '../overlay';
17
+ import type {
18
+ ConsoleListItem,
19
+ ConsoleOverlayStatus,
20
+ ConsoleTabName,
21
+ } from '../types';
22
+ import { CONSOLE_TABS } from '../types';
23
+ import { ConsoleItemDetailContainer } from './ConsoleItemDetailContainer';
24
+
25
+ const emptyCounts = (): Record<ConsoleTabName, number> => {
26
+ const result = {} as Record<ConsoleTabName, number>;
27
+ for (const tab of CONSOLE_TABS) {
28
+ result[tab.name] = 0;
29
+ }
30
+ return result;
31
+ };
32
+
33
+ const OVERLAY_NAMESPACE_FALLBACK = 'console';
8
34
 
9
35
  export const ConsolePage = () => {
10
36
  const pjcode = useConsolePjcode();
37
+ const { snapshots, isLoading, error } = useConsoleTabData(pjcode);
11
38
  const [activeTab, setActiveTab] = useState<ConsoleTabName>(
12
39
  CONSOLE_TABS[0].name,
13
40
  );
14
- const { items, isLoading, error } = useConsoleList(pjcode, activeTab);
41
+ const [selectedItem, setSelectedItem] = useState<ConsoleListItem | null>(
42
+ null,
43
+ );
44
+
45
+ const overlayState = useConsoleOverlay(pjcode ?? OVERLAY_NAMESPACE_FALLBACK);
46
+ const caches = useConsoleCaches();
47
+ const operations = useConsoleOperations(pjcode, activeTab, overlayState);
48
+ const now = Date.now();
49
+
50
+ const counts = useMemo(() => {
51
+ const result = emptyCounts();
52
+ for (const tab of CONSOLE_TABS) {
53
+ const snapshot = snapshots[tab.name];
54
+ if (snapshot === null) {
55
+ continue;
56
+ }
57
+ result[tab.name] = countPendingItems(
58
+ snapshot.items,
59
+ overlayState.overlay,
60
+ parseGeneratedAtMs(snapshot.generatedAt),
61
+ tab.name,
62
+ );
63
+ }
64
+ return result;
65
+ }, [snapshots, overlayState.overlay]);
66
+
67
+ const activeSnapshot = snapshots[activeTab];
68
+ const pendingItems = useMemo(() => {
69
+ if (activeSnapshot === null) {
70
+ return [];
71
+ }
72
+ return filterPendingItems(
73
+ activeSnapshot.items,
74
+ overlayState.overlay,
75
+ parseGeneratedAtMs(activeSnapshot.generatedAt),
76
+ activeTab,
77
+ );
78
+ }, [activeSnapshot, overlayState.overlay, activeTab]);
79
+
80
+ const rows = useMemo(
81
+ () => buildConsoleListRows(pendingItems, overlayState.overlay),
82
+ [pendingItems, overlayState.overlay],
83
+ );
84
+
85
+ const storyColors = activeSnapshot?.storyColors ?? {};
86
+ const statusOptions = activeSnapshot?.statusOptions ?? [];
87
+ const storyOptions = activeSnapshot?.storyOptions ?? [];
88
+
89
+ const selectTab = (tab: ConsoleTabName): void => {
90
+ setActiveTab(tab);
91
+ setSelectedItem(null);
92
+ };
93
+
94
+ const overlayStatusForSelected = ((): ConsoleOverlayStatus | null => {
95
+ if (selectedItem === null) {
96
+ return null;
97
+ }
98
+ const entry = overlayState.overlay[overlayKeyForItem(selectedItem)];
99
+ return entry?.status ?? null;
100
+ })();
101
+
102
+ const storyNameForSelected =
103
+ selectedItem !== null
104
+ ? resolveItemStory(selectedItem, overlayState.overlay)
105
+ : null;
15
106
 
16
107
  return (
17
- <main className="mx-auto flex max-w-3xl flex-col">
108
+ <main className="console-app">
18
109
  <ConsoleProjectHeader pjcode={pjcode} />
19
- <ConsoleTabBar activeTab={activeTab} onSelectTab={setActiveTab} />
20
- <ConsoleListView items={items} isLoading={isLoading} error={error} />
110
+ <ConsoleTabBar
111
+ activeTab={activeTab}
112
+ counts={counts}
113
+ onSelectTab={selectTab}
114
+ />
115
+ {selectedItem === null ? (
116
+ <ConsoleListView
117
+ rows={rows}
118
+ storyColors={storyColors}
119
+ activeItemId={null}
120
+ isLoading={isLoading}
121
+ error={error}
122
+ onSelectItem={setSelectedItem}
123
+ />
124
+ ) : (
125
+ <div className="console-detail-screen">
126
+ <button
127
+ type="button"
128
+ className="console-back-button"
129
+ onClick={() => setSelectedItem(null)}
130
+ >
131
+ ← Back to list
132
+ </button>
133
+ <ConsoleItemDetailContainer
134
+ tab={activeTab}
135
+ item={selectedItem}
136
+ caches={caches}
137
+ operations={operations}
138
+ statusOptions={statusOptions}
139
+ storyOptions={storyOptions}
140
+ storyColors={storyColors}
141
+ storyName={storyNameForSelected}
142
+ overlayStatus={overlayStatusForSelected}
143
+ now={now}
144
+ />
145
+ </div>
146
+ )}
21
147
  </main>
22
148
  );
23
149
  };
@@ -0,0 +1,52 @@
1
+ import { formatFullTimestamp, formatRelativeTime } from './relativeTime';
2
+
3
+ const now = Date.parse('2026-06-19T12:00:00.000Z');
4
+
5
+ describe('formatRelativeTime', () => {
6
+ it('returns just now for very recent times', () => {
7
+ expect(formatRelativeTime('2026-06-19T11:59:30.000Z', now)).toBe(
8
+ 'just now',
9
+ );
10
+ });
11
+
12
+ it('returns minutes ago', () => {
13
+ expect(formatRelativeTime('2026-06-19T11:45:00.000Z', now)).toBe(
14
+ '15 minutes ago',
15
+ );
16
+ });
17
+
18
+ it('uses singular for one minute', () => {
19
+ expect(formatRelativeTime('2026-06-19T11:59:00.000Z', now)).toBe(
20
+ '1 minute ago',
21
+ );
22
+ });
23
+
24
+ it('returns hours ago', () => {
25
+ expect(formatRelativeTime('2026-06-19T09:00:00.000Z', now)).toBe(
26
+ '3 hours ago',
27
+ );
28
+ });
29
+
30
+ it('returns yesterday for one day', () => {
31
+ expect(formatRelativeTime('2026-06-18T12:00:00.000Z', now)).toBe(
32
+ 'yesterday',
33
+ );
34
+ });
35
+
36
+ it('returns days ago for under a month', () => {
37
+ expect(formatRelativeTime('2026-06-09T12:00:00.000Z', now)).toBe(
38
+ '10 days ago',
39
+ );
40
+ });
41
+
42
+ it('returns an empty string for an invalid date', () => {
43
+ expect(formatRelativeTime('nonsense', now)).toBe('');
44
+ });
45
+ });
46
+
47
+ describe('formatFullTimestamp', () => {
48
+ it('formats a timestamp and returns empty for invalid input', () => {
49
+ expect(formatFullTimestamp('2026-06-19T12:00:00.000Z')).not.toBe('');
50
+ expect(formatFullTimestamp('nonsense')).toBe('');
51
+ });
52
+ });
@@ -0,0 +1,51 @@
1
+ const MS_PER_SECOND = 1000;
2
+ const MS_PER_MINUTE = 60 * MS_PER_SECOND;
3
+ const MS_PER_HOUR = 60 * MS_PER_MINUTE;
4
+ const MS_PER_DAY = 24 * MS_PER_HOUR;
5
+
6
+ export const formatRelativeTime = (iso: string, now: number): string => {
7
+ const then = Date.parse(iso);
8
+ if (Number.isNaN(then)) {
9
+ return '';
10
+ }
11
+ const diff = now - then;
12
+ if (diff < 45 * MS_PER_SECOND) {
13
+ return 'just now';
14
+ }
15
+ if (diff < MS_PER_HOUR) {
16
+ const minutes = Math.round(diff / MS_PER_MINUTE);
17
+ return `${minutes} minute${minutes === 1 ? '' : 's'} ago`;
18
+ }
19
+ if (diff < MS_PER_DAY) {
20
+ const hours = Math.round(diff / MS_PER_HOUR);
21
+ return `${hours} hour${hours === 1 ? '' : 's'} ago`;
22
+ }
23
+ const days = Math.round(diff / MS_PER_DAY);
24
+ if (days === 1) {
25
+ return 'yesterday';
26
+ }
27
+ if (days < 30) {
28
+ return `${days} days ago`;
29
+ }
30
+ const date = new Date(then);
31
+ const sameYear = date.getFullYear() === new Date(now).getFullYear();
32
+ return date.toLocaleDateString('en-US', {
33
+ year: sameYear ? undefined : 'numeric',
34
+ month: 'short',
35
+ day: 'numeric',
36
+ });
37
+ };
38
+
39
+ export const formatFullTimestamp = (iso: string): string => {
40
+ const then = Date.parse(iso);
41
+ if (Number.isNaN(then)) {
42
+ return '';
43
+ }
44
+ return new Date(then).toLocaleString('ja-JP', {
45
+ year: 'numeric',
46
+ month: '2-digit',
47
+ day: '2-digit',
48
+ hour: '2-digit',
49
+ minute: '2-digit',
50
+ });
51
+ };
@@ -48,6 +48,78 @@ export type ConsoleTriageTab = {
48
48
 
49
49
  export type ConsoleTabData = ConsoleStatusTab | ConsoleTriageTab;
50
50
 
51
+ export type ConsoleStoryColorSource = Record<
52
+ string,
53
+ ConsoleColor | { color: ConsoleColor }
54
+ >;
55
+
56
+ export type ConsoleIssueState = {
57
+ state: string;
58
+ merged: boolean;
59
+ isPullRequest: boolean;
60
+ };
61
+
62
+ export type ConsoleComment = {
63
+ author: string;
64
+ body: string;
65
+ createdAt: string;
66
+ };
67
+
68
+ export type ConsoleChangedFile = {
69
+ path: string;
70
+ additions: number;
71
+ deletions: number;
72
+ status: string;
73
+ patch: string | null;
74
+ };
75
+
76
+ export type ConsoleCommit = {
77
+ sha: string;
78
+ message: string;
79
+ author: string;
80
+ authoredAt: string;
81
+ };
82
+
83
+ export type ConsoleRelatedPullRequest = {
84
+ url: string;
85
+ branchName: string | null;
86
+ createdAt: string;
87
+ isDraft: boolean;
88
+ isConflicted: boolean;
89
+ isPassedAllCiJob: boolean;
90
+ isCiStateSuccess: boolean;
91
+ isResolvedAllReviewComments: boolean;
92
+ isBranchOutOfDate: boolean;
93
+ missingRequiredCheckNames: string[];
94
+ summary: {
95
+ title: string;
96
+ body: string;
97
+ additions: number;
98
+ deletions: number;
99
+ changedFiles: number;
100
+ } | null;
101
+ };
102
+
103
+ export type ConsoleOverlayStatus = {
104
+ name: string;
105
+ color: ConsoleColor;
106
+ };
107
+
108
+ export type ConsoleOverlayStory = {
109
+ name: string;
110
+ color: ConsoleColor;
111
+ };
112
+
113
+ export type ConsoleOverlayEntry = {
114
+ done?: boolean;
115
+ status?: ConsoleOverlayStatus;
116
+ story?: ConsoleOverlayStory;
117
+ ts: number;
118
+ mode: ConsoleTabName;
119
+ };
120
+
121
+ export type ConsoleOverlay = Record<string, ConsoleOverlayEntry>;
122
+
51
123
  export type ConsoleTabName =
52
124
  | 'prs'
53
125
  | 'triage'
@@ -65,5 +137,5 @@ export const CONSOLE_TABS: ConsoleTab[] = [
65
137
  { name: 'triage', label: 'Triage' },
66
138
  { name: 'unread', label: 'Unread' },
67
139
  { name: 'failed-preparation', label: 'Failed Preparation' },
68
- { name: 'todo-by-human', label: 'Todo By Human' },
140
+ { name: 'todo-by-human', label: 'Todo by human' },
69
141
  ];
@@ -23,9 +23,359 @@
23
23
 
24
24
  body {
25
25
  margin: 0;
26
- background-color: var(--color-background);
27
- color: var(--color-foreground);
26
+ background-color: #0d1117;
27
+ color: #e6edf3;
28
28
  font-family:
29
29
  ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI",
30
30
  Roboto, Helvetica, Arial, sans-serif;
31
31
  }
32
+
33
+ .console-app {
34
+ max-width: 920px;
35
+ margin: 0 auto;
36
+ display: flex;
37
+ flex-direction: column;
38
+ }
39
+
40
+ .console-tabbar {
41
+ display: flex;
42
+ flex-wrap: wrap;
43
+ gap: 4px;
44
+ padding: 8px;
45
+ border-bottom: 1px solid #30363d;
46
+ }
47
+
48
+ .console-tab {
49
+ display: inline-flex;
50
+ align-items: center;
51
+ gap: 6px;
52
+ padding: 4px 10px;
53
+ border: 1px solid transparent;
54
+ border-radius: 6px;
55
+ background: transparent;
56
+ color: #8b949e;
57
+ font-size: 13px;
58
+ cursor: pointer;
59
+ }
60
+
61
+ .console-tab[data-active="true"] {
62
+ background: #161b22;
63
+ border-color: #30363d;
64
+ color: #e6edf3;
65
+ font-weight: 600;
66
+ }
67
+
68
+ .console-tab-badge {
69
+ min-width: 18px;
70
+ text-align: center;
71
+ border-radius: 999px;
72
+ background: #21262d;
73
+ padding: 0 6px;
74
+ font-size: 12px;
75
+ }
76
+
77
+ .console-tab-badge[data-zero="true"] {
78
+ opacity: 0.5;
79
+ }
80
+
81
+ .console-list {
82
+ list-style: none;
83
+ margin: 0;
84
+ padding: 0;
85
+ }
86
+
87
+ .console-list-group {
88
+ list-style: none;
89
+ }
90
+
91
+ .console-group-header {
92
+ display: flex;
93
+ align-items: center;
94
+ justify-content: space-between;
95
+ padding: 6px 12px;
96
+ background: #0b0f14;
97
+ border-bottom: 1px solid #21262d;
98
+ }
99
+
100
+ .console-storytag {
101
+ display: inline-flex;
102
+ align-items: center;
103
+ gap: 8px;
104
+ font-size: 13px;
105
+ font-weight: 700;
106
+ }
107
+
108
+ .console-story-dot {
109
+ width: 10px;
110
+ height: 10px;
111
+ border-radius: 999px;
112
+ display: inline-block;
113
+ }
114
+
115
+ .console-group-count {
116
+ color: #8b949e;
117
+ font-size: 12px;
118
+ }
119
+
120
+ .console-item-row {
121
+ display: flex;
122
+ align-items: center;
123
+ gap: 8px;
124
+ width: 100%;
125
+ padding: 8px 12px;
126
+ background: transparent;
127
+ border: none;
128
+ border-bottom: 1px solid #21262d;
129
+ color: #e6edf3;
130
+ text-align: left;
131
+ cursor: pointer;
132
+ }
133
+
134
+ .console-item-row:hover,
135
+ .console-item-row[data-active="true"] {
136
+ background: #161b22;
137
+ }
138
+
139
+ .console-item-title {
140
+ flex: 1;
141
+ font-size: 14px;
142
+ }
143
+
144
+ .console-item-number {
145
+ color: #8b949e;
146
+ font-size: 12px;
147
+ }
148
+
149
+ .console-list-message {
150
+ padding: 16px;
151
+ color: #8b949e;
152
+ font-size: 14px;
153
+ }
154
+
155
+ .console-list-error,
156
+ .console-comment-error,
157
+ .console-files-error,
158
+ .console-commits-error,
159
+ .console-detail-body-error {
160
+ color: #f85149;
161
+ }
162
+
163
+ .console-detail {
164
+ display: flex;
165
+ flex-direction: column;
166
+ gap: 12px;
167
+ padding: 16px;
168
+ }
169
+
170
+ .console-detail-title {
171
+ display: flex;
172
+ align-items: center;
173
+ gap: 8px;
174
+ font-size: 20px;
175
+ margin: 0;
176
+ }
177
+
178
+ .console-detail-title-text {
179
+ flex: 1;
180
+ }
181
+
182
+ .console-detail-number {
183
+ color: #8b949e;
184
+ font-weight: 400;
185
+ }
186
+
187
+ .console-detail-closed-label {
188
+ color: #a371f7;
189
+ font-size: 13px;
190
+ }
191
+
192
+ .console-detail-subbar {
193
+ display: flex;
194
+ align-items: center;
195
+ gap: 12px;
196
+ font-size: 13px;
197
+ }
198
+
199
+ .console-detail-link {
200
+ color: #4493f8;
201
+ }
202
+
203
+ .console-detail-repo {
204
+ font-family: ui-monospace, SFMono-Regular, Menlo, monospace;
205
+ color: #8b949e;
206
+ }
207
+
208
+ .console-detail-pill,
209
+ .console-label-chip,
210
+ .console-detail-status-chip {
211
+ display: inline-block;
212
+ padding: 2px 8px;
213
+ border-radius: 999px;
214
+ border: 1px solid #30363d;
215
+ font-size: 12px;
216
+ }
217
+
218
+ .console-detail-labels {
219
+ display: flex;
220
+ flex-wrap: wrap;
221
+ gap: 6px;
222
+ }
223
+
224
+ .console-detail-createdat {
225
+ color: #8b949e;
226
+ font-size: 12px;
227
+ }
228
+
229
+ .console-panel {
230
+ border: 1px solid #30363d;
231
+ border-radius: 8px;
232
+ overflow: hidden;
233
+ }
234
+
235
+ .console-panel-header {
236
+ display: flex;
237
+ align-items: center;
238
+ justify-content: space-between;
239
+ background: #161b22;
240
+ padding: 6px 12px;
241
+ }
242
+
243
+ .console-panel-toggle {
244
+ display: inline-flex;
245
+ align-items: center;
246
+ gap: 8px;
247
+ background: transparent;
248
+ border: none;
249
+ color: #e6edf3;
250
+ font-size: 14px;
251
+ font-weight: 600;
252
+ cursor: pointer;
253
+ }
254
+
255
+ .console-panel-body {
256
+ padding: 12px;
257
+ }
258
+
259
+ .console-markdown {
260
+ font-size: 14px;
261
+ line-height: 1.5;
262
+ word-break: break-word;
263
+ }
264
+
265
+ .console-mermaid-error {
266
+ color: #f85149;
267
+ font-size: 13px;
268
+ }
269
+
270
+ .console-comment {
271
+ border-top: 1px solid #21262d;
272
+ padding-top: 8px;
273
+ }
274
+
275
+ .console-comment-header {
276
+ display: flex;
277
+ gap: 8px;
278
+ font-size: 12px;
279
+ color: #8b949e;
280
+ }
281
+
282
+ .console-comment-author {
283
+ font-weight: 600;
284
+ color: #e6edf3;
285
+ }
286
+
287
+ .console-files,
288
+ .console-commits {
289
+ list-style: none;
290
+ margin: 0;
291
+ padding: 0;
292
+ }
293
+
294
+ .console-file,
295
+ .console-commit {
296
+ display: flex;
297
+ align-items: center;
298
+ gap: 8px;
299
+ padding: 4px 0;
300
+ font-size: 13px;
301
+ }
302
+
303
+ .console-file-badge {
304
+ width: 18px;
305
+ text-align: center;
306
+ border: 1px solid;
307
+ border-radius: 4px;
308
+ font-size: 11px;
309
+ }
310
+
311
+ .console-file-path,
312
+ .console-commit-message {
313
+ flex: 1;
314
+ font-family: ui-monospace, SFMono-Regular, Menlo, monospace;
315
+ }
316
+
317
+ .console-file-add,
318
+ .console-pr-add {
319
+ color: #3fb950;
320
+ }
321
+
322
+ .console-file-del,
323
+ .console-pr-del {
324
+ color: #f85149;
325
+ }
326
+
327
+ .console-commit-sha {
328
+ font-family: ui-monospace, SFMono-Regular, Menlo, monospace;
329
+ color: #8b949e;
330
+ }
331
+
332
+ .console-operation-bar {
333
+ display: flex;
334
+ flex-direction: column;
335
+ gap: 10px;
336
+ border-top: 1px solid #30363d;
337
+ padding-top: 12px;
338
+ }
339
+
340
+ .console-op-group {
341
+ display: flex;
342
+ flex-wrap: wrap;
343
+ gap: 8px;
344
+ }
345
+
346
+ .console-op-button {
347
+ padding: 6px 12px;
348
+ border: 1px solid #30363d;
349
+ border-radius: 6px;
350
+ background: #161b22;
351
+ color: #e6edf3;
352
+ font-size: 13px;
353
+ cursor: pointer;
354
+ }
355
+
356
+ .console-pr-section {
357
+ border: 1px solid #30363d;
358
+ border-radius: 8px;
359
+ padding: 12px;
360
+ display: flex;
361
+ flex-direction: column;
362
+ gap: 10px;
363
+ }
364
+
365
+ .console-pr-statbar {
366
+ display: flex;
367
+ gap: 12px;
368
+ font-size: 12px;
369
+ color: #8b949e;
370
+ }
371
+
372
+ .console-back-button {
373
+ align-self: flex-start;
374
+ margin: 12px 16px 0;
375
+ background: transparent;
376
+ border: 1px solid #30363d;
377
+ border-radius: 6px;
378
+ color: #e6edf3;
379
+ padding: 4px 10px;
380
+ cursor: pointer;
381
+ }
@@ -16,6 +16,7 @@
16
16
  "noUnusedParameters": true,
17
17
  "noFallthroughCasesInSwitch": true,
18
18
  "noUncheckedSideEffectImports": true,
19
+ "types": ["jest", "node", "@testing-library/jest-dom"],
19
20
  "paths": {
20
21
  "@/*": ["./src/*"]
21
22
  }