Implements Phases 1-8 of the TFTSR implementation plan. Rust backend (Tauri 2.x, src-tauri/): - Multi-provider AI: OpenAI-compatible, Anthropic, Gemini, Mistral, Ollama - PII detection engine: 11 regex patterns with overlap resolution - SQLCipher AES-256 encrypted database with 10 versioned migrations - 28 Tauri IPC commands for triage, analysis, document, and system ops - Ollama: hardware probe, model recommendations, pull/delete with events - RCA and blameless post-mortem Markdown document generators - PDF export via printpdf - Audit log: SHA-256 hash of every external data send - Integration stubs for Confluence, ServiceNow, Azure DevOps (v0.2) Frontend (React 18 + TypeScript + Vite, src/): - 9 pages: full triage workflow NewIssue→LogUpload→Triage→Resolution→RCA→Postmortem→History+Settings - 7 components: ChatWindow, TriageProgress, PiiDiffViewer, DocEditor, HardwareReport, ModelSelector, UI primitives - 3 Zustand stores: session, settings (persisted), history - Type-safe tauriCommands.ts matching Rust backend types exactly - 8 IT domain system prompts (Linux, Windows, Network, K8s, DB, Virt, HW, Obs) DevOps: - .woodpecker/test.yml: rustfmt, clippy, cargo test, tsc, vitest on every push - .woodpecker/release.yml: linux/amd64 + linux/arm64 builds, Gogs release upload Verified: - cargo check: zero errors - tsc --noEmit: zero errors - vitest run: 13/13 unit tests passing Co-Authored-By: Claude Sonnet 4.6 (1M context) <noreply@anthropic.com>
54 lines
2.4 KiB
JavaScript
54 lines
2.4 KiB
JavaScript
import '../utils/dataTransfer/Clipboard.js';
|
|
import { getWindow } from '../utils/misc/getWindow.js';
|
|
import { setUIValueClean, setUISelection, hasUISelection } from './UI.js';
|
|
|
|
const TrackChanges = Symbol('Track programmatic changes for React workaround');
|
|
// When the input event happens in the browser, React executes all event handlers
|
|
// and if they change state of a controlled value, nothing happens.
|
|
// But when we trigger the event handlers in test environment with React@17,
|
|
// the changes are rolled back before the state update is applied.
|
|
// This results in a reset cursor.
|
|
// There might be a better way to work around if we figure out
|
|
// why the batched update is executed differently in our test environment.
|
|
function isReact17Element(element) {
|
|
return Object.getOwnPropertyNames(element).some((k)=>k.startsWith('__react')) && getWindow(element).REACT_VERSION === 17;
|
|
}
|
|
function startTrackValue(element) {
|
|
if (!isReact17Element(element)) {
|
|
return;
|
|
}
|
|
element[TrackChanges] = {
|
|
previousValue: String(element.value),
|
|
tracked: []
|
|
};
|
|
}
|
|
function trackOrSetValue(element, v) {
|
|
var _element_TrackChanges_tracked, _element_TrackChanges;
|
|
(_element_TrackChanges = element[TrackChanges]) === null || _element_TrackChanges === undefined ? undefined : (_element_TrackChanges_tracked = _element_TrackChanges.tracked) === null || _element_TrackChanges_tracked === undefined ? undefined : _element_TrackChanges_tracked.push(v);
|
|
if (!element[TrackChanges]) {
|
|
setUIValueClean(element);
|
|
setUISelection(element, {
|
|
focusOffset: v.length
|
|
});
|
|
}
|
|
}
|
|
function commitValueAfterInput(element, cursorOffset) {
|
|
var _changes_tracked;
|
|
const changes = element[TrackChanges];
|
|
element[TrackChanges] = undefined;
|
|
if (!(changes === null || changes === undefined ? undefined : (_changes_tracked = changes.tracked) === null || _changes_tracked === undefined ? undefined : _changes_tracked.length)) {
|
|
return;
|
|
}
|
|
const isJustReactStateUpdate = changes.tracked.length === 2 && changes.tracked[0] === changes.previousValue && changes.tracked[1] === element.value;
|
|
if (!isJustReactStateUpdate) {
|
|
setUIValueClean(element);
|
|
}
|
|
if (hasUISelection(element)) {
|
|
setUISelection(element, {
|
|
focusOffset: isJustReactStateUpdate ? cursorOffset : element.value.length
|
|
});
|
|
}
|
|
}
|
|
|
|
export { commitValueAfterInput, startTrackValue, trackOrSetValue };
|