agent-ecosystem/src/renderer/utils/chipUtils.ts
iliya 52ef9fd0a8 feat: add FAQ section to README and improve cross-platform handling in code
- Introduced a comprehensive FAQ section in the README to address common user queries regarding app installation, code handling, agent communication, and project management.
- Enhanced cross-platform keyboard shortcut handling in the Electron app for better user experience on macOS and Windows/Linux.
- Updated signal handling in the standalone process to ensure proper shutdown behavior across platforms.
- Improved WSL user resolution logic to support default user retrieval for better compatibility.
- Enhanced notification handling to support cross-platform features and improve user feedback.
- Refactored SSH connection management to include additional key file types and improve authentication handling.
- Updated team management services to ensure consistent process termination across platforms.
- Improved project path handling in team provisioning to accommodate different operating systems.
- Enhanced editor components to utilize shared utility functions for path management, improving code maintainability.
2026-03-02 22:56:56 +02:00

268 lines
8.2 KiB
TypeScript

/**
* Utility functions for working with inline code chip tokens in text.
*/
import { chipToken } from '@renderer/types/inlineChip';
import { getCodeFenceLanguage } from '@renderer/utils/buildSelectionAction';
import { getBasename } from '@shared/utils/platformPath';
import type { InlineChip } from '@renderer/types/inlineChip';
import type { EditorSelectionAction } from '@shared/types/editor';
// =============================================================================
// Chip creation
// =============================================================================
let chipCounter = 0;
/**
* Creates an InlineChip from an EditorSelectionAction.
* Returns null if a chip with the same filePath + line range already exists.
*/
export function createChipFromSelection(
action: EditorSelectionAction,
existingChips: InlineChip[]
): InlineChip | null {
const isFileMention = !action.selectedText || action.fromLine == null || action.toLine == null;
if (isFileMention) {
// File-level mention: deduplicate by filePath + null lines
const isDuplicate = existingChips.some(
(c) => c.filePath === action.filePath && c.fromLine == null
);
if (isDuplicate) return null;
const fileName = getBasename(action.filePath) || 'file';
return {
id: `chip-${++chipCounter}-${Date.now()}`,
filePath: action.filePath,
fileName,
fromLine: null,
toLine: null,
codeText: '',
language: getCodeFenceLanguage(fileName),
displayPath: action.displayPath,
};
}
// Code selection chip
const isDuplicate = existingChips.some(
(c) =>
c.filePath === action.filePath && c.fromLine === action.fromLine && c.toLine === action.toLine
);
if (isDuplicate) return null;
const fileName = getBasename(action.filePath) || 'file';
const language = getCodeFenceLanguage(fileName);
return {
id: `chip-${++chipCounter}-${Date.now()}`,
filePath: action.filePath,
fileName,
fromLine: action.fromLine,
toLine: action.toLine,
codeText: action.selectedText,
language,
};
}
// =============================================================================
// Chip boundary detection
// =============================================================================
export interface ChipBoundary {
start: number;
end: number;
chip: InlineChip;
}
/**
* Finds the chip token boundary that contains or is adjacent to the cursor position.
* Returns null if cursor is not at/inside any chip token.
*/
export function findChipBoundary(
text: string,
chips: InlineChip[],
cursorPos: number
): ChipBoundary | null {
for (const chip of chips) {
const token = chipToken(chip);
let searchFrom = 0;
while (searchFrom < text.length) {
const idx = text.indexOf(token, searchFrom);
if (idx === -1) break;
const end = idx + token.length;
if (cursorPos >= idx && cursorPos <= end) {
return { start: idx, end, chip };
}
searchFrom = idx + 1;
}
}
return null;
}
/**
* Returns true if cursor is strictly inside a chip token (not at boundaries).
*/
export function isInsideChip(text: string, chips: InlineChip[], cursorPos: number): boolean {
const boundary = findChipBoundary(text, chips, cursorPos);
if (!boundary) return false;
return cursorPos > boundary.start && cursorPos < boundary.end;
}
/**
* Snaps cursor to the nearest chip boundary (start or end) if inside a chip.
* Returns the original position if not inside any chip.
*/
export function snapCursorToChipBoundary(
text: string,
chips: InlineChip[],
cursorPos: number
): number {
const boundary = findChipBoundary(text, chips, cursorPos);
if (!boundary) return cursorPos;
if (cursorPos <= boundary.start || cursorPos >= boundary.end) return cursorPos;
const distToStart = cursorPos - boundary.start;
const distToEnd = boundary.end - cursorPos;
return distToStart <= distToEnd ? boundary.start : boundary.end;
}
// =============================================================================
// Reconciliation
// =============================================================================
/**
* Returns only those chips whose tokens are still present in the text.
* Used to keep chip state in sync after paste/cut/undo operations.
*/
export function reconcileChips(oldChips: InlineChip[], newText: string): InlineChip[] {
return oldChips.filter((chip) => newText.includes(chipToken(chip)));
}
/**
* Removes a chip token from text, including a trailing newline if present.
* This prevents orphan blank lines after chip removal.
*/
export function removeChipTokenFromText(text: string, chip: InlineChip): string {
const token = chipToken(chip);
const idx = text.indexOf(token);
if (idx === -1) return text;
const end = idx + token.length;
// Remove trailing newline if present
const removeEnd = end < text.length && text[end] === '\n' ? end + 1 : end;
return text.slice(0, idx) + text.slice(removeEnd);
}
// =============================================================================
// Chip position calculation (mirror div technique)
// =============================================================================
export interface ChipPosition {
chip: InlineChip;
top: number;
left: number;
width: number;
height: number;
}
/**
* Calculates screen positions of chip tokens in textarea using the mirror div technique.
* Creates a temporary mirror div that replicates textarea layout and measures chip spans.
*/
export function calculateChipPositions(
textarea: HTMLTextAreaElement,
text: string,
chips: InlineChip[]
): ChipPosition[] {
if (chips.length === 0) return [];
const cs = window.getComputedStyle(textarea);
const mirror = document.createElement('div');
// Copy all relevant styles to mirror div
mirror.style.font = cs.font;
mirror.style.letterSpacing = cs.letterSpacing;
mirror.style.wordSpacing = cs.wordSpacing;
mirror.style.textIndent = cs.textIndent;
mirror.style.textTransform = cs.textTransform;
mirror.style.tabSize = cs.tabSize;
mirror.style.whiteSpace = cs.whiteSpace;
mirror.style.overflowWrap = cs.overflowWrap;
mirror.style.paddingTop = cs.paddingTop;
mirror.style.paddingRight = cs.paddingRight;
mirror.style.paddingBottom = cs.paddingBottom;
mirror.style.paddingLeft = cs.paddingLeft;
mirror.style.borderTopWidth = cs.borderTopWidth;
mirror.style.borderRightWidth = cs.borderRightWidth;
mirror.style.borderBottomWidth = cs.borderBottomWidth;
mirror.style.borderLeftWidth = cs.borderLeftWidth;
mirror.style.boxSizing = cs.boxSizing;
mirror.style.width = cs.width;
mirror.style.lineHeight = cs.lineHeight;
mirror.style.position = 'absolute';
mirror.style.top = '-9999px';
mirror.style.left = '-9999px';
mirror.style.visibility = 'hidden';
mirror.style.overflow = 'hidden';
mirror.style.height = 'auto';
// Build content with chip tokens wrapped in spans
const chipSpans = new Map<string, HTMLSpanElement>();
const tokenPositions: { chip: InlineChip; token: string; index: number }[] = [];
// Find all chip token positions in text
for (const chip of chips) {
const token = chipToken(chip);
const idx = text.indexOf(token);
if (idx !== -1) {
tokenPositions.push({ chip, token, index: idx });
}
}
// Sort by position in text
tokenPositions.sort((a, b) => a.index - b.index);
// Build mirror content
let lastEnd = 0;
for (const { chip, token, index } of tokenPositions) {
// Text before this chip
if (index > lastEnd) {
const textNode = document.createTextNode(text.slice(lastEnd, index));
mirror.appendChild(textNode);
}
// Chip span
const span = document.createElement('span');
span.textContent = token;
mirror.appendChild(span);
chipSpans.set(chip.id, span);
lastEnd = index + token.length;
}
// Text after last chip
if (lastEnd < text.length) {
mirror.appendChild(document.createTextNode(text.slice(lastEnd)));
}
document.body.appendChild(mirror);
const positions: ChipPosition[] = [];
for (const { chip } of tokenPositions) {
const span = chipSpans.get(chip.id);
if (!span) continue;
positions.push({
chip,
top: span.offsetTop,
left: span.offsetLeft,
width: span.offsetWidth,
height: span.offsetHeight,
});
}
document.body.removeChild(mirror);
return positions;
}