@chuzi/shared 1.2.0 → 1.2.2

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.
@@ -78,6 +78,12 @@ interface CatalogResponse {
78
78
  progress: Record<string, StoryProgress>;
79
79
  };
80
80
  }
81
+ interface MineResponse {
82
+ data: StoryListItem[];
83
+ meta: {
84
+ coverboxes: Record<string, string | null>;
85
+ };
86
+ }
81
87
  interface SceneChoice {
82
88
  id: string;
83
89
  label: string;
@@ -261,6 +267,70 @@ interface UpdateSceneRequest {
261
267
  choice_pause_for_choice?: boolean;
262
268
  choice_end_time_seconds?: number | null;
263
269
  }
270
+ interface CreateSceneRequest {
271
+ story_id: string;
272
+ title: string;
273
+ order?: number;
274
+ }
275
+ type AiGenerationStatus = "pending" | "generating" | "ready" | "failed" | "accepted" | "rejected";
276
+ interface GenerateImageRequest {
277
+ story_id: string;
278
+ scene_id?: string;
279
+ prompt: string;
280
+ mood?: string;
281
+ visual_style?: string;
282
+ rejection_feedback?: string;
283
+ previous_generation_id?: string;
284
+ }
285
+ interface AiGeneration {
286
+ id: string;
287
+ story_id: string;
288
+ scene_id: string | null;
289
+ status: AiGenerationStatus;
290
+ user_prompt: string;
291
+ style_context: {
292
+ mood?: string;
293
+ visual_style?: string;
294
+ subject_prompt: string;
295
+ full_prompt: string;
296
+ negative_prompt?: string;
297
+ };
298
+ preview_url?: string;
299
+ rejection_feedback: string | null;
300
+ media_id: string | null;
301
+ attempt_number: number;
302
+ credits_charged: number;
303
+ created_at: string;
304
+ updated_at: string;
305
+ }
306
+ interface GenerateImageResponse {
307
+ generation: AiGeneration;
308
+ balance: {
309
+ watch: number;
310
+ create: number;
311
+ };
312
+ }
313
+ interface AiGenerationShowResponse {
314
+ generation: AiGeneration;
315
+ }
316
+ interface AcceptGenerationRequest {
317
+ scene_id: string;
318
+ }
319
+ interface AcceptGenerationResponse {
320
+ generation: AiGeneration;
321
+ scene: SceneListItem;
322
+ media: MediaItem;
323
+ }
324
+ interface RejectGenerationRequest {
325
+ feedback: string;
326
+ }
327
+ interface RejectGenerationResponse {
328
+ generation: AiGeneration;
329
+ }
330
+ interface StoryStyleResponse {
331
+ has_style: boolean;
332
+ style_context: AiGeneration["style_context"] | null;
333
+ }
264
334
  interface MediaItem {
265
335
  id: string;
266
336
  user_id: string;
@@ -364,4 +434,4 @@ interface UpdateLocaleResponse {
364
434
  locale: LocaleId;
365
435
  }
366
436
 
367
- export type { BookmarkListItem, BookmarkListResponse, BookmarkResponse, CatalogResponse, ContentRating, ContentRatingDefinition, CreateStoryRequest, EngagementResponse, HistoryEntry, LocaleId, LoginRequest, LoginResponse, MediaItem, PaginatedLink, PaginatedResponse, PlayUrlResponse, PopularChoice, RealmConfigResponse, RealmDefinition, RealmId, RegisterMediaRequest, RegisterMediaResponse, SaveBookmarkRequest, SceneChoice, SceneListItem, SceneMapEntry, SceneMapResponse, SceneNode, SourceUrlResponse, StateUpdate, StoryListItem, StoryPreview, StoryProgress, TrackEngagementRequest, TranscodeRequest, TranscodeResponse, TreeGraph, TreeGraphEdge, TreeGraphNode, UpdateLocaleRequest, UpdateLocaleResponse, UpdateProfileRequest, UpdateProfileResponse, UpdateRealmRequest, UpdateRealmResponse, UpdateSceneRequest, UploadUrlRequest, UploadUrlResponse, UserProfile, VisibilityCondition, VisibilityRules, WatchSnapshot };
437
+ export type { AcceptGenerationRequest, AcceptGenerationResponse, AiGeneration, AiGenerationShowResponse, AiGenerationStatus, BookmarkListItem, BookmarkListResponse, BookmarkResponse, CatalogResponse, ContentRating, ContentRatingDefinition, CreateSceneRequest, CreateStoryRequest, EngagementResponse, GenerateImageRequest, GenerateImageResponse, HistoryEntry, LocaleId, LoginRequest, LoginResponse, MediaItem, MineResponse, PaginatedLink, PaginatedResponse, PlayUrlResponse, PopularChoice, RealmConfigResponse, RealmDefinition, RealmId, RegisterMediaRequest, RegisterMediaResponse, RejectGenerationRequest, RejectGenerationResponse, SaveBookmarkRequest, SceneChoice, SceneListItem, SceneMapEntry, SceneMapResponse, SceneNode, SourceUrlResponse, StateUpdate, StoryListItem, StoryPreview, StoryProgress, StoryStyleResponse, TrackEngagementRequest, TranscodeRequest, TranscodeResponse, TreeGraph, TreeGraphEdge, TreeGraphNode, UpdateLocaleRequest, UpdateLocaleResponse, UpdateProfileRequest, UpdateProfileResponse, UpdateRealmRequest, UpdateRealmResponse, UpdateSceneRequest, UploadUrlRequest, UploadUrlResponse, UserProfile, VisibilityCondition, VisibilityRules, WatchSnapshot };
package/dist/ui/index.js CHANGED
@@ -58,9 +58,9 @@ var REALMS = {
58
58
  editor_media_preview: "Transmission Preview",
59
59
  editor_alt_media: "Additional Transmission",
60
60
  editor_choices: "Trajectories",
61
- editor_scene_settings: "Planet Settings",
62
- editor_title_scene: "Title Planet",
63
- editor_ending_scene: "Final Orbit",
61
+ editor_scene_settings: "Star Settings",
62
+ editor_title_scene: "Title Star",
63
+ editor_ending_scene: "Final Star",
64
64
  editor_play: "Engage",
65
65
  editor_pause: "Hold",
66
66
  editor_prev_frame: "Previous Frame",
@@ -69,9 +69,27 @@ var REALMS = {
69
69
  editor_no_media: "No transmission assigned",
70
70
  editor_uploading: "Transmitting\u2026",
71
71
  editor_drop_media: "Drop transmission here",
72
- editor_scene_type: "Planet Type",
72
+ editor_scene_type: "Star Type",
73
73
  editor_save: "Lock In",
74
- editor_scene_color: "Planet Hue"
74
+ editor_scene_color: "Star Hue",
75
+ editor_ai_generate: "Generate Transmission",
76
+ editor_ai_generating: "Transmitting Signal\u2026",
77
+ editor_ai_mood_question: "What energy should this planet radiate?",
78
+ editor_ai_style_question: "What vision style for this transmission?",
79
+ editor_ai_prompt_question: "Describe this planet\u2019s landscape",
80
+ editor_ai_accept: "Lock Transmission",
81
+ editor_ai_reject: "Reject Signal",
82
+ editor_ai_retry: "Retransmit",
83
+ editor_ai_feedback_prompt: "How should we adjust the signal?",
84
+ editor_ai_credit_cost: "1 Stardust per signal",
85
+ editor_ai_generating_scene: "Planet forming\u2026",
86
+ editor_ai_style_badge: "Signal Style",
87
+ editor_ai_failed: "Signal lost. Try retransmitting.",
88
+ editor_media_attached: "This star already has a transmission locked in.",
89
+ editor_remove_media: "Remove Transmission",
90
+ constellation_no_coverbox: "Uncharted",
91
+ hud_edit_control: "Engage",
92
+ hud_delete_control: "Jettison"
75
93
  },
76
94
  locales: {
77
95
  es: {
@@ -185,7 +203,25 @@ var REALMS = {
185
203
  editor_drop_media: "Drop bloom here",
186
204
  editor_scene_type: "Clearing Type",
187
205
  editor_save: "Engrave",
188
- editor_scene_color: "Clearing Tint"
206
+ editor_scene_color: "Clearing Tint",
207
+ editor_ai_generate: "Grow Bloom",
208
+ editor_ai_generating: "Seed Sprouting\u2026",
209
+ editor_ai_mood_question: "What feeling should this clearing evoke?",
210
+ editor_ai_style_question: "What growth pattern for this bloom?",
211
+ editor_ai_prompt_question: "Describe this clearing",
212
+ editor_ai_accept: "Root Bloom",
213
+ editor_ai_reject: "Prune Bloom",
214
+ editor_ai_retry: "Regrow",
215
+ editor_ai_feedback_prompt: "How should we reshape the growth?",
216
+ editor_ai_credit_cost: "1 Pollen per bloom",
217
+ editor_ai_generating_scene: "Clearing emerging\u2026",
218
+ editor_ai_style_badge: "Growth Style",
219
+ editor_ai_failed: "Growth withered. Try regrowing.",
220
+ editor_media_attached: "This clearing already has a bloom rooted.",
221
+ editor_remove_media: "Uproot Bloom",
222
+ constellation_no_coverbox: "Unseen",
223
+ hud_edit_control: "Tend",
224
+ hud_delete_control: "Uproot"
189
225
  },
190
226
  locales: {
191
227
  es: {
@@ -297,7 +333,25 @@ var FALLBACK_LEXICON = {
297
333
  editor_drop_media: "Drop media here",
298
334
  editor_scene_type: "Scene Type",
299
335
  editor_save: "Save",
300
- editor_scene_color: "Scene Color"
336
+ editor_scene_color: "Scene Color",
337
+ editor_ai_generate: "Generate with AI",
338
+ editor_ai_generating: "Generating\u2026",
339
+ editor_ai_mood_question: "What mood should this scene convey?",
340
+ editor_ai_style_question: "What visual style do you want?",
341
+ editor_ai_prompt_question: "Describe this scene",
342
+ editor_ai_accept: "Accept",
343
+ editor_ai_reject: "Reject",
344
+ editor_ai_retry: "Try Again",
345
+ editor_ai_feedback_prompt: "What should we change?",
346
+ editor_ai_credit_cost: "1 credit per generation",
347
+ editor_ai_generating_scene: "Scene generating\u2026",
348
+ editor_ai_style_badge: "Style",
349
+ editor_ai_failed: "Generation failed. Please try again.",
350
+ editor_media_attached: "This scene already has media attached.",
351
+ editor_remove_media: "Remove Media",
352
+ constellation_no_coverbox: "No cover",
353
+ hud_edit_control: "Edit",
354
+ hud_delete_control: "Delete"
301
355
  };
302
356
  var FALLBACK_LOCALES = {
303
357
  es: {
@@ -1 +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":";;;;;;;AAiHO,IAAM,MAAA,GAA2C;AAAA,EACtD,MAAA,EAAQ;AAAA,IACN,KAAA,EAAO,cAAA;AAAA,IACP,WAAA,EAAa,QAAA;AAAA,IACb,OAAA,EAAS;AAAA,MACP,KAAA,EAAO,QAAA;AAAA,MACP,KAAA,EAAO,aAAA;AAAA,MACP,KAAA,EAAO,QAAA;AAAA,MACP,YAAA,EAAc,YAAA;AAAA,MACd,IAAA,EAAM,QAAA;AAAA,MACN,gBAAA,EAAkB,YAAA;AAAA,MAClB,eAAA,EAAiB,cAAA;AAAA,MACjB,qBAAA,EAAuB,MAAA;AAAA,MACvB,MAAA,EAAQ,SAAA;AAAA,MACR,OAAA,EAAS,YAAA;AAAA,MACT,YAAA,EAAc,iBAAA;AAAA,MACd,aAAA,EAAe,kBAAA;AAAA,MACf,MAAA,EAAQ,iBAAA;AAAA,MACR,OAAA,EAAS,QAAA;AAAA,MACT,OAAA,EAAS,aAAA;AAAA,MACT,UAAA,EAAY,WAAA;AAAA,MACZ,WAAA,EAAa,YAAA;AAAA,MACb,UAAA,EAAY,iBAAA;AAAA,MACZ,cAAA,EAAgB,sBAAA;AAAA,MAChB,aAAA,EAAe,gBAAA;AAAA,MACf,YAAA,EAAc,SAAA;AAAA,MACd,aAAA,EAAe,cAAA;AAAA,MACf,gBAAA,EAAkB,mBAAA;AAAA,MAClB,aAAA,EAAe,eAAA;AAAA,MACf,KAAA,EAAO,OAAA;AAAA,MACP,aAAA,EAAe,+BAAA;AAAA,MACf,WAAA,EAAa,4BAAA;AAAA,MACb,iBAAA,EAAmB,0CAAA;AAAA,MACnB,oBAAA,EAAsB,mCAAA;AAAA,MACtB,sBAAA,EAAwB,yBAAA;AAAA,MACxB,iBAAA,EAAmB,iBAAA;AAAA,MACnB,gBAAA,EAAkB,aAAA;AAAA,MAClB,sBAAA,EAAwB,mBAAA;AAAA,MACxB,YAAA,EAAc,eAAA;AAAA,MACd,gBAAA,EAAkB,QAAA;AAAA,MAClB,iBAAA,EAAmB,oBAAA;AAAA,MACnB,mBAAA,EAAqB,cAAA;AAAA,MACrB,iBAAA,EAAmB,oBAAA;AAAA,MACnB,cAAA,EAAgB,gBAAA;AAAA,MAChB,yBAAA,EAA2B,gBAAA;AAAA,MAC3B,qBAAA,EAAuB,uBAAA;AAAA,MACvB,OAAA,EAAS,cAAA;AAAA,MACT,cAAA,EAAgB,YAAA;AAAA,MAChB,eAAA,EAAiB,UAAA;AAAA,MACjB,mBAAA,EAAqB,qBAAA;AAAA,MACrB,oBAAA,EAAsB,sBAAA;AAAA,MACtB,gBAAA,EAAkB,yBAAA;AAAA,MAClB,cAAA,EAAgB,cAAA;AAAA,MAChB,qBAAA,EAAuB,iBAAA;AAAA,MACvB,kBAAA,EAAoB,cAAA;AAAA,MACpB,mBAAA,EAAqB,aAAA;AAAA,MACrB,WAAA,EAAa,QAAA;AAAA,MACb,YAAA,EAAc,MAAA;AAAA,MACd,iBAAA,EAAmB,gBAAA;AAAA,MACnB,iBAAA,EAAmB,YAAA;AAAA,MACnB,WAAA,EAAa,MAAA;AAAA,MACb,eAAA,EAAiB,0BAAA;AAAA,MACjB,gBAAA,EAAkB,oBAAA;AAAA,MAClB,iBAAA,EAAmB,wBAAA;AAAA,MACnB,iBAAA,EAAmB,aAAA;AAAA,MACnB,WAAA,EAAa,SAAA;AAAA,MACb,kBAAA,EAAoB;AAAA,KACtB;AAAA,IACA,OAAA,EAAS;AAAA,MACP,EAAA,EAAI;AAAA,QACF,KAAA,EAAO,SAAA;AAAA,QACP,KAAA,EAAO,iBAAA;AAAA,QACP,KAAA,EAAO,SAAA;AAAA,QACP,YAAA,EAAc,aAAA;AAAA,QACd,MAAA,EAAQ,SAAA;AAAA,QACR,OAAA,EAAS,eAAA;AAAA,QACT,MAAA,EAAQ,sBAAA;AAAA,QACR,OAAA,EAAS,QAAA;AAAA,QACT,OAAA,EAAS,iBAAA;AAAA,QACT,cAAA,EAAgB,+BAAA;AAAA,QAChB,iBAAA,EAAmB,uBAAA;AAAA,QACnB,sBAAA,EAAwB,+BAAA;AAAA,QACxB,YAAA,EAAc;AAAA,OAChB;AAAA,MACA,EAAA,EAAI;AAAA,QACF,KAAA,EAAO,SAAA;AAAA,QACP,KAAA,EAAO,sBAAA;AAAA,QACP,KAAA,EAAO,YAAA;AAAA,QACP,YAAA,EAAc,aAAA;AAAA,QACd,MAAA,EAAQ,UAAA;AAAA,QACR,OAAA,EAAS,iBAAA;AAAA,QACT,MAAA,EAAQ,wBAAA;AAAA,QACR,OAAA,EAAS,QAAA;AAAA,QACT,OAAA,EAAS,eAAA;AAAA,QACT,cAAA,EAAgB,iCAAA;AAAA,QAChB,iBAAA,EAAmB,8BAAA;AAAA,QACnB,sBAAA,EAAwB,+BAAA;AAAA,QACxB,YAAA,EAAc;AAAA,OAChB;AAAA,MACA,EAAA,EAAI;AAAA,QACF,KAAA,EAAO,SAAA;AAAA,QACP,KAAA,EAAO,aAAA;AAAA,QACP,KAAA,EAAO,QAAA;AAAA,QACP,MAAA,EAAQ,WAAA;AAAA,QACR,OAAA,EAAS;AAAA,OACX;AAAA,MACA,EAAA,EAAI;AAAA,QACF,KAAA,EAAO,YAAA;AAAA,QACP,KAAA,EAAO,iBAAA;AAAA,QACP,KAAA,EAAO,SAAA;AAAA,QACP,MAAA,EAAQ,UAAA;AAAA,QACR,OAAA,EAAS;AAAA;AACX;AACF,GACF;AAAA,EACA,KAAA,EAAO;AAAA,IACL,KAAA,EAAO,aAAA;AAAA,IACP,WAAA,EAAa,OAAA;AAAA,IACb,OAAA,EAAS;AAAA,MACP,KAAA,EAAO,OAAA;AAAA,MACP,KAAA,EAAO,OAAA;AAAA,MACP,KAAA,EAAO,UAAA;AAAA,MACP,YAAA,EAAc,OAAA;AAAA,MACd,IAAA,EAAM,MAAA;AAAA,MACN,gBAAA,EAAkB,OAAA;AAAA,MAClB,eAAA,EAAiB,OAAA;AAAA,MACjB,qBAAA,EAAuB,UAAA;AAAA,MACvB,MAAA,EAAQ,UAAA;AAAA,MACR,OAAA,EAAS,QAAA;AAAA,MACT,YAAA,EAAc,aAAA;AAAA,MACd,aAAA,EAAe,cAAA;AAAA,MACf,MAAA,EAAQ,WAAA;AAAA,MACR,OAAA,EAAS,OAAA;AAAA,MACT,OAAA,EAAS,OAAA;AAAA,MACT,UAAA,EAAY,aAAA;AAAA,MACZ,WAAA,EAAa,QAAA;AAAA,MACb,UAAA,EAAY,YAAA;AAAA,MACZ,cAAA,EAAgB,gBAAA;AAAA,MAChB,aAAA,EAAe,eAAA;AAAA,MACf,YAAA,EAAc,WAAA;AAAA,MACd,aAAA,EAAe,QAAA;AAAA,MACf,gBAAA,EAAkB,kBAAA;AAAA,MAClB,aAAA,EAAe,cAAA;AAAA,MACf,KAAA,EAAO,OAAA;AAAA,MACP,aAAA,EAAe,kCAAA;AAAA,MACf,WAAA,EAAa,uBAAA;AAAA,MACb,iBAAA,EAAmB,oCAAA;AAAA,MACnB,oBAAA,EAAsB,kCAAA;AAAA,MACtB,sBAAA,EAAwB,gBAAA;AAAA,MACxB,iBAAA,EAAmB,WAAA;AAAA,MACnB,gBAAA,EAAkB,aAAA;AAAA,MAClB,sBAAA,EAAwB,aAAA;AAAA,MACxB,YAAA,EAAc,cAAA;AAAA,MACd,gBAAA,EAAkB,OAAA;AAAA,MAClB,iBAAA,EAAmB,cAAA;AAAA,MACnB,mBAAA,EAAqB,gBAAA;AAAA,MACrB,iBAAA,EAAmB,cAAA;AAAA,MACnB,cAAA,EAAgB,aAAA;AAAA,MAChB,yBAAA,EAA2B,aAAA;AAAA,MAC3B,qBAAA,EAAuB,oBAAA;AAAA,MACvB,OAAA,EAAS,UAAA;AAAA,MACT,cAAA,EAAgB,KAAA;AAAA,MAChB,eAAA,EAAiB,QAAA;AAAA,MACjB,mBAAA,EAAqB,cAAA;AAAA,MACrB,oBAAA,EAAsB,eAAA;AAAA,MACtB,gBAAA,EAAkB,kBAAA;AAAA,MAClB,cAAA,EAAgB,QAAA;AAAA,MAChB,qBAAA,EAAuB,mBAAA;AAAA,MACvB,kBAAA,EAAoB,gBAAA;AAAA,MACpB,mBAAA,EAAqB,OAAA;AAAA,MACrB,WAAA,EAAa,QAAA;AAAA,MACb,YAAA,EAAc,MAAA;AAAA,MACd,iBAAA,EAAmB,gBAAA;AAAA,MACnB,iBAAA,EAAmB,YAAA;AAAA,MACnB,WAAA,EAAa,MAAA;AAAA,MACb,eAAA,EAAiB,mBAAA;AAAA,MACjB,gBAAA,EAAkB,gBAAA;AAAA,MAClB,iBAAA,EAAmB,iBAAA;AAAA,MACnB,iBAAA,EAAmB,eAAA;AAAA,MACnB,WAAA,EAAa,SAAA;AAAA,MACb,kBAAA,EAAoB;AAAA,KACtB;AAAA,IACA,OAAA,EAAS;AAAA,MACP,EAAA,EAAI;AAAA,QACF,KAAA,EAAO,OAAA;AAAA,QACP,KAAA,EAAO,UAAA;AAAA,QACP,KAAA,EAAO,OAAA;AAAA,QACP,YAAA,EAAc,SAAA;AAAA,QACd,MAAA,EAAQ,SAAA;AAAA,QACR,OAAA,EAAS,OAAA;AAAA,QACT,MAAA,EAAQ,SAAA;AAAA,QACR,OAAA,EAAS,SAAA;AAAA,QACT,OAAA,EAAS,WAAA;AAAA,QACT,cAAA,EAAgB,wBAAA;AAAA,QAChB,iBAAA,EAAmB,gBAAA;AAAA,QACnB,sBAAA,EAAwB,0BAAA;AAAA,QACxB,YAAA,EAAc;AAAA,OAChB;AAAA,MACA,EAAA,EAAI;AAAA,QACF,KAAA,EAAO,OAAA;AAAA,QACP,KAAA,EAAO,SAAA;AAAA,QACP,KAAA,EAAO,cAAA;AAAA,QACP,YAAA,EAAc,SAAA;AAAA,QACd,MAAA,EAAQ,UAAA;AAAA,QACR,OAAA,EAAS,YAAA;AAAA,QACT,MAAA,EAAQ,mBAAA;AAAA,QACR,OAAA,EAAS,SAAA;AAAA,QACT,OAAA,EAAS,SAAA;AAAA,QACT,cAAA,EAAgB,oBAAA;AAAA,QAChB,iBAAA,EAAmB,iBAAA;AAAA,QACnB,sBAAA,EAAwB,kBAAA;AAAA,QACxB,YAAA,EAAc;AAAA,OAChB;AAAA,MACA,EAAA,EAAI;AAAA,QACF,KAAA,EAAO,MAAA;AAAA,QACP,KAAA,EAAO,MAAA;AAAA,QACP,KAAA,EAAO,UAAA;AAAA,QACP,MAAA,EAAQ,UAAA;AAAA,QACR,OAAA,EAAS;AAAA,OACX;AAAA,MACA,EAAA,EAAI;AAAA,QACF,KAAA,EAAO,OAAA;AAAA,QACP,KAAA,EAAO,QAAA;AAAA,QACP,KAAA,EAAO,UAAA;AAAA,QACP,MAAA,EAAQ,WAAA;AAAA,QACR,OAAA,EAAS;AAAA;AACX;AACF;AAEJ,CAAA;AAEO,IAAM,gBAAA,GAA2C;AAAA,EACtD,KAAA,EAAO,OAAA;AAAA,EACP,KAAA,EAAO,OAAA;AAAA,EACP,KAAA,EAAO,OAAA;AAAA,EACP,YAAA,EAAc,QAAA;AAAA,EACd,IAAA,EAAM,MAAA;AAAA,EACN,gBAAA,EAAkB,QAAA;AAAA,EAClB,eAAA,EAAiB,OAAA;AAAA,EACjB,qBAAA,EAAuB,aAAA;AAAA,EACvB,MAAA,EAAQ,QAAA;AAAA,EACR,OAAA,EAAS,SAAA;AAAA,EACT,YAAA,EAAc,iBAAA;AAAA,EACd,aAAA,EAAe,kBAAA;AAAA,EACf,MAAA,EAAQ,QAAA;AAAA,EACR,OAAA,EAAS,SAAA;AAAA,EACT,OAAA,EAAS,SAAA;AAAA,EACT,UAAA,EAAY,YAAA;AAAA,EACZ,WAAA,EAAa,YAAA;AAAA,EACb,UAAA,EAAY,WAAA;AAAA,EACZ,cAAA,EAAgB,gBAAA;AAAA,EAChB,aAAA,EAAe,eAAA;AAAA,EACf,YAAA,EAAc,QAAA;AAAA,EACd,aAAA,EAAe,SAAA;AAAA,EACf,gBAAA,EAAkB,oBAAA;AAAA,EAClB,aAAA,EAAe,gBAAA;AAAA,EACf,KAAA,EAAO,OAAA;AAAA,EACP,aAAA,EAAe,0BAAA;AAAA,EACf,WAAA,EAAa,sBAAA;AAAA,EACb,iBAAA,EAAmB,0CAAA;AAAA,EACnB,oBAAA,EAAsB,2BAAA;AAAA,EACtB,sBAAA,EAAwB,iBAAA;AAAA,EACxB,iBAAA,EAAmB,cAAA;AAAA,EACnB,gBAAA,EAAkB,aAAA;AAAA,EAClB,sBAAA,EAAwB,YAAA;AAAA,EACxB,YAAA,EAAc,cAAA;AAAA,EACd,gBAAA,EAAkB,YAAA;AAAA,EAClB,iBAAA,EAAmB,aAAA;AAAA,EACnB,mBAAA,EAAqB,aAAA;AAAA,EACrB,iBAAA,EAAmB,cAAA;AAAA,EACnB,cAAA,EAAgB,QAAA;AAAA,EAChB,yBAAA,EAA2B,gBAAA;AAAA,EAC3B,qBAAA,EAAuB,eAAA;AAAA,EACvB,OAAA,EAAS,WAAA;AAAA,EACT,cAAA,EAAgB,gBAAA;AAAA,EAChB,eAAA,EAAiB,iBAAA;AAAA,EACjB,mBAAA,EAAqB,cAAA;AAAA,EACrB,oBAAA,EAAsB,eAAA;AAAA,EACtB,gBAAA,EAAkB,kBAAA;AAAA,EAClB,cAAA,EAAgB,SAAA;AAAA,EAChB,qBAAA,EAAuB,gBAAA;AAAA,EACvB,kBAAA,EAAoB,aAAA;AAAA,EACpB,mBAAA,EAAqB,cAAA;AAAA,EACrB,WAAA,EAAa,MAAA;AAAA,EACb,YAAA,EAAc,OAAA;AAAA,EACd,iBAAA,EAAmB,gBAAA;AAAA,EACnB,iBAAA,EAAmB,YAAA;AAAA,EACnB,WAAA,EAAa,MAAA;AAAA,EACb,eAAA,EAAiB,mBAAA;AAAA,EACjB,gBAAA,EAAkB,iBAAA;AAAA,EAClB,iBAAA,EAAmB,iBAAA;AAAA,EACnB,iBAAA,EAAmB,YAAA;AAAA,EACnB,WAAA,EAAa,MAAA;AAAA,EACb,kBAAA,EAAoB;AACtB,CAAA;AAMO,IAAM,gBAAA,GAET;AAAA,EACF,EAAA,EAAI;AAAA,IACF,KAAA,EAAO,WAAA;AAAA,IACP,KAAA,EAAO,UAAA;AAAA,IACP,KAAA,EAAO,QAAA;AAAA,IACP,YAAA,EAAc,aAAA;AAAA,IACd,MAAA,EAAQ,aAAA;AAAA,IACR,OAAA,EAAS,YAAA;AAAA,IACT,MAAA,EAAQ,QAAA;AAAA,IACR,OAAA,EAAS,UAAA;AAAA,IACT,OAAA,EAAS,KAAA;AAAA,IACT,cAAA,EAAgB,wBAAA;AAAA,IAChB,iBAAA,EAAmB,gBAAA;AAAA,IACnB,gBAAA,EAAkB,mBAAA;AAAA,IAClB,sBAAA,EAAwB,6BAAA;AAAA,IACxB,YAAA,EAAc,uBAAA;AAAA,IACd,iBAAA,EAAmB;AAAA,GACrB;AAAA,EACA,EAAA,EAAI;AAAA,IACF,KAAA,EAAO,OAAA;AAAA,IACP,KAAA,EAAO,UAAA;AAAA,IACP,KAAA,EAAO,UAAA;AAAA,IACP,YAAA,EAAc,OAAA;AAAA,IACd,MAAA,EAAQ,SAAA;AAAA,IACR,OAAA,EAAS,iBAAA;AAAA,IACT,MAAA,EAAQ,YAAA;AAAA,IACR,OAAA,EAAS,SAAA;AAAA,IACT,OAAA,EAAS,KAAA;AAAA,IACT,cAAA,EAAgB,qBAAA;AAAA,IAChB,iBAAA,EAAmB,uBAAA;AAAA,IACnB,gBAAA,EAAkB,kBAAA;AAAA,IAClB,sBAAA,EAAwB,eAAA;AAAA,IACxB,YAAA,EAAc,0BAAA;AAAA,IACd,iBAAA,EAAmB;AAAA,GACrB;AAAA,EACA,EAAA,EAAI;AAAA,IACF,KAAA,EAAO,OAAA;AAAA,IACP,KAAA,EAAO,YAAA;AAAA,IACP,KAAA,EAAO,OAAA;AAAA,IACP,OAAA,EAAS,oBAAA;AAAA,IACT,OAAA,EAAS,MAAA;AAAA,IACT,cAAA,EAAgB,uBAAA;AAAA,IAChB,iBAAA,EAAmB,sBAAA;AAAA,IACnB,sBAAA,EAAwB,WAAA;AAAA,IACxB,YAAA,EAAc;AAAA,GAChB;AAAA,EACA,EAAA,EAAI;AAAA,IACF,KAAA,EAAO,WAAA;AAAA,IACP,KAAA,EAAO,aAAA;AAAA,IACP,KAAA,EAAO,MAAA;AAAA,IACP,OAAA,EAAS,UAAA;AAAA,IACT,OAAA,EAAS,KAAA;AAAA,IACT,cAAA,EAAgB,2BAAA;AAAA,IAChB,iBAAA,EAAmB,mBAAA;AAAA,IACnB,sBAAA,EAAwB,oBAAA;AAAA,IACxB,YAAA,EAAc;AAAA;AAElB,CAAA;AAEA,SAAS,gBAAgB,MAAA,EAA6D;AACpF,EAAA,MAAM,SAAA,GAAY,MAAA,IAAU,gBAAA,CAAiB,MAAM,CAAA;AACnD,EAAA,OAAO,SAAA,GAAY,EAAE,GAAG,gBAAA,EAAkB,GAAG,SAAA,EAAU,GAAI,EAAE,GAAG,gBAAA,EAAiB;AACnF;AAEA,SAAS,YAAA,CACP,SACA,MAAA,EACwB;AACxB,EAAA,MAAM,KAAA,GAAQ,OAAO,OAAO,CAAA;AAC5B,EAAA,MAAM,YAAY,MAAA,IAAU,KAAA,CAAM,UAAU,KAAA,CAAM,OAAA,CAAQ,MAAM,CAAA,GAAI,MAAA;AACpE,EAAA,OAAO,SAAA,GAAY,EAAE,GAAG,KAAA,CAAM,OAAA,EAAS,GAAG,SAAA,EAAU,GAAI,EAAE,GAAG,KAAA,CAAM,OAAA,EAAQ;AAC7E;AAMO,SAAS,EACd,OAAA,EACA,GAAA,EACA,QAAA,GAAW,EAAA,EACX,SAAsC,IAAA,EAC9B;AACR,EAAA,IAAI,OAAA,IAAW,MAAA,CAAO,OAAO,CAAA,EAAG;AAC9B,IAAA,MAAM,QAAA,GAAW,YAAA,CAAa,OAAA,EAAS,MAAM,CAAA;AAC7C,IAAA,IAAI,SAAS,GAAG,CAAA,KAAM,MAAA,EAAW,OAAO,SAAS,GAAG,CAAA;AAAA,EACtD;AACA,EAAA,MAAM,EAAA,GAAK,gBAAgB,MAAM,CAAA;AACjC,EAAA,OAAO,EAAA,CAAG,GAAG,CAAA,IAAK,QAAA;AACpB;AAMO,SAAS,eAAA,CACd,OAAA,EACA,MAAA,GAAsC,IAAA,EACd;AACxB,EAAA,MAAM,EAAA,GAAK,gBAAgB,MAAM,CAAA;AACjC,EAAA,IAAI,CAAC,OAAA,IAAW,CAAC,MAAA,CAAO,OAAO,CAAA,EAAG;AAChC,IAAA,OAAO,EAAA;AAAA,EACT;AACA,EAAA,OAAO,EAAE,GAAG,EAAA,EAAI,GAAG,YAAA,CAAa,OAAA,EAAS,MAAM,CAAA,EAAE;AACnD;;;ACxfO,IAAM,YAAA,GAAkD;AAAA,EAC7D,MAAA,EAAQ;AAAA,IACN,MAAA,EAAQ,SAAA;AAAA,IACR,KAAA,EAAO,SAAA;AAAA,IACP,MAAA,EAAQ,SAAA;AAAA,IACR,UAAA,EAAY,2BAAA;AAAA,IACZ,IAAA,EAAM,SAAA;AAAA,IACN,KAAA,EAAO;AAAA,GACT;AAAA,EACA,KAAA,EAAO;AAAA,IACL,MAAA,EAAQ,SAAA;AAAA,IACR,KAAA,EAAO,SAAA;AAAA,IACP,MAAA,EAAQ,SAAA;AAAA,IACR,UAAA,EAAY,2BAAA;AAAA,IACZ,IAAA,EAAM,SAAA;AAAA,IACN,KAAA,EAAO;AAAA;AAEX,CAAA;AA6DO,SAAS,eAAe,OAAA,EAAuD;AACpF,EAAA,OAAO,YAAA,CAAa,OAAA,IAAW,OAAO,CAAA,IAAK,YAAA,CAAa,KAAA;AAC1D;AC1EA,IAAM,iBAAA,GAAoB,cAAiC,IAAI,CAAA;AAQxD,SAAS,kBAAA,CAAmB;AAAA,EACjC,OAAA;AAAA,EACA,MAAA,GAAS,IAAA;AAAA,EACT;AACF,CAAA,EAA4B;AAC1B,EAAA,MAAM,KAAA,GAAQ,QAAoB,MAAM;AACtC,IAAA,OAAO;AAAA,MACL,OAAA;AAAA,MACA,MAAA;AAAA,MACA,MAAA,EAAQ,eAAe,OAAO,CAAA;AAAA,MAC9B,OAAA,EAAS,eAAA,CAAgB,OAAA,EAAS,MAAM,CAAA;AAAA,MACxC,CAAA,EAAG,CAAC,GAAA,EAAK,QAAA,GAAW,OAAO,CAAA,CAAe,OAAA,EAAS,GAAA,EAAK,QAAA,EAAU,MAAM;AAAA,KAC1E;AAAA,EACF,CAAA,EAAG,CAAC,OAAA,EAAS,MAAM,CAAC,CAAA;AAEpB,EAAA,uBACE,GAAA,CAAC,iBAAA,CAAkB,QAAA,EAAlB,EAA2B,OACzB,QAAA,EACH,CAAA;AAEJ;AAEO,SAAS,aAAA,GAA4B;AAC1C,EAAA,MAAM,GAAA,GAAM,WAAW,iBAAiB,CAAA;AACxC,EAAA,IAAI,CAAC,GAAA,EAAK;AACR,IAAA,MAAM,IAAI,KAAA;AAAA,MACR;AAAA,KACF;AAAA,EACF;AACA,EAAA,OAAO,GAAA;AACT;AClCO,SAAS,MAAA,CAAO;AAAA,EACrB,KAAA;AAAA,EACA,OAAA,GAAU,SAAA;AAAA,EACV,KAAA;AAAA,EACA,GAAG;AACL,CAAA,EAAgB;AACd,EAAA,MAAM,EAAE,MAAA,EAAO,GAAI,aAAA,EAAc;AACjC,EAAA,uBACEA,GAAAA;AAAA,IAAC,SAAA;AAAA,IAAA;AAAA,MACC,iBAAA,EAAkB,QAAA;AAAA,MAClB,KAAA,EAAO,CAAC,EAAE,OAAA,EAAQ,KAAM;AAAA,QACtB,MAAA,CAAO,IAAA;AAAA,QACP,YAAY,SAAA,GACR,EAAE,eAAA,EAAiB,MAAA,CAAO,QAAO,GACjC;AAAA,UACE,eAAA,EAAiB,aAAA;AAAA,UACjB,aAAa,MAAA,CAAO,MAAA;AAAA,UACpB,WAAA,EAAa;AAAA,SACf;AAAA,QACJ,OAAA,IAAW,EAAE,OAAA,EAAS,IAAA,EAAK;AAAA,QAC3B;AAAA,OACF;AAAA,MACC,GAAG,IAAA;AAAA,MAEJ,QAAA,kBAAAA,GAAAA;AAAA,QAAC,IAAA;AAAA,QAAA;AAAA,UACC,KAAA,EAAO;AAAA,YACL,MAAA,CAAO,KAAA;AAAA,YACP;AAAA,cACE,KAAA,EAAO,OAAA,KAAY,SAAA,GAAY,MAAA,CAAO,SAAS,MAAA,CAAO;AAAA;AACxD,WACF;AAAA,UAEC,QAAA,EAAA;AAAA;AAAA;AACH;AAAA,GACF;AAEJ;AAEA,IAAM,MAAA,GAAS,WAAW,MAAA,CAAO;AAAA,EAC/B,IAAA,EAAM;AAAA,IACJ,iBAAA,EAAmB,EAAA;AAAA,IACnB,eAAA,EAAiB,EAAA;AAAA,IACjB,YAAA,EAAc,CAAA;AAAA,IACd,UAAA,EAAY,QAAA;AAAA,IACZ,cAAA,EAAgB;AAAA,GAClB;AAAA,EACA,KAAA,EAAO;AAAA,IACL,QAAA,EAAU,EAAA;AAAA,IACV,UAAA,EAAY,KAAA;AAAA,IACZ,aAAA,EAAe;AAAA;AAEnB,CAAC,CAAA;ACnDM,SAAS,QAAA,CAAS,EAAE,IAAA,EAAM,OAAA,EAAS,OAAM,EAAkB;AAChE,EAAA,MAAM,EAAE,MAAA,EAAQ,CAAA,EAAAC,EAAAA,KAAM,aAAA,EAAc;AAEpC,EAAA,MAAM,QAAQ,IAAA,CAAK,KAAA,EAAO,MAAK,IAAKA,EAAAA,CAAE,kBAAkB,UAAU,CAAA;AAClE,EAAA,MAAM,QAAQ,IAAA,CAAK,KAAA,EAAO,MAAK,IAAKA,EAAAA,CAAE,iBAAiB,SAAS,CAAA;AAChE,EAAA,MAAM,UAAA,GAAaA,EAAAA,CAAE,cAAA,EAAgB,QAAQ,CAAA;AAC7C,EAAA,MAAM,WAAA,GAAcA,EAAAA,CAAE,eAAA,EAAiB,SAAS,CAAA;AAEhD,EAAA,MAAM,wBACJ,IAAA,CAAA,QAAA,EAAA,EACE,QAAA,EAAA;AAAA,oBAAAD,GAAAA;AAAA,MAACE,IAAAA;AAAA,MAAA;AAAA,QACC,KAAA,EAAO,CAACC,OAAAA,CAAO,KAAA,EAAO,EAAE,KAAA,EAAO,MAAA,CAAO,MAAM,CAAA;AAAA,QAC5C,aAAA,EAAe,CAAA;AAAA,QAEd,QAAA,EAAA;AAAA;AAAA,KACH;AAAA,oBACA,IAAA,CAACD,IAAAA,EAAA,EAAK,KAAA,EAAO,CAACC,OAAAA,CAAO,IAAA,EAAM,EAAE,KAAA,EAAO,MAAA,CAAO,KAAA,EAAO,CAAA,EAC/C,QAAA,EAAA;AAAA,MAAA,KAAA;AAAA,MAAM,QAAA;AAAA,MAAI,IAAA,CAAK,YAAA;AAAA,MAAa,GAAA;AAAA,MAAE,UAAA;AAAA,MAAW,QAAA;AAAA,MAAI,IAAA,CAAK,aAAA;AAAA,MAAe,GAAA;AAAA,MACjE;AAAA,KAAA,EACH,CAAA;AAAA,IACC,IAAA,CAAK,SAAS,IAAA,mBACbH,IAACE,IAAAA,EAAA,EAAK,OAAO,CAACC,OAAAA,CAAO,SAAS,EAAE,KAAA,EAAO,OAAO,MAAA,EAAQ,GACnD,QAAA,EAAA,IAAA,CAAK,OAAA,CAAQ,MAChB,CAAA,GACE;AAAA,GAAA,EACN,CAAA;AAGF,EAAA,IAAI,OAAA,EAAS;AACX,IAAA,uBACEH,GAAAA;AAAA,MAACI,SAAAA;AAAA,MAAA;AAAA,QACC,iBAAA,EAAkB,QAAA;AAAA,QAClB,OAAA;AAAA,QACA,KAAA,EAAO,CAAC,EAAE,OAAA,EAAQ,KAAM;AAAA,UACtBD,OAAAA,CAAO,IAAA;AAAA,UACP,EAAE,eAAA,EAAiB,MAAA,CAAO,KAAA,EAAO,WAAA,EAAa,OAAO,UAAA,EAAW;AAAA,UAChE,OAAA,IAAW,EAAE,OAAA,EAAS,IAAA,EAAK;AAAA,UAC3B;AAAA,SACF;AAAA,QAEC,QAAA,EAAA;AAAA;AAAA,KACH;AAAA,EAEJ;AAEA,EAAA,uBACEH,GAAAA;AAAA,IAAC,IAAA;AAAA,IAAA;AAAA,MACC,KAAA,EAAO;AAAA,QACLG,OAAAA,CAAO,IAAA;AAAA,QACP,EAAE,eAAA,EAAiB,MAAA,CAAO,KAAA,EAAO,WAAA,EAAa,OAAO,UAAA,EAAW;AAAA,QAChE;AAAA,OACF;AAAA,MAEC,QAAA,EAAA;AAAA;AAAA,GACH;AAEJ;AAEA,IAAMA,OAAAA,GAASE,WAAW,MAAA,CAAO;AAAA,EAC/B,IAAA,EAAM;AAAA,IACJ,OAAA,EAAS,EAAA;AAAA,IACT,YAAA,EAAc,EAAA;AAAA,IACd,WAAA,EAAa,CAAA;AAAA,IACb,GAAA,EAAK,CAAA;AAAA,IACL,QAAA,EAAU;AAAA,GACZ;AAAA,EACA,KAAA,EAAO;AAAA,IACL,QAAA,EAAU,EAAA;AAAA,IACV,UAAA,EAAY;AAAA,GACd;AAAA,EACA,IAAA,EAAM;AAAA,IACJ,QAAA,EAAU;AAAA,GACZ;AAAA,EACA,OAAA,EAAS;AAAA,IACP,QAAA,EAAU,EAAA;AAAA,IACV,UAAA,EAAY;AAAA;AAEhB,CAAC,CAAA;AChFM,SAAS,WAAA,CAAY,EAAE,IAAA,EAAM,KAAA,EAAO,OAAM,EAAqB;AACpE,EAAA,MAAM,EAAE,MAAA,EAAO,GAAI,aAAA,EAAc;AAEjC,EAAA,MAAM,KAAA,GAAQ,IAAA,KAAS,QAAA,GAAW,gBAAA,GAAmB,iBAAA;AAErD,EAAA,uBACEC,IAAAA;AAAA,IAACC,IAAAA;AAAA,IAAA;AAAA,MACC,KAAA,EAAO;AAAA,QACLJ,OAAAA,CAAO,SAAA;AAAA,QACP;AAAA,UACE,aAAa,MAAA,CAAO,MAAA;AAAA,UACpB,iBAAiB,MAAA,CAAO;AAAA,SAC1B;AAAA,QACA;AAAA,OACF;AAAA,MAEA,QAAA,EAAA;AAAA,wBAAAH,GAAAA,CAACE,IAAAA,EAAA,EAAK,KAAA,EAAO,CAACC,OAAAA,CAAO,KAAA,EAAO,EAAE,KAAA,EAAO,MAAA,CAAO,KAAA,EAAO,GAAI,QAAA,EAAA,KAAA,EAAM,CAAA;AAAA,wBAC7DH,GAAAA,CAACE,IAAAA,EAAA,EAAK,KAAA,EAAO,CAACC,OAAAA,CAAO,KAAA,EAAO,EAAE,KAAA,EAAO,OAAO,MAAA,EAAQ,CAAA,EACjD,QAAA,EAAA,KAAA,CAAM,gBAAe,EACxB;AAAA;AAAA;AAAA,GACF;AAEJ;AAEA,IAAMA,OAAAA,GAASE,WAAW,MAAA,CAAO;AAAA,EAC/B,SAAA,EAAW;AAAA,IACT,aAAA,EAAe,KAAA;AAAA,IACf,UAAA,EAAY,QAAA;AAAA,IACZ,GAAA,EAAK,CAAA;AAAA,IACL,iBAAA,EAAmB,EAAA;AAAA,IACnB,eAAA,EAAiB,CAAA;AAAA,IACjB,YAAA,EAAc,CAAA;AAAA,IACd,WAAA,EAAa;AAAA,GACf;AAAA,EACA,KAAA,EAAO;AAAA,IACL,QAAA,EAAU,EAAA;AAAA,IACV,UAAA,EAAY,KAAA;AAAA,IACZ,aAAA,EAAe,GAAA;AAAA,IACf,aAAA,EAAe;AAAA,GACjB;AAAA,EACA,KAAA,EAAO;AAAA,IACL,QAAA,EAAU,EAAA;AAAA,IACV,UAAA,EAAY;AAAA;AAEhB,CAAC,CAAA","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 editor_media_upload: \"Upload Transmission\",\n editor_media_preview: \"Transmission Preview\",\n editor_alt_media: \"Additional Transmission\",\n editor_choices: \"Trajectories\",\n editor_scene_settings: \"Planet Settings\",\n editor_title_scene: \"Title Planet\",\n editor_ending_scene: \"Final Orbit\",\n editor_play: \"Engage\",\n editor_pause: \"Hold\",\n editor_prev_frame: \"Previous Frame\",\n editor_next_frame: \"Next Frame\",\n editor_seek: \"Seek\",\n editor_no_media: \"No transmission assigned\",\n editor_uploading: \"Transmitting\\u2026\",\n editor_drop_media: \"Drop transmission here\",\n editor_scene_type: \"Planet Type\",\n editor_save: \"Lock In\",\n editor_scene_color: \"Planet Hue\",\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 editor_media_upload: \"Upload Bloom\",\n editor_media_preview: \"Bloom Preview\",\n editor_alt_media: \"Additional Bloom\",\n editor_choices: \"Trails\",\n editor_scene_settings: \"Clearing Settings\",\n editor_title_scene: \"Title Clearing\",\n editor_ending_scene: \"Roots\",\n editor_play: \"Unfurl\",\n editor_pause: \"Rest\",\n editor_prev_frame: \"Previous Frame\",\n editor_next_frame: \"Next Frame\",\n editor_seek: \"Seek\",\n editor_no_media: \"No bloom assigned\",\n editor_uploading: \"Blooming\\u2026\",\n editor_drop_media: \"Drop bloom here\",\n editor_scene_type: \"Clearing Type\",\n editor_save: \"Engrave\",\n editor_scene_color: \"Clearing Tint\",\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 editor_media_upload: \"Upload Media\",\n editor_media_preview: \"Media Preview\",\n editor_alt_media: \"Additional Media\",\n editor_choices: \"Choices\",\n editor_scene_settings: \"Scene Settings\",\n editor_title_scene: \"Title Scene\",\n editor_ending_scene: \"Ending Scene\",\n editor_play: \"Play\",\n editor_pause: \"Pause\",\n editor_prev_frame: \"Previous Frame\",\n editor_next_frame: \"Next Frame\",\n editor_seek: \"Seek\",\n editor_no_media: \"No media assigned\",\n editor_uploading: \"Uploading\\u2026\",\n editor_drop_media: \"Drop media here\",\n editor_scene_type: \"Scene Type\",\n editor_save: \"Save\",\n editor_scene_color: \"Scene Color\",\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"]}
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 editor_media_upload: \"Upload Transmission\",\n editor_media_preview: \"Transmission Preview\",\n editor_alt_media: \"Additional Transmission\",\n editor_choices: \"Trajectories\",\n editor_scene_settings: \"Star Settings\",\n editor_title_scene: \"Title Star\",\n editor_ending_scene: \"Final Star\",\n editor_play: \"Engage\",\n editor_pause: \"Hold\",\n editor_prev_frame: \"Previous Frame\",\n editor_next_frame: \"Next Frame\",\n editor_seek: \"Seek\",\n editor_no_media: \"No transmission assigned\",\n editor_uploading: \"Transmitting\\u2026\",\n editor_drop_media: \"Drop transmission here\",\n editor_scene_type: \"Star Type\",\n editor_save: \"Lock In\",\n editor_scene_color: \"Star Hue\",\n editor_ai_generate: \"Generate Transmission\",\n editor_ai_generating: \"Transmitting Signal\\u2026\",\n editor_ai_mood_question: \"What energy should this planet radiate?\",\n editor_ai_style_question: \"What vision style for this transmission?\",\n editor_ai_prompt_question: \"Describe this planet\\u2019s landscape\",\n editor_ai_accept: \"Lock Transmission\",\n editor_ai_reject: \"Reject Signal\",\n editor_ai_retry: \"Retransmit\",\n editor_ai_feedback_prompt: \"How should we adjust the signal?\",\n editor_ai_credit_cost: \"1 Stardust per signal\",\n editor_ai_generating_scene: \"Planet forming\\u2026\",\n editor_ai_style_badge: \"Signal Style\",\n editor_ai_failed: \"Signal lost. Try retransmitting.\",\n editor_media_attached: \"This star already has a transmission locked in.\",\n editor_remove_media: \"Remove Transmission\",\n constellation_no_coverbox: \"Uncharted\",\n hud_edit_control: \"Engage\",\n hud_delete_control: \"Jettison\",\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 editor_media_upload: \"Upload Bloom\",\n editor_media_preview: \"Bloom Preview\",\n editor_alt_media: \"Additional Bloom\",\n editor_choices: \"Trails\",\n editor_scene_settings: \"Clearing Settings\",\n editor_title_scene: \"Title Clearing\",\n editor_ending_scene: \"Roots\",\n editor_play: \"Unfurl\",\n editor_pause: \"Rest\",\n editor_prev_frame: \"Previous Frame\",\n editor_next_frame: \"Next Frame\",\n editor_seek: \"Seek\",\n editor_no_media: \"No bloom assigned\",\n editor_uploading: \"Blooming\\u2026\",\n editor_drop_media: \"Drop bloom here\",\n editor_scene_type: \"Clearing Type\",\n editor_save: \"Engrave\",\n editor_scene_color: \"Clearing Tint\",\n editor_ai_generate: \"Grow Bloom\",\n editor_ai_generating: \"Seed Sprouting\\u2026\",\n editor_ai_mood_question: \"What feeling should this clearing evoke?\",\n editor_ai_style_question: \"What growth pattern for this bloom?\",\n editor_ai_prompt_question: \"Describe this clearing\",\n editor_ai_accept: \"Root Bloom\",\n editor_ai_reject: \"Prune Bloom\",\n editor_ai_retry: \"Regrow\",\n editor_ai_feedback_prompt: \"How should we reshape the growth?\",\n editor_ai_credit_cost: \"1 Pollen per bloom\",\n editor_ai_generating_scene: \"Clearing emerging\\u2026\",\n editor_ai_style_badge: \"Growth Style\",\n editor_ai_failed: \"Growth withered. Try regrowing.\",\n editor_media_attached: \"This clearing already has a bloom rooted.\",\n editor_remove_media: \"Uproot Bloom\",\n constellation_no_coverbox: \"Unseen\",\n hud_edit_control: \"Tend\",\n hud_delete_control: \"Uproot\",\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 editor_media_upload: \"Upload Media\",\n editor_media_preview: \"Media Preview\",\n editor_alt_media: \"Additional Media\",\n editor_choices: \"Choices\",\n editor_scene_settings: \"Scene Settings\",\n editor_title_scene: \"Title Scene\",\n editor_ending_scene: \"Ending Scene\",\n editor_play: \"Play\",\n editor_pause: \"Pause\",\n editor_prev_frame: \"Previous Frame\",\n editor_next_frame: \"Next Frame\",\n editor_seek: \"Seek\",\n editor_no_media: \"No media assigned\",\n editor_uploading: \"Uploading\\u2026\",\n editor_drop_media: \"Drop media here\",\n editor_scene_type: \"Scene Type\",\n editor_save: \"Save\",\n editor_scene_color: \"Scene Color\",\n editor_ai_generate: \"Generate with AI\",\n editor_ai_generating: \"Generating\\u2026\",\n editor_ai_mood_question: \"What mood should this scene convey?\",\n editor_ai_style_question: \"What visual style do you want?\",\n editor_ai_prompt_question: \"Describe this scene\",\n editor_ai_accept: \"Accept\",\n editor_ai_reject: \"Reject\",\n editor_ai_retry: \"Try Again\",\n editor_ai_feedback_prompt: \"What should we change?\",\n editor_ai_credit_cost: \"1 credit per generation\",\n editor_ai_generating_scene: \"Scene generating\\u2026\",\n editor_ai_style_badge: \"Style\",\n editor_ai_failed: \"Generation failed. Please try again.\",\n editor_media_attached: \"This scene already has media attached.\",\n editor_remove_media: \"Remove Media\",\n constellation_no_coverbox: \"No cover\",\n hud_edit_control: \"Edit\",\n hud_delete_control: \"Delete\",\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"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@chuzi/shared",
3
- "version": "1.2.0",
3
+ "version": "1.2.2",
4
4
  "description": "Shared types, realm config, theme tokens, API client, input layer, and realm rendering primitives for CHUZI apps",
5
5
  "type": "module",
6
6
  "sideEffects": false,
@@ -74,13 +74,27 @@
74
74
  "three": ">=0.160.0"
75
75
  },
76
76
  "peerDependenciesMeta": {
77
- "react": { "optional": true },
78
- "react-dom": { "optional": true },
79
- "react-native": { "optional": true },
80
- "react-native-web": { "optional": true },
81
- "three": { "optional": true },
82
- "@react-three/fiber": { "optional": true },
83
- "@react-three/drei": { "optional": true }
77
+ "react": {
78
+ "optional": true
79
+ },
80
+ "react-dom": {
81
+ "optional": true
82
+ },
83
+ "react-native": {
84
+ "optional": true
85
+ },
86
+ "react-native-web": {
87
+ "optional": true
88
+ },
89
+ "three": {
90
+ "optional": true
91
+ },
92
+ "@react-three/fiber": {
93
+ "optional": true
94
+ },
95
+ "@react-three/drei": {
96
+ "optional": true
97
+ }
84
98
  },
85
99
  "devDependencies": {
86
100
  "@react-three/drei": "^9.122.0",