@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.
- package/README.md +2 -2
- package/dist/cli/commands/create.js +54 -17
- package/dist/cli/commands/dev.js +41 -77
- package/dist/cli/commands/init.js +29 -11
- package/package.json +6 -6
- package/renderer/components/docs-viewer/index.tsx +41 -2
- package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +20 -0
- package/templates/basic/README.md +0 -139
- package/templates/basic/assets/favicon.svg +0 -4
- package/templates/basic/assets/logo.svg +0 -9
- package/templates/basic/docs.json +0 -47
- package/templates/basic/guides/configuration.mdx +0 -149
- package/templates/basic/guides/overview.mdx +0 -96
- package/templates/basic/index.mdx +0 -39
- package/templates/basic/package.json +0 -14
- package/templates/basic/quickstart.mdx +0 -92
- package/templates/basic/vercel.json +0 -6
- package/templates/graphql/README.md +0 -139
- package/templates/graphql/api-reference/schema.graphql +0 -305
- package/templates/graphql/assets/favicon.svg +0 -4
- package/templates/graphql/assets/logo.svg +0 -9
- package/templates/graphql/docs.json +0 -54
- package/templates/graphql/guides/configuration.mdx +0 -149
- package/templates/graphql/guides/overview.mdx +0 -96
- package/templates/graphql/index.mdx +0 -39
- package/templates/graphql/package.json +0 -14
- package/templates/graphql/quickstart.mdx +0 -92
- package/templates/graphql/vercel.json +0 -6
- package/templates/openapi/README.md +0 -139
- package/templates/openapi/api-reference/openapi.json +0 -419
- package/templates/openapi/assets/favicon.svg +0 -4
- package/templates/openapi/assets/logo.svg +0 -9
- package/templates/openapi/docs.json +0 -61
- package/templates/openapi/guides/configuration.mdx +0 -149
- package/templates/openapi/guides/overview.mdx +0 -96
- package/templates/openapi/index.mdx +0 -39
- package/templates/openapi/package.json +0 -14
- package/templates/openapi/quickstart.mdx +0 -92
- 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
|
-
|
|
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
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
logger_1.logger.
|
|
157
|
-
|
|
158
|
-
|
|
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.
|
|
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
|
|
16
|
-
"test:check": "npm run build:cli && cd ../../devdoc
|
|
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
|
-
[](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
|