@chuzi/shared 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +52 -0
- package/dist/api/index.d.ts +122 -0
- package/dist/api/index.js +108 -0
- package/dist/api/index.js.map +1 -0
- package/dist/config/index.d.ts +39 -0
- package/dist/config/index.js +404 -0
- package/dist/config/index.js.map +1 -0
- package/dist/index.d.ts +6 -0
- package/dist/index.js +627 -0
- package/dist/index.js.map +1 -0
- package/dist/input/index.d.ts +70 -0
- package/dist/input/index.js +41 -0
- package/dist/input/index.js.map +1 -0
- package/dist/realms/cosmos/components/index.d.ts +68 -0
- package/dist/realms/cosmos/components/index.js +172 -0
- package/dist/realms/cosmos/components/index.js.map +1 -0
- package/dist/realms/cosmos/index.d.ts +8 -0
- package/dist/realms/cosmos/index.js +76 -0
- package/dist/realms/cosmos/index.js.map +1 -0
- package/dist/realms/index.d.ts +109 -0
- package/dist/realms/index.js +23 -0
- package/dist/realms/index.js.map +1 -0
- package/dist/realms/wilds/components/index.d.ts +88 -0
- package/dist/realms/wilds/components/index.js +359 -0
- package/dist/realms/wilds/components/index.js.map +1 -0
- package/dist/realms/wilds/index.d.ts +8 -0
- package/dist/realms/wilds/index.js +70 -0
- package/dist/realms/wilds/index.js.map +1 -0
- package/dist/themes/index.d.ts +46 -0
- package/dist/themes/index.js +63 -0
- package/dist/themes/index.js.map +1 -0
- package/dist/types/index.d.ts +292 -0
- package/dist/types/index.js +3 -0
- package/dist/types/index.js.map +1 -0
- package/dist/ui/index.d.ts +71 -0
- package/dist/ui/index.js +551 -0
- package/dist/ui/index.js.map +1 -0
- package/package.json +107 -0
package/dist/ui/index.js
ADDED
|
@@ -0,0 +1,551 @@
|
|
|
1
|
+
import { createContext, useMemo, useContext } from 'react';
|
|
2
|
+
import { jsx, jsxs, Fragment } from 'react/jsx-runtime';
|
|
3
|
+
import { StyleSheet, Pressable, Text, View } from 'react-native';
|
|
4
|
+
|
|
5
|
+
// src/ui/theme/RealmThemeProvider.tsx
|
|
6
|
+
|
|
7
|
+
// src/config/index.ts
|
|
8
|
+
var REALMS = {
|
|
9
|
+
cosmos: {
|
|
10
|
+
label: "CHUZI COSMOS",
|
|
11
|
+
short_label: "Cosmos",
|
|
12
|
+
lexicon: {
|
|
13
|
+
genre: "Galaxy",
|
|
14
|
+
story: "Star System",
|
|
15
|
+
scene: "Planet",
|
|
16
|
+
scene_choice: "Trajectory",
|
|
17
|
+
node: "Signal",
|
|
18
|
+
node_type_choice: "Trajectory",
|
|
19
|
+
node_type_media: "Transmission",
|
|
20
|
+
node_type_go_to_scene: "Warp",
|
|
21
|
+
player: "Voyager",
|
|
22
|
+
library: "Star Chart",
|
|
23
|
+
library_open: "Open Star Chart",
|
|
24
|
+
library_close: "Close Star Chart",
|
|
25
|
+
editor: "Mission Control",
|
|
26
|
+
publish: "Launch",
|
|
27
|
+
the_end: "Final Orbit",
|
|
28
|
+
media_pool: "Cargo Bay",
|
|
29
|
+
scene_graph: "Star Chart",
|
|
30
|
+
watch_path: "Your trajectory",
|
|
31
|
+
untitled_story: "Untitled Star System",
|
|
32
|
+
unknown_genre: "Unknown Galaxy",
|
|
33
|
+
scenes_count: "planets",
|
|
34
|
+
choices_count: "trajectories",
|
|
35
|
+
drafts_published: "Drafts + Launched",
|
|
36
|
+
published_ver: "Launched ver.",
|
|
37
|
+
draft: "Draft",
|
|
38
|
+
no_scenes_yet: "No planets in this chart yet.",
|
|
39
|
+
no_path_yet: "No trajectory to show yet.",
|
|
40
|
+
menu_empty_drafts: "No star systems yet. Chart one to begin.",
|
|
41
|
+
menu_empty_published: "No launched systems in the chart.",
|
|
42
|
+
menu_published_heading: "Launched constellations",
|
|
43
|
+
menu_create_story: "New star system",
|
|
44
|
+
menu_create_film: "Create film",
|
|
45
|
+
film_title_placeholder: "Star system title",
|
|
46
|
+
select_genre: "Select galaxy",
|
|
47
|
+
genre_field_aria: "Galaxy",
|
|
48
|
+
delete_story_verb: "Delete star system",
|
|
49
|
+
title_scene_default: "Title planet",
|
|
50
|
+
choices_made_stat: "trajectories taken",
|
|
51
|
+
content_rating: "Audience Class",
|
|
52
|
+
content_rating_field_aria: "Audience class",
|
|
53
|
+
select_content_rating: "Select audience class",
|
|
54
|
+
unrated: "Unclassified",
|
|
55
|
+
viewer_credits: "Fuel Cells",
|
|
56
|
+
creator_credits: "Stardust"
|
|
57
|
+
},
|
|
58
|
+
locales: {
|
|
59
|
+
es: {
|
|
60
|
+
genre: "Galaxia",
|
|
61
|
+
story: "Sistema Estelar",
|
|
62
|
+
scene: "Planeta",
|
|
63
|
+
scene_choice: "Trayectoria",
|
|
64
|
+
player: "Viajero",
|
|
65
|
+
library: "Carta Estelar",
|
|
66
|
+
editor: "Control de Misi\xF3n",
|
|
67
|
+
publish: "Lanzar",
|
|
68
|
+
the_end: "\xD3rbita Final",
|
|
69
|
+
untitled_story: "Sistema Estelar sin t\xEDtulo",
|
|
70
|
+
menu_create_story: "Nuevo sistema estelar",
|
|
71
|
+
film_title_placeholder: "T\xEDtulo del sistema estelar",
|
|
72
|
+
select_genre: "Seleccionar galaxia"
|
|
73
|
+
},
|
|
74
|
+
fr: {
|
|
75
|
+
genre: "Galaxie",
|
|
76
|
+
story: "Syst\xE8me Stellaire",
|
|
77
|
+
scene: "Plan\xE8te",
|
|
78
|
+
scene_choice: "Trajectoire",
|
|
79
|
+
player: "Voyageur",
|
|
80
|
+
library: "Carte Stellaire",
|
|
81
|
+
editor: "Contr\xF4le de Mission",
|
|
82
|
+
publish: "Lancer",
|
|
83
|
+
the_end: "Orbite Finale",
|
|
84
|
+
untitled_story: "Syst\xE8me stellaire sans titre",
|
|
85
|
+
menu_create_story: "Nouveau syst\xE8me stellaire",
|
|
86
|
+
film_title_placeholder: "Titre du syst\xE8me stellaire",
|
|
87
|
+
select_genre: "S\xE9lectionner une galaxie"
|
|
88
|
+
},
|
|
89
|
+
de: {
|
|
90
|
+
genre: "Galaxie",
|
|
91
|
+
story: "Sternsystem",
|
|
92
|
+
scene: "Planet",
|
|
93
|
+
player: "Reisender",
|
|
94
|
+
publish: "Starten"
|
|
95
|
+
},
|
|
96
|
+
pt: {
|
|
97
|
+
genre: "Gal\xE1xia",
|
|
98
|
+
story: "Sistema Estelar",
|
|
99
|
+
scene: "Planeta",
|
|
100
|
+
player: "Viajante",
|
|
101
|
+
publish: "Lan\xE7ar"
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
},
|
|
105
|
+
wilds: {
|
|
106
|
+
label: "CHUZI WILDS",
|
|
107
|
+
short_label: "Wilds",
|
|
108
|
+
lexicon: {
|
|
109
|
+
genre: "Biome",
|
|
110
|
+
story: "Grove",
|
|
111
|
+
scene: "Clearing",
|
|
112
|
+
scene_choice: "Trail",
|
|
113
|
+
node: "Seed",
|
|
114
|
+
node_type_choice: "Trail",
|
|
115
|
+
node_type_media: "Bloom",
|
|
116
|
+
node_type_go_to_scene: "Crossing",
|
|
117
|
+
player: "Wanderer",
|
|
118
|
+
library: "Canopy",
|
|
119
|
+
library_open: "Open Canopy",
|
|
120
|
+
library_close: "Close Canopy",
|
|
121
|
+
editor: "Heartwood",
|
|
122
|
+
publish: "Plant",
|
|
123
|
+
the_end: "Roots",
|
|
124
|
+
media_pool: "Undergrowth",
|
|
125
|
+
scene_graph: "Canopy",
|
|
126
|
+
watch_path: "Your trail",
|
|
127
|
+
untitled_story: "Untitled Grove",
|
|
128
|
+
unknown_genre: "Unknown Biome",
|
|
129
|
+
scenes_count: "clearings",
|
|
130
|
+
choices_count: "trails",
|
|
131
|
+
drafts_published: "Drafts + Planted",
|
|
132
|
+
published_ver: "Planted ver.",
|
|
133
|
+
draft: "Draft",
|
|
134
|
+
no_scenes_yet: "No clearings in this canopy yet.",
|
|
135
|
+
no_path_yet: "No trail to show yet.",
|
|
136
|
+
menu_empty_drafts: "No groves yet. Plant one to begin.",
|
|
137
|
+
menu_empty_published: "No planted groves in the canopy.",
|
|
138
|
+
menu_published_heading: "Planted groves",
|
|
139
|
+
menu_create_story: "New grove",
|
|
140
|
+
menu_create_film: "Create film",
|
|
141
|
+
film_title_placeholder: "Grove title",
|
|
142
|
+
select_genre: "Select biome",
|
|
143
|
+
genre_field_aria: "Biome",
|
|
144
|
+
delete_story_verb: "Delete grove",
|
|
145
|
+
title_scene_default: "Title clearing",
|
|
146
|
+
choices_made_stat: "trails taken",
|
|
147
|
+
content_rating: "Field Guide",
|
|
148
|
+
content_rating_field_aria: "Field guide",
|
|
149
|
+
select_content_rating: "Select field guide",
|
|
150
|
+
unrated: "Unmarked",
|
|
151
|
+
viewer_credits: "Sap",
|
|
152
|
+
creator_credits: "Pollen"
|
|
153
|
+
},
|
|
154
|
+
locales: {
|
|
155
|
+
es: {
|
|
156
|
+
genre: "Bioma",
|
|
157
|
+
story: "Arboleda",
|
|
158
|
+
scene: "Claro",
|
|
159
|
+
scene_choice: "Sendero",
|
|
160
|
+
player: "Errante",
|
|
161
|
+
library: "Dosel",
|
|
162
|
+
editor: "Duramen",
|
|
163
|
+
publish: "Plantar",
|
|
164
|
+
the_end: "Ra\xEDces",
|
|
165
|
+
untitled_story: "Arboleda sin t\xEDtulo",
|
|
166
|
+
menu_create_story: "Nueva arboleda",
|
|
167
|
+
film_title_placeholder: "T\xEDtulo de la arboleda",
|
|
168
|
+
select_genre: "Seleccionar bioma"
|
|
169
|
+
},
|
|
170
|
+
fr: {
|
|
171
|
+
genre: "Biome",
|
|
172
|
+
story: "Bosquet",
|
|
173
|
+
scene: "Clairi\xE8re",
|
|
174
|
+
scene_choice: "Sentier",
|
|
175
|
+
player: "Vagabond",
|
|
176
|
+
library: "Canop\xE9e",
|
|
177
|
+
editor: "C\u0153ur de bois",
|
|
178
|
+
publish: "Planter",
|
|
179
|
+
the_end: "Racines",
|
|
180
|
+
untitled_story: "Bosquet sans titre",
|
|
181
|
+
menu_create_story: "Nouveau bosquet",
|
|
182
|
+
film_title_placeholder: "Titre du bosquet",
|
|
183
|
+
select_genre: "S\xE9lectionner un biome"
|
|
184
|
+
},
|
|
185
|
+
de: {
|
|
186
|
+
genre: "Biom",
|
|
187
|
+
story: "Hain",
|
|
188
|
+
scene: "Lichtung",
|
|
189
|
+
player: "Wanderer",
|
|
190
|
+
publish: "Pflanzen"
|
|
191
|
+
},
|
|
192
|
+
pt: {
|
|
193
|
+
genre: "Bioma",
|
|
194
|
+
story: "Bosque",
|
|
195
|
+
scene: "Clareira",
|
|
196
|
+
player: "Andarilho",
|
|
197
|
+
publish: "Plantar"
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
};
|
|
202
|
+
var FALLBACK_LEXICON = {
|
|
203
|
+
genre: "Genre",
|
|
204
|
+
story: "Story",
|
|
205
|
+
scene: "Scene",
|
|
206
|
+
scene_choice: "Choice",
|
|
207
|
+
node: "Node",
|
|
208
|
+
node_type_choice: "Choice",
|
|
209
|
+
node_type_media: "Media",
|
|
210
|
+
node_type_go_to_scene: "Go to scene",
|
|
211
|
+
player: "Player",
|
|
212
|
+
library: "Library",
|
|
213
|
+
library_open: "Open scene tree",
|
|
214
|
+
library_close: "Close scene tree",
|
|
215
|
+
editor: "Editor",
|
|
216
|
+
publish: "Publish",
|
|
217
|
+
the_end: "The End",
|
|
218
|
+
media_pool: "Media Pool",
|
|
219
|
+
scene_graph: "Scene tree",
|
|
220
|
+
watch_path: "Your path",
|
|
221
|
+
untitled_story: "Untitled Story",
|
|
222
|
+
unknown_genre: "Unknown Genre",
|
|
223
|
+
scenes_count: "scenes",
|
|
224
|
+
choices_count: "choices",
|
|
225
|
+
drafts_published: "Drafts + Published",
|
|
226
|
+
published_ver: "Published ver.",
|
|
227
|
+
draft: "Draft",
|
|
228
|
+
no_scenes_yet: "No scenes available yet.",
|
|
229
|
+
no_path_yet: "No path to show yet.",
|
|
230
|
+
menu_empty_drafts: "No films yet. Create one to get started.",
|
|
231
|
+
menu_empty_published: "No published films found.",
|
|
232
|
+
menu_published_heading: "Published Films",
|
|
233
|
+
menu_create_story: "Create Story",
|
|
234
|
+
menu_create_film: "Create Film",
|
|
235
|
+
film_title_placeholder: "Film title",
|
|
236
|
+
select_genre: "Select genre",
|
|
237
|
+
genre_field_aria: "Film genre",
|
|
238
|
+
delete_story_verb: "Delete film",
|
|
239
|
+
title_scene_default: "Title Scene",
|
|
240
|
+
choices_made_stat: "choices made",
|
|
241
|
+
content_rating: "Rating",
|
|
242
|
+
content_rating_field_aria: "Content rating",
|
|
243
|
+
select_content_rating: "Select rating",
|
|
244
|
+
unrated: "Not Rated",
|
|
245
|
+
viewer_credits: "Viewer Credits",
|
|
246
|
+
creator_credits: "Creator Credits"
|
|
247
|
+
};
|
|
248
|
+
var FALLBACK_LOCALES = {
|
|
249
|
+
es: {
|
|
250
|
+
genre: "G\xE9nero",
|
|
251
|
+
story: "Historia",
|
|
252
|
+
scene: "Escena",
|
|
253
|
+
scene_choice: "Elecci\xF3n",
|
|
254
|
+
player: "Reproductor",
|
|
255
|
+
library: "Biblioteca",
|
|
256
|
+
editor: "Editor",
|
|
257
|
+
publish: "Publicar",
|
|
258
|
+
the_end: "Fin",
|
|
259
|
+
untitled_story: "Historia sin t\xEDtulo",
|
|
260
|
+
menu_create_story: "Crear historia",
|
|
261
|
+
menu_create_film: "Crear pel\xEDcula",
|
|
262
|
+
film_title_placeholder: "T\xEDtulo de la pel\xEDcula",
|
|
263
|
+
select_genre: "Seleccionar g\xE9nero",
|
|
264
|
+
delete_story_verb: "Eliminar pel\xEDcula"
|
|
265
|
+
},
|
|
266
|
+
fr: {
|
|
267
|
+
genre: "Genre",
|
|
268
|
+
story: "Histoire",
|
|
269
|
+
scene: "Sc\xE8ne",
|
|
270
|
+
scene_choice: "Choix",
|
|
271
|
+
player: "Lecteur",
|
|
272
|
+
library: "Biblioth\xE8que",
|
|
273
|
+
editor: "\xC9diteur",
|
|
274
|
+
publish: "Publier",
|
|
275
|
+
the_end: "Fin",
|
|
276
|
+
untitled_story: "Histoire sans titre",
|
|
277
|
+
menu_create_story: "Cr\xE9er une histoire",
|
|
278
|
+
menu_create_film: "Cr\xE9er un film",
|
|
279
|
+
film_title_placeholder: "Titre du film",
|
|
280
|
+
select_genre: "S\xE9lectionner un genre",
|
|
281
|
+
delete_story_verb: "Supprimer le film"
|
|
282
|
+
},
|
|
283
|
+
de: {
|
|
284
|
+
genre: "Genre",
|
|
285
|
+
story: "Geschichte",
|
|
286
|
+
scene: "Szene",
|
|
287
|
+
publish: "Ver\xF6ffentlichen",
|
|
288
|
+
the_end: "Ende",
|
|
289
|
+
untitled_story: "Unbenannte Geschichte",
|
|
290
|
+
menu_create_story: "Geschichte erstellen",
|
|
291
|
+
film_title_placeholder: "Filmtitel",
|
|
292
|
+
select_genre: "Genre ausw\xE4hlen"
|
|
293
|
+
},
|
|
294
|
+
pt: {
|
|
295
|
+
genre: "G\xEAnero",
|
|
296
|
+
story: "Hist\xF3ria",
|
|
297
|
+
scene: "Cena",
|
|
298
|
+
publish: "Publicar",
|
|
299
|
+
the_end: "Fim",
|
|
300
|
+
untitled_story: "Hist\xF3ria sem t\xEDtulo",
|
|
301
|
+
menu_create_story: "Criar hist\xF3ria",
|
|
302
|
+
film_title_placeholder: "T\xEDtulo do filme",
|
|
303
|
+
select_genre: "Selecionar g\xEAnero"
|
|
304
|
+
}
|
|
305
|
+
};
|
|
306
|
+
function resolveFallback(locale) {
|
|
307
|
+
const overrides = locale && FALLBACK_LOCALES[locale];
|
|
308
|
+
return overrides ? { ...FALLBACK_LEXICON, ...overrides } : { ...FALLBACK_LEXICON };
|
|
309
|
+
}
|
|
310
|
+
function resolveRealm(realmId, locale) {
|
|
311
|
+
const realm = REALMS[realmId];
|
|
312
|
+
const overrides = locale && realm.locales ? realm.locales[locale] : void 0;
|
|
313
|
+
return overrides ? { ...realm.lexicon, ...overrides } : { ...realm.lexicon };
|
|
314
|
+
}
|
|
315
|
+
function t(realmId, key, fallback = "", locale = null) {
|
|
316
|
+
if (realmId && REALMS[realmId]) {
|
|
317
|
+
const realmLex = resolveRealm(realmId, locale);
|
|
318
|
+
if (realmLex[key] !== void 0) return realmLex[key];
|
|
319
|
+
}
|
|
320
|
+
const fb = resolveFallback(locale);
|
|
321
|
+
return fb[key] ?? fallback;
|
|
322
|
+
}
|
|
323
|
+
function lexiconForRealm(realmId, locale = null) {
|
|
324
|
+
const fb = resolveFallback(locale);
|
|
325
|
+
if (!realmId || !REALMS[realmId]) {
|
|
326
|
+
return fb;
|
|
327
|
+
}
|
|
328
|
+
return { ...fb, ...resolveRealm(realmId, locale) };
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
// src/themes/index.ts
|
|
332
|
+
var THEME_TOKENS = {
|
|
333
|
+
cosmos: {
|
|
334
|
+
bgDeep: "#04070d",
|
|
335
|
+
bgMid: "#0a1020",
|
|
336
|
+
accent: "#7eb8ff",
|
|
337
|
+
accentSoft: "rgba(126, 184, 255, 0.35)",
|
|
338
|
+
text: "#e8f0ff",
|
|
339
|
+
muted: "rgba(232, 240, 255, 0.65)"
|
|
340
|
+
},
|
|
341
|
+
wilds: {
|
|
342
|
+
bgDeep: "#0d120c",
|
|
343
|
+
bgMid: "#152018",
|
|
344
|
+
accent: "#7bc96f",
|
|
345
|
+
accentSoft: "rgba(123, 201, 111, 0.35)",
|
|
346
|
+
text: "#eef6ea",
|
|
347
|
+
muted: "rgba(238, 246, 234, 0.7)"
|
|
348
|
+
}
|
|
349
|
+
};
|
|
350
|
+
function getThemeTokens(realmId) {
|
|
351
|
+
return THEME_TOKENS[realmId ?? "wilds"] ?? THEME_TOKENS.wilds;
|
|
352
|
+
}
|
|
353
|
+
var RealmThemeContext = createContext(null);
|
|
354
|
+
function RealmThemeProvider({
|
|
355
|
+
realmId,
|
|
356
|
+
locale = null,
|
|
357
|
+
children
|
|
358
|
+
}) {
|
|
359
|
+
const value = useMemo(() => {
|
|
360
|
+
return {
|
|
361
|
+
realmId,
|
|
362
|
+
locale,
|
|
363
|
+
tokens: getThemeTokens(realmId),
|
|
364
|
+
lexicon: lexiconForRealm(realmId, locale),
|
|
365
|
+
t: (key, fallback = "") => t(realmId, key, fallback, locale)
|
|
366
|
+
};
|
|
367
|
+
}, [realmId, locale]);
|
|
368
|
+
return /* @__PURE__ */ jsx(RealmThemeContext.Provider, { value, children });
|
|
369
|
+
}
|
|
370
|
+
function useRealmTheme() {
|
|
371
|
+
const ctx = useContext(RealmThemeContext);
|
|
372
|
+
if (!ctx) {
|
|
373
|
+
throw new Error(
|
|
374
|
+
"useRealmTheme: no RealmThemeProvider in tree. Wrap your app root with <RealmThemeProvider realmId={...} />."
|
|
375
|
+
);
|
|
376
|
+
}
|
|
377
|
+
return ctx;
|
|
378
|
+
}
|
|
379
|
+
function Button({
|
|
380
|
+
label,
|
|
381
|
+
variant = "primary",
|
|
382
|
+
style,
|
|
383
|
+
...rest
|
|
384
|
+
}) {
|
|
385
|
+
const { tokens } = useRealmTheme();
|
|
386
|
+
return /* @__PURE__ */ jsx(
|
|
387
|
+
Pressable,
|
|
388
|
+
{
|
|
389
|
+
accessibilityRole: "button",
|
|
390
|
+
style: ({ pressed }) => [
|
|
391
|
+
styles.base,
|
|
392
|
+
variant === "primary" ? { backgroundColor: tokens.accent } : {
|
|
393
|
+
backgroundColor: "transparent",
|
|
394
|
+
borderColor: tokens.accent,
|
|
395
|
+
borderWidth: 1
|
|
396
|
+
},
|
|
397
|
+
pressed && { opacity: 0.85 },
|
|
398
|
+
style
|
|
399
|
+
],
|
|
400
|
+
...rest,
|
|
401
|
+
children: /* @__PURE__ */ jsx(
|
|
402
|
+
Text,
|
|
403
|
+
{
|
|
404
|
+
style: [
|
|
405
|
+
styles.label,
|
|
406
|
+
{
|
|
407
|
+
color: variant === "primary" ? tokens.bgDeep : tokens.accent
|
|
408
|
+
}
|
|
409
|
+
],
|
|
410
|
+
children: label
|
|
411
|
+
}
|
|
412
|
+
)
|
|
413
|
+
}
|
|
414
|
+
);
|
|
415
|
+
}
|
|
416
|
+
var styles = StyleSheet.create({
|
|
417
|
+
base: {
|
|
418
|
+
paddingHorizontal: 18,
|
|
419
|
+
paddingVertical: 10,
|
|
420
|
+
borderRadius: 8,
|
|
421
|
+
alignItems: "center",
|
|
422
|
+
justifyContent: "center"
|
|
423
|
+
},
|
|
424
|
+
label: {
|
|
425
|
+
fontSize: 14,
|
|
426
|
+
fontWeight: "600",
|
|
427
|
+
letterSpacing: 0.5
|
|
428
|
+
}
|
|
429
|
+
});
|
|
430
|
+
function FilmCard({ film, onPress, style }) {
|
|
431
|
+
const { tokens, t: t2 } = useRealmTheme();
|
|
432
|
+
const title = film.title?.trim() || t2("untitled_story", "Untitled");
|
|
433
|
+
const genre = film.genre?.trim() || t2("unknown_genre", "Unknown");
|
|
434
|
+
const sceneLabel = t2("scenes_count", "scenes");
|
|
435
|
+
const choiceLabel = t2("choices_count", "choices");
|
|
436
|
+
const inner = /* @__PURE__ */ jsxs(Fragment, { children: [
|
|
437
|
+
/* @__PURE__ */ jsx(
|
|
438
|
+
Text,
|
|
439
|
+
{
|
|
440
|
+
style: [styles2.title, { color: tokens.text }],
|
|
441
|
+
numberOfLines: 2,
|
|
442
|
+
children: title
|
|
443
|
+
}
|
|
444
|
+
),
|
|
445
|
+
/* @__PURE__ */ jsxs(Text, { style: [styles2.meta, { color: tokens.muted }], children: [
|
|
446
|
+
genre,
|
|
447
|
+
" \xB7 ",
|
|
448
|
+
film.scenes_count,
|
|
449
|
+
" ",
|
|
450
|
+
sceneLabel,
|
|
451
|
+
" \xB7 ",
|
|
452
|
+
film.choices_count,
|
|
453
|
+
" ",
|
|
454
|
+
choiceLabel
|
|
455
|
+
] }),
|
|
456
|
+
film.creator?.name ? /* @__PURE__ */ jsx(Text, { style: [styles2.creator, { color: tokens.accent }], children: film.creator.name }) : null
|
|
457
|
+
] });
|
|
458
|
+
if (onPress) {
|
|
459
|
+
return /* @__PURE__ */ jsx(
|
|
460
|
+
Pressable,
|
|
461
|
+
{
|
|
462
|
+
accessibilityRole: "button",
|
|
463
|
+
onPress,
|
|
464
|
+
style: ({ pressed }) => [
|
|
465
|
+
styles2.base,
|
|
466
|
+
{ backgroundColor: tokens.bgMid, borderColor: tokens.accentSoft },
|
|
467
|
+
pressed && { opacity: 0.92 },
|
|
468
|
+
style
|
|
469
|
+
],
|
|
470
|
+
children: inner
|
|
471
|
+
}
|
|
472
|
+
);
|
|
473
|
+
}
|
|
474
|
+
return /* @__PURE__ */ jsx(
|
|
475
|
+
View,
|
|
476
|
+
{
|
|
477
|
+
style: [
|
|
478
|
+
styles2.base,
|
|
479
|
+
{ backgroundColor: tokens.bgMid, borderColor: tokens.accentSoft },
|
|
480
|
+
style
|
|
481
|
+
],
|
|
482
|
+
children: inner
|
|
483
|
+
}
|
|
484
|
+
);
|
|
485
|
+
}
|
|
486
|
+
var styles2 = StyleSheet.create({
|
|
487
|
+
base: {
|
|
488
|
+
padding: 16,
|
|
489
|
+
borderRadius: 12,
|
|
490
|
+
borderWidth: 1,
|
|
491
|
+
gap: 6,
|
|
492
|
+
minWidth: 240
|
|
493
|
+
},
|
|
494
|
+
title: {
|
|
495
|
+
fontSize: 16,
|
|
496
|
+
fontWeight: "700"
|
|
497
|
+
},
|
|
498
|
+
meta: {
|
|
499
|
+
fontSize: 12
|
|
500
|
+
},
|
|
501
|
+
creator: {
|
|
502
|
+
fontSize: 12,
|
|
503
|
+
fontWeight: "600"
|
|
504
|
+
}
|
|
505
|
+
});
|
|
506
|
+
function CreditBadge({ role, value, style }) {
|
|
507
|
+
const { tokens } = useRealmTheme();
|
|
508
|
+
const label = role === "viewer" ? "Viewer Credits" : "Creator Credits";
|
|
509
|
+
return /* @__PURE__ */ jsxs(
|
|
510
|
+
View,
|
|
511
|
+
{
|
|
512
|
+
style: [
|
|
513
|
+
styles3.container,
|
|
514
|
+
{
|
|
515
|
+
borderColor: tokens.accent,
|
|
516
|
+
backgroundColor: tokens.bgMid
|
|
517
|
+
},
|
|
518
|
+
style
|
|
519
|
+
],
|
|
520
|
+
children: [
|
|
521
|
+
/* @__PURE__ */ jsx(Text, { style: [styles3.label, { color: tokens.muted }], children: label }),
|
|
522
|
+
/* @__PURE__ */ jsx(Text, { style: [styles3.value, { color: tokens.accent }], children: value.toLocaleString() })
|
|
523
|
+
]
|
|
524
|
+
}
|
|
525
|
+
);
|
|
526
|
+
}
|
|
527
|
+
var styles3 = StyleSheet.create({
|
|
528
|
+
container: {
|
|
529
|
+
flexDirection: "row",
|
|
530
|
+
alignItems: "center",
|
|
531
|
+
gap: 8,
|
|
532
|
+
paddingHorizontal: 14,
|
|
533
|
+
paddingVertical: 8,
|
|
534
|
+
borderRadius: 8,
|
|
535
|
+
borderWidth: 1
|
|
536
|
+
},
|
|
537
|
+
label: {
|
|
538
|
+
fontSize: 12,
|
|
539
|
+
fontWeight: "600",
|
|
540
|
+
letterSpacing: 0.4,
|
|
541
|
+
textTransform: "uppercase"
|
|
542
|
+
},
|
|
543
|
+
value: {
|
|
544
|
+
fontSize: 16,
|
|
545
|
+
fontWeight: "700"
|
|
546
|
+
}
|
|
547
|
+
});
|
|
548
|
+
|
|
549
|
+
export { Button, CreditBadge, FilmCard, RealmThemeProvider, useRealmTheme };
|
|
550
|
+
//# sourceMappingURL=index.js.map
|
|
551
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/config/index.ts","../../src/themes/index.ts","../../src/ui/theme/RealmThemeProvider.tsx","../../src/ui/Button.tsx","../../src/ui/FilmCard.tsx","../../src/ui/CreditBadge.tsx"],"names":["jsx","t","Text","styles","Pressable","StyleSheet","jsxs","View"],"mappings":"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","file":"index.js","sourcesContent":["import type {\n RealmId,\n RealmDefinition,\n ContentRating,\n ContentRatingDefinition,\n LocaleId,\n} from \"../types/index.js\";\n\nexport const REALM_IDS: readonly RealmId[] = [\"cosmos\", \"wilds\"] as const;\n\nexport const SUPPORTED_LOCALES: readonly LocaleId[] = [\n \"en\",\n \"es\",\n \"fr\",\n \"de\",\n \"pt\",\n] as const;\n\nexport const LOCALE_LABELS: Record<LocaleId, string> = {\n en: \"English\",\n es: \"Español\",\n fr: \"Français\",\n de: \"Deutsch\",\n pt: \"Português\",\n};\n\nexport const DEFAULT_LOCALE: LocaleId = \"en\";\n\nexport function isSupportedLocale(value: unknown): value is LocaleId {\n return (\n typeof value === \"string\" &&\n SUPPORTED_LOCALES.includes(value as LocaleId)\n );\n}\n\n/**\n * Normalize a raw locale tag (e.g. \"en-US\", \"pt_BR\") to a supported LocaleId,\n * matching the PHP `LocaleResolver::normalize` behavior.\n */\nexport function normalizeLocale(value: string | null | undefined): LocaleId | null {\n if (!value) return null;\n const lower = value.toLowerCase().trim();\n if (isSupportedLocale(lower)) return lower;\n const base = lower.split(/[-_]/, 1)[0];\n return isSupportedLocale(base) ? base : null;\n}\n\n/**\n * Pick the best supported locale from a browser Accept-Language string.\n * Mirrors PHP `LocaleResolver::matchAcceptLanguage` (q-value aware).\n */\nexport function matchAcceptLanguage(\n accept: string | null | undefined\n): LocaleId | null {\n if (!accept) return null;\n const tags = accept\n .split(\",\")\n .map((part) => {\n const [tag, ...params] = part.trim().split(\";\");\n const qParam = params.find((p) => p.trim().startsWith(\"q=\"));\n const q = qParam ? parseFloat(qParam.split(\"=\")[1]) || 0 : 1;\n return { tag: tag.trim(), q };\n })\n .filter((c) => c.tag && c.tag !== \"*\")\n .sort((a, b) => b.q - a.q);\n\n for (const c of tags) {\n const normalized = normalizeLocale(c.tag);\n if (normalized) return normalized;\n }\n return null;\n}\n\nexport const CONTENT_RATING_IDS: readonly ContentRating[] = [\n \"G\",\n \"PG\",\n \"PG-13\",\n \"R\",\n \"NC-17\",\n] as const;\n\nexport const CONTENT_RATINGS: Record<ContentRating, ContentRatingDefinition> = {\n G: {\n id: \"G\",\n label: \"G\",\n description: \"General Audiences — all ages admitted.\",\n },\n PG: {\n id: \"PG\",\n label: \"PG\",\n description: \"Parental Guidance Suggested — some material may not be suitable for children.\",\n },\n \"PG-13\": {\n id: \"PG-13\",\n label: \"PG-13\",\n description: \"Parents Strongly Cautioned — some material may be inappropriate for children under 13.\",\n },\n R: {\n id: \"R\",\n label: \"R\",\n description: \"Restricted — under 17 requires accompanying parent or adult guardian.\",\n },\n \"NC-17\": {\n id: \"NC-17\",\n label: \"NC-17\",\n description: \"Adults Only — no one 17 and under admitted.\",\n },\n};\n\nexport function isContentRating(value: unknown): value is ContentRating {\n return typeof value === \"string\" && CONTENT_RATING_IDS.includes(value as ContentRating);\n}\n\nexport const REALMS: Record<RealmId, RealmDefinition> = {\n cosmos: {\n label: \"CHUZI COSMOS\",\n short_label: \"Cosmos\",\n lexicon: {\n genre: \"Galaxy\",\n story: \"Star System\",\n scene: \"Planet\",\n scene_choice: \"Trajectory\",\n node: \"Signal\",\n node_type_choice: \"Trajectory\",\n node_type_media: \"Transmission\",\n node_type_go_to_scene: \"Warp\",\n player: \"Voyager\",\n library: \"Star Chart\",\n library_open: \"Open Star Chart\",\n library_close: \"Close Star Chart\",\n editor: \"Mission Control\",\n publish: \"Launch\",\n the_end: \"Final Orbit\",\n media_pool: \"Cargo Bay\",\n scene_graph: \"Star Chart\",\n watch_path: \"Your trajectory\",\n untitled_story: \"Untitled Star System\",\n unknown_genre: \"Unknown Galaxy\",\n scenes_count: \"planets\",\n choices_count: \"trajectories\",\n drafts_published: \"Drafts + Launched\",\n published_ver: \"Launched ver.\",\n draft: \"Draft\",\n no_scenes_yet: \"No planets in this chart yet.\",\n no_path_yet: \"No trajectory to show yet.\",\n menu_empty_drafts: \"No star systems yet. Chart one to begin.\",\n menu_empty_published: \"No launched systems in the chart.\",\n menu_published_heading: \"Launched constellations\",\n menu_create_story: \"New star system\",\n menu_create_film: \"Create film\",\n film_title_placeholder: \"Star system title\",\n select_genre: \"Select galaxy\",\n genre_field_aria: \"Galaxy\",\n delete_story_verb: \"Delete star system\",\n title_scene_default: \"Title planet\",\n choices_made_stat: \"trajectories taken\",\n content_rating: \"Audience Class\",\n content_rating_field_aria: \"Audience class\",\n select_content_rating: \"Select audience class\",\n unrated: \"Unclassified\",\n viewer_credits: \"Fuel Cells\",\n creator_credits: \"Stardust\",\n },\n locales: {\n es: {\n genre: \"Galaxia\",\n story: \"Sistema Estelar\",\n scene: \"Planeta\",\n scene_choice: \"Trayectoria\",\n player: \"Viajero\",\n library: \"Carta Estelar\",\n editor: \"Control de Misión\",\n publish: \"Lanzar\",\n the_end: \"Órbita Final\",\n untitled_story: \"Sistema Estelar sin título\",\n menu_create_story: \"Nuevo sistema estelar\",\n film_title_placeholder: \"Título del sistema estelar\",\n select_genre: \"Seleccionar galaxia\",\n },\n fr: {\n genre: \"Galaxie\",\n story: \"Système Stellaire\",\n scene: \"Planète\",\n scene_choice: \"Trajectoire\",\n player: \"Voyageur\",\n library: \"Carte Stellaire\",\n editor: \"Contrôle de Mission\",\n publish: \"Lancer\",\n the_end: \"Orbite Finale\",\n untitled_story: \"Système stellaire sans titre\",\n menu_create_story: \"Nouveau système stellaire\",\n film_title_placeholder: \"Titre du système stellaire\",\n select_genre: \"Sélectionner une galaxie\",\n },\n de: {\n genre: \"Galaxie\",\n story: \"Sternsystem\",\n scene: \"Planet\",\n player: \"Reisender\",\n publish: \"Starten\",\n },\n pt: {\n genre: \"Galáxia\",\n story: \"Sistema Estelar\",\n scene: \"Planeta\",\n player: \"Viajante\",\n publish: \"Lançar\",\n },\n },\n },\n wilds: {\n label: \"CHUZI WILDS\",\n short_label: \"Wilds\",\n lexicon: {\n genre: \"Biome\",\n story: \"Grove\",\n scene: \"Clearing\",\n scene_choice: \"Trail\",\n node: \"Seed\",\n node_type_choice: \"Trail\",\n node_type_media: \"Bloom\",\n node_type_go_to_scene: \"Crossing\",\n player: \"Wanderer\",\n library: \"Canopy\",\n library_open: \"Open Canopy\",\n library_close: \"Close Canopy\",\n editor: \"Heartwood\",\n publish: \"Plant\",\n the_end: \"Roots\",\n media_pool: \"Undergrowth\",\n scene_graph: \"Canopy\",\n watch_path: \"Your trail\",\n untitled_story: \"Untitled Grove\",\n unknown_genre: \"Unknown Biome\",\n scenes_count: \"clearings\",\n choices_count: \"trails\",\n drafts_published: \"Drafts + Planted\",\n published_ver: \"Planted ver.\",\n draft: \"Draft\",\n no_scenes_yet: \"No clearings in this canopy yet.\",\n no_path_yet: \"No trail to show yet.\",\n menu_empty_drafts: \"No groves yet. Plant one to begin.\",\n menu_empty_published: \"No planted groves in the canopy.\",\n menu_published_heading: \"Planted groves\",\n menu_create_story: \"New grove\",\n menu_create_film: \"Create film\",\n film_title_placeholder: \"Grove title\",\n select_genre: \"Select biome\",\n genre_field_aria: \"Biome\",\n delete_story_verb: \"Delete grove\",\n title_scene_default: \"Title clearing\",\n choices_made_stat: \"trails taken\",\n content_rating: \"Field Guide\",\n content_rating_field_aria: \"Field guide\",\n select_content_rating: \"Select field guide\",\n unrated: \"Unmarked\",\n viewer_credits: \"Sap\",\n creator_credits: \"Pollen\",\n },\n locales: {\n es: {\n genre: \"Bioma\",\n story: \"Arboleda\",\n scene: \"Claro\",\n scene_choice: \"Sendero\",\n player: \"Errante\",\n library: \"Dosel\",\n editor: \"Duramen\",\n publish: \"Plantar\",\n the_end: \"Raíces\",\n untitled_story: \"Arboleda sin título\",\n menu_create_story: \"Nueva arboleda\",\n film_title_placeholder: \"Título de la arboleda\",\n select_genre: \"Seleccionar bioma\",\n },\n fr: {\n genre: \"Biome\",\n story: \"Bosquet\",\n scene: \"Clairière\",\n scene_choice: \"Sentier\",\n player: \"Vagabond\",\n library: \"Canopée\",\n editor: \"Cœur de bois\",\n publish: \"Planter\",\n the_end: \"Racines\",\n untitled_story: \"Bosquet sans titre\",\n menu_create_story: \"Nouveau bosquet\",\n film_title_placeholder: \"Titre du bosquet\",\n select_genre: \"Sélectionner un biome\",\n },\n de: {\n genre: \"Biom\",\n story: \"Hain\",\n scene: \"Lichtung\",\n player: \"Wanderer\",\n publish: \"Pflanzen\",\n },\n pt: {\n genre: \"Bioma\",\n story: \"Bosque\",\n scene: \"Clareira\",\n player: \"Andarilho\",\n publish: \"Plantar\",\n },\n },\n },\n};\n\nexport const FALLBACK_LEXICON: Record<string, string> = {\n genre: \"Genre\",\n story: \"Story\",\n scene: \"Scene\",\n scene_choice: \"Choice\",\n node: \"Node\",\n node_type_choice: \"Choice\",\n node_type_media: \"Media\",\n node_type_go_to_scene: \"Go to scene\",\n player: \"Player\",\n library: \"Library\",\n library_open: \"Open scene tree\",\n library_close: \"Close scene tree\",\n editor: \"Editor\",\n publish: \"Publish\",\n the_end: \"The End\",\n media_pool: \"Media Pool\",\n scene_graph: \"Scene tree\",\n watch_path: \"Your path\",\n untitled_story: \"Untitled Story\",\n unknown_genre: \"Unknown Genre\",\n scenes_count: \"scenes\",\n choices_count: \"choices\",\n drafts_published: \"Drafts + Published\",\n published_ver: \"Published ver.\",\n draft: \"Draft\",\n no_scenes_yet: \"No scenes available yet.\",\n no_path_yet: \"No path to show yet.\",\n menu_empty_drafts: \"No films yet. Create one to get started.\",\n menu_empty_published: \"No published films found.\",\n menu_published_heading: \"Published Films\",\n menu_create_story: \"Create Story\",\n menu_create_film: \"Create Film\",\n film_title_placeholder: \"Film title\",\n select_genre: \"Select genre\",\n genre_field_aria: \"Film genre\",\n delete_story_verb: \"Delete film\",\n title_scene_default: \"Title Scene\",\n choices_made_stat: \"choices made\",\n content_rating: \"Rating\",\n content_rating_field_aria: \"Content rating\",\n select_content_rating: \"Select rating\",\n unrated: \"Not Rated\",\n viewer_credits: \"Viewer Credits\",\n creator_credits: \"Creator Credits\",\n};\n\n/**\n * Per-locale overrides for FALLBACK_LEXICON. Missing keys fall through to\n * the English fallback. Mirrors PHP `chuzi_realms.fallback_locales`.\n */\nexport const FALLBACK_LOCALES: Partial<\n Record<LocaleId, Record<string, string>>\n> = {\n es: {\n genre: \"Género\",\n story: \"Historia\",\n scene: \"Escena\",\n scene_choice: \"Elección\",\n player: \"Reproductor\",\n library: \"Biblioteca\",\n editor: \"Editor\",\n publish: \"Publicar\",\n the_end: \"Fin\",\n untitled_story: \"Historia sin título\",\n menu_create_story: \"Crear historia\",\n menu_create_film: \"Crear película\",\n film_title_placeholder: \"Título de la película\",\n select_genre: \"Seleccionar género\",\n delete_story_verb: \"Eliminar película\",\n },\n fr: {\n genre: \"Genre\",\n story: \"Histoire\",\n scene: \"Scène\",\n scene_choice: \"Choix\",\n player: \"Lecteur\",\n library: \"Bibliothèque\",\n editor: \"Éditeur\",\n publish: \"Publier\",\n the_end: \"Fin\",\n untitled_story: \"Histoire sans titre\",\n menu_create_story: \"Créer une histoire\",\n menu_create_film: \"Créer un film\",\n film_title_placeholder: \"Titre du film\",\n select_genre: \"Sélectionner un genre\",\n delete_story_verb: \"Supprimer le film\",\n },\n de: {\n genre: \"Genre\",\n story: \"Geschichte\",\n scene: \"Szene\",\n publish: \"Veröffentlichen\",\n the_end: \"Ende\",\n untitled_story: \"Unbenannte Geschichte\",\n menu_create_story: \"Geschichte erstellen\",\n film_title_placeholder: \"Filmtitel\",\n select_genre: \"Genre auswählen\",\n },\n pt: {\n genre: \"Gênero\",\n story: \"História\",\n scene: \"Cena\",\n publish: \"Publicar\",\n the_end: \"Fim\",\n untitled_story: \"História sem título\",\n menu_create_story: \"Criar história\",\n film_title_placeholder: \"Título do filme\",\n select_genre: \"Selecionar gênero\",\n },\n};\n\nfunction resolveFallback(locale: LocaleId | null | undefined): Record<string, string> {\n const overrides = locale && FALLBACK_LOCALES[locale];\n return overrides ? { ...FALLBACK_LEXICON, ...overrides } : { ...FALLBACK_LEXICON };\n}\n\nfunction resolveRealm(\n realmId: RealmId,\n locale: LocaleId | null | undefined\n): Record<string, string> {\n const realm = REALMS[realmId];\n const overrides = locale && realm.locales ? realm.locales[locale] : undefined;\n return overrides ? { ...realm.lexicon, ...overrides } : { ...realm.lexicon };\n}\n\n/**\n * Get a lexicon value for a realm + locale, falling back to the neutral\n * lexicon (which itself respects locale).\n */\nexport function t(\n realmId: RealmId | null | undefined,\n key: string,\n fallback = \"\",\n locale: LocaleId | null | undefined = null\n): string {\n if (realmId && REALMS[realmId]) {\n const realmLex = resolveRealm(realmId, locale);\n if (realmLex[key] !== undefined) return realmLex[key];\n }\n const fb = resolveFallback(locale);\n return fb[key] ?? fallback;\n}\n\n/**\n * Get the full merged lexicon for a realm + locale (realm lexicon on top of\n * locale-aware fallback).\n */\nexport function lexiconForRealm(\n realmId: RealmId | null | undefined,\n locale: LocaleId | null | undefined = null\n): Record<string, string> {\n const fb = resolveFallback(locale);\n if (!realmId || !REALMS[realmId]) {\n return fb;\n }\n return { ...fb, ...resolveRealm(realmId, locale) };\n}\n","import type { RealmId } from \"../types/index.js\";\n\n/**\n * CSS custom property tokens, mirroring public/css/chuzi-realms.css.\n * Use these for any non-DOM rendering (e.g. React Native, canvas).\n */\nexport interface RealmThemeTokens {\n bgDeep: string;\n bgMid: string;\n accent: string;\n accentSoft: string;\n text: string;\n muted: string;\n}\n\nexport const THEME_TOKENS: Record<RealmId, RealmThemeTokens> = {\n cosmos: {\n bgDeep: \"#04070d\",\n bgMid: \"#0a1020\",\n accent: \"#7eb8ff\",\n accentSoft: \"rgba(126, 184, 255, 0.35)\",\n text: \"#e8f0ff\",\n muted: \"rgba(232, 240, 255, 0.65)\",\n },\n wilds: {\n bgDeep: \"#0d120c\",\n bgMid: \"#152018\",\n accent: \"#7bc96f\",\n accentSoft: \"rgba(123, 201, 111, 0.35)\",\n text: \"#eef6ea\",\n muted: \"rgba(238, 246, 234, 0.7)\",\n },\n};\n\n/**\n * Scene tree visualization theme, mirroring the THEMES object in\n * resources/js/scene-tree-viewer.js.\n */\nexport interface SceneTreeTheme {\n bg: string;\n edgeChoice: string;\n edgeGoto: string;\n nodeDefault: string;\n nodeActive: string;\n borderDefault: string;\n borderActive: string;\n labelDefault: string;\n labelActive: string;\n nodeLockedFill: string;\n nodeLockedBorder: string;\n labelLocked: string;\n nodeShape: \"star\" | \"rect\";\n rectRx: number;\n}\n\nexport const SCENE_TREE_THEMES: Record<RealmId, SceneTreeTheme> = {\n cosmos: {\n bg: \"#020408\",\n edgeChoice: \"#e8f0ff\",\n edgeGoto: \"#4a9fff\",\n nodeDefault: \"#ffffff\",\n nodeActive: \"#ffd47e\",\n borderDefault: \"#3a5080\",\n borderActive: \"#fff6d0\",\n labelDefault: \"rgba(220, 232, 255, 0.92)\",\n labelActive: \"rgba(255, 246, 220, 0.98)\",\n nodeLockedFill: \"#151a28\",\n nodeLockedBorder: \"#2a3348\",\n labelLocked: \"rgba(200, 210, 230, 0.35)\",\n nodeShape: \"star\",\n rectRx: 2,\n },\n wilds: {\n bg: \"#04070d\",\n edgeChoice: \"#ffffff\",\n edgeGoto: \"#6ecf7a\",\n nodeDefault: \"#e8f5e4\",\n nodeActive: \"#d31e2f\",\n borderDefault: \"#2d4a32\",\n borderActive: \"#ff9ea8\",\n labelDefault: \"rgba(232, 245, 228, 0.92)\",\n labelActive: \"rgba(255, 214, 219, 0.98)\",\n nodeLockedFill: \"#1a221c\",\n nodeLockedBorder: \"#2a3d30\",\n labelLocked: \"rgba(200, 220, 200, 0.38)\",\n nodeShape: \"rect\",\n rectRx: 3,\n },\n};\n\n/**\n * Get theme tokens for a realm, defaulting to wilds.\n */\nexport function getThemeTokens(realmId: RealmId | null | undefined): RealmThemeTokens {\n return THEME_TOKENS[realmId ?? \"wilds\"] ?? THEME_TOKENS.wilds;\n}\n\n/**\n * Get scene tree theme for a realm, defaulting to wilds.\n */\nexport function getSceneTreeTheme(realmId: RealmId | null | undefined): SceneTreeTheme {\n return SCENE_TREE_THEMES[realmId ?? \"wilds\"] ?? SCENE_TREE_THEMES.wilds;\n}\n","import { createContext, useContext, useMemo, type ReactNode } from \"react\";\nimport { lexiconForRealm, t as translateRealm } from \"../../config/index.js\";\nimport { getThemeTokens, type RealmThemeTokens } from \"../../themes/index.js\";\nimport type { LocaleId, RealmId } from \"../../types/index.js\";\n\n/**\n * Realm-aware theme + lexicon context. Every chuzi-shared/ui primitive reads\n * from this; consuming apps wrap their tree once at the top level. Switching\n * realms or locales re-renders all consumers without prop drilling.\n */\n\nexport interface RealmTheme {\n realmId: RealmId | null;\n locale: LocaleId | null;\n tokens: RealmThemeTokens;\n /** Realm + locale aware lexicon lookup. */\n t: (key: string, fallback?: string) => string;\n /** Full merged lexicon for the current realm + locale. */\n lexicon: Record<string, string>;\n}\n\nconst RealmThemeContext = createContext<RealmTheme | null>(null);\n\nexport interface RealmThemeProviderProps {\n realmId: RealmId | null;\n locale?: LocaleId | null;\n children: ReactNode;\n}\n\nexport function RealmThemeProvider({\n realmId,\n locale = null,\n children,\n}: RealmThemeProviderProps) {\n const value = useMemo<RealmTheme>(() => {\n return {\n realmId,\n locale,\n tokens: getThemeTokens(realmId),\n lexicon: lexiconForRealm(realmId, locale),\n t: (key, fallback = \"\") => translateRealm(realmId, key, fallback, locale),\n };\n }, [realmId, locale]);\n\n return (\n <RealmThemeContext.Provider value={value}>\n {children}\n </RealmThemeContext.Provider>\n );\n}\n\nexport function useRealmTheme(): RealmTheme {\n const ctx = useContext(RealmThemeContext);\n if (!ctx) {\n throw new Error(\n \"useRealmTheme: no RealmThemeProvider in tree. Wrap your app root with <RealmThemeProvider realmId={...} />.\",\n );\n }\n return ctx;\n}\n","import {\n Pressable,\n StyleSheet,\n Text,\n type PressableProps,\n type StyleProp,\n type ViewStyle,\n} from \"react-native\";\nimport { useRealmTheme } from \"./theme/RealmThemeProvider.js\";\n\nexport type ButtonVariant = \"primary\" | \"ghost\";\n\nexport interface ButtonProps\n extends Omit<PressableProps, \"style\" | \"children\"> {\n label: string;\n variant?: ButtonVariant;\n style?: StyleProp<ViewStyle>;\n}\n\n/**\n * Realm-themed button. Primitives come from react-native so the same\n * component renders on web (via react-native-web) and native. Colors come\n * from the active realm's theme tokens — primary fills with accent, ghost\n * draws an accent outline.\n */\nexport function Button({\n label,\n variant = \"primary\",\n style,\n ...rest\n}: ButtonProps) {\n const { tokens } = useRealmTheme();\n return (\n <Pressable\n accessibilityRole=\"button\"\n style={({ pressed }) => [\n styles.base,\n variant === \"primary\"\n ? { backgroundColor: tokens.accent }\n : {\n backgroundColor: \"transparent\",\n borderColor: tokens.accent,\n borderWidth: 1,\n },\n pressed && { opacity: 0.85 },\n style,\n ]}\n {...rest}\n >\n <Text\n style={[\n styles.label,\n {\n color: variant === \"primary\" ? tokens.bgDeep : tokens.accent,\n },\n ]}\n >\n {label}\n </Text>\n </Pressable>\n );\n}\n\nconst styles = StyleSheet.create({\n base: {\n paddingHorizontal: 18,\n paddingVertical: 10,\n borderRadius: 8,\n alignItems: \"center\",\n justifyContent: \"center\",\n },\n label: {\n fontSize: 14,\n fontWeight: \"600\",\n letterSpacing: 0.5,\n },\n});\n","import {\n Pressable,\n StyleSheet,\n Text,\n View,\n type StyleProp,\n type ViewStyle,\n} from \"react-native\";\nimport type { StoryListItem } from \"../types/index.js\";\nimport { useRealmTheme } from \"./theme/RealmThemeProvider.js\";\n\nexport interface FilmCardProps {\n film: StoryListItem;\n /** Tap/click handler. Without one, the card is a static block. */\n onPress?: () => void;\n style?: StyleProp<ViewStyle>;\n}\n\n/**\n * Realm-themed film card. Title falls back to the realm's \"untitled\" lexicon\n * (Untitled Star System / Untitled Grove). Genre falls back to the realm's\n * \"unknown genre\" string. Pressable when an onPress is provided; otherwise\n * renders a non-interactive View — important for list contexts that want\n * the press handled at the row level instead.\n */\nexport function FilmCard({ film, onPress, style }: FilmCardProps) {\n const { tokens, t } = useRealmTheme();\n\n const title = film.title?.trim() || t(\"untitled_story\", \"Untitled\");\n const genre = film.genre?.trim() || t(\"unknown_genre\", \"Unknown\");\n const sceneLabel = t(\"scenes_count\", \"scenes\");\n const choiceLabel = t(\"choices_count\", \"choices\");\n\n const inner = (\n <>\n <Text\n style={[styles.title, { color: tokens.text }]}\n numberOfLines={2}\n >\n {title}\n </Text>\n <Text style={[styles.meta, { color: tokens.muted }]}>\n {genre} · {film.scenes_count} {sceneLabel} · {film.choices_count}{\" \"}\n {choiceLabel}\n </Text>\n {film.creator?.name ? (\n <Text style={[styles.creator, { color: tokens.accent }]}>\n {film.creator.name}\n </Text>\n ) : null}\n </>\n );\n\n if (onPress) {\n return (\n <Pressable\n accessibilityRole=\"button\"\n onPress={onPress}\n style={({ pressed }) => [\n styles.base,\n { backgroundColor: tokens.bgMid, borderColor: tokens.accentSoft },\n pressed && { opacity: 0.92 },\n style,\n ]}\n >\n {inner}\n </Pressable>\n );\n }\n\n return (\n <View\n style={[\n styles.base,\n { backgroundColor: tokens.bgMid, borderColor: tokens.accentSoft },\n style,\n ]}\n >\n {inner}\n </View>\n );\n}\n\nconst styles = StyleSheet.create({\n base: {\n padding: 16,\n borderRadius: 12,\n borderWidth: 1,\n gap: 6,\n minWidth: 240,\n },\n title: {\n fontSize: 16,\n fontWeight: \"700\",\n },\n meta: {\n fontSize: 12,\n },\n creator: {\n fontSize: 12,\n fontWeight: \"600\",\n },\n});\n","import {\n StyleSheet,\n Text,\n View,\n type StyleProp,\n type ViewStyle,\n} from \"react-native\";\nimport { useRealmTheme } from \"./theme/RealmThemeProvider.js\";\n\nexport type CreditRole = \"viewer\" | \"creator\";\n\nexport interface CreditBadgeProps {\n role: CreditRole;\n value: number;\n style?: StyleProp<ViewStyle>;\n}\n\n/**\n * Realm-themed credit badge — a bordered label showing a viewer or creator\n * credit balance. The label text adapts to the active realm's lexicon and\n * colors pull from the realm's theme tokens.\n */\nexport function CreditBadge({ role, value, style }: CreditBadgeProps) {\n const { tokens } = useRealmTheme();\n\n const label = role === \"viewer\" ? \"Viewer Credits\" : \"Creator Credits\";\n\n return (\n <View\n style={[\n styles.container,\n {\n borderColor: tokens.accent,\n backgroundColor: tokens.bgMid,\n },\n style,\n ]}\n >\n <Text style={[styles.label, { color: tokens.muted }]}>{label}</Text>\n <Text style={[styles.value, { color: tokens.accent }]}>\n {value.toLocaleString()}\n </Text>\n </View>\n );\n}\n\nconst styles = StyleSheet.create({\n container: {\n flexDirection: \"row\",\n alignItems: \"center\",\n gap: 8,\n paddingHorizontal: 14,\n paddingVertical: 8,\n borderRadius: 8,\n borderWidth: 1,\n },\n label: {\n fontSize: 12,\n fontWeight: \"600\",\n letterSpacing: 0.4,\n textTransform: \"uppercase\",\n },\n value: {\n fontSize: 16,\n fontWeight: \"700\",\n },\n});\n"]}
|