@brainfish-ai/devdoc 0.1.24 → 0.1.26

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 (39) hide show
  1. package/README.md +2 -2
  2. package/dist/cli/commands/create.js +54 -17
  3. package/dist/cli/commands/dev.js +41 -77
  4. package/dist/cli/commands/init.js +29 -11
  5. package/package.json +6 -6
  6. package/renderer/components/docs-viewer/index.tsx +41 -2
  7. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +20 -0
  8. package/templates/basic/README.md +0 -139
  9. package/templates/basic/assets/favicon.svg +0 -4
  10. package/templates/basic/assets/logo.svg +0 -9
  11. package/templates/basic/docs.json +0 -47
  12. package/templates/basic/guides/configuration.mdx +0 -149
  13. package/templates/basic/guides/overview.mdx +0 -96
  14. package/templates/basic/index.mdx +0 -39
  15. package/templates/basic/package.json +0 -14
  16. package/templates/basic/quickstart.mdx +0 -92
  17. package/templates/basic/vercel.json +0 -6
  18. package/templates/graphql/README.md +0 -139
  19. package/templates/graphql/api-reference/schema.graphql +0 -305
  20. package/templates/graphql/assets/favicon.svg +0 -4
  21. package/templates/graphql/assets/logo.svg +0 -9
  22. package/templates/graphql/docs.json +0 -54
  23. package/templates/graphql/guides/configuration.mdx +0 -149
  24. package/templates/graphql/guides/overview.mdx +0 -96
  25. package/templates/graphql/index.mdx +0 -39
  26. package/templates/graphql/package.json +0 -14
  27. package/templates/graphql/quickstart.mdx +0 -92
  28. package/templates/graphql/vercel.json +0 -6
  29. package/templates/openapi/README.md +0 -139
  30. package/templates/openapi/api-reference/openapi.json +0 -419
  31. package/templates/openapi/assets/favicon.svg +0 -4
  32. package/templates/openapi/assets/logo.svg +0 -9
  33. package/templates/openapi/docs.json +0 -61
  34. package/templates/openapi/guides/configuration.mdx +0 -149
  35. package/templates/openapi/guides/overview.mdx +0 -96
  36. package/templates/openapi/index.mdx +0 -39
  37. package/templates/openapi/package.json +0 -14
  38. package/templates/openapi/quickstart.mdx +0 -92
  39. package/templates/openapi/vercel.json +0 -6
@@ -143,24 +143,42 @@ async function init(options) {
143
143
  subdomain = subdomain.toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/^-|-$/g, '');
144
144
  }
145
145
  // Validate subdomain format locally
146
- const formatCheck = isValidSubdomainFormat(subdomain);
146
+ let formatCheck = isValidSubdomainFormat(subdomain);
147
147
  if (!formatCheck.valid) {
148
148
  logger_1.logger.error(formatCheck.error);
149
149
  process.exit(1);
150
150
  }
151
151
  // Check subdomain availability via API (server validates blacklist)
152
152
  // Note: This only checks availability, it does NOT reserve the subdomain
