feat: Reinitialize frontend with SvelteKit and TypeScript

- Delete old Vite+Svelte frontend
- Initialize new SvelteKit project with TypeScript
- Configure Tailwind CSS v4 + DaisyUI
- Implement JWT authentication with auto-refresh
- Create login page with form validation (Zod)
- Add protected route guards
- Update Docker configuration for single-stage build
- Add E2E tests with Playwright (6/11 passing)
- Fix Svelte 5 reactivity with $state() runes

Known issues:
- 5 E2E tests failing (timing/async issues)
- Token refresh implementation needs debugging
- Validation error display timing
This commit is contained in:
2026-02-17 16:19:59 -05:00
parent 54df6018f5
commit de2d83092e
28274 changed files with 3816354 additions and 90 deletions

137
frontend/node_modules/happy-dom/cjs/nodes/text/Text.cjs generated vendored Normal file
View File

@@ -0,0 +1,137 @@
"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
var ownKeys = function(o) {
ownKeys = Object.getOwnPropertyNames || function (o) {
var ar = [];
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
return ar;
};
return ownKeys(o);
};
return function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
__setModuleDefault(result, mod);
return result;
};
})();
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
const PropertySymbol = __importStar(require("../../PropertySymbol.cjs"));
const CharacterData_js_1 = __importDefault(require("../character-data/CharacterData.cjs"));
const DOMExceptionNameEnum_js_1 = __importDefault(require("../../exception/DOMExceptionNameEnum.cjs"));
const NodeTypeEnum_js_1 = __importDefault(require("../node/NodeTypeEnum.cjs"));
/**
* Text node.
*/
class Text extends CharacterData_js_1.default {
[PropertySymbol.nodeType] = NodeTypeEnum_js_1.default.textNode;
[PropertySymbol.textAreaNode] = null;
[PropertySymbol.styleNode] = null;
/**
* Node name.
*
* @returns Node name.
*/
get nodeName() {
return '#text';
}
/**
* @override
*/
get data() {
return this[PropertySymbol.data];
}
/**
* @override
*/
set data(data) {
super.data = data;
if (this[PropertySymbol.textAreaNode]) {
this[PropertySymbol.textAreaNode][PropertySymbol.resetSelection]();
}
if (this[PropertySymbol.styleNode]) {
this[PropertySymbol.styleNode][PropertySymbol.updateSheet]();
}
}
/**
* Breaks the Text node into two nodes at the specified offset, keeping both nodes in the tree as siblings.
*
* @see https://dom.spec.whatwg.org/#dom-text-splittext
* @param offset Offset.
* @returns New text node.
*/
splitText(offset) {
const length = this[PropertySymbol.data].length;
if (offset < 0 || offset > length) {
throw new this[PropertySymbol.window].DOMException('The index is not in the allowed range.', DOMExceptionNameEnum_js_1.default.indexSizeError);
}
const count = length - offset;
const newData = this.substringData(offset, count);
const newNode = this[PropertySymbol.ownerDocument].createTextNode(newData);
if (this[PropertySymbol.parentNode] !== null) {
this[PropertySymbol.parentNode].insertBefore(newNode, this.nextSibling);
}
this.replaceData(offset, count, '');
return newNode;
}
/**
* Converts to string.
*
* @returns String.
*/
toString() {
return '[object Text]';
}
/**
* @override
*/
[PropertySymbol.cloneNode](deep = false) {
return super[PropertySymbol.cloneNode](deep);
}
/**
* @override
*/
[PropertySymbol.connectedToNode]() {
super[PropertySymbol.connectedToNode]();
if (this[PropertySymbol.textAreaNode]) {
this[PropertySymbol.textAreaNode][PropertySymbol.resetSelection]();
}
if (this[PropertySymbol.styleNode] && this[PropertySymbol.data]) {
this[PropertySymbol.styleNode][PropertySymbol.updateSheet]();
}
}
/**
* @override
*/
[PropertySymbol.disconnectedFromNode]() {
if (this[PropertySymbol.textAreaNode]) {
this[PropertySymbol.textAreaNode][PropertySymbol.resetSelection]();
}
if (this[PropertySymbol.styleNode] && this[PropertySymbol.data]) {
this[PropertySymbol.styleNode][PropertySymbol.updateSheet]();
}
super[PropertySymbol.disconnectedFromNode]();
}
}
exports.default = Text;
//# sourceMappingURL=Text.cjs.map

View File

@@ -0,0 +1 @@
{"version":3,"file":"Text.cjs","sourceRoot":"","sources":["../../../src/nodes/text/Text.ts"],"names":[],"mappings":"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"}

View File

@@ -0,0 +1,55 @@
import * as PropertySymbol from '../../PropertySymbol.cjs';
import CharacterData from '../character-data/CharacterData.cjs';
import HTMLTextAreaElement from '../html-text-area-element/HTMLTextAreaElement.cjs';
import NodeTypeEnum from '../node/NodeTypeEnum.cjs';
import HTMLStyleElement from '../html-style-element/HTMLStyleElement.cjs';
/**
* Text node.
*/
export default class Text extends CharacterData {
cloneNode: (deep?: boolean) => Text;
[PropertySymbol.nodeType]: NodeTypeEnum;
[PropertySymbol.textAreaNode]: HTMLTextAreaElement | null;
[PropertySymbol.styleNode]: HTMLStyleElement | null;
/**
* Node name.
*
* @returns Node name.
*/
get nodeName(): string;
/**
* @override
*/
get data(): string;
/**
* @override
*/
set data(data: string);
/**
* Breaks the Text node into two nodes at the specified offset, keeping both nodes in the tree as siblings.
*
* @see https://dom.spec.whatwg.org/#dom-text-splittext
* @param offset Offset.
* @returns New text node.
*/
splitText(offset: number): Text;
/**
* Converts to string.
*
* @returns String.
*/
toString(): string;
/**
* @override
*/
[PropertySymbol.cloneNode](deep?: boolean): Text;
/**
* @override
*/
[PropertySymbol.connectedToNode](): void;
/**
* @override
*/
[PropertySymbol.disconnectedFromNode](): void;
}
//# sourceMappingURL=Text.d.ts.map

View File

@@ -0,0 +1 @@
{"version":3,"file":"Text.d.ts","sourceRoot":"","sources":["../../../src/nodes/text/Text.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,cAAc,MAAM,yBAAyB,CAAC;AAC1D,OAAO,aAAa,MAAM,oCAAoC,CAAC;AAE/D,OAAO,mBAAmB,MAAM,kDAAkD,CAAC;AACnF,OAAO,YAAY,MAAM,yBAAyB,CAAC;AACnD,OAAO,gBAAgB,MAAM,2CAA2C,CAAC;AAEzE;;GAEG;AACH,MAAM,CAAC,OAAO,OAAO,IAAK,SAAQ,aAAa;IAC/B,SAAS,EAAE,CAAC,IAAI,CAAC,EAAE,OAAO,KAAK,IAAI,CAAC;IACnC,CAAC,cAAc,CAAC,QAAQ,CAAC,eAAyB;IAClD,CAAC,cAAc,CAAC,YAAY,CAAC,EAAE,mBAAmB,GAAG,IAAI,CAAQ;IACjE,CAAC,cAAc,CAAC,SAAS,CAAC,EAAE,gBAAgB,GAAG,IAAI,CAAQ;IAE3E;;;;OAIG;IACH,IAAW,QAAQ,IAAI,MAAM,CAE5B;IAED;;OAEG;IACH,IAAoB,IAAI,IAAI,MAAM,CAEjC;IAED;;OAEG;IACH,IAAoB,IAAI,CAAC,IAAI,EAAE,MAAM,EAUpC;IAED;;;;;;OAMG;IACI,SAAS,CAAC,MAAM,EAAE,MAAM,GAAG,IAAI;IAuBtC;;;;OAIG;IACI,QAAQ,IAAI,MAAM;IAIzB;;OAEG;IACa,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC,IAAI,UAAQ,GAAG,IAAI;IAI9D;;OAEG;IACa,CAAC,cAAc,CAAC,eAAe,CAAC,IAAI,IAAI;IAYxD;;OAEG;IACa,CAAC,cAAc,CAAC,oBAAoB,CAAC,IAAI,IAAI;CAW7D"}