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.
- package/bin/cli.mjs +204 -2
- package/lib/doctor.mjs +38 -1
- package/lib/hydrate.mjs +15 -0
- package/lib/scaffold.mjs +5 -0
- package/lib/setup-wizard.mjs +35 -2
- package/package.json +1 -1
- package/scaffold/.context/project.yaml.example +19 -0
- package/scaffold/mcp-notification-server/package.json +18 -0
- package/scaffold/mcp-notification-server/src/index.ts +275 -0
- package/scaffold/mcp-notification-server/src/turso-client.ts +142 -0
- package/scaffold/mcp-notification-server/tsconfig.json +14 -0
- package/scaffold/mcp-pm/package.json +19 -0
- package/scaffold/mcp-pm/src/api-client.ts +69 -0
- package/scaffold/mcp-pm/src/index.ts +660 -0
- package/scaffold/mcp-pm/tsconfig.json +14 -0
- package/scaffold/pm-api/package.json +21 -0
- package/scaffold/pm-api/sql/001-memo-v2.sql +49 -0
- package/scaffold/pm-api/sql/002-notifications.sql +18 -0
- package/scaffold/pm-api/sql/003-content.sql +66 -0
- package/scaffold/pm-api/sql/004-agent-events.sql +21 -0
- package/scaffold/pm-api/sql/005-epic-sprint-decoupling.sql +6 -0
- package/scaffold/pm-api/sql/schema-core.sql +331 -0
- package/scaffold/pm-api/sql/schema-docs.sql +25 -0
- package/scaffold/pm-api/sql/schema-meetings.sql +17 -0
- package/scaffold/pm-api/sql/schema-rewards.sql +16 -0
- package/scaffold/pm-api/src/auth.ts +28 -0
- package/scaffold/pm-api/src/blockchain/adapter.ts +20 -0
- package/scaffold/pm-api/src/blockchain/tron.ts +62 -0
- package/scaffold/pm-api/src/db/adapter.ts +36 -0
- package/scaffold/pm-api/src/db/turso.ts +147 -0
- package/scaffold/pm-api/src/index.ts +114 -0
- package/scaffold/pm-api/src/mcp-tools/dashboard.ts +40 -0
- package/scaffold/pm-api/src/mcp-tools/epic.ts +67 -0
- package/scaffold/pm-api/src/mcp-tools/event.ts +89 -0
- package/scaffold/pm-api/src/mcp-tools/index.ts +11 -0
- package/scaffold/pm-api/src/mcp-tools/initiative.ts +51 -0
- package/scaffold/pm-api/src/mcp-tools/memo.ts +164 -0
- package/scaffold/pm-api/src/mcp-tools/notification.ts +37 -0
- package/scaffold/pm-api/src/mcp-tools/retro.ts +183 -0
- package/scaffold/pm-api/src/mcp-tools/sprint.ts +204 -0
- package/scaffold/pm-api/src/mcp-tools/standup.ts +136 -0
- package/scaffold/pm-api/src/mcp-tools/story.ts +230 -0
- package/scaffold/pm-api/src/mcp-tools/task.ts +187 -0
- package/scaffold/pm-api/src/mcp-tools/utils.ts +83 -0
- package/scaffold/pm-api/src/mcp.ts +871 -0
- package/scaffold/pm-api/src/nudge.ts +283 -0
- package/scaffold/pm-api/src/routes/auth.ts +32 -0
- package/scaffold/pm-api/src/routes/v2-activity.ts +27 -0
- package/scaffold/pm-api/src/routes/v2-admin.ts +165 -0
- package/scaffold/pm-api/src/routes/v2-dashboard.ts +189 -0
- package/scaffold/pm-api/src/routes/v2-docs.ts +34 -0
- package/scaffold/pm-api/src/routes/v2-initiatives.ts +118 -0
- package/scaffold/pm-api/src/routes/v2-kickoff.ts +265 -0
- package/scaffold/pm-api/src/routes/v2-meetings.ts +324 -0
- package/scaffold/pm-api/src/routes/v2-memos.ts +257 -0
- package/scaffold/pm-api/src/routes/v2-nav.ts +260 -0
- package/scaffold/pm-api/src/routes/v2-notifications.ts +79 -0
- package/scaffold/pm-api/src/routes/v2-page-content.ts +35 -0
- package/scaffold/pm-api/src/routes/v2-pm.ts +380 -0
- package/scaffold/pm-api/src/routes/v2-policy.ts +58 -0
- package/scaffold/pm-api/src/routes/v2-retro.ts +221 -0
- package/scaffold/pm-api/src/routes/v2-rewards.ts +132 -0
- package/scaffold/pm-api/src/routes/v2-scenarios.ts +48 -0
- package/scaffold/pm-api/src/routes/v2-search.ts +32 -0
- package/scaffold/pm-api/src/routes/v2-standup.ts +127 -0
- package/scaffold/pm-api/src/routes/v2-user.ts +38 -0
- package/scaffold/pm-api/src/types.ts +11 -0
- package/scaffold/pm-api/src/utils/activity.ts +22 -0
- package/scaffold/pm-api/src/utils/admin.ts +9 -0
- package/scaffold/pm-api/src/utils/agent-notify.ts +62 -0
- package/scaffold/pm-api/src/utils/assignee.ts +69 -0
- package/scaffold/pm-api/src/utils/db.ts +45 -0
- package/scaffold/pm-api/src/utils/initiative.ts +23 -0
- package/scaffold/pm-api/src/utils/retro-link.ts +32 -0
- package/scaffold/pm-api/src/utils/sprint-lifecycle.ts +96 -0
- package/scaffold/pm-api/tsconfig.json +15 -0
- package/scaffold/pm-api/wrangler.toml.hbs +11 -0
- package/scaffold/spec-site/package-lock.json +892 -0
- package/scaffold/spec-site/package.json +15 -1
- package/scaffold/spec-site/src/api/types.ts +6 -0
- package/scaffold/spec-site/src/components/AppHeader.vue +429 -55
- package/scaffold/spec-site/src/components/AuthGate.vue +117 -0
- package/scaffold/spec-site/src/components/BurndownChart.vue +78 -0
- package/scaffold/spec-site/src/components/DocComments.vue +137 -0
- package/scaffold/spec-site/src/components/DocEditor.vue +118 -0
- package/scaffold/spec-site/src/components/DocExportBar.vue +110 -0
- package/scaffold/spec-site/src/components/DocsSidebar.vue +309 -0
- package/scaffold/spec-site/src/components/EmptyState.vue +30 -0
- package/scaffold/spec-site/src/components/ErrorBanner.vue +38 -0
- package/scaffold/spec-site/src/components/Icon.vue +58 -0
- package/scaffold/spec-site/src/components/MemberSelect.vue +48 -0
- package/scaffold/spec-site/src/components/MemoChecklist.vue +88 -0
- package/scaffold/spec-site/src/components/MemoGraph.vue +75 -0
- package/scaffold/spec-site/src/components/MemoItem.vue +353 -0
- package/scaffold/spec-site/src/components/MemoRelations.vue +101 -0
- package/scaffold/spec-site/src/components/MemoTimeline.vue +53 -0
- package/scaffold/spec-site/src/components/MentionInput.vue +174 -0
- package/scaffold/spec-site/src/components/NotificationDropdown.vue +116 -0
- package/scaffold/spec-site/src/components/PriorityBadge.vue +23 -0
- package/scaffold/spec-site/src/components/SearchModal.vue +102 -0
- package/scaffold/spec-site/src/components/SlashCommand.ts +123 -0
- package/scaffold/spec-site/src/components/StateDisplay.vue +54 -0
- package/scaffold/spec-site/src/components/TreeNode.vue +82 -0
- package/scaffold/spec-site/src/components/UserAvatar.vue +24 -0
- package/scaffold/spec-site/src/components/VelocityChart.vue +77 -0
- package/scaffold/spec-site/src/composables/navTypes.ts +3 -0
- package/scaffold/spec-site/src/composables/pmTypes.ts +15 -2
- package/scaffold/spec-site/src/composables/useBottomSheet.ts +103 -0
- package/scaffold/spec-site/src/composables/useDashboard.ts +221 -0
- package/scaffold/spec-site/src/composables/useMediaQuery.ts +28 -0
- package/scaffold/spec-site/src/composables/useMemo.ts +39 -0
- package/scaffold/spec-site/src/composables/useNotification.ts +200 -0
- package/scaffold/spec-site/src/composables/usePmStore.ts +48 -1
- package/scaffold/spec-site/src/composables/useRetro.ts +6 -0
- package/scaffold/spec-site/src/composables/useStandup.ts +201 -0
- package/scaffold/spec-site/src/composables/useTheme.ts +37 -0
- package/scaffold/spec-site/src/composables/useTurso.ts +17 -0
- package/scaffold/spec-site/src/composables/useUser.ts +19 -1
- package/scaffold/spec-site/src/composables/useViewport.ts +26 -0
- package/scaffold/spec-site/src/features.ts +108 -0
- package/scaffold/spec-site/src/mockup/ComponentPalette.vue +61 -0
- package/scaffold/spec-site/src/mockup/MockupCanvas.vue +459 -0
- package/scaffold/spec-site/src/mockup/PropertyPanel.vue +217 -0
- package/scaffold/spec-site/src/mockup/componentCatalog.ts +68 -0
- package/scaffold/spec-site/src/mockup/useScenarios.ts +67 -0
- package/scaffold/spec-site/src/pages/AdminPage.vue +299 -0
- package/scaffold/spec-site/src/pages/DashboardPage.vue +650 -0
- package/scaffold/spec-site/src/pages/DocsEditor.vue +119 -0
- package/scaffold/spec-site/src/pages/DocsHub.vue +157 -0
- package/scaffold/spec-site/src/pages/DocsPage.vue +444 -0
- package/scaffold/spec-site/src/pages/InboxPage.vue +156 -0
- package/scaffold/spec-site/src/pages/MeetingsPage.vue +294 -0
- package/scaffold/spec-site/src/pages/MemosPage.vue +857 -0
- package/scaffold/spec-site/src/pages/MockupEditorPage.vue +611 -0
- package/scaffold/spec-site/src/pages/MockupListPage.vue +121 -0
- package/scaffold/spec-site/src/pages/MockupViewerPage.vue +199 -0
- package/scaffold/spec-site/src/pages/MyPage.vue +343 -0
- package/scaffold/spec-site/src/pages/NotificationSettingsPage.vue +59 -0
- package/scaffold/spec-site/src/pages/RewardsPage.vue +266 -0
- package/scaffold/spec-site/src/pages/SprintAdmin.vue +521 -0
- package/scaffold/spec-site/src/pages/SprintTimeline.vue +159 -0
- package/scaffold/spec-site/src/pages/board/BoardAdmin.vue +422 -0
- package/scaffold/spec-site/src/pages/board/BoardEpicSection.vue +54 -0
- package/scaffold/spec-site/src/pages/board/BoardPage.vue +884 -0
- package/scaffold/spec-site/src/pages/board/BoardStoryCard.vue +67 -0
- package/scaffold/spec-site/src/pages/board/BoardTaskItem.vue +52 -0
- package/scaffold/spec-site/src/pages/board/KanbanBoard.vue +93 -0
- package/scaffold/spec-site/src/pages/board/MyTasksPage.vue +202 -0
- package/scaffold/spec-site/src/pages/board/SprintClose.vue +167 -0
- package/scaffold/spec-site/src/pages/board/SprintColumn.vue +49 -0
- package/scaffold/spec-site/src/pages/board/SprintKickoff.vue +389 -0
- package/scaffold/spec-site/src/pages/board/StatusBadge.vue +52 -0
- package/scaffold/spec-site/src/pages/board/StoryDetailPanel.vue +495 -0
- package/scaffold/spec-site/src/pages/board/TaskCard.vue +42 -0
- package/scaffold/spec-site/src/pages/retro/RetroCard.vue +36 -2
- package/scaffold/spec-site/src/pages/retro/RetroHeader.vue +82 -66
- package/scaffold/spec-site/src/pages/retro/RetroPage.vue +47 -18
- package/scaffold/spec-site/src/pages/standup/StandupEntryCard.vue +551 -0
- package/scaffold/spec-site/src/pages/standup/StandupForm.vue +68 -0
- package/scaffold/spec-site/src/pages/standup/StandupList.vue +71 -0
- package/scaffold/spec-site/src/pages/standup/StandupPage.vue +225 -0
- package/scaffold/spec-site/src/router.ts +141 -0
- package/scaffold/spec-site/src/styles/buttons.css +124 -0
- package/scaffold/spec-site/src/utils/parseMentions.ts +56 -0
- 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">⚠</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) ? '▼' : '▶' }}
|
|
72
|
+
</span>
|
|
73
|
+
<span class="tree-icon tree-leaf" v-else>●</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">●</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>
|