153
- logger_1.logger.info(`Checking if ${subdomain}.devdoc.sh is available...`);
154
- const availability = await checkSubdomainAvailability(subdomain, apiUrl);
155
- if (!availability.available) {
156
- logger_1.logger.error(availability.error || `Subdomain "${subdomain}" is not available`);
157
- if (availability.suggestion) {
158
- logger_1.logger.info(`Suggestion: Try "${availability.suggestion}"`);
153
+ // Loop until we find an available subdomain
154
+ let subdomainAvailable = false;
155
+ while (!subdomainAvailable) {
156
+ logger_1.logger.info(`Checking if ${subdomain}.devdoc.sh is available...`);
157
+ const availability = await checkSubdomainAvailability(subdomain, apiUrl);
158
+ if (!availability.available) {
159
+ console.log('');
160
+ logger_1.logger.warn(availability.error || `Subdomain "${subdomain}" is not available`);
161
+ const suggestion = availability.suggestion || `${subdomain}-docs`;
162
+ console.log('');
163
+ logger_1.logger.info(`Suggestion: ${suggestion}.devdoc.sh`);
164
+ console.log('');
165
+ // Prompt for new subdomain
166
+ subdomain = await prompt('Enter a different subdomain', suggestion);
167
+ subdomain = subdomain.toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/^-|-$/g, '');
168
+ // Validate format
169
+ formatCheck = isValidSubdomainFormat(subdomain);
170
+ if (!formatCheck.valid) {
171
+ logger_1.logger.error(formatCheck.error);
172
+ continue;
173
+ }
174
+ console.log('');
175
+ }
176
+ else {
177
+ subdomainAvailable = true;
178
+ logger_1.logger.success(`✓ ${subdomain}.devdoc.sh is available!`);
179
+ console.log('');
159
180
  }
160
- process.exit(1);
161
181
  }
162
- logger_1.logger.success(`✓ ${subdomain}.devdoc.sh is available!`);
163
- console.log('');
164
182
  // Create .devdoc.json with subdomain (but NO API key - not registered yet)
165
183
  // The subdomain will only be reserved when the user actually deploys
166
184
  const projectId = `${slug}-${generateId()}`;
@@ -197,4 +215,4 @@ function generateSlug(name) {
197
215
  function generateId() {
198
216
  return Math.random().toString(36).substring(2, 8);
199
217
  }
200
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"init.js","sourceRoot":"","sources":["../../../src/cli/commands/init.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path'\nimport fs from 'fs-extra'\nimport { loadConfig } from '../../config'\nimport { logger } from '../../utils/logger'\nimport { DEFAULT_API_URL } from '../../constants'\n\ninterface InitOptions {\n  slug?: string\n  subdomain?: string\n  force?: boolean\n  url?: string\n}\n\ninterface DevDocConfig {\n  projectId: string\n  name: string\n  slug: string\n  subdomain: string\n  apiKey?: string\n  createdAt: string\n}\n\ninterface CheckSubdomainResponse {\n  available: boolean\n  error?: string\n  suggestion?: string\n}\n\n// Simple prompt helper using readline\nasync function prompt(question: string, defaultValue?: string): Promise<string> {\n  const readline = await import('readline')\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout,\n  })\n\n  return new Promise((resolve) => {\n    const defaultHint = defaultValue ? ` (${defaultValue})` : ''\n    rl.question(`${question}${defaultHint}: `, (answer) => {\n      rl.close()\n      resolve(answer.trim() || defaultValue || '')\n    })\n  })\n}\n\n/**\n * Basic subdomain format validation (server does full validation)\n */\nfunction isValidSubdomainFormat(subdomain: string): { valid: boolean; error?: string } {\n  if (!subdomain) {\n    return { valid: false, error: 'Subdomain is required' }\n  }\n  \n  if (subdomain.length < 3) {\n    return { valid: false, error: 'Subdomain must be at least 3 characters' }\n  }\n  \n  if (subdomain.length > 63) {\n    return { valid: false, error: 'Subdomain must be 63 characters or less' }\n  }\n  \n  if (!/^[a-z0-9]([a-z0-9-]*[a-z0-9])?$/.test(subdomain)) {\n    return { valid: false, error: 'Subdomain must start and end with alphanumeric, can contain hyphens' }\n  }\n  \n  if (/--/.test(subdomain)) {\n    return { valid: false, error: 'Subdomain cannot contain consecutive hyphens' }\n  }\n  \n  return { valid: true }\n}\n\n/**\n * Check subdomain availability via API (also validates against server blacklist)\n */\nasync function checkSubdomainAvailability(\n  subdomain: string,\n  apiUrl: string\n): Promise<CheckSubdomainResponse> {\n  try {\n    const response = await fetch(`${apiUrl}/api/subdomains/check`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n      },\n      body: JSON.stringify({ subdomain }),\n    })\n    \n    const result = await response.json() as CheckSubdomainResponse\n    return result\n  } catch {\n    // If API is unavailable, allow proceeding (will fail at registration if invalid)\n    return { available: true }\n  }\n}\n\n/**\n * Initialize a DevDoc project with .devdoc.json\n * Note: Subdomain is only reserved when documentation is actually deployed (devdoc deploy)\n */\nexport async function init(options: InitOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  logger.info('Initializing DevDoc project...\\n')\n  \n  // Check for docs.json\n  const configPath = path.join(projectRoot, 'docs.json')\n  if (!fs.existsSync(configPath)) {\n    logger.error('docs.json not found in current directory')\n    logger.info('Make sure you are in a DevDoc documentation project directory')\n    process.exit(1)\n  }\n  \n  // Check if .devdoc.json already exists with API key (already deployed)\n  const devdocConfigPath = path.join(projectRoot, '.devdoc.json')\n  if (fs.existsSync(devdocConfigPath) && !options.force) {\n    const existingConfig = fs.readJsonSync(devdocConfigPath) as DevDocConfig\n    if (existingConfig.apiKey) {\n      logger.warn('.devdoc.json already exists with API key (project is deployed)')\n      logger.info('Use --force to overwrite and create a new project')\n      process.exit(1)\n    }\n  }\n  \n  // Load docs.json to get project name\n  let projectName = 'My Documentation'\n  try {\n    const config = await loadConfig(projectRoot)\n    projectName = config.name || projectName\n  } catch {\n    // Use default name\n  }\n  \n  // Generate slug from project name or use provided slug\n  const slug = options.slug || generateSlug(projectName)\n  \n  // Get subdomain - prompt if not provided\n  let subdomain = options.subdomain\n  \n  if (!subdomain) {\n    const suggestedSubdomain = slug\n    console.log('')\n    subdomain = await prompt(`Enter subdomain for ${suggestedSubdomain}.devdoc.sh`, suggestedSubdomain)\n    subdomain = subdomain.toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/^-|-$/g, '')\n  }\n  \n  // Validate subdomain format locally\n  const formatCheck = isValidSubdomainFormat(subdomain)\n  if (!formatCheck.valid) {\n    logger.error(formatCheck.error!)\n    process.exit(1)\n  }\n  \n  // Check subdomain availability via API (server validates blacklist)\n  // Note: This only checks availability, it does NOT reserve the subdomain\n  logger.info(`Checking if ${subdomain}.devdoc.sh is available...`)\n  const availability = await checkSubdomainAvailability(subdomain, apiUrl)\n  \n  if (!availability.available) {\n    logger.error(availability.error || `Subdomain \"${subdomain}\" is not available`)\n    if (availability.suggestion) {\n      logger.info(`Suggestion: Try \"${availability.suggestion}\"`)\n    }\n    process.exit(1)\n  }\n  \n  logger.success(`✓ ${subdomain}.devdoc.sh is available!`)\n  console.log('')\n  \n  // Create .devdoc.json with subdomain (but NO API key - not registered yet)\n  // The subdomain will only be reserved when the user actually deploys\n  const projectId = `${slug}-${generateId()}`\n  const devdocConfig: DevDocConfig = {\n    projectId,\n    name: projectName,\n    slug,\n    subdomain,\n    createdAt: new Date().toISOString(),\n    // Note: No apiKey - subdomain is not reserved until deploy\n  }\n  \n  fs.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 })\n  \n  logger.success('✓ Project initialized')\n  console.log('')\n  console.log('  Subdomain:', `${subdomain}.devdoc.sh`)\n  console.log('  Status:', 'Not yet deployed')\n  console.log('')\n  logger.info('Note: The subdomain is not reserved until you deploy.')\n  logger.info('Run \"devdoc deploy\" to deploy and claim your subdomain.')\n}\n\n/**\n * Generate a URL-friendly slug from a name\n */\nfunction generateSlug(name: string): string {\n  return name\n    .toLowerCase()\n    .replace(/[^a-z0-9]+/g, '-')\n    .replace(/^-|-$/g, '')\n    .substring(0, 50)\n}\n\n/**\n * Generate a short random ID\n */\nfunction generateId(): string {\n  return Math.random().toString(36).substring(2, 8)\n}\n"]}
218
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"init.js","sourceRoot":"","sources":["../../../src/cli/commands/init.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path'\nimport fs from 'fs-extra'\nimport { loadConfig } from '../../config'\nimport { logger } from '../../utils/logger'\nimport { DEFAULT_API_URL } from '../../constants'\n\ninterface InitOptions {\n  slug?: string\n  subdomain?: string\n  force?: boolean\n  url?: string\n}\n\ninterface DevDocConfig {\n  projectId: string\n  name: string\n  slug: string\n  subdomain: string\n  apiKey?: string\n  createdAt: string\n}\n\ninterface CheckSubdomainResponse {\n  available: boolean\n  error?: string\n  suggestion?: string\n}\n\n// Simple prompt helper using readline\nasync function prompt(question: string, defaultValue?: string): Promise<string> {\n  const readline = await import('readline')\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout,\n  })\n\n  return new Promise((resolve) => {\n    const defaultHint = defaultValue ? ` (${defaultValue})` : ''\n    rl.question(`${question}${defaultHint}: `, (answer) => {\n      rl.close()\n      resolve(answer.trim() || defaultValue || '')\n    })\n  })\n}\n\n/**\n * Basic subdomain format validation (server does full validation)\n */\nfunction isValidSubdomainFormat(subdomain: string): { valid: boolean; error?: string } {\n  if (!subdomain) {\n    return { valid: false, error: 'Subdomain is required' }\n  }\n  \n  if (subdomain.length < 3) {\n    return { valid: false, error: 'Subdomain must be at least 3 characters' }\n  }\n  \n  if (subdomain.length > 63) {\n    return { valid: false, error: 'Subdomain must be 63 characters or less' }\n  }\n  \n  if (!/^[a-z0-9]([a-z0-9-]*[a-z0-9])?$/.test(subdomain)) {\n    return { valid: false, error: 'Subdomain must start and end with alphanumeric, can contain hyphens' }\n  }\n  \n  if (/--/.test(subdomain)) {\n    return { valid: false, error: 'Subdomain cannot contain consecutive hyphens' }\n  }\n  \n  return { valid: true }\n}\n\n/**\n * Check subdomain availability via API (also validates against server blacklist)\n */\nasync function checkSubdomainAvailability(\n  subdomain: string,\n  apiUrl: string\n): Promise<CheckSubdomainResponse> {\n  try {\n    const response = await fetch(`${apiUrl}/api/subdomains/check`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n      },\n      body: JSON.stringify({ subdomain }),\n    })\n    \n    const result = await response.json() as CheckSubdomainResponse\n    return result\n  } catch {\n    // If API is unavailable, allow proceeding (will fail at registration if invalid)\n    return { available: true }\n  }\n}\n\n/**\n * Initialize a DevDoc project with .devdoc.json\n * Note: Subdomain is only reserved when documentation is actually deployed (devdoc deploy)\n */\nexport async function init(options: InitOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  logger.info('Initializing DevDoc project...\\n')\n  \n  // Check for docs.json\n  const configPath = path.join(projectRoot, 'docs.json')\n  if (!fs.existsSync(configPath)) {\n    logger.error('docs.json not found in current directory')\n    logger.info('Make sure you are in a DevDoc documentation project directory')\n    process.exit(1)\n  }\n  \n  // Check if .devdoc.json already exists with API key (already deployed)\n  const devdocConfigPath = path.join(projectRoot, '.devdoc.json')\n  if (fs.existsSync(devdocConfigPath) && !options.force) {\n    const existingConfig = fs.readJsonSync(devdocConfigPath) as DevDocConfig\n    if (existingConfig.apiKey) {\n      logger.warn('.devdoc.json already exists with API key (project is deployed)')\n      logger.info('Use --force to overwrite and create a new project')\n      process.exit(1)\n    }\n  }\n  \n  // Load docs.json to get project name\n  let projectName = 'My Documentation'\n  try {\n    const config = await loadConfig(projectRoot)\n    projectName = config.name || projectName\n  } catch {\n    // Use default name\n  }\n  \n  // Generate slug from project name or use provided slug\n  const slug = options.slug || generateSlug(projectName)\n  \n  // Get subdomain - prompt if not provided\n  let subdomain = options.subdomain\n  \n  if (!subdomain) {\n    const suggestedSubdomain = slug\n    console.log('')\n    subdomain = await prompt(`Enter subdomain for ${suggestedSubdomain}.devdoc.sh`, suggestedSubdomain)\n    subdomain = subdomain.toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/^-|-$/g, '')\n  }\n  \n  // Validate subdomain format locally\n  let formatCheck = isValidSubdomainFormat(subdomain)\n  if (!formatCheck.valid) {\n    logger.error(formatCheck.error!)\n    process.exit(1)\n  }\n  \n  // Check subdomain availability via API (server validates blacklist)\n  // Note: This only checks availability, it does NOT reserve the subdomain\n  // Loop until we find an available subdomain\n  let subdomainAvailable = false\n  while (!subdomainAvailable) {\n    logger.info(`Checking if ${subdomain}.devdoc.sh is available...`)\n    const availability = await checkSubdomainAvailability(subdomain, apiUrl)\n    \n    if (!availability.available) {\n      console.log('')\n      logger.warn(availability.error || `Subdomain \"${subdomain}\" is not available`)\n      \n      const suggestion = availability.suggestion || `${subdomain}-docs`\n      console.log('')\n      logger.info(`Suggestion: ${suggestion}.devdoc.sh`)\n      console.log('')\n      \n      // Prompt for new subdomain\n      subdomain = await prompt('Enter a different subdomain', suggestion)\n      subdomain = subdomain.toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/^-|-$/g, '')\n      \n      // Validate format\n      formatCheck = isValidSubdomainFormat(subdomain)\n      if (!formatCheck.valid) {\n        logger.error(formatCheck.error!)\n        continue\n      }\n      \n      console.log('')\n    } else {\n      subdomainAvailable = true\n      logger.success(`✓ ${subdomain}.devdoc.sh is available!`)\n      console.log('')\n    }\n  }\n  \n  // Create .devdoc.json with subdomain (but NO API key - not registered yet)\n  // The subdomain will only be reserved when the user actually deploys\n  const projectId = `${slug}-${generateId()}`\n  const devdocConfig: DevDocConfig = {\n    projectId,\n    name: projectName,\n    slug,\n    subdomain,\n    createdAt: new Date().toISOString(),\n    // Note: No apiKey - subdomain is not reserved until deploy\n  }\n  \n  fs.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 })\n  \n  logger.success('✓ Project initialized')\n  console.log('')\n  console.log('  Subdomain:', `${subdomain}.devdoc.sh`)\n  console.log('  Status:', 'Not yet deployed')\n  console.log('')\n  logger.info('Note: The subdomain is not reserved until you deploy.')\n  logger.info('Run \"devdoc deploy\" to deploy and claim your subdomain.')\n}\n\n/**\n * Generate a URL-friendly slug from a name\n */\nfunction generateSlug(name: string): string {\n  return name\n    .toLowerCase()\n    .replace(/[^a-z0-9]+/g, '-')\n    .replace(/^-|-$/g, '')\n    .substring(0, 50)\n}\n\n/**\n * Generate a short random ID\n */\nfunction generateId(): string {\n  return Math.random().toString(36).substring(2, 8)\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@brainfish-ai/devdoc",
3
- "version": "0.1.24",
3
+ "version": "0.1.26",
4
4
  "description": "Documentation framework for developers. Write docs in MDX, preview locally, deploy to Brainfish.",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
@@ -12,8 +12,8 @@
12
12
  "build:bundle": "node scripts/build.js",
13
13
  "build": "npm run build:cli",
14
14
  "dev": "tsc --watch",
15
- "test:dev": "npm run build:cli && cd ../../devdoc-docs && node ../../packages/devdoc/bin/devdoc.js dev",
16
- "test:check": "npm run build:cli && cd ../../devdoc-docs && node ../../packages/devdoc/bin/devdoc.js check",
15
+ "test:dev": "npm run build:cli && cd ../../devdoc/docs && node ../../packages/devdoc/bin/devdoc.js dev",
16
+ "test:check": "npm run build:cli && cd ../../devdoc/docs && node ../../packages/devdoc/bin/devdoc.js check",
17
17
  "prepublishOnly": "npm run build:bundle"
18
18
  },
19
19
  "keywords": [
@@ -31,7 +31,8 @@
31
31
  },
32
32
  "dependencies": {
33
33
  "commander": "^12.0.0",
34
- "fs-extra": "^11.2.0"
34
+ "fs-extra": "^11.2.0",
35
+ "giget": "^1.2.3"
35
36
  },
36
37
  "devDependencies": {
37
38
  "@types/node": "^20.0.0",
@@ -41,8 +42,7 @@
41
42
  "files": [
42
43
  "bin",
43
44
  "dist",
44
- "renderer",
45
- "templates"
45
+ "renderer"
46
46
  ],
47
47
  "repository": {
48
48
  "type": "git",
@@ -1340,6 +1340,7 @@ function DocsWithMode({
1340
1340
  if (!showGraphQL || !schemaPath) {
1341
1341
  setGraphqlOperations([])
1342
1342
  setGraphqlCollection(null)
1343
+ setSelectedGraphQLOperation(null) // Clear selection when leaving GraphQL tab
1343
1344
  return
1344
1345
  }
1345
1346
 
@@ -1364,6 +1365,44 @@ function DocsWithMode({
1364
1365
  loadGraphQLSchema()
1365
1366
  }, [showGraphQL, schemaPath, schemaEndpoint])
1366
1367
 
1368
+ // Auto-select GraphQL operation from URL hash or default to first operation
1369
+ useEffect(() => {
1370
+ if (!showGraphQL || graphqlOperations.length === 0) {
1371
+ return
1372
+ }
1373
+
1374
+ // Check if there's a selection in the URL hash first
1375
+ const hash = window.location.hash.slice(1) // Remove #
1376
+ if (hash) {
1377
+ const parts = hash.split('/')
1378
+ const hashType = parts[1]
1379
+ const hashId = parts.slice(2).join('/')
1380
+
1381
+ if (hashType === 'endpoint' && hashId) {
1382
+ // Try to find and select the operation from the URL hash
1383
+ const operation = graphqlOperations.find(op => op.id === hashId)
1384
+ if (operation) {
1385
+ setSelectedGraphQLOperation(operation)
1386
+ return
1387
+ }
1388
+ }
1389
+ }
1390
+
1391
+ // If already have a selection, don't override it
1392
+ if (selectedGraphQLOperation) {
1393
+ return
1394
+ }
1395
+
1396
+ // Check if there are doc groups for this tab first
1397
+ const hasGroups = hasDocGroupsForTab(collection?.docGroups, activeTab)
1398
+ if (!hasGroups) {
1399
+ // No doc groups - auto-select first GraphQL operation
1400
+ const firstOperation = graphqlOperations[0]
1401
+ setSelectedGraphQLOperation(firstOperation)
1402
+ window.history.pushState(null, '', `#${activeTab}/endpoint/${firstOperation.id}`)
1403
+ }
1404
+ }, [showGraphQL, graphqlOperations, selectedGraphQLOperation, activeTab, collection?.docGroups])
1405
+
1367
1406
  // Handle GraphQL operation selection from sidebar
1368
1407
  const handleSelectGraphQLOperation = useCallback((request: BrainfishRESTRequest) => {
1369
1408
  // Find the matching GraphQL operation
@@ -1500,10 +1539,10 @@ function DocsWithMode({
1500
1539
  ref={contentRef}
1501
1540
  className={cn(
1502
1541
  "docs-content-area flex-1 bg-background min-w-0",
1503
- (showChangelog || showGraphQL) ? "overflow-hidden flex flex-col" : "overflow-y-auto scroll-smooth"
1542
+ (showChangelog || (showGraphQL && selectedGraphQLOperation)) ? "overflow-hidden flex flex-col" : "overflow-y-auto scroll-smooth"
1504
1543
  )}
1505
1544
  >
1506
- {showGraphQL ? (
1545
+ {showGraphQL && selectedGraphQLOperation ? (
1507
1546
  <div className="flex-1 flex flex-col h-full">
1508
1547
  <GraphQLPlayground
1509
1548
  endpoint={activeGraphQLSchemas[0]?.endpoint || ''}
@@ -77,6 +77,26 @@ export function CollectionTree({
77
77
  }
78
78
  return expanded
79
79
  })
80
+
81
+ // Re-expand all folders when collection folders change (e.g., tab switch)
82
+ // This ensures folders are expanded when switching between tabs with different folder sets
83
+ const folderIds = useMemo(() =>
84
+ collection.folders.map(f => f.id).sort().join(','),
85
+ [collection.folders]
86
+ )
87
+
88
+ useEffect(() => {
89
+ if (level === 0 && collection.folders.length > 0) {
90
+ setExpandedFolders(prev => {
91
+ const newExpanded = new Set(prev)
92
+ // Add all current folder IDs to expanded set
93
+ collection.folders.forEach((folder) => {
94
+ newExpanded.add(folder.id)
95
+ })
96
+ return newExpanded
97
+ })
98
+ }
99
+ }, [folderIds, level, collection.folders])
80
100
 
81
101
  // Auto-expand folder when selectedRequest changes (e.g., via agent navigation)
82
102
  useEffect(() => {
@@ -1,139 +0,0 @@
1
- # My Documentation
2
-
3
- Documentation site built with [DevDoc](https://github.com/brainfish-ai/devdoc).
4
-
5
- [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/clone?repository-url=https://github.com/brainfish-ai/devdoc-starter)
6
-
7
- ## Quick Start
8
-
9
- ### Prerequisites
10
-
11
- - [Node.js](https://nodejs.org/) 18.0 or later
12
- - npm, yarn, or pnpm
13
-
14
- ### Installation
15
-
16
- 1. **Clone this repository**
17
-
18
- ```bash
19
- git clone https://github.com/your-username/your-docs.git
20
- cd your-docs
21
- ```
22
-
23
- Or use the GitHub template button above.
24
-
25
- 2. **Install dependencies**
26
-
27
- ```bash
28
- npm install
29
- ```
30
-
31
- 3. **Start the development server**
32
-
33
- ```bash
34
- npm run dev
35
- ```
36
-
37
- 4. **Open your browser**
38
-
39
- Visit [http://localhost:3000](http://localhost:3000) to see your docs.
40
-
41
- ## Project Structure
42
-
43
- ```
44
- .
45
- ├── docs.json # Site configuration
46
- ├── index.mdx # Homepage
47
- ├── quickstart.mdx # Quickstart guide
48
- ├── guides/
49
- │ ├── overview.mdx # Overview page
50
- │ └── configuration.mdx # Configuration guide
51
- ├── public/
52
- │ ├── logo.svg # Site logo
53
- │ └── favicon.svg # Favicon
54
- ├── package.json
55
- ├── vercel.json # Vercel deployment config
56
- └── README.md
57
- ```
58
-
59
- ## Configuration
60
-
61
- Edit `docs.json` to customize your site:
62
-
63
- ```json
64
- {
65
- "name": "My Documentation",
66
- "logo": "/logo.svg",
67
- "navigation": {
68
- "tabs": [
69
- {
70
- "tab": "Documentation",
71
- "type": "docs",
72
- "groups": [
73
- {
74
- "group": "Getting Started",
75
- "pages": ["index", "quickstart"]
76
- }
77
- ]
78
- }
79
- ]
80
- }
81
- }
82
- ```
83
-
84
- ## Writing Content
85
-
86
- Create `.mdx` files with frontmatter:
87
-
88
- ```mdx
89
- ---
90
- title: Page Title
91
- description: Page description for SEO
92
- ---
93
-
94
- # Your Content Here
95
-
96
- Write Markdown with React components.
97
-
98
- <Note>
99
- Use components like Note, Card, Steps, and more.
100
- </Note>
101
- ```
102
-
103
- ## Available Commands
104
-
105
- | Command | Description |
106
- |---------|-------------|
107
- | `npm run dev` | Start development server |
108
- | `npm run build` | Build for production |
109
- | `npm run start` | Start production server |
110
- | `npm run check` | Validate configuration |
111
-
112
- ## Deployment
113
-
114
- ### Vercel (Recommended)
115
-
116
- Click the "Deploy with Vercel" button above, or:
117
-
118
- 1. Push your code to GitHub
119
- 2. Import the repository in [Vercel](https://vercel.com)
120
- 3. Deploy!
121
-
122
- ### Other Platforms
123
-
124
- ```bash
125
- # Build the site
126
- npm run build
127
-
128
- # Output is in the dist/ directory
129
- ```
130
-
131
- ## Learn More
132
-
133
- - [DevDoc Documentation](https://devdoc.sh/docs)
134
- - [MDX Documentation](https://mdxjs.com/)
135
- - [Vercel Documentation](https://vercel.com/docs)
136
-
137
- ## License
138
-
139
- MIT
@@ -1,4 +0,0 @@
1
- <svg width="32" height="32" viewBox="0 0 32 32" fill="none" xmlns="http://www.w3.org/2000/svg">
2
- <rect width="32" height="32" rx="8" fill="#10b981"/>
3
- <path d="M8 16L14 22L24 10" stroke="white" stroke-width="3" stroke-linecap="round" stroke-linejoin="round"/>
4
- </svg>