@novely/core 0.0.0-alpha.11 → 0.0.0-alpha.12

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/dist/index.d.ts CHANGED
@@ -117,15 +117,6 @@ type ActionProxyProvider<Characters extends Record<string, Character>> = {
117
117
  type DefaultActionProxyProvider = ActionProxyProvider<Record<string, Character>>;
118
118
  type GetActionParameters<T extends Capitalize<keyof DefaultActionProxyProvider>> = Parameters<DefaultActionProxyProvider[Uncapitalize<T>]>;
119
119
 
120
- interface LocalStorageStorageSettings {
121
- key: string;
122
- }
123
- interface Storage {
124
- get: () => Promise<StorageData>;
125
- set: (data: StorageData) => Promise<void>;
126
- }
127
- declare const localStorageStorage: (options: LocalStorageStorageSettings) => Storage;
128
-
129
120
  type Stored<T> = {
130
121
  subscribe: (cb: (value: T) => void) => () => void;
131
122
  update: (fn: (prev: T) => T) => void;
@@ -169,10 +160,9 @@ type Renderer = {
169
160
  };
170
161
  type RendererInit = {
171
162
  characters: Record<string, Character>;
172
- storage: Storage;
173
163
  set: (save: Save) => Promise<void>;
174
164
  restore: (save?: Save) => Promise<void>;
175
- save: (override?: boolean, type?: Save[2][1]) => Promise<void>;
165
+ save: (override?: boolean, type?: Save[2][1]) => void;
176
166
  newGame: () => void;
177
167
  stack: Stack;
178
168
  languages: string[];
@@ -186,6 +176,15 @@ type RendererInit = {
186
176
  $: Stored<StorageData>;
187
177
  };
188
178
 
179
+ interface LocalStorageStorageSettings {
180
+ key: string;
181
+ }
182
+ interface Storage {
183
+ get: () => Promise<StorageData>;
184
+ set: (data: StorageData) => Promise<void>;
185
+ }
186
+ declare const localStorageStorage: (options: LocalStorageStorageSettings) => Storage;
187
+
189
188
  interface NovelyInit<Languages extends string, Characters extends Record<string, Character<Languages>>, Inter extends ReturnType<SetupT9N<Languages>>> {
190
189
  /**
191
190
  * An array of languages supported by the game.
@@ -220,7 +219,7 @@ interface NovelyInit<Languages extends string, Characters extends Record<string,
220
219
  */
221
220
  singleSave?: boolean;
222
221
  }
223
- declare const novely: <Languages extends string, Characters extends Record<string, Character<Languages>>, Inter extends _novely_t9n.T9N<Languages, string, string>>({ characters, storage, renderer: createRenderer, initialScreen, t9n, languages, assetsPreload }: NovelyInit<Languages, Characters, Inter>) => Promise<{
222
+ declare const novely: <Languages extends string, Characters extends Record<string, Character<Languages>>, Inter extends _novely_t9n.T9N<Languages, string, string>>({ characters, storage, renderer: createRenderer, initialScreen, t9n, languages, assetsPreload }: NovelyInit<Languages, Characters, Inter>) => {
224
223
  withStory: (s: Story) => void;
225
224
  action: ActionProxyProvider<Characters>;
226
225
  render: () => void;
@@ -229,6 +228,6 @@ declare const novely: <Languages extends string, Characters extends Record<strin
229
228
  (): State;
230
229
  };
231
230
  t: (key: string) => (lang: (string & {}) | Languages, obj: Record<string, unknown>) => string;
232
- }>;
231
+ };
233
232
 
234
233
  export { ActionProxyProvider, AudioHandle, Character, CharacterHandle, CustomHandler, CustomHandlerGetResult, CustomHandlerGetResultDataFunction, DefaultActionProxyProvider, Emotions, GetActionParameters, Path, Renderer, RendererInit, RendererStore, Storage, StorageData, Stored, Story, Thenable, ValidAction, localStorageStorage, novely };
@@ -1,2 +1,2 @@
1
- "use strict";var Novely=(()=>{var Se=Object.create;var L=Object.defineProperty;var ve=Object.getOwnPropertyDescriptor;var he=Object.getOwnPropertyNames;var Ae=Object.getPrototypeOf,be=Object.prototype.hasOwnProperty;var Ce=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),De=(e,t)=>{for(var a in t)L(e,a,{get:t[a],enumerable:!0})},$=(e,t,a,o)=>{if(t&&typeof t=="object"||typeof t=="function")for(let s of he(t))!be.call(e,s)&&s!==a&&L(e,s,{get:()=>t[s],enumerable:!(o=ve(t,s))||o.enumerable});return e};var Ee=(e,t,a)=>(a=e!=null?Se(Ae(e)):{},$(t||!e||!e.__esModule?L(a,"default",{value:e,enumerable:!0}):a,e)),we=e=>$(L({},"__esModule",{value:!0}),e);var re=Ce((Je,te)=>{"use strict";var xe=function(t){return Pe(t)&&!Te(t)};function Pe(e){return!!e&&typeof e=="object"}function Te(e){var t=Object.prototype.toString.call(e);return t==="[object RegExp]"||t==="[object Date]"||Oe(e)}var Be=typeof Symbol=="function"&&Symbol.for,Me=Be?Symbol.for("react.element"):60103;function Oe(e){return e.$$typeof===Me}function Re(e){return Array.isArray(e)?[]:{}}function M(e,t){return t.clone!==!1&&t.isMergeableObject(e)?x(Re(e),e,t):e}function je(e,t,a){return e.concat(t).map(function(o){return M(o,a)})}function Le(e,t){if(!t.customMerge)return x;var a=t.customMerge(e);return typeof a=="function"?a:x}function Fe(e){return Object.getOwnPropertySymbols?Object.getOwnPropertySymbols(e).filter(function(t){return Object.propertyIsEnumerable.call(e,t)}):[]}function Z(e){return Object.keys(e).concat(Fe(e))}function ee(e,t){try{return t in e}catch{return!1}}function Ne(e,t){return ee(e,t)&&!(Object.hasOwnProperty.call(e,t)&&Object.propertyIsEnumerable.call(e,t))}function Ie(e,t,a){var o={};return a.isMergeableObject(e)&&Z(e).forEach(function(s){o[s]=M(e[s],a)}),Z(t).forEach(function(s){Ne(e,s)||(ee(e,s)&&a.isMergeableObject(t[s])?o[s]=Le(s,a)(e[s],t[s],a):o[s]=M(t[s],a))}),o}function x(e,t,a){a=a||{},a.arrayMerge=a.arrayMerge||je,a.isMergeableObject=a.isMergeableObject||xe,a.cloneUnlessOtherwiseSpecified=M;var o=Array.isArray(t),s=Array.isArray(e),p=o===s;return p?o?a.arrayMerge(e,t,a):Ie(e,t,a):M(t,a)}x.all=function(t,a){if(!Array.isArray(t))throw new Error("first argument should be an array");return t.reduce(function(o,s){return x(o,s,a)},{})};var Ge=x;te.exports=Ge});var He={};De(He,{localStorageStorage:()=>se,novely:()=>oe});var J=e=>(t,a)=>e[t](a),F=e=>typeof e=="number",G=e=>e===null,W=e=>typeof e=="string",z=e=>{let t=String.prototype.startsWith.bind(e);return t("http")||t("/")||t(".")||t("data")},k=e=>String(e),Q=(e,t)=>e==="custom"&&t[0]&&t[0].requireUserAction,N=()=>[[[null,"start"],[null,0]],{},[Date.now(),"auto"]],H=(e,t=navigator.language)=>e.includes(t)||e.includes(t=t.substring(0,2))||(t=e.find(a=>navigator.languages.includes(a)))?t:e[0],Y=(e,t)=>{let a=!1,o,s;function p(){if(a){o=arguments,s=this;return}e.apply(this,arguments),a=!1}return setTimeout(function(){a=!1,o&&(p.apply(s,o),o=s=null)},t),p};var X=(e,t=new Set)=>{let a=g=>(t.add(g),g(e),()=>{t.delete(g)}),o=g=>{t.forEach(A=>A(g))};return{subscribe:a,update:g=>{o(e=g(e))},get:()=>e}};var ue=Ee(re(),1);function D(e){var t,a,o;if(Array.isArray(e)){for(a=Array(t=e.length);t--;)a[t]=(o=e[t])&&typeof o=="object"?D(o):o;return a}if(Object.prototype.toString.call(e)==="[object Object]"){a={};for(t in e)t==="__proto__"?Object.defineProperty(a,t,{value:D(e[t]),configurable:!0,enumerable:!0,writable:!0}):a[t]=(o=e[t])&&typeof o=="object"?D(o):o;return a}return e}var ae=new Set(["dialog","input","vibrate"]);var ke=/{{(.*?)}}/g,ne=(e,t,a,o)=>e.replace(ke,(s,p,g)=>{s=0,g=t,p=p.trim();let A=p.split("@"),P;for(A.length>1&&([p,P]=A),p=p.split(".");g&&s<p.length;)g=g[p[s++]];return P&&a&&o&&g&&(g=a[P][o.select(g)]),g??""});var oe=async({characters:e,storage:t,renderer:a,initialScreen:o="mainmenu",t9n:s,languages:p,assetsPreload:g})=>{let A,P=r=>{A=Object.fromEntries(Object.entries(r).map(([u,n])=>{let c=l=>l.flatMap(v=>{let h=v[0];return Array.isArray(h)?c(v):[v]});return[u,c(n)]})),ge()},_={background:new Set},ie=new Proxy({},{get(r,u){return(...n)=>(u==="showBackground"&&z(n[0])&&_.background.add(n[0]),[u,...n])}});function U(r){if(!r)return i.value[1];let u=i.value[1],n=typeof r=="function"?r(u):(0,ue.all)([u,r]);i.value[1]=n}let ce=(r,u=[r])=>({get value(){return u.at(-1)},set value(n){u[u.length-1]=n},back(){u.length>1&&(u.pop(),E=!0)},push(n){u.push(n)},clear(){u=[N()]}}),V=await t.get();V.meta[0]||=H(p);let T=X(V),le=Y(r=>t.set(r),120);T.subscribe(le);let I=(r=>r.saves.length>0&&r.saves.at(-1))(T.get())||N(),i=ce(I),q=async(r=!1,u=r?"auto":"manual")=>{let n=await t.get(),c=i.value[2][0],l=n.saves.findIndex(v=>v[2][0]===c)===n.saves.length-1;return i.value[2][0]=Date.now(),i.value[2][1]=u,r&&l?n.saves[n.saves.length-1]=i.value:n.saves.push(i.value),await t.set(n)},me=()=>{T.update(r=>{let u=N();return r.saves.push(u),O(u),r})},fe=r=>(i.value=r,O(r)),S=!1,E=!1,O=async r=>{let u=r||await t.get().then(m=>m.saves.at(-1));u||(await t.set({saves:[I],meta:[H(p)]}),u=D(I)),S=!0,i.value=u,d.ui.showScreen("game"),B("clear",[E]);let n=A,c=0,l=i.value[0].reduce((m,[f,b])=>G(f)&&F(b)?m+1:m,0),v=[];for(let[m,f]of i.value[0])if(m===null){if(W(f))n=n[f];else if(F(f)){c++;for(let b=0;b<f;b++){let[C,...j]=n[b];if(ae.has(C)||Q(C,j))if(c===l&&b===f)v.push([C,j]);else continue;v.push([C,j])}n=n[f]}}else m==="choice"?n=n[f+1][1]:m==="condition"&&(n=n[2][f]);let h=v.map((m,f)=>m.concat(f));for await(let[m,f,b]of h)if(m==="function"||m==="custom"){if(m==="custom"&&f[0].callOnlyLatest&&!!h.slice(b+1).some(([Ve,ye])=>k(ye[0])===k(f[0])))continue;let C=B(m,f);C&&"then"in C&&await C}else B(m,f);S=!1,E=!1,w()},pe=()=>{let r=A;for(let[u,n]of i.value[0])u===null?r=r[n]:u==="choice"?r=r[n+1][1]:u==="condition"&&(r=r[2][n]);return r},d=a({characters:e,storage:t,set:fe,restore:O,save:q,newGame:me,stack:i,languages:p,t:s.i,$:T}),ge=()=>{if(!g){o==="game"?O():d.ui.showScreen(o);return}let r=[];d.ui.showScreen("loading");for(let[u,{emotions:n}]of Object.entries(e))for(let c of Object.keys(n))r.push(Promise.resolve(d.character(u).withEmotion(c)));for(let u of _.background)r.push(new Promise((n,c)=>{let l=document.createElement("img");l.crossOrigin="*",l.src=u,l.complete&&l.naturalHeight!==0?n(void 0):(l.onload=n,l.onerror=c)}));Promise.all(r).then(()=>{d.ui.showScreen(o)})},B=J({wait([r]){S||setTimeout(y,r)},showBackground([r]){d.background(r),y()},playMusic([r]){d.music(r,"music").play(),y()},stopMusic([r]){d.music(r,"music").stop(),y()},showCharacter([r,u,n,c]){let l=d.character(r);l.append(n,c),l.withEmotion(u)(),y()},hideCharacter([r,u,n,c]){d.character(r).remove(u,n,c)(y)},dialog([r,u,n]){d.dialog(K(u),r,n)(()=>{R(),y()})},function([r]){let u=r();return S||(u?u.then(y):y()),u},choice(r){let u=r.map(([n,c,l])=>[K(n),c,l]);d.choices(u)(n=>{R(),i.value[0].push(["choice",n],[null,0]),w()})},jump([r]){i.value[0]=[[null,r],[null,0]],d.clear(!1)(()=>{S||w()})},clear(){try{navigator.vibrate(0)}finally{d.clear(E)(y)}},condition([r]){let u=r();S||(i.value[0].push(["condition",u],[null,0]),w())},end(){q(!1,"auto").then(()=>{B("clear",[]),d.ui.showScreen("mainmenu")})},input([r,u,n]){d.input(r,u,n)(()=>{R(),y()})},custom([r]){return d.custom(r,E,()=>{!S&&r.requireUserAction&&R(),S||y()})},vibrate(r){try{navigator.vibrate(r)}finally{y()}},next(){y()},animateCharacter([r,u,...n]){let c=l=>{let h=l("novely-animate-character",!1).root.querySelector(`div[data-characters] > canvas[data-character="${r}"]`);if(!h)return;let m=n.filter(f=>!h.classList.contains(f));h.classList.add(...m),setTimeout(()=>{h.classList.remove(...m)},u)};return c.callOnlyLatest=!0,d.custom(c,E,()=>{}),y()}}),R=()=>{if(S)return;let r=D(i.value);r[0]=D(i.value[0]),r[2][0]=new Date().valueOf(),r[2][1]="auto",i.push(r)},de=()=>{let r=i.value[0][i.value[0].length-1];if(G(r[0])&&F(r[1])){r[1]=r[1]+1;return}i.value[0].push([null,0])},w=()=>{let r=pe();if(r){let[u,...n]=r;B(u,n)}},y=()=>{S||(de(),w())},K=r=>{let u=T.get().meta[0],n=U();return ne(typeof r=="function"?r(u,n):r,n)};return{withStory:P,action:ie,render:w,state:U,t:s.t}};var se=e=>({async get(){let t=localStorage.getItem(e.key);return t?JSON.parse(t):{saves:[],meta:[]}},async set(t){localStorage.setItem(e.key,JSON.stringify(t))}});return we(He);})();
1
+ "use strict";var Novely=(()=>{var ve=Object.create;var j=Object.defineProperty;var he=Object.getOwnPropertyDescriptor;var Ae=Object.getOwnPropertyNames;var be=Object.getPrototypeOf,Ce=Object.prototype.hasOwnProperty;var De=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),Ee=(e,t)=>{for(var a in t)j(e,a,{get:t[a],enumerable:!0})},$=(e,t,a,o)=>{if(t&&typeof t=="object"||typeof t=="function")for(let s of Ae(t))!Ce.call(e,s)&&s!==a&&j(e,s,{get:()=>t[s],enumerable:!(o=he(t,s))||o.enumerable});return e};var xe=(e,t,a)=>(a=e!=null?ve(be(e)):{},$(t||!e||!e.__esModule?j(a,"default",{value:e,enumerable:!0}):a,e)),we=e=>$(j({},"__esModule",{value:!0}),e);var re=De((We,te)=>{"use strict";var Pe=function(t){return Te(t)&&!Be(t)};function Te(e){return!!e&&typeof e=="object"}function Be(e){var t=Object.prototype.toString.call(e);return t==="[object RegExp]"||t==="[object Date]"||Re(e)}var Me=typeof Symbol=="function"&&Symbol.for,Oe=Me?Symbol.for("react.element"):60103;function Re(e){return e.$$typeof===Oe}function Le(e){return Array.isArray(e)?[]:{}}function M(e,t){return t.clone!==!1&&t.isMergeableObject(e)?P(Le(e),e,t):e}function je(e,t,a){return e.concat(t).map(function(o){return M(o,a)})}function Fe(e,t){if(!t.customMerge)return P;var a=t.customMerge(e);return typeof a=="function"?a:P}function Ne(e){return Object.getOwnPropertySymbols?Object.getOwnPropertySymbols(e).filter(function(t){return Object.propertyIsEnumerable.call(e,t)}):[]}function Z(e){return Object.keys(e).concat(Ne(e))}function ee(e,t){try{return t in e}catch{return!1}}function Ie(e,t){return ee(e,t)&&!(Object.hasOwnProperty.call(e,t)&&Object.propertyIsEnumerable.call(e,t))}function Ge(e,t,a){var o={};return a.isMergeableObject(e)&&Z(e).forEach(function(s){o[s]=M(e[s],a)}),Z(t).forEach(function(s){Ie(e,s)||(ee(e,s)&&a.isMergeableObject(t[s])?o[s]=Fe(s,a)(e[s],t[s],a):o[s]=M(t[s],a))}),o}function P(e,t,a){a=a||{},a.arrayMerge=a.arrayMerge||je,a.isMergeableObject=a.isMergeableObject||Pe,a.cloneUnlessOtherwiseSpecified=M;var o=Array.isArray(t),s=Array.isArray(e),m=o===s;return m?o?a.arrayMerge(e,t,a):Ge(e,t,a):M(t,a)}P.all=function(t,a){if(!Array.isArray(t))throw new Error("first argument should be an array");return t.reduce(function(o,s){return P(o,s,a)},{})};var ke=P;te.exports=ke});var _e={};Ee(_e,{localStorageStorage:()=>se,novely:()=>oe});var J=e=>(t,a)=>e[t](a),F=e=>typeof e=="number",k=e=>e===null,W=e=>typeof e=="string",z=e=>{let t=String.prototype.startsWith.bind(e);return t("http")||t("/")||t(".")||t("data")},H=e=>String(e),Q=(e,t)=>e==="custom"&&t[0]&&t[0].requireUserAction,N=()=>[[[null,"start"],[null,0]],{},[Date.now(),"auto"]],I=(e,t=navigator.language)=>e.includes(t)||e.includes(t=t.substring(0,2))||(t=e.find(a=>navigator.languages.includes(a)))?t:e[0],Y=(e,t)=>{let a=!1,o,s;function m(){if(a){o=arguments,s=this;return}e.apply(this,arguments),a=!1}return setTimeout(function(){a=!1,o&&(m.apply(s,o),o=s=null)},t),m};var X=(e,t=new Set)=>{let a=p=>(t.add(p),p(e),()=>{t.delete(p)}),o=p=>{t.forEach(b=>b(p))};return{subscribe:a,update:p=>{o(e=p(e))},get:()=>e}};var ue=xe(re(),1);function E(e){var t,a,o;if(Array.isArray(e)){for(a=Array(t=e.length);t--;)a[t]=(o=e[t])&&typeof o=="object"?E(o):o;return a}if(Object.prototype.toString.call(e)==="[object Object]"){a={};for(t in e)t==="__proto__"?Object.defineProperty(a,t,{value:E(e[t]),configurable:!0,enumerable:!0,writable:!0}):a[t]=(o=e[t])&&typeof o=="object"?E(o):o;return a}return e}var ae=new Set(["dialog","input","vibrate"]);var He=/{{(.*?)}}/g,ne=(e,t,a,o)=>e.replace(He,(s,m,p)=>{s=0,p=t,m=m.trim();let b=m.split("@"),T;for(b.length>1&&([m,T]=b),m=m.split(".");p&&s<m.length;)p=p[m[s++]];return T&&a&&o&&p&&(p=a[T][o.select(p)]),p??""});var oe=({characters:e,storage:t,renderer:a,initialScreen:o="mainmenu",t9n:s,languages:m,assetsPreload:p})=>{let b,T=r=>{b=Object.fromEntries(Object.entries(r).map(([u,n])=>{let c=l=>l.flatMap(v=>{let A=v[0];return Array.isArray(A)?c(v):[v]});return[u,c(n)]})),de()},_={background:new Set},ie=new Proxy({},{get(r,u){return(...n)=>(u==="showBackground"&&z(n[0])&&_.background.add(n[0]),[u,...n])}});function U(r){if(!r)return i.value[1];let u=i.value[1],n=typeof r=="function"?r(u):(0,ue.all)([u,r]);i.value[1]=n}let ce=(r,u=[r])=>({get value(){return u.at(-1)},set value(n){u[u.length-1]=n},back(){u.length>1&&(u.pop(),x=!0)},push(n){u.push(n)},clear(){u=[N()]}}),le={saves:[],meta:[I(m)]},h=X(le),V=!1,me=Y(r=>{V&&t.set(r)},120);h.subscribe(me),t.get().then(r=>{r.meta[0]||=I(m),V=!0,h.update(()=>r)});let G=(r=>r.saves.length>0&&r.saves.at(-1))(h.get())||N(),i=ce(G),q=(r=!1,u=r?"auto":"manual")=>{h.update(n=>{let c=i.value[2][0],l=n.saves.findIndex(v=>v[2][0]===c)===n.saves.length-1;return i.value[2][0]=Date.now(),i.value[2][1]=u,r&&l?n.saves[n.saves.length-1]=i.value:n.saves.push(i.value),n})},fe=()=>{h.update(r=>{let u=N();return r.saves.push(u),O(u),r})},pe=r=>(i.value=r,O(r)),y=!1,x=!1,O=async r=>{let u=r||h.get().saves.at(-1);u||(h.update(()=>({saves:[G],meta:[I(m)]})),u=E(G)),y=!0,i.value=u,g.ui.showScreen("game"),B("clear",[x]);let n=b,c=0,l=i.value[0].reduce((d,[f,C])=>k(f)&&F(C)?d+1:d,0),v=[];for(let[d,f]of i.value[0])if(d===null){if(W(f))n=n[f];else if(F(f)){c++;for(let C=0;C<f;C++){let[D,...L]=n[C];if(ae.has(D)||Q(D,L))if(c===l&&C===f)v.push([D,L]);else continue;v.push([D,L])}n=n[f]}}else d==="choice"?n=n[f+1][1]:d==="condition"&&(n=n[2][f]);let A=v.map((d,f)=>d.concat(f));for await(let[d,f,C]of A)if(d==="function"||d==="custom"){if(d==="custom"&&f[0].callOnlyLatest&&!!A.slice(C+1).some(([qe,ye])=>H(ye[0])===H(f[0])))continue;let D=B(d,f);D&&"then"in D&&await D}else B(d,f);y=!1,x=!1,w()},ge=()=>{let r=b;for(let[u,n]of i.value[0])u===null?r=r[n]:u==="choice"?r=r[n+1][1]:u==="condition"&&(r=r[2][n]);return r},g=a({characters:e,set:pe,restore:O,save:q,newGame:fe,stack:i,languages:m,t:s.i,$:h}),de=()=>{if(!p){o==="game"?O():g.ui.showScreen(o);return}let r=[];g.ui.showScreen("loading");for(let[u,{emotions:n}]of Object.entries(e))for(let c of Object.keys(n))r.push(Promise.resolve(g.character(u).withEmotion(c)));for(let u of _.background)r.push(new Promise((n,c)=>{let l=document.createElement("img");l.crossOrigin="*",l.src=u,l.complete&&l.naturalHeight!==0?n(void 0):(l.onload=n,l.onerror=c)}));Promise.all(r).then(()=>{g.ui.showScreen(o)})},B=J({wait([r]){y||setTimeout(S,r)},showBackground([r]){g.background(r),S()},playMusic([r]){g.music(r,"music").play(),S()},stopMusic([r]){g.music(r,"music").stop(),S()},showCharacter([r,u,n,c]){let l=g.character(r);l.append(n,c),l.withEmotion(u)(),S()},hideCharacter([r,u,n,c]){g.character(r).remove(u,n,c)(S)},dialog([r,u,n]){g.dialog(K(u),r,n)(()=>{R(),S()})},function([r]){let u=r();return y||(u?u.then(S):S()),u},choice(r){let u=r.map(([n,c,l])=>[K(n),c,l]);g.choices(u)(n=>{R(),i.value[0].push(["choice",n],[null,0]),w()})},jump([r]){i.value[0]=[[null,r],[null,0]],g.clear(!1)(()=>{y||w()})},clear(){try{navigator.vibrate(0)}finally{g.clear(x)(S)}},condition([r]){let u=r();y||(i.value[0].push(["condition",u],[null,0]),w())},end(){q(!1,"auto"),B("clear",[]),g.ui.showScreen("mainmenu")},input([r,u,n]){g.input(r,u,n)(()=>{R(),S()})},custom([r]){return g.custom(r,x,()=>{!y&&r.requireUserAction&&R(),y||S()})},vibrate(r){try{navigator.vibrate(r)}finally{S()}},next(){S()},animateCharacter([r,u,...n]){let c=l=>{let A=l("novely-animate-character",!1).root.querySelector(`div[data-characters] > canvas[data-character="${r}"]`);if(!A)return;let d=n.filter(f=>!A.classList.contains(f));A.classList.add(...d),setTimeout(()=>{A.classList.remove(...d)},u)};return c.callOnlyLatest=!0,g.custom(c,x,()=>{}),S()}}),R=()=>{if(y)return;let r=E(i.value);r[0]=E(i.value[0]),r[2][0]=new Date().valueOf(),r[2][1]="auto",i.push(r)},Se=()=>{let r=i.value[0][i.value[0].length-1];if(k(r[0])&&F(r[1])){r[1]=r[1]+1;return}i.value[0].push([null,0])},w=()=>{let r=ge();if(r){let[u,...n]=r;B(u,n)}},S=()=>{y||(Se(),w())},K=r=>{let u=h.get().meta[0],n=U();return ne(typeof r=="function"?r(u,n):r,n)};return{withStory:T,action:ie,render:w,state:U,t:s.t}};var se=e=>({async get(){let t=localStorage.getItem(e.key);return t?JSON.parse(t):{saves:[],meta:[]}},async set(t){localStorage.setItem(e.key,JSON.stringify(t))}});return we(_e);})();
2
2
  //# sourceMappingURL=index.global.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../node_modules/.pnpm/deepmerge@4.3.0/node_modules/deepmerge/dist/cjs.js","../src/index.ts","../src/utils.ts","../src/store.ts","../src/novely.ts","../../../node_modules/.pnpm/klona@2.0.6/node_modules/klona/json/index.mjs","../src/constants.ts","../../t9n/src/translation.ts","../../t9n/src/translations.ts","../src/storage.ts"],"sourcesContent":["'use strict';\n\nvar isMergeableObject = function isMergeableObject(value) {\n\treturn isNonNullObject(value)\n\t\t&& !isSpecial(value)\n};\n\nfunction isNonNullObject(value) {\n\treturn !!value && typeof value === 'object'\n}\n\nfunction isSpecial(value) {\n\tvar stringValue = Object.prototype.toString.call(value);\n\n\treturn stringValue === '[object RegExp]'\n\t\t|| stringValue === '[object Date]'\n\t\t|| isReactElement(value)\n}\n\n// see https://github.com/facebook/react/blob/b5ac963fb791d1298e7f396236383bc955f916c1/src/isomorphic/classic/element/ReactElement.js#L21-L25\nvar canUseSymbol = typeof Symbol === 'function' && Symbol.for;\nvar REACT_ELEMENT_TYPE = canUseSymbol ? Symbol.for('react.element') : 0xeac7;\n\nfunction isReactElement(value) {\n\treturn value.$$typeof === REACT_ELEMENT_TYPE\n}\n\nfunction emptyTarget(val) {\n\treturn Array.isArray(val) ? [] : {}\n}\n\nfunction cloneUnlessOtherwiseSpecified(value, options) {\n\treturn (options.clone !== false && options.isMergeableObject(value))\n\t\t? deepmerge(emptyTarget(value), value, options)\n\t\t: value\n}\n\nfunction defaultArrayMerge(target, source, options) {\n\treturn target.concat(source).map(function(element) {\n\t\treturn cloneUnlessOtherwiseSpecified(element, options)\n\t})\n}\n\nfunction getMergeFunction(key, options) {\n\tif (!options.customMerge) {\n\t\treturn deepmerge\n\t}\n\tvar customMerge = options.customMerge(key);\n\treturn typeof customMerge === 'function' ? customMerge : deepmerge\n}\n\nfunction getEnumerableOwnPropertySymbols(target) {\n\treturn Object.getOwnPropertySymbols\n\t\t? Object.getOwnPropertySymbols(target).filter(function(symbol) {\n\t\t\treturn Object.propertyIsEnumerable.call(target, symbol)\n\t\t})\n\t\t: []\n}\n\nfunction getKeys(target) {\n\treturn Object.keys(target).concat(getEnumerableOwnPropertySymbols(target))\n}\n\nfunction propertyIsOnObject(object, property) {\n\ttry {\n\t\treturn property in object\n\t} catch(_) {\n\t\treturn false\n\t}\n}\n\n// Protects from prototype poisoning and unexpected merging up the prototype chain.\nfunction propertyIsUnsafe(target, key) {\n\treturn propertyIsOnObject(target, key) // Properties are safe to merge if they don't exist in the target yet,\n\t\t&& !(Object.hasOwnProperty.call(target, key) // unsafe if they exist up the prototype chain,\n\t\t\t&& Object.propertyIsEnumerable.call(target, key)) // and also unsafe if they're nonenumerable.\n}\n\nfunction mergeObject(target, source, options) {\n\tvar destination = {};\n\tif (options.isMergeableObject(target)) {\n\t\tgetKeys(target).forEach(function(key) {\n\t\t\tdestination[key] = cloneUnlessOtherwiseSpecified(target[key], options);\n\t\t});\n\t}\n\tgetKeys(source).forEach(function(key) {\n\t\tif (propertyIsUnsafe(target, key)) {\n\t\t\treturn\n\t\t}\n\n\t\tif (propertyIsOnObject(target, key) && options.isMergeableObject(source[key])) {\n\t\t\tdestination[key] = getMergeFunction(key, options)(target[key], source[key], options);\n\t\t} else {\n\t\t\tdestination[key] = cloneUnlessOtherwiseSpecified(source[key], options);\n\t\t}\n\t});\n\treturn destination\n}\n\nfunction deepmerge(target, source, options) {\n\toptions = options || {};\n\toptions.arrayMerge = options.arrayMerge || defaultArrayMerge;\n\toptions.isMergeableObject = options.isMergeableObject || isMergeableObject;\n\t// cloneUnlessOtherwiseSpecified is added to `options` so that custom arrayMerge()\n\t// implementations can use it. The caller may not replace it.\n\toptions.cloneUnlessOtherwiseSpecified = cloneUnlessOtherwiseSpecified;\n\n\tvar sourceIsArray = Array.isArray(source);\n\tvar targetIsArray = Array.isArray(target);\n\tvar sourceAndTargetTypesMatch = sourceIsArray === targetIsArray;\n\n\tif (!sourceAndTargetTypesMatch) {\n\t\treturn cloneUnlessOtherwiseSpecified(source, options)\n\t} else if (sourceIsArray) {\n\t\treturn options.arrayMerge(target, source, options)\n\t} else {\n\t\treturn mergeObject(target, source, options)\n\t}\n}\n\ndeepmerge.all = function deepmergeAll(array, options) {\n\tif (!Array.isArray(array)) {\n\t\tthrow new Error('first argument should be an array')\n\t}\n\n\treturn array.reduce(function(prev, next) {\n\t\treturn deepmerge(prev, next, options)\n\t}, {})\n};\n\nvar deepmerge_1 = deepmerge;\n\nmodule.exports = deepmerge_1;\n","export type { ValidAction, Story, ActionProxyProvider, DefaultActionProxyProvider, GetActionParameters, CustomHandler, CustomHandlerGetResult, CustomHandlerGetResultDataFunction } from './action'\nexport type { Emotions, Character } from './character'\nexport type { CharacterHandle, AudioHandle, RendererStore, Renderer, RendererInit } from './renderer'\nexport type { Storage } from './storage'\nexport type { Thenable, Path, StorageData } from './types'\nexport type { Stored } from './store'\nexport type { BaseTranslationStrings } from '@novely/t9n';\n\nexport { novely } from './novely'\nexport { localStorageStorage } from './storage'","import type { ActionProxyProvider, CustomHandler } from './action'\nimport type { Character } from './character'\nimport type { Save, Thenable } from './types'\n\ntype MatchActionMap = {\n [Key in keyof ActionProxyProvider<Record<string, Character>>]: (data: Parameters<ActionProxyProvider<Record<string, Character>>[Key]>) => void;\n}\n\ntype MatchActionMapComplete = Omit<MatchActionMap, 'custom'> & {\n custom: (value: [handler: CustomHandler]) => Thenable<void>;\n}\n\nconst matchAction = <M extends MatchActionMapComplete>(values: M) => {\n return (action: keyof MatchActionMap, props: any) => {\n return values[action](props);\n }\n}\n\nconst isNumber = (val: unknown): val is number => {\n return typeof val === 'number';\n}\n\nconst isNull = (val: unknown): val is null => {\n return val === null;\n}\n\nconst isString = (val: unknown): val is string => {\n return typeof val === 'string';\n}\n\nconst isCSSImage = (str: string) => {\n const startsWith = String.prototype.startsWith.bind(str);\n\n return startsWith('http') || startsWith('/') || startsWith('.') || startsWith('data');\n}\n\nconst str = (value: unknown) => {\n return String(value);\n}\n\nconst isUserRequiredAction = (action: keyof MatchActionMapComplete, meta: Parameters<MatchActionMapComplete[keyof MatchActionMapComplete]>) => {\n return action === 'custom' && meta[0] && (meta[0] as unknown as CustomHandler).requireUserAction;\n}\n\nconst getDefaultSave = () => {\n return [[[null, 'start'], [null, 0]], {}, [Date.now(), 'auto']] as Save;\n}\n\nconst getLanguage = (languages: string[], language = navigator.language) => {\n if (languages.includes(language)) {\n return language;\n } else if (languages.includes((language = language.substring(0, 2)))) {\n return language;\n } else if ((language = languages.find((value) => navigator.languages.includes(value))!)) {\n return language\n }\n\n /**\n * We'v checked the `en-GB` format, `en` format, and maybe any second languages, but there were no matches\n */\n return languages[0];\n}\n\n/**\n * @copyright Techlead LLC\n * @see https://learn.javascript.ru/task/throttle\n */\nconst throttle = <Fn extends ((...args: any[]) => any)>(fn: Fn, ms: number) => {\n let throttled = false, savedArgs: any, savedThis: any;\n\n function wrapper() {\n if (throttled) {\n savedArgs = arguments;\n // @ts-ignore\n savedThis = this;\n\n return;\n }\n\n // @ts-ignore\n fn.apply(this, arguments);\n\n throttled = false;\n }\n\n setTimeout(function () {\n throttled = false;\n\n if (savedArgs) {\n wrapper.apply(savedThis, savedArgs);\n savedArgs = savedThis = null;\n }\n }, ms);\n\n return wrapper as unknown as (...args: Parameters<Fn>) => void;\n}\n\nexport { matchAction, isNumber, isNull, isString, isCSSImage, str, isUserRequiredAction, getDefaultSave, getLanguage, throttle }","type Stored<T> = {\n subscribe: (cb: (value: T) => void) => () => void;\n update: (fn: (prev: T) => T) => void;\n get: () => T;\n}\n\nconst store = <T>(current: T, subscribers = new Set<(value: T) => void>()): Stored<T> => {\n const subscribe = (cb: (value: T) => void) => {\n subscribers.add(cb), cb(current);\n\n return () => {\n subscribers.delete(cb);\n }\n };\n\n const push = (value: T) => {\n subscribers.forEach((cb) => cb(value))\n };\n\n const update = (fn: (prev: T) => T) => {\n push((current = fn(current)));\n };\n\n const get = () => {\n return current;\n };\n\n return { subscribe, update, get } as const;\n};\n\nexport { store }\nexport type { Stored }","import type { Character } from './character';\nimport type { ActionProxyProvider, GetActionParameters, Story, ValidAction, DialogContent, ChoiceContent, CustomHandler } from './action';\nimport type { Storage } from './storage';\nimport type { Save, State, StorageData } from './types'\nimport type { Renderer, RendererInit } from './renderer'\nimport type { SetupT9N } from '@novely/t9n'\nimport { matchAction, isNumber, isNull, isString, isCSSImage, str, isUserRequiredAction, getDefaultSave, getLanguage, throttle } from './utils';\nimport { store } from './store';\nimport { all as deepmerge } from 'deepmerge'\nimport { klona } from 'klona/json';\nimport { SKIPPED_DURING_RESTORE } from './constants';\nimport { replace as replaceT9N } from '@novely/t9n';\n\ninterface NovelyInit<Languages extends string, Characters extends Record<string, Character<Languages>>, Inter extends ReturnType<SetupT9N<Languages>>> {\n /**\n * An array of languages supported by the game.\n */\n languages: Languages[];\n /**\n * An object containing the characters in the game.\n */\n characters: Characters;\n /**\n * An object that provides access to the game's storage system.\n */\n storage: Storage;\n /**\n * A function that returns a Renderer object used to display the game's content\n */\n renderer: (characters: RendererInit) => Renderer;\n /**\n * An optional property that specifies the initial screen to display when the game starts\n */\n initialScreen?: \"mainmenu\" | \"game\" | \"saves\" | \"settings\";\n /**\n * An object containing the translation functions used in the game\n */\n t9n: Inter;\n /**\n * An optional property that specifies whether to preload assets when the game starts\n */\n assetsPreload?: boolean;\n /**\n * An optional property that specifies whether the game should use a single save.\n */\n singleSave?: boolean;\n}\n\nconst novely = async <Languages extends string, Characters extends Record<string, Character<Languages>>, Inter extends ReturnType<SetupT9N<Languages>>>({ characters, storage, renderer: createRenderer, initialScreen = \"mainmenu\", t9n, languages, assetsPreload }: NovelyInit<Languages, Characters, Inter>) => {\n let story: Story;\n\n const withStory = (s: Story) => {\n /**\n * Transforms `(ValidAction | ValidAction[])[]` to `ValidAction[]`\n */\n story = Object.fromEntries(Object.entries(s).map(([name, items]) => {\n const flat = (item: (ValidAction | ValidAction[])[]): ValidAction[] => {\n return item.flatMap((data) => {\n const type = data[0];\n\n /**\n * This is not just an action like `['name', ...arguments]`, but an array of actions\n */\n if (Array.isArray(type)) return flat(data as ValidAction[]);\n\n return [data as ValidAction];\n });\n };\n\n return [name, flat(items)];\n }));\n\n /**\n * Load assets after the `action` scripts are executed\n */\n preloadAssets();\n }\n\n /**\n * This is used when background is loading\n */\n const preload = {\n background: new Set<string>(),\n }\n\n const action = new Proxy({} as ActionProxyProvider<Characters>, {\n get(_, prop) {\n return (...props: Parameters<ActionProxyProvider<Record<string, Character>>[keyof ActionProxyProvider<Record<string, Character>>]>) => {\n if (prop === 'showBackground') {\n if (isCSSImage(props[0] as string)) preload.background.add(props[0] as string);\n }\n\n return [prop, ...props];\n }\n }\n });\n\n function state(value: State | ((prev: State) => State)): void;\n function state(): State;\n function state(value?: State | ((prev: State) => State)): State | void {\n if (!value) return stack.value[1]\n\n const prev = stack.value[1];\n const val = typeof value === 'function' ? value(prev as State) : deepmerge([prev, value]);\n\n stack.value[1] = val as State;\n }\n\n const createStack = (current: Save, stack = [current]) => {\n return {\n get value() {\n return stack.at(-1)!;\n },\n set value(value: Save) {\n stack[stack.length - 1] = value;\n },\n back() {\n if (stack.length > 1) stack.pop(), goingBack = true;\n },\n push(value: Save) {\n stack.push(value);\n },\n clear() {\n stack = [getDefaultSave()];\n }\n }\n }\n\n const stored = await storage.get();\n\n /**\n * Default `localStorageStorage` cannot determine preferred language, and returns empty array\n */\n stored.meta[0] ||= getLanguage(languages);\n\n const $ = store(stored);\n\n const onStorageDataChange = (value: StorageData) => storage.set(value);\n const throttledOnStorageDataChange = throttle(onStorageDataChange, 120);\n\n $.subscribe(throttledOnStorageDataChange);\n\n const initial = ((value) => value.saves.length > 0 && value.saves.at(-1))($.get()) || getDefaultSave();\n const stack = createStack(initial);\n\n const save = async (override = false, type: Save[2][1] = override ? 'auto' : 'manual') => {\n /**\n * Получаем предыдущее значение\n */\n const prev = await storage.get();\n\n const date = stack.value[2][0];\n const isLatest = prev.saves.findIndex(value => value[2][0] === date) === prev.saves.length - 1;\n\n /**\n * Обновим дату и тип\n */\n stack.value[2][0] = Date.now();\n stack.value[2][1] = type;\n\n if (override) {\n /**\n * Перезапишем\n */\n if (isLatest) {\n /**\n * Сохранения хранятся в массиве. Нельзя перезаписать любое последнее\n * \n * Если перезаписывать старое сохранение, то они не будут идти в хронологическом порядке\n */\n prev.saves[prev.saves.length - 1] = stack.value;\n } else {\n prev.saves.push(stack.value);\n }\n } else {\n /**\n * Добавляем текущее сохранение\n */\n prev.saves.push(stack.value);\n }\n\n /**\n * Устанавливаем новое значение\n */\n return await storage.set(prev);\n }\n\n const newGame = () => {\n $.update(prev => {\n const save = getDefaultSave();\n\n prev.saves.push(save), restore(save);\n\n return prev;\n });\n }\n\n /**\n * Устанавливает сохранение\n */\n const set = (save: Save) => {\n stack.value = save;\n\n return restore(save);\n }\n\n let restoring = false;\n let goingBack = false;\n\n /**\n * Визуально восстанавливает историю\n */\n const restore = async (save?: Save) => {\n let latest = save ? save : await storage.get().then(value => value.saves.at(-1));\n\n /**\n * Если нет сохранённой игры, то запустим ту, которая уже есть\n */\n if (!latest) {\n await storage.set({ saves: [initial], meta: [getLanguage(languages)] });\n\n latest = klona(initial);\n }\n\n restoring = true, stack.value = latest;\n\n /**\n * Показать экран игры\n */\n renderer.ui.showScreen('game');\n\n match('clear', [goingBack]);\n\n /**\n * Текущий элемент в истории\n */\n let current: any = story;\n /**\n * Текущий элемент `[null, int]`\n */\n let index = 0;\n\n /**\n * Число элементов `[null, int]`\n */\n const max = stack.value[0].reduce((acc, [type, val]) => {\n if (isNull(type) && isNumber(val)) return acc + 1;\n\n return acc;\n }, 0);\n\n const queue = [] as [any, any][];\n\n for (const [type, val] of stack.value[0]) {\n if (type === null) {\n if (isString(val)) {\n current = current[val];\n } else if (isNumber(val)) {\n index++;\n\n /**\n * Запустим все экшены которые идут в `[null, int]` от `0` до `int`\n */\n for (let i = 0; i < val; i++) {\n const [action, ...meta] = current[i];\n\n /**\n * Экшены, для закрытия которых пользователь должен с ними взаимодействовать\n * Также в эту группу входят экшены, которые не должны быть вызваны при восстановлении\n */\n if (SKIPPED_DURING_RESTORE.has(action) || isUserRequiredAction(action, meta)) {\n if (index === max && i === val) {\n queue.push([action, meta]);\n } else {\n continue;\n }\n }\n\n queue.push([action, meta]);\n }\n\n current = current[val];\n }\n } else if (type === 'choice') {\n current = current[val as number + 1][1];\n } else if (type === 'condition') {\n current = current[2][val];\n }\n }\n\n const indexedQueue = queue.map((value, index) => value.concat(index) as [ValidAction[0], ValidAction[1], number]);\n\n for await (const [action, meta, i] of indexedQueue) {\n if (action === 'function' || action === 'custom') {\n /**\n * Если `callOnlyLatest` - `true`\n */\n if (action === 'custom' && (meta as GetActionParameters<'Custom'>)[0].callOnlyLatest) {\n /**\n * Вычислим `latest` или нет\n */\n const next = indexedQueue.slice(i + 1);\n const latest = !next.some(([_action, _meta]) => str(_meta[0]) === str(meta[0]));\n\n if (!latest) continue;\n }\n\n /**\n * Action может возвращать Promise. Нужно подожать его `resolve`\n */\n const result = match(action, meta);\n\n /**\n * Дождёмся окончания\n */\n if (result && 'then' in result) await result;\n } else {\n match(action as keyof ActionProxyProvider<Record<string, Character<string>>>, meta);\n }\n }\n\n restoring = false, goingBack = false, render();\n }\n\n const refer = () => {\n let current: any = story;\n\n for (const [type, val] of stack.value[0]) {\n if (type === null) {\n current = current[val];\n } else if (type === 'choice') {\n current = current[val as number + 1][1];\n } else if (type === 'condition') {\n current = current[2][val];\n }\n }\n\n return current;\n }\n\n const renderer = createRenderer({\n characters,\n storage,\n set,\n restore,\n save,\n newGame,\n stack,\n languages,\n t: t9n.i,\n $\n });\n\n const preloadAssets = () => {\n if (!assetsPreload) {\n initialScreen === 'game' ? restore() : renderer.ui.showScreen(initialScreen);\n\n return;\n }\n\n /**\n * We need to load all the characters and their emotions\n */\n\n let promises: Promise<unknown>[] = [];\n\n renderer.ui.showScreen('loading');\n\n for (const [name, { emotions }] of Object.entries(characters)) {\n for (const emotion of Object.keys(emotions)) {\n promises.push(Promise.resolve(renderer.character(name).withEmotion(emotion)));\n }\n }\n\n for (const bg of preload.background) {\n promises.push(new Promise((res, rej) => {\n /**\n * Create `img` element\n */\n const img = document.createElement('img');\n\n /**\n * Set `src`\n */\n img.crossOrigin = '*';\n img.src = bg;\n\n if (img.complete && img.naturalHeight !== 0) {\n /**\n * Image is already loaded\n */\n res(void 0);\n } else {\n /**\n * Image is uniq, it is safe to use `onload`\n */\n img.onload = res;\n img.onerror = rej;\n }\n }));\n }\n\n Promise.all(promises).then(() => {\n renderer.ui.showScreen(initialScreen);\n });\n }\n\n const match = matchAction({\n wait([time]) {\n /**\n * `restoring` может поменяться на `true` перед тем как запуститься `push` из `setTimeout`\n */\n if (!restoring) setTimeout(push, time);\n },\n showBackground([background]) {\n renderer.background(background);\n push()\n },\n playMusic([source]) {\n renderer.music(source, 'music').play();\n push()\n },\n stopMusic([source]) {\n renderer.music(source, 'music').stop();\n push()\n },\n showCharacter([character, emotion, className, style]) {\n const handle = renderer.character(character);\n\n handle.append(className, style);\n handle.withEmotion(emotion)();\n\n push()\n },\n hideCharacter([character, className, style, duration]) {\n const handle = renderer.character(character);\n\n handle.remove(className, style, duration)(push);\n },\n dialog([person, content, emotion]) {\n renderer.dialog(unwrap(content), person, emotion)(() => {\n enmemory();\n push();\n });\n },\n function([fn]) {\n const result = fn();\n\n if (!restoring) result ? result.then(push) : push();\n\n return result;\n },\n choice(choices) {\n const unwrapped = choices.map(([content, action, visible]) => {\n return [unwrap(content), action, visible] as [string, ValidAction[], () => boolean];\n });\n\n renderer.choices(unwrapped)((selected) => {\n enmemory();\n\n stack.value[0].push(['choice', selected], [null, 0]), render();\n });\n },\n jump([scene]) {\n stack.value[0] = [[null, scene], [null, 0]];\n\n renderer.clear(false)(() => {\n if (!restoring) render();\n })\n },\n clear() {\n try {\n navigator.vibrate(0)\n } finally {\n renderer.clear(goingBack)(push);\n }\n },\n condition([condition]) {\n const value = condition();\n\n if (!restoring) stack.value[0].push(['condition', value], [null, 0]), render();\n },\n end() {\n save(false, 'auto').then(() => {\n match('clear', []);\n\n renderer.ui.showScreen('mainmenu');\n });\n },\n input([question, onInput, setup]) {\n renderer.input(question, onInput, setup)(() => {\n enmemory();\n push();\n });\n },\n custom([handler]) {\n const result = renderer.custom(handler, goingBack, () => {\n if (!restoring && handler.requireUserAction) enmemory();\n if (!restoring) push();\n });\n\n return result;\n },\n vibrate(pattern) {\n try {\n navigator.vibrate(pattern)\n } finally {\n push()\n }\n },\n next() {\n push();\n },\n animateCharacter([character, timeout, ...classes]) {\n const handler: CustomHandler = (get) => {\n const root = get('novely-animate-character', false).root;\n const target = root.querySelector(`div[data-characters] > canvas[data-character=\"${character}\"]`);\n\n /**\n * Character is not found in the DOM\n */\n if (!target) return;\n\n const classNames = classes.filter(className => !target.classList.contains(className));\n\n target.classList.add(...classNames);\n\n setTimeout(() => {\n target.classList.remove(...classNames);\n }, timeout);\n }\n\n handler.callOnlyLatest = true;\n\n return renderer.custom(handler, goingBack, () => { }), push();\n }\n });\n\n const enmemory = () => {\n if (restoring) return;\n\n const current = klona(stack.value);\n\n current[0] = klona(stack.value[0]);\n\n current[2][0] = new Date().valueOf();\n current[2][1] = 'auto';\n\n stack.push(current);\n }\n\n const next = () => {\n /**\n * Последний элемент пути\n */\n const last = stack.value[0][stack.value[0].length - 1]!;\n\n /**\n * Если он вида `[null, int]` - увеличивает `int`\n */\n if (isNull(last[0]) && isNumber(last[1])) {\n last[1] = last[1] + 1;\n return;\n }\n\n /**\n * Иначе добавляет новое `[null int]`\n */\n stack.value[0].push([null, 0]);\n }\n\n const render = () => {\n const referred = refer();\n\n if (referred) {\n const [action, ...props] = referred;\n\n match(action, props);\n }\n }\n\n const push = () => {\n if (!restoring) next(), render();\n }\n\n const unwrap = (content: DialogContent | ChoiceContent) => {\n const lang = $.get().meta[0];\n const data = state();\n\n return replaceT9N(typeof content === 'function' ? content(lang, data) : content, data);\n }\n\n return {\n withStory,\n action,\n render,\n state,\n t: t9n.t,\n }\n}\n\nexport { novely }\n","export function klona(val) {\n\tvar k, out, tmp;\n\n\tif (Array.isArray(val)) {\n\t\tout = Array(k=val.length);\n\t\twhile (k--) out[k] = (tmp=val[k]) && typeof tmp === 'object' ? klona(tmp) : tmp;\n\t\treturn out;\n\t}\n\n\tif (Object.prototype.toString.call(val) === '[object Object]') {\n\t\tout = {}; // null\n\t\tfor (k in val) {\n\t\t\tif (k === '__proto__') {\n\t\t\t\tObject.defineProperty(out, k, {\n\t\t\t\t\tvalue: klona(val[k]),\n\t\t\t\t\tconfigurable: true,\n\t\t\t\t\tenumerable: true,\n\t\t\t\t\twritable: true,\n\t\t\t\t});\n\t\t\t} else {\n\t\t\t\tout[k] = (tmp=val[k]) && typeof tmp === 'object' ? klona(tmp) : tmp;\n\t\t\t}\n\t\t}\n\t\treturn out;\n\t}\n\n\treturn val;\n}\n","const SKIPPED_DURING_RESTORE = new Set([\n 'dialog',\n 'input',\n 'vibrate'\n] as const);\n\nexport { SKIPPED_DURING_RESTORE }","import type { BaseTranslationStrings } from './translations';\n\ntype PluralType = Intl.LDMLPluralRule;\ntype FunctionalSetupT9N = <LanguageKey extends string, PluralKey extends string, StringKey extends string>(parameters: { [Lang in LanguageKey]: { pluralization: { [Plural in PluralKey]: Partial<Record<PluralType, string>> }; internal: { [Key in BaseTranslationStrings]: string }; strings: { [Str in StringKey]: string } } }) => T9N<LanguageKey, PluralKey, StringKey>\ntype SetupT9N<LanguageKey extends string> = <PluralKey extends string, StringKey extends string>(parameters: { [Lang in LanguageKey]: { pluralization: { [Plural in PluralKey]: Partial<Record<PluralType, string>> }; internal: { [Key in BaseTranslationStrings]: string }; strings: { [Str in StringKey]: string } } }) => T9N<LanguageKey, PluralKey, StringKey>\n\ntype T9N<LanguageKey extends string, _PluralKey extends string, StringKey extends string> = {\n t(key: StringKey): (lang: LanguageKey | (string & {}), obj: Record<string, unknown>) => string;\n i(key: StringKey, lang: LanguageKey | (string & {})): string;\n}\n\nconst RGX = /{{(.*?)}}/g;\n\nconst replace = (str: string, obj: Record<string, unknown>, pluralization?: Record<string, Record<string, PluralType>>, pr?: Intl.PluralRules) => {\n return str.replace(RGX, (x: any, key: any, y: any) => {\n x = 0;\n y = obj;\n\n key = (key as string).trim();\n\n let at = (key as string).split('@');\n let plural: string | undefined;\n\n if (at.length > 1) {\n ([key, plural] = at);\n }\n\n key = (key as string).split('.');\n\n while (y && x < key.length) {\n y = y[key[x++]];\n }\n\n if (plural && pluralization && pr && y) {\n y = pluralization[plural][pr!.select(y)];\n }\n\n return y != null ? y : '';\n });\n}\n\nconst createT9N: FunctionalSetupT9N = (parameters) => {\n let locale: string | undefined;\n let pr: Intl.PluralRules | undefined;\n\n return {\n t(key) {\n return (lang, obj) => {\n /**\n * At first run `locale` and `pr` are not defined.\n * When `locale` changes, `pr` should be updated`\n */\n if (!locale || !pr || lang != locale) {\n pr = new Intl.PluralRules(locale = lang);\n }\n\n // @ts-ignore `(string & {})` cannot be used to index type `LanguageKey`.\n return replace(parameters[lang]['strings'][key], obj, parameters[lang]['pluralization']);\n }\n },\n i(key, lang) {\n // @ts-ignore `(string & {})` cannot be used to index type `LanguageKey`.\n return parameters[lang]['internal'][key] as string;\n }\n }\n}\n\nexport { createT9N, replace }\nexport type { SetupT9N, T9N, FunctionalSetupT9N } ","const RU = {\n 'NewGame': 'Новая игра',\n 'HomeScreen': 'Главный экран',\n 'ToTheGame': 'К игре',\n 'Language': 'Язык',\n 'NoSaves': 'Сохранений нет',\n 'LoadSave': 'Загрузить',\n 'Saves': 'Сохранения',\n 'Settings': 'Настройки',\n 'Sumbit': 'Подтвердить',\n 'GoBack': 'Назад',\n 'DoSave': 'Сохранение',\n 'Auto': 'Авто',\n 'Stop': 'Стоп',\n 'Exit': 'Выход',\n 'Automatic': 'Автоматическое',\n 'Manual': 'Ручное',\n 'Remove': 'Удалить',\n 'LoadASaveFrom': 'Загрузить сохранение от',\n 'DeleteASaveFrom': 'Удалить сохранение от',\n};\n\nconst EN = {\n 'NewGame': 'New Game',\n 'HomeScreen': 'Home Screen',\n 'ToTheGame': 'To the Game',\n 'Language': 'Language',\n 'NoSaves': 'No saves',\n 'LoadSave': 'Load',\n 'Saves': 'Saves',\n 'Settings': 'Settings',\n 'Sumbit': 'Submit',\n 'GoBack': 'Go back',\n 'DoSave': 'Save',\n 'Auto': 'Auto',\n 'Stop': 'Stop',\n 'Exit': 'Exit',\n 'Automatic': 'Automatic',\n 'Manual': 'Manual',\n 'Remove': 'Remove',\n 'LoadASaveFrom': 'Load a save from',\n 'DeleteASaveFrom': 'Delete a save from',\n};\n\nconst JP = {\n 'NewGame': '新しいゲーム',\n 'HomeScreen': 'ホーム画面',\n 'ToTheGame': 'ゲームへ',\n 'Language': '言語',\n 'NoSaves': 'セーブがありません',\n 'LoadSave': 'ロード',\n 'Saves': 'セーブ',\n 'Settings': '設定',\n 'Sumbit': '提出する',\n 'GoBack': '戻る',\n 'DoSave': 'セーブする',\n 'Auto': 'オート',\n 'Stop': '停止',\n 'Exit': '終了する',\n 'Automatic': '自動',\n 'Manual': '手動',\n 'Remove': '削除する',\n 'LoadASaveFrom': 'からセーブを読み込む',\n 'DeleteASaveFrom': 'からセーブを削除する',\n};\n\ntype BaseTranslationStrings = keyof typeof RU;\n\nexport { RU, EN, JP }\nexport type { BaseTranslationStrings }","import type { StorageData } from './types'\n\ninterface LocalStorageStorageSettings {\n key: string\n}\n\ninterface Storage {\n get: () => Promise<StorageData>;\n set: (data: StorageData) => Promise<void>;\n}\n\nconst localStorageStorage = (options: LocalStorageStorageSettings): Storage => {\n return {\n async get() {\n const value = localStorage.getItem(options.key);\n\n return value ? JSON.parse(value) : { saves: [], meta: [] };\n },\n async set(data) {\n localStorage.setItem(options.key, JSON.stringify(data));\n }\n }\n}\n\nexport type { Storage }\nexport { localStorageStorage }"],"mappings":"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","names":["require_cjs","__commonJSMin","exports","module","isMergeableObject","value","isNonNullObject","isSpecial","stringValue","isReactElement","canUseSymbol","REACT_ELEMENT_TYPE","emptyTarget","val","cloneUnlessOtherwiseSpecified","options","deepmerge","defaultArrayMerge","target","source","element","getMergeFunction","key","customMerge","getEnumerableOwnPropertySymbols","symbol","getKeys","propertyIsOnObject","object","property","propertyIsUnsafe","mergeObject","destination","sourceIsArray","targetIsArray","sourceAndTargetTypesMatch","array","prev","next","deepmerge_1","src_exports","__export","localStorageStorage","novely","matchAction","values","action","props","isNumber","val","isNull","isString","isCSSImage","str","startsWith","value","isUserRequiredAction","meta","getDefaultSave","getLanguage","languages","language","throttle","fn","ms","throttled","savedArgs","savedThis","wrapper","store","current","subscribers","subscribe","cb","push","value","fn","import_deepmerge","klona","val","k","out","tmp","SKIPPED_DURING_RESTORE","RGX","replace","str","obj","pluralization","pr","x","key","y","at","plural","novely","characters","storage","createRenderer","initialScreen","t9n","languages","assetsPreload","story","withStory","s","name","items","flat","item","data","type","preloadAssets","preload","action","_","prop","props","isCSSImage","state","value","stack","prev","val","deepmerge","createStack","current","goingBack","getDefaultSave","stored","getLanguage","$","store","throttledOnStorageDataChange","throttle","initial","save","override","date","isLatest","newGame","restore","set","restoring","latest","klona","renderer","match","index","max","acc","isNull","isNumber","queue","isString","i","meta","SKIPPED_DURING_RESTORE","isUserRequiredAction","indexedQueue","_action","_meta","str","result","render","refer","promises","emotions","emotion","bg","res","rej","img","matchAction","time","push","background","source","character","className","style","handle","duration","person","content","unwrap","enmemory","fn","choices","unwrapped","visible","selected","scene","condition","question","onInput","setup","handler","pattern","timeout","classes","get","target","classNames","next","last","referred","lang","g","localStorageStorage","options","value","data"]}
1
+ {"version":3,"sources":["../../../node_modules/.pnpm/deepmerge@4.3.0/node_modules/deepmerge/dist/cjs.js","../src/index.ts","../src/utils.ts","../src/store.ts","../src/novely.ts","../../../node_modules/.pnpm/klona@2.0.6/node_modules/klona/json/index.mjs","../src/constants.ts","../../t9n/src/translation.ts","../../t9n/src/translations.ts","../src/storage.ts"],"sourcesContent":["'use strict';\n\nvar isMergeableObject = function isMergeableObject(value) {\n\treturn isNonNullObject(value)\n\t\t&& !isSpecial(value)\n};\n\nfunction isNonNullObject(value) {\n\treturn !!value && typeof value === 'object'\n}\n\nfunction isSpecial(value) {\n\tvar stringValue = Object.prototype.toString.call(value);\n\n\treturn stringValue === '[object RegExp]'\n\t\t|| stringValue === '[object Date]'\n\t\t|| isReactElement(value)\n}\n\n// see https://github.com/facebook/react/blob/b5ac963fb791d1298e7f396236383bc955f916c1/src/isomorphic/classic/element/ReactElement.js#L21-L25\nvar canUseSymbol = typeof Symbol === 'function' && Symbol.for;\nvar REACT_ELEMENT_TYPE = canUseSymbol ? Symbol.for('react.element') : 0xeac7;\n\nfunction isReactElement(value) {\n\treturn value.$$typeof === REACT_ELEMENT_TYPE\n}\n\nfunction emptyTarget(val) {\n\treturn Array.isArray(val) ? [] : {}\n}\n\nfunction cloneUnlessOtherwiseSpecified(value, options) {\n\treturn (options.clone !== false && options.isMergeableObject(value))\n\t\t? deepmerge(emptyTarget(value), value, options)\n\t\t: value\n}\n\nfunction defaultArrayMerge(target, source, options) {\n\treturn target.concat(source).map(function(element) {\n\t\treturn cloneUnlessOtherwiseSpecified(element, options)\n\t})\n}\n\nfunction getMergeFunction(key, options) {\n\tif (!options.customMerge) {\n\t\treturn deepmerge\n\t}\n\tvar customMerge = options.customMerge(key);\n\treturn typeof customMerge === 'function' ? customMerge : deepmerge\n}\n\nfunction getEnumerableOwnPropertySymbols(target) {\n\treturn Object.getOwnPropertySymbols\n\t\t? Object.getOwnPropertySymbols(target).filter(function(symbol) {\n\t\t\treturn Object.propertyIsEnumerable.call(target, symbol)\n\t\t})\n\t\t: []\n}\n\nfunction getKeys(target) {\n\treturn Object.keys(target).concat(getEnumerableOwnPropertySymbols(target))\n}\n\nfunction propertyIsOnObject(object, property) {\n\ttry {\n\t\treturn property in object\n\t} catch(_) {\n\t\treturn false\n\t}\n}\n\n// Protects from prototype poisoning and unexpected merging up the prototype chain.\nfunction propertyIsUnsafe(target, key) {\n\treturn propertyIsOnObject(target, key) // Properties are safe to merge if they don't exist in the target yet,\n\t\t&& !(Object.hasOwnProperty.call(target, key) // unsafe if they exist up the prototype chain,\n\t\t\t&& Object.propertyIsEnumerable.call(target, key)) // and also unsafe if they're nonenumerable.\n}\n\nfunction mergeObject(target, source, options) {\n\tvar destination = {};\n\tif (options.isMergeableObject(target)) {\n\t\tgetKeys(target).forEach(function(key) {\n\t\t\tdestination[key] = cloneUnlessOtherwiseSpecified(target[key], options);\n\t\t});\n\t}\n\tgetKeys(source).forEach(function(key) {\n\t\tif (propertyIsUnsafe(target, key)) {\n\t\t\treturn\n\t\t}\n\n\t\tif (propertyIsOnObject(target, key) && options.isMergeableObject(source[key])) {\n\t\t\tdestination[key] = getMergeFunction(key, options)(target[key], source[key], options);\n\t\t} else {\n\t\t\tdestination[key] = cloneUnlessOtherwiseSpecified(source[key], options);\n\t\t}\n\t});\n\treturn destination\n}\n\nfunction deepmerge(target, source, options) {\n\toptions = options || {};\n\toptions.arrayMerge = options.arrayMerge || defaultArrayMerge;\n\toptions.isMergeableObject = options.isMergeableObject || isMergeableObject;\n\t// cloneUnlessOtherwiseSpecified is added to `options` so that custom arrayMerge()\n\t// implementations can use it. The caller may not replace it.\n\toptions.cloneUnlessOtherwiseSpecified = cloneUnlessOtherwiseSpecified;\n\n\tvar sourceIsArray = Array.isArray(source);\n\tvar targetIsArray = Array.isArray(target);\n\tvar sourceAndTargetTypesMatch = sourceIsArray === targetIsArray;\n\n\tif (!sourceAndTargetTypesMatch) {\n\t\treturn cloneUnlessOtherwiseSpecified(source, options)\n\t} else if (sourceIsArray) {\n\t\treturn options.arrayMerge(target, source, options)\n\t} else {\n\t\treturn mergeObject(target, source, options)\n\t}\n}\n\ndeepmerge.all = function deepmergeAll(array, options) {\n\tif (!Array.isArray(array)) {\n\t\tthrow new Error('first argument should be an array')\n\t}\n\n\treturn array.reduce(function(prev, next) {\n\t\treturn deepmerge(prev, next, options)\n\t}, {})\n};\n\nvar deepmerge_1 = deepmerge;\n\nmodule.exports = deepmerge_1;\n","export type { ValidAction, Story, ActionProxyProvider, DefaultActionProxyProvider, GetActionParameters, CustomHandler, CustomHandlerGetResult, CustomHandlerGetResultDataFunction } from './action'\nexport type { Emotions, Character } from './character'\nexport type { CharacterHandle, AudioHandle, RendererStore, Renderer, RendererInit } from './renderer'\nexport type { Storage } from './storage'\nexport type { Thenable, Path, StorageData } from './types'\nexport type { Stored } from './store'\nexport type { BaseTranslationStrings } from '@novely/t9n';\n\nexport { novely } from './novely'\nexport { localStorageStorage } from './storage'","import type { ActionProxyProvider, CustomHandler } from './action'\nimport type { Character } from './character'\nimport type { Save, Thenable } from './types'\n\ntype MatchActionMap = {\n [Key in keyof ActionProxyProvider<Record<string, Character>>]: (data: Parameters<ActionProxyProvider<Record<string, Character>>[Key]>) => void;\n}\n\ntype MatchActionMapComplete = Omit<MatchActionMap, 'custom'> & {\n custom: (value: [handler: CustomHandler]) => Thenable<void>;\n}\n\nconst matchAction = <M extends MatchActionMapComplete>(values: M) => {\n return (action: keyof MatchActionMap, props: any) => {\n return values[action](props);\n }\n}\n\nconst isNumber = (val: unknown): val is number => {\n return typeof val === 'number';\n}\n\nconst isNull = (val: unknown): val is null => {\n return val === null;\n}\n\nconst isString = (val: unknown): val is string => {\n return typeof val === 'string';\n}\n\nconst isCSSImage = (str: string) => {\n const startsWith = String.prototype.startsWith.bind(str);\n\n return startsWith('http') || startsWith('/') || startsWith('.') || startsWith('data');\n}\n\nconst str = (value: unknown) => {\n return String(value);\n}\n\nconst isUserRequiredAction = (action: keyof MatchActionMapComplete, meta: Parameters<MatchActionMapComplete[keyof MatchActionMapComplete]>) => {\n return action === 'custom' && meta[0] && (meta[0] as unknown as CustomHandler).requireUserAction;\n}\n\nconst getDefaultSave = () => {\n return [[[null, 'start'], [null, 0]], {}, [Date.now(), 'auto']] as Save;\n}\n\nconst getLanguage = (languages: string[], language = navigator.language) => {\n if (languages.includes(language)) {\n return language;\n } else if (languages.includes((language = language.substring(0, 2)))) {\n return language;\n } else if ((language = languages.find((value) => navigator.languages.includes(value))!)) {\n return language\n }\n\n /**\n * We'v checked the `en-GB` format, `en` format, and maybe any second languages, but there were no matches\n */\n return languages[0];\n}\n\n/**\n * @copyright Techlead LLC\n * @see https://learn.javascript.ru/task/throttle\n */\nconst throttle = <Fn extends ((...args: any[]) => any)>(fn: Fn, ms: number) => {\n let throttled = false, savedArgs: any, savedThis: any;\n\n function wrapper() {\n if (throttled) {\n savedArgs = arguments;\n // @ts-ignore\n savedThis = this;\n\n return;\n }\n\n // @ts-ignore\n fn.apply(this, arguments);\n\n throttled = false;\n }\n\n setTimeout(function () {\n throttled = false;\n\n if (savedArgs) {\n wrapper.apply(savedThis, savedArgs);\n savedArgs = savedThis = null;\n }\n }, ms);\n\n return wrapper as unknown as (...args: Parameters<Fn>) => void;\n}\n\nexport { matchAction, isNumber, isNull, isString, isCSSImage, str, isUserRequiredAction, getDefaultSave, getLanguage, throttle }","type Stored<T> = {\n subscribe: (cb: (value: T) => void) => () => void;\n update: (fn: (prev: T) => T) => void;\n get: () => T;\n}\n\nconst store = <T>(current: T, subscribers = new Set<(value: T) => void>()): Stored<T> => {\n const subscribe = (cb: (value: T) => void) => {\n subscribers.add(cb), cb(current);\n\n return () => {\n subscribers.delete(cb);\n }\n };\n\n const push = (value: T) => {\n subscribers.forEach((cb) => cb(value))\n };\n\n const update = (fn: (prev: T) => T) => {\n push((current = fn(current)));\n };\n\n const get = () => {\n return current;\n };\n\n return { subscribe, update, get } as const;\n};\n\nexport { store }\nexport type { Stored }","import type { Character } from './character';\nimport type { ActionProxyProvider, GetActionParameters, Story, ValidAction, DialogContent, ChoiceContent, CustomHandler } from './action';\nimport type { Storage } from './storage';\nimport type { Save, State, StorageData } from './types'\nimport type { Renderer, RendererInit } from './renderer'\nimport type { SetupT9N } from '@novely/t9n'\nimport { matchAction, isNumber, isNull, isString, isCSSImage, str, isUserRequiredAction, getDefaultSave, getLanguage, throttle } from './utils';\nimport { store } from './store';\nimport { all as deepmerge } from 'deepmerge'\nimport { klona } from 'klona/json';\nimport { SKIPPED_DURING_RESTORE } from './constants';\nimport { replace as replaceT9N } from '@novely/t9n';\n\ninterface NovelyInit<Languages extends string, Characters extends Record<string, Character<Languages>>, Inter extends ReturnType<SetupT9N<Languages>>> {\n /**\n * An array of languages supported by the game.\n */\n languages: Languages[];\n /**\n * An object containing the characters in the game.\n */\n characters: Characters;\n /**\n * An object that provides access to the game's storage system.\n */\n storage: Storage;\n /**\n * A function that returns a Renderer object used to display the game's content\n */\n renderer: (characters: RendererInit) => Renderer;\n /**\n * An optional property that specifies the initial screen to display when the game starts\n */\n initialScreen?: \"mainmenu\" | \"game\" | \"saves\" | \"settings\";\n /**\n * An object containing the translation functions used in the game\n */\n t9n: Inter;\n /**\n * An optional property that specifies whether to preload assets when the game starts\n */\n assetsPreload?: boolean;\n /**\n * An optional property that specifies whether the game should use a single save.\n */\n singleSave?: boolean;\n}\n\nconst novely = <Languages extends string, Characters extends Record<string, Character<Languages>>, Inter extends ReturnType<SetupT9N<Languages>>>({ characters, storage, renderer: createRenderer, initialScreen = \"mainmenu\", t9n, languages, assetsPreload }: NovelyInit<Languages, Characters, Inter>) => {\n let story: Story;\n\n const withStory = (s: Story) => {\n /**\n * Transforms `(ValidAction | ValidAction[])[]` to `ValidAction[]`\n */\n story = Object.fromEntries(Object.entries(s).map(([name, items]) => {\n const flat = (item: (ValidAction | ValidAction[])[]): ValidAction[] => {\n return item.flatMap((data) => {\n const type = data[0];\n\n /**\n * This is not just an action like `['name', ...arguments]`, but an array of actions\n */\n if (Array.isArray(type)) return flat(data as ValidAction[]);\n\n return [data as ValidAction];\n });\n };\n\n return [name, flat(items)];\n }));\n\n /**\n * Load assets after the `action` scripts are executed\n */\n preloadAssets();\n }\n\n /**\n * This is used when background is loading\n */\n const preload = {\n background: new Set<string>(),\n }\n\n const action = new Proxy({} as ActionProxyProvider<Characters>, {\n get(_, prop) {\n return (...props: Parameters<ActionProxyProvider<Record<string, Character>>[keyof ActionProxyProvider<Record<string, Character>>]>) => {\n if (prop === 'showBackground') {\n if (isCSSImage(props[0] as string)) preload.background.add(props[0] as string);\n }\n\n return [prop, ...props];\n }\n }\n });\n\n function state(value: State | ((prev: State) => State)): void;\n function state(): State;\n function state(value?: State | ((prev: State) => State)): State | void {\n if (!value) return stack.value[1]\n\n const prev = stack.value[1];\n const val = typeof value === 'function' ? value(prev as State) : deepmerge([prev, value]);\n\n stack.value[1] = val as State;\n }\n\n const createStack = (current: Save, stack = [current]) => {\n return {\n get value() {\n return stack.at(-1)!;\n },\n set value(value: Save) {\n stack[stack.length - 1] = value;\n },\n back() {\n if (stack.length > 1) stack.pop(), goingBack = true;\n },\n push(value: Save) {\n stack.push(value);\n },\n clear() {\n stack = [getDefaultSave()];\n }\n }\n }\n\n /**\n * 1) Novely rendered using the `initialData`, you can still start new game or `load` an empty one - this is scary, imagine losing your progress\n * 2) Actual stored data is loaded, language and etc is changed \n */\n const initialData: StorageData = {\n saves: [],\n meta: [getLanguage(languages)]\n };\n\n const $ = store(initialData);\n\n let initialDataLoaded = false;\n\n const onStorageDataChange = (value: StorageData) => {\n if (initialDataLoaded) storage.set(value);\n };\n\n const throttledOnStorageDataChange = throttle(onStorageDataChange, 120);\n\n $.subscribe(throttledOnStorageDataChange);\n\n storage.get().then(stored => {\n /**\n * Default `localStorageStorage` cannot determine preferred language, and returns empty array\n */\n stored.meta[0] ||= getLanguage(languages);\n\n /**\n * Now the next store updates will entail saving via storage.set\n */\n initialDataLoaded = true;\n\n $.update(() => stored);\n });\n\n const initial = ((value) => value.saves.length > 0 && value.saves.at(-1))($.get()) || getDefaultSave();\n const stack = createStack(initial);\n\n const save = (override = false, type: Save[2][1] = override ? 'auto' : 'manual') => {\n $.update(prev => {\n const date = stack.value[2][0];\n const isLatest = prev.saves.findIndex(value => value[2][0] === date) === prev.saves.length - 1;\n\n /**\n * Обновим дату и тип\n */\n stack.value[2][0] = Date.now();\n stack.value[2][1] = type;\n\n if (override) {\n /**\n * Перезапишем\n */\n if (isLatest) {\n /**\n * Сохранения хранятся в массиве. Нельзя перезаписать любое последнее\n * \n * Если перезаписывать старое сохранение, то они не будут идти в хронологическом порядке\n */\n prev.saves[prev.saves.length - 1] = stack.value;\n } else {\n prev.saves.push(stack.value);\n }\n } else {\n /**\n * Добавляем текущее сохранение\n */\n prev.saves.push(stack.value);\n }\n\n /**\n * Устанавливаем новое значение\n */\n return prev;\n });\n }\n\n const newGame = () => {\n $.update(prev => {\n const save = getDefaultSave();\n\n prev.saves.push(save), restore(save);\n\n return prev;\n });\n }\n\n /**\n * Устанавливает сохранение\n */\n const set = (save: Save) => {\n stack.value = save;\n\n return restore(save);\n }\n\n let restoring = false;\n let goingBack = false;\n\n /**\n * Визуально восстанавливает историю\n */\n const restore = async (save?: Save) => {\n let latest = save ? save : $.get().saves.at(-1);\n\n /**\n * Если нет сохранённой игры, то запустим ту, которая уже есть\n */\n if (!latest) {\n $.update(() => ({ saves: [initial], meta: [getLanguage(languages)] }));\n\n latest = klona(initial);\n }\n\n restoring = true, stack.value = latest;\n\n /**\n * Показать экран игры\n */\n renderer.ui.showScreen('game');\n\n match('clear', [goingBack]);\n\n /**\n * Текущий элемент в истории\n */\n let current: any = story;\n /**\n * Текущий элемент `[null, int]`\n */\n let index = 0;\n\n /**\n * Число элементов `[null, int]`\n */\n const max = stack.value[0].reduce((acc, [type, val]) => {\n if (isNull(type) && isNumber(val)) return acc + 1;\n\n return acc;\n }, 0);\n\n const queue = [] as [any, any][];\n\n for (const [type, val] of stack.value[0]) {\n if (type === null) {\n if (isString(val)) {\n current = current[val];\n } else if (isNumber(val)) {\n index++;\n\n /**\n * Запустим все экшены которые идут в `[null, int]` от `0` до `int`\n */\n for (let i = 0; i < val; i++) {\n const [action, ...meta] = current[i];\n\n /**\n * Экшены, для закрытия которых пользователь должен с ними взаимодействовать\n * Также в эту группу входят экшены, которые не должны быть вызваны при восстановлении\n */\n if (SKIPPED_DURING_RESTORE.has(action) || isUserRequiredAction(action, meta)) {\n if (index === max && i === val) {\n queue.push([action, meta]);\n } else {\n continue;\n }\n }\n\n queue.push([action, meta]);\n }\n\n current = current[val];\n }\n } else if (type === 'choice') {\n current = current[val as number + 1][1];\n } else if (type === 'condition') {\n current = current[2][val];\n }\n }\n\n const indexedQueue = queue.map((value, index) => value.concat(index) as [ValidAction[0], ValidAction[1], number]);\n\n for await (const [action, meta, i] of indexedQueue) {\n if (action === 'function' || action === 'custom') {\n /**\n * Если `callOnlyLatest` - `true`\n */\n if (action === 'custom' && (meta as GetActionParameters<'Custom'>)[0].callOnlyLatest) {\n /**\n * Вычислим `latest` или нет\n */\n const next = indexedQueue.slice(i + 1);\n const latest = !next.some(([_action, _meta]) => str(_meta[0]) === str(meta[0]));\n\n if (!latest) continue;\n }\n\n /**\n * Action может возвращать Promise. Нужно подожать его `resolve`\n */\n const result = match(action, meta);\n\n /**\n * Дождёмся окончания\n */\n if (result && 'then' in result) await result;\n } else {\n match(action as keyof ActionProxyProvider<Record<string, Character<string>>>, meta);\n }\n }\n\n restoring = false, goingBack = false, render();\n }\n\n const refer = () => {\n let current: any = story;\n\n for (const [type, val] of stack.value[0]) {\n if (type === null) {\n current = current[val];\n } else if (type === 'choice') {\n current = current[val as number + 1][1];\n } else if (type === 'condition') {\n current = current[2][val];\n }\n }\n\n return current;\n }\n\n const renderer = createRenderer({\n characters,\n set,\n restore,\n save,\n newGame,\n stack,\n languages,\n t: t9n.i,\n $\n });\n\n const preloadAssets = () => {\n if (!assetsPreload) {\n initialScreen === 'game' ? restore() : renderer.ui.showScreen(initialScreen);\n\n return;\n }\n\n /**\n * We need to load all the characters and their emotions\n */\n\n let promises: Promise<unknown>[] = [];\n\n renderer.ui.showScreen('loading');\n\n for (const [name, { emotions }] of Object.entries(characters)) {\n for (const emotion of Object.keys(emotions)) {\n promises.push(Promise.resolve(renderer.character(name).withEmotion(emotion)));\n }\n }\n\n for (const bg of preload.background) {\n promises.push(new Promise((res, rej) => {\n /**\n * Create `img` element\n */\n const img = document.createElement('img');\n\n /**\n * Set `src`\n */\n img.crossOrigin = '*';\n img.src = bg;\n\n if (img.complete && img.naturalHeight !== 0) {\n /**\n * Image is already loaded\n */\n res(void 0);\n } else {\n /**\n * Image is uniq, it is safe to use `onload`\n */\n img.onload = res;\n img.onerror = rej;\n }\n }));\n }\n\n Promise.all(promises).then(() => {\n renderer.ui.showScreen(initialScreen);\n });\n }\n\n const match = matchAction({\n wait([time]) {\n /**\n * `restoring` может поменяться на `true` перед тем как запуститься `push` из `setTimeout`\n */\n if (!restoring) setTimeout(push, time);\n },\n showBackground([background]) {\n renderer.background(background);\n push()\n },\n playMusic([source]) {\n renderer.music(source, 'music').play();\n push()\n },\n stopMusic([source]) {\n renderer.music(source, 'music').stop();\n push()\n },\n showCharacter([character, emotion, className, style]) {\n const handle = renderer.character(character);\n\n handle.append(className, style);\n handle.withEmotion(emotion)();\n\n push()\n },\n hideCharacter([character, className, style, duration]) {\n const handle = renderer.character(character);\n\n handle.remove(className, style, duration)(push);\n },\n dialog([person, content, emotion]) {\n renderer.dialog(unwrap(content), person, emotion)(() => {\n enmemory();\n push();\n });\n },\n function([fn]) {\n const result = fn();\n\n if (!restoring) result ? result.then(push) : push();\n\n return result;\n },\n choice(choices) {\n const unwrapped = choices.map(([content, action, visible]) => {\n return [unwrap(content), action, visible] as [string, ValidAction[], () => boolean];\n });\n\n renderer.choices(unwrapped)((selected) => {\n enmemory();\n\n stack.value[0].push(['choice', selected], [null, 0]), render();\n });\n },\n jump([scene]) {\n stack.value[0] = [[null, scene], [null, 0]];\n\n renderer.clear(false)(() => {\n if (!restoring) render();\n })\n },\n clear() {\n try {\n navigator.vibrate(0)\n } finally {\n renderer.clear(goingBack)(push);\n }\n },\n condition([condition]) {\n const value = condition();\n\n if (!restoring) stack.value[0].push(['condition', value], [null, 0]), render();\n },\n end() {\n /**\n * Save Current Game\n */\n save(false, 'auto');\n /**\n * Clear the Scene\n */\n match('clear', []);\n /**\n * Go to the main menu\n */\n renderer.ui.showScreen('mainmenu');\n },\n input([question, onInput, setup]) {\n renderer.input(question, onInput, setup)(() => {\n enmemory();\n push();\n });\n },\n custom([handler]) {\n const result = renderer.custom(handler, goingBack, () => {\n if (!restoring && handler.requireUserAction) enmemory();\n if (!restoring) push();\n });\n\n return result;\n },\n vibrate(pattern) {\n try {\n navigator.vibrate(pattern)\n } finally {\n push()\n }\n },\n next() {\n push();\n },\n animateCharacter([character, timeout, ...classes]) {\n const handler: CustomHandler = (get) => {\n const root = get('novely-animate-character', false).root;\n const target = root.querySelector(`div[data-characters] > canvas[data-character=\"${character}\"]`);\n\n /**\n * Character is not found in the DOM\n */\n if (!target) return;\n\n const classNames = classes.filter(className => !target.classList.contains(className));\n\n target.classList.add(...classNames);\n\n setTimeout(() => {\n target.classList.remove(...classNames);\n }, timeout);\n }\n\n handler.callOnlyLatest = true;\n\n return renderer.custom(handler, goingBack, () => { }), push();\n }\n });\n\n const enmemory = () => {\n if (restoring) return;\n\n const current = klona(stack.value);\n\n current[0] = klona(stack.value[0]);\n\n current[2][0] = new Date().valueOf();\n current[2][1] = 'auto';\n\n stack.push(current);\n }\n\n const next = () => {\n /**\n * Последний элемент пути\n */\n const last = stack.value[0][stack.value[0].length - 1]!;\n\n /**\n * Если он вида `[null, int]` - увеличивает `int`\n */\n if (isNull(last[0]) && isNumber(last[1])) {\n last[1] = last[1] + 1;\n return;\n }\n\n /**\n * Иначе добавляет новое `[null int]`\n */\n stack.value[0].push([null, 0]);\n }\n\n const render = () => {\n const referred = refer();\n\n if (referred) {\n const [action, ...props] = referred;\n\n match(action, props);\n }\n }\n\n const push = () => {\n if (!restoring) next(), render();\n }\n\n const unwrap = (content: DialogContent | ChoiceContent) => {\n const lang = $.get().meta[0];\n const data = state();\n\n return replaceT9N(typeof content === 'function' ? content(lang, data) : content, data);\n }\n\n return {\n withStory,\n action,\n render,\n state,\n t: t9n.t,\n }\n}\n\nexport { novely }\n","export function klona(val) {\n\tvar k, out, tmp;\n\n\tif (Array.isArray(val)) {\n\t\tout = Array(k=val.length);\n\t\twhile (k--) out[k] = (tmp=val[k]) && typeof tmp === 'object' ? klona(tmp) : tmp;\n\t\treturn out;\n\t}\n\n\tif (Object.prototype.toString.call(val) === '[object Object]') {\n\t\tout = {}; // null\n\t\tfor (k in val) {\n\t\t\tif (k === '__proto__') {\n\t\t\t\tObject.defineProperty(out, k, {\n\t\t\t\t\tvalue: klona(val[k]),\n\t\t\t\t\tconfigurable: true,\n\t\t\t\t\tenumerable: true,\n\t\t\t\t\twritable: true,\n\t\t\t\t});\n\t\t\t} else {\n\t\t\t\tout[k] = (tmp=val[k]) && typeof tmp === 'object' ? klona(tmp) : tmp;\n\t\t\t}\n\t\t}\n\t\treturn out;\n\t}\n\n\treturn val;\n}\n","const SKIPPED_DURING_RESTORE = new Set([\n 'dialog',\n 'input',\n 'vibrate'\n] as const);\n\nexport { SKIPPED_DURING_RESTORE }","import type { BaseTranslationStrings } from './translations';\n\ntype PluralType = Intl.LDMLPluralRule;\ntype FunctionalSetupT9N = <LanguageKey extends string, PluralKey extends string, StringKey extends string>(parameters: { [Lang in LanguageKey]: { pluralization: { [Plural in PluralKey]: Partial<Record<PluralType, string>> }; internal: { [Key in BaseTranslationStrings]: string }; strings: { [Str in StringKey]: string } } }) => T9N<LanguageKey, PluralKey, StringKey>\ntype SetupT9N<LanguageKey extends string> = <PluralKey extends string, StringKey extends string>(parameters: { [Lang in LanguageKey]: { pluralization: { [Plural in PluralKey]: Partial<Record<PluralType, string>> }; internal: { [Key in BaseTranslationStrings]: string }; strings: { [Str in StringKey]: string } } }) => T9N<LanguageKey, PluralKey, StringKey>\n\ntype T9N<LanguageKey extends string, _PluralKey extends string, StringKey extends string> = {\n t(key: StringKey): (lang: LanguageKey | (string & {}), obj: Record<string, unknown>) => string;\n i(key: StringKey, lang: LanguageKey | (string & {})): string;\n}\n\nconst RGX = /{{(.*?)}}/g;\n\nconst replace = (str: string, obj: Record<string, unknown>, pluralization?: Record<string, Record<string, PluralType>>, pr?: Intl.PluralRules) => {\n return str.replace(RGX, (x: any, key: any, y: any) => {\n x = 0;\n y = obj;\n\n key = (key as string).trim();\n\n let at = (key as string).split('@');\n let plural: string | undefined;\n\n if (at.length > 1) {\n ([key, plural] = at);\n }\n\n key = (key as string).split('.');\n\n while (y && x < key.length) {\n y = y[key[x++]];\n }\n\n if (plural && pluralization && pr && y) {\n y = pluralization[plural][pr!.select(y)];\n }\n\n return y != null ? y : '';\n });\n}\n\nconst createT9N: FunctionalSetupT9N = (parameters) => {\n let locale: string | undefined;\n let pr: Intl.PluralRules | undefined;\n\n return {\n t(key) {\n return (lang, obj) => {\n /**\n * At first run `locale` and `pr` are not defined.\n * When `locale` changes, `pr` should be updated`\n */\n if (!locale || !pr || lang != locale) {\n pr = new Intl.PluralRules(locale = lang);\n }\n\n // @ts-ignore `(string & {})` cannot be used to index type `LanguageKey`.\n return replace(parameters[lang]['strings'][key], obj, parameters[lang]['pluralization']);\n }\n },\n i(key, lang) {\n // @ts-ignore `(string & {})` cannot be used to index type `LanguageKey`.\n return parameters[lang]['internal'][key] as string;\n }\n }\n}\n\nexport { createT9N, replace }\nexport type { SetupT9N, T9N, FunctionalSetupT9N } ","const RU = {\n 'NewGame': 'Новая игра',\n 'HomeScreen': 'Главный экран',\n 'ToTheGame': 'К игре',\n 'Language': 'Язык',\n 'NoSaves': 'Сохранений нет',\n 'LoadSave': 'Загрузить',\n 'Saves': 'Сохранения',\n 'Settings': 'Настройки',\n 'Sumbit': 'Подтвердить',\n 'GoBack': 'Назад',\n 'DoSave': 'Сохранение',\n 'Auto': 'Авто',\n 'Stop': 'Стоп',\n 'Exit': 'Выход',\n 'Automatic': 'Автоматическое',\n 'Manual': 'Ручное',\n 'Remove': 'Удалить',\n 'LoadASaveFrom': 'Загрузить сохранение от',\n 'DeleteASaveFrom': 'Удалить сохранение от',\n};\n\nconst EN = {\n 'NewGame': 'New Game',\n 'HomeScreen': 'Home Screen',\n 'ToTheGame': 'To the Game',\n 'Language': 'Language',\n 'NoSaves': 'No saves',\n 'LoadSave': 'Load',\n 'Saves': 'Saves',\n 'Settings': 'Settings',\n 'Sumbit': 'Submit',\n 'GoBack': 'Go back',\n 'DoSave': 'Save',\n 'Auto': 'Auto',\n 'Stop': 'Stop',\n 'Exit': 'Exit',\n 'Automatic': 'Automatic',\n 'Manual': 'Manual',\n 'Remove': 'Remove',\n 'LoadASaveFrom': 'Load a save from',\n 'DeleteASaveFrom': 'Delete a save from',\n};\n\nconst JP = {\n 'NewGame': '新しいゲーム',\n 'HomeScreen': 'ホーム画面',\n 'ToTheGame': 'ゲームへ',\n 'Language': '言語',\n 'NoSaves': 'セーブがありません',\n 'LoadSave': 'ロード',\n 'Saves': 'セーブ',\n 'Settings': '設定',\n 'Sumbit': '提出する',\n 'GoBack': '戻る',\n 'DoSave': 'セーブする',\n 'Auto': 'オート',\n 'Stop': '停止',\n 'Exit': '終了する',\n 'Automatic': '自動',\n 'Manual': '手動',\n 'Remove': '削除する',\n 'LoadASaveFrom': 'からセーブを読み込む',\n 'DeleteASaveFrom': 'からセーブを削除する',\n};\n\ntype BaseTranslationStrings = keyof typeof RU;\n\nexport { RU, EN, JP }\nexport type { BaseTranslationStrings }","import type { StorageData } from './types'\n\ninterface LocalStorageStorageSettings {\n key: string\n}\n\ninterface Storage {\n get: () => Promise<StorageData>;\n set: (data: StorageData) => Promise<void>;\n}\n\nconst localStorageStorage = (options: LocalStorageStorageSettings): Storage => {\n return {\n async get() {\n const value = localStorage.getItem(options.key);\n\n return value ? JSON.parse(value) : { saves: [], meta: [] };\n },\n async set(data) {\n localStorage.setItem(options.key, JSON.stringify(data));\n }\n }\n}\n\nexport type { Storage }\nexport { localStorageStorage }"],"mappings":"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","names":["require_cjs","__commonJSMin","exports","module","isMergeableObject","value","isNonNullObject","isSpecial","stringValue","isReactElement","canUseSymbol","REACT_ELEMENT_TYPE","emptyTarget","val","cloneUnlessOtherwiseSpecified","options","deepmerge","defaultArrayMerge","target","source","element","getMergeFunction","key","customMerge","getEnumerableOwnPropertySymbols","symbol","getKeys","propertyIsOnObject","object","property","propertyIsUnsafe","mergeObject","destination","sourceIsArray","targetIsArray","sourceAndTargetTypesMatch","array","prev","next","deepmerge_1","src_exports","__export","localStorageStorage","novely","matchAction","values","action","props","isNumber","val","isNull","isString","isCSSImage","str","startsWith","value","isUserRequiredAction","meta","getDefaultSave","getLanguage","languages","language","throttle","fn","ms","throttled","savedArgs","savedThis","wrapper","store","current","subscribers","subscribe","cb","push","value","fn","import_deepmerge","klona","val","k","out","tmp","SKIPPED_DURING_RESTORE","RGX","replace","str","obj","pluralization","pr","x","key","y","at","plural","novely","characters","storage","createRenderer","initialScreen","t9n","languages","assetsPreload","story","withStory","s","name","items","flat","item","data","type","preloadAssets","preload","action","_","prop","props","isCSSImage","state","value","stack","prev","val","deepmerge","createStack","current","goingBack","getDefaultSave","initialData","getLanguage","$","store","initialDataLoaded","throttledOnStorageDataChange","throttle","stored","initial","save","override","date","isLatest","newGame","restore","set","restoring","latest","klona","renderer","match","index","max","acc","isNull","isNumber","queue","isString","i","meta","SKIPPED_DURING_RESTORE","isUserRequiredAction","indexedQueue","_action","_meta","str","result","render","refer","promises","emotions","emotion","bg","res","rej","img","matchAction","time","push","background","source","character","className","style","handle","duration","person","content","unwrap","enmemory","fn","choices","unwrapped","visible","selected","scene","condition","question","onInput","setup","handler","pattern","timeout","classes","get","target","classNames","next","last","referred","lang","g","localStorageStorage","options","value","data"]}
package/dist/index.js CHANGED
@@ -1,2 +1,2 @@
1
- var j=n=>(a,d)=>n[a](d),M=n=>typeof n=="number",N=n=>n===null,B=n=>typeof n=="string",F=n=>{let a=String.prototype.startsWith.bind(n);return a("http")||a("/")||a(".")||a("data")},O=n=>String(n),K=(n,a)=>n==="custom"&&a[0]&&a[0].requireUserAction,I=()=>[[[null,"start"],[null,0]],{},[Date.now(),"auto"]],E=(n,a=navigator.language)=>n.includes(a)||n.includes(a=a.substring(0,2))||(a=n.find(d=>navigator.languages.includes(d)))?a:n[0],J=(n,a)=>{let d=!1,p,C;function x(){if(d){p=arguments,C=this;return}n.apply(this,arguments),d=!1}return setTimeout(function(){d=!1,p&&(x.apply(C,p),p=C=null)},a),x};var W=(n,a=new Set)=>{let d=h=>(a.add(h),h(n),()=>{a.delete(h)}),p=h=>{a.forEach(w=>w(h))};return{subscribe:d,update:h=>{p(n=h(n))},get:()=>n}};import{all as oe}from"deepmerge";import{klona as H}from"klona/json";var $=new Set(["dialog","input","vibrate"]);import{replace as se}from"@novely/t9n";var ie=async({characters:n,storage:a,renderer:d,initialScreen:p="mainmenu",t9n:C,languages:x,assetsPreload:h})=>{let w,Q=e=>{w=Object.fromEntries(Object.entries(e).map(([r,t])=>{let s=i=>i.flatMap(g=>{let v=g[0];return Array.isArray(v)?s(g):[g]});return[r,s(t)]})),re()},V={background:new Set},z=new Proxy({},{get(e,r){return(...t)=>(r==="showBackground"&&F(t[0])&&V.background.add(t[0]),[r,...t])}});function G(e){if(!e)return o.value[1];let r=o.value[1],t=typeof e=="function"?e(r):oe([r,e]);o.value[1]=t}let X=(e,r=[e])=>({get value(){return r.at(-1)},set value(t){r[r.length-1]=t},back(){r.length>1&&(r.pop(),A=!0)},push(t){r.push(t)},clear(){r=[I()]}}),q=await a.get();q.meta[0]||=E(x);let b=W(q),Y=J(e=>a.set(e),120);b.subscribe(Y);let L=(e=>e.saves.length>0&&e.saves.at(-1))(b.get())||I(),o=X(L),U=async(e=!1,r=e?"auto":"manual")=>{let t=await a.get(),s=o.value[2][0],i=t.saves.findIndex(g=>g[2][0]===s)===t.saves.length-1;return o.value[2][0]=Date.now(),o.value[2][1]=r,e&&i?t.saves[t.saves.length-1]=o.value:t.saves.push(o.value),await a.set(t)},Z=()=>{b.update(e=>{let r=I();return e.saves.push(r),R(r),e})},ee=e=>(o.value=e,R(e)),f=!1,A=!1,R=async e=>{let r=e||await a.get().then(c=>c.saves.at(-1));r||(await a.set({saves:[L],meta:[E(x)]}),r=H(L)),f=!0,o.value=r,l.ui.showScreen("game"),T("clear",[A]);let t=w,s=0,i=o.value[0].reduce((c,[u,y])=>N(u)&&M(y)?c+1:c,0),g=[];for(let[c,u]of o.value[0])if(c===null){if(B(u))t=t[u];else if(M(u)){s++;for(let y=0;y<u;y++){let[S,...D]=t[y];if($.has(S)||K(S,D))if(s===i&&y===u)g.push([S,D]);else continue;g.push([S,D])}t=t[u]}}else c==="choice"?t=t[u+1][1]:c==="condition"&&(t=t[2][u]);let v=g.map((c,u)=>c.concat(u));for await(let[c,u,y]of v)if(c==="function"||c==="custom"){if(c==="custom"&&u[0].callOnlyLatest&&!!v.slice(y+1).some(([me,ne])=>O(ne[0])===O(u[0])))continue;let S=T(c,u);S&&"then"in S&&await S}else T(c,u);f=!1,A=!1,P()},te=()=>{let e=w;for(let[r,t]of o.value[0])r===null?e=e[t]:r==="choice"?e=e[t+1][1]:r==="condition"&&(e=e[2][t]);return e},l=d({characters:n,storage:a,set:ee,restore:R,save:U,newGame:Z,stack:o,languages:x,t:C.i,$:b}),re=()=>{if(!h){p==="game"?R():l.ui.showScreen(p);return}let e=[];l.ui.showScreen("loading");for(let[r,{emotions:t}]of Object.entries(n))for(let s of Object.keys(t))e.push(Promise.resolve(l.character(r).withEmotion(s)));for(let r of V.background)e.push(new Promise((t,s)=>{let i=document.createElement("img");i.crossOrigin="*",i.src=r,i.complete&&i.naturalHeight!==0?t(void 0):(i.onload=t,i.onerror=s)}));Promise.all(e).then(()=>{l.ui.showScreen(p)})},T=j({wait([e]){f||setTimeout(m,e)},showBackground([e]){l.background(e),m()},playMusic([e]){l.music(e,"music").play(),m()},stopMusic([e]){l.music(e,"music").stop(),m()},showCharacter([e,r,t,s]){let i=l.character(e);i.append(t,s),i.withEmotion(r)(),m()},hideCharacter([e,r,t,s]){l.character(e).remove(r,t,s)(m)},dialog([e,r,t]){l.dialog(_(r),e,t)(()=>{k(),m()})},function([e]){let r=e();return f||(r?r.then(m):m()),r},choice(e){let r=e.map(([t,s,i])=>[_(t),s,i]);l.choices(r)(t=>{k(),o.value[0].push(["choice",t],[null,0]),P()})},jump([e]){o.value[0]=[[null,e],[null,0]],l.clear(!1)(()=>{f||P()})},clear(){try{navigator.vibrate(0)}finally{l.clear(A)(m)}},condition([e]){let r=e();f||(o.value[0].push(["condition",r],[null,0]),P())},end(){U(!1,"auto").then(()=>{T("clear",[]),l.ui.showScreen("mainmenu")})},input([e,r,t]){l.input(e,r,t)(()=>{k(),m()})},custom([e]){return l.custom(e,A,()=>{!f&&e.requireUserAction&&k(),f||m()})},vibrate(e){try{navigator.vibrate(e)}finally{m()}},next(){m()},animateCharacter([e,r,...t]){let s=i=>{let v=i("novely-animate-character",!1).root.querySelector(`div[data-characters] > canvas[data-character="${e}"]`);if(!v)return;let c=t.filter(u=>!v.classList.contains(u));v.classList.add(...c),setTimeout(()=>{v.classList.remove(...c)},r)};return s.callOnlyLatest=!0,l.custom(s,A,()=>{}),m()}}),k=()=>{if(f)return;let e=H(o.value);e[0]=H(o.value[0]),e[2][0]=new Date().valueOf(),e[2][1]="auto",o.push(e)},ae=()=>{let e=o.value[0][o.value[0].length-1];if(N(e[0])&&M(e[1])){e[1]=e[1]+1;return}o.value[0].push([null,0])},P=()=>{let e=te();if(e){let[r,...t]=e;T(r,t)}},m=()=>{f||(ae(),P())},_=e=>{let r=b.get().meta[0],t=G();return se(typeof e=="function"?e(r,t):e,t)};return{withStory:Q,action:z,render:P,state:G,t:C.t}};var ce=n=>({async get(){let a=localStorage.getItem(n.key);return a?JSON.parse(a):{saves:[],meta:[]}},async set(a){localStorage.setItem(n.key,JSON.stringify(a))}});export{ce as localStorageStorage,ie as novely};
1
+ var j=a=>(n,d)=>a[n](d),M=a=>typeof a=="number",O=a=>a===null,B=a=>typeof a=="string",F=a=>{let n=String.prototype.startsWith.bind(a);return n("http")||n("/")||n(".")||n("data")},E=a=>String(a),K=(a,n)=>a==="custom"&&n[0]&&n[0].requireUserAction,L=()=>[[[null,"start"],[null,0]],{},[Date.now(),"auto"]],I=(a,n=navigator.language)=>a.includes(n)||a.includes(n=n.substring(0,2))||(n=a.find(d=>navigator.languages.includes(d)))?n:a[0],J=(a,n)=>{let d=!1,p,A;function S(){if(d){p=arguments,A=this;return}a.apply(this,arguments),d=!1}return setTimeout(function(){d=!1,p&&(S.apply(A,p),p=A=null)},n),S};var W=(a,n=new Set)=>{let d=h=>(n.add(h),h(a),()=>{n.delete(h)}),p=h=>{n.forEach(b=>b(h))};return{subscribe:d,update:h=>{p(a=h(a))},get:()=>a}};import{all as se}from"deepmerge";import{klona as H}from"klona/json";var $=new Set(["dialog","input","vibrate"]);import{replace as ie}from"@novely/t9n";var ce=({characters:a,storage:n,renderer:d,initialScreen:p="mainmenu",t9n:A,languages:S,assetsPreload:h})=>{let b,Q=e=>{b=Object.fromEntries(Object.entries(e).map(([r,t])=>{let s=i=>i.flatMap(g=>{let y=g[0];return Array.isArray(y)?s(g):[g]});return[r,s(t)]})),ae()},V={background:new Set},z=new Proxy({},{get(e,r){return(...t)=>(r==="showBackground"&&F(t[0])&&V.background.add(t[0]),[r,...t])}});function G(e){if(!e)return o.value[1];let r=o.value[1],t=typeof e=="function"?e(r):se([r,e]);o.value[1]=t}let X=(e,r=[e])=>({get value(){return r.at(-1)},set value(t){r[r.length-1]=t},back(){r.length>1&&(r.pop(),P=!0)},push(t){r.push(t)},clear(){r=[L()]}}),Y={saves:[],meta:[I(S)]},v=W(Y),q=!1,Z=J(e=>{q&&n.set(e)},120);v.subscribe(Z),n.get().then(e=>{e.meta[0]||=I(S),q=!0,v.update(()=>e)});let N=(e=>e.saves.length>0&&e.saves.at(-1))(v.get())||L(),o=X(N),U=(e=!1,r=e?"auto":"manual")=>{v.update(t=>{let s=o.value[2][0],i=t.saves.findIndex(g=>g[2][0]===s)===t.saves.length-1;return o.value[2][0]=Date.now(),o.value[2][1]=r,e&&i?t.saves[t.saves.length-1]=o.value:t.saves.push(o.value),t})},ee=()=>{v.update(e=>{let r=L();return e.saves.push(r),R(r),e})},te=e=>(o.value=e,R(e)),f=!1,P=!1,R=async e=>{let r=e||v.get().saves.at(-1);r||(v.update(()=>({saves:[N],meta:[I(S)]})),r=H(N)),f=!0,o.value=r,u.ui.showScreen("game"),T("clear",[P]);let t=b,s=0,i=o.value[0].reduce((l,[c,C])=>O(c)&&M(C)?l+1:l,0),g=[];for(let[l,c]of o.value[0])if(l===null){if(B(c))t=t[c];else if(M(c)){s++;for(let C=0;C<c;C++){let[x,...k]=t[C];if($.has(x)||K(x,k))if(s===i&&C===c)g.push([x,k]);else continue;g.push([x,k])}t=t[c]}}else l==="choice"?t=t[c+1][1]:l==="condition"&&(t=t[2][c]);let y=g.map((l,c)=>l.concat(c));for await(let[l,c,C]of y)if(l==="function"||l==="custom"){if(l==="custom"&&c[0].callOnlyLatest&&!!y.slice(C+1).some(([de,oe])=>E(oe[0])===E(c[0])))continue;let x=T(l,c);x&&"then"in x&&await x}else T(l,c);f=!1,P=!1,w()},re=()=>{let e=b;for(let[r,t]of o.value[0])r===null?e=e[t]:r==="choice"?e=e[t+1][1]:r==="condition"&&(e=e[2][t]);return e},u=d({characters:a,set:te,restore:R,save:U,newGame:ee,stack:o,languages:S,t:A.i,$:v}),ae=()=>{if(!h){p==="game"?R():u.ui.showScreen(p);return}let e=[];u.ui.showScreen("loading");for(let[r,{emotions:t}]of Object.entries(a))for(let s of Object.keys(t))e.push(Promise.resolve(u.character(r).withEmotion(s)));for(let r of V.background)e.push(new Promise((t,s)=>{let i=document.createElement("img");i.crossOrigin="*",i.src=r,i.complete&&i.naturalHeight!==0?t(void 0):(i.onload=t,i.onerror=s)}));Promise.all(e).then(()=>{u.ui.showScreen(p)})},T=j({wait([e]){f||setTimeout(m,e)},showBackground([e]){u.background(e),m()},playMusic([e]){u.music(e,"music").play(),m()},stopMusic([e]){u.music(e,"music").stop(),m()},showCharacter([e,r,t,s]){let i=u.character(e);i.append(t,s),i.withEmotion(r)(),m()},hideCharacter([e,r,t,s]){u.character(e).remove(r,t,s)(m)},dialog([e,r,t]){u.dialog(_(r),e,t)(()=>{D(),m()})},function([e]){let r=e();return f||(r?r.then(m):m()),r},choice(e){let r=e.map(([t,s,i])=>[_(t),s,i]);u.choices(r)(t=>{D(),o.value[0].push(["choice",t],[null,0]),w()})},jump([e]){o.value[0]=[[null,e],[null,0]],u.clear(!1)(()=>{f||w()})},clear(){try{navigator.vibrate(0)}finally{u.clear(P)(m)}},condition([e]){let r=e();f||(o.value[0].push(["condition",r],[null,0]),w())},end(){U(!1,"auto"),T("clear",[]),u.ui.showScreen("mainmenu")},input([e,r,t]){u.input(e,r,t)(()=>{D(),m()})},custom([e]){return u.custom(e,P,()=>{!f&&e.requireUserAction&&D(),f||m()})},vibrate(e){try{navigator.vibrate(e)}finally{m()}},next(){m()},animateCharacter([e,r,...t]){let s=i=>{let y=i("novely-animate-character",!1).root.querySelector(`div[data-characters] > canvas[data-character="${e}"]`);if(!y)return;let l=t.filter(c=>!y.classList.contains(c));y.classList.add(...l),setTimeout(()=>{y.classList.remove(...l)},r)};return s.callOnlyLatest=!0,u.custom(s,P,()=>{}),m()}}),D=()=>{if(f)return;let e=H(o.value);e[0]=H(o.value[0]),e[2][0]=new Date().valueOf(),e[2][1]="auto",o.push(e)},ne=()=>{let e=o.value[0][o.value[0].length-1];if(O(e[0])&&M(e[1])){e[1]=e[1]+1;return}o.value[0].push([null,0])},w=()=>{let e=re();if(e){let[r,...t]=e;T(r,t)}},m=()=>{f||(ne(),w())},_=e=>{let r=v.get().meta[0],t=G();return ie(typeof e=="function"?e(r,t):e,t)};return{withStory:Q,action:z,render:w,state:G,t:A.t}};var ue=a=>({async get(){let n=localStorage.getItem(a.key);return n?JSON.parse(n):{saves:[],meta:[]}},async set(n){localStorage.setItem(a.key,JSON.stringify(n))}});export{ue as localStorageStorage,ce as novely};
2
2
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/utils.ts","../src/store.ts","../src/novely.ts","../src/constants.ts","../src/storage.ts"],"sourcesContent":["import type { ActionProxyProvider, CustomHandler } from './action'\nimport type { Character } from './character'\nimport type { Save, Thenable } from './types'\n\ntype MatchActionMap = {\n [Key in keyof ActionProxyProvider<Record<string, Character>>]: (data: Parameters<ActionProxyProvider<Record<string, Character>>[Key]>) => void;\n}\n\ntype MatchActionMapComplete = Omit<MatchActionMap, 'custom'> & {\n custom: (value: [handler: CustomHandler]) => Thenable<void>;\n}\n\nconst matchAction = <M extends MatchActionMapComplete>(values: M) => {\n return (action: keyof MatchActionMap, props: any) => {\n return values[action](props);\n }\n}\n\nconst isNumber = (val: unknown): val is number => {\n return typeof val === 'number';\n}\n\nconst isNull = (val: unknown): val is null => {\n return val === null;\n}\n\nconst isString = (val: unknown): val is string => {\n return typeof val === 'string';\n}\n\nconst isCSSImage = (str: string) => {\n const startsWith = String.prototype.startsWith.bind(str);\n\n return startsWith('http') || startsWith('/') || startsWith('.') || startsWith('data');\n}\n\nconst str = (value: unknown) => {\n return String(value);\n}\n\nconst isUserRequiredAction = (action: keyof MatchActionMapComplete, meta: Parameters<MatchActionMapComplete[keyof MatchActionMapComplete]>) => {\n return action === 'custom' && meta[0] && (meta[0] as unknown as CustomHandler).requireUserAction;\n}\n\nconst getDefaultSave = () => {\n return [[[null, 'start'], [null, 0]], {}, [Date.now(), 'auto']] as Save;\n}\n\nconst getLanguage = (languages: string[], language = navigator.language) => {\n if (languages.includes(language)) {\n return language;\n } else if (languages.includes((language = language.substring(0, 2)))) {\n return language;\n } else if ((language = languages.find((value) => navigator.languages.includes(value))!)) {\n return language\n }\n\n /**\n * We'v checked the `en-GB` format, `en` format, and maybe any second languages, but there were no matches\n */\n return languages[0];\n}\n\n/**\n * @copyright Techlead LLC\n * @see https://learn.javascript.ru/task/throttle\n */\nconst throttle = <Fn extends ((...args: any[]) => any)>(fn: Fn, ms: number) => {\n let throttled = false, savedArgs: any, savedThis: any;\n\n function wrapper() {\n if (throttled) {\n savedArgs = arguments;\n // @ts-ignore\n savedThis = this;\n\n return;\n }\n\n // @ts-ignore\n fn.apply(this, arguments);\n\n throttled = false;\n }\n\n setTimeout(function () {\n throttled = false;\n\n if (savedArgs) {\n wrapper.apply(savedThis, savedArgs);\n savedArgs = savedThis = null;\n }\n }, ms);\n\n return wrapper as unknown as (...args: Parameters<Fn>) => void;\n}\n\nexport { matchAction, isNumber, isNull, isString, isCSSImage, str, isUserRequiredAction, getDefaultSave, getLanguage, throttle }","type Stored<T> = {\n subscribe: (cb: (value: T) => void) => () => void;\n update: (fn: (prev: T) => T) => void;\n get: () => T;\n}\n\nconst store = <T>(current: T, subscribers = new Set<(value: T) => void>()): Stored<T> => {\n const subscribe = (cb: (value: T) => void) => {\n subscribers.add(cb), cb(current);\n\n return () => {\n subscribers.delete(cb);\n }\n };\n\n const push = (value: T) => {\n subscribers.forEach((cb) => cb(value))\n };\n\n const update = (fn: (prev: T) => T) => {\n push((current = fn(current)));\n };\n\n const get = () => {\n return current;\n };\n\n return { subscribe, update, get } as const;\n};\n\nexport { store }\nexport type { Stored }","import type { Character } from './character';\nimport type { ActionProxyProvider, GetActionParameters, Story, ValidAction, DialogContent, ChoiceContent, CustomHandler } from './action';\nimport type { Storage } from './storage';\nimport type { Save, State, StorageData } from './types'\nimport type { Renderer, RendererInit } from './renderer'\nimport type { SetupT9N } from '@novely/t9n'\nimport { matchAction, isNumber, isNull, isString, isCSSImage, str, isUserRequiredAction, getDefaultSave, getLanguage, throttle } from './utils';\nimport { store } from './store';\nimport { all as deepmerge } from 'deepmerge'\nimport { klona } from 'klona/json';\nimport { SKIPPED_DURING_RESTORE } from './constants';\nimport { replace as replaceT9N } from '@novely/t9n';\n\ninterface NovelyInit<Languages extends string, Characters extends Record<string, Character<Languages>>, Inter extends ReturnType<SetupT9N<Languages>>> {\n /**\n * An array of languages supported by the game.\n */\n languages: Languages[];\n /**\n * An object containing the characters in the game.\n */\n characters: Characters;\n /**\n * An object that provides access to the game's storage system.\n */\n storage: Storage;\n /**\n * A function that returns a Renderer object used to display the game's content\n */\n renderer: (characters: RendererInit) => Renderer;\n /**\n * An optional property that specifies the initial screen to display when the game starts\n */\n initialScreen?: \"mainmenu\" | \"game\" | \"saves\" | \"settings\";\n /**\n * An object containing the translation functions used in the game\n */\n t9n: Inter;\n /**\n * An optional property that specifies whether to preload assets when the game starts\n */\n assetsPreload?: boolean;\n /**\n * An optional property that specifies whether the game should use a single save.\n */\n singleSave?: boolean;\n}\n\nconst novely = async <Languages extends string, Characters extends Record<string, Character<Languages>>, Inter extends ReturnType<SetupT9N<Languages>>>({ characters, storage, renderer: createRenderer, initialScreen = \"mainmenu\", t9n, languages, assetsPreload }: NovelyInit<Languages, Characters, Inter>) => {\n let story: Story;\n\n const withStory = (s: Story) => {\n /**\n * Transforms `(ValidAction | ValidAction[])[]` to `ValidAction[]`\n */\n story = Object.fromEntries(Object.entries(s).map(([name, items]) => {\n const flat = (item: (ValidAction | ValidAction[])[]): ValidAction[] => {\n return item.flatMap((data) => {\n const type = data[0];\n\n /**\n * This is not just an action like `['name', ...arguments]`, but an array of actions\n */\n if (Array.isArray(type)) return flat(data as ValidAction[]);\n\n return [data as ValidAction];\n });\n };\n\n return [name, flat(items)];\n }));\n\n /**\n * Load assets after the `action` scripts are executed\n */\n preloadAssets();\n }\n\n /**\n * This is used when background is loading\n */\n const preload = {\n background: new Set<string>(),\n }\n\n const action = new Proxy({} as ActionProxyProvider<Characters>, {\n get(_, prop) {\n return (...props: Parameters<ActionProxyProvider<Record<string, Character>>[keyof ActionProxyProvider<Record<string, Character>>]>) => {\n if (prop === 'showBackground') {\n if (isCSSImage(props[0] as string)) preload.background.add(props[0] as string);\n }\n\n return [prop, ...props];\n }\n }\n });\n\n function state(value: State | ((prev: State) => State)): void;\n function state(): State;\n function state(value?: State | ((prev: State) => State)): State | void {\n if (!value) return stack.value[1]\n\n const prev = stack.value[1];\n const val = typeof value === 'function' ? value(prev as State) : deepmerge([prev, value]);\n\n stack.value[1] = val as State;\n }\n\n const createStack = (current: Save, stack = [current]) => {\n return {\n get value() {\n return stack.at(-1)!;\n },\n set value(value: Save) {\n stack[stack.length - 1] = value;\n },\n back() {\n if (stack.length > 1) stack.pop(), goingBack = true;\n },\n push(value: Save) {\n stack.push(value);\n },\n clear() {\n stack = [getDefaultSave()];\n }\n }\n }\n\n const stored = await storage.get();\n\n /**\n * Default `localStorageStorage` cannot determine preferred language, and returns empty array\n */\n stored.meta[0] ||= getLanguage(languages);\n\n const $ = store(stored);\n\n const onStorageDataChange = (value: StorageData) => storage.set(value);\n const throttledOnStorageDataChange = throttle(onStorageDataChange, 120);\n\n $.subscribe(throttledOnStorageDataChange);\n\n const initial = ((value) => value.saves.length > 0 && value.saves.at(-1))($.get()) || getDefaultSave();\n const stack = createStack(initial);\n\n const save = async (override = false, type: Save[2][1] = override ? 'auto' : 'manual') => {\n /**\n * Получаем предыдущее значение\n */\n const prev = await storage.get();\n\n const date = stack.value[2][0];\n const isLatest = prev.saves.findIndex(value => value[2][0] === date) === prev.saves.length - 1;\n\n /**\n * Обновим дату и тип\n */\n stack.value[2][0] = Date.now();\n stack.value[2][1] = type;\n\n if (override) {\n /**\n * Перезапишем\n */\n if (isLatest) {\n /**\n * Сохранения хранятся в массиве. Нельзя перезаписать любое последнее\n * \n * Если перезаписывать старое сохранение, то они не будут идти в хронологическом порядке\n */\n prev.saves[prev.saves.length - 1] = stack.value;\n } else {\n prev.saves.push(stack.value);\n }\n } else {\n /**\n * Добавляем текущее сохранение\n */\n prev.saves.push(stack.value);\n }\n\n /**\n * Устанавливаем новое значение\n */\n return await storage.set(prev);\n }\n\n const newGame = () => {\n $.update(prev => {\n const save = getDefaultSave();\n\n prev.saves.push(save), restore(save);\n\n return prev;\n });\n }\n\n /**\n * Устанавливает сохранение\n */\n const set = (save: Save) => {\n stack.value = save;\n\n return restore(save);\n }\n\n let restoring = false;\n let goingBack = false;\n\n /**\n * Визуально восстанавливает историю\n */\n const restore = async (save?: Save) => {\n let latest = save ? save : await storage.get().then(value => value.saves.at(-1));\n\n /**\n * Если нет сохранённой игры, то запустим ту, которая уже есть\n */\n if (!latest) {\n await storage.set({ saves: [initial], meta: [getLanguage(languages)] });\n\n latest = klona(initial);\n }\n\n restoring = true, stack.value = latest;\n\n /**\n * Показать экран игры\n */\n renderer.ui.showScreen('game');\n\n match('clear', [goingBack]);\n\n /**\n * Текущий элемент в истории\n */\n let current: any = story;\n /**\n * Текущий элемент `[null, int]`\n */\n let index = 0;\n\n /**\n * Число элементов `[null, int]`\n */\n const max = stack.value[0].reduce((acc, [type, val]) => {\n if (isNull(type) && isNumber(val)) return acc + 1;\n\n return acc;\n }, 0);\n\n const queue = [] as [any, any][];\n\n for (const [type, val] of stack.value[0]) {\n if (type === null) {\n if (isString(val)) {\n current = current[val];\n } else if (isNumber(val)) {\n index++;\n\n /**\n * Запустим все экшены которые идут в `[null, int]` от `0` до `int`\n */\n for (let i = 0; i < val; i++) {\n const [action, ...meta] = current[i];\n\n /**\n * Экшены, для закрытия которых пользователь должен с ними взаимодействовать\n * Также в эту группу входят экшены, которые не должны быть вызваны при восстановлении\n */\n if (SKIPPED_DURING_RESTORE.has(action) || isUserRequiredAction(action, meta)) {\n if (index === max && i === val) {\n queue.push([action, meta]);\n } else {\n continue;\n }\n }\n\n queue.push([action, meta]);\n }\n\n current = current[val];\n }\n } else if (type === 'choice') {\n current = current[val as number + 1][1];\n } else if (type === 'condition') {\n current = current[2][val];\n }\n }\n\n const indexedQueue = queue.map((value, index) => value.concat(index) as [ValidAction[0], ValidAction[1], number]);\n\n for await (const [action, meta, i] of indexedQueue) {\n if (action === 'function' || action === 'custom') {\n /**\n * Если `callOnlyLatest` - `true`\n */\n if (action === 'custom' && (meta as GetActionParameters<'Custom'>)[0].callOnlyLatest) {\n /**\n * Вычислим `latest` или нет\n */\n const next = indexedQueue.slice(i + 1);\n const latest = !next.some(([_action, _meta]) => str(_meta[0]) === str(meta[0]));\n\n if (!latest) continue;\n }\n\n /**\n * Action может возвращать Promise. Нужно подожать его `resolve`\n */\n const result = match(action, meta);\n\n /**\n * Дождёмся окончания\n */\n if (result && 'then' in result) await result;\n } else {\n match(action as keyof ActionProxyProvider<Record<string, Character<string>>>, meta);\n }\n }\n\n restoring = false, goingBack = false, render();\n }\n\n const refer = () => {\n let current: any = story;\n\n for (const [type, val] of stack.value[0]) {\n if (type === null) {\n current = current[val];\n } else if (type === 'choice') {\n current = current[val as number + 1][1];\n } else if (type === 'condition') {\n current = current[2][val];\n }\n }\n\n return current;\n }\n\n const renderer = createRenderer({\n characters,\n storage,\n set,\n restore,\n save,\n newGame,\n stack,\n languages,\n t: t9n.i,\n $\n });\n\n const preloadAssets = () => {\n if (!assetsPreload) {\n initialScreen === 'game' ? restore() : renderer.ui.showScreen(initialScreen);\n\n return;\n }\n\n /**\n * We need to load all the characters and their emotions\n */\n\n let promises: Promise<unknown>[] = [];\n\n renderer.ui.showScreen('loading');\n\n for (const [name, { emotions }] of Object.entries(characters)) {\n for (const emotion of Object.keys(emotions)) {\n promises.push(Promise.resolve(renderer.character(name).withEmotion(emotion)));\n }\n }\n\n for (const bg of preload.background) {\n promises.push(new Promise((res, rej) => {\n /**\n * Create `img` element\n */\n const img = document.createElement('img');\n\n /**\n * Set `src`\n */\n img.crossOrigin = '*';\n img.src = bg;\n\n if (img.complete && img.naturalHeight !== 0) {\n /**\n * Image is already loaded\n */\n res(void 0);\n } else {\n /**\n * Image is uniq, it is safe to use `onload`\n */\n img.onload = res;\n img.onerror = rej;\n }\n }));\n }\n\n Promise.all(promises).then(() => {\n renderer.ui.showScreen(initialScreen);\n });\n }\n\n const match = matchAction({\n wait([time]) {\n /**\n * `restoring` может поменяться на `true` перед тем как запуститься `push` из `setTimeout`\n */\n if (!restoring) setTimeout(push, time);\n },\n showBackground([background]) {\n renderer.background(background);\n push()\n },\n playMusic([source]) {\n renderer.music(source, 'music').play();\n push()\n },\n stopMusic([source]) {\n renderer.music(source, 'music').stop();\n push()\n },\n showCharacter([character, emotion, className, style]) {\n const handle = renderer.character(character);\n\n handle.append(className, style);\n handle.withEmotion(emotion)();\n\n push()\n },\n hideCharacter([character, className, style, duration]) {\n const handle = renderer.character(character);\n\n handle.remove(className, style, duration)(push);\n },\n dialog([person, content, emotion]) {\n renderer.dialog(unwrap(content), person, emotion)(() => {\n enmemory();\n push();\n });\n },\n function([fn]) {\n const result = fn();\n\n if (!restoring) result ? result.then(push) : push();\n\n return result;\n },\n choice(choices) {\n const unwrapped = choices.map(([content, action, visible]) => {\n return [unwrap(content), action, visible] as [string, ValidAction[], () => boolean];\n });\n\n renderer.choices(unwrapped)((selected) => {\n enmemory();\n\n stack.value[0].push(['choice', selected], [null, 0]), render();\n });\n },\n jump([scene]) {\n stack.value[0] = [[null, scene], [null, 0]];\n\n renderer.clear(false)(() => {\n if (!restoring) render();\n })\n },\n clear() {\n try {\n navigator.vibrate(0)\n } finally {\n renderer.clear(goingBack)(push);\n }\n },\n condition([condition]) {\n const value = condition();\n\n if (!restoring) stack.value[0].push(['condition', value], [null, 0]), render();\n },\n end() {\n save(false, 'auto').then(() => {\n match('clear', []);\n\n renderer.ui.showScreen('mainmenu');\n });\n },\n input([question, onInput, setup]) {\n renderer.input(question, onInput, setup)(() => {\n enmemory();\n push();\n });\n },\n custom([handler]) {\n const result = renderer.custom(handler, goingBack, () => {\n if (!restoring && handler.requireUserAction) enmemory();\n if (!restoring) push();\n });\n\n return result;\n },\n vibrate(pattern) {\n try {\n navigator.vibrate(pattern)\n } finally {\n push()\n }\n },\n next() {\n push();\n },\n animateCharacter([character, timeout, ...classes]) {\n const handler: CustomHandler = (get) => {\n const root = get('novely-animate-character', false).root;\n const target = root.querySelector(`div[data-characters] > canvas[data-character=\"${character}\"]`);\n\n /**\n * Character is not found in the DOM\n */\n if (!target) return;\n\n const classNames = classes.filter(className => !target.classList.contains(className));\n\n target.classList.add(...classNames);\n\n setTimeout(() => {\n target.classList.remove(...classNames);\n }, timeout);\n }\n\n handler.callOnlyLatest = true;\n\n return renderer.custom(handler, goingBack, () => { }), push();\n }\n });\n\n const enmemory = () => {\n if (restoring) return;\n\n const current = klona(stack.value);\n\n current[0] = klona(stack.value[0]);\n\n current[2][0] = new Date().valueOf();\n current[2][1] = 'auto';\n\n stack.push(current);\n }\n\n const next = () => {\n /**\n * Последний элемент пути\n */\n const last = stack.value[0][stack.value[0].length - 1]!;\n\n /**\n * Если он вида `[null, int]` - увеличивает `int`\n */\n if (isNull(last[0]) && isNumber(last[1])) {\n last[1] = last[1] + 1;\n return;\n }\n\n /**\n * Иначе добавляет новое `[null int]`\n */\n stack.value[0].push([null, 0]);\n }\n\n const render = () => {\n const referred = refer();\n\n if (referred) {\n const [action, ...props] = referred;\n\n match(action, props);\n }\n }\n\n const push = () => {\n if (!restoring) next(), render();\n }\n\n const unwrap = (content: DialogContent | ChoiceContent) => {\n const lang = $.get().meta[0];\n const data = state();\n\n return replaceT9N(typeof content === 'function' ? content(lang, data) : content, data);\n }\n\n return {\n withStory,\n action,\n render,\n state,\n t: t9n.t,\n }\n}\n\nexport { novely }\n","const SKIPPED_DURING_RESTORE = new Set([\n 'dialog',\n 'input',\n 'vibrate'\n] as const);\n\nexport { SKIPPED_DURING_RESTORE }","import type { StorageData } from './types'\n\ninterface LocalStorageStorageSettings {\n key: string\n}\n\ninterface Storage {\n get: () => Promise<StorageData>;\n set: (data: StorageData) => Promise<void>;\n}\n\nconst localStorageStorage = (options: LocalStorageStorageSettings): Storage => {\n return {\n async get() {\n const value = localStorage.getItem(options.key);\n\n return value ? JSON.parse(value) : { saves: [], meta: [] };\n },\n async set(data) {\n localStorage.setItem(options.key, JSON.stringify(data));\n }\n }\n}\n\nexport type { Storage }\nexport { localStorageStorage }"],"mappings":"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","names":["matchAction","values","action","props","isNumber","val","isNull","isString","isCSSImage","str","startsWith","value","isUserRequiredAction","meta","getDefaultSave","getLanguage","languages","language","throttle","fn","ms","throttled","savedArgs","savedThis","wrapper","store","current","subscribers","subscribe","cb","push","value","fn","deepmerge","klona","SKIPPED_DURING_RESTORE","replaceT9N","novely","characters","storage","createRenderer","initialScreen","t9n","languages","assetsPreload","story","withStory","s","name","items","flat","item","data","type","preloadAssets","preload","action","_","prop","props","isCSSImage","state","value","stack","prev","val","deepmerge","createStack","current","goingBack","getDefaultSave","stored","getLanguage","$","store","throttledOnStorageDataChange","throttle","initial","save","override","date","isLatest","newGame","restore","set","restoring","latest","klona","renderer","match","index","max","acc","isNull","isNumber","queue","isString","i","meta","SKIPPED_DURING_RESTORE","isUserRequiredAction","indexedQueue","_action","_meta","str","result","render","refer","promises","emotions","emotion","bg","res","rej","img","matchAction","time","push","background","source","character","className","style","handle","duration","person","content","unwrap","enmemory","fn","choices","unwrapped","visible","selected","scene","condition","question","onInput","setup","handler","pattern","timeout","classes","get","target","classNames","next","last","referred","lang","localStorageStorage","options","value","data"]}
1
+ {"version":3,"sources":["../src/utils.ts","../src/store.ts","../src/novely.ts","../src/constants.ts","../src/storage.ts"],"sourcesContent":["import type { ActionProxyProvider, CustomHandler } from './action'\nimport type { Character } from './character'\nimport type { Save, Thenable } from './types'\n\ntype MatchActionMap = {\n [Key in keyof ActionProxyProvider<Record<string, Character>>]: (data: Parameters<ActionProxyProvider<Record<string, Character>>[Key]>) => void;\n}\n\ntype MatchActionMapComplete = Omit<MatchActionMap, 'custom'> & {\n custom: (value: [handler: CustomHandler]) => Thenable<void>;\n}\n\nconst matchAction = <M extends MatchActionMapComplete>(values: M) => {\n return (action: keyof MatchActionMap, props: any) => {\n return values[action](props);\n }\n}\n\nconst isNumber = (val: unknown): val is number => {\n return typeof val === 'number';\n}\n\nconst isNull = (val: unknown): val is null => {\n return val === null;\n}\n\nconst isString = (val: unknown): val is string => {\n return typeof val === 'string';\n}\n\nconst isCSSImage = (str: string) => {\n const startsWith = String.prototype.startsWith.bind(str);\n\n return startsWith('http') || startsWith('/') || startsWith('.') || startsWith('data');\n}\n\nconst str = (value: unknown) => {\n return String(value);\n}\n\nconst isUserRequiredAction = (action: keyof MatchActionMapComplete, meta: Parameters<MatchActionMapComplete[keyof MatchActionMapComplete]>) => {\n return action === 'custom' && meta[0] && (meta[0] as unknown as CustomHandler).requireUserAction;\n}\n\nconst getDefaultSave = () => {\n return [[[null, 'start'], [null, 0]], {}, [Date.now(), 'auto']] as Save;\n}\n\nconst getLanguage = (languages: string[], language = navigator.language) => {\n if (languages.includes(language)) {\n return language;\n } else if (languages.includes((language = language.substring(0, 2)))) {\n return language;\n } else if ((language = languages.find((value) => navigator.languages.includes(value))!)) {\n return language\n }\n\n /**\n * We'v checked the `en-GB` format, `en` format, and maybe any second languages, but there were no matches\n */\n return languages[0];\n}\n\n/**\n * @copyright Techlead LLC\n * @see https://learn.javascript.ru/task/throttle\n */\nconst throttle = <Fn extends ((...args: any[]) => any)>(fn: Fn, ms: number) => {\n let throttled = false, savedArgs: any, savedThis: any;\n\n function wrapper() {\n if (throttled) {\n savedArgs = arguments;\n // @ts-ignore\n savedThis = this;\n\n return;\n }\n\n // @ts-ignore\n fn.apply(this, arguments);\n\n throttled = false;\n }\n\n setTimeout(function () {\n throttled = false;\n\n if (savedArgs) {\n wrapper.apply(savedThis, savedArgs);\n savedArgs = savedThis = null;\n }\n }, ms);\n\n return wrapper as unknown as (...args: Parameters<Fn>) => void;\n}\n\nexport { matchAction, isNumber, isNull, isString, isCSSImage, str, isUserRequiredAction, getDefaultSave, getLanguage, throttle }","type Stored<T> = {\n subscribe: (cb: (value: T) => void) => () => void;\n update: (fn: (prev: T) => T) => void;\n get: () => T;\n}\n\nconst store = <T>(current: T, subscribers = new Set<(value: T) => void>()): Stored<T> => {\n const subscribe = (cb: (value: T) => void) => {\n subscribers.add(cb), cb(current);\n\n return () => {\n subscribers.delete(cb);\n }\n };\n\n const push = (value: T) => {\n subscribers.forEach((cb) => cb(value))\n };\n\n const update = (fn: (prev: T) => T) => {\n push((current = fn(current)));\n };\n\n const get = () => {\n return current;\n };\n\n return { subscribe, update, get } as const;\n};\n\nexport { store }\nexport type { Stored }","import type { Character } from './character';\nimport type { ActionProxyProvider, GetActionParameters, Story, ValidAction, DialogContent, ChoiceContent, CustomHandler } from './action';\nimport type { Storage } from './storage';\nimport type { Save, State, StorageData } from './types'\nimport type { Renderer, RendererInit } from './renderer'\nimport type { SetupT9N } from '@novely/t9n'\nimport { matchAction, isNumber, isNull, isString, isCSSImage, str, isUserRequiredAction, getDefaultSave, getLanguage, throttle } from './utils';\nimport { store } from './store';\nimport { all as deepmerge } from 'deepmerge'\nimport { klona } from 'klona/json';\nimport { SKIPPED_DURING_RESTORE } from './constants';\nimport { replace as replaceT9N } from '@novely/t9n';\n\ninterface NovelyInit<Languages extends string, Characters extends Record<string, Character<Languages>>, Inter extends ReturnType<SetupT9N<Languages>>> {\n /**\n * An array of languages supported by the game.\n */\n languages: Languages[];\n /**\n * An object containing the characters in the game.\n */\n characters: Characters;\n /**\n * An object that provides access to the game's storage system.\n */\n storage: Storage;\n /**\n * A function that returns a Renderer object used to display the game's content\n */\n renderer: (characters: RendererInit) => Renderer;\n /**\n * An optional property that specifies the initial screen to display when the game starts\n */\n initialScreen?: \"mainmenu\" | \"game\" | \"saves\" | \"settings\";\n /**\n * An object containing the translation functions used in the game\n */\n t9n: Inter;\n /**\n * An optional property that specifies whether to preload assets when the game starts\n */\n assetsPreload?: boolean;\n /**\n * An optional property that specifies whether the game should use a single save.\n */\n singleSave?: boolean;\n}\n\nconst novely = <Languages extends string, Characters extends Record<string, Character<Languages>>, Inter extends ReturnType<SetupT9N<Languages>>>({ characters, storage, renderer: createRenderer, initialScreen = \"mainmenu\", t9n, languages, assetsPreload }: NovelyInit<Languages, Characters, Inter>) => {\n let story: Story;\n\n const withStory = (s: Story) => {\n /**\n * Transforms `(ValidAction | ValidAction[])[]` to `ValidAction[]`\n */\n story = Object.fromEntries(Object.entries(s).map(([name, items]) => {\n const flat = (item: (ValidAction | ValidAction[])[]): ValidAction[] => {\n return item.flatMap((data) => {\n const type = data[0];\n\n /**\n * This is not just an action like `['name', ...arguments]`, but an array of actions\n */\n if (Array.isArray(type)) return flat(data as ValidAction[]);\n\n return [data as ValidAction];\n });\n };\n\n return [name, flat(items)];\n }));\n\n /**\n * Load assets after the `action` scripts are executed\n */\n preloadAssets();\n }\n\n /**\n * This is used when background is loading\n */\n const preload = {\n background: new Set<string>(),\n }\n\n const action = new Proxy({} as ActionProxyProvider<Characters>, {\n get(_, prop) {\n return (...props: Parameters<ActionProxyProvider<Record<string, Character>>[keyof ActionProxyProvider<Record<string, Character>>]>) => {\n if (prop === 'showBackground') {\n if (isCSSImage(props[0] as string)) preload.background.add(props[0] as string);\n }\n\n return [prop, ...props];\n }\n }\n });\n\n function state(value: State | ((prev: State) => State)): void;\n function state(): State;\n function state(value?: State | ((prev: State) => State)): State | void {\n if (!value) return stack.value[1]\n\n const prev = stack.value[1];\n const val = typeof value === 'function' ? value(prev as State) : deepmerge([prev, value]);\n\n stack.value[1] = val as State;\n }\n\n const createStack = (current: Save, stack = [current]) => {\n return {\n get value() {\n return stack.at(-1)!;\n },\n set value(value: Save) {\n stack[stack.length - 1] = value;\n },\n back() {\n if (stack.length > 1) stack.pop(), goingBack = true;\n },\n push(value: Save) {\n stack.push(value);\n },\n clear() {\n stack = [getDefaultSave()];\n }\n }\n }\n\n /**\n * 1) Novely rendered using the `initialData`, you can still start new game or `load` an empty one - this is scary, imagine losing your progress\n * 2) Actual stored data is loaded, language and etc is changed \n */\n const initialData: StorageData = {\n saves: [],\n meta: [getLanguage(languages)]\n };\n\n const $ = store(initialData);\n\n let initialDataLoaded = false;\n\n const onStorageDataChange = (value: StorageData) => {\n if (initialDataLoaded) storage.set(value);\n };\n\n const throttledOnStorageDataChange = throttle(onStorageDataChange, 120);\n\n $.subscribe(throttledOnStorageDataChange);\n\n storage.get().then(stored => {\n /**\n * Default `localStorageStorage` cannot determine preferred language, and returns empty array\n */\n stored.meta[0] ||= getLanguage(languages);\n\n /**\n * Now the next store updates will entail saving via storage.set\n */\n initialDataLoaded = true;\n\n $.update(() => stored);\n });\n\n const initial = ((value) => value.saves.length > 0 && value.saves.at(-1))($.get()) || getDefaultSave();\n const stack = createStack(initial);\n\n const save = (override = false, type: Save[2][1] = override ? 'auto' : 'manual') => {\n $.update(prev => {\n const date = stack.value[2][0];\n const isLatest = prev.saves.findIndex(value => value[2][0] === date) === prev.saves.length - 1;\n\n /**\n * Обновим дату и тип\n */\n stack.value[2][0] = Date.now();\n stack.value[2][1] = type;\n\n if (override) {\n /**\n * Перезапишем\n */\n if (isLatest) {\n /**\n * Сохранения хранятся в массиве. Нельзя перезаписать любое последнее\n * \n * Если перезаписывать старое сохранение, то они не будут идти в хронологическом порядке\n */\n prev.saves[prev.saves.length - 1] = stack.value;\n } else {\n prev.saves.push(stack.value);\n }\n } else {\n /**\n * Добавляем текущее сохранение\n */\n prev.saves.push(stack.value);\n }\n\n /**\n * Устанавливаем новое значение\n */\n return prev;\n });\n }\n\n const newGame = () => {\n $.update(prev => {\n const save = getDefaultSave();\n\n prev.saves.push(save), restore(save);\n\n return prev;\n });\n }\n\n /**\n * Устанавливает сохранение\n */\n const set = (save: Save) => {\n stack.value = save;\n\n return restore(save);\n }\n\n let restoring = false;\n let goingBack = false;\n\n /**\n * Визуально восстанавливает историю\n */\n const restore = async (save?: Save) => {\n let latest = save ? save : $.get().saves.at(-1);\n\n /**\n * Если нет сохранённой игры, то запустим ту, которая уже есть\n */\n if (!latest) {\n $.update(() => ({ saves: [initial], meta: [getLanguage(languages)] }));\n\n latest = klona(initial);\n }\n\n restoring = true, stack.value = latest;\n\n /**\n * Показать экран игры\n */\n renderer.ui.showScreen('game');\n\n match('clear', [goingBack]);\n\n /**\n * Текущий элемент в истории\n */\n let current: any = story;\n /**\n * Текущий элемент `[null, int]`\n */\n let index = 0;\n\n /**\n * Число элементов `[null, int]`\n */\n const max = stack.value[0].reduce((acc, [type, val]) => {\n if (isNull(type) && isNumber(val)) return acc + 1;\n\n return acc;\n }, 0);\n\n const queue = [] as [any, any][];\n\n for (const [type, val] of stack.value[0]) {\n if (type === null) {\n if (isString(val)) {\n current = current[val];\n } else if (isNumber(val)) {\n index++;\n\n /**\n * Запустим все экшены которые идут в `[null, int]` от `0` до `int`\n */\n for (let i = 0; i < val; i++) {\n const [action, ...meta] = current[i];\n\n /**\n * Экшены, для закрытия которых пользователь должен с ними взаимодействовать\n * Также в эту группу входят экшены, которые не должны быть вызваны при восстановлении\n */\n if (SKIPPED_DURING_RESTORE.has(action) || isUserRequiredAction(action, meta)) {\n if (index === max && i === val) {\n queue.push([action, meta]);\n } else {\n continue;\n }\n }\n\n queue.push([action, meta]);\n }\n\n current = current[val];\n }\n } else if (type === 'choice') {\n current = current[val as number + 1][1];\n } else if (type === 'condition') {\n current = current[2][val];\n }\n }\n\n const indexedQueue = queue.map((value, index) => value.concat(index) as [ValidAction[0], ValidAction[1], number]);\n\n for await (const [action, meta, i] of indexedQueue) {\n if (action === 'function' || action === 'custom') {\n /**\n * Если `callOnlyLatest` - `true`\n */\n if (action === 'custom' && (meta as GetActionParameters<'Custom'>)[0].callOnlyLatest) {\n /**\n * Вычислим `latest` или нет\n */\n const next = indexedQueue.slice(i + 1);\n const latest = !next.some(([_action, _meta]) => str(_meta[0]) === str(meta[0]));\n\n if (!latest) continue;\n }\n\n /**\n * Action может возвращать Promise. Нужно подожать его `resolve`\n */\n const result = match(action, meta);\n\n /**\n * Дождёмся окончания\n */\n if (result && 'then' in result) await result;\n } else {\n match(action as keyof ActionProxyProvider<Record<string, Character<string>>>, meta);\n }\n }\n\n restoring = false, goingBack = false, render();\n }\n\n const refer = () => {\n let current: any = story;\n\n for (const [type, val] of stack.value[0]) {\n if (type === null) {\n current = current[val];\n } else if (type === 'choice') {\n current = current[val as number + 1][1];\n } else if (type === 'condition') {\n current = current[2][val];\n }\n }\n\n return current;\n }\n\n const renderer = createRenderer({\n characters,\n set,\n restore,\n save,\n newGame,\n stack,\n languages,\n t: t9n.i,\n $\n });\n\n const preloadAssets = () => {\n if (!assetsPreload) {\n initialScreen === 'game' ? restore() : renderer.ui.showScreen(initialScreen);\n\n return;\n }\n\n /**\n * We need to load all the characters and their emotions\n */\n\n let promises: Promise<unknown>[] = [];\n\n renderer.ui.showScreen('loading');\n\n for (const [name, { emotions }] of Object.entries(characters)) {\n for (const emotion of Object.keys(emotions)) {\n promises.push(Promise.resolve(renderer.character(name).withEmotion(emotion)));\n }\n }\n\n for (const bg of preload.background) {\n promises.push(new Promise((res, rej) => {\n /**\n * Create `img` element\n */\n const img = document.createElement('img');\n\n /**\n * Set `src`\n */\n img.crossOrigin = '*';\n img.src = bg;\n\n if (img.complete && img.naturalHeight !== 0) {\n /**\n * Image is already loaded\n */\n res(void 0);\n } else {\n /**\n * Image is uniq, it is safe to use `onload`\n */\n img.onload = res;\n img.onerror = rej;\n }\n }));\n }\n\n Promise.all(promises).then(() => {\n renderer.ui.showScreen(initialScreen);\n });\n }\n\n const match = matchAction({\n wait([time]) {\n /**\n * `restoring` может поменяться на `true` перед тем как запуститься `push` из `setTimeout`\n */\n if (!restoring) setTimeout(push, time);\n },\n showBackground([background]) {\n renderer.background(background);\n push()\n },\n playMusic([source]) {\n renderer.music(source, 'music').play();\n push()\n },\n stopMusic([source]) {\n renderer.music(source, 'music').stop();\n push()\n },\n showCharacter([character, emotion, className, style]) {\n const handle = renderer.character(character);\n\n handle.append(className, style);\n handle.withEmotion(emotion)();\n\n push()\n },\n hideCharacter([character, className, style, duration]) {\n const handle = renderer.character(character);\n\n handle.remove(className, style, duration)(push);\n },\n dialog([person, content, emotion]) {\n renderer.dialog(unwrap(content), person, emotion)(() => {\n enmemory();\n push();\n });\n },\n function([fn]) {\n const result = fn();\n\n if (!restoring) result ? result.then(push) : push();\n\n return result;\n },\n choice(choices) {\n const unwrapped = choices.map(([content, action, visible]) => {\n return [unwrap(content), action, visible] as [string, ValidAction[], () => boolean];\n });\n\n renderer.choices(unwrapped)((selected) => {\n enmemory();\n\n stack.value[0].push(['choice', selected], [null, 0]), render();\n });\n },\n jump([scene]) {\n stack.value[0] = [[null, scene], [null, 0]];\n\n renderer.clear(false)(() => {\n if (!restoring) render();\n })\n },\n clear() {\n try {\n navigator.vibrate(0)\n } finally {\n renderer.clear(goingBack)(push);\n }\n },\n condition([condition]) {\n const value = condition();\n\n if (!restoring) stack.value[0].push(['condition', value], [null, 0]), render();\n },\n end() {\n /**\n * Save Current Game\n */\n save(false, 'auto');\n /**\n * Clear the Scene\n */\n match('clear', []);\n /**\n * Go to the main menu\n */\n renderer.ui.showScreen('mainmenu');\n },\n input([question, onInput, setup]) {\n renderer.input(question, onInput, setup)(() => {\n enmemory();\n push();\n });\n },\n custom([handler]) {\n const result = renderer.custom(handler, goingBack, () => {\n if (!restoring && handler.requireUserAction) enmemory();\n if (!restoring) push();\n });\n\n return result;\n },\n vibrate(pattern) {\n try {\n navigator.vibrate(pattern)\n } finally {\n push()\n }\n },\n next() {\n push();\n },\n animateCharacter([character, timeout, ...classes]) {\n const handler: CustomHandler = (get) => {\n const root = get('novely-animate-character', false).root;\n const target = root.querySelector(`div[data-characters] > canvas[data-character=\"${character}\"]`);\n\n /**\n * Character is not found in the DOM\n */\n if (!target) return;\n\n const classNames = classes.filter(className => !target.classList.contains(className));\n\n target.classList.add(...classNames);\n\n setTimeout(() => {\n target.classList.remove(...classNames);\n }, timeout);\n }\n\n handler.callOnlyLatest = true;\n\n return renderer.custom(handler, goingBack, () => { }), push();\n }\n });\n\n const enmemory = () => {\n if (restoring) return;\n\n const current = klona(stack.value);\n\n current[0] = klona(stack.value[0]);\n\n current[2][0] = new Date().valueOf();\n current[2][1] = 'auto';\n\n stack.push(current);\n }\n\n const next = () => {\n /**\n * Последний элемент пути\n */\n const last = stack.value[0][stack.value[0].length - 1]!;\n\n /**\n * Если он вида `[null, int]` - увеличивает `int`\n */\n if (isNull(last[0]) && isNumber(last[1])) {\n last[1] = last[1] + 1;\n return;\n }\n\n /**\n * Иначе добавляет новое `[null int]`\n */\n stack.value[0].push([null, 0]);\n }\n\n const render = () => {\n const referred = refer();\n\n if (referred) {\n const [action, ...props] = referred;\n\n match(action, props);\n }\n }\n\n const push = () => {\n if (!restoring) next(), render();\n }\n\n const unwrap = (content: DialogContent | ChoiceContent) => {\n const lang = $.get().meta[0];\n const data = state();\n\n return replaceT9N(typeof content === 'function' ? content(lang, data) : content, data);\n }\n\n return {\n withStory,\n action,\n render,\n state,\n t: t9n.t,\n }\n}\n\nexport { novely }\n","const SKIPPED_DURING_RESTORE = new Set([\n 'dialog',\n 'input',\n 'vibrate'\n] as const);\n\nexport { SKIPPED_DURING_RESTORE }","import type { StorageData } from './types'\n\ninterface LocalStorageStorageSettings {\n key: string\n}\n\ninterface Storage {\n get: () => Promise<StorageData>;\n set: (data: StorageData) => Promise<void>;\n}\n\nconst localStorageStorage = (options: LocalStorageStorageSettings): Storage => {\n return {\n async get() {\n const value = localStorage.getItem(options.key);\n\n return value ? JSON.parse(value) : { saves: [], meta: [] };\n },\n async set(data) {\n localStorage.setItem(options.key, JSON.stringify(data));\n }\n }\n}\n\nexport type { Storage }\nexport { localStorageStorage }"],"mappings":"AAYA,IAAMA,EAAiDC,GAC9C,CAACC,EAA8BC,IAC7BF,EAAOC,CAAM,EAAEC,CAAK,EAIzBC,EAAYC,GACT,OAAOA,GAAQ,SAGlBC,EAAUD,GACPA,IAAQ,KAGXE,EAAYF,GACT,OAAOA,GAAQ,SAGlBG,EAAcC,GAAgB,CAClC,IAAMC,EAAa,OAAO,UAAU,WAAW,KAAKD,CAAG,EAEvD,OAAOC,EAAW,MAAM,GAAKA,EAAW,GAAG,GAAKA,EAAW,GAAG,GAAKA,EAAW,MAAM,CACtF,EAEMD,EAAOE,GACJ,OAAOA,CAAK,EAGfC,EAAuB,CAACV,EAAsCW,IAC3DX,IAAW,UAAYW,EAAK,CAAC,GAAMA,EAAK,CAAC,EAA+B,kBAG3EC,EAAiB,IACd,CAAC,CAAC,CAAC,KAAM,OAAO,EAAG,CAAC,KAAM,CAAC,CAAC,EAAG,CAAC,EAAG,CAAC,KAAK,IAAI,EAAG,MAAM,CAAC,EAG1DC,EAAc,CAACC,EAAqBC,EAAW,UAAU,WACzDD,EAAU,SAASC,CAAQ,GAEpBD,EAAU,SAAUC,EAAWA,EAAS,UAAU,EAAG,CAAC,CAAE,IAEvDA,EAAWD,EAAU,KAAML,GAAU,UAAU,UAAU,SAASA,CAAK,CAAC,GAH3EM,EAUFD,EAAU,CAAC,EAOdE,EAAW,CAAuCC,EAAQC,IAAe,CAC7E,IAAIC,EAAY,GAAOC,EAAgBC,EAEvC,SAASC,GAAU,CACjB,GAAIH,EAAW,CACbC,EAAY,UAEZC,EAAY,KAEZ,OAIFJ,EAAG,MAAM,KAAM,SAAS,EAExBE,EAAY,EACd,CAEA,kBAAW,UAAY,CACrBA,EAAY,GAERC,IACFE,EAAQ,MAAMD,EAAWD,CAAS,EAClCA,EAAYC,EAAY,KAE5B,EAAGH,CAAE,EAEEI,CACT,ECzFA,IAAMC,EAAQ,CAAIC,EAAYC,EAAc,IAAI,MAAyC,CACvF,IAAMC,EAAaC,IACjBF,EAAY,IAAIE,CAAE,EAAGA,EAAGH,CAAO,EAExB,IAAM,CACXC,EAAY,OAAOE,CAAE,CACvB,GAGIC,EAAQC,GAAa,CACzBJ,EAAY,QAASE,GAAOA,EAAGE,CAAK,CAAC,CACvC,EAUA,MAAO,CAAE,UAAAH,EAAW,OARJI,GAAuB,CACrCF,EAAMJ,EAAUM,EAAGN,CAAO,CAAE,CAC9B,EAM4B,IAJhB,IACHA,CAGuB,CAClC,ECpBA,OAAS,OAAOO,OAAiB,YACjC,OAAS,SAAAC,MAAa,aCTtB,IAAMC,EAAyB,IAAI,IAAI,CACrC,SACA,QACA,SACF,CAAU,EDOV,OAAS,WAAWC,OAAkB,cAqCtC,IAAMC,GAAS,CAAmI,CAAE,WAAAC,EAAY,QAAAC,EAAS,SAAUC,EAAgB,cAAAC,EAAgB,WAAY,IAAAC,EAAK,UAAAC,EAAW,cAAAC,CAAc,IAAgD,CAC3S,IAAIC,EAEEC,EAAaC,GAAa,CAI9BF,EAAQ,OAAO,YAAY,OAAO,QAAQE,CAAC,EAAE,IAAI,CAAC,CAACC,EAAMC,CAAK,IAAM,CAClE,IAAMC,EAAQC,GACLA,EAAK,QAASC,GAAS,CAC5B,IAAMC,EAAOD,EAAK,CAAC,EAKnB,OAAI,MAAM,QAAQC,CAAI,EAAUH,EAAKE,CAAqB,EAEnD,CAACA,CAAmB,CAC7B,CAAC,EAGH,MAAO,CAACJ,EAAME,EAAKD,CAAK,CAAC,CAC3B,CAAC,CAAC,EAKFK,GAAc,CAChB,EAKMC,EAAU,CACd,WAAY,IAAI,GAClB,EAEMC,EAAS,IAAI,MAAM,CAAC,EAAsC,CAC9D,IAAIC,EAAGC,EAAM,CACX,MAAO,IAAIC,KACLD,IAAS,kBACPE,EAAWD,EAAM,CAAC,CAAW,GAAGJ,EAAQ,WAAW,IAAII,EAAM,CAAC,CAAW,EAGxE,CAACD,EAAM,GAAGC,CAAK,EAE1B,CACF,CAAC,EAID,SAASE,EAAMC,EAAwD,CACrE,GAAI,CAACA,EAAO,OAAOC,EAAM,MAAM,CAAC,EAEhC,IAAMC,EAAOD,EAAM,MAAM,CAAC,EACpBE,EAAM,OAAOH,GAAU,WAAaA,EAAME,CAAa,EAAIE,GAAU,CAACF,EAAMF,CAAK,CAAC,EAExFC,EAAM,MAAM,CAAC,EAAIE,CACnB,CAEA,IAAME,EAAc,CAACC,EAAeL,EAAQ,CAACK,CAAO,KAC3C,CACL,IAAI,OAAQ,CACV,OAAOL,EAAM,GAAG,EAAE,CACpB,EACA,IAAI,MAAMD,EAAa,CACrBC,EAAMA,EAAM,OAAS,CAAC,EAAID,CAC5B,EACA,MAAO,CACDC,EAAM,OAAS,IAAGA,EAAM,IAAI,EAAGM,EAAY,GACjD,EACA,KAAKP,EAAa,CAChBC,EAAM,KAAKD,CAAK,CAClB,EACA,OAAQ,CACNC,EAAQ,CAACO,EAAe,CAAC,CAC3B,CACF,GAOIC,EAA2B,CAC/B,MAAO,CAAC,EACR,KAAM,CAACC,EAAY7B,CAAS,CAAC,CAC/B,EAEM8B,EAAIC,EAAMH,CAAW,EAEvBI,EAAoB,GAMlBC,EAA+BC,EAJRf,GAAuB,CAC9Ca,GAAmBpC,EAAQ,IAAIuB,CAAK,CAC1C,EAEmE,GAAG,EAEtEW,EAAE,UAAUG,CAA4B,EAExCrC,EAAQ,IAAI,EAAE,KAAKuC,GAAU,CAI3BA,EAAO,KAAK,CAAC,IAAMN,EAAY7B,CAAS,EAKxCgC,EAAoB,GAEpBF,EAAE,OAAO,IAAMK,CAAM,CACvB,CAAC,EAED,IAAMC,GAAYjB,GAAUA,EAAM,MAAM,OAAS,GAAKA,EAAM,MAAM,GAAG,EAAE,GAAGW,EAAE,IAAI,CAAC,GAAKH,EAAe,EAC/FP,EAAQI,EAAYY,CAAO,EAE3BC,EAAO,CAACC,EAAW,GAAO5B,EAAmB4B,EAAW,OAAS,WAAa,CAClFR,EAAE,OAAOT,GAAQ,CACf,IAAMkB,EAAOnB,EAAM,MAAM,CAAC,EAAE,CAAC,EACvBoB,EAAWnB,EAAK,MAAM,UAAUF,GAASA,EAAM,CAAC,EAAE,CAAC,IAAMoB,CAAI,IAAMlB,EAAK,MAAM,OAAS,EAK7F,OAAAD,EAAM,MAAM,CAAC,EAAE,CAAC,EAAI,KAAK,IAAI,EAC7BA,EAAM,MAAM,CAAC,EAAE,CAAC,EAAIV,EAEhB4B,GAIEE,EAMFnB,EAAK,MAAMA,EAAK,MAAM,OAAS,CAAC,EAAID,EAAM,MAQ5CC,EAAK,MAAM,KAAKD,EAAM,KAAK,EAMtBC,CACT,CAAC,CACH,EAEMoB,GAAU,IAAM,CACpBX,EAAE,OAAOT,GAAQ,CACf,IAAMgB,EAAOV,EAAe,EAE5B,OAAAN,EAAK,MAAM,KAAKgB,CAAI,EAAGK,EAAQL,CAAI,EAE5BhB,CACT,CAAC,CACH,EAKMsB,GAAON,IACXjB,EAAM,MAAQiB,EAEPK,EAAQL,CAAI,GAGjBO,EAAY,GACZlB,EAAY,GAKVgB,EAAU,MAAOL,GAAgB,CACrC,IAAIQ,EAASR,GAAcP,EAAE,IAAI,EAAE,MAAM,GAAG,EAAE,EAKzCe,IACHf,EAAE,OAAO,KAAO,CAAE,MAAO,CAACM,CAAO,EAAG,KAAM,CAACP,EAAY7B,CAAS,CAAC,CAAE,EAAE,EAErE6C,EAASC,EAAMV,CAAO,GAGxBQ,EAAY,GAAMxB,EAAM,MAAQyB,EAKhCE,EAAS,GAAG,WAAW,MAAM,EAE7BC,EAAM,QAAS,CAACtB,CAAS,CAAC,EAK1B,IAAID,EAAevB,EAIf+C,EAAQ,EAKNC,EAAM9B,EAAM,MAAM,CAAC,EAAE,OAAO,CAAC+B,EAAK,CAACzC,EAAMY,CAAG,IAC5C8B,EAAO1C,CAAI,GAAK2C,EAAS/B,CAAG,EAAU6B,EAAM,EAEzCA,EACN,CAAC,EAEEG,EAAQ,CAAC,EAEf,OAAW,CAAC5C,EAAMY,CAAG,IAAKF,EAAM,MAAM,CAAC,EACrC,GAAIV,IAAS,MACX,GAAI6C,EAASjC,CAAG,EACdG,EAAUA,EAAQH,CAAG,UACZ+B,EAAS/B,CAAG,EAAG,CACxB2B,IAKA,QAASO,EAAI,EAAGA,EAAIlC,EAAKkC,IAAK,CAC5B,GAAM,CAAC3C,EAAQ,GAAG4C,CAAI,EAAIhC,EAAQ+B,CAAC,EAMnC,GAAIE,EAAuB,IAAI7C,CAAM,GAAK8C,EAAqB9C,EAAQ4C,CAAI,EACzE,GAAIR,IAAUC,GAAOM,IAAMlC,EACzBgC,EAAM,KAAK,CAACzC,EAAQ4C,CAAI,CAAC,MAEzB,UAIJH,EAAM,KAAK,CAACzC,EAAQ4C,CAAI,CAAC,EAG3BhC,EAAUA,EAAQH,CAAG,QAEdZ,IAAS,SAClBe,EAAUA,EAAQH,EAAgB,CAAC,EAAE,CAAC,EAC7BZ,IAAS,cAClBe,EAAUA,EAAQ,CAAC,EAAEH,CAAG,GAI5B,IAAMsC,EAAeN,EAAM,IAAI,CAACnC,EAAO8B,IAAU9B,EAAM,OAAO8B,CAAK,CAA6C,EAEhH,aAAiB,CAACpC,EAAQ4C,EAAMD,CAAC,IAAKI,EACpC,GAAI/C,IAAW,YAAcA,IAAW,SAAU,CAIhD,GAAIA,IAAW,UAAa4C,EAAuC,CAAC,EAAE,gBAOhE,CAFW,CADFG,EAAa,MAAMJ,EAAI,CAAC,EAChB,KAAK,CAAC,CAACK,GAASC,EAAK,IAAMC,EAAID,GAAM,CAAC,CAAC,IAAMC,EAAIN,EAAK,CAAC,CAAC,CAAC,EAEjE,SAMf,IAAMO,EAAShB,EAAMnC,EAAQ4C,CAAI,EAK7BO,GAAU,SAAUA,GAAQ,MAAMA,OAEtChB,EAAMnC,EAAwE4C,CAAI,EAItFb,EAAY,GAAOlB,EAAY,GAAOuC,EAAO,CAC/C,EAEMC,GAAQ,IAAM,CAClB,IAAIzC,EAAevB,EAEnB,OAAW,CAACQ,EAAMY,CAAG,IAAKF,EAAM,MAAM,CAAC,EACjCV,IAAS,KACXe,EAAUA,EAAQH,CAAG,EACZZ,IAAS,SAClBe,EAAUA,EAAQH,EAAgB,CAAC,EAAE,CAAC,EAC7BZ,IAAS,cAClBe,EAAUA,EAAQ,CAAC,EAAEH,CAAG,GAI5B,OAAOG,CACT,EAEMsB,EAAWlD,EAAe,CAC9B,WAAAF,EACA,IAAAgD,GACA,QAAAD,EACA,KAAAL,EACA,QAAAI,GACA,MAAArB,EACA,UAAApB,EACA,EAAGD,EAAI,EACP,EAAA+B,CACF,CAAC,EAEKnB,GAAgB,IAAM,CAC1B,GAAI,CAACV,EAAe,CAClBH,IAAkB,OAAS4C,EAAQ,EAAIK,EAAS,GAAG,WAAWjD,CAAa,EAE3E,OAOF,IAAIqE,EAA+B,CAAC,EAEpCpB,EAAS,GAAG,WAAW,SAAS,EAEhC,OAAW,CAAC1C,EAAM,CAAE,SAAA+D,CAAS,CAAC,IAAK,OAAO,QAAQzE,CAAU,EAC1D,QAAW0E,KAAW,OAAO,KAAKD,CAAQ,EACxCD,EAAS,KAAK,QAAQ,QAAQpB,EAAS,UAAU1C,CAAI,EAAE,YAAYgE,CAAO,CAAC,CAAC,EAIhF,QAAWC,KAAM1D,EAAQ,WACvBuD,EAAS,KAAK,IAAI,QAAQ,CAACI,EAAKC,IAAQ,CAItC,IAAMC,EAAM,SAAS,cAAc,KAAK,EAKxCA,EAAI,YAAc,IAClBA,EAAI,IAAMH,EAENG,EAAI,UAAYA,EAAI,gBAAkB,EAIxCF,EAAI,MAAM,GAKVE,EAAI,OAASF,EACbE,EAAI,QAAUD,EAElB,CAAC,CAAC,EAGJ,QAAQ,IAAIL,CAAQ,EAAE,KAAK,IAAM,CAC/BpB,EAAS,GAAG,WAAWjD,CAAa,CACtC,CAAC,CACH,EAEMkD,EAAQ0B,EAAY,CACxB,KAAK,CAACC,CAAI,EAAG,CAIN/B,GAAW,WAAWgC,EAAMD,CAAI,CACvC,EACA,eAAe,CAACE,CAAU,EAAG,CAC3B9B,EAAS,WAAW8B,CAAU,EAC9BD,EAAK,CACP,EACA,UAAU,CAACE,CAAM,EAAG,CAClB/B,EAAS,MAAM+B,EAAQ,OAAO,EAAE,KAAK,EACrCF,EAAK,CACP,EACA,UAAU,CAACE,CAAM,EAAG,CAClB/B,EAAS,MAAM+B,EAAQ,OAAO,EAAE,KAAK,EACrCF,EAAK,CACP,EACA,cAAc,CAACG,EAAWV,EAASW,EAAWC,CAAK,EAAG,CACpD,IAAMC,EAASnC,EAAS,UAAUgC,CAAS,EAE3CG,EAAO,OAAOF,EAAWC,CAAK,EAC9BC,EAAO,YAAYb,CAAO,EAAE,EAE5BO,EAAK,CACP,EACA,cAAc,CAACG,EAAWC,EAAWC,EAAOE,CAAQ,EAAG,CACtCpC,EAAS,UAAUgC,CAAS,EAEpC,OAAOC,EAAWC,EAAOE,CAAQ,EAAEP,CAAI,CAChD,EACA,OAAO,CAACQ,EAAQC,EAAShB,CAAO,EAAG,CACjCtB,EAAS,OAAOuC,EAAOD,CAAO,EAAGD,EAAQf,CAAO,EAAE,IAAM,CACtDkB,EAAS,EACTX,EAAK,CACP,CAAC,CACH,EACA,SAAS,CAACY,CAAE,EAAG,CACb,IAAMxB,EAASwB,EAAG,EAElB,OAAK5C,IAAWoB,EAASA,EAAO,KAAKY,CAAI,EAAIA,EAAK,GAE3CZ,CACT,EACA,OAAOyB,EAAS,CACd,IAAMC,EAAYD,EAAQ,IAAI,CAAC,CAACJ,EAASxE,EAAQ8E,CAAO,IAC/C,CAACL,EAAOD,CAAO,EAAGxE,EAAQ8E,CAAO,CACzC,EAED5C,EAAS,QAAQ2C,CAAS,EAAGE,GAAa,CACxCL,EAAS,EAETnE,EAAM,MAAM,CAAC,EAAE,KAAK,CAAC,SAAUwE,CAAQ,EAAG,CAAC,KAAM,CAAC,CAAC,EAAG3B,EAAO,CAC/D,CAAC,CACH,EACA,KAAK,CAAC4B,CAAK,EAAG,CACZzE,EAAM,MAAM,CAAC,EAAI,CAAC,CAAC,KAAMyE,CAAK,EAAG,CAAC,KAAM,CAAC,CAAC,EAE1C9C,EAAS,MAAM,EAAK,EAAE,IAAM,CACrBH,GAAWqB,EAAO,CACzB,CAAC,CACH,EACA,OAAQ,CACN,GAAI,CACF,UAAU,QAAQ,CAAC,CACrB,QAAE,CACAlB,EAAS,MAAMrB,CAAS,EAAEkD,CAAI,CAChC,CACF,EACA,UAAU,CAACkB,CAAS,EAAG,CACrB,IAAM3E,EAAQ2E,EAAU,EAEnBlD,IAAWxB,EAAM,MAAM,CAAC,EAAE,KAAK,CAAC,YAAaD,CAAK,EAAG,CAAC,KAAM,CAAC,CAAC,EAAG8C,EAAO,EAC/E,EACA,KAAM,CAIJ5B,EAAK,GAAO,MAAM,EAIlBW,EAAM,QAAS,CAAC,CAAC,EAIjBD,EAAS,GAAG,WAAW,UAAU,CACnC,EACA,MAAM,CAACgD,EAAUC,EAASC,CAAK,EAAG,CAChClD,EAAS,MAAMgD,EAAUC,EAASC,CAAK,EAAE,IAAM,CAC7CV,EAAS,EACTX,EAAK,CACP,CAAC,CACH,EACA,OAAO,CAACsB,CAAO,EAAG,CAMhB,OALenD,EAAS,OAAOmD,EAASxE,EAAW,IAAM,CACnD,CAACkB,GAAasD,EAAQ,mBAAmBX,EAAS,EACjD3C,GAAWgC,EAAK,CACvB,CAAC,CAGH,EACA,QAAQuB,EAAS,CACf,GAAI,CACF,UAAU,QAAQA,CAAO,CAC3B,QAAE,CACAvB,EAAK,CACP,CACF,EACA,MAAO,CACLA,EAAK,CACP,EACA,iBAAiB,CAACG,EAAWqB,EAAS,GAAGC,CAAO,EAAG,CACjD,IAAMH,EAA0BI,GAAQ,CAEtC,IAAMC,EADOD,EAAI,2BAA4B,EAAK,EAAE,KAChC,cAAc,iDAAiDvB,KAAa,EAKhG,GAAI,CAACwB,EAAQ,OAEb,IAAMC,EAAaH,EAAQ,OAAOrB,GAAa,CAACuB,EAAO,UAAU,SAASvB,CAAS,CAAC,EAEpFuB,EAAO,UAAU,IAAI,GAAGC,CAAU,EAElC,WAAW,IAAM,CACfD,EAAO,UAAU,OAAO,GAAGC,CAAU,CACvC,EAAGJ,CAAO,CACZ,EAEA,OAAAF,EAAQ,eAAiB,GAElBnD,EAAS,OAAOmD,EAASxE,EAAW,IAAM,CAAE,CAAC,EAAGkD,EAAK,CAC9D,CACF,CAAC,EAEKW,EAAW,IAAM,CACrB,GAAI3C,EAAW,OAEf,IAAMnB,EAAUqB,EAAM1B,EAAM,KAAK,EAEjCK,EAAQ,CAAC,EAAIqB,EAAM1B,EAAM,MAAM,CAAC,CAAC,EAEjCK,EAAQ,CAAC,EAAE,CAAC,EAAI,IAAI,KAAK,EAAE,QAAQ,EACnCA,EAAQ,CAAC,EAAE,CAAC,EAAI,OAEhBL,EAAM,KAAKK,CAAO,CACpB,EAEMgF,GAAO,IAAM,CAIjB,IAAMC,EAAOtF,EAAM,MAAM,CAAC,EAAEA,EAAM,MAAM,CAAC,EAAE,OAAS,CAAC,EAKrD,GAAIgC,EAAOsD,EAAK,CAAC,CAAC,GAAKrD,EAASqD,EAAK,CAAC,CAAC,EAAG,CACxCA,EAAK,CAAC,EAAIA,EAAK,CAAC,EAAI,EACpB,OAMFtF,EAAM,MAAM,CAAC,EAAE,KAAK,CAAC,KAAM,CAAC,CAAC,CAC/B,EAEM6C,EAAS,IAAM,CACnB,IAAM0C,EAAWzC,GAAM,EAEvB,GAAIyC,EAAU,CACZ,GAAM,CAAC9F,EAAQ,GAAGG,CAAK,EAAI2F,EAE3B3D,EAAMnC,EAAQG,CAAK,EAEvB,EAEM4D,EAAO,IAAM,CACZhC,IAAW6D,GAAK,EAAGxC,EAAO,EACjC,EAEMqB,EAAUD,GAA2C,CACzD,IAAMuB,EAAO9E,EAAE,IAAI,EAAE,KAAK,CAAC,EACrBrB,EAAOS,EAAM,EAEnB,OAAOzB,GAAW,OAAO4F,GAAY,WAAaA,EAAQuB,EAAMnG,CAAI,EAAI4E,EAAS5E,CAAI,CACvF,EAEA,MAAO,CACL,UAAAN,EACA,OAAAU,EACA,OAAAoD,EACA,MAAA/C,EACA,EAAGnB,EAAI,CACT,CACF,EEpmBA,IAAM8G,GAAuBC,IACpB,CACL,MAAM,KAAM,CACV,IAAMC,EAAQ,aAAa,QAAQD,EAAQ,GAAG,EAE9C,OAAOC,EAAQ,KAAK,MAAMA,CAAK,EAAI,CAAE,MAAO,CAAC,EAAG,KAAM,CAAC,CAAE,CAC3D,EACA,MAAM,IAAIC,EAAM,CACd,aAAa,QAAQF,EAAQ,IAAK,KAAK,UAAUE,CAAI,CAAC,CACxD,CACF","names":["matchAction","values","action","props","isNumber","val","isNull","isString","isCSSImage","str","startsWith","value","isUserRequiredAction","meta","getDefaultSave","getLanguage","languages","language","throttle","fn","ms","throttled","savedArgs","savedThis","wrapper","store","current","subscribers","subscribe","cb","push","value","fn","deepmerge","klona","SKIPPED_DURING_RESTORE","replaceT9N","novely","characters","storage","createRenderer","initialScreen","t9n","languages","assetsPreload","story","withStory","s","name","items","flat","item","data","type","preloadAssets","preload","action","_","prop","props","isCSSImage","state","value","stack","prev","val","deepmerge","createStack","current","goingBack","getDefaultSave","initialData","getLanguage","$","store","initialDataLoaded","throttledOnStorageDataChange","throttle","stored","initial","save","override","date","isLatest","newGame","restore","set","restoring","latest","klona","renderer","match","index","max","acc","isNull","isNumber","queue","isString","i","meta","SKIPPED_DURING_RESTORE","isUserRequiredAction","indexedQueue","_action","_meta","str","result","render","refer","promises","emotions","emotion","bg","res","rej","img","matchAction","time","push","background","source","character","className","style","handle","duration","person","content","unwrap","enmemory","fn","choices","unwrapped","visible","selected","scene","condition","question","onInput","setup","handler","pattern","timeout","classes","get","target","classNames","next","last","referred","lang","localStorageStorage","options","value","data"]}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@novely/core",
3
3
  "description": "Novely - powerful visual novel engine for creating interactive stories and games with branching narratives and rich multimedia content.",
4
- "version": "0.0.0-alpha.11",
4
+ "version": "0.0.0-alpha.12",
5
5
  "type": "module",
6
6
  "sideEffects": false,
7
7
  "types": "./dist/index.d.ts",