* perf(terminal): smooth layout drags and faster tab switching Defer xterm refit during split, sidebar, and host-tree drags while keeping pane containers in sync with live layout measurements. Refactor TerminalLayer into focused sections with TabBridge/memo optimizations and add the terminal host tree sidebar. Co-authored-by: Cursor <cursoragent@cursor.com> * fix(terminal): keep side panels alive and guard session attach races Prevent terminal boot unmount from leaking backend sessions, keep SFTP/scripts/theme/AI state when switching side tabs, and defer heavy SFTP UI mount so first entry stays responsive. Co-authored-by: Cursor <cursoragent@cursor.com> * perf(terminal): reduce tab switch jank --------- Co-authored-by: Cursor <cursoragent@cursor.com>
37 lines
972 B
TypeScript
37 lines
972 B
TypeScript
import { useSyncExternalStore } from 'react';
|
|
|
|
type Listener = () => void;
|
|
|
|
class HostTreeInlineGroupDeleteStore {
|
|
private targetPath: string | null = null;
|
|
private listeners = new Set<Listener>();
|
|
|
|
getTargetPath = () => this.targetPath;
|
|
|
|
open = (groupPath: string) => {
|
|
this.targetPath = groupPath;
|
|
this.listeners.forEach((listener) => listener());
|
|
};
|
|
|
|
close = () => {
|
|
if (!this.targetPath) return;
|
|
this.targetPath = null;
|
|
this.listeners.forEach((listener) => listener());
|
|
};
|
|
|
|
subscribe = (listener: Listener) => {
|
|
this.listeners.add(listener);
|
|
return () => this.listeners.delete(listener);
|
|
};
|
|
}
|
|
|
|
export const hostTreeInlineGroupDeleteStore = new HostTreeInlineGroupDeleteStore();
|
|
|
|
export const useHostTreeInlineGroupDeleteTarget = () => {
|
|
return useSyncExternalStore(
|
|
hostTreeInlineGroupDeleteStore.subscribe,
|
|
hostTreeInlineGroupDeleteStore.getTargetPath,
|
|
hostTreeInlineGroupDeleteStore.getTargetPath,
|
|
);
|
|
};
|