chore: follow the general comments style
This commit is contained in:
@@ -34,11 +34,17 @@
|
|||||||
* A breadcrumb item representing a tracked section
|
* A breadcrumb item representing a tracked section
|
||||||
*/
|
*/
|
||||||
export interface BreadcrumbItem {
|
export interface BreadcrumbItem {
|
||||||
/** Unique index for ordering */
|
/**
|
||||||
|
* Unique index for ordering
|
||||||
|
*/
|
||||||
index: number;
|
index: number;
|
||||||
/** Display title for the breadcrumb */
|
/**
|
||||||
|
* Display title for the breadcrumb
|
||||||
|
*/
|
||||||
title: string;
|
title: string;
|
||||||
/** DOM element to track */
|
/**
|
||||||
|
* DOM element to track
|
||||||
|
*/
|
||||||
element: HTMLElement;
|
element: HTMLElement;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -50,21 +56,37 @@ export interface BreadcrumbItem {
|
|||||||
* past while moving down the page.
|
* past while moving down the page.
|
||||||
*/
|
*/
|
||||||
class ScrollBreadcrumbsStore {
|
class ScrollBreadcrumbsStore {
|
||||||
/** All tracked breadcrumb items */
|
/**
|
||||||
|
* All tracked breadcrumb items
|
||||||
|
*/
|
||||||
#items = $state<BreadcrumbItem[]>([]);
|
#items = $state<BreadcrumbItem[]>([]);
|
||||||
/** Set of indices that have scrolled past (exited viewport while scrolling down) */
|
/**
|
||||||
|
* Set of indices that have scrolled past (exited viewport while scrolling down)
|
||||||
|
*/
|
||||||
#scrolledPast = $state<Set<number>>(new Set());
|
#scrolledPast = $state<Set<number>>(new Set());
|
||||||
/** Intersection Observer instance */
|
/**
|
||||||
|
* Intersection Observer instance
|
||||||
|
*/
|
||||||
#observer: IntersectionObserver | null = null;
|
#observer: IntersectionObserver | null = null;
|
||||||
/** Offset for smooth scrolling (sticky header height) */
|
/**
|
||||||
|
* Offset for smooth scrolling (sticky header height)
|
||||||
|
*/
|
||||||
#scrollOffset = 0;
|
#scrollOffset = 0;
|
||||||
/** Current scroll direction */
|
/**
|
||||||
|
* Current scroll direction
|
||||||
|
*/
|
||||||
#isScrollingDown = $state(false);
|
#isScrollingDown = $state(false);
|
||||||
/** Previous scroll Y position to determine direction */
|
/**
|
||||||
|
* Previous scroll Y position to determine direction
|
||||||
|
*/
|
||||||
#prevScrollY = 0;
|
#prevScrollY = 0;
|
||||||
/** Throttled scroll handler */
|
/**
|
||||||
|
* Throttled scroll handler
|
||||||
|
*/
|
||||||
#handleScroll: (() => void) | null = null;
|
#handleScroll: (() => void) | null = null;
|
||||||
/** Listener count for cleanup */
|
/**
|
||||||
|
* Listener count for cleanup
|
||||||
|
*/
|
||||||
#listenerCount = 0;
|
#listenerCount = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -141,17 +163,23 @@ class ScrollBreadcrumbsStore {
|
|||||||
this.#detachScrollListener();
|
this.#detachScrollListener();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** All tracked items sorted by index */
|
/**
|
||||||
|
* All tracked items sorted by index
|
||||||
|
*/
|
||||||
get items(): BreadcrumbItem[] {
|
get items(): BreadcrumbItem[] {
|
||||||
return this.#items.slice().sort((a, b) => a.index - b.index);
|
return this.#items.slice().sort((a, b) => a.index - b.index);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Items that have scrolled past viewport top (visible in breadcrumbs) */
|
/**
|
||||||
|
* Items that have scrolled past viewport top (visible in breadcrumbs)
|
||||||
|
*/
|
||||||
get scrolledPastItems(): BreadcrumbItem[] {
|
get scrolledPastItems(): BreadcrumbItem[] {
|
||||||
return this.items.filter(item => this.#scrolledPast.has(item.index));
|
return this.items.filter(item => this.#scrolledPast.has(item.index));
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Index of the most recently scrolled item (active section) */
|
/**
|
||||||
|
* Index of the most recently scrolled item (active section)
|
||||||
|
*/
|
||||||
get activeIndex(): number | null {
|
get activeIndex(): number | null {
|
||||||
const past = this.scrolledPastItems;
|
const past = this.scrolledPastItems;
|
||||||
return past.length > 0 ? past[past.length - 1].index : null;
|
return past.length > 0 ? past[past.length - 1].index : null;
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
/** @vitest-environment jsdom */
|
/**
|
||||||
|
* @vitest-environment jsdom
|
||||||
|
*/
|
||||||
import {
|
import {
|
||||||
afterEach,
|
afterEach,
|
||||||
beforeEach,
|
beforeEach,
|
||||||
|
|||||||
@@ -97,16 +97,24 @@ export interface ProxyFontsParams extends QueryParams {
|
|||||||
* Includes pagination metadata alongside font data
|
* Includes pagination metadata alongside font data
|
||||||
*/
|
*/
|
||||||
export interface ProxyFontsResponse {
|
export interface ProxyFontsResponse {
|
||||||
/** Array of unified font objects */
|
/**
|
||||||
|
* List of font objects returned by the proxy
|
||||||
|
*/
|
||||||
fonts: UnifiedFont[];
|
fonts: UnifiedFont[];
|
||||||
|
|
||||||
/** Total number of fonts matching the query */
|
/**
|
||||||
|
* Total number of matching fonts (ignoring limit/offset)
|
||||||
|
*/
|
||||||
total: number;
|
total: number;
|
||||||
|
|
||||||
/** Limit used for this request */
|
/**
|
||||||
|
* Page size used for the request
|
||||||
|
*/
|
||||||
limit: number;
|
limit: number;
|
||||||
|
|
||||||
/** Offset used for this request */
|
/**
|
||||||
|
* Start index for the result set
|
||||||
|
*/
|
||||||
offset: number;
|
offset: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -3,7 +3,9 @@ import type {
|
|||||||
UnifiedFont,
|
UnifiedFont,
|
||||||
} from '../../model';
|
} from '../../model';
|
||||||
|
|
||||||
/** Valid font weight values (100-900 in increments of 100) */
|
/**
|
||||||
|
* Valid font weight values (100-900 in increments of 100)
|
||||||
|
*/
|
||||||
const SIZES = [100, 200, 300, 400, 500, 600, 700, 800, 900];
|
const SIZES = [100, 200, 300, 400, 500, 600, 700, 800, 900];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,31 +1,3 @@
|
|||||||
/**
|
|
||||||
* Mock font filter data
|
|
||||||
*
|
|
||||||
* Factory functions and preset mock data for font-related filters.
|
|
||||||
* Used in Storybook stories for font filtering components.
|
|
||||||
*
|
|
||||||
* ## Usage
|
|
||||||
*
|
|
||||||
* ```ts
|
|
||||||
* import {
|
|
||||||
* createMockFilter,
|
|
||||||
* MOCK_FILTERS,
|
|
||||||
* } from '$entities/Font/lib/mocks';
|
|
||||||
*
|
|
||||||
* // Create a custom filter
|
|
||||||
* const customFilter = createMockFilter({
|
|
||||||
* properties: [
|
|
||||||
* { id: 'option1', name: 'Option 1', value: 'option1' },
|
|
||||||
* { id: 'option2', name: 'Option 2', value: 'option2', selected: true },
|
|
||||||
* ],
|
|
||||||
* });
|
|
||||||
*
|
|
||||||
* // Use preset filters
|
|
||||||
* const categoriesFilter = MOCK_FILTERS.categories;
|
|
||||||
* const subsetsFilter = MOCK_FILTERS.subsets;
|
|
||||||
* ```
|
|
||||||
*/
|
|
||||||
|
|
||||||
import type {
|
import type {
|
||||||
FontCategory,
|
FontCategory,
|
||||||
FontProvider,
|
FontProvider,
|
||||||
@@ -34,13 +6,13 @@ import type {
|
|||||||
import type { Property } from '$shared/lib';
|
import type { Property } from '$shared/lib';
|
||||||
import { createFilter } from '$shared/lib';
|
import { createFilter } from '$shared/lib';
|
||||||
|
|
||||||
// TYPE DEFINITIONS
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Options for creating a mock filter
|
* Options for creating a mock filter
|
||||||
*/
|
*/
|
||||||
export interface MockFilterOptions {
|
export interface MockFilterOptions {
|
||||||
/** Filter properties */
|
/**
|
||||||
|
* Initial set of properties for the mock filter
|
||||||
|
*/
|
||||||
properties: Property<string>[];
|
properties: Property<string>[];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -48,16 +20,20 @@ export interface MockFilterOptions {
|
|||||||
* Preset mock filters for font filtering
|
* Preset mock filters for font filtering
|
||||||
*/
|
*/
|
||||||
export interface MockFilters {
|
export interface MockFilters {
|
||||||
/** Provider filter (Google, Fontshare) */
|
/**
|
||||||
|
* Provider filter (Google, Fontshare)
|
||||||
|
*/
|
||||||
providers: ReturnType<typeof createFilter<'google' | 'fontshare'>>;
|
providers: ReturnType<typeof createFilter<'google' | 'fontshare'>>;
|
||||||
/** Category filter (sans-serif, serif, display, etc.) */
|
/**
|
||||||
|
* Category filter (sans-serif, serif, display, etc.)
|
||||||
|
*/
|
||||||
categories: ReturnType<typeof createFilter<FontCategory>>;
|
categories: ReturnType<typeof createFilter<FontCategory>>;
|
||||||
/** Subset filter (latin, latin-ext, cyrillic, etc.) */
|
/**
|
||||||
|
* Subset filter (latin, latin-ext, cyrillic, etc.)
|
||||||
|
*/
|
||||||
subsets: ReturnType<typeof createFilter<FontSubset>>;
|
subsets: ReturnType<typeof createFilter<FontSubset>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
// FONT CATEGORIES
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Unified categories (combines both providers)
|
* Unified categories (combines both providers)
|
||||||
*/
|
*/
|
||||||
@@ -71,8 +47,6 @@ export const UNIFIED_CATEGORIES: Property<FontCategory>[] = [
|
|||||||
{ id: 'script', name: 'Script', value: 'script' },
|
{ id: 'script', name: 'Script', value: 'script' },
|
||||||
];
|
];
|
||||||
|
|
||||||
// FONT SUBSETS
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Common font subsets
|
* Common font subsets
|
||||||
*/
|
*/
|
||||||
@@ -85,8 +59,6 @@ export const FONT_SUBSETS: Property<FontSubset>[] = [
|
|||||||
{ id: 'devanagari', name: 'Devanagari', value: 'devanagari' },
|
{ id: 'devanagari', name: 'Devanagari', value: 'devanagari' },
|
||||||
];
|
];
|
||||||
|
|
||||||
// FONT PROVIDERS
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Font providers
|
* Font providers
|
||||||
*/
|
*/
|
||||||
@@ -95,8 +67,6 @@ export const FONT_PROVIDERS: Property<FontProvider>[] = [
|
|||||||
{ id: 'fontshare', name: 'Fontshare', value: 'fontshare' },
|
{ id: 'fontshare', name: 'Fontshare', value: 'fontshare' },
|
||||||
];
|
];
|
||||||
|
|
||||||
// FILTER FACTORIES
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a mock filter from properties
|
* Create a mock filter from properties
|
||||||
*/
|
*/
|
||||||
@@ -139,8 +109,6 @@ export function createProvidersFilter(options?: { selected?: FontProvider[] }) {
|
|||||||
return createFilter<FontProvider>({ properties });
|
return createFilter<FontProvider>({ properties });
|
||||||
}
|
}
|
||||||
|
|
||||||
// PRESET FILTERS
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Preset mock filters - use these directly in stories
|
* Preset mock filters - use these directly in stories
|
||||||
*/
|
*/
|
||||||
@@ -216,8 +184,6 @@ export const MOCK_FILTERS_ALL_SELECTED: MockFilters = {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
// GENERIC FILTER MOCKS
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a mock filter with generic string properties
|
* Create a mock filter with generic string properties
|
||||||
* Useful for testing generic filter components
|
* Useful for testing generic filter components
|
||||||
@@ -239,7 +205,9 @@ export function createGenericFilter(
|
|||||||
* Preset generic filters for testing
|
* Preset generic filters for testing
|
||||||
*/
|
*/
|
||||||
export const GENERIC_FILTERS = {
|
export const GENERIC_FILTERS = {
|
||||||
/** Small filter with 3 items */
|
/**
|
||||||
|
* Small filter with 3 items
|
||||||
|
*/
|
||||||
small: createFilter({
|
small: createFilter({
|
||||||
properties: [
|
properties: [
|
||||||
{ id: 'option-1', name: 'Option 1', value: 'option-1' },
|
{ id: 'option-1', name: 'Option 1', value: 'option-1' },
|
||||||
@@ -247,7 +215,9 @@ export const GENERIC_FILTERS = {
|
|||||||
{ id: 'option-3', name: 'Option 3', value: 'option-3' },
|
{ id: 'option-3', name: 'Option 3', value: 'option-3' },
|
||||||
],
|
],
|
||||||
}),
|
}),
|
||||||
/** Medium filter with 6 items */
|
/**
|
||||||
|
* Medium filter with 6 items
|
||||||
|
*/
|
||||||
medium: createFilter({
|
medium: createFilter({
|
||||||
properties: [
|
properties: [
|
||||||
{ id: 'alpha', name: 'Alpha', value: 'alpha' },
|
{ id: 'alpha', name: 'Alpha', value: 'alpha' },
|
||||||
@@ -258,7 +228,9 @@ export const GENERIC_FILTERS = {
|
|||||||
{ id: 'zeta', name: 'Zeta', value: 'zeta' },
|
{ id: 'zeta', name: 'Zeta', value: 'zeta' },
|
||||||
],
|
],
|
||||||
}),
|
}),
|
||||||
/** Large filter with 12 items */
|
/**
|
||||||
|
* Large filter with 12 items
|
||||||
|
*/
|
||||||
large: createFilter({
|
large: createFilter({
|
||||||
properties: [
|
properties: [
|
||||||
{ id: 'jan', name: 'January', value: 'jan' },
|
{ id: 'jan', name: 'January', value: 'jan' },
|
||||||
@@ -275,7 +247,9 @@ export const GENERIC_FILTERS = {
|
|||||||
{ id: 'dec', name: 'December', value: 'dec' },
|
{ id: 'dec', name: 'December', value: 'dec' },
|
||||||
],
|
],
|
||||||
}),
|
}),
|
||||||
/** Filter with some pre-selected items */
|
/**
|
||||||
|
* Filter with some pre-selected items
|
||||||
|
*/
|
||||||
partial: createFilter({
|
partial: createFilter({
|
||||||
properties: [
|
properties: [
|
||||||
{ id: 'red', name: 'Red', value: 'red', selected: true },
|
{ id: 'red', name: 'Red', value: 'red', selected: true },
|
||||||
@@ -284,7 +258,9 @@ export const GENERIC_FILTERS = {
|
|||||||
{ id: 'yellow', name: 'Yellow', value: 'yellow', selected: false },
|
{ id: 'yellow', name: 'Yellow', value: 'yellow', selected: false },
|
||||||
],
|
],
|
||||||
}),
|
}),
|
||||||
/** Filter with all items selected */
|
/**
|
||||||
|
* Filter with all items selected
|
||||||
|
*/
|
||||||
allSelected: createFilter({
|
allSelected: createFilter({
|
||||||
properties: [
|
properties: [
|
||||||
{ id: 'cat', name: 'Cat', value: 'cat', selected: true },
|
{ id: 'cat', name: 'Cat', value: 'cat', selected: true },
|
||||||
@@ -292,7 +268,9 @@ export const GENERIC_FILTERS = {
|
|||||||
{ id: 'bird', name: 'Bird', value: 'bird', selected: true },
|
{ id: 'bird', name: 'Bird', value: 'bird', selected: true },
|
||||||
],
|
],
|
||||||
}),
|
}),
|
||||||
/** Empty filter (no items) */
|
/**
|
||||||
|
* Empty filter (no items)
|
||||||
|
*/
|
||||||
empty: createFilter({
|
empty: createFilter({
|
||||||
properties: [],
|
properties: [],
|
||||||
}),
|
}),
|
||||||
|
|||||||
@@ -51,23 +51,41 @@ import type {
|
|||||||
* Options for creating a mock UnifiedFont
|
* Options for creating a mock UnifiedFont
|
||||||
*/
|
*/
|
||||||
export interface MockUnifiedFontOptions {
|
export interface MockUnifiedFontOptions {
|
||||||
/** Unique identifier (default: derived from name) */
|
/**
|
||||||
|
* Unique identifier (default: derived from name)
|
||||||
|
*/
|
||||||
id?: string;
|
id?: string;
|
||||||
/** Font display name (default: 'Mock Font') */
|
/**
|
||||||
|
* Font display name (default: 'Mock Font')
|
||||||
|
*/
|
||||||
name?: string;
|
name?: string;
|
||||||
/** Font provider (default: 'google') */
|
/**
|
||||||
|
* Font provider (default: 'google')
|
||||||
|
*/
|
||||||
provider?: FontProvider;
|
provider?: FontProvider;
|
||||||
/** Font category (default: 'sans-serif') */
|
/**
|
||||||
|
* Font category (default: 'sans-serif')
|
||||||
|
*/
|
||||||
category?: FontCategory;
|
category?: FontCategory;
|
||||||
/** Font subsets (default: ['latin']) */
|
/**
|
||||||
|
* Font subsets (default: ['latin'])
|
||||||
|
*/
|
||||||
subsets?: FontSubset[];
|
subsets?: FontSubset[];
|
||||||
/** Font variants (default: ['regular', '700', 'italic', '700italic']) */
|
/**
|
||||||
|
* Font variants (default: ['regular', '700', 'italic', '700italic'])
|
||||||
|
*/
|
||||||
variants?: FontVariant[];
|
variants?: FontVariant[];
|
||||||
/** Style URLs (if not provided, mock URLs are generated) */
|
/**
|
||||||
|
* Style URLs (if not provided, mock URLs are generated)
|
||||||
|
*/
|
||||||
styles?: FontStyleUrls;
|
styles?: FontStyleUrls;
|
||||||
/** Metadata overrides */
|
/**
|
||||||
|
* Metadata overrides
|
||||||
|
*/
|
||||||
metadata?: Partial<FontMetadata>;
|
metadata?: Partial<FontMetadata>;
|
||||||
/** Features overrides */
|
/**
|
||||||
|
* Features overrides
|
||||||
|
*/
|
||||||
features?: Partial<FontFeatures>;
|
features?: Partial<FontFeatures>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,4 @@
|
|||||||
/**
|
/**
|
||||||
* ============================================================================
|
|
||||||
* MOCK FONT STORE HELPERS
|
|
||||||
* ============================================================================
|
|
||||||
*
|
|
||||||
* Factory functions and preset mock data for TanStack Query stores and state management.
|
* Factory functions and preset mock data for TanStack Query stores and state management.
|
||||||
* Used in Storybook stories for components that use reactive stores.
|
* Used in Storybook stories for components that use reactive stores.
|
||||||
*
|
*
|
||||||
@@ -35,27 +31,73 @@ import {
|
|||||||
generateMockFonts,
|
generateMockFonts,
|
||||||
} from './fonts.mock';
|
} from './fonts.mock';
|
||||||
|
|
||||||
// TANSTACK QUERY MOCK TYPES
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Mock TanStack Query state
|
* Mock TanStack Query state
|
||||||
*/
|
*/
|
||||||
export interface MockQueryState<TData = unknown, TError = Error> {
|
export interface MockQueryState<TData = unknown, TError = Error> {
|
||||||
|
/**
|
||||||
|
* Primary query status (pending, success, error)
|
||||||
|
*/
|
||||||
status: QueryStatus;
|
status: QueryStatus;
|
||||||
|
/**
|
||||||
|
* Payload data (present on success)
|
||||||
|
*/
|
||||||
data?: TData;
|
data?: TData;
|
||||||
|
/**
|
||||||
|
* Caught error object (present on error)
|
||||||
|
*/
|
||||||
error?: TError;
|
error?: TError;
|
||||||
|
/**
|
||||||
|
* True if initial load is in progress
|
||||||
|
*/
|
||||||
isLoading?: boolean;
|
isLoading?: boolean;
|
||||||
|
/**
|
||||||
|
* True if background fetch is in progress
|
||||||
|
*/
|
||||||
isFetching?: boolean;
|
isFetching?: boolean;
|
||||||
|
/**
|
||||||
|
* True if query resolved successfully
|
||||||
|
*/
|
||||||
isSuccess?: boolean;
|
isSuccess?: boolean;
|
||||||
|
/**
|
||||||
|
* True if query failed
|
||||||
|
*/
|
||||||
isError?: boolean;
|
isError?: boolean;
|
||||||
|
/**
|
||||||
|
* True if query is waiting to be executed
|
||||||
|
*/
|
||||||
isPending?: boolean;
|
isPending?: boolean;
|
||||||
|
/**
|
||||||
|
* Timestamp of last successful data retrieval
|
||||||
|
*/
|
||||||
dataUpdatedAt?: number;
|
dataUpdatedAt?: number;
|
||||||
|
/**
|
||||||
|
* Timestamp of last recorded error
|
||||||
|
*/
|
||||||
errorUpdatedAt?: number;
|
errorUpdatedAt?: number;
|
||||||
|
/**
|
||||||
|
* Total number of consecutive failures
|
||||||
|
*/
|
||||||
failureCount?: number;
|
failureCount?: number;
|
||||||
|
/**
|
||||||
|
* Detailed reason for the last failure
|
||||||
|
*/
|
||||||
failureReason?: TError;
|
failureReason?: TError;
|
||||||
|
/**
|
||||||
|
* Number of times an error has been caught
|
||||||
|
*/
|
||||||
errorUpdateCount?: number;
|
errorUpdateCount?: number;
|
||||||
|
/**
|
||||||
|
* True if currently refetching in background
|
||||||
|
*/
|
||||||
isRefetching?: boolean;
|
isRefetching?: boolean;
|
||||||
|
/**
|
||||||
|
* True if refetch attempt failed
|
||||||
|
*/
|
||||||
isRefetchError?: boolean;
|
isRefetchError?: boolean;
|
||||||
|
/**
|
||||||
|
* True if query is paused (e.g. offline)
|
||||||
|
*/
|
||||||
isPaused?: boolean;
|
isPaused?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -63,26 +105,72 @@ export interface MockQueryState<TData = unknown, TError = Error> {
|
|||||||
* Mock TanStack Query observer result
|
* Mock TanStack Query observer result
|
||||||
*/
|
*/
|
||||||
export interface MockQueryObserverResult<TData = unknown, TError = Error> {
|
export interface MockQueryObserverResult<TData = unknown, TError = Error> {
|
||||||
|
/**
|
||||||
|
* Current observer status
|
||||||
|
*/
|
||||||
status?: QueryStatus;
|
status?: QueryStatus;
|
||||||
|
/**
|
||||||
|
* Cached or active data payload
|
||||||
|
*/
|
||||||
data?: TData;
|
data?: TData;
|
||||||
|
/**
|
||||||
|
* Caught error from the observer
|
||||||
|
*/
|
||||||
error?: TError;
|
error?: TError;
|
||||||
|
/**
|
||||||
|
* Loading flag for the observer
|
||||||
|
*/
|
||||||
isLoading?: boolean;
|
isLoading?: boolean;
|
||||||
|
/**
|
||||||
|
* Fetching flag for the observer
|
||||||
|
*/
|
||||||
isFetching?: boolean;
|
isFetching?: boolean;
|
||||||
|
/**
|
||||||
|
* Success flag for the observer
|
||||||
|
*/
|
||||||
isSuccess?: boolean;
|
isSuccess?: boolean;
|
||||||
|
/**
|
||||||
|
* Error flag for the observer
|
||||||
|
*/
|
||||||
isError?: boolean;
|
isError?: boolean;
|
||||||
|
/**
|
||||||
|
* Pending flag for the observer
|
||||||
|
*/
|
||||||
isPending?: boolean;
|
isPending?: boolean;
|
||||||
|
/**
|
||||||
|
* Last update time for data
|
||||||
|
*/
|
||||||
dataUpdatedAt?: number;
|
dataUpdatedAt?: number;
|
||||||
|
/**
|
||||||
|
* Last update time for error
|
||||||
|
*/
|
||||||
errorUpdatedAt?: number;
|
errorUpdatedAt?: number;
|
||||||
|
/**
|
||||||
|
* Consecutive failure count
|
||||||
|
*/
|
||||||
failureCount?: number;
|
failureCount?: number;
|
||||||
|
/**
|
||||||
|
* Failure reason object
|
||||||
|
*/
|
||||||
failureReason?: TError;
|
failureReason?: TError;
|
||||||
|
/**
|
||||||
|
* Error count for the observer
|
||||||
|
*/
|
||||||
errorUpdateCount?: number;
|
errorUpdateCount?: number;
|
||||||
|
/**
|
||||||
|
* Refetching flag
|
||||||
|
*/
|
||||||
isRefetching?: boolean;
|
isRefetching?: boolean;
|
||||||
|
/**
|
||||||
|
* Refetch error flag
|
||||||
|
*/
|
||||||
isRefetchError?: boolean;
|
isRefetchError?: boolean;
|
||||||
|
/**
|
||||||
|
* Paused flag
|
||||||
|
*/
|
||||||
isPaused?: boolean;
|
isPaused?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
// TANSTACK QUERY MOCK FACTORIES
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a mock query state for TanStack Query
|
* Create a mock query state for TanStack Query
|
||||||
*/
|
*/
|
||||||
@@ -138,33 +226,53 @@ export function createSuccessState<TData>(data: TData): MockQueryObserverResult<
|
|||||||
return createMockQueryState<TData>({ status: 'success', data, error: undefined });
|
return createMockQueryState<TData>({ status: 'success', data, error: undefined });
|
||||||
}
|
}
|
||||||
|
|
||||||
// FONT STORE MOCKS
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Mock UnifiedFontStore state
|
* Mock UnifiedFontStore state
|
||||||
*/
|
*/
|
||||||
export interface MockFontStoreState {
|
export interface MockFontStoreState {
|
||||||
/** All cached fonts */
|
/**
|
||||||
|
* Map of mock fonts indexed by ID
|
||||||
|
*/
|
||||||
fonts: Record<string, UnifiedFont>;
|
fonts: Record<string, UnifiedFont>;
|
||||||
/** Current page */
|
/**
|
||||||
|
* Currently active page number
|
||||||
|
*/
|
||||||
page: number;
|
page: number;
|
||||||
/** Total pages available */
|
/**
|
||||||
|
* Total number of pages calculated from limit
|
||||||
|
*/
|
||||||
totalPages: number;
|
totalPages: number;
|
||||||
/** Items per page */
|
/**
|
||||||
|
* Number of items per page
|
||||||
|
*/
|
||||||
limit: number;
|
limit: number;
|
||||||
/** Total font count */
|
/**
|
||||||
|
* Total number of available fonts
|
||||||
|
*/
|
||||||
total: number;
|
total: number;
|
||||||
/** Loading state */
|
/**
|
||||||
|
* Store-level loading status
|
||||||
|
*/
|
||||||
isLoading: boolean;
|
isLoading: boolean;
|
||||||
/** Error state */
|
/**
|
||||||
|
* Caught error object
|
||||||
|
*/
|
||||||
error: Error | null;
|
error: Error | null;
|
||||||
/** Search query */
|
/**
|
||||||
|
* Mock search filter string
|
||||||
|
*/
|
||||||
searchQuery: string;
|
searchQuery: string;
|
||||||
/** Selected provider */
|
/**
|
||||||
|
* Mock provider filter selection
|
||||||
|
*/
|
||||||
provider: 'google' | 'fontshare' | 'all';
|
provider: 'google' | 'fontshare' | 'all';
|
||||||
/** Selected category */
|
/**
|
||||||
|
* Mock category filter selection
|
||||||
|
*/
|
||||||
category: string | null;
|
category: string | null;
|
||||||
/** Selected subset */
|
/**
|
||||||
|
* Mock subset filter selection
|
||||||
|
*/
|
||||||
subset: string | null;
|
subset: string | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -210,10 +318,12 @@ export function createMockFontStoreState(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Preset font store states
|
* Preset font store states for UI testing
|
||||||
*/
|
*/
|
||||||
export const MOCK_FONT_STORE_STATES = {
|
export const MOCK_FONT_STORE_STATES = {
|
||||||
/** Initial loading state */
|
/**
|
||||||
|
* Initial loading state with no data
|
||||||
|
*/
|
||||||
loading: createMockFontStoreState({
|
loading: createMockFontStoreState({
|
||||||
isLoading: true,
|
isLoading: true,
|
||||||
fonts: {},
|
fonts: {},
|
||||||
@@ -221,7 +331,9 @@ export const MOCK_FONT_STORE_STATES = {
|
|||||||
page: 1,
|
page: 1,
|
||||||
}),
|
}),
|
||||||
|
|
||||||
/** Empty state (no fonts found) */
|
/**
|
||||||
|
* State with no fonts matching filters
|
||||||
|
*/
|
||||||
empty: createMockFontStoreState({
|
empty: createMockFontStoreState({
|
||||||
fonts: {},
|
fonts: {},
|
||||||
total: 0,
|
total: 0,
|
||||||
@@ -229,7 +341,9 @@ export const MOCK_FONT_STORE_STATES = {
|
|||||||
isLoading: false,
|
isLoading: false,
|
||||||
}),
|
}),
|
||||||
|
|
||||||
/** First page with fonts */
|
/**
|
||||||
|
* First page of results (10 items)
|
||||||
|
*/
|
||||||
firstPage: createMockFontStoreState({
|
firstPage: createMockFontStoreState({
|
||||||
fonts: Object.fromEntries(
|
fonts: Object.fromEntries(
|
||||||
Object.values(UNIFIED_FONTS).slice(0, 10).map(font => [font.id, font]),
|
Object.values(UNIFIED_FONTS).slice(0, 10).map(font => [font.id, font]),
|
||||||
@@ -241,7 +355,9 @@ export const MOCK_FONT_STORE_STATES = {
|
|||||||
isLoading: false,
|
isLoading: false,
|
||||||
}),
|
}),
|
||||||
|
|
||||||
/** Second page with fonts */
|
/**
|
||||||
|
* Second page of results (10 items)
|
||||||
|
*/
|
||||||
secondPage: createMockFontStoreState({
|
secondPage: createMockFontStoreState({
|
||||||
fonts: Object.fromEntries(
|
fonts: Object.fromEntries(
|
||||||
Object.values(UNIFIED_FONTS).slice(10, 20).map(font => [font.id, font]),
|
Object.values(UNIFIED_FONTS).slice(10, 20).map(font => [font.id, font]),
|
||||||
@@ -253,7 +369,9 @@ export const MOCK_FONT_STORE_STATES = {
|
|||||||
isLoading: false,
|
isLoading: false,
|
||||||
}),
|
}),
|
||||||
|
|
||||||
/** Last page with fonts */
|
/**
|
||||||
|
* Final page of results (5 items)
|
||||||
|
*/
|
||||||
lastPage: createMockFontStoreState({
|
lastPage: createMockFontStoreState({
|
||||||
fonts: Object.fromEntries(
|
fonts: Object.fromEntries(
|
||||||
Object.values(UNIFIED_FONTS).slice(0, 5).map(font => [font.id, font]),
|
Object.values(UNIFIED_FONTS).slice(0, 5).map(font => [font.id, font]),
|
||||||
@@ -265,7 +383,9 @@ export const MOCK_FONT_STORE_STATES = {
|
|||||||
isLoading: false,
|
isLoading: false,
|
||||||
}),
|
}),
|
||||||
|
|
||||||
/** Error state */
|
/**
|
||||||
|
* Terminal failure state
|
||||||
|
*/
|
||||||
error: createMockFontStoreState({
|
error: createMockFontStoreState({
|
||||||
fonts: {},
|
fonts: {},
|
||||||
error: new Error('Failed to load fonts'),
|
error: new Error('Failed to load fonts'),
|
||||||
@@ -274,7 +394,9 @@ export const MOCK_FONT_STORE_STATES = {
|
|||||||
isLoading: false,
|
isLoading: false,
|
||||||
}),
|
}),
|
||||||
|
|
||||||
/** With search query */
|
/**
|
||||||
|
* State with active search query
|
||||||
|
*/
|
||||||
withSearch: createMockFontStoreState({
|
withSearch: createMockFontStoreState({
|
||||||
fonts: Object.fromEntries(
|
fonts: Object.fromEntries(
|
||||||
Object.values(UNIFIED_FONTS).slice(0, 3).map(font => [font.id, font]),
|
Object.values(UNIFIED_FONTS).slice(0, 3).map(font => [font.id, font]),
|
||||||
@@ -285,7 +407,9 @@ export const MOCK_FONT_STORE_STATES = {
|
|||||||
searchQuery: 'Roboto',
|
searchQuery: 'Roboto',
|
||||||
}),
|
}),
|
||||||
|
|
||||||
/** Filtered by category */
|
/**
|
||||||
|
* State with active category filter
|
||||||
|
*/
|
||||||
filteredByCategory: createMockFontStoreState({
|
filteredByCategory: createMockFontStoreState({
|
||||||
fonts: Object.fromEntries(
|
fonts: Object.fromEntries(
|
||||||
Object.values(UNIFIED_FONTS)
|
Object.values(UNIFIED_FONTS)
|
||||||
@@ -299,7 +423,9 @@ export const MOCK_FONT_STORE_STATES = {
|
|||||||
category: 'serif',
|
category: 'serif',
|
||||||
}),
|
}),
|
||||||
|
|
||||||
/** Filtered by provider */
|
/**
|
||||||
|
* State with active provider filter
|
||||||
|
*/
|
||||||
filteredByProvider: createMockFontStoreState({
|
filteredByProvider: createMockFontStoreState({
|
||||||
fonts: Object.fromEntries(
|
fonts: Object.fromEntries(
|
||||||
Object.values(UNIFIED_FONTS)
|
Object.values(UNIFIED_FONTS)
|
||||||
@@ -313,7 +439,9 @@ export const MOCK_FONT_STORE_STATES = {
|
|||||||
provider: 'google',
|
provider: 'google',
|
||||||
}),
|
}),
|
||||||
|
|
||||||
/** Large dataset */
|
/**
|
||||||
|
* Large collection for performance testing (50 items)
|
||||||
|
*/
|
||||||
largeDataset: createMockFontStoreState({
|
largeDataset: createMockFontStoreState({
|
||||||
fonts: Object.fromEntries(
|
fonts: Object.fromEntries(
|
||||||
generateMockFonts(50).map(font => [font.id, font]),
|
generateMockFonts(50).map(font => [font.id, font]),
|
||||||
@@ -326,17 +454,30 @@ export const MOCK_FONT_STORE_STATES = {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
// MOCK STORE OBJECT
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a mock store object that mimics TanStack Query behavior
|
* Create a mock store object that mimics TanStack Query behavior
|
||||||
* Useful for components that subscribe to store properties
|
* Useful for components that subscribe to store properties
|
||||||
*/
|
*/
|
||||||
export function createMockStore<T>(config: {
|
export function createMockStore<T>(config: {
|
||||||
|
/**
|
||||||
|
* Reactive data payload
|
||||||
|
*/
|
||||||
data?: T;
|
data?: T;
|
||||||
|
/**
|
||||||
|
* Loading status flag
|
||||||
|
*/
|
||||||
isLoading?: boolean;
|
isLoading?: boolean;
|
||||||
|
/**
|
||||||
|
* Error status flag
|
||||||
|
*/
|
||||||
isError?: boolean;
|
isError?: boolean;
|
||||||
|
/**
|
||||||
|
* Catch-all error object
|
||||||
|
*/
|
||||||
error?: Error;
|
error?: Error;
|
||||||
|
/**
|
||||||
|
* Background fetching flag
|
||||||
|
*/
|
||||||
isFetching?: boolean;
|
isFetching?: boolean;
|
||||||
}) {
|
}) {
|
||||||
const {
|
const {
|
||||||
@@ -348,24 +489,45 @@ export function createMockStore<T>(config: {
|
|||||||
} = config;
|
} = config;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
/**
|
||||||
|
* Returns the active data payload
|
||||||
|
*/
|
||||||
get data() {
|
get data() {
|
||||||
return data;
|
return data;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* True if initially loading
|
||||||
|
*/
|
||||||
get isLoading() {
|
get isLoading() {
|
||||||
return isLoading;
|
return isLoading;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* True if last request failed
|
||||||
|
*/
|
||||||
get isError() {
|
get isError() {
|
||||||
return isError;
|
return isError;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Returns the caught error object
|
||||||
|
*/
|
||||||
get error() {
|
get error() {
|
||||||
return error;
|
return error;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* True if fetching in background
|
||||||
|
*/
|
||||||
get isFetching() {
|
get isFetching() {
|
||||||
return isFetching;
|
return isFetching;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* True if query is stable and has data
|
||||||
|
*/
|
||||||
get isSuccess() {
|
get isSuccess() {
|
||||||
return !isLoading && !isError && data !== undefined;
|
return !isLoading && !isError && data !== undefined;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Returns semantic status string
|
||||||
|
*/
|
||||||
get status() {
|
get status() {
|
||||||
if (isLoading) return 'pending';
|
if (isLoading) return 'pending';
|
||||||
if (isError) return 'error';
|
if (isError) return 'error';
|
||||||
@@ -375,23 +537,29 @@ export function createMockStore<T>(config: {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Preset mock stores
|
* Preset mock stores for common UI states
|
||||||
*/
|
*/
|
||||||
export const MOCK_STORES = {
|
export const MOCK_STORES = {
|
||||||
/** Font store in loading state */
|
/**
|
||||||
|
* Initial loading state
|
||||||
|
*/
|
||||||
loadingFontStore: createMockStore<UnifiedFont[]>({
|
loadingFontStore: createMockStore<UnifiedFont[]>({
|
||||||
isLoading: true,
|
isLoading: true,
|
||||||
data: undefined,
|
data: undefined,
|
||||||
}),
|
}),
|
||||||
|
|
||||||
/** Font store with fonts loaded */
|
/**
|
||||||
|
* Successful data load state
|
||||||
|
*/
|
||||||
successFontStore: createMockStore<UnifiedFont[]>({
|
successFontStore: createMockStore<UnifiedFont[]>({
|
||||||
data: Object.values(UNIFIED_FONTS),
|
data: Object.values(UNIFIED_FONTS),
|
||||||
isLoading: false,
|
isLoading: false,
|
||||||
isError: false,
|
isError: false,
|
||||||
}),
|
}),
|
||||||
|
|
||||||
/** Font store with error */
|
/**
|
||||||
|
* API error state
|
||||||
|
*/
|
||||||
errorFontStore: createMockStore<UnifiedFont[]>({
|
errorFontStore: createMockStore<UnifiedFont[]>({
|
||||||
data: undefined,
|
data: undefined,
|
||||||
isLoading: false,
|
isLoading: false,
|
||||||
@@ -399,7 +567,9 @@ export const MOCK_STORES = {
|
|||||||
error: new Error('Failed to load fonts'),
|
error: new Error('Failed to load fonts'),
|
||||||
}),
|
}),
|
||||||
|
|
||||||
/** Font store with empty results */
|
/**
|
||||||
|
* Empty result set state
|
||||||
|
*/
|
||||||
emptyFontStore: createMockStore<UnifiedFont[]>({
|
emptyFontStore: createMockStore<UnifiedFont[]>({
|
||||||
data: [],
|
data: [],
|
||||||
isLoading: false,
|
isLoading: false,
|
||||||
@@ -414,36 +584,69 @@ export const MOCK_STORES = {
|
|||||||
const mockState = createMockFontStoreState(state);
|
const mockState = createMockFontStoreState(state);
|
||||||
return {
|
return {
|
||||||
// State properties
|
// State properties
|
||||||
|
/**
|
||||||
|
* Collection of mock fonts
|
||||||
|
*/
|
||||||
get fonts() {
|
get fonts() {
|
||||||
return mockState.fonts;
|
return mockState.fonts;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Current mock page
|
||||||
|
*/
|
||||||
get page() {
|
get page() {
|
||||||
return mockState.page;
|
return mockState.page;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Total mock pages
|
||||||
|
*/
|
||||||
get totalPages() {
|
get totalPages() {
|
||||||
return mockState.totalPages;
|
return mockState.totalPages;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock items per page
|
||||||
|
*/
|
||||||
get limit() {
|
get limit() {
|
||||||
return mockState.limit;
|
return mockState.limit;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Total mock items
|
||||||
|
*/
|
||||||
get total() {
|
get total() {
|
||||||
return mockState.total;
|
return mockState.total;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock loading status
|
||||||
|
*/
|
||||||
get isLoading() {
|
get isLoading() {
|
||||||
return mockState.isLoading;
|
return mockState.isLoading;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock error status
|
||||||
|
*/
|
||||||
get error() {
|
get error() {
|
||||||
return mockState.error;
|
return mockState.error;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock search string
|
||||||
|
*/
|
||||||
get searchQuery() {
|
get searchQuery() {
|
||||||
return mockState.searchQuery;
|
return mockState.searchQuery;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock provider filter
|
||||||
|
*/
|
||||||
get provider() {
|
get provider() {
|
||||||
return mockState.provider;
|
return mockState.provider;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock category filter
|
||||||
|
*/
|
||||||
get category() {
|
get category() {
|
||||||
return mockState.category;
|
return mockState.category;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock subset filter
|
||||||
|
*/
|
||||||
get subset() {
|
get subset() {
|
||||||
return mockState.subset;
|
return mockState.subset;
|
||||||
},
|
},
|
||||||
@@ -464,15 +667,45 @@ export const MOCK_STORES = {
|
|||||||
* Matches FontStore's public API for Storybook use
|
* Matches FontStore's public API for Storybook use
|
||||||
*/
|
*/
|
||||||
fontStore: (config: {
|
fontStore: (config: {
|
||||||
|
/**
|
||||||
|
* Preset font list
|
||||||
|
*/
|
||||||
fonts?: UnifiedFont[];
|
fonts?: UnifiedFont[];
|
||||||
|
/**
|
||||||
|
* Total item count
|
||||||
|
*/
|
||||||
total?: number;
|
total?: number;
|
||||||
|
/**
|
||||||
|
* Items per page
|
||||||
|
*/
|
||||||
limit?: number;
|
limit?: number;
|
||||||
|
/**
|
||||||
|
* Pagination offset
|
||||||
|
*/
|
||||||
offset?: number;
|
offset?: number;
|
||||||
|
/**
|
||||||
|
* Loading flag
|
||||||
|
*/
|
||||||
isLoading?: boolean;
|
isLoading?: boolean;
|
||||||
|
/**
|
||||||
|
* Fetching flag
|
||||||
|
*/
|
||||||
isFetching?: boolean;
|
isFetching?: boolean;
|
||||||
|
/**
|
||||||
|
* Error flag
|
||||||
|
*/
|
||||||
isError?: boolean;
|
isError?: boolean;
|
||||||
|
/**
|
||||||
|
* Catch-all error object
|
||||||
|
*/
|
||||||
error?: Error | null;
|
error?: Error | null;
|
||||||
|
/**
|
||||||
|
* Has more pages flag
|
||||||
|
*/
|
||||||
hasMore?: boolean;
|
hasMore?: boolean;
|
||||||
|
/**
|
||||||
|
* Current page number
|
||||||
|
*/
|
||||||
page?: number;
|
page?: number;
|
||||||
} = {}) => {
|
} = {}) => {
|
||||||
const {
|
const {
|
||||||
@@ -495,27 +728,51 @@ export const MOCK_STORES = {
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
// State getters
|
// State getters
|
||||||
|
/**
|
||||||
|
* Current mock parameters
|
||||||
|
*/
|
||||||
get params() {
|
get params() {
|
||||||
return state.params;
|
return state.params;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock font list
|
||||||
|
*/
|
||||||
get fonts() {
|
get fonts() {
|
||||||
return mockFonts;
|
return mockFonts;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock loading state
|
||||||
|
*/
|
||||||
get isLoading() {
|
get isLoading() {
|
||||||
return isLoading;
|
return isLoading;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock fetching state
|
||||||
|
*/
|
||||||
get isFetching() {
|
get isFetching() {
|
||||||
return isFetching;
|
return isFetching;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock error state
|
||||||
|
*/
|
||||||
get isError() {
|
get isError() {
|
||||||
return isError;
|
return isError;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock error object
|
||||||
|
*/
|
||||||
get error() {
|
get error() {
|
||||||
return error;
|
return error;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock empty state check
|
||||||
|
*/
|
||||||
get isEmpty() {
|
get isEmpty() {
|
||||||
return !isLoading && !isFetching && mockFonts.length === 0;
|
return !isLoading && !isFetching && mockFonts.length === 0;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mock pagination metadata
|
||||||
|
*/
|
||||||
get pagination() {
|
get pagination() {
|
||||||
return {
|
return {
|
||||||
total: mockTotal,
|
total: mockTotal,
|
||||||
@@ -527,18 +784,33 @@ export const MOCK_STORES = {
|
|||||||
};
|
};
|
||||||
},
|
},
|
||||||
// Category getters
|
// Category getters
|
||||||
|
/**
|
||||||
|
* Derived sans-serif filter
|
||||||
|
*/
|
||||||
get sansSerifFonts() {
|
get sansSerifFonts() {
|
||||||
return mockFonts.filter(f => f.category === 'sans-serif');
|
return mockFonts.filter(f => f.category === 'sans-serif');
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Derived serif filter
|
||||||
|
*/
|
||||||
get serifFonts() {
|
get serifFonts() {
|
||||||
return mockFonts.filter(f => f.category === 'serif');
|
return mockFonts.filter(f => f.category === 'serif');
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Derived display filter
|
||||||
|
*/
|
||||||
get displayFonts() {
|
get displayFonts() {
|
||||||
return mockFonts.filter(f => f.category === 'display');
|
return mockFonts.filter(f => f.category === 'display');
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Derived handwriting filter
|
||||||
|
*/
|
||||||
get handwritingFonts() {
|
get handwritingFonts() {
|
||||||
return mockFonts.filter(f => f.category === 'handwriting');
|
return mockFonts.filter(f => f.category === 'handwriting');
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Derived monospace filter
|
||||||
|
*/
|
||||||
get monospaceFonts() {
|
get monospaceFonts() {
|
||||||
return mockFonts.filter(f => f.category === 'monospace');
|
return mockFonts.filter(f => f.category === 'monospace');
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -13,15 +13,25 @@ import type {
|
|||||||
* (e.g. `SvelteMap.get()`) is automatically tracked as a dependency.
|
* (e.g. `SvelteMap.get()`) is automatically tracked as a dependency.
|
||||||
*/
|
*/
|
||||||
export interface FontRowSizeResolverOptions {
|
export interface FontRowSizeResolverOptions {
|
||||||
/** Returns the current fonts array. Index `i` corresponds to row `i`. */
|
/**
|
||||||
|
* Returns the current fonts array. Index `i` corresponds to row `i`.
|
||||||
|
*/
|
||||||
getFonts: () => UnifiedFont[];
|
getFonts: () => UnifiedFont[];
|
||||||
/** Returns the active font weight (e.g. 400). */
|
/**
|
||||||
|
* Returns the active font weight (e.g. 400).
|
||||||
|
*/
|
||||||
getWeight: () => number;
|
getWeight: () => number;
|
||||||
/** Returns the preview text string. */
|
/**
|
||||||
|
* Returns the preview text string.
|
||||||
|
*/
|
||||||
getPreviewText: () => string;
|
getPreviewText: () => string;
|
||||||
/** Returns the scroll container's inner width in pixels. Returns 0 before mount. */
|
/**
|
||||||
|
* Returns the scroll container's inner width in pixels. Returns 0 before mount.
|
||||||
|
*/
|
||||||
getContainerWidth: () => number;
|
getContainerWidth: () => number;
|
||||||
/** Returns the font size in pixels (e.g. `controlManager.renderedSize`). */
|
/**
|
||||||
|
* Returns the font size in pixels (e.g. `controlManager.renderedSize`).
|
||||||
|
*/
|
||||||
getFontSizePx: () => number;
|
getFontSizePx: () => number;
|
||||||
/**
|
/**
|
||||||
* Returns the computed line height in pixels.
|
* Returns the computed line height in pixels.
|
||||||
@@ -44,9 +54,13 @@ export interface FontRowSizeResolverOptions {
|
|||||||
* the content width is never over-estimated, keeping the height estimate safe.
|
* the content width is never over-estimated, keeping the height estimate safe.
|
||||||
*/
|
*/
|
||||||
contentHorizontalPadding: number;
|
contentHorizontalPadding: number;
|
||||||
/** Fixed height in pixels of chrome that is not text content (header bar, etc.). */
|
/**
|
||||||
|
* Fixed height in pixels of chrome that is not text content (header bar, etc.).
|
||||||
|
*/
|
||||||
chromeHeight: number;
|
chromeHeight: number;
|
||||||
/** Height in pixels to return when the font is not loaded or container width is 0. */
|
/**
|
||||||
|
* Height in pixels to return when the font is not loaded or container width is 0.
|
||||||
|
*/
|
||||||
fallbackHeight: number;
|
fallbackHeight: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
/** @vitest-environment jsdom */
|
/**
|
||||||
|
* @vitest-environment jsdom
|
||||||
|
*/
|
||||||
import { AppliedFontsManager } from './appliedFontsStore.svelte';
|
import { AppliedFontsManager } from './appliedFontsStore.svelte';
|
||||||
import { FontFetchError } from './errors';
|
import { FontFetchError } from './errors';
|
||||||
import { FontEvictionPolicy } from './utils/fontEvictionPolicy/FontEvictionPolicy';
|
import { FontEvictionPolicy } from './utils/fontEvictionPolicy/FontEvictionPolicy';
|
||||||
|
|
||||||
// ── Fake collaborators ────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
class FakeBufferCache {
|
class FakeBufferCache {
|
||||||
async get(_url: string): Promise<ArrayBuffer> {
|
async get(_url: string): Promise<ArrayBuffer> {
|
||||||
return new ArrayBuffer(8);
|
return new ArrayBuffer(8);
|
||||||
@@ -13,7 +13,9 @@ class FakeBufferCache {
|
|||||||
clear(): void {}
|
clear(): void {}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Throws {@link FontFetchError} on every `get()` — simulates network/HTTP failure. */
|
/**
|
||||||
|
* Throws {@link FontFetchError} on every `get()` — simulates network/HTTP failure.
|
||||||
|
*/
|
||||||
class FailingBufferCache {
|
class FailingBufferCache {
|
||||||
async get(url: string): Promise<never> {
|
async get(url: string): Promise<never> {
|
||||||
throw new FontFetchError(url, new Error('network error'), 500);
|
throw new FontFetchError(url, new Error('network error'), 500);
|
||||||
@@ -22,8 +24,6 @@ class FailingBufferCache {
|
|||||||
clear(): void {}
|
clear(): void {}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ── Helpers ───────────────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
const makeConfig = (id: string, overrides: Partial<{ weight: number; isVariable: boolean }> = {}) => ({
|
const makeConfig = (id: string, overrides: Partial<{ weight: number; isVariable: boolean }> = {}) => ({
|
||||||
id,
|
id,
|
||||||
name: id,
|
name: id,
|
||||||
@@ -32,8 +32,6 @@ const makeConfig = (id: string, overrides: Partial<{ weight: number; isVariable:
|
|||||||
...overrides,
|
...overrides,
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── Suite ─────────────────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('AppliedFontsManager', () => {
|
describe('AppliedFontsManager', () => {
|
||||||
let manager: AppliedFontsManager;
|
let manager: AppliedFontsManager;
|
||||||
let eviction: FontEvictionPolicy;
|
let eviction: FontEvictionPolicy;
|
||||||
@@ -66,8 +64,6 @@ describe('AppliedFontsManager', () => {
|
|||||||
vi.unstubAllGlobals();
|
vi.unstubAllGlobals();
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── touch() ───────────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('touch()', () => {
|
describe('touch()', () => {
|
||||||
it('queues and loads a new font', async () => {
|
it('queues and loads a new font', async () => {
|
||||||
manager.touch([makeConfig('roboto')]);
|
manager.touch([makeConfig('roboto')]);
|
||||||
@@ -131,8 +127,6 @@ describe('AppliedFontsManager', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── queue processing ──────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('queue processing', () => {
|
describe('queue processing', () => {
|
||||||
it('filters non-critical weights in data-saver mode', async () => {
|
it('filters non-critical weights in data-saver mode', async () => {
|
||||||
(navigator as any).connection = { saveData: true };
|
(navigator as any).connection = { saveData: true };
|
||||||
@@ -163,8 +157,6 @@ describe('AppliedFontsManager', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── Phase 1: fetch ────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('Phase 1 — fetch', () => {
|
describe('Phase 1 — fetch', () => {
|
||||||
it('sets status to error on fetch failure', async () => {
|
it('sets status to error on fetch failure', async () => {
|
||||||
const consoleSpy = vi.spyOn(console, 'error').mockImplementation(() => {});
|
const consoleSpy = vi.spyOn(console, 'error').mockImplementation(() => {});
|
||||||
@@ -209,8 +201,6 @@ describe('AppliedFontsManager', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── Phase 2: parse ────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('Phase 2 — parse', () => {
|
describe('Phase 2 — parse', () => {
|
||||||
it('sets status to error on parse failure', async () => {
|
it('sets status to error on parse failure', async () => {
|
||||||
const consoleSpy = vi.spyOn(console, 'error').mockImplementation(() => {});
|
const consoleSpy = vi.spyOn(console, 'error').mockImplementation(() => {});
|
||||||
@@ -241,8 +231,6 @@ describe('AppliedFontsManager', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── #purgeUnused ──────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('#purgeUnused', () => {
|
describe('#purgeUnused', () => {
|
||||||
it('evicts fonts after TTL expires', async () => {
|
it('evicts fonts after TTL expires', async () => {
|
||||||
manager.touch([makeConfig('ephemeral')]);
|
manager.touch([makeConfig('ephemeral')]);
|
||||||
@@ -300,8 +288,6 @@ describe('AppliedFontsManager', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── destroy() ─────────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('destroy()', () => {
|
describe('destroy()', () => {
|
||||||
it('clears all statuses', async () => {
|
it('clears all statuses', async () => {
|
||||||
manager.touch([makeConfig('roboto')]);
|
manager.touch([makeConfig('roboto')]);
|
||||||
|
|||||||
@@ -156,7 +156,9 @@ export class AppliedFontsManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Returns true if data-saver mode is enabled (defers non-critical weights). */
|
/**
|
||||||
|
* Returns true if data-saver mode is enabled (defers non-critical weights).
|
||||||
|
*/
|
||||||
#shouldDeferNonCritical(): boolean {
|
#shouldDeferNonCritical(): boolean {
|
||||||
return (navigator as any).connection?.saveData === true;
|
return (navigator as any).connection?.saveData === true;
|
||||||
}
|
}
|
||||||
@@ -188,13 +190,11 @@ export class AppliedFontsManager {
|
|||||||
const concurrency = getEffectiveConcurrency();
|
const concurrency = getEffectiveConcurrency();
|
||||||
const buffers = new Map<string, ArrayBuffer>();
|
const buffers = new Map<string, ArrayBuffer>();
|
||||||
|
|
||||||
// ==================== PHASE 1: Concurrent Fetching ====================
|
|
||||||
// Fetch multiple font files in parallel since network I/O is non-blocking
|
// Fetch multiple font files in parallel since network I/O is non-blocking
|
||||||
for (let i = 0; i < entries.length; i += concurrency) {
|
for (let i = 0; i < entries.length; i += concurrency) {
|
||||||
await this.#fetchChunk(entries.slice(i, i + concurrency), buffers);
|
await this.#fetchChunk(entries.slice(i, i + concurrency), buffers);
|
||||||
}
|
}
|
||||||
|
|
||||||
// ==================== PHASE 2: Sequential Parsing ====================
|
|
||||||
// Parse buffers one at a time with periodic yields to avoid blocking UI
|
// Parse buffers one at a time with periodic yields to avoid blocking UI
|
||||||
const hasInputPending = !!(navigator as any).scheduling?.isInputPending;
|
const hasInputPending = !!(navigator as any).scheduling?.isInputPending;
|
||||||
let lastYield = performance.now();
|
let lastYield = performance.now();
|
||||||
@@ -279,7 +279,9 @@ export class AppliedFontsManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Removes fonts unused within TTL (LRU-style cleanup). Runs every PURGE_INTERVAL. Pinned fonts are never evicted. */
|
/**
|
||||||
|
* Removes fonts unused within TTL (LRU-style cleanup). Runs every PURGE_INTERVAL. Pinned fonts are never evicted.
|
||||||
|
*/
|
||||||
#purgeUnused() {
|
#purgeUnused() {
|
||||||
const now = Date.now();
|
const now = Date.now();
|
||||||
// Iterate through all tracked font keys
|
// Iterate through all tracked font keys
|
||||||
@@ -307,7 +309,9 @@ export class AppliedFontsManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Returns current loading status for a font, or undefined if never requested. */
|
/**
|
||||||
|
* Returns current loading status for a font, or undefined if never requested.
|
||||||
|
*/
|
||||||
getFontStatus(id: string, weight: number, isVariable = false) {
|
getFontStatus(id: string, weight: number, isVariable = false) {
|
||||||
try {
|
try {
|
||||||
return this.statuses.get(generateFontKey({ id, weight, isVariable }));
|
return this.statuses.get(generateFontKey({ id, weight, isVariable }));
|
||||||
@@ -316,17 +320,23 @@ export class AppliedFontsManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Pins a font so it is never evicted by #purgeUnused(), regardless of TTL. */
|
/**
|
||||||
|
* Pins a font so it is never evicted by #purgeUnused(), regardless of TTL.
|
||||||
|
*/
|
||||||
pin(id: string, weight: number, isVariable = false): void {
|
pin(id: string, weight: number, isVariable = false): void {
|
||||||
this.#eviction.pin(generateFontKey({ id, weight, isVariable }));
|
this.#eviction.pin(generateFontKey({ id, weight, isVariable }));
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Unpins a font, allowing it to be evicted by #purgeUnused() once its TTL expires. */
|
/**
|
||||||
|
* Unpins a font, allowing it to be evicted by #purgeUnused() once its TTL expires.
|
||||||
|
*/
|
||||||
unpin(id: string, weight: number, isVariable = false): void {
|
unpin(id: string, weight: number, isVariable = false): void {
|
||||||
this.#eviction.unpin(generateFontKey({ id, weight, isVariable }));
|
this.#eviction.unpin(generateFontKey({ id, weight, isVariable }));
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Waits for all fonts to finish loading using document.fonts.ready. */
|
/**
|
||||||
|
* Waits for all fonts to finish loading using document.fonts.ready.
|
||||||
|
*/
|
||||||
async ready(): Promise<void> {
|
async ready(): Promise<void> {
|
||||||
if (typeof document === 'undefined') {
|
if (typeof document === 'undefined') {
|
||||||
return;
|
return;
|
||||||
@@ -336,7 +346,9 @@ export class AppliedFontsManager {
|
|||||||
} catch { /* document unloaded */ }
|
} catch { /* document unloaded */ }
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Aborts all operations, removes fonts from document, and clears state. Manager cannot be reused after. */
|
/**
|
||||||
|
* Aborts all operations, removes fonts from document, and clears state. Manager cannot be reused after.
|
||||||
|
*/
|
||||||
destroy() {
|
destroy() {
|
||||||
// Abort all in-flight network requests
|
// Abort all in-flight network requests
|
||||||
this.#abortController.abort();
|
this.#abortController.abort();
|
||||||
@@ -375,5 +387,7 @@ export class AppliedFontsManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Singleton instance — use throughout the application for unified font loading state. */
|
/**
|
||||||
|
* Singleton instance — use throughout the application for unified font loading state.
|
||||||
|
*/
|
||||||
export const appliedFontsManager = new AppliedFontsManager();
|
export const appliedFontsManager = new AppliedFontsManager();
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
/** @vitest-environment jsdom */
|
/**
|
||||||
|
* @vitest-environment jsdom
|
||||||
|
*/
|
||||||
import { FontFetchError } from '../../errors';
|
import { FontFetchError } from '../../errors';
|
||||||
import { FontBufferCache } from './FontBufferCache';
|
import { FontBufferCache } from './FontBufferCache';
|
||||||
|
|
||||||
|
|||||||
@@ -3,9 +3,13 @@ import { FontFetchError } from '../../errors';
|
|||||||
type Fetcher = (url: string, init?: RequestInit) => Promise<Response>;
|
type Fetcher = (url: string, init?: RequestInit) => Promise<Response>;
|
||||||
|
|
||||||
interface FontBufferCacheOptions {
|
interface FontBufferCacheOptions {
|
||||||
/** Custom fetch implementation. Defaults to `globalThis.fetch`. Inject in tests for isolation. */
|
/**
|
||||||
|
* Custom fetch implementation. Defaults to `globalThis.fetch`. Inject in tests for isolation.
|
||||||
|
*/
|
||||||
fetcher?: Fetcher;
|
fetcher?: Fetcher;
|
||||||
/** Cache API cache name. Defaults to `'font-cache-v1'`. */
|
/**
|
||||||
|
* Cache API cache name. Defaults to `'font-cache-v1'`.
|
||||||
|
*/
|
||||||
cacheName?: string;
|
cacheName?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -85,12 +89,16 @@ export class FontBufferCache {
|
|||||||
return buffer;
|
return buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Removes a URL from the in-memory cache. Next call to `get()` will re-fetch. */
|
/**
|
||||||
|
* Removes a URL from the in-memory cache. Next call to `get()` will re-fetch.
|
||||||
|
*/
|
||||||
evict(url: string): void {
|
evict(url: string): void {
|
||||||
this.#buffersByUrl.delete(url);
|
this.#buffersByUrl.delete(url);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Clears all in-memory cached buffers. */
|
/**
|
||||||
|
* Clears all in-memory cached buffers.
|
||||||
|
*/
|
||||||
clear(): void {
|
clear(): void {
|
||||||
this.#buffersByUrl.clear();
|
this.#buffersByUrl.clear();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
interface FontEvictionPolicyOptions {
|
interface FontEvictionPolicyOptions {
|
||||||
/** TTL in milliseconds. Defaults to 5 minutes. */
|
/**
|
||||||
|
* TTL in milliseconds. Defaults to 5 minutes.
|
||||||
|
*/
|
||||||
ttl?: number;
|
ttl?: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -28,12 +30,16 @@ export class FontEvictionPolicy {
|
|||||||
this.#usageTracker.set(key, now);
|
this.#usageTracker.set(key, now);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Pins a font key so it is never evicted regardless of TTL. */
|
/**
|
||||||
|
* Pins a font key so it is never evicted regardless of TTL.
|
||||||
|
*/
|
||||||
pin(key: string): void {
|
pin(key: string): void {
|
||||||
this.#pinnedFonts.add(key);
|
this.#pinnedFonts.add(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Unpins a font key, allowing it to be evicted once its TTL expires. */
|
/**
|
||||||
|
* Unpins a font key, allowing it to be evicted once its TTL expires.
|
||||||
|
*/
|
||||||
unpin(key: string): void {
|
unpin(key: string): void {
|
||||||
this.#pinnedFonts.delete(key);
|
this.#pinnedFonts.delete(key);
|
||||||
}
|
}
|
||||||
@@ -57,18 +63,24 @@ export class FontEvictionPolicy {
|
|||||||
return now - lastUsed >= this.#TTL;
|
return now - lastUsed >= this.#TTL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Returns an iterator over all tracked font keys. */
|
/**
|
||||||
|
* Returns an iterator over all tracked font keys.
|
||||||
|
*/
|
||||||
keys(): IterableIterator<string> {
|
keys(): IterableIterator<string> {
|
||||||
return this.#usageTracker.keys();
|
return this.#usageTracker.keys();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Removes a font key from tracking. Called by the orchestrator after eviction. */
|
/**
|
||||||
|
* Removes a font key from tracking. Called by the orchestrator after eviction.
|
||||||
|
*/
|
||||||
remove(key: string): void {
|
remove(key: string): void {
|
||||||
this.#usageTracker.delete(key);
|
this.#usageTracker.delete(key);
|
||||||
this.#pinnedFonts.delete(key);
|
this.#pinnedFonts.delete(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Clears all usage timestamps and pinned keys. */
|
/**
|
||||||
|
* Clears all usage timestamps and pinned keys.
|
||||||
|
*/
|
||||||
clear(): void {
|
clear(): void {
|
||||||
this.#usageTracker.clear();
|
this.#usageTracker.clear();
|
||||||
this.#pinnedFonts.clear();
|
this.#pinnedFonts.clear();
|
||||||
|
|||||||
@@ -34,22 +34,30 @@ export class FontLoadQueue {
|
|||||||
return entries;
|
return entries;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Returns `true` if the key is currently in the queue. */
|
/**
|
||||||
|
* Returns `true` if the key is currently in the queue.
|
||||||
|
*/
|
||||||
has(key: string): boolean {
|
has(key: string): boolean {
|
||||||
return this.#queue.has(key);
|
return this.#queue.has(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Increments the retry count for a font key. */
|
/**
|
||||||
|
* Increments the retry count for a font key.
|
||||||
|
*/
|
||||||
incrementRetry(key: string): void {
|
incrementRetry(key: string): void {
|
||||||
this.#retryCounts.set(key, (this.#retryCounts.get(key) ?? 0) + 1);
|
this.#retryCounts.set(key, (this.#retryCounts.get(key) ?? 0) + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Returns `true` if the font has reached or exceeded the maximum retry limit. */
|
/**
|
||||||
|
* Returns `true` if the font has reached or exceeded the maximum retry limit.
|
||||||
|
*/
|
||||||
isMaxRetriesReached(key: string): boolean {
|
isMaxRetriesReached(key: string): boolean {
|
||||||
return (this.#retryCounts.get(key) ?? 0) >= this.#MAX_RETRIES;
|
return (this.#retryCounts.get(key) ?? 0) >= this.#MAX_RETRIES;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Clears all queued fonts and resets all retry counts. */
|
/**
|
||||||
|
* Clears all queued fonts and resets all retry counts.
|
||||||
|
*/
|
||||||
clear(): void {
|
clear(): void {
|
||||||
this.#queue.clear();
|
this.#queue.clear();
|
||||||
this.#retryCounts.clear();
|
this.#retryCounts.clear();
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
/** @vitest-environment jsdom */
|
/**
|
||||||
|
* @vitest-environment jsdom
|
||||||
|
*/
|
||||||
import { FontParseError } from '../../errors';
|
import { FontParseError } from '../../errors';
|
||||||
import { loadFont } from './loadFont';
|
import { loadFont } from './loadFont';
|
||||||
|
|
||||||
|
|||||||
@@ -61,7 +61,6 @@ describe('FontStore', () => {
|
|||||||
vi.resetAllMocks();
|
vi.resetAllMocks();
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('construction', () => {
|
describe('construction', () => {
|
||||||
it('stores initial params', () => {
|
it('stores initial params', () => {
|
||||||
const store = makeStore({ limit: 20 });
|
const store = makeStore({ limit: 20 });
|
||||||
@@ -90,7 +89,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('state after fetch', () => {
|
describe('state after fetch', () => {
|
||||||
it('exposes loaded fonts', async () => {
|
it('exposes loaded fonts', async () => {
|
||||||
const store = await fetchedStore({}, generateMockFonts(7));
|
const store = await fetchedStore({}, generateMockFonts(7));
|
||||||
@@ -129,7 +127,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('error states', () => {
|
describe('error states', () => {
|
||||||
it('isError is false before any fetch', () => {
|
it('isError is false before any fetch', () => {
|
||||||
const store = makeStore();
|
const store = makeStore();
|
||||||
@@ -178,7 +175,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('font accumulation', () => {
|
describe('font accumulation', () => {
|
||||||
it('replaces fonts when refetching the first page', async () => {
|
it('replaces fonts when refetching the first page', async () => {
|
||||||
const store = makeStore();
|
const store = makeStore();
|
||||||
@@ -212,7 +208,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('pagination state', () => {
|
describe('pagination state', () => {
|
||||||
it('returns zero-value defaults before any fetch', () => {
|
it('returns zero-value defaults before any fetch', () => {
|
||||||
const store = makeStore();
|
const store = makeStore();
|
||||||
@@ -248,7 +243,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('setParams', () => {
|
describe('setParams', () => {
|
||||||
it('merges updates into existing params', () => {
|
it('merges updates into existing params', () => {
|
||||||
const store = makeStore({ limit: 10 });
|
const store = makeStore({ limit: 10 });
|
||||||
@@ -266,7 +260,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('filter change resets', () => {
|
describe('filter change resets', () => {
|
||||||
it('clears accumulated fonts when a filter changes', async () => {
|
it('clears accumulated fonts when a filter changes', async () => {
|
||||||
const store = await fetchedStore({}, generateMockFonts(5));
|
const store = await fetchedStore({}, generateMockFonts(5));
|
||||||
@@ -302,7 +295,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('staleTime in buildOptions', () => {
|
describe('staleTime in buildOptions', () => {
|
||||||
it('is 5 minutes with no active filters', () => {
|
it('is 5 minutes with no active filters', () => {
|
||||||
const store = makeStore();
|
const store = makeStore();
|
||||||
@@ -331,7 +323,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('buildQueryKey', () => {
|
describe('buildQueryKey', () => {
|
||||||
it('omits empty-string params', () => {
|
it('omits empty-string params', () => {
|
||||||
const store = makeStore();
|
const store = makeStore();
|
||||||
@@ -366,7 +357,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('destroy', () => {
|
describe('destroy', () => {
|
||||||
it('does not throw', () => {
|
it('does not throw', () => {
|
||||||
const store = makeStore();
|
const store = makeStore();
|
||||||
@@ -380,7 +370,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('refetch', () => {
|
describe('refetch', () => {
|
||||||
it('triggers a fetch', async () => {
|
it('triggers a fetch', async () => {
|
||||||
const store = makeStore();
|
const store = makeStore();
|
||||||
@@ -400,7 +389,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('nextPage', () => {
|
describe('nextPage', () => {
|
||||||
let store: FontStore;
|
let store: FontStore;
|
||||||
|
|
||||||
@@ -437,7 +425,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('prevPage and goToPage', () => {
|
describe('prevPage and goToPage', () => {
|
||||||
it('prevPage is a no-op — infinite scroll does not support backward navigation', async () => {
|
it('prevPage is a no-op — infinite scroll does not support backward navigation', async () => {
|
||||||
const store = await fetchedStore({}, generateMockFonts(5));
|
const store = await fetchedStore({}, generateMockFonts(5));
|
||||||
@@ -454,7 +441,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('prefetch', () => {
|
describe('prefetch', () => {
|
||||||
it('triggers a fetch for the provided params', async () => {
|
it('triggers a fetch for the provided params', async () => {
|
||||||
const store = makeStore();
|
const store = makeStore();
|
||||||
@@ -465,7 +451,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('getCachedData / setQueryData', () => {
|
describe('getCachedData / setQueryData', () => {
|
||||||
it('getCachedData returns undefined before any fetch', () => {
|
it('getCachedData returns undefined before any fetch', () => {
|
||||||
queryClient.clear();
|
queryClient.clear();
|
||||||
@@ -497,7 +482,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('invalidate', () => {
|
describe('invalidate', () => {
|
||||||
it('calls invalidateQueries', async () => {
|
it('calls invalidateQueries', async () => {
|
||||||
const store = await fetchedStore();
|
const store = await fetchedStore();
|
||||||
@@ -508,7 +492,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('setLimit', () => {
|
describe('setLimit', () => {
|
||||||
it('updates the limit param', () => {
|
it('updates the limit param', () => {
|
||||||
const store = makeStore({ limit: 10 });
|
const store = makeStore({ limit: 10 });
|
||||||
@@ -518,7 +501,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('filter shortcut methods', () => {
|
describe('filter shortcut methods', () => {
|
||||||
let store: FontStore;
|
let store: FontStore;
|
||||||
|
|
||||||
@@ -561,7 +543,6 @@ describe('FontStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
describe('category getters', () => {
|
describe('category getters', () => {
|
||||||
it('each getter returns only fonts of that category', async () => {
|
it('each getter returns only fonts of that category', async () => {
|
||||||
const fonts = generateMixedCategoryFonts(2); // 2 of each category = 10 total
|
const fonts = generateMixedCategoryFonts(2); // 2 of each category = 10 total
|
||||||
|
|||||||
@@ -18,7 +18,9 @@ import type { UnifiedFont } from '../../types';
|
|||||||
|
|
||||||
type PageParam = { offset: number };
|
type PageParam = { offset: number };
|
||||||
|
|
||||||
/** Filter params + limit — offset is managed by TQ as a page param, not a user param. */
|
/**
|
||||||
|
* Filter params + limit — offset is managed by TQ as a page param, not a user param.
|
||||||
|
*/
|
||||||
type FontStoreParams = Omit<ProxyFontsParams, 'offset'>;
|
type FontStoreParams = Omit<ProxyFontsParams, 'offset'>;
|
||||||
|
|
||||||
type FontStoreResult = InfiniteQueryObserverResult<InfiniteData<ProxyFontsResponse, PageParam>, Error>;
|
type FontStoreResult = InfiniteQueryObserverResult<InfiniteData<ProxyFontsResponse, PageParam>, Error>;
|
||||||
@@ -44,34 +46,53 @@ export class FontStore {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// -- Public state --
|
/**
|
||||||
|
* Current filter and limit configuration
|
||||||
|
*/
|
||||||
get params(): FontStoreParams {
|
get params(): FontStoreParams {
|
||||||
return this.#params;
|
return this.#params;
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* Flattened list of all fonts loaded across all pages (reactive)
|
||||||
|
*/
|
||||||
get fonts(): UnifiedFont[] {
|
get fonts(): UnifiedFont[] {
|
||||||
return this.#result.data?.pages.flatMap((p: ProxyFontsResponse) => p.fonts) ?? [];
|
return this.#result.data?.pages.flatMap((p: ProxyFontsResponse) => p.fonts) ?? [];
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* True if the first page is currently being fetched
|
||||||
|
*/
|
||||||
get isLoading(): boolean {
|
get isLoading(): boolean {
|
||||||
return this.#result.isLoading;
|
return this.#result.isLoading;
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* True if any background fetch is in progress (initial or pagination)
|
||||||
|
*/
|
||||||
get isFetching(): boolean {
|
get isFetching(): boolean {
|
||||||
return this.#result.isFetching;
|
return this.#result.isFetching;
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* True if the last fetch attempt resulted in an error
|
||||||
|
*/
|
||||||
get isError(): boolean {
|
get isError(): boolean {
|
||||||
return this.#result.isError;
|
return this.#result.isError;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Last caught error from the query observer
|
||||||
|
*/
|
||||||
get error(): Error | null {
|
get error(): Error | null {
|
||||||
return this.#result.error ?? null;
|
return this.#result.error ?? null;
|
||||||
}
|
}
|
||||||
// isEmpty is false during loading/fetching so the UI never flashes "no results"
|
/**
|
||||||
// while a fetch is in progress. The !isFetching guard is specifically for the filter-change
|
* True if no fonts were found for the current filter criteria
|
||||||
// transition: fonts clear synchronously → isFetching becomes true → isEmpty stays false.
|
*/
|
||||||
get isEmpty(): boolean {
|
get isEmpty(): boolean {
|
||||||
return !this.isLoading && !this.isFetching && this.fonts.length === 0;
|
return !this.isLoading && !this.isFetching && this.fonts.length === 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pagination metadata derived from the last loaded page
|
||||||
|
*/
|
||||||
get pagination() {
|
get pagination() {
|
||||||
const pages = this.#result.data?.pages;
|
const pages = this.#result.data?.pages;
|
||||||
const last = pages?.at(-1);
|
const last = pages?.at(-1);
|
||||||
@@ -95,37 +116,52 @@ export class FontStore {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// -- Lifecycle --
|
/**
|
||||||
|
* Cleans up subscriptions and destroys the observer
|
||||||
|
*/
|
||||||
destroy() {
|
destroy() {
|
||||||
this.#unsubscribe();
|
this.#unsubscribe();
|
||||||
this.#observer.destroy();
|
this.#observer.destroy();
|
||||||
}
|
}
|
||||||
|
|
||||||
// -- Param management --
|
/**
|
||||||
|
* Merge new parameters into existing state and trigger a refetch
|
||||||
|
*/
|
||||||
setParams(updates: Partial<FontStoreParams>) {
|
setParams(updates: Partial<FontStoreParams>) {
|
||||||
this.#params = { ...this.#params, ...updates };
|
this.#params = { ...this.#params, ...updates };
|
||||||
this.#observer.setOptions(this.buildOptions());
|
this.#observer.setOptions(this.buildOptions());
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* Forcefully invalidate and refetch the current query from the network
|
||||||
|
*/
|
||||||
invalidate() {
|
invalidate() {
|
||||||
this.#qc.invalidateQueries({ queryKey: this.buildQueryKey(this.#params) });
|
this.#qc.invalidateQueries({ queryKey: this.buildQueryKey(this.#params) });
|
||||||
}
|
}
|
||||||
|
|
||||||
// -- Async operations --
|
/**
|
||||||
|
* Manually trigger a query refetch
|
||||||
|
*/
|
||||||
async refetch() {
|
async refetch() {
|
||||||
await this.#observer.refetch();
|
await this.#observer.refetch();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prime the cache with data for a specific parameter set
|
||||||
|
*/
|
||||||
async prefetch(params: FontStoreParams) {
|
async prefetch(params: FontStoreParams) {
|
||||||
await this.#qc.prefetchInfiniteQuery(this.buildOptions(params));
|
await this.#qc.prefetchInfiniteQuery(this.buildOptions(params));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Abort any active network requests for this store
|
||||||
|
*/
|
||||||
cancel() {
|
cancel() {
|
||||||
this.#qc.cancelQueries({ queryKey: this.buildQueryKey(this.#params) });
|
this.#qc.cancelQueries({ queryKey: this.buildQueryKey(this.#params) });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieve current font list from cache without triggering a fetch
|
||||||
|
*/
|
||||||
getCachedData(): UnifiedFont[] | undefined {
|
getCachedData(): UnifiedFont[] | undefined {
|
||||||
const data = this.#qc.getQueryData<InfiniteData<ProxyFontsResponse, PageParam>>(
|
const data = this.#qc.getQueryData<InfiniteData<ProxyFontsResponse, PageParam>>(
|
||||||
this.buildQueryKey(this.#params),
|
this.buildQueryKey(this.#params),
|
||||||
@@ -134,6 +170,9 @@ export class FontStore {
|
|||||||
return data.pages.flatMap(p => p.fonts);
|
return data.pages.flatMap(p => p.fonts);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Manually update the cached font data (useful for optimistic updates)
|
||||||
|
*/
|
||||||
setQueryData(updater: (old: UnifiedFont[] | undefined) => UnifiedFont[]) {
|
setQueryData(updater: (old: UnifiedFont[] | undefined) => UnifiedFont[]) {
|
||||||
const key = this.buildQueryKey(this.#params);
|
const key = this.buildQueryKey(this.#params);
|
||||||
this.#qc.setQueryData<InfiniteData<ProxyFontsResponse, PageParam>>(
|
this.#qc.setQueryData<InfiniteData<ProxyFontsResponse, PageParam>>(
|
||||||
@@ -164,56 +203,90 @@ export class FontStore {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// -- Filter shortcuts --
|
/**
|
||||||
|
* Shortcut to update provider filters
|
||||||
|
*/
|
||||||
setProviders(v: ProxyFontsParams['providers']) {
|
setProviders(v: ProxyFontsParams['providers']) {
|
||||||
this.setParams({ providers: v });
|
this.setParams({ providers: v });
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* Shortcut to update category filters
|
||||||
|
*/
|
||||||
setCategories(v: ProxyFontsParams['categories']) {
|
setCategories(v: ProxyFontsParams['categories']) {
|
||||||
this.setParams({ categories: v });
|
this.setParams({ categories: v });
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* Shortcut to update subset filters
|
||||||
|
*/
|
||||||
setSubsets(v: ProxyFontsParams['subsets']) {
|
setSubsets(v: ProxyFontsParams['subsets']) {
|
||||||
this.setParams({ subsets: v });
|
this.setParams({ subsets: v });
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* Shortcut to update search query
|
||||||
|
*/
|
||||||
setSearch(v: string) {
|
setSearch(v: string) {
|
||||||
this.setParams({ q: v || undefined });
|
this.setParams({ q: v || undefined });
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* Shortcut to update sort order
|
||||||
|
*/
|
||||||
setSort(v: ProxyFontsParams['sort']) {
|
setSort(v: ProxyFontsParams['sort']) {
|
||||||
this.setParams({ sort: v });
|
this.setParams({ sort: v });
|
||||||
}
|
}
|
||||||
|
|
||||||
// -- Pagination navigation --
|
/**
|
||||||
|
* Fetch the next page of results if available
|
||||||
|
*/
|
||||||
async nextPage(): Promise<void> {
|
async nextPage(): Promise<void> {
|
||||||
await this.#observer.fetchNextPage();
|
await this.#observer.fetchNextPage();
|
||||||
}
|
}
|
||||||
prevPage(): void {} // no-op: infinite scroll accumulates forward only; method kept for API compatibility
|
/**
|
||||||
goToPage(_page: number): void {} // no-op
|
* Backward pagination (no-op: infinite scroll accumulates forward only)
|
||||||
|
*/
|
||||||
|
prevPage(): void {}
|
||||||
|
/**
|
||||||
|
* Jump to specific page (no-op for infinite scroll)
|
||||||
|
*/
|
||||||
|
goToPage(_page: number): void {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update the number of items fetched per page
|
||||||
|
*/
|
||||||
setLimit(limit: number) {
|
setLimit(limit: number) {
|
||||||
this.setParams({ limit });
|
this.setParams({ limit });
|
||||||
}
|
}
|
||||||
|
|
||||||
// -- Category views --
|
/**
|
||||||
|
* Derived list of sans-serif fonts in the current set
|
||||||
|
*/
|
||||||
get sansSerifFonts() {
|
get sansSerifFonts() {
|
||||||
return this.fonts.filter(f => f.category === 'sans-serif');
|
return this.fonts.filter(f => f.category === 'sans-serif');
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* Derived list of serif fonts in the current set
|
||||||
|
*/
|
||||||
get serifFonts() {
|
get serifFonts() {
|
||||||
return this.fonts.filter(f => f.category === 'serif');
|
return this.fonts.filter(f => f.category === 'serif');
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* Derived list of display fonts in the current set
|
||||||
|
*/
|
||||||
get displayFonts() {
|
get displayFonts() {
|
||||||
return this.fonts.filter(f => f.category === 'display');
|
return this.fonts.filter(f => f.category === 'display');
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* Derived list of handwriting fonts in the current set
|
||||||
|
*/
|
||||||
get handwritingFonts() {
|
get handwritingFonts() {
|
||||||
return this.fonts.filter(f => f.category === 'handwriting');
|
return this.fonts.filter(f => f.category === 'handwriting');
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* Derived list of monospace fonts in the current set
|
||||||
|
*/
|
||||||
get monospaceFonts() {
|
get monospaceFonts() {
|
||||||
return this.fonts.filter(f => f.category === 'monospace');
|
return this.fonts.filter(f => f.category === 'monospace');
|
||||||
}
|
}
|
||||||
|
|
||||||
// -- Private helpers (TypeScript-private so tests can spy via `as any`) --
|
|
||||||
|
|
||||||
private buildQueryKey(params: FontStoreParams): readonly unknown[] {
|
private buildQueryKey(params: FontStoreParams): readonly unknown[] {
|
||||||
const filtered: Record<string, any> = {};
|
const filtered: Record<string, any> = {};
|
||||||
|
|
||||||
|
|||||||
@@ -31,18 +31,28 @@ export type FontSubset = 'latin' | 'latin-ext' | 'cyrillic' | 'greek' | 'arabic'
|
|||||||
* Combined filter state for font queries
|
* Combined filter state for font queries
|
||||||
*/
|
*/
|
||||||
export interface FontFilters {
|
export interface FontFilters {
|
||||||
/** Selected font providers */
|
/**
|
||||||
|
* Active font providers to fetch from
|
||||||
|
*/
|
||||||
providers: FontProvider[];
|
providers: FontProvider[];
|
||||||
/** Selected font categories */
|
/**
|
||||||
|
* Visual classifications (sans, serif, etc.)
|
||||||
|
*/
|
||||||
categories: FontCategory[];
|
categories: FontCategory[];
|
||||||
/** Selected character subsets */
|
/**
|
||||||
|
* Character sets required for the sample text
|
||||||
|
*/
|
||||||
subsets: FontSubset[];
|
subsets: FontSubset[];
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Filter group identifier */
|
/**
|
||||||
|
* Filter group identifier
|
||||||
|
*/
|
||||||
export type FilterGroup = 'providers' | 'categories' | 'subsets';
|
export type FilterGroup = 'providers' | 'categories' | 'subsets';
|
||||||
|
|
||||||
/** Filter type including search query */
|
/**
|
||||||
|
* Filter type including search query
|
||||||
|
*/
|
||||||
export type FilterType = FilterGroup | 'searchQuery';
|
export type FilterType = FilterGroup | 'searchQuery';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -80,15 +90,25 @@ export type UnifiedFontVariant = FontVariant;
|
|||||||
* Font style URLs
|
* Font style URLs
|
||||||
*/
|
*/
|
||||||
export interface FontStyleUrls {
|
export interface FontStyleUrls {
|
||||||
/** Regular weight URL */
|
/**
|
||||||
|
* URL for the regular (400) weight
|
||||||
|
*/
|
||||||
regular?: string;
|
regular?: string;
|
||||||
/** Italic URL */
|
/**
|
||||||
|
* URL for the italic (400) style
|
||||||
|
*/
|
||||||
italic?: string;
|
italic?: string;
|
||||||
/** Bold weight URL */
|
/**
|
||||||
|
* URL for the bold (700) weight
|
||||||
|
*/
|
||||||
bold?: string;
|
bold?: string;
|
||||||
/** Bold italic URL */
|
/**
|
||||||
|
* URL for the bold-italic (700) style
|
||||||
|
*/
|
||||||
boldItalic?: string;
|
boldItalic?: string;
|
||||||
/** Additional variant mapping */
|
/**
|
||||||
|
* Mapping for all other numeric/custom variants
|
||||||
|
*/
|
||||||
variants?: Partial<Record<UnifiedFontVariant, string>>;
|
variants?: Partial<Record<UnifiedFontVariant, string>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -96,19 +116,24 @@ export interface FontStyleUrls {
|
|||||||
* Font metadata
|
* Font metadata
|
||||||
*/
|
*/
|
||||||
export interface FontMetadata {
|
export interface FontMetadata {
|
||||||
/** Timestamp when font was cached */
|
/**
|
||||||
|
* Epoch timestamp of last successful fetch
|
||||||
|
*/
|
||||||
cachedAt: number;
|
cachedAt: number;
|
||||||
/** Font version from provider */
|
/**
|
||||||
|
* Semantic version string from upstream
|
||||||
|
*/
|
||||||
version?: string;
|
version?: string;
|
||||||
/** Last modified date from provider */
|
/**
|
||||||
|
* ISO date string of last remote update
|
||||||
|
*/
|
||||||
lastModified?: string;
|
lastModified?: string;
|
||||||
/** Popularity rank (if available from provider) */
|
/**
|
||||||
|
* Raw ranking integer from provider
|
||||||
|
*/
|
||||||
popularity?: number;
|
popularity?: number;
|
||||||
/**
|
/**
|
||||||
* Normalized popularity score (0-100)
|
* Normalized score (0-100) used for global sorting
|
||||||
*
|
|
||||||
* Normalized across all fonts for consistent ranking
|
|
||||||
* Higher values indicate more popular fonts
|
|
||||||
*/
|
*/
|
||||||
popularityScore?: number;
|
popularityScore?: number;
|
||||||
}
|
}
|
||||||
@@ -117,17 +142,38 @@ export interface FontMetadata {
|
|||||||
* Font features (variable fonts, axes, tags)
|
* Font features (variable fonts, axes, tags)
|
||||||
*/
|
*/
|
||||||
export interface FontFeatures {
|
export interface FontFeatures {
|
||||||
/** Whether this is a variable font */
|
/**
|
||||||
|
* Whether the font supports fluid weight/width axes
|
||||||
|
*/
|
||||||
isVariable?: boolean;
|
isVariable?: boolean;
|
||||||
/** Variable font axes (for Fontshare) */
|
/**
|
||||||
|
* Definable axes for variable font interpolation
|
||||||
|
*/
|
||||||
axes?: Array<{
|
axes?: Array<{
|
||||||
|
/**
|
||||||
|
* Human-readable axis name (e.g., 'Weight')
|
||||||
|
*/
|
||||||
name: string;
|
name: string;
|
||||||
|
/**
|
||||||
|
* CSS property name (e.g., 'wght')
|
||||||
|
*/
|
||||||
property: string;
|
property: string;
|
||||||
|
/**
|
||||||
|
* Default numeric value for the axis
|
||||||
|
*/
|
||||||
default: number;
|
default: number;
|
||||||
|
/**
|
||||||
|
* Minimum inclusive bound
|
||||||
|
*/
|
||||||
min: number;
|
min: number;
|
||||||
|
/**
|
||||||
|
* Maximum inclusive bound
|
||||||
|
*/
|
||||||
max: number;
|
max: number;
|
||||||
}>;
|
}>;
|
||||||
/** Usage tags (for Fontshare) */
|
/**
|
||||||
|
* Descriptive keywords for search indexing
|
||||||
|
*/
|
||||||
tags?: string[];
|
tags?: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -138,29 +184,44 @@ export interface FontFeatures {
|
|||||||
* for consistent font handling across the application.
|
* for consistent font handling across the application.
|
||||||
*/
|
*/
|
||||||
export interface UnifiedFont {
|
export interface UnifiedFont {
|
||||||
/** Unique identifier (Google: family name, Fontshare: slug) */
|
/**
|
||||||
|
* Unique ID (family name for Google, slug for Fontshare)
|
||||||
|
*/
|
||||||
id: string;
|
id: string;
|
||||||
/** Font display name */
|
/**
|
||||||
|
* Canonical family name for CSS font-family
|
||||||
|
*/
|
||||||
name: string;
|
name: string;
|
||||||
/** Font provider (google | fontshare) */
|
/**
|
||||||
|
* Upstream data source
|
||||||
|
*/
|
||||||
provider: FontProvider;
|
provider: FontProvider;
|
||||||
/**
|
/**
|
||||||
* Provider badge display name
|
* Display label for provider badges
|
||||||
*
|
|
||||||
* Human-readable provider name for UI display
|
|
||||||
* e.g., "Google Fonts" or "Fontshare"
|
|
||||||
*/
|
*/
|
||||||
providerBadge?: string;
|
providerBadge?: string;
|
||||||
/** Font category classification */
|
/**
|
||||||
|
* Primary typographic category
|
||||||
|
*/
|
||||||
category: FontCategory;
|
category: FontCategory;
|
||||||
/** Supported character subsets */
|
/**
|
||||||
|
* All supported character sets
|
||||||
|
*/
|
||||||
subsets: FontSubset[];
|
subsets: FontSubset[];
|
||||||
/** Available font variants (weights, styles) */
|
/**
|
||||||
|
* List of available weights and styles
|
||||||
|
*/
|
||||||
variants: UnifiedFontVariant[];
|
variants: UnifiedFontVariant[];
|
||||||
/** URL mapping for font file downloads */
|
/**
|
||||||
|
* Remote assets for font loading
|
||||||
|
*/
|
||||||
styles: FontStyleUrls;
|
styles: FontStyleUrls;
|
||||||
/** Additional metadata */
|
/**
|
||||||
|
* Technical metadata and rankings
|
||||||
|
*/
|
||||||
metadata: FontMetadata;
|
metadata: FontMetadata;
|
||||||
/** Advanced font features */
|
/**
|
||||||
|
* Variable font details and tags
|
||||||
|
*/
|
||||||
features: FontFeatures;
|
features: FontFeatures;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,3 @@
|
|||||||
/**
|
|
||||||
* ============================================================================
|
|
||||||
* SINGLE EXPORT POINT
|
|
||||||
* ============================================================================
|
|
||||||
*
|
|
||||||
* This is the single export point for all Font types.
|
|
||||||
* All imports should use: `import { X } from '$entities/Font/model/types'`
|
|
||||||
*/
|
|
||||||
|
|
||||||
// Font domain and model types
|
// Font domain and model types
|
||||||
export type {
|
export type {
|
||||||
FilterGroup,
|
FilterGroup,
|
||||||
|
|||||||
@@ -1,9 +1,3 @@
|
|||||||
/**
|
|
||||||
* ============================================================================
|
|
||||||
* STORE TYPES
|
|
||||||
* ============================================================================
|
|
||||||
*/
|
|
||||||
|
|
||||||
import type {
|
import type {
|
||||||
FontCategory,
|
FontCategory,
|
||||||
FontProvider,
|
FontProvider,
|
||||||
@@ -12,37 +6,55 @@ import type {
|
|||||||
} from './font';
|
} from './font';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Font collection state
|
* Global state for the local font collection
|
||||||
*/
|
*/
|
||||||
export interface FontCollectionState {
|
export interface FontCollectionState {
|
||||||
/** All cached fonts */
|
/**
|
||||||
|
* Map of cached fonts indexed by their unique family ID
|
||||||
|
*/
|
||||||
fonts: Record<string, UnifiedFont>;
|
fonts: Record<string, UnifiedFont>;
|
||||||
/** Active filters */
|
/**
|
||||||
|
* Set of active user-defined filters
|
||||||
|
*/
|
||||||
filters: FontCollectionFilters;
|
filters: FontCollectionFilters;
|
||||||
/** Sort configuration */
|
/**
|
||||||
|
* Current sorting parameters for the display list
|
||||||
|
*/
|
||||||
sort: FontCollectionSort;
|
sort: FontCollectionSort;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Font collection filters
|
* Filter configuration for narrow collections
|
||||||
*/
|
*/
|
||||||
export interface FontCollectionFilters {
|
export interface FontCollectionFilters {
|
||||||
/** Search query */
|
/**
|
||||||
|
* Partial family name to match against
|
||||||
|
*/
|
||||||
searchQuery: string;
|
searchQuery: string;
|
||||||
/** Filter by providers */
|
/**
|
||||||
|
* Data sources (Google, Fontshare) to include
|
||||||
|
*/
|
||||||
providers?: FontProvider[];
|
providers?: FontProvider[];
|
||||||
/** Filter by categories */
|
/**
|
||||||
|
* Typographic categories (Serif, Sans, etc.) to include
|
||||||
|
*/
|
||||||
categories?: FontCategory[];
|
categories?: FontCategory[];
|
||||||
/** Filter by subsets */
|
/**
|
||||||
|
* Character sets (Latin, Cyrillic, etc.) to include
|
||||||
|
*/
|
||||||
subsets?: FontSubset[];
|
subsets?: FontSubset[];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Font collection sort configuration
|
* Ordering configuration for the font list
|
||||||
*/
|
*/
|
||||||
export interface FontCollectionSort {
|
export interface FontCollectionSort {
|
||||||
/** Sort field */
|
/**
|
||||||
|
* The font property to order by
|
||||||
|
*/
|
||||||
field: 'name' | 'popularity' | 'category';
|
field: 'name' | 'popularity' | 'category';
|
||||||
/** Sort direction */
|
/**
|
||||||
|
* The sort order (Ascending or Descending)
|
||||||
|
*/
|
||||||
direction: 'asc' | 'desc';
|
direction: 'asc' | 'desc';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -41,15 +41,25 @@ type ThemeSource = 'system' | 'user';
|
|||||||
*/
|
*/
|
||||||
class ThemeManager {
|
class ThemeManager {
|
||||||
// Private reactive state
|
// Private reactive state
|
||||||
/** Current theme value ('light' or 'dark') */
|
/**
|
||||||
|
* Current theme value ('light' or 'dark')
|
||||||
|
*/
|
||||||
#theme = $state<Theme>('light');
|
#theme = $state<Theme>('light');
|
||||||
/** Whether theme is controlled by user or follows system */
|
/**
|
||||||
|
* Whether theme is controlled by user or follows system
|
||||||
|
*/
|
||||||
#source = $state<ThemeSource>('system');
|
#source = $state<ThemeSource>('system');
|
||||||
/** MediaQueryList for detecting system theme changes */
|
/**
|
||||||
|
* MediaQueryList for detecting system theme changes
|
||||||
|
*/
|
||||||
#mediaQuery: MediaQueryList | null = null;
|
#mediaQuery: MediaQueryList | null = null;
|
||||||
/** Persistent storage for user's theme preference */
|
/**
|
||||||
|
* Persistent storage for user's theme preference
|
||||||
|
*/
|
||||||
#store = createPersistentStore<Theme | null>('glyphdiff:theme', null);
|
#store = createPersistentStore<Theme | null>('glyphdiff:theme', null);
|
||||||
/** Bound handler for system theme change events */
|
/**
|
||||||
|
* Bound handler for system theme change events
|
||||||
|
*/
|
||||||
#systemChangeHandler = this.#onSystemChange.bind(this);
|
#systemChangeHandler = this.#onSystemChange.bind(this);
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
@@ -64,22 +74,30 @@ class ThemeManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Current theme value */
|
/**
|
||||||
|
* Current theme value
|
||||||
|
*/
|
||||||
get value(): Theme {
|
get value(): Theme {
|
||||||
return this.#theme;
|
return this.#theme;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Source of current theme ('system' or 'user') */
|
/**
|
||||||
|
* Source of current theme ('system' or 'user')
|
||||||
|
*/
|
||||||
get source(): ThemeSource {
|
get source(): ThemeSource {
|
||||||
return this.#source;
|
return this.#source;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Whether dark theme is active */
|
/**
|
||||||
|
* Whether dark theme is active
|
||||||
|
*/
|
||||||
get isDark(): boolean {
|
get isDark(): boolean {
|
||||||
return this.#theme === 'dark';
|
return this.#theme === 'dark';
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Whether theme is controlled by user (not following system) */
|
/**
|
||||||
|
* Whether theme is controlled by user (not following system)
|
||||||
|
*/
|
||||||
get isUserControlled(): boolean {
|
get isUserControlled(): boolean {
|
||||||
return this.#source === 'user';
|
return this.#source === 'user';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
/** @vitest-environment jsdom */
|
/**
|
||||||
|
* @vitest-environment jsdom
|
||||||
|
*/
|
||||||
|
|
||||||
// ============================================================
|
|
||||||
// Mock MediaQueryListEvent for system theme change simulations
|
// Mock MediaQueryListEvent for system theme change simulations
|
||||||
// Note: Other mocks (ResizeObserver, localStorage, matchMedia) are set up in vitest.setup.unit.ts
|
// Note: Other mocks (ResizeObserver, localStorage, matchMedia) are set up in vitest.setup.unit.ts
|
||||||
// ============================================================
|
|
||||||
|
|
||||||
class MockMediaQueryListEvent extends Event {
|
class MockMediaQueryListEvent extends Event {
|
||||||
matches: boolean;
|
matches: boolean;
|
||||||
@@ -16,9 +16,7 @@ class MockMediaQueryListEvent extends Event {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ============================================================
|
|
||||||
// NOW IT'S SAFE TO IMPORT
|
// NOW IT'S SAFE TO IMPORT
|
||||||
// ============================================================
|
|
||||||
|
|
||||||
import {
|
import {
|
||||||
afterEach,
|
afterEach,
|
||||||
|
|||||||
@@ -15,19 +15,29 @@ const PROXY_API_URL = 'https://api.glyphdiff.com/api/v1/filters' as const;
|
|||||||
* Filter metadata type from backend
|
* Filter metadata type from backend
|
||||||
*/
|
*/
|
||||||
export interface FilterMetadata {
|
export interface FilterMetadata {
|
||||||
/** Filter ID (e.g., "providers", "categories", "subsets") */
|
/**
|
||||||
|
* Filter ID (e.g., "providers", "categories", "subsets")
|
||||||
|
*/
|
||||||
id: string;
|
id: string;
|
||||||
|
|
||||||
/** Display name (e.g., "Font Providers", "Categories", "Character Subsets") */
|
/**
|
||||||
|
* Display name (e.g., "Font Providers", "Categories", "Character Subsets")
|
||||||
|
*/
|
||||||
name: string;
|
name: string;
|
||||||
|
|
||||||
/** Filter description */
|
/**
|
||||||
|
* Filter description
|
||||||
|
*/
|
||||||
description: string;
|
description: string;
|
||||||
|
|
||||||
/** Filter type */
|
/**
|
||||||
|
* Filter type
|
||||||
|
*/
|
||||||
type: 'enum' | 'string' | 'array';
|
type: 'enum' | 'string' | 'array';
|
||||||
|
|
||||||
/** Available filter options */
|
/**
|
||||||
|
* Available filter options
|
||||||
|
*/
|
||||||
options: FilterOption[];
|
options: FilterOption[];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -35,16 +45,24 @@ export interface FilterMetadata {
|
|||||||
* Filter option type
|
* Filter option type
|
||||||
*/
|
*/
|
||||||
export interface FilterOption {
|
export interface FilterOption {
|
||||||
/** Option ID (e.g., "google", "serif", "latin") */
|
/**
|
||||||
|
* Option ID (e.g., "google", "serif", "latin")
|
||||||
|
*/
|
||||||
id: string;
|
id: string;
|
||||||
|
|
||||||
/** Display name (e.g., "Google Fonts", "Serif", "Latin") */
|
/**
|
||||||
|
* Display name (e.g., "Google Fonts", "Serif", "Latin")
|
||||||
|
*/
|
||||||
name: string;
|
name: string;
|
||||||
|
|
||||||
/** Option value (e.g., "google", "serif", "latin") */
|
/**
|
||||||
|
* Option value (e.g., "google", "serif", "latin")
|
||||||
|
*/
|
||||||
value: string;
|
value: string;
|
||||||
|
|
||||||
/** Number of fonts with this value */
|
/**
|
||||||
|
* Number of fonts with this value
|
||||||
|
*/
|
||||||
count: number;
|
count: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -52,7 +70,9 @@ export interface FilterOption {
|
|||||||
* Proxy filters API response
|
* Proxy filters API response
|
||||||
*/
|
*/
|
||||||
export interface ProxyFiltersResponse {
|
export interface ProxyFiltersResponse {
|
||||||
/** Array of filter metadata */
|
/**
|
||||||
|
* Array of filter metadata
|
||||||
|
*/
|
||||||
filters: FilterMetadata[];
|
filters: FilterMetadata[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,15 +1,56 @@
|
|||||||
export type {
|
export type {
|
||||||
|
/**
|
||||||
|
* Top-level configuration for all filters
|
||||||
|
*/
|
||||||
FilterConfig,
|
FilterConfig,
|
||||||
|
/**
|
||||||
|
* Configuration for a single grouping of filter properties
|
||||||
|
*/
|
||||||
FilterGroupConfig,
|
FilterGroupConfig,
|
||||||
} from './types/filter';
|
} from './types/filter';
|
||||||
|
|
||||||
export { filtersStore } from './state/filters.svelte';
|
/**
|
||||||
export { filterManager } from './state/manager.svelte';
|
* Global reactive filter state
|
||||||
|
*/
|
||||||
export {
|
export {
|
||||||
|
/**
|
||||||
|
* Low-level property selection store
|
||||||
|
*/
|
||||||
|
filtersStore,
|
||||||
|
} from './state/filters.svelte';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Main filter controller
|
||||||
|
*/
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* High-level manager for syncing search and filters
|
||||||
|
*/
|
||||||
|
filterManager,
|
||||||
|
} from './state/manager.svelte';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sorting logic
|
||||||
|
*/
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Map of human-readable labels to API sort keys
|
||||||
|
*/
|
||||||
SORT_MAP,
|
SORT_MAP,
|
||||||
|
/**
|
||||||
|
* List of all available sort options for the UI
|
||||||
|
*/
|
||||||
SORT_OPTIONS,
|
SORT_OPTIONS,
|
||||||
|
/**
|
||||||
|
* Valid sort key values
|
||||||
|
*/
|
||||||
type SortApiValue,
|
type SortApiValue,
|
||||||
|
/**
|
||||||
|
* UI model for a single sort option
|
||||||
|
*/
|
||||||
type SortOption,
|
type SortOption,
|
||||||
|
/**
|
||||||
|
* Reactive store for the current sort selection
|
||||||
|
*/
|
||||||
sortStore,
|
sortStore,
|
||||||
} from './store/sortStore.svelte';
|
} from './store/sortStore.svelte';
|
||||||
|
|||||||
@@ -32,13 +32,19 @@ import {
|
|||||||
* Provides reactive access to filter data
|
* Provides reactive access to filter data
|
||||||
*/
|
*/
|
||||||
class FiltersStore {
|
class FiltersStore {
|
||||||
/** TanStack Query result state */
|
/**
|
||||||
|
* TanStack Query result state
|
||||||
|
*/
|
||||||
protected result = $state<QueryObserverResult<FilterMetadata[], Error>>({} as any);
|
protected result = $state<QueryObserverResult<FilterMetadata[], Error>>({} as any);
|
||||||
|
|
||||||
/** TanStack Query observer instance */
|
/**
|
||||||
|
* TanStack Query observer instance
|
||||||
|
*/
|
||||||
protected observer: QueryObserver<FilterMetadata[], Error>;
|
protected observer: QueryObserver<FilterMetadata[], Error>;
|
||||||
|
|
||||||
/** Shared query client */
|
/**
|
||||||
|
* Shared query client
|
||||||
|
*/
|
||||||
protected qc = queryClient;
|
protected qc = queryClient;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -21,17 +21,23 @@ function createSortStore(initial: SortOption = 'Popularity') {
|
|||||||
let current = $state<SortOption>(initial);
|
let current = $state<SortOption>(initial);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
/** Current display label (e.g. 'Popularity') */
|
/**
|
||||||
|
* Current display label (e.g. 'Popularity')
|
||||||
|
*/
|
||||||
get value() {
|
get value() {
|
||||||
return current;
|
return current;
|
||||||
},
|
},
|
||||||
|
|
||||||
/** Mapped API value (e.g. 'popularity') */
|
/**
|
||||||
|
* Mapped API value (e.g. 'popularity')
|
||||||
|
*/
|
||||||
get apiValue(): SortApiValue {
|
get apiValue(): SortApiValue {
|
||||||
return SORT_MAP[current];
|
return SORT_MAP[current];
|
||||||
},
|
},
|
||||||
|
|
||||||
/** Set the active sort option by its display label */
|
/**
|
||||||
|
* Set the active sort option by its display label
|
||||||
|
*/
|
||||||
set(option: SortOption) {
|
set(option: SortOption) {
|
||||||
current = option;
|
current = option;
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,12 +1,27 @@
|
|||||||
import type { Property } from '$shared/lib';
|
import type { Property } from '$shared/lib';
|
||||||
|
|
||||||
export interface FilterGroupConfig<TValue extends string> {
|
export interface FilterGroupConfig<TValue extends string> {
|
||||||
|
/**
|
||||||
|
* Unique identifier for the filter group (e.g. 'categories')
|
||||||
|
*/
|
||||||
id: string;
|
id: string;
|
||||||
|
/**
|
||||||
|
* Human-readable label displayed in the UI header
|
||||||
|
*/
|
||||||
label: string;
|
label: string;
|
||||||
|
/**
|
||||||
|
* List of toggleable properties within this group
|
||||||
|
*/
|
||||||
properties: Property<TValue>[];
|
properties: Property<TValue>[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface FilterConfig<TValue extends string> {
|
export interface FilterConfig<TValue extends string> {
|
||||||
|
/**
|
||||||
|
* Optional string to filter results by name
|
||||||
|
*/
|
||||||
queryValue?: string;
|
queryValue?: string;
|
||||||
|
/**
|
||||||
|
* Collection of filter groups to display
|
||||||
|
*/
|
||||||
groups: FilterGroupConfig<TValue>[];
|
groups: FilterGroupConfig<TValue>[];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -30,9 +30,12 @@ import { SvelteMap } from 'svelte/reactivity';
|
|||||||
type ControlOnlyFields<T extends string = string> = Omit<ControlModel<T>, keyof ControlDataModel>;
|
type ControlOnlyFields<T extends string = string> = Omit<ControlModel<T>, keyof ControlDataModel>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A control with its instance
|
* A control with its associated instance
|
||||||
*/
|
*/
|
||||||
export interface Control extends ControlOnlyFields<ControlId> {
|
export interface Control extends ControlOnlyFields<ControlId> {
|
||||||
|
/**
|
||||||
|
* The reactive typography control instance
|
||||||
|
*/
|
||||||
instance: TypographyControl;
|
instance: TypographyControl;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -40,9 +43,21 @@ export interface Control extends ControlOnlyFields<ControlId> {
|
|||||||
* Storage schema for typography settings
|
* Storage schema for typography settings
|
||||||
*/
|
*/
|
||||||
export interface TypographySettings {
|
export interface TypographySettings {
|
||||||
|
/**
|
||||||
|
* Base font size (User preference, unscaled)
|
||||||
|
*/
|
||||||
fontSize: number;
|
fontSize: number;
|
||||||
|
/**
|
||||||
|
* Numeric font weight (100-900)
|
||||||
|
*/
|
||||||
fontWeight: number;
|
fontWeight: number;
|
||||||
|
/**
|
||||||
|
* Line height multiplier (e.g. 1.5)
|
||||||
|
*/
|
||||||
lineHeight: number;
|
lineHeight: number;
|
||||||
|
/**
|
||||||
|
* Letter spacing in em/px
|
||||||
|
*/
|
||||||
letterSpacing: number;
|
letterSpacing: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -53,13 +68,21 @@ export interface TypographySettings {
|
|||||||
* responsive scaling support for font size.
|
* responsive scaling support for font size.
|
||||||
*/
|
*/
|
||||||
export class TypographySettingsManager {
|
export class TypographySettingsManager {
|
||||||
/** Map of controls keyed by ID */
|
/**
|
||||||
|
* Internal map of reactive controls keyed by their identifier
|
||||||
|
*/
|
||||||
#controls = new SvelteMap<string, Control>();
|
#controls = new SvelteMap<string, Control>();
|
||||||
/** Responsive multiplier for font size display */
|
/**
|
||||||
|
* Global multiplier for responsive font size scaling
|
||||||
|
*/
|
||||||
#multiplier = $state(1);
|
#multiplier = $state(1);
|
||||||
/** Persistent storage for settings */
|
/**
|
||||||
|
* LocalStorage-backed storage for persistence
|
||||||
|
*/
|
||||||
#storage: PersistentStore<TypographySettings>;
|
#storage: PersistentStore<TypographySettings>;
|
||||||
/** Base font size (user preference, unscaled) */
|
/**
|
||||||
|
* The underlying font size before responsive scaling is applied
|
||||||
|
*/
|
||||||
#baseSize = $state(DEFAULT_FONT_SIZE);
|
#baseSize = $state(DEFAULT_FONT_SIZE);
|
||||||
|
|
||||||
constructor(configs: ControlModel<ControlId>[], storage: PersistentStore<TypographySettings>) {
|
constructor(configs: ControlModel<ControlId>[], storage: PersistentStore<TypographySettings>) {
|
||||||
@@ -131,16 +154,15 @@ export class TypographySettingsManager {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Current multiplier for responsive scaling */
|
/**
|
||||||
|
* Active scaling factor for the rendered font size
|
||||||
|
*/
|
||||||
get multiplier() {
|
get multiplier() {
|
||||||
return this.#multiplier;
|
return this.#multiplier;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the multiplier and update font size display
|
* Updates the multiplier and recalculates dependent control values
|
||||||
*
|
|
||||||
* When multiplier changes, the font size control's display value
|
|
||||||
* is updated to reflect the new scale while preserving base size.
|
|
||||||
*/
|
*/
|
||||||
set multiplier(value: number) {
|
set multiplier(value: number) {
|
||||||
if (this.#multiplier === value) return;
|
if (this.#multiplier === value) return;
|
||||||
@@ -154,14 +176,15 @@ export class TypographySettingsManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The scaled size for CSS usage
|
* The actual pixel value for CSS font-size (baseSize * multiplier)
|
||||||
* Returns baseSize * multiplier for actual rendering
|
|
||||||
*/
|
*/
|
||||||
get renderedSize() {
|
get renderedSize() {
|
||||||
return this.#baseSize * this.#multiplier;
|
return this.#baseSize * this.#multiplier;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** The base size (User Preference) */
|
/**
|
||||||
|
* The raw font size preference before scaling
|
||||||
|
*/
|
||||||
get baseSize() {
|
get baseSize() {
|
||||||
return this.#baseSize;
|
return this.#baseSize;
|
||||||
}
|
}
|
||||||
@@ -173,45 +196,63 @@ export class TypographySettingsManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Getters for controls
|
* List of all managed typography controls
|
||||||
*/
|
*/
|
||||||
get controls() {
|
get controls() {
|
||||||
return Array.from(this.#controls.values());
|
return Array.from(this.#controls.values());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reactive instance for weight manipulation
|
||||||
|
*/
|
||||||
get weightControl() {
|
get weightControl() {
|
||||||
return this.#controls.get('font_weight')?.instance;
|
return this.#controls.get('font_weight')?.instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reactive instance for size manipulation
|
||||||
|
*/
|
||||||
get sizeControl() {
|
get sizeControl() {
|
||||||
return this.#controls.get('font_size')?.instance;
|
return this.#controls.get('font_size')?.instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reactive instance for line-height manipulation
|
||||||
|
*/
|
||||||
get heightControl() {
|
get heightControl() {
|
||||||
return this.#controls.get('line_height')?.instance;
|
return this.#controls.get('line_height')?.instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reactive instance for letter-spacing manipulation
|
||||||
|
*/
|
||||||
get spacingControl() {
|
get spacingControl() {
|
||||||
return this.#controls.get('letter_spacing')?.instance;
|
return this.#controls.get('letter_spacing')?.instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Getters for values (besides font-size)
|
* Current numeric font weight (reactive)
|
||||||
*/
|
*/
|
||||||
get weight() {
|
get weight() {
|
||||||
return this.#controls.get('font_weight')?.instance.value ?? DEFAULT_FONT_WEIGHT;
|
return this.#controls.get('font_weight')?.instance.value ?? DEFAULT_FONT_WEIGHT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Current numeric line height (reactive)
|
||||||
|
*/
|
||||||
get height() {
|
get height() {
|
||||||
return this.#controls.get('line_height')?.instance.value ?? DEFAULT_LINE_HEIGHT;
|
return this.#controls.get('line_height')?.instance.value ?? DEFAULT_LINE_HEIGHT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Current numeric letter spacing (reactive)
|
||||||
|
*/
|
||||||
get spacing() {
|
get spacing() {
|
||||||
return this.#controls.get('letter_spacing')?.instance.value ?? DEFAULT_LETTER_SPACING;
|
return this.#controls.get('letter_spacing')?.instance.value ?? DEFAULT_LETTER_SPACING;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Reset all controls to default values
|
* Reset all controls to project-defined defaults
|
||||||
*/
|
*/
|
||||||
reset() {
|
reset() {
|
||||||
this.#storage.clear();
|
this.#storage.clear();
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
/** @vitest-environment jsdom */
|
/**
|
||||||
|
* @vitest-environment jsdom
|
||||||
|
*/
|
||||||
import {
|
import {
|
||||||
DEFAULT_FONT_SIZE,
|
DEFAULT_FONT_SIZE,
|
||||||
DEFAULT_FONT_WEIGHT,
|
DEFAULT_FONT_WEIGHT,
|
||||||
|
|||||||
@@ -1,3 +1,9 @@
|
|||||||
|
/**
|
||||||
|
* Application entry point
|
||||||
|
*
|
||||||
|
* Mounts the main App component to the DOM and initializes
|
||||||
|
* global styles.
|
||||||
|
*/
|
||||||
import App from '$app/App.svelte';
|
import App from '$app/App.svelte';
|
||||||
import { mount } from 'svelte';
|
import { mount } from 'svelte';
|
||||||
import '$app/styles/app.css';
|
import '$app/styles/app.css';
|
||||||
|
|||||||
@@ -41,10 +41,14 @@ export class ApiError extends Error {
|
|||||||
* @param response - Original fetch Response object
|
* @param response - Original fetch Response object
|
||||||
*/
|
*/
|
||||||
constructor(
|
constructor(
|
||||||
/** HTTP status code */
|
/**
|
||||||
|
* HTTP status code
|
||||||
|
*/
|
||||||
public status: number,
|
public status: number,
|
||||||
message: string,
|
message: string,
|
||||||
/** Original Response object for inspection */
|
/**
|
||||||
|
* Original Response object for inspection
|
||||||
|
*/
|
||||||
public response?: Response,
|
public response?: Response,
|
||||||
) {
|
) {
|
||||||
super(message);
|
super(message);
|
||||||
|
|||||||
@@ -15,15 +15,25 @@ import { QueryClient } from '@tanstack/query-core';
|
|||||||
export const queryClient = new QueryClient({
|
export const queryClient = new QueryClient({
|
||||||
defaultOptions: {
|
defaultOptions: {
|
||||||
queries: {
|
queries: {
|
||||||
/** Data remains fresh for 5 minutes after fetch */
|
/**
|
||||||
|
* Data remains fresh for 5 minutes after fetch
|
||||||
|
*/
|
||||||
staleTime: 5 * 60 * 1000,
|
staleTime: 5 * 60 * 1000,
|
||||||
/** Unused cache entries are removed after 10 minutes */
|
/**
|
||||||
|
* Unused cache entries are removed after 10 minutes
|
||||||
|
*/
|
||||||
gcTime: 10 * 60 * 1000,
|
gcTime: 10 * 60 * 1000,
|
||||||
/** Don't refetch when window regains focus */
|
/**
|
||||||
|
* Don't refetch when window regains focus
|
||||||
|
*/
|
||||||
refetchOnWindowFocus: false,
|
refetchOnWindowFocus: false,
|
||||||
/** Refetch on mount if data is stale */
|
/**
|
||||||
|
* Refetch on mount if data is stale
|
||||||
|
*/
|
||||||
refetchOnMount: true,
|
refetchOnMount: true,
|
||||||
/** Retry failed requests up to 3 times */
|
/**
|
||||||
|
* Retry failed requests up to 3 times
|
||||||
|
*/
|
||||||
retry: 3,
|
retry: 3,
|
||||||
/**
|
/**
|
||||||
* Exponential backoff for retries
|
* Exponential backoff for retries
|
||||||
|
|||||||
@@ -3,21 +3,35 @@
|
|||||||
* Ensures consistent serialization for batch requests by sorting IDs.
|
* Ensures consistent serialization for batch requests by sorting IDs.
|
||||||
*/
|
*/
|
||||||
export const fontKeys = {
|
export const fontKeys = {
|
||||||
/** Base key for all font queries */
|
/**
|
||||||
|
* Base key for all font queries
|
||||||
|
*/
|
||||||
all: ['fonts'] as const,
|
all: ['fonts'] as const,
|
||||||
|
|
||||||
/** Keys for font list queries */
|
/**
|
||||||
|
* Keys for font list queries
|
||||||
|
*/
|
||||||
lists: () => [...fontKeys.all, 'list'] as const,
|
lists: () => [...fontKeys.all, 'list'] as const,
|
||||||
/** Specific font list key with filter parameters */
|
/**
|
||||||
|
* Specific font list key with filter parameters
|
||||||
|
*/
|
||||||
list: (params: object) => [...fontKeys.lists(), params] as const,
|
list: (params: object) => [...fontKeys.lists(), params] as const,
|
||||||
|
|
||||||
/** Keys for font batch queries */
|
/**
|
||||||
|
* Keys for font batch queries
|
||||||
|
*/
|
||||||
batches: () => [...fontKeys.all, 'batch'] as const,
|
batches: () => [...fontKeys.all, 'batch'] as const,
|
||||||
/** Specific batch key, sorted for stability */
|
/**
|
||||||
|
* Specific batch key, sorted for stability
|
||||||
|
*/
|
||||||
batch: (ids: string[]) => [...fontKeys.batches(), [...ids].sort()] as const,
|
batch: (ids: string[]) => [...fontKeys.batches(), [...ids].sort()] as const,
|
||||||
|
|
||||||
/** Keys for font detail queries */
|
/**
|
||||||
|
* Keys for font detail queries
|
||||||
|
*/
|
||||||
details: () => [...fontKeys.all, 'detail'] as const,
|
details: () => [...fontKeys.all, 'detail'] as const,
|
||||||
/** Specific font detail key by ID */
|
/**
|
||||||
|
* Specific font detail key by ID
|
||||||
|
*/
|
||||||
detail: (id: string) => [...fontKeys.details(), id] as const,
|
detail: (id: string) => [...fontKeys.details(), id] as const,
|
||||||
} as const;
|
} as const;
|
||||||
|
|||||||
@@ -12,20 +12,37 @@ import {
|
|||||||
* each font's actual advance widths independently.
|
* each font's actual advance widths independently.
|
||||||
*/
|
*/
|
||||||
export interface ComparisonLine {
|
export interface ComparisonLine {
|
||||||
/** Full text of this line as returned by pretext. */
|
/**
|
||||||
|
* Full text of this line as returned by pretext.
|
||||||
|
*/
|
||||||
text: string;
|
text: string;
|
||||||
/** Rendered width of this line in pixels — maximum across font A and font B. */
|
/**
|
||||||
|
* Rendered width of this line in pixels — maximum across font A and font B.
|
||||||
|
*/
|
||||||
width: number;
|
width: number;
|
||||||
|
/**
|
||||||
|
* Individual character metadata for both fonts in this line
|
||||||
|
*/
|
||||||
chars: Array<{
|
chars: Array<{
|
||||||
/** The grapheme cluster string (may be >1 code unit for emoji, etc.). */
|
/**
|
||||||
|
* The grapheme cluster string (may be >1 code unit for emoji, etc.).
|
||||||
|
*/
|
||||||
char: string;
|
char: string;
|
||||||
/** X offset from the start of the line in font A, in pixels. */
|
/**
|
||||||
|
* X offset from the start of the line in font A, in pixels.
|
||||||
|
*/
|
||||||
xA: number;
|
xA: number;
|
||||||
/** Advance width of this grapheme in font A, in pixels. */
|
/**
|
||||||
|
* Advance width of this grapheme in font A, in pixels.
|
||||||
|
*/
|
||||||
widthA: number;
|
widthA: number;
|
||||||
/** X offset from the start of the line in font B, in pixels. */
|
/**
|
||||||
|
* X offset from the start of the line in font B, in pixels.
|
||||||
|
*/
|
||||||
xB: number;
|
xB: number;
|
||||||
/** Advance width of this grapheme in font B, in pixels. */
|
/**
|
||||||
|
* Advance width of this grapheme in font B, in pixels.
|
||||||
|
*/
|
||||||
widthB: number;
|
widthB: number;
|
||||||
}>;
|
}>;
|
||||||
}
|
}
|
||||||
@@ -34,9 +51,13 @@ export interface ComparisonLine {
|
|||||||
* Aggregated output of a dual-font layout pass.
|
* Aggregated output of a dual-font layout pass.
|
||||||
*/
|
*/
|
||||||
export interface ComparisonResult {
|
export interface ComparisonResult {
|
||||||
/** Per-line grapheme data for both fonts. Empty when input text is empty. */
|
/**
|
||||||
|
* Per-line grapheme data for both fonts. Empty when input text is empty.
|
||||||
|
*/
|
||||||
lines: ComparisonLine[];
|
lines: ComparisonLine[];
|
||||||
/** Total height in pixels. Equals `lines.length * lineHeight` (pretext guarantee). */
|
/**
|
||||||
|
* Total height in pixels. Equals `lines.length * lineHeight` (pretext guarantee).
|
||||||
|
*/
|
||||||
totalHeight: number;
|
totalHeight: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -28,8 +28,6 @@ describe('CharacterComparisonEngine', () => {
|
|||||||
engine = new CharacterComparisonEngine();
|
engine = new CharacterComparisonEngine();
|
||||||
});
|
});
|
||||||
|
|
||||||
// --- layout() ---
|
|
||||||
|
|
||||||
it('returns empty result for empty string', () => {
|
it('returns empty result for empty string', () => {
|
||||||
const result = engine.layout('', '400 16px "FontA"', '400 16px "FontB"', 500, 20);
|
const result = engine.layout('', '400 16px "FontA"', '400 16px "FontB"', 500, 20);
|
||||||
expect(result.lines).toHaveLength(0);
|
expect(result.lines).toHaveLength(0);
|
||||||
@@ -111,8 +109,6 @@ describe('CharacterComparisonEngine', () => {
|
|||||||
expect(r2).not.toBe(r1);
|
expect(r2).not.toBe(r1);
|
||||||
});
|
});
|
||||||
|
|
||||||
// --- getCharState() ---
|
|
||||||
|
|
||||||
it('getCharState returns proximity 1 when slider is exactly over char center', () => {
|
it('getCharState returns proximity 1 when slider is exactly over char center', () => {
|
||||||
// 'A' only: FontA width=10. Container=500px. Line centered.
|
// 'A' only: FontA width=10. Container=500px. Line centered.
|
||||||
// lineXOffset = (500 - maxWidth) / 2. maxWidth = max(10, 15) = 15 (FontB is wider).
|
// lineXOffset = (500 - maxWidth) / 2. maxWidth = max(10, 15) = 15 (FontB is wider).
|
||||||
|
|||||||
@@ -10,16 +10,29 @@ import {
|
|||||||
* sequences and combining characters each produce exactly one entry.
|
* sequences and combining characters each produce exactly one entry.
|
||||||
*/
|
*/
|
||||||
export interface LayoutLine {
|
export interface LayoutLine {
|
||||||
/** Full text of this line as returned by pretext. */
|
/**
|
||||||
|
* Full text of this line as returned by pretext.
|
||||||
|
*/
|
||||||
text: string;
|
text: string;
|
||||||
/** Rendered width of this line in pixels. */
|
/**
|
||||||
|
* Rendered width of this line in pixels.
|
||||||
|
*/
|
||||||
width: number;
|
width: number;
|
||||||
|
/**
|
||||||
|
* Individual character metadata for this line
|
||||||
|
*/
|
||||||
chars: Array<{
|
chars: Array<{
|
||||||
/** The grapheme cluster string (may be >1 code unit for emoji, etc.). */
|
/**
|
||||||
|
* The grapheme cluster string (may be >1 code unit for emoji, etc.).
|
||||||
|
*/
|
||||||
char: string;
|
char: string;
|
||||||
/** X offset from the start of the line, in pixels. */
|
/**
|
||||||
|
* X offset from the start of the line, in pixels.
|
||||||
|
*/
|
||||||
x: number;
|
x: number;
|
||||||
/** Advance width of this grapheme, in pixels. */
|
/**
|
||||||
|
* Advance width of this grapheme, in pixels.
|
||||||
|
*/
|
||||||
width: number;
|
width: number;
|
||||||
}>;
|
}>;
|
||||||
}
|
}
|
||||||
@@ -28,9 +41,13 @@ export interface LayoutLine {
|
|||||||
* Aggregated output of a single-font layout pass.
|
* Aggregated output of a single-font layout pass.
|
||||||
*/
|
*/
|
||||||
export interface LayoutResult {
|
export interface LayoutResult {
|
||||||
/** Per-line grapheme data. Empty when input text is empty. */
|
/**
|
||||||
|
* Per-line grapheme data. Empty when input text is empty.
|
||||||
|
*/
|
||||||
lines: LayoutLine[];
|
lines: LayoutLine[];
|
||||||
/** Total height in pixels. Equals `lines.length * lineHeight` (pretext guarantee). */
|
/**
|
||||||
|
* Total height in pixels. Equals `lines.length * lineHeight` (pretext guarantee).
|
||||||
|
*/
|
||||||
totalHeight: number;
|
totalHeight: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -65,7 +82,9 @@ export class TextLayoutEngine {
|
|||||||
*/
|
*/
|
||||||
#segmenter: Intl.Segmenter;
|
#segmenter: Intl.Segmenter;
|
||||||
|
|
||||||
/** @param locale BCP 47 language tag passed to Intl.Segmenter. Defaults to the runtime locale. */
|
/**
|
||||||
|
* @param locale BCP 47 language tag passed to Intl.Segmenter. Defaults to the runtime locale.
|
||||||
|
*/
|
||||||
constructor(locale?: string) {
|
constructor(locale?: string) {
|
||||||
this.#segmenter = new Intl.Segmenter(locale, { granularity: 'grapheme' });
|
this.#segmenter = new Intl.Segmenter(locale, { granularity: 'grapheme' });
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -32,7 +32,9 @@ export function createDebouncedState<T>(initialValue: T, wait: number = 300) {
|
|||||||
}, wait);
|
}, wait);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
/** Current value with immediate updates (for UI binding) */
|
/**
|
||||||
|
* Current value with immediate updates (for UI binding)
|
||||||
|
*/
|
||||||
get immediate() {
|
get immediate() {
|
||||||
return immediate;
|
return immediate;
|
||||||
},
|
},
|
||||||
@@ -41,7 +43,9 @@ export function createDebouncedState<T>(initialValue: T, wait: number = 300) {
|
|||||||
// Manually trigger the debounce on write
|
// Manually trigger the debounce on write
|
||||||
updateDebounced(value);
|
updateDebounced(value);
|
||||||
},
|
},
|
||||||
/** Current value with debounced updates (for logic/operations) */
|
/**
|
||||||
|
* Current value with debounced updates (for logic/operations)
|
||||||
|
*/
|
||||||
get debounced() {
|
get debounced() {
|
||||||
return debounced;
|
return debounced;
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -28,7 +28,9 @@ import { SvelteMap } from 'svelte/reactivity';
|
|||||||
* Base entity interface requiring an ID field
|
* Base entity interface requiring an ID field
|
||||||
*/
|
*/
|
||||||
export interface Entity {
|
export interface Entity {
|
||||||
/** Unique identifier for the entity */
|
/**
|
||||||
|
* Unique identifier for the entity
|
||||||
|
*/
|
||||||
id: string;
|
id: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -39,7 +41,9 @@ export interface Entity {
|
|||||||
* triggers updates when entities are added, removed, or modified.
|
* triggers updates when entities are added, removed, or modified.
|
||||||
*/
|
*/
|
||||||
export class EntityStore<T extends Entity> {
|
export class EntityStore<T extends Entity> {
|
||||||
/** Reactive map of entities keyed by ID */
|
/**
|
||||||
|
* Reactive map of entities keyed by ID
|
||||||
|
*/
|
||||||
#entities = new SvelteMap<string, T>();
|
#entities = new SvelteMap<string, T>();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -29,13 +29,21 @@
|
|||||||
* @template TValue - The type of the property value (typically string)
|
* @template TValue - The type of the property value (typically string)
|
||||||
*/
|
*/
|
||||||
export interface Property<TValue extends string> {
|
export interface Property<TValue extends string> {
|
||||||
/** Unique identifier for the property */
|
/**
|
||||||
|
* Unique string identifier for the filterable property
|
||||||
|
*/
|
||||||
id: string;
|
id: string;
|
||||||
/** Human-readable display name */
|
/**
|
||||||
|
* Human-readable label for UI display
|
||||||
|
*/
|
||||||
name: string;
|
name: string;
|
||||||
/** Underlying value for filtering logic */
|
/**
|
||||||
|
* Underlying machine-readable value used for filtering logic
|
||||||
|
*/
|
||||||
value: TValue;
|
value: TValue;
|
||||||
/** Whether the property is currently selected */
|
/**
|
||||||
|
* Current selection status (reactive)
|
||||||
|
*/
|
||||||
selected?: boolean;
|
selected?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -45,7 +53,9 @@ export interface Property<TValue extends string> {
|
|||||||
* @template TValue - The type of property values
|
* @template TValue - The type of property values
|
||||||
*/
|
*/
|
||||||
export interface FilterModel<TValue extends string> {
|
export interface FilterModel<TValue extends string> {
|
||||||
/** Array of filterable properties */
|
/**
|
||||||
|
* Collection of properties that can be toggled in this filter
|
||||||
|
*/
|
||||||
properties: Property<TValue>[];
|
properties: Property<TValue>[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
/** @vitest-environment jsdom */
|
/**
|
||||||
|
* @vitest-environment jsdom
|
||||||
|
*/
|
||||||
import {
|
import {
|
||||||
afterEach,
|
afterEach,
|
||||||
beforeEach,
|
beforeEach,
|
||||||
|
|||||||
@@ -32,19 +32,33 @@ import { Spring } from 'svelte/motion';
|
|||||||
* Configuration options for perspective effects
|
* Configuration options for perspective effects
|
||||||
*/
|
*/
|
||||||
export interface PerspectiveConfig {
|
export interface PerspectiveConfig {
|
||||||
/** Z-axis translation per level in pixels */
|
/**
|
||||||
|
* Z-axis translation per level in pixels
|
||||||
|
*/
|
||||||
depthStep?: number;
|
depthStep?: number;
|
||||||
/** Scale reduction per level (0-1) */
|
/**
|
||||||
|
* Scale reduction per level (0-1)
|
||||||
|
*/
|
||||||
scaleStep?: number;
|
scaleStep?: number;
|
||||||
/** Blur amount per level in pixels */
|
/**
|
||||||
|
* Blur amount per level in pixels
|
||||||
|
*/
|
||||||
blurStep?: number;
|
blurStep?: number;
|
||||||
/** Opacity reduction per level (0-1) */
|
/**
|
||||||
|
* Opacity reduction per level (0-1)
|
||||||
|
*/
|
||||||
opacityStep?: number;
|
opacityStep?: number;
|
||||||
/** Parallax movement intensity per level */
|
/**
|
||||||
|
* Parallax movement intensity per level
|
||||||
|
*/
|
||||||
parallaxIntensity?: number;
|
parallaxIntensity?: number;
|
||||||
/** Horizontal offset - positive for right, negative for left */
|
/**
|
||||||
|
* Horizontal offset - positive for right, negative for left
|
||||||
|
*/
|
||||||
horizontalOffset?: number;
|
horizontalOffset?: number;
|
||||||
/** Layout mode: 'center' for centered, 'split' for side-by-side */
|
/**
|
||||||
|
* Layout mode: 'center' for centered, 'split' for side-by-side
|
||||||
|
*/
|
||||||
layoutMode?: 'center' | 'split';
|
layoutMode?: 'center' | 'split';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -39,15 +39,25 @@
|
|||||||
* Customize to match your design system's breakpoints.
|
* Customize to match your design system's breakpoints.
|
||||||
*/
|
*/
|
||||||
export interface Breakpoints {
|
export interface Breakpoints {
|
||||||
/** Mobile devices - default 640px */
|
/**
|
||||||
|
* Mobile devices - default 640px
|
||||||
|
*/
|
||||||
mobile: number;
|
mobile: number;
|
||||||
/** Tablet portrait - default 768px */
|
/**
|
||||||
|
* Tablet portrait - default 768px
|
||||||
|
*/
|
||||||
tabletPortrait: number;
|
tabletPortrait: number;
|
||||||
/** Tablet landscape - default 1024px */
|
/**
|
||||||
|
* Tablet landscape - default 1024px
|
||||||
|
*/
|
||||||
tablet: number;
|
tablet: number;
|
||||||
/** Desktop - default 1280px */
|
/**
|
||||||
|
* Desktop - default 1280px
|
||||||
|
*/
|
||||||
desktop: number;
|
desktop: number;
|
||||||
/** Large desktop - default 1536px */
|
/**
|
||||||
|
* Large desktop - default 1536px
|
||||||
|
*/
|
||||||
desktopLarge: number;
|
desktopLarge: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -206,66 +216,108 @@ export function createResponsiveManager(customBreakpoints?: Partial<Breakpoints>
|
|||||||
);
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
/** Viewport width in pixels */
|
/**
|
||||||
|
* Current viewport width in pixels (reactive)
|
||||||
|
*/
|
||||||
get width() {
|
get width() {
|
||||||
return width;
|
return width;
|
||||||
},
|
},
|
||||||
/** Viewport height in pixels */
|
/**
|
||||||
|
* Current viewport height in pixels (reactive)
|
||||||
|
*/
|
||||||
get height() {
|
get height() {
|
||||||
return height;
|
return height;
|
||||||
},
|
},
|
||||||
|
|
||||||
// Standard breakpoints
|
/**
|
||||||
|
* True if viewport width is below the mobile threshold
|
||||||
|
*/
|
||||||
get isMobile() {
|
get isMobile() {
|
||||||
return isMobile;
|
return isMobile;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* True if viewport width is between mobile and tablet portrait thresholds
|
||||||
|
*/
|
||||||
get isTabletPortrait() {
|
get isTabletPortrait() {
|
||||||
return isTabletPortrait;
|
return isTabletPortrait;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* True if viewport width is between tablet portrait and desktop thresholds
|
||||||
|
*/
|
||||||
get isTablet() {
|
get isTablet() {
|
||||||
return isTablet;
|
return isTablet;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* True if viewport width is between desktop and large desktop thresholds
|
||||||
|
*/
|
||||||
get isDesktop() {
|
get isDesktop() {
|
||||||
return isDesktop;
|
return isDesktop;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* True if viewport width is at or above the large desktop threshold
|
||||||
|
*/
|
||||||
get isDesktopLarge() {
|
get isDesktopLarge() {
|
||||||
return isDesktopLarge;
|
return isDesktopLarge;
|
||||||
},
|
},
|
||||||
|
|
||||||
// Convenience groupings
|
/**
|
||||||
|
* True if viewport width is below the desktop threshold
|
||||||
|
*/
|
||||||
get isMobileOrTablet() {
|
get isMobileOrTablet() {
|
||||||
return isMobileOrTablet;
|
return isMobileOrTablet;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* True if viewport width is at or above the tablet portrait threshold
|
||||||
|
*/
|
||||||
get isTabletOrDesktop() {
|
get isTabletOrDesktop() {
|
||||||
return isTabletOrDesktop;
|
return isTabletOrDesktop;
|
||||||
},
|
},
|
||||||
|
|
||||||
// Orientation
|
/**
|
||||||
|
* Current screen orientation (portrait | landscape)
|
||||||
|
*/
|
||||||
get orientation() {
|
get orientation() {
|
||||||
return orientation;
|
return orientation;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* True if screen height is greater than width
|
||||||
|
*/
|
||||||
get isPortrait() {
|
get isPortrait() {
|
||||||
return isPortrait;
|
return isPortrait;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* True if screen width is greater than height
|
||||||
|
*/
|
||||||
get isLandscape() {
|
get isLandscape() {
|
||||||
return isLandscape;
|
return isLandscape;
|
||||||
},
|
},
|
||||||
|
|
||||||
// Device capabilities
|
/**
|
||||||
|
* True if the device supports touch interaction
|
||||||
|
*/
|
||||||
get isTouchDevice() {
|
get isTouchDevice() {
|
||||||
return isTouchDevice;
|
return isTouchDevice;
|
||||||
},
|
},
|
||||||
|
|
||||||
// Current breakpoint
|
/**
|
||||||
|
* Name of the currently active breakpoint (reactive)
|
||||||
|
*/
|
||||||
get currentBreakpoint() {
|
get currentBreakpoint() {
|
||||||
return currentBreakpoint;
|
return currentBreakpoint;
|
||||||
},
|
},
|
||||||
|
|
||||||
// Methods
|
/**
|
||||||
|
* Initialization function to start event listeners
|
||||||
|
*/
|
||||||
init,
|
init,
|
||||||
|
/**
|
||||||
|
* Helper to check for custom width ranges
|
||||||
|
*/
|
||||||
matches,
|
matches,
|
||||||
|
|
||||||
// Breakpoint values (for custom logic)
|
/**
|
||||||
|
* Underlying breakpoint pixel values
|
||||||
|
*/
|
||||||
breakpoints,
|
breakpoints,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -34,13 +34,21 @@ import {
|
|||||||
* Defines the bounds and stepping behavior for a control
|
* Defines the bounds and stepping behavior for a control
|
||||||
*/
|
*/
|
||||||
export interface ControlDataModel {
|
export interface ControlDataModel {
|
||||||
/** Current numeric value */
|
/**
|
||||||
|
* Initial or current numeric value
|
||||||
|
*/
|
||||||
value: number;
|
value: number;
|
||||||
/** Minimum allowed value (inclusive) */
|
/**
|
||||||
|
* Lower inclusive bound
|
||||||
|
*/
|
||||||
min: number;
|
min: number;
|
||||||
/** Maximum allowed value (inclusive) */
|
/**
|
||||||
|
* Upper inclusive bound
|
||||||
|
*/
|
||||||
max: number;
|
max: number;
|
||||||
/** Step size for increment/decrement operations */
|
/**
|
||||||
|
* Precision for increment/decrement operations
|
||||||
|
*/
|
||||||
step: number;
|
step: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -50,13 +58,21 @@ export interface ControlDataModel {
|
|||||||
* @template T - Type for the control identifier
|
* @template T - Type for the control identifier
|
||||||
*/
|
*/
|
||||||
export interface ControlModel<T extends string = string> extends ControlDataModel {
|
export interface ControlModel<T extends string = string> extends ControlDataModel {
|
||||||
/** Unique identifier for the control */
|
/**
|
||||||
|
* Unique string identifier for the control
|
||||||
|
*/
|
||||||
id: T;
|
id: T;
|
||||||
/** ARIA label for the increase button */
|
/**
|
||||||
|
* Label used by screen readers for the increase button
|
||||||
|
*/
|
||||||
increaseLabel?: string;
|
increaseLabel?: string;
|
||||||
/** ARIA label for the decrease button */
|
/**
|
||||||
|
* Label used by screen readers for the decrease button
|
||||||
|
*/
|
||||||
decreaseLabel?: string;
|
decreaseLabel?: string;
|
||||||
/** ARIA label for the control area */
|
/**
|
||||||
|
* Overall label describing the control's purpose
|
||||||
|
*/
|
||||||
controlLabel?: string;
|
controlLabel?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -109,8 +125,7 @@ export function createTypographyControl<T extends ControlDataModel>(
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
/**
|
/**
|
||||||
* Current control value (getter/setter)
|
* Clamped and rounded control value (reactive)
|
||||||
* Setting automatically clamps to bounds and rounds to step precision
|
|
||||||
*/
|
*/
|
||||||
get value() {
|
get value() {
|
||||||
return value;
|
return value;
|
||||||
@@ -122,27 +137,37 @@ export function createTypographyControl<T extends ControlDataModel>(
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
/** Maximum allowed value */
|
/**
|
||||||
|
* Upper limit for the control value
|
||||||
|
*/
|
||||||
get max() {
|
get max() {
|
||||||
return max;
|
return max;
|
||||||
},
|
},
|
||||||
|
|
||||||
/** Minimum allowed value */
|
/**
|
||||||
|
* Lower limit for the control value
|
||||||
|
*/
|
||||||
get min() {
|
get min() {
|
||||||
return min;
|
return min;
|
||||||
},
|
},
|
||||||
|
|
||||||
/** Step increment size */
|
/**
|
||||||
|
* Configured step increment
|
||||||
|
*/
|
||||||
get step() {
|
get step() {
|
||||||
return step;
|
return step;
|
||||||
},
|
},
|
||||||
|
|
||||||
/** Whether the value is at or exceeds the maximum */
|
/**
|
||||||
|
* True if current value is equal to or greater than max
|
||||||
|
*/
|
||||||
get isAtMax() {
|
get isAtMax() {
|
||||||
return isAtMax;
|
return isAtMax;
|
||||||
},
|
},
|
||||||
|
|
||||||
/** Whether the value is at or below the minimum */
|
/**
|
||||||
|
* True if current value is equal to or less than min
|
||||||
|
*/
|
||||||
get isAtMin() {
|
get isAtMin() {
|
||||||
return isAtMin;
|
return isAtMin;
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -45,7 +45,9 @@ export interface VirtualItem {
|
|||||||
* Options are reactive - pass them through a function getter to enable updates.
|
* Options are reactive - pass them through a function getter to enable updates.
|
||||||
*/
|
*/
|
||||||
export interface VirtualizerOptions {
|
export interface VirtualizerOptions {
|
||||||
/** Total number of items in the data array */
|
/**
|
||||||
|
* Total number of items in the underlying data array
|
||||||
|
*/
|
||||||
count: number;
|
count: number;
|
||||||
/**
|
/**
|
||||||
* Function to estimate the size of an item at a given index.
|
* Function to estimate the size of an item at a given index.
|
||||||
@@ -60,7 +62,10 @@ export interface VirtualizerOptions {
|
|||||||
* as fonts finish loading, eliminating the DOM-measurement snap on load.
|
* as fonts finish loading, eliminating the DOM-measurement snap on load.
|
||||||
*/
|
*/
|
||||||
estimateSize: (index: number) => number;
|
estimateSize: (index: number) => number;
|
||||||
/** Number of extra items to render outside viewport for smoother scrolling (default: 5) */
|
/**
|
||||||
|
* Number of extra items to render outside viewport for smoother scrolling
|
||||||
|
* @default 5
|
||||||
|
*/
|
||||||
overscan?: number;
|
overscan?: number;
|
||||||
/**
|
/**
|
||||||
* Function to get the key of an item at a given index.
|
* Function to get the key of an item at a given index.
|
||||||
@@ -464,27 +469,45 @@ export function createVirtualizer<T>(
|
|||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
/**
|
||||||
|
* Current vertical scroll position in pixels (reactive)
|
||||||
|
*/
|
||||||
get scrollOffset() {
|
get scrollOffset() {
|
||||||
return scrollOffset;
|
return scrollOffset;
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Measured height of the visible container area (reactive)
|
||||||
|
*/
|
||||||
get containerHeight() {
|
get containerHeight() {
|
||||||
return containerHeight;
|
return containerHeight;
|
||||||
},
|
},
|
||||||
/** Computed array of visible items to render (reactive) */
|
/**
|
||||||
|
* Computed array of visible items to render (reactive)
|
||||||
|
*/
|
||||||
get items() {
|
get items() {
|
||||||
return items;
|
return items;
|
||||||
},
|
},
|
||||||
/** Total height of all items in pixels (reactive) */
|
/**
|
||||||
|
* Total height of all items in pixels (reactive)
|
||||||
|
*/
|
||||||
get totalSize() {
|
get totalSize() {
|
||||||
return totalSize;
|
return totalSize;
|
||||||
},
|
},
|
||||||
/** Svelte action for the scrollable container element */
|
/**
|
||||||
|
* Svelte action for the scrollable container element
|
||||||
|
*/
|
||||||
container,
|
container,
|
||||||
/** Svelte action for measuring individual item elements */
|
/**
|
||||||
|
* Svelte action for measuring individual item elements
|
||||||
|
*/
|
||||||
measureElement,
|
measureElement,
|
||||||
/** Programmatic scroll method to scroll to a specific item */
|
/**
|
||||||
|
* Programmatic scroll method to scroll to a specific item
|
||||||
|
*/
|
||||||
scrollToIndex,
|
scrollToIndex,
|
||||||
/** Programmatic scroll method to scroll to a specific pixel offset */
|
/**
|
||||||
|
* Programmatic scroll method to scroll to a specific pixel offset
|
||||||
|
*/
|
||||||
scrollToOffset,
|
scrollToOffset,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
/** @vitest-environment jsdom */
|
/**
|
||||||
|
* @vitest-environment jsdom
|
||||||
|
*/
|
||||||
import {
|
import {
|
||||||
afterEach,
|
afterEach,
|
||||||
describe,
|
describe,
|
||||||
|
|||||||
@@ -22,59 +22,178 @@
|
|||||||
* ```
|
* ```
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Filter management
|
||||||
|
*/
|
||||||
export {
|
export {
|
||||||
|
/**
|
||||||
|
* Reactive filter factory
|
||||||
|
*/
|
||||||
createFilter,
|
createFilter,
|
||||||
|
/**
|
||||||
|
* Filter instance type
|
||||||
|
*/
|
||||||
type Filter,
|
type Filter,
|
||||||
|
/**
|
||||||
|
* Initial state model
|
||||||
|
*/
|
||||||
type FilterModel,
|
type FilterModel,
|
||||||
|
/**
|
||||||
|
* Filterable property definition
|
||||||
|
*/
|
||||||
type Property,
|
type Property,
|
||||||
} from './createFilter/createFilter.svelte';
|
} from './createFilter/createFilter.svelte';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Bounded numeric controls
|
||||||
|
*/
|
||||||
export {
|
export {
|
||||||
|
/**
|
||||||
|
* Base numeric configuration
|
||||||
|
*/
|
||||||
type ControlDataModel,
|
type ControlDataModel,
|
||||||
|
/**
|
||||||
|
* Extended model with labels
|
||||||
|
*/
|
||||||
type ControlModel,
|
type ControlModel,
|
||||||
|
/**
|
||||||
|
* Reactive control factory
|
||||||
|
*/
|
||||||
createTypographyControl,
|
createTypographyControl,
|
||||||
|
/**
|
||||||
|
* Control instance type
|
||||||
|
*/
|
||||||
type TypographyControl,
|
type TypographyControl,
|
||||||
} from './createTypographyControl/createTypographyControl.svelte';
|
} from './createTypographyControl/createTypographyControl.svelte';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* List virtualization
|
||||||
|
*/
|
||||||
export {
|
export {
|
||||||
|
/**
|
||||||
|
* Reactive virtualizer factory
|
||||||
|
*/
|
||||||
createVirtualizer,
|
createVirtualizer,
|
||||||
|
/**
|
||||||
|
* Rendered item layout data
|
||||||
|
*/
|
||||||
type VirtualItem,
|
type VirtualItem,
|
||||||
|
/**
|
||||||
|
* Virtualizer instance type
|
||||||
|
*/
|
||||||
type Virtualizer,
|
type Virtualizer,
|
||||||
|
/**
|
||||||
|
* Configuration options
|
||||||
|
*/
|
||||||
type VirtualizerOptions,
|
type VirtualizerOptions,
|
||||||
} from './createVirtualizer/createVirtualizer.svelte';
|
} from './createVirtualizer/createVirtualizer.svelte';
|
||||||
|
|
||||||
export { createDebouncedState } from './createDebouncedState/createDebouncedState.svelte';
|
/**
|
||||||
|
* UI State
|
||||||
|
*/
|
||||||
export {
|
export {
|
||||||
|
/**
|
||||||
|
* Immediate/debounced state factory
|
||||||
|
*/
|
||||||
|
createDebouncedState,
|
||||||
|
} from './createDebouncedState/createDebouncedState.svelte';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Entity collections
|
||||||
|
*/
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Reactive entity store factory
|
||||||
|
*/
|
||||||
createEntityStore,
|
createEntityStore,
|
||||||
|
/**
|
||||||
|
* Base entity requirement
|
||||||
|
*/
|
||||||
type Entity,
|
type Entity,
|
||||||
|
/**
|
||||||
|
* Entity store instance type
|
||||||
|
*/
|
||||||
type EntityStore,
|
type EntityStore,
|
||||||
} from './createEntityStore/createEntityStore.svelte';
|
} from './createEntityStore/createEntityStore.svelte';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Comparison logic
|
||||||
|
*/
|
||||||
export {
|
export {
|
||||||
|
/**
|
||||||
|
* Character-by-character comparison utility
|
||||||
|
*/
|
||||||
CharacterComparisonEngine,
|
CharacterComparisonEngine,
|
||||||
|
/**
|
||||||
|
* Single line of comparison results
|
||||||
|
*/
|
||||||
type ComparisonLine,
|
type ComparisonLine,
|
||||||
|
/**
|
||||||
|
* Full comparison output
|
||||||
|
*/
|
||||||
type ComparisonResult,
|
type ComparisonResult,
|
||||||
} from './CharacterComparisonEngine/CharacterComparisonEngine.svelte';
|
} from './CharacterComparisonEngine/CharacterComparisonEngine.svelte';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Text layout
|
||||||
|
*/
|
||||||
export {
|
export {
|
||||||
|
/**
|
||||||
|
* Single line layout information
|
||||||
|
*/
|
||||||
type LayoutLine as TextLayoutLine,
|
type LayoutLine as TextLayoutLine,
|
||||||
|
/**
|
||||||
|
* Full multi-line layout information
|
||||||
|
*/
|
||||||
type LayoutResult as TextLayoutResult,
|
type LayoutResult as TextLayoutResult,
|
||||||
|
/**
|
||||||
|
* High-level text measurement engine
|
||||||
|
*/
|
||||||
TextLayoutEngine,
|
TextLayoutEngine,
|
||||||
} from './TextLayoutEngine/TextLayoutEngine.svelte';
|
} from './TextLayoutEngine/TextLayoutEngine.svelte';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Persistence
|
||||||
|
*/
|
||||||
export {
|
export {
|
||||||
|
/**
|
||||||
|
* LocalStorage-backed reactive store factory
|
||||||
|
*/
|
||||||
createPersistentStore,
|
createPersistentStore,
|
||||||
|
/**
|
||||||
|
* Persistent store instance type
|
||||||
|
*/
|
||||||
type PersistentStore,
|
type PersistentStore,
|
||||||
} from './createPersistentStore/createPersistentStore.svelte';
|
} from './createPersistentStore/createPersistentStore.svelte';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Responsive design
|
||||||
|
*/
|
||||||
export {
|
export {
|
||||||
|
/**
|
||||||
|
* Breakpoint tracking factory
|
||||||
|
*/
|
||||||
createResponsiveManager,
|
createResponsiveManager,
|
||||||
|
/**
|
||||||
|
* Responsive manager instance type
|
||||||
|
*/
|
||||||
type ResponsiveManager,
|
type ResponsiveManager,
|
||||||
|
/**
|
||||||
|
* Singleton manager for global usage
|
||||||
|
*/
|
||||||
responsiveManager,
|
responsiveManager,
|
||||||
} from './createResponsiveManager/createResponsiveManager.svelte';
|
} from './createResponsiveManager/createResponsiveManager.svelte';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 3D Perspectives
|
||||||
|
*/
|
||||||
export {
|
export {
|
||||||
|
/**
|
||||||
|
* Motion-aware perspective factory
|
||||||
|
*/
|
||||||
createPerspectiveManager,
|
createPerspectiveManager,
|
||||||
|
/**
|
||||||
|
* Perspective manager instance type
|
||||||
|
*/
|
||||||
type PerspectiveManager,
|
type PerspectiveManager,
|
||||||
} from './createPerspectiveManager/createPerspectiveManager.svelte';
|
} from './createPerspectiveManager/createPerspectiveManager.svelte';
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
/** @vitest-environment jsdom */
|
/**
|
||||||
|
* @vitest-environment jsdom
|
||||||
|
*/
|
||||||
import {
|
import {
|
||||||
afterEach,
|
afterEach,
|
||||||
beforeEach,
|
beforeEach,
|
||||||
|
|||||||
@@ -7,8 +7,12 @@
|
|||||||
* @template T - Type of the response data
|
* @template T - Type of the response data
|
||||||
*/
|
*/
|
||||||
export interface ApiResponse<T> {
|
export interface ApiResponse<T> {
|
||||||
/** Response payload data */
|
/**
|
||||||
|
* Primary data payload returned by the server
|
||||||
|
*/
|
||||||
data: T;
|
data: T;
|
||||||
/** HTTP status code */
|
/**
|
||||||
|
* HTTP status code (e.g. 200, 404, 500)
|
||||||
|
*/
|
||||||
status: number;
|
status: number;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,9 @@ import type {
|
|||||||
InputVariant,
|
InputVariant,
|
||||||
} from './types';
|
} from './types';
|
||||||
|
|
||||||
/** Size-specific layout classes: padding, text size, height, and clear-icon pixel size. */
|
/**
|
||||||
|
* Size-specific layout classes: padding, text size, height, and clear-icon pixel size.
|
||||||
|
*/
|
||||||
export const inputSizeConfig: Record<InputSize, { input: string; text: string; height: string; clearIcon: number }> = {
|
export const inputSizeConfig: Record<InputSize, { input: string; text: string; height: string; clearIcon: number }> = {
|
||||||
sm: { input: 'px-3 py-1.5', text: 'text-sm', height: 'h-8', clearIcon: 12 },
|
sm: { input: 'px-3 py-1.5', text: 'text-sm', height: 'h-8', clearIcon: 12 },
|
||||||
md: { input: 'px-4 py-2', text: 'text-base', height: 'h-10', clearIcon: 14 },
|
md: { input: 'px-4 py-2', text: 'text-base', height: 'h-10', clearIcon: 14 },
|
||||||
@@ -11,7 +13,9 @@ export const inputSizeConfig: Record<InputSize, { input: string; text: string; h
|
|||||||
xl: { input: 'px-4 py-3', text: 'text-xl md:text-2xl', height: 'h-14', clearIcon: 18 },
|
xl: { input: 'px-4 py-3', text: 'text-xl md:text-2xl', height: 'h-14', clearIcon: 18 },
|
||||||
};
|
};
|
||||||
|
|
||||||
/** Variant-specific classes: base background/border, focus ring, and error state. */
|
/**
|
||||||
|
* Variant-specific classes: base background/border, focus ring, and error state.
|
||||||
|
*/
|
||||||
export const inputVariantConfig: Record<InputVariant, { base: string; focus: string; error: string }> = {
|
export const inputVariantConfig: Record<InputVariant, { base: string; focus: string; error: string }> = {
|
||||||
default: {
|
default: {
|
||||||
base: 'bg-paper dark:bg-paper border border-subtle',
|
base: 'bg-paper dark:bg-paper border border-subtle',
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
export type InputVariant = 'default' | 'underline' | 'filled';
|
export type InputVariant = 'default' | 'underline' | 'filled';
|
||||||
export type InputSize = 'sm' | 'md' | 'lg' | 'xl';
|
export type InputSize = 'sm' | 'md' | 'lg' | 'xl';
|
||||||
/** Convenience map for consumers sizing icons to match the input. */
|
/**
|
||||||
|
* Convenience map for consumers sizing icons to match the input.
|
||||||
|
*/
|
||||||
export const inputIconSize: Record<InputSize, number> = {
|
export const inputIconSize: Record<InputSize, number> = {
|
||||||
sm: 14,
|
sm: 14,
|
||||||
md: 16,
|
md: 16,
|
||||||
|
|||||||
@@ -1,28 +1,150 @@
|
|||||||
export { default as Badge } from './Badge/Badge.svelte';
|
|
||||||
export {
|
export {
|
||||||
|
/**
|
||||||
|
* Pill-shaped status indicator
|
||||||
|
*/
|
||||||
|
default as Badge,
|
||||||
|
} from './Badge/Badge.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Main action trigger
|
||||||
|
*/
|
||||||
Button,
|
Button,
|
||||||
|
/**
|
||||||
|
* Horizontal layout for related buttons
|
||||||
|
*/
|
||||||
ButtonGroup,
|
ButtonGroup,
|
||||||
|
/**
|
||||||
|
* Button optimized for single-icon display
|
||||||
|
*/
|
||||||
IconButton,
|
IconButton,
|
||||||
|
/**
|
||||||
|
* State-aware toggle switch
|
||||||
|
*/
|
||||||
ToggleButton,
|
ToggleButton,
|
||||||
} from './Button';
|
} from './Button';
|
||||||
export { default as ComboControl } from './ComboControl/ComboControl.svelte';
|
export {
|
||||||
export { default as ContentEditable } from './ContentEditable/ContentEditable.svelte';
|
/**
|
||||||
export { default as ControlGroup } from './ControlGroup/ControlGroup.svelte';
|
* Input with associated increment/decrement controls
|
||||||
export { default as Divider } from './Divider/Divider.svelte';
|
*/
|
||||||
export { default as FilterGroup } from './FilterGroup/FilterGroup.svelte';
|
default as ComboControl,
|
||||||
export { default as Footnote } from './Footnote/Footnote.svelte';
|
} from './ComboControl/ComboControl.svelte';
|
||||||
export { default as Input } from './Input/Input.svelte';
|
export {
|
||||||
export { default as Label } from './Label/Label.svelte';
|
/**
|
||||||
export { default as Loader } from './Loader/Loader.svelte';
|
* Rich text input using contenteditable attribute
|
||||||
export { default as Logo } from './Logo/Logo.svelte';
|
*/
|
||||||
export { default as PerspectivePlan } from './PerspectivePlan/PerspectivePlan.svelte';
|
default as ContentEditable,
|
||||||
export { default as SearchBar } from './SearchBar/SearchBar.svelte';
|
} from './ContentEditable/ContentEditable.svelte';
|
||||||
export { default as Section } from './Section/Section.svelte';
|
export {
|
||||||
export type { TitleStatusChangeHandler } from './Section/types';
|
/**
|
||||||
export { default as SidebarContainer } from './SidebarContainer/SidebarContainer.svelte';
|
* Semantic grouping for related UI controls
|
||||||
export { default as Skeleton } from './Skeleton/Skeleton.svelte';
|
*/
|
||||||
export { default as Slider } from './Slider/Slider.svelte';
|
default as ControlGroup,
|
||||||
export { default as Stat } from './Stat/Stat.svelte';
|
} from './ControlGroup/ControlGroup.svelte';
|
||||||
export { default as StatGroup } from './Stat/StatGroup.svelte';
|
export {
|
||||||
export { default as TechText } from './TechText/TechText.svelte';
|
/**
|
||||||
export { default as VirtualList } from './VirtualList/VirtualList.svelte';
|
* Simple horizontal line separator
|
||||||
|
*/
|
||||||
|
default as Divider,
|
||||||
|
} from './Divider/Divider.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Filterable property set with selection logic
|
||||||
|
*/
|
||||||
|
default as FilterGroup,
|
||||||
|
} from './FilterGroup/FilterGroup.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Small text for secondary meta-information
|
||||||
|
*/
|
||||||
|
default as Footnote,
|
||||||
|
} from './Footnote/Footnote.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Design-system standard text input
|
||||||
|
*/
|
||||||
|
default as Input,
|
||||||
|
} from './Input/Input.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Text label for input fields
|
||||||
|
*/
|
||||||
|
default as Label,
|
||||||
|
} from './Label/Label.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Full-page or component-level progress spinner
|
||||||
|
*/
|
||||||
|
default as Loader,
|
||||||
|
} from './Loader/Loader.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Main application logo
|
||||||
|
*/
|
||||||
|
default as Logo,
|
||||||
|
} from './Logo/Logo.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* 3D perspective background/container
|
||||||
|
*/
|
||||||
|
default as PerspectivePlan,
|
||||||
|
} from './PerspectivePlan/PerspectivePlan.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Specialized input with search icon and clear state
|
||||||
|
*/
|
||||||
|
default as SearchBar,
|
||||||
|
} from './SearchBar/SearchBar.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Content section with header and optional title tracking
|
||||||
|
*/
|
||||||
|
default as Section,
|
||||||
|
} from './Section/Section.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Callback for section visibility status changes
|
||||||
|
*/
|
||||||
|
type TitleStatusChangeHandler,
|
||||||
|
} from './Section/types';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Structural sidebar component
|
||||||
|
*/
|
||||||
|
default as SidebarContainer,
|
||||||
|
} from './SidebarContainer/SidebarContainer.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Loading placeholder with pulsing animation
|
||||||
|
*/
|
||||||
|
default as Skeleton,
|
||||||
|
} from './Skeleton/Skeleton.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Range selector with numeric feedback
|
||||||
|
*/
|
||||||
|
default as Slider,
|
||||||
|
} from './Slider/Slider.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Individual numeric statistic display
|
||||||
|
*/
|
||||||
|
default as Stat,
|
||||||
|
} from './Stat/Stat.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Grouping for multiple statistics
|
||||||
|
*/
|
||||||
|
default as StatGroup,
|
||||||
|
} from './Stat/StatGroup.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* Mono-spaced technical/metadata text
|
||||||
|
*/
|
||||||
|
default as TechText,
|
||||||
|
} from './TechText/TechText.svelte';
|
||||||
|
export {
|
||||||
|
/**
|
||||||
|
* High-performance list renderer for large datasets
|
||||||
|
*/
|
||||||
|
default as VirtualList,
|
||||||
|
} from './VirtualList/VirtualList.svelte';
|
||||||
|
|||||||
@@ -29,9 +29,13 @@ import { untrack } from 'svelte';
|
|||||||
* Storage schema for comparison state
|
* Storage schema for comparison state
|
||||||
*/
|
*/
|
||||||
interface ComparisonState {
|
interface ComparisonState {
|
||||||
/** Font ID for side A (left/top) */
|
/**
|
||||||
|
* Unique identifier for the primary font being compared (Side A)
|
||||||
|
*/
|
||||||
fontAId: string | null;
|
fontAId: string | null;
|
||||||
/** Font ID for side B (right/bottom) */
|
/**
|
||||||
|
* Unique identifier for the secondary font being compared (Side B)
|
||||||
|
*/
|
||||||
fontBId: string | null;
|
fontBId: string | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -53,21 +57,33 @@ const storage = createPersistentStore<ComparisonState>('glyphdiff:comparison', {
|
|||||||
* storage is empty.
|
* storage is empty.
|
||||||
*/
|
*/
|
||||||
export class ComparisonStore {
|
export class ComparisonStore {
|
||||||
/** Font for side A */
|
/**
|
||||||
|
* The primary font model for Side A (left/top)
|
||||||
|
*/
|
||||||
#fontA = $state<UnifiedFont | undefined>();
|
#fontA = $state<UnifiedFont | undefined>();
|
||||||
/** Font for side B */
|
/**
|
||||||
|
* The secondary font model for Side B (right/bottom)
|
||||||
|
*/
|
||||||
#fontB = $state<UnifiedFont | undefined>();
|
#fontB = $state<UnifiedFont | undefined>();
|
||||||
/** Sample text to display */
|
/**
|
||||||
|
* The preview text string displayed in the comparison area
|
||||||
|
*/
|
||||||
#sampleText = $state('The quick brown fox jumps over the lazy dog');
|
#sampleText = $state('The quick brown fox jumps over the lazy dog');
|
||||||
/** Whether fonts are loaded and ready to display */
|
/**
|
||||||
|
* Flag indicating if both fonts are successfully loaded and ready for rendering
|
||||||
|
*/
|
||||||
#fontsReady = $state(false);
|
#fontsReady = $state(false);
|
||||||
/** Active side for single-font operations */
|
/**
|
||||||
|
* Currently active side (A or B) for single-font adjustments
|
||||||
|
*/
|
||||||
#side = $state<Side>('A');
|
#side = $state<Side>('A');
|
||||||
/** Slider position for character morphing (0-100) */
|
/**
|
||||||
|
* Interactive slider position (0-100) used for morphing/layout transitions
|
||||||
|
*/
|
||||||
#sliderPosition = $state(50);
|
#sliderPosition = $state(50);
|
||||||
// /** Typography controls for this comparison */
|
/**
|
||||||
// #typography = createTypographyControlManager(DEFAULT_TYPOGRAPHY_CONTROLS_DATA, 'glyphdiff:comparison:typography');
|
* TanStack Query-backed store for efficient batch font retrieval
|
||||||
/** TanStack Query-backed batch font fetcher */
|
*/
|
||||||
#batchStore: BatchFontStore;
|
#batchStore: BatchFontStore;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
@@ -211,14 +227,9 @@ export class ComparisonStore {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// // ── Getters / Setters ─────────────────────────────────────────────────────
|
/**
|
||||||
|
* Primary font for comparison (reactive)
|
||||||
// /** Typography control manager */
|
*/
|
||||||
// get typography() {
|
|
||||||
// return typographySettingsStore;
|
|
||||||
// }
|
|
||||||
|
|
||||||
/** Font for side A */
|
|
||||||
get fontA() {
|
get fontA() {
|
||||||
return this.#fontA;
|
return this.#fontA;
|
||||||
}
|
}
|
||||||
@@ -228,7 +239,9 @@ export class ComparisonStore {
|
|||||||
this.updateStorage();
|
this.updateStorage();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Font for side B */
|
/**
|
||||||
|
* Secondary font for comparison (reactive)
|
||||||
|
*/
|
||||||
get fontB() {
|
get fontB() {
|
||||||
return this.#fontB;
|
return this.#fontB;
|
||||||
}
|
}
|
||||||
@@ -238,7 +251,9 @@ export class ComparisonStore {
|
|||||||
this.updateStorage();
|
this.updateStorage();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Sample text to display */
|
/**
|
||||||
|
* Shared preview text string (reactive)
|
||||||
|
*/
|
||||||
get text() {
|
get text() {
|
||||||
return this.#sampleText;
|
return this.#sampleText;
|
||||||
}
|
}
|
||||||
@@ -247,7 +262,9 @@ export class ComparisonStore {
|
|||||||
this.#sampleText = value;
|
this.#sampleText = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Active side for single-font operations */
|
/**
|
||||||
|
* Side currently selected for focused manipulation (reactive)
|
||||||
|
*/
|
||||||
get side() {
|
get side() {
|
||||||
return this.#side;
|
return this.#side;
|
||||||
}
|
}
|
||||||
@@ -256,7 +273,9 @@ export class ComparisonStore {
|
|||||||
this.#side = value;
|
this.#side = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Slider position (0-100) for character morphing */
|
/**
|
||||||
|
* Morphing slider position (0-100) used by Character components (reactive)
|
||||||
|
*/
|
||||||
get sliderPosition() {
|
get sliderPosition() {
|
||||||
return this.#sliderPosition;
|
return this.#sliderPosition;
|
||||||
}
|
}
|
||||||
@@ -265,12 +284,16 @@ export class ComparisonStore {
|
|||||||
this.#sliderPosition = value;
|
this.#sliderPosition = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Whether both fonts are selected and loaded */
|
/**
|
||||||
|
* True if both fonts are ready for side-by-side display (reactive)
|
||||||
|
*/
|
||||||
get isReady() {
|
get isReady() {
|
||||||
return !!this.#fontA && !!this.#fontB && this.#fontsReady;
|
return !!this.#fontA && !!this.#fontB && this.#fontsReady;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Whether currently loading (batch fetch in flight or fonts not yet painted) */
|
/**
|
||||||
|
* True if any font is currently being fetched or loaded (reactive)
|
||||||
|
*/
|
||||||
get isLoading() {
|
get isLoading() {
|
||||||
return this.#batchStore.isLoading || !this.#fontsReady;
|
return this.#batchStore.isLoading || !this.#fontsReady;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,7 +5,9 @@
|
|||||||
* Controls network behaviour via vi.spyOn on the proxyFonts API layer.
|
* Controls network behaviour via vi.spyOn on the proxyFonts API layer.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/** @vitest-environment jsdom */
|
/**
|
||||||
|
* @vitest-environment jsdom
|
||||||
|
*/
|
||||||
|
|
||||||
import type { UnifiedFont } from '$entities/Font';
|
import type { UnifiedFont } from '$entities/Font';
|
||||||
import { UNIFIED_FONTS } from '$entities/Font/lib/mocks';
|
import { UNIFIED_FONTS } from '$entities/Font/lib/mocks';
|
||||||
@@ -18,8 +20,6 @@ import {
|
|||||||
vi,
|
vi,
|
||||||
} from 'vitest';
|
} from 'vitest';
|
||||||
|
|
||||||
// ── Persistent-store mock ─────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
const mockStorage = vi.hoisted(() => {
|
const mockStorage = vi.hoisted(() => {
|
||||||
const storage: any = {};
|
const storage: any = {};
|
||||||
storage._value = { fontAId: null, fontBId: null };
|
storage._value = { fontAId: null, fontBId: null };
|
||||||
@@ -51,8 +51,6 @@ vi.mock('$shared/lib/helpers/createPersistentStore/createPersistentStore.svelte'
|
|||||||
createPersistentStore: vi.fn(() => mockStorage),
|
createPersistentStore: vi.fn(() => mockStorage),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// ── $entities/Font mock — keep real BatchFontStore, stub singletons ───────────
|
|
||||||
|
|
||||||
vi.mock('$entities/Font', async importOriginal => {
|
vi.mock('$entities/Font', async importOriginal => {
|
||||||
const actual = await importOriginal<typeof import('$entities/Font')>();
|
const actual = await importOriginal<typeof import('$entities/Font')>();
|
||||||
const { BatchFontStore } = await import(
|
const { BatchFontStore } = await import(
|
||||||
@@ -73,8 +71,6 @@ vi.mock('$entities/Font', async importOriginal => {
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── $features/SetupFont mock ──────────────────────────────────────────────────
|
|
||||||
|
|
||||||
vi.mock('$features/SetupFont', () => ({
|
vi.mock('$features/SetupFont', () => ({
|
||||||
DEFAULT_TYPOGRAPHY_CONTROLS_DATA: [],
|
DEFAULT_TYPOGRAPHY_CONTROLS_DATA: [],
|
||||||
createTypographyControlManager: vi.fn(() => ({
|
createTypographyControlManager: vi.fn(() => ({
|
||||||
@@ -92,8 +88,6 @@ vi.mock('$features/SetupFont/model', () => ({
|
|||||||
},
|
},
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// ── Imports (after mocks) ─────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
import {
|
import {
|
||||||
appliedFontsManager,
|
appliedFontsManager,
|
||||||
fontStore,
|
fontStore,
|
||||||
@@ -101,8 +95,6 @@ import {
|
|||||||
import * as proxyFonts from '$entities/Font/api/proxy/proxyFonts';
|
import * as proxyFonts from '$entities/Font/api/proxy/proxyFonts';
|
||||||
import { ComparisonStore } from './comparisonStore.svelte';
|
import { ComparisonStore } from './comparisonStore.svelte';
|
||||||
|
|
||||||
// ── Tests ─────────────────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('ComparisonStore', () => {
|
describe('ComparisonStore', () => {
|
||||||
const mockFontA: UnifiedFont = UNIFIED_FONTS.roboto; // id: 'roboto'
|
const mockFontA: UnifiedFont = UNIFIED_FONTS.roboto; // id: 'roboto'
|
||||||
const mockFontB: UnifiedFont = UNIFIED_FONTS.openSans; // id: 'open-sans'
|
const mockFontB: UnifiedFont = UNIFIED_FONTS.openSans; // id: 'open-sans'
|
||||||
@@ -129,8 +121,6 @@ describe('ComparisonStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── Initialization ────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('Initialization', () => {
|
describe('Initialization', () => {
|
||||||
it('should create store with initial empty state', () => {
|
it('should create store with initial empty state', () => {
|
||||||
const store = new ComparisonStore();
|
const store = new ComparisonStore();
|
||||||
@@ -139,8 +129,6 @@ describe('ComparisonStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── Restoration from Storage ──────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('Restoration from Storage (via BatchFontStore)', () => {
|
describe('Restoration from Storage (via BatchFontStore)', () => {
|
||||||
it('should restore fontA and fontB from stored IDs', async () => {
|
it('should restore fontA and fontB from stored IDs', async () => {
|
||||||
mockStorage._value.fontAId = mockFontA.id;
|
mockStorage._value.fontAId = mockFontA.id;
|
||||||
@@ -169,8 +157,6 @@ describe('ComparisonStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── Default Fallbacks ─────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('Default Fallbacks', () => {
|
describe('Default Fallbacks', () => {
|
||||||
it('should update storage with default IDs when storage is empty', async () => {
|
it('should update storage with default IDs when storage is empty', async () => {
|
||||||
(fontStore as any).fonts = [mockFontA, mockFontB];
|
(fontStore as any).fonts = [mockFontA, mockFontB];
|
||||||
@@ -185,8 +171,6 @@ describe('ComparisonStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── Loading State ─────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('Aggregate Loading State', () => {
|
describe('Aggregate Loading State', () => {
|
||||||
it('should be loading initially when storage has IDs', async () => {
|
it('should be loading initially when storage has IDs', async () => {
|
||||||
mockStorage._value.fontAId = mockFontA.id;
|
mockStorage._value.fontAId = mockFontA.id;
|
||||||
@@ -202,8 +186,6 @@ describe('ComparisonStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── Reset ─────────────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('Reset Functionality', () => {
|
describe('Reset Functionality', () => {
|
||||||
it('should reset all state and clear storage', () => {
|
it('should reset all state and clear storage', () => {
|
||||||
const store = new ComparisonStore();
|
const store = new ComparisonStore();
|
||||||
@@ -225,8 +207,6 @@ describe('ComparisonStore', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ── Pin / Unpin ───────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
describe('Pin / Unpin (eviction guard)', () => {
|
describe('Pin / Unpin (eviction guard)', () => {
|
||||||
it('pins fontA and fontB when they are loaded', async () => {
|
it('pins fontA and fontB when they are loaded', async () => {
|
||||||
mockStorage._value.fontAId = mockFontA.id;
|
mockStorage._value.fontAId = mockFontA.id;
|
||||||
|
|||||||
@@ -1,3 +1,8 @@
|
|||||||
|
<!--
|
||||||
|
Component: Search
|
||||||
|
Typeface search input for the comparison view.
|
||||||
|
Updates the global filterManager query to filter the font list.
|
||||||
|
-->
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { filterManager } from '$features/GetFonts';
|
import { filterManager } from '$features/GetFonts';
|
||||||
import { SearchBar } from '$shared/ui';
|
import { SearchBar } from '$shared/ui';
|
||||||
|
|||||||
@@ -22,6 +22,9 @@ import { responsiveManager } from '$shared/lib';
|
|||||||
export type LayoutMode = 'list' | 'grid';
|
export type LayoutMode = 'list' | 'grid';
|
||||||
|
|
||||||
interface LayoutConfig {
|
interface LayoutConfig {
|
||||||
|
/**
|
||||||
|
* Active display mode (list | grid)
|
||||||
|
*/
|
||||||
mode: LayoutMode;
|
mode: LayoutMode;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -40,9 +43,13 @@ const DEFAULT_CONFIG: LayoutConfig = {
|
|||||||
* calculation. Persists user preference to localStorage.
|
* calculation. Persists user preference to localStorage.
|
||||||
*/
|
*/
|
||||||
class LayoutManager {
|
class LayoutManager {
|
||||||
/** Current layout mode */
|
/**
|
||||||
|
* Reactive layout mode state
|
||||||
|
*/
|
||||||
#mode = $state<LayoutMode>(DEFAULT_CONFIG.mode);
|
#mode = $state<LayoutMode>(DEFAULT_CONFIG.mode);
|
||||||
/** Persistent storage for layout preference */
|
/**
|
||||||
|
* Persistence layer for saving layout between sessions
|
||||||
|
*/
|
||||||
#store = createPersistentStore<LayoutConfig>(STORAGE_KEY, DEFAULT_CONFIG);
|
#store = createPersistentStore<LayoutConfig>(STORAGE_KEY, DEFAULT_CONFIG);
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
@@ -53,7 +60,9 @@ class LayoutManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Current layout mode ('list' or 'grid') */
|
/**
|
||||||
|
* Current active layout mode
|
||||||
|
*/
|
||||||
get mode(): LayoutMode {
|
get mode(): LayoutMode {
|
||||||
return this.#mode;
|
return this.#mode;
|
||||||
}
|
}
|
||||||
@@ -66,12 +75,16 @@ class LayoutManager {
|
|||||||
return responsiveManager.isMobile || responsiveManager.isTabletPortrait ? SM_GAP_PX : MD_GAP_PX;
|
return responsiveManager.isMobile || responsiveManager.isTabletPortrait ? SM_GAP_PX : MD_GAP_PX;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Whether currently in list mode */
|
/**
|
||||||
|
* True if currently showing a single-column list
|
||||||
|
*/
|
||||||
get isListMode(): boolean {
|
get isListMode(): boolean {
|
||||||
return this.#mode === 'list';
|
return this.#mode === 'list';
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Whether currently in grid mode */
|
/**
|
||||||
|
* True if currently showing a multi-column grid
|
||||||
|
*/
|
||||||
get isGridMode(): boolean {
|
get isGridMode(): boolean {
|
||||||
return this.#mode === 'grid';
|
return this.#mode === 'grid';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
/** @vitest-environment jsdom */
|
/**
|
||||||
|
* @vitest-environment jsdom
|
||||||
|
*/
|
||||||
|
|
||||||
import {
|
import {
|
||||||
afterEach,
|
afterEach,
|
||||||
|
|||||||
Reference in New Issue
Block a user