d94a11d466
RotatedImage: размер контейнера через clientWidth/Height (не getBoundingClientRect), чтобы cover при 90°/270° работал под zoom React Flow; убраны отладочные логи. Главное меню в dev: пункт «Вид» с DevTools (Ctrl+Shift+I без пустого application menu). Список проектов: project.list без лицензии; список подгружается при неактивной лицензии; ProjectPicker с подсказками; listProjects пропускает битые zip. Сохранение проектов: atomicReplace — замена zip без rm до commit; восстановление *.dnd.zip.tmp при старте; тесты. EditorApp: блокировка UI при открытых окнах презентации и пульта; стили оверлея. Made-with: Cursor
29 lines
1.2 KiB
TypeScript
29 lines
1.2 KiB
TypeScript
import assert from 'node:assert/strict';
|
|
import fs from 'node:fs';
|
|
import path from 'node:path';
|
|
import test from 'node:test';
|
|
import { fileURLToPath } from 'node:url';
|
|
|
|
const here = path.dirname(fileURLToPath(import.meta.url));
|
|
|
|
void test('projectState: list/get after delete invalidates in-flight initial load (epoch guard)', () => {
|
|
const src = fs.readFileSync(path.join(here, 'projectState.ts'), 'utf8');
|
|
assert.match(src, /projectDataEpochRef/);
|
|
assert.match(src, /const epoch = \+\+projectDataEpochRef\.current/);
|
|
assert.match(src, /if \(projectDataEpochRef\.current !== epoch\) return/);
|
|
assert.match(
|
|
src,
|
|
/const deleteProject = async[\s\S]+?projectDataEpochRef\.current \+= 1[\s\S]+?await api\.invoke/,
|
|
);
|
|
assert.match(
|
|
src,
|
|
/const openProject = async[\s\S]+?projectDataEpochRef\.current \+= 1[\s\S]+?await api\.invoke/,
|
|
);
|
|
assert.match(src, /const refreshProjects = async \(\) => \{[\s\S]+?projectDataEpochRef\.current \+= 1/);
|
|
});
|
|
|
|
void test('ipc router: project.list does not require license', () => {
|
|
const routerSrc = fs.readFileSync(path.join(here, '..', '..', '..', 'main', 'ipc', 'router.ts'), 'utf8');
|
|
assert.match(routerSrc, /if \(channel === ipcChannels\.project\.list\) return false/);
|
|
});
|