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

View File

@@ -0,0 +1,38 @@
import SVGElement from '../svg-element/SVGElement.js';
import SVGStringList from '../../svg/SVGStringList.js';
import * as PropertySymbol from '../../PropertySymbol.js';
import Event from '../../event/Event.js';
/**
* SVG Animation Element.
*
* @see https://developer.mozilla.org/en-US/docs/Web/API/SVGAnimationElement
*/
export default class SVGAnimationElement extends SVGElement {
[PropertySymbol.requiredExtensions]: SVGStringList | null;
[PropertySymbol.systemLanguage]: SVGStringList | null;
get onbegin(): ((event: Event) => void) | null;
set onbegin(value: ((event: Event) => void) | null);
get onend(): ((event: Event) => void) | null;
set onend(value: ((event: Event) => void) | null);
get onrepeat(): ((event: Event) => void) | null;
set onrepeat(value: ((event: Event) => void) | null);
/**
* Returns required extensions.
*
* @returns Required extensions.
*/
get requiredExtensions(): SVGStringList;
/**
* Returns system language.
*
* @returns System language.
*/
get systemLanguage(): SVGStringList;
/**
* Returns target element.
*
* @returns Target element.
*/
get targetElement(): SVGElement | null;
}
//# sourceMappingURL=SVGAnimationElement.d.ts.map

View File

@@ -0,0 +1 @@
{"version":3,"file":"SVGAnimationElement.d.ts","sourceRoot":"","sources":["../../../src/nodes/svg-animation-element/SVGAnimationElement.ts"],"names":[],"mappings":"AAAA,OAAO,UAAU,MAAM,8BAA8B,CAAC;AACtD,OAAO,aAAa,MAAM,4BAA4B,CAAC;AACvD,OAAO,KAAK,cAAc,MAAM,yBAAyB,CAAC;AAC1D,OAAO,KAAK,MAAM,sBAAsB,CAAC;AAGzC;;;;GAIG;AACH,MAAM,CAAC,OAAO,OAAO,mBAAoB,SAAQ,UAAU;IAEnD,CAAC,cAAc,CAAC,kBAAkB,CAAC,EAAE,aAAa,GAAG,IAAI,CAAQ;IACjE,CAAC,cAAc,CAAC,cAAc,CAAC,EAAE,aAAa,GAAG,IAAI,CAAQ;IAMpE,IAAW,OAAO,IAAI,CAAC,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC,GAAG,IAAI,CAEpD;IAED,IAAW,OAAO,CAAC,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC,GAAG,IAAI,EAExD;IAED,IAAW,KAAK,IAAI,CAAC,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC,GAAG,IAAI,CAElD;IAED,IAAW,KAAK,CAAC,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC,GAAG,IAAI,EAEtD;IAED,IAAW,QAAQ,IAAI,CAAC,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC,GAAG,IAAI,CAErD;IAED,IAAW,QAAQ,CAAC,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC,GAAG,IAAI,EAEzD;IAID;;;;OAIG;IACH,IAAW,kBAAkB,IAAI,aAAa,CAY7C;IAED;;;;OAIG;IACH,IAAW,cAAc,IAAI,aAAa,CAYzC;IAED;;;;OAIG;IACH,IAAW,aAAa,IAAI,UAAU,GAAG,IAAI,CAG5C;CACD"}

View File

@@ -0,0 +1,73 @@
import SVGElement from '../svg-element/SVGElement.js';
import SVGStringList from '../../svg/SVGStringList.js';
import * as PropertySymbol from '../../PropertySymbol.js';
import ElementEventAttributeUtility from '../element/ElementEventAttributeUtility.js';
/**
* SVG Animation Element.
*
* @see https://developer.mozilla.org/en-US/docs/Web/API/SVGAnimationElement
*/
export default class SVGAnimationElement extends SVGElement {
// Internal properties
[PropertySymbol.requiredExtensions] = null;
[PropertySymbol.systemLanguage] = null;
// Events
/* eslint-disable jsdoc/require-jsdoc */
get onbegin() {
return ElementEventAttributeUtility.getEventListener(this, 'onbegin');
}
set onbegin(value) {
this[PropertySymbol.propertyEventListeners].set('onbegin', value);
}
get onend() {
return ElementEventAttributeUtility.getEventListener(this, 'onend');
}
set onend(value) {
this[PropertySymbol.propertyEventListeners].set('onend', value);
}
get onrepeat() {
return ElementEventAttributeUtility.getEventListener(this, 'onrepeat');
}
set onrepeat(value) {
this[PropertySymbol.propertyEventListeners].set('onrepeat', value);
}
/* eslint-enable jsdoc/require-jsdoc */
/**
* Returns required extensions.
*
* @returns Required extensions.
*/
get requiredExtensions() {
if (!this[PropertySymbol.requiredExtensions]) {
this[PropertySymbol.requiredExtensions] = new SVGStringList(PropertySymbol.illegalConstructor, this[PropertySymbol.window], {
getAttribute: () => this.getAttribute('requiredExtensions'),
setAttribute: (value) => this.setAttribute('requiredExtensions', value)
});
}
return this[PropertySymbol.requiredExtensions];
}
/**
* Returns system language.
*
* @returns System language.
*/
get systemLanguage() {
if (!this[PropertySymbol.systemLanguage]) {
this[PropertySymbol.systemLanguage] = new SVGStringList(PropertySymbol.illegalConstructor, this[PropertySymbol.window], {
getAttribute: () => this.getAttribute('systemLanguage'),
setAttribute: (value) => this.setAttribute('systemLanguage', value)
});
}
return this[PropertySymbol.systemLanguage];
}
/**
* Returns target element.
*
* @returns Target element.
*/
get targetElement() {
// TODO: Implement targetElement
return null;
}
}
//# sourceMappingURL=SVGAnimationElement.js.map

View File

@@ -0,0 +1 @@
{"version":3,"file":"SVGAnimationElement.js","sourceRoot":"","sources":["../../../src/nodes/svg-animation-element/SVGAnimationElement.ts"],"names":[],"mappings":"AAAA,OAAO,UAAU,MAAM,8BAA8B,CAAC;AACtD,OAAO,aAAa,MAAM,4BAA4B,CAAC;AACvD,OAAO,KAAK,cAAc,MAAM,yBAAyB,CAAC;AAE1D,OAAO,4BAA4B,MAAM,4CAA4C,CAAC;AAEtF;;;;GAIG;AACH,MAAM,CAAC,OAAO,OAAO,mBAAoB,SAAQ,UAAU;IAC1D,sBAAsB;IACf,CAAC,cAAc,CAAC,kBAAkB,CAAC,GAAyB,IAAI,CAAC;IACjE,CAAC,cAAc,CAAC,cAAc,CAAC,GAAyB,IAAI,CAAC;IAEpE,SAAS;IAET,wCAAwC;IAExC,IAAW,OAAO;QACjB,OAAO,4BAA4B,CAAC,gBAAgB,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;IACvE,CAAC;IAED,IAAW,OAAO,CAAC,KAAsC;QACxD,IAAI,CAAC,cAAc,CAAC,sBAAsB,CAAC,CAAC,GAAG,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;IACnE,CAAC;IAED,IAAW,KAAK;QACf,OAAO,4BAA4B,CAAC,gBAAgB,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;IACrE,CAAC;IAED,IAAW,KAAK,CAAC,KAAsC;QACtD,IAAI,CAAC,cAAc,CAAC,sBAAsB,CAAC,CAAC,GAAG,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;IACjE,CAAC;IAED,IAAW,QAAQ;QAClB,OAAO,4BAA4B,CAAC,gBAAgB,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;IACxE,CAAC;IAED,IAAW,QAAQ,CAAC,KAAsC;QACzD,IAAI,CAAC,cAAc,CAAC,sBAAsB,CAAC,CAAC,GAAG,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC;IACpE,CAAC;IAED,uCAAuC;IAEvC;;;;OAIG;IACH,IAAW,kBAAkB;QAC5B,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,kBAAkB,CAAC,EAAE,CAAC;YAC9C,IAAI,CAAC,cAAc,CAAC,kBAAkB,CAAC,GAAG,IAAI,aAAa,CAC1D,cAAc,CAAC,kBAAkB,EACjC,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,EAC3B;gBACC,YAAY,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,oBAAoB,CAAC;gBAC3D,YAAY,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,oBAAoB,EAAE,KAAK,CAAC;aACvE,CACD,CAAC;QACH,CAAC;QACD,OAAO,IAAI,CAAC,cAAc,CAAC,kBAAkB,CAAC,CAAC;IAChD,CAAC;IAED;;;;OAIG;IACH,IAAW,cAAc;QACxB,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC,EAAE,CAAC;YAC1C,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC,GAAG,IAAI,aAAa,CACtD,cAAc,CAAC,kBAAkB,EACjC,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,EAC3B;gBACC,YAAY,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,gBAAgB,CAAC;gBACvD,YAAY,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,gBAAgB,EAAE,KAAK,CAAC;aACnE,CACD,CAAC;QACH,CAAC;QACD,OAAO,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;IAC5C,CAAC;IAED;;;;OAIG;IACH,IAAW,aAAa;QACvB,gCAAgC;QAChC,OAAO,IAAI,CAAC;IACb,CAAC;CACD"}