Filemedium importancesource

export.tsx

commands/export/export.tsx

91
Lines
15104
Bytes
3
Exports
9
Imports
10
Keywords

What this is

This page documents one file from the repository and includes its full source so you can read it without leaving the docs site.

Beginner explanation

This file lives in the command layer. It likely turns a user action into concrete program behavior.

How it is used

Start from the exports list and related files. Those are the easiest clues for where this file fits into the system.

Expert explanation

Architecturally, this file intersects with commands. It contains 91 lines, 9 detected imports, and 3 detected exports.

Important relationships

Detected exports

  • extractFirstPrompt
  • sanitizeFilename
  • call

Keywords

contentresultdatereplacemessagecontextfilenamepadstarttextondone

Detected imports

  • path
  • react
  • ../../components/ExportDialog.js
  • ../../Tool.js
  • ../../types/command.js
  • ../../types/message.js
  • ../../utils/cwd.js
  • ../../utils/exportRenderer.js
  • ../../utils/slowOperations.js

Source notes

This page embeds the full file contents. Small or leaf files are still indexed honestly instead of being over-explained.

Open parent directory

Full source

import { join } from 'path';
import React from 'react';
import { ExportDialog } from '../../components/ExportDialog.js';
import type { ToolUseContext } from '../../Tool.js';
import type { LocalJSXCommandOnDone } from '../../types/command.js';
import type { Message } from '../../types/message.js';
import { getCwd } from '../../utils/cwd.js';
import { renderMessagesToPlainText } from '../../utils/exportRenderer.js';
import { writeFileSync_DEPRECATED } from '../../utils/slowOperations.js';
function formatTimestamp(date: Date): string {
  const year = date.getFullYear();
  const month = String(date.getMonth() + 1).padStart(2, '0');
  const day = String(date.getDate()).padStart(2, '0');
  const hours = String(date.getHours()).padStart(2, '0');
  const minutes = String(date.getMinutes()).padStart(2, '0');
  const seconds = String(date.getSeconds()).padStart(2, '0');
  return `${year}-${month}-${day}-${hours}${minutes}${seconds}`;
}
export function extractFirstPrompt(messages: Message[]): string {
  const firstUserMessage = messages.find(msg => msg.type === 'user');
  if (!firstUserMessage || firstUserMessage.type !== 'user') {
    return '';
  }
  const content = firstUserMessage.message?.content;
  let result = '';
  if (typeof content === 'string') {
    result = content.trim();
  } else if (Array.isArray(content)) {
    const textContent = content.find(item => item.type === 'text');
    if (textContent && 'text' in textContent) {
      result = textContent.text.trim();
    }
  }

  // Take first line only and limit length
  result = result.split('\n')[0] || '';
  if (result.length > 50) {
    result = result.substring(0, 49) + '…';
  }
  return result;
}
export function sanitizeFilename(text: string): string {
  // Replace special characters with hyphens
  return text.toLowerCase().replace(/[^a-z0-9\s-]/g, '') // Remove special chars
  .replace(/\s+/g, '-') // Replace spaces with hyphens
  .replace(/-+/g, '-') // Replace multiple hyphens with single
  .replace(/^-|-$/g, ''); // Remove leading/trailing hyphens
}
async function exportWithReactRenderer(context: ToolUseContext): Promise<string> {
  const tools = context.options.tools || [];
  return renderMessagesToPlainText(context.messages, tools);
}
export async function call(onDone: LocalJSXCommandOnDone, context: ToolUseContext, args: string): Promise<React.ReactNode> {
  // Render the conversation content
  const content = await exportWithReactRenderer(context);

  // If args are provided, write directly to file and skip dialog
  const filename = args.trim();
  if (filename) {
    const finalFilename = filename.endsWith('.txt') ? filename : filename.replace(/\.[^.]+$/, '') + '.txt';
    const filepath = join(getCwd(), finalFilename);
    try {
      writeFileSync_DEPRECATED(filepath, content, {
        encoding: 'utf-8',
        flush: true
      });
      onDone(`Conversation exported to: ${filepath}`);
      return null;
    } catch (error) {
      onDone(`Failed to export conversation: ${error instanceof Error ? error.message : 'Unknown error'}`);
      return null;
    }
  }

  // Generate default filename from first prompt or timestamp
  const firstPrompt = extractFirstPrompt(context.messages);
  const timestamp = formatTimestamp(new Date());
  let defaultFilename: string;
  if (firstPrompt) {
    const sanitized = sanitizeFilename(firstPrompt);
    defaultFilename = sanitized ? `${timestamp}-${sanitized}.txt` : `conversation-${timestamp}.txt`;
  } else {
    defaultFilename = `conversation-${timestamp}.txt`;
  }

  // Return the dialog component when no args provided
  return <ExportDialog content={content} defaultFilename={defaultFilename} onDone={result => {
    onDone(result.message);
  }} />;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["join","React","ExportDialog","ToolUseContext","LocalJSXCommandOnDone","Message","getCwd","renderMessagesToPlainText","writeFileSync_DEPRECATED","formatTimestamp","date","Date","year","getFullYear","month","String","getMonth","padStart","day","getDate","hours","getHours","minutes","getMinutes","seconds","getSeconds","extractFirstPrompt","messages","firstUserMessage","find","msg","type","content","message","result","trim","Array","isArray","textContent","item","text","split","length","substring","sanitizeFilename","toLowerCase","replace","exportWithReactRenderer","context","Promise","tools","options","call","onDone","args","ReactNode","filename","finalFilename","endsWith","filepath","encoding","flush","error","Error","firstPrompt","timestamp","defaultFilename","sanitized"],"sources":["export.tsx"],"sourcesContent":["import { join } from 'path'\nimport React from 'react'\nimport { ExportDialog } from '../../components/ExportDialog.js'\nimport type { ToolUseContext } from '../../Tool.js'\nimport type { LocalJSXCommandOnDone } from '../../types/command.js'\nimport type { Message } from '../../types/message.js'\nimport { getCwd } from '../../utils/cwd.js'\nimport { renderMessagesToPlainText } from '../../utils/exportRenderer.js'\nimport { writeFileSync_DEPRECATED } from '../../utils/slowOperations.js'\n\nfunction formatTimestamp(date: Date): string {\n  const year = date.getFullYear()\n  const month = String(date.getMonth() + 1).padStart(2, '0')\n  const day = String(date.getDate()).padStart(2, '0')\n  const hours = String(date.getHours()).padStart(2, '0')\n  const minutes = String(date.getMinutes()).padStart(2, '0')\n  const seconds = String(date.getSeconds()).padStart(2, '0')\n  return `${year}-${month}-${day}-${hours}${minutes}${seconds}`\n}\n\nexport function extractFirstPrompt(messages: Message[]): string {\n  const firstUserMessage = messages.find(msg => msg.type === 'user')\n\n  if (!firstUserMessage || firstUserMessage.type !== 'user') {\n    return ''\n  }\n\n  const content = firstUserMessage.message?.content\n  let result = ''\n\n  if (typeof content === 'string') {\n    result = content.trim()\n  } else if (Array.isArray(content)) {\n    const textContent = content.find(item => item.type === 'text')\n    if (textContent && 'text' in textContent) {\n      result = textContent.text.trim()\n    }\n  }\n\n  // Take first line only and limit length\n  result = result.split('\\n')[0] || ''\n  if (result.length > 50) {\n    result = result.substring(0, 49) + '…'\n  }\n\n  return result\n}\n\nexport function sanitizeFilename(text: string): string {\n  // Replace special characters with hyphens\n  return text\n    .toLowerCase()\n    .replace(/[^a-z0-9\\s-]/g, '') // Remove special chars\n    .replace(/\\s+/g, '-') // Replace spaces with hyphens\n    .replace(/-+/g, '-') // Replace multiple hyphens with single\n    .replace(/^-|-$/g, '') // Remove leading/trailing hyphens\n}\n\nasync function exportWithReactRenderer(\n  context: ToolUseContext,\n): Promise<string> {\n  const tools = context.options.tools || []\n  return renderMessagesToPlainText(context.messages, tools)\n}\n\nexport async function call(\n  onDone: LocalJSXCommandOnDone,\n  context: ToolUseContext,\n  args: string,\n): Promise<React.ReactNode> {\n  // Render the conversation content\n  const content = await exportWithReactRenderer(context)\n\n  // If args are provided, write directly to file and skip dialog\n  const filename = args.trim()\n  if (filename) {\n    const finalFilename = filename.endsWith('.txt')\n      ? filename\n      : filename.replace(/\\.[^.]+$/, '') + '.txt'\n    const filepath = join(getCwd(), finalFilename)\n\n    try {\n      writeFileSync_DEPRECATED(filepath, content, {\n        encoding: 'utf-8',\n        flush: true,\n      })\n      onDone(`Conversation exported to: ${filepath}`)\n      return null\n    } catch (error) {\n      onDone(\n        `Failed to export conversation: ${error instanceof Error ? error.message : 'Unknown error'}`,\n      )\n      return null\n    }\n  }\n\n  // Generate default filename from first prompt or timestamp\n  const firstPrompt = extractFirstPrompt(context.messages)\n  const timestamp = formatTimestamp(new Date())\n\n  let defaultFilename: string\n  if (firstPrompt) {\n    const sanitized = sanitizeFilename(firstPrompt)\n    defaultFilename = sanitized\n      ? `${timestamp}-${sanitized}.txt`\n      : `conversation-${timestamp}.txt`\n  } else {\n    defaultFilename = `conversation-${timestamp}.txt`\n  }\n\n  // Return the dialog component when no args provided\n  return (\n    <ExportDialog\n      content={content}\n      defaultFilename={defaultFilename}\n      onDone={result => {\n        onDone(result.message)\n      }}\n    />\n  )\n}\n"],"mappings":"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","ignoreList":[]}