summaryrefslogtreecommitdiff
path: root/frontend/src/components/FileDetail.tsx
diff options
context:
space:
mode:
authorsoryu <soryu@soryu.co>2026-01-25 00:01:25 +0000
committerGitHub <noreply@github.com>2026-01-25 00:01:25 +0000
commita279ec29efb863fefd1ca82e5b490f2e8784cf3c (patch)
treeaf207e559e7eef5557b2229714384bf78c530976 /frontend/src/components/FileDetail.tsx
parent6364363d1418728351f252b799d397b756e1f985 (diff)
downloadsoryu-a279ec29efb863fefd1ca82e5b490f2e8784cf3c.tar.gz
soryu-a279ec29efb863fefd1ca82e5b490f2e8784cf3c.zip
Move files tab and file pages to be accessible via contracts (#27)
* feat: remove Files from top-level navigation Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * feat: update file links to use contract-scoped routes Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * feat: add contract context to FileDetail component - Add contractId, contractName, and onContractClick props to FileDetailProps - Update breadcrumb navigation to show contract name with path separator when viewing file within a contract context - Fall back to "Back to list" when no contract context is provided - This enables the FileDetail component to be used within the /contracts/:contractId/files/:fileId route Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * feat: update routes to nest files under contracts - Add react-router-dom for client-side routing - Create ContractList component to list all contracts - Create ContractDetail component with tabs (overview, files, tasks, repos) - Create FileDetail component to view individual files - Configure routes: - /contracts - list all contracts - /contracts/:id - view contract details with Files tab - /contracts/:contractId/files/:fileId - view file in contract context - Remove standalone file routes (/files, /files/:id) Files are now only accessible through their parent contract. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Task completion checkpoint * Task completion checkpoint * Task completion checkpoint * Task completion checkpoint * Task completion checkpoint * Task completion checkpoint * Task completion checkpoint * Task completion checkpoint --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com>
Diffstat (limited to 'frontend/src/components/FileDetail.tsx')
-rw-r--r--frontend/src/components/FileDetail.tsx97
1 files changed, 97 insertions, 0 deletions
diff --git a/frontend/src/components/FileDetail.tsx b/frontend/src/components/FileDetail.tsx
new file mode 100644
index 0000000..31228ef
--- /dev/null
+++ b/frontend/src/components/FileDetail.tsx
@@ -0,0 +1,97 @@
+import React, { useEffect, useState } from 'react'
+import { useParams, Link } from 'react-router-dom'
+
+interface File {
+ id: string
+ name: string
+ description?: string
+ body?: string
+ contract_id?: string
+ version: number
+ created_at: string
+}
+
+export function FileDetail() {
+ const { contractId, fileId } = useParams<{ contractId: string; fileId: string }>()
+ const [file, setFile] = useState<File | null>(null)
+ const [loading, setLoading] = useState(true)
+ const [error, setError] = useState<string | null>(null)
+
+ useEffect(() => {
+ async function fetchFile() {
+ if (!fileId) return
+
+ try {
+ setLoading(true)
+ const response = await fetch(`/api/v1/files/${fileId}`)
+ if (!response.ok) {
+ throw new Error(`Failed to fetch file: ${response.statusText}`)
+ }
+ const data = await response.json()
+ setFile(data)
+ } catch (err) {
+ setError(err instanceof Error ? err.message : 'Unknown error')
+ } finally {
+ setLoading(false)
+ }
+ }
+
+ fetchFile()
+ }, [fileId])
+
+ if (loading) {
+ return (
+ <div className="file-detail-container">
+ <div className="loading">Loading file...</div>
+ </div>
+ )
+ }
+
+ if (error) {
+ return (
+ <div className="file-detail-container">
+ <div className="error">Error: {error}</div>
+ <Link to={`/contracts/${contractId}`} className="back-link">
+ Back to Contract
+ </Link>
+ </div>
+ )
+ }
+
+ if (!file) {
+ return (
+ <div className="file-detail-container">
+ <div className="not-found">File not found</div>
+ <Link to={`/contracts/${contractId}`} className="back-link">
+ Back to Contract
+ </Link>
+ </div>
+ )
+ }
+
+ return (
+ <div className="file-detail-container">
+ <div className="file-detail-header">
+ <Link to={`/contracts/${contractId}`} className="back-link">
+ Back to Contract
+ </Link>
+ <h1 className="file-title">{file.name}</h1>
+ {file.description && (
+ <p className="file-description">{file.description}</p>
+ )}
+ <div className="file-meta">
+ <span>Version: {file.version}</span>
+ <span>Created: {new Date(file.created_at).toLocaleString()}</span>
+ </div>
+ </div>
+
+ <div className="file-detail-body">
+ {file.body ? (
+ <pre className="file-content">{file.body}</pre>
+ ) : (
+ <p className="no-content">No content</p>
+ )}
+ </div>
+ </div>
+ )
+}