popilot 0.6.0 → 0.8.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 (165) hide show
  1. package/bin/cli.mjs +204 -2
  2. package/lib/doctor.mjs +38 -1
  3. package/lib/hydrate.mjs +15 -0
  4. package/lib/scaffold.mjs +5 -0
  5. package/lib/setup-wizard.mjs +35 -2
  6. package/package.json +1 -1
  7. package/scaffold/.context/project.yaml.example +19 -0
  8. package/scaffold/mcp-notification-server/package.json +18 -0
  9. package/scaffold/mcp-notification-server/src/index.ts +275 -0
  10. package/scaffold/mcp-notification-server/src/turso-client.ts +142 -0
  11. package/scaffold/mcp-notification-server/tsconfig.json +14 -0
  12. package/scaffold/mcp-pm/package.json +19 -0
  13. package/scaffold/mcp-pm/src/api-client.ts +69 -0
  14. package/scaffold/mcp-pm/src/index.ts +660 -0
  15. package/scaffold/mcp-pm/tsconfig.json +14 -0
  16. package/scaffold/pm-api/package.json +21 -0
  17. package/scaffold/pm-api/sql/001-memo-v2.sql +49 -0
  18. package/scaffold/pm-api/sql/002-notifications.sql +18 -0
  19. package/scaffold/pm-api/sql/003-content.sql +66 -0
  20. package/scaffold/pm-api/sql/004-agent-events.sql +21 -0
  21. package/scaffold/pm-api/sql/005-epic-sprint-decoupling.sql +6 -0
  22. package/scaffold/pm-api/sql/schema-core.sql +331 -0
  23. package/scaffold/pm-api/sql/schema-docs.sql +25 -0
  24. package/scaffold/pm-api/sql/schema-meetings.sql +17 -0
  25. package/scaffold/pm-api/sql/schema-rewards.sql +16 -0
  26. package/scaffold/pm-api/src/auth.ts +28 -0
  27. package/scaffold/pm-api/src/blockchain/adapter.ts +20 -0
  28. package/scaffold/pm-api/src/blockchain/tron.ts +62 -0
  29. package/scaffold/pm-api/src/db/adapter.ts +36 -0
  30. package/scaffold/pm-api/src/db/turso.ts +147 -0
  31. package/scaffold/pm-api/src/index.ts +114 -0
  32. package/scaffold/pm-api/src/mcp-tools/dashboard.ts +40 -0
  33. package/scaffold/pm-api/src/mcp-tools/epic.ts +67 -0
  34. package/scaffold/pm-api/src/mcp-tools/event.ts +89 -0
  35. package/scaffold/pm-api/src/mcp-tools/index.ts +11 -0
  36. package/scaffold/pm-api/src/mcp-tools/initiative.ts +51 -0
  37. package/scaffold/pm-api/src/mcp-tools/memo.ts +164 -0
  38. package/scaffold/pm-api/src/mcp-tools/notification.ts +37 -0
  39. package/scaffold/pm-api/src/mcp-tools/retro.ts +183 -0
  40. package/scaffold/pm-api/src/mcp-tools/sprint.ts +204 -0
  41. package/scaffold/pm-api/src/mcp-tools/standup.ts +136 -0
  42. package/scaffold/pm-api/src/mcp-tools/story.ts +230 -0
  43. package/scaffold/pm-api/src/mcp-tools/task.ts +187 -0
  44. package/scaffold/pm-api/src/mcp-tools/utils.ts +83 -0
  45. package/scaffold/pm-api/src/mcp.ts +871 -0
  46. package/scaffold/pm-api/src/nudge.ts +283 -0
  47. package/scaffold/pm-api/src/routes/auth.ts +32 -0
  48. package/scaffold/pm-api/src/routes/v2-activity.ts +27 -0
  49. package/scaffold/pm-api/src/routes/v2-admin.ts +165 -0
  50. package/scaffold/pm-api/src/routes/v2-dashboard.ts +189 -0
  51. package/scaffold/pm-api/src/routes/v2-docs.ts +34 -0
  52. package/scaffold/pm-api/src/routes/v2-initiatives.ts +118 -0
  53. package/scaffold/pm-api/src/routes/v2-kickoff.ts +265 -0
  54. package/scaffold/pm-api/src/routes/v2-meetings.ts +324 -0
  55. package/scaffold/pm-api/src/routes/v2-memos.ts +257 -0
  56. package/scaffold/pm-api/src/routes/v2-nav.ts +260 -0
  57. package/scaffold/pm-api/src/routes/v2-notifications.ts +79 -0
  58. package/scaffold/pm-api/src/routes/v2-page-content.ts +35 -0
  59. package/scaffold/pm-api/src/routes/v2-pm.ts +380 -0
  60. package/scaffold/pm-api/src/routes/v2-policy.ts +58 -0
  61. package/scaffold/pm-api/src/routes/v2-retro.ts +221 -0
  62. package/scaffold/pm-api/src/routes/v2-rewards.ts +132 -0
  63. package/scaffold/pm-api/src/routes/v2-scenarios.ts +48 -0
  64. package/scaffold/pm-api/src/routes/v2-search.ts +32 -0
  65. package/scaffold/pm-api/src/routes/v2-standup.ts +127 -0
  66. package/scaffold/pm-api/src/routes/v2-user.ts +38 -0
  67. package/scaffold/pm-api/src/types.ts +11 -0
  68. package/scaffold/pm-api/src/utils/activity.ts +22 -0
  69. package/scaffold/pm-api/src/utils/admin.ts +9 -0
  70. package/scaffold/pm-api/src/utils/agent-notify.ts +62 -0
  71. package/scaffold/pm-api/src/utils/assignee.ts +69 -0
  72. package/scaffold/pm-api/src/utils/db.ts +45 -0
  73. package/scaffold/pm-api/src/utils/initiative.ts +23 -0
  74. package/scaffold/pm-api/src/utils/retro-link.ts +32 -0
  75. package/scaffold/pm-api/src/utils/sprint-lifecycle.ts +96 -0
  76. package/scaffold/pm-api/tsconfig.json +15 -0
  77. package/scaffold/pm-api/wrangler.toml.hbs +11 -0
  78. package/scaffold/spec-site/package-lock.json +892 -0
  79. package/scaffold/spec-site/package.json +15 -1
  80. package/scaffold/spec-site/src/api/types.ts +6 -0
  81. package/scaffold/spec-site/src/components/AppHeader.vue +429 -55
  82. package/scaffold/spec-site/src/components/AuthGate.vue +117 -0
  83. package/scaffold/spec-site/src/components/BurndownChart.vue +78 -0
  84. package/scaffold/spec-site/src/components/DocComments.vue +137 -0
  85. package/scaffold/spec-site/src/components/DocEditor.vue +118 -0
  86. package/scaffold/spec-site/src/components/DocExportBar.vue +110 -0
  87. package/scaffold/spec-site/src/components/DocsSidebar.vue +309 -0
  88. package/scaffold/spec-site/src/components/EmptyState.vue +30 -0
  89. package/scaffold/spec-site/src/components/ErrorBanner.vue +38 -0
  90. package/scaffold/spec-site/src/components/Icon.vue +58 -0
  91. package/scaffold/spec-site/src/components/MemberSelect.vue +48 -0
  92. package/scaffold/spec-site/src/components/MemoChecklist.vue +88 -0
  93. package/scaffold/spec-site/src/components/MemoGraph.vue +75 -0
  94. package/scaffold/spec-site/src/components/MemoItem.vue +353 -0
  95. package/scaffold/spec-site/src/components/MemoRelations.vue +101 -0
  96. package/scaffold/spec-site/src/components/MemoTimeline.vue +53 -0
  97. package/scaffold/spec-site/src/components/MentionInput.vue +174 -0
  98. package/scaffold/spec-site/src/components/NotificationDropdown.vue +116 -0
  99. package/scaffold/spec-site/src/components/PriorityBadge.vue +23 -0
  100. package/scaffold/spec-site/src/components/SearchModal.vue +102 -0
  101. package/scaffold/spec-site/src/components/SlashCommand.ts +123 -0
  102. package/scaffold/spec-site/src/components/StateDisplay.vue +54 -0
  103. package/scaffold/spec-site/src/components/TreeNode.vue +82 -0
  104. package/scaffold/spec-site/src/components/UserAvatar.vue +24 -0
  105. package/scaffold/spec-site/src/components/VelocityChart.vue +77 -0
  106. package/scaffold/spec-site/src/composables/navTypes.ts +3 -0
  107. package/scaffold/spec-site/src/composables/pmTypes.ts +15 -2
  108. package/scaffold/spec-site/src/composables/useBottomSheet.ts +103 -0
  109. package/scaffold/spec-site/src/composables/useDashboard.ts +221 -0
  110. package/scaffold/spec-site/src/composables/useMediaQuery.ts +28 -0
  111. package/scaffold/spec-site/src/composables/useMemo.ts +39 -0
  112. package/scaffold/spec-site/src/composables/useNotification.ts +200 -0
  113. package/scaffold/spec-site/src/composables/usePmStore.ts +48 -1
  114. package/scaffold/spec-site/src/composables/useRetro.ts +6 -0
  115. package/scaffold/spec-site/src/composables/useStandup.ts +201 -0
  116. package/scaffold/spec-site/src/composables/useTheme.ts +37 -0
  117. package/scaffold/spec-site/src/composables/useTurso.ts +17 -0
  118. package/scaffold/spec-site/src/composables/useUser.ts +19 -1
  119. package/scaffold/spec-site/src/composables/useViewport.ts +26 -0
  120. package/scaffold/spec-site/src/features.ts +108 -0
  121. package/scaffold/spec-site/src/mockup/ComponentPalette.vue +61 -0
  122. package/scaffold/spec-site/src/mockup/MockupCanvas.vue +459 -0
  123. package/scaffold/spec-site/src/mockup/PropertyPanel.vue +217 -0
  124. package/scaffold/spec-site/src/mockup/componentCatalog.ts +68 -0
  125. package/scaffold/spec-site/src/mockup/useScenarios.ts +67 -0
  126. package/scaffold/spec-site/src/pages/AdminPage.vue +299 -0
  127. package/scaffold/spec-site/src/pages/DashboardPage.vue +650 -0
  128. package/scaffold/spec-site/src/pages/DocsEditor.vue +119 -0
  129. package/scaffold/spec-site/src/pages/DocsHub.vue +157 -0
  130. package/scaffold/spec-site/src/pages/DocsPage.vue +444 -0
  131. package/scaffold/spec-site/src/pages/InboxPage.vue +156 -0
  132. package/scaffold/spec-site/src/pages/MeetingsPage.vue +294 -0
  133. package/scaffold/spec-site/src/pages/MemosPage.vue +857 -0
  134. package/scaffold/spec-site/src/pages/MockupEditorPage.vue +611 -0
  135. package/scaffold/spec-site/src/pages/MockupListPage.vue +121 -0
  136. package/scaffold/spec-site/src/pages/MockupViewerPage.vue +199 -0
  137. package/scaffold/spec-site/src/pages/MyPage.vue +343 -0
  138. package/scaffold/spec-site/src/pages/NotificationSettingsPage.vue +59 -0
  139. package/scaffold/spec-site/src/pages/RewardsPage.vue +266 -0
  140. package/scaffold/spec-site/src/pages/SprintAdmin.vue +521 -0
  141. package/scaffold/spec-site/src/pages/SprintTimeline.vue +159 -0
  142. package/scaffold/spec-site/src/pages/board/BoardAdmin.vue +422 -0
  143. package/scaffold/spec-site/src/pages/board/BoardEpicSection.vue +54 -0
  144. package/scaffold/spec-site/src/pages/board/BoardPage.vue +884 -0
  145. package/scaffold/spec-site/src/pages/board/BoardStoryCard.vue +67 -0
  146. package/scaffold/spec-site/src/pages/board/BoardTaskItem.vue +52 -0
  147. package/scaffold/spec-site/src/pages/board/KanbanBoard.vue +93 -0
  148. package/scaffold/spec-site/src/pages/board/MyTasksPage.vue +202 -0
  149. package/scaffold/spec-site/src/pages/board/SprintClose.vue +167 -0
  150. package/scaffold/spec-site/src/pages/board/SprintColumn.vue +49 -0
  151. package/scaffold/spec-site/src/pages/board/SprintKickoff.vue +389 -0
  152. package/scaffold/spec-site/src/pages/board/StatusBadge.vue +52 -0
  153. package/scaffold/spec-site/src/pages/board/StoryDetailPanel.vue +495 -0
  154. package/scaffold/spec-site/src/pages/board/TaskCard.vue +42 -0
  155. package/scaffold/spec-site/src/pages/retro/RetroCard.vue +36 -2
  156. package/scaffold/spec-site/src/pages/retro/RetroHeader.vue +82 -66
  157. package/scaffold/spec-site/src/pages/retro/RetroPage.vue +47 -18
  158. package/scaffold/spec-site/src/pages/standup/StandupEntryCard.vue +551 -0
  159. package/scaffold/spec-site/src/pages/standup/StandupForm.vue +68 -0
  160. package/scaffold/spec-site/src/pages/standup/StandupList.vue +71 -0
  161. package/scaffold/spec-site/src/pages/standup/StandupPage.vue +225 -0
  162. package/scaffold/spec-site/src/router.ts +141 -0
  163. package/scaffold/spec-site/src/styles/buttons.css +124 -0
  164. package/scaffold/spec-site/src/utils/parseMentions.ts +56 -0
  165. package/scaffold/spec-site/src/utils/timezone.ts +18 -0
@@ -0,0 +1,119 @@
1
+ <script setup lang="ts">
2
+ import { ref, computed, onMounted } from 'vue'
3
+ import { useRoute, useRouter } from 'vue-router'
4
+ import { apiGet, apiPut } from '@/composables/useTurso'
5
+ import { renderMarkdown } from '@/utils/markdown'
6
+
7
+ const route = useRoute()
8
+ const router = useRouter()
9
+ const docId = computed(() => route.params.docId as string | undefined)
10
+ const isNew = computed(() => !docId.value)
11
+
12
+ const title = ref('')
13
+ const content = ref('')
14
+ const saving = ref(false)
15
+
16
+ onMounted(async () => {
17
+ if (docId.value) {
18
+ const { data } = await apiGet<{ doc: { title: string; content: string } }>(`/api/v2/docs/${docId.value}`)
19
+ if (data?.doc) {
20
+ title.value = data.doc.title
21
+ content.value = data.doc.content
22
+ }
23
+ }
24
+ })
25
+
26
+ function generateId(title: string): string {
27
+ return title.toLowerCase().replace(/[^a-z0-9]/g, '-').replace(/-+/g, '-').replace(/^-|-$/g, '').slice(0, 50) || 'untitled'
28
+ }
29
+
30
+ async function save() {
31
+ if (!title.value.trim()) { alert('Please enter a title'); return }
32
+ saving.value = true
33
+ const id = docId.value || generateId(title.value)
34
+ const { error } = await apiPut(`/api/v2/docs/${id}`, { title: title.value, content: content.value })
35
+ saving.value = false
36
+ if (error) { alert(error); return }
37
+ router.push(`/docs/${id}`)
38
+ }
39
+
40
+ // renderMarkdown imported from @/utils/markdown
41
+ </script>
42
+
43
+ <template>
44
+ <div class="editor-page">
45
+ <div class="editor-header">
46
+ <h1>{{ isNew ? 'New Document' : 'Edit Document' }}</h1>
47
+ <div class="editor-actions">
48
+ <button class="btn" @click="router.back()">Cancel</button>
49
+ <button class="btn btn--primary" :disabled="saving || !title.trim()" @click="save">
50
+ {{ saving ? 'Saving...' : 'Save' }}
51
+ </button>
52
+ </div>
53
+ </div>
54
+
55
+ <input
56
+ v-model="title"
57
+ class="editor-title"
58
+ placeholder="Document title"
59
+ />
60
+
61
+ <div class="editor-split">
62
+ <div class="editor-pane">
63
+ <div class="pane-label">Markdown</div>
64
+ <textarea
65
+ v-model="content"
66
+ class="editor-textarea"
67
+ placeholder="Write in markdown..."
68
+ />
69
+ </div>
70
+ <div class="preview-pane">
71
+ <div class="pane-label">Preview</div>
72
+ <div class="preview-content" v-html="renderMarkdown(content)" />
73
+ </div>
74
+ </div>
75
+ </div>
76
+ </template>
77
+
78
+ <style scoped>
79
+ .editor-page { max-width: 1200px; margin: 0 auto; padding: 24px; }
80
+ .editor-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 16px; }
81
+ .editor-header h1 { font-size: 20px; font-weight: 700; }
82
+ .editor-actions { display: flex; gap: 8px; }
83
+
84
+ .editor-title {
85
+ width: 100%; padding: 12px 16px; font-size: 18px; font-weight: 600;
86
+ border: 1px solid rgba(0,0,0,0.1); border-radius: 12px; margin-bottom: 16px;
87
+ box-sizing: border-box;
88
+ }
89
+
90
+ .editor-split { display: grid; grid-template-columns: 1fr 1fr; gap: 16px; min-height: 60vh; }
91
+ .pane-label { font-size: 12px; font-weight: 600; color: var(--text-muted, #888); margin-bottom: 8px; }
92
+
93
+ .editor-textarea {
94
+ width: 100%; height: 100%; min-height: 500px;
95
+ border: 1px solid rgba(0,0,0,0.1); border-radius: 12px;
96
+ padding: 16px; font-size: 14px; font-family: 'Fira Code', monospace;
97
+ resize: vertical; box-sizing: border-box; line-height: 1.6;
98
+ }
99
+
100
+ .preview-pane { overflow-y: auto; }
101
+ .preview-content {
102
+ border: 1px solid rgba(0,0,0,0.1); border-radius: 12px;
103
+ padding: 16px; min-height: 500px; font-size: 14px; line-height: 1.7;
104
+ }
105
+ .preview-content :deep(h1) { font-size: 22px; font-weight: 700; margin: 16px 0 8px; }
106
+ .preview-content :deep(h2) { font-size: 18px; font-weight: 600; margin: 24px 0 12px; border-bottom: 1px solid #eee; padding-bottom: 6px; }
107
+ .preview-content :deep(h3) { font-size: 15px; font-weight: 600; margin: 16px 0 8px; }
108
+ .preview-content :deep(code) { background: rgba(0,0,0,0.06); padding: 2px 5px; border-radius: 3px; font-size: 12px; }
109
+ .preview-content :deep(.code-block) { background: #1e293b; color: #e2e8f0; border-radius: 8px; padding: 12px; overflow-x: auto; }
110
+ .preview-content :deep(.code-block code) { background: none; color: inherit; }
111
+ .preview-content :deep(blockquote) { border-left: 4px solid #3b82f6; background: #f9fafb; padding: 8px 12px; margin: 12px 0; border-radius: 0 8px 8px 0; }
112
+ .preview-content :deep(hr) { border: none; border-top: 1px solid #e5e7eb; margin: 24px 0; }
113
+ .preview-content :deep(a) { color: #3b82f6; }
114
+ .preview-content :deep(ul) { padding-left: 20px; }
115
+
116
+ @media (max-width: 768px) {
117
+ .editor-split { grid-template-columns: 1fr; }
118
+ }
119
+ </style>
@@ -0,0 +1,157 @@
1
+ <script setup lang="ts">
2
+ import { ref, onMounted } from 'vue'
3
+ import { apiGet, isStaticMode } from '@/api/client'
4
+
5
+ interface DocEntry {
6
+ id: number
7
+ title: string
8
+ path: string
9
+ updated_by: string
10
+ updated_at: string
11
+ children?: DocEntry[]
12
+ }
13
+
14
+ interface DocsData {
15
+ documents: DocEntry[]
16
+ recent_edits: { id: number; title: string; user: string; updated_at: string }[]
17
+ }
18
+
19
+ const loading = ref(true)
20
+ const error = ref<string | null>(null)
21
+ const docs = ref<DocEntry[]>([])
22
+ const recentEdits = ref<DocsData['recent_edits']>([])
23
+ const expandedIds = ref<Set<number>>(new Set())
24
+
25
+ function toggleExpand(id: number) {
26
+ if (expandedIds.value.has(id)) {
27
+ expandedIds.value.delete(id)
28
+ } else {
29
+ expandedIds.value.add(id)
30
+ }
31
+ }
32
+
33
+ onMounted(async () => {
34
+ if (isStaticMode()) { loading.value = false; return }
35
+ const { data, error: err } = await apiGet<DocsData>('/api/v2/docs')
36
+ if (err) {
37
+ error.value = err
38
+ } else if (data) {
39
+ docs.value = data.documents
40
+ recentEdits.value = data.recent_edits
41
+ }
42
+ loading.value = false
43
+ })
44
+ </script>
45
+
46
+ <template>
47
+ <div class="docs-page">
48
+ <h1>Documents</h1>
49
+ <p class="page-desc">Team documentation hub.</p>
50
+
51
+ <div v-if="loading" class="loading-state">
52
+ <div class="loading-spinner" />
53
+ <span>Loading documents...</span>
54
+ </div>
55
+
56
+ <div v-else-if="error" class="error-state">
57
+ <div class="error-icon">&#9888;</div>
58
+ <p>{{ error }}</p>
59
+ </div>
60
+
61
+ <template v-else>
62
+ <div class="docs-layout">
63
+ <!-- Document tree -->
64
+ <div class="docs-tree">
65
+ <h2>All Documents</h2>
66
+ <div v-if="!docs.length" class="empty-msg">No documents yet.</div>
67
+ <ul class="tree-list">
68
+ <li v-for="doc in docs" :key="doc.id" class="tree-item">
69
+ <div class="tree-row" @click="toggleExpand(doc.id)">
70
+ <span class="tree-icon" v-if="doc.children?.length">
71
+ {{ expandedIds.has(doc.id) ? '&#9660;' : '&#9654;' }}
72
+ </span>
73
+ <span class="tree-icon tree-leaf" v-else>&#9679;</span>
74
+ <span class="tree-title">{{ doc.title }}</span>
75
+ <span class="tree-meta">{{ doc.updated_by }}</span>
76
+ </div>
77
+ <ul v-if="doc.children?.length && expandedIds.has(doc.id)" class="tree-children">
78
+ <li v-for="child in doc.children" :key="child.id" class="tree-item tree-child">
79
+ <div class="tree-row">
80
+ <span class="tree-icon tree-leaf">&#9679;</span>
81
+ <span class="tree-title">{{ child.title }}</span>
82
+ <span class="tree-meta">{{ child.updated_by }}</span>
83
+ </div>
84
+ </li>
85
+ </ul>
86
+ </li>
87
+ </ul>
88
+ </div>
89
+
90
+ <!-- Recent edits -->
91
+ <div class="docs-recent">
92
+ <h2>Recent Edits</h2>
93
+ <div v-if="!recentEdits.length" class="empty-msg">No recent edits.</div>
94
+ <ul class="edit-list">
95
+ <li v-for="edit in recentEdits" :key="edit.id" class="edit-item">
96
+ <div class="edit-title">{{ edit.title }}</div>
97
+ <div class="edit-meta">
98
+ <span>{{ edit.user }}</span>
99
+ <span>{{ edit.updated_at }}</span>
100
+ </div>
101
+ </li>
102
+ </ul>
103
+ </div>
104
+ </div>
105
+ </template>
106
+ </div>
107
+ </template>
108
+
109
+ <style scoped>
110
+ .docs-page { padding: 32px 40px; max-width: 960px; margin: 0 auto; }
111
+ h1 { font-size: 22px; font-weight: 700; margin-bottom: 4px; }
112
+ .page-desc { font-size: 13px; color: var(--text-secondary); margin-bottom: 24px; }
113
+
114
+ .loading-state, .error-state {
115
+ display: flex; flex-direction: column; align-items: center; justify-content: center;
116
+ padding: 80px 0; gap: 12px; color: var(--text-muted);
117
+ }
118
+ .loading-spinner {
119
+ width: 28px; height: 28px; border: 3px solid var(--border);
120
+ border-top-color: var(--primary); border-radius: 50%; animation: spin 0.8s linear infinite;
121
+ }
122
+ @keyframes spin { to { transform: rotate(360deg); } }
123
+ .error-icon { font-size: 32px; }
124
+ .error-state p { color: var(--red); font-size: 14px; }
125
+ .empty-msg { font-size: 13px; color: var(--text-muted); }
126
+
127
+ .docs-layout { display: grid; grid-template-columns: 2fr 1fr; gap: 24px; }
128
+ @media (max-width: 768px) { .docs-layout { grid-template-columns: 1fr; } }
129
+
130
+ .docs-tree, .docs-recent {
131
+ background: var(--card-bg); border: 1px solid var(--border-light);
132
+ border-radius: var(--radius); padding: 20px;
133
+ }
134
+ h2 { font-size: 15px; font-weight: 600; margin-bottom: 16px; }
135
+
136
+ .tree-list { list-style: none; padding: 0; margin: 0; }
137
+ .tree-item { margin-bottom: 2px; }
138
+ .tree-row {
139
+ display: flex; align-items: center; gap: 8px; padding: 8px 10px;
140
+ border-radius: 6px; cursor: pointer; font-size: 13px;
141
+ }
142
+ .tree-row:hover { background: var(--bg); }
143
+ .tree-icon { font-size: 8px; color: var(--text-muted); width: 12px; text-align: center; flex-shrink: 0; }
144
+ .tree-leaf { font-size: 6px; }
145
+ .tree-title { flex: 1; font-weight: 500; }
146
+ .tree-meta { font-size: 11px; color: var(--text-muted); }
147
+ .tree-children { list-style: none; padding-left: 20px; margin: 0; }
148
+ .tree-child .tree-row { padding: 6px 10px; }
149
+
150
+ .edit-list { list-style: none; padding: 0; margin: 0; }
151
+ .edit-item {
152
+ padding: 10px 0; border-bottom: 1px solid var(--border-light);
153
+ }
154
+ .edit-item:last-child { border-bottom: none; }
155
+ .edit-title { font-size: 13px; font-weight: 500; margin-bottom: 4px; }
156
+ .edit-meta { display: flex; justify-content: space-between; font-size: 11px; color: var(--text-muted); }
157
+ </style>