qwerty-cli 0.0.1-alpha.16 → 0.0.1-alpha.18
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/{ConfigEditor-KAL2MLQO.js → ConfigEditor-ZJ52VG3I.js} +2 -2
- package/dist/DictBrowser-ZQ773KQV.js +2 -0
- package/dist/DictBrowser-ZQ773KQV.js.map +1 -0
- package/dist/{HelpScreen-O5RSN3C4.js → HelpScreen-Y3TWEEWZ.js} +2 -2
- package/dist/{PracticeScreen-FOCGEFWL.js → PracticeScreen-PWXXG33U.js} +2 -2
- package/dist/{StatsViewer-XOOMUPGL.js → StatsViewer-XARAMVZW.js} +2 -2
- package/dist/{StatsViewer-XOOMUPGL.js.map → StatsViewer-XARAMVZW.js.map} +1 -1
- package/dist/{WordLookup-MWWYQBCM.js → WordLookup-LKBEPDTB.js} +2 -2
- package/dist/chunk-FQ2MEK7M.js +2 -0
- package/dist/chunk-FQ2MEK7M.js.map +1 -0
- package/dist/chunk-GULN5HRV.js +2 -0
- package/dist/chunk-GULN5HRV.js.map +1 -0
- package/dist/chunk-TCYEMBFW.js +3 -0
- package/dist/chunk-TCYEMBFW.js.map +1 -0
- package/dist/cli.js +1 -1
- package/dist/menu.impl-HBP6CFSW.js +2 -0
- package/dist/{menu.impl-4C7HT3VD.js.map → menu.impl-HBP6CFSW.js.map} +1 -1
- package/dist/practice.impl-AVTRDL7X.js +2 -0
- package/dist/practice.impl-AVTRDL7X.js.map +1 -0
- package/dist/{stats.impl-3I7BB7X3.js → stats.impl-WX3BFWI3.js} +3 -3
- package/dist/{stats.impl-3I7BB7X3.js.map → stats.impl-WX3BFWI3.js.map} +1 -1
- package/package.json +1 -1
- package/dist/DictBrowser-OF274IOX.js +0 -2
- package/dist/DictBrowser-OF274IOX.js.map +0 -1
- package/dist/chunk-4EJYJITR.js +0 -3
- package/dist/chunk-4EJYJITR.js.map +0 -1
- package/dist/chunk-HJIGZU3E.js +0 -2
- package/dist/chunk-HJIGZU3E.js.map +0 -1
- package/dist/chunk-UWTJMVJ6.js +0 -2
- package/dist/chunk-UWTJMVJ6.js.map +0 -1
- package/dist/menu.impl-4C7HT3VD.js +0 -2
- package/dist/practice.impl-OLNLJLIA.js +0 -2
- package/dist/practice.impl-OLNLJLIA.js.map +0 -1
- /package/dist/{ConfigEditor-KAL2MLQO.js.map → ConfigEditor-ZJ52VG3I.js.map} +0 -0
- /package/dist/{HelpScreen-O5RSN3C4.js.map → HelpScreen-Y3TWEEWZ.js.map} +0 -0
- /package/dist/{PracticeScreen-FOCGEFWL.js.map → PracticeScreen-PWXXG33U.js.map} +0 -0
- /package/dist/{WordLookup-MWWYQBCM.js.map → WordLookup-LKBEPDTB.js.map} +0 -0
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{b as P}from"./chunk-CQRKGMPU.js";import{d as E}from"./chunk-7LTZGB7F.js";import{b as $,c as B,e as C}from"./chunk-QG7ZTS2G.js";import{b as F,d as M,f as l}from"./chunk-
|
|
2
|
-
//# sourceMappingURL=ConfigEditor-
|
|
1
|
+
import{b as P}from"./chunk-CQRKGMPU.js";import{d as E}from"./chunk-7LTZGB7F.js";import{b as $,c as B,e as C}from"./chunk-QG7ZTS2G.js";import{b as F,d as M,f as l}from"./chunk-FQ2MEK7M.js";import"./chunk-NA5UNUVL.js";import"./chunk-E6BBQALJ.js";import{useState as k}from"react";import{Box as g,Text as d,useInput as V}from"ink";import{jsx as c,jsxs as S}from"react/jsx-runtime";var f=[{kind:"dictRef",path:"defaultDict",labelKey:"defaultDict"},{kind:"enum",path:"defaultMode",labelKey:"defaultMode",options:["order","dictation","review","random","loop"]},{kind:"enum",path:"accent",labelKey:"accent",options:["us","uk"]},{kind:"enum",path:"language",labelKey:"language",options:["auto","zh","en"]},{kind:"enum",path:"mirror",labelKey:"mirror",options:["jsdelivr","github"]},{kind:"enum",path:"stealth",labelKey:"stealth",options:["off","menu","default"]},{kind:"int",path:"chapterSize",labelKey:"chapterSize",min:1,max:200},{kind:"bool",path:"autoplayPronunciation",labelKey:"autoplayPronunciation"},{kind:"bool",path:"sounds.master",labelKey:"soundsMaster"},{kind:"bool",path:"sounds.keystroke",labelKey:"soundsKeystroke"},{kind:"bool",path:"sounds.feedback",labelKey:"soundsFeedback"},{kind:"enum",path:"sounds.pronunciationRate",labelKey:"soundsPronunciationRate",options:["0.5","0.75","1","1.25","1.5"]},{kind:"enum",path:"sounds.pronunciationSource",labelKey:"soundsPronunciationSource",options:["youdao","dictapi"]}];function N(a,e){return e.split(".").reduce((o,s)=>{if(o&&typeof o=="object")return o[s]},a)}function U(){let a=F(),{cfg:e,setCfg:o}=P(),s=M(),x=$(e.defaultDict),[y,K]=k(0),[m,h]=k(!1),[w,p]=k(""),[v,u]=k(null),r=f[y],T=N(e,r.path),b=async i=>{try{let n=E(e,r.path,i);await o(n),h(!1),u(null)}catch(n){u(n.message)}};V((i,n)=>{if(m&&r.kind==="string"){if(n.escape){h(!1),u(null);return}if(n.return){b(w);return}if(n.backspace||n.delete){p(t=>t.slice(0,-1));return}i&&!n.ctrl&&!n.meta&&p(t=>t+i);return}if(m&&r.kind==="int"){if(n.escape){h(!1),u(null);return}if(n.return){b(w);return}if(n.backspace||n.delete){p(t=>t.slice(0,-1));return}/^[0-9]$/.test(i)&&p(t=>t+i);return}if(n.escape){a.back();return}if(n.upArrow){K(t=>(t-1+f.length)%f.length),u(null);return}if(n.downArrow){K(t=>(t+1)%f.length),u(null);return}if(r.kind==="bool"&&(i===" "||n.return)){b(T?"false":"true");return}if(r.kind==="enum"&&(n.leftArrow||n.rightArrow)){let t=r.options.indexOf(String(T)),D=n.rightArrow?1:-1,R=r.options[(t+D+r.options.length)%r.options.length];b(R);return}if(r.kind==="dictRef"&&n.return){a.navigate({name:"dict",params:{pickerMode:"set-default"}});return}(r.kind==="string"||r.kind==="int")&&n.return&&(p(String(T??"")),h(!0),u(null))});let z=Math.max(...f.map(i=>B(s.config.fields[i.labelKey])))+4;return S(g,{flexDirection:"column",paddingX:2,paddingY:1,width:"100%",height:"100%",children:[c(d,{bold:!0,color:l.accent,children:s.config.title}),c(g,{marginTop:1,flexDirection:"column",flexGrow:1,children:f.map((i,n)=>{let t=n===y,D=N(e,i.path),R=j(i,D,t&&m?w:null,s,i.path==="defaultDict"?x:""),A=s.config.fields[i.labelKey],I=" ".repeat(Math.max(0,z-B(A)));return S(g,{children:[c(d,{color:t?l.accent:l.muted,children:t?"\u258C ":" "}),S(d,{bold:t,color:t?l.text:l.muted,children:[A,I]}),c(d,{color:t?l.accent:l.muted,children:R})]},i.path)})}),v&&c(g,{marginTop:1,children:S(d,{color:l.error,children:["! ",v]})}),c(g,{marginTop:1,children:c(d,{color:l.muted,children:L(r,m,s)})})]})}function j(a,e,o,s,x){return o!==null?`${o}_`:a.kind==="bool"?e?`\u2713 ${s.common.on}`:`\u2717 ${s.common.off}`:a.kind==="dictRef"?e?C(x||String(e),24):"\u2014":a.kind==="enum"?`< ${s.config.enumValues[a.labelKey]?.[String(e)]??String(e)} >`:String(e??"")}function L(a,e,o){return e?o.config.hints.editing:a.kind==="bool"?o.config.hints.bool:a.kind==="enum"?o.config.hints.enum:a.kind==="dictRef"?o.config.hints.dictRef:o.config.hints.stringOrInt}export{U as ConfigEditor};
|
|
2
|
+
//# sourceMappingURL=ConfigEditor-ZJ52VG3I.js.map
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
import{a as _,d as H,f as N}from"./chunk-DURXS5MX.js";import{b as z}from"./chunk-CQRKGMPU.js";import"./chunk-7LTZGB7F.js";import{b as K,e as Q}from"./chunk-GULN5HRV.js";import{b as q,d as J,f as e}from"./chunk-FQ2MEK7M.js";import{b as X,c as Y}from"./chunk-NA5UNUVL.js";import"./chunk-E6BBQALJ.js";import{useEffect as st,useMemo as mt,useState as B}from"react";import{Box as l,Text as c,useInput as ut,useStdout as ft}from"ink";import{useState as at}from"react";import{Box as M,Text as I,useInput as dt}from"ink";import{jsx as A,jsxs as U}from"react/jsx-runtime";function G({title:D,items:f,onClose:T}){let g=f.map((s,u)=>s.disabled?-1:u).filter(s=>s>=0),o=g[0]??0,[P,v]=at(o);dt((s,u)=>{if(u.escape){T();return}if(u.upArrow){let a=g.indexOf(P),d=g[(a-1+g.length)%g.length];d!==void 0&&v(d);return}if(u.downArrow){let a=g.indexOf(P),d=g[(a+1)%g.length];d!==void 0&&v(d);return}if(u.return){let a=f[P];a&&!a.disabled&&a.run();return}for(let a=0;a<f.length;a++){let d=f[a];if(!d.disabled&&d.key&&s===d.key){d.run();return}}});let S=Math.max(...f.map(s=>s.label.length)),L=Math.max(S+8,D.length+4,24);return U(M,{flexDirection:"column",borderStyle:"round",borderColor:e.accent,paddingX:2,paddingY:1,width:L,children:[A(M,{marginBottom:1,children:A(I,{bold:!0,color:e.accent,children:D})}),f.map((s,u)=>{let a=u===P,d=s.disabled?e.muted:a?e.text:e.muted;return U(M,{children:[A(I,{color:a?e.accent:e.muted,children:a?"\u258C ":" "}),A(I,{bold:a,color:d,children:s.label})]},u)}),A(M,{marginTop:1,children:A(I,{color:e.muted,children:"\u2191/\u2193 \xB7 Enter \xB7 Esc"})})]})}import{Fragment as gt,jsx as t,jsxs as h}from"react/jsx-runtime";function Lt({params:D}){let f=q(),{cfg:T,setCfg:g}=z(),o=J(),{stdout:P}=ft(),[v,S]=B([]),[L,s]=B(!0),[u,a]=B(0),[d,O]=B(""),[x,y]=B(null),[V,E]=B(0),[C,p]=B(null),W=async()=>{let n=await Y(),r=await Promise.all(n.map(async m=>({entry:m,local:await H(m.id)})));S(r),s(!1)};st(()=>{W()},[V]);let b=mt(()=>d?v.filter(n=>X([n.entry],d).length>0):v,[d,v]),R=Math.max(0,Math.min(b.length-1,u)),i=b[R],Z=P?.rows??24,$=Math.max(6,Z-8),tt=Math.floor($/2),j=Math.max(0,Math.min(b.length-$,R-tt)),et=Math.min(b.length,j+$),F=async n=>{let r=Q(await K(),n);f.replace({name:"practice",params:{dictId:n,chapterIndex:r,mode:T.defaultMode,stealth:T.stealth==="default"}})},nt=async(n,r=!0)=>{await g({...T,defaultDict:n}),p(null),r&&(D?.pickerMode==="choose-then-practice"?await F(n):f.back())},ot=n=>{p(null),y({kind:"removing",id:n}),(async()=>{try{await N(n),y(null),E(r=>r+1)}catch(r){y({kind:"error",id:n,msg:r.message})}})()},rt=n=>{p(null),y({kind:"pulling",id:n}),(async()=>{try{await _(n),y(null),E(r=>r+1)}catch(r){y({kind:"error",id:n,msg:r.message})}})()},it=()=>{p(null),y({kind:"refreshing"}),E(n=>n+1),y(null)};if(ut((n,r)=>{if(C===null){if(r.escape){f.back();return}if(r.upArrow){a(m=>Math.max(0,m-1));return}if(r.downArrow){a(m=>Math.min(b.length-1,m+1));return}if(r.ctrl&&n==="k"){p("more");return}if(r.return){i&&p("item");return}if(r.backspace||r.delete){O(m=>m.slice(0,-1)),a(0);return}if(n&&!r.ctrl&&!r.meta){let m=[...n].filter(w=>{let k=w.codePointAt(0);return k>=32&&k!==127}).join("");if(m.length===0)return;O(w=>w+m),a(0)}}}),L)return t(l,{alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:t(c,{color:e.muted,children:o.dict.loading})});let lt=i?[{label:o.dict.action.setDefault,run:()=>{nt(i.entry.id,D?.pickerMode!==void 0)}},{label:o.dict.action.practice,run:()=>{F(i.entry.id)}},{label:o.dict.action.delete,disabled:!i.local,run:()=>ot(i.entry.id)},{label:o.common.cancel,run:()=>p(null)}]:[],ct=[{label:o.dict.command.pull,disabled:!i,run:()=>i&&rt(i.entry.id)},{label:o.dict.command.import,disabled:!0,run:()=>{}},{label:o.dict.command.refreshList,run:()=>it()},{label:o.common.cancel,run:()=>p(null)}];return C==="item"&&i?t(l,{alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:t(G,{title:`${o.dict.action.title} \xB7 ${i.entry.name}`,items:lt,onClose:()=>p(null)})}):C==="more"?t(l,{alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:t(G,{title:o.dict.command.title,items:ct,onClose:()=>p(null)})}):h(l,{flexDirection:"column",paddingX:2,paddingY:1,width:"100%",height:"100%",children:[h(l,{children:[t(c,{bold:!0,color:e.accent,children:o.dict.title}),t(l,{flexGrow:1}),t(c,{color:e.muted,children:d?`${o.dict.filterPlaceholder}: ${d}_`:`${o.dict.filterPlaceholder}_`}),h(c,{color:e.muted,children:[" ",o.dict.entries(b.length)]})]}),h(l,{marginTop:1,flexGrow:1,children:[t(l,{flexDirection:"column",width:"75%",paddingRight:1,children:b.slice(j,et).map((n,r)=>{let w=j+r===R,k=T.defaultDict===n.entry.id;return h(l,{children:[t(l,{width:2,children:t(c,{color:w?e.accent:e.muted,children:w?"\u258C ":" "})}),t(l,{width:2,children:t(c,{color:n.local?e.accent:e.muted,children:n.local?"\u25CF":"\u25CB"})}),t(l,{width:2,children:t(c,{color:k?e.success:e.muted,children:k?"\u2605":" "})}),t(l,{flexGrow:1,children:t(c,{bold:w,color:w?e.text:e.muted,wrap:"truncate",children:n.entry.name})}),t(l,{width:6,children:t(c,{color:e.muted,children:String(n.entry.length).padStart(5)})})]},n.entry.id)})}),t(l,{flexDirection:"column",width:"25%",paddingLeft:1,children:i&&h(gt,{children:[t(c,{bold:!0,color:e.text,wrap:"wrap",children:i.entry.name}),t(c,{color:e.muted,children:i.entry.id}),t(l,{marginTop:1,children:h(c,{color:e.muted,wrap:"wrap",children:[i.entry.language," \xB7 ",i.entry.category]})}),t(l,{marginTop:1,children:t(c,{color:e.muted,children:o.dict.wordsLabel(i.entry.length)})}),i.entry.description&&t(l,{marginTop:1,children:t(c,{color:e.primary,wrap:"wrap",children:i.entry.description})}),i.entry.tags.length>0&&t(l,{marginTop:1,children:t(c,{color:e.muted,wrap:"wrap",children:o.dict.tagsLabel(i.entry.tags.join(", "))})}),t(l,{marginTop:1,children:t(c,{color:i.local?e.accent:e.muted,children:i.local?o.dict.local:o.dict.notLocal})}),T.defaultDict===i.entry.id&&t(l,{children:t(c,{color:e.success,children:o.dict.defaultMark})})]})})]}),x&&h(l,{marginTop:1,children:[x.kind==="pulling"&&t(c,{color:e.warning,children:o.dict.pulling(x.id)}),x.kind==="removing"&&t(c,{color:e.warning,children:o.dict.removing(x.id)}),x.kind==="refreshing"&&h(c,{color:e.warning,children:[o.dict.command.refreshList,"\u2026"]}),x.kind==="error"&&t(c,{color:e.error,children:o.dict.errorOn(x.id,x.msg)})]}),t(l,{marginTop:1,children:t(c,{color:e.muted,children:o.dict.footer})})]})}export{Lt as DictBrowser};
|
|
2
|
+
//# sourceMappingURL=DictBrowser-ZQ773KQV.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/ui/screens/DictBrowser.tsx","../src/ui/components/ActionPanel.tsx"],"sourcesContent":["import { useEffect, useMemo, useState } from 'react';\nimport { Box, Text, useInput, useStdout } from 'ink';\nimport { useNav, type DictParams } from '../nav.js';\nimport { useAppState } from '../app-state.js';\nimport { useStrings } from '../../i18n/context.js';\nimport { PALETTE } from '../components/BigWord.js';\nimport { ActionPanel, type ActionItem } from '../components/ActionPanel.js';\nimport { loadRegistry } from '../../infra/registry-store.js';\nimport {\n isLocallyAvailable,\n pullDictionary,\n removeDictionary,\n} from '../../infra/dict-downloader.js';\nimport type { DictionaryEntry } from '../../domain/dictionary.js';\nimport { filterRegistry } from '../../domain/dictionary.js';\nimport { loadSessions, resumeChapterFor } from '../../domain/stats.js';\n\ntype Row = { entry: DictionaryEntry; local: boolean };\ntype PendingAction =\n | { kind: 'pulling'; id: string }\n | { kind: 'removing'; id: string }\n | { kind: 'refreshing' }\n | { kind: 'error'; id: string; msg: string }\n | null;\n\ntype Panel = null | 'item' | 'more';\n\nexport function DictBrowser({ params }: { params?: DictParams }) {\n const nav = useNav();\n const { cfg, setCfg } = useAppState();\n const t = useStrings();\n const { stdout } = useStdout();\n const [rows, setRows] = useState<Row[]>([]);\n const [loading, setLoading] = useState(true);\n const [selected, setSelected] = useState(0);\n const [filter, setFilter] = useState('');\n const [pending, setPending] = useState<PendingAction>(null);\n const [tick, setTick] = useState(0);\n const [panel, setPanel] = useState<Panel>(null);\n\n const refresh = async () => {\n const reg = await loadRegistry();\n const flagged = await Promise.all(\n reg.map(async (e) => ({ entry: e, local: await isLocallyAvailable(e.id) })),\n );\n setRows(flagged);\n setLoading(false);\n };\n\n useEffect(() => {\n void refresh();\n }, [tick]);\n\n const filtered = useMemo(\n () => (filter ? rows.filter((r) => filterRegistry([r.entry], filter).length > 0) : rows),\n [filter, rows],\n );\n const safeSelected = Math.max(0, Math.min(filtered.length - 1, selected));\n const current = filtered[safeSelected];\n\n const rowsTotal = stdout?.rows ?? 24;\n const visibleH = Math.max(6, rowsTotal - 8);\n const half = Math.floor(visibleH / 2);\n const start = Math.max(0, Math.min(filtered.length - visibleH, safeSelected - half));\n const end = Math.min(filtered.length, start + visibleH);\n\n const goPractice = async (id: string) => {\n const chapterIndex = resumeChapterFor(await loadSessions(), id);\n nav.replace({\n name: 'practice',\n params: {\n dictId: id,\n chapterIndex,\n mode: cfg.defaultMode,\n stealth: cfg.stealth === 'default',\n },\n });\n };\n\n const doSetDefault = async (id: string, navigate = true) => {\n await setCfg({ ...cfg, defaultDict: id });\n setPanel(null);\n if (navigate) {\n if (params?.pickerMode === 'choose-then-practice') {\n await goPractice(id);\n } else {\n nav.back();\n }\n }\n };\n\n const doDelete = (id: string) => {\n setPanel(null);\n setPending({ kind: 'removing', id });\n void (async () => {\n try {\n await removeDictionary(id);\n setPending(null);\n setTick((n) => n + 1);\n } catch (err) {\n setPending({ kind: 'error', id, msg: (err as Error).message });\n }\n })();\n };\n\n const doPull = (id: string) => {\n setPanel(null);\n setPending({ kind: 'pulling', id });\n void (async () => {\n try {\n await pullDictionary(id);\n setPending(null);\n setTick((n) => n + 1);\n } catch (err) {\n setPending({ kind: 'error', id, msg: (err as Error).message });\n }\n })();\n };\n\n const doRefreshList = () => {\n setPanel(null);\n setPending({ kind: 'refreshing' });\n setTick((n) => n + 1);\n setPending(null);\n };\n\n useInput((input, key) => {\n if (panel !== null) return;\n\n if (key.escape) {\n nav.back();\n return;\n }\n if (key.upArrow) {\n setSelected((i) => Math.max(0, i - 1));\n return;\n }\n if (key.downArrow) {\n setSelected((i) => Math.min(filtered.length - 1, i + 1));\n return;\n }\n if (key.ctrl && input === 'k') {\n setPanel('more');\n return;\n }\n if (key.return) {\n if (current) setPanel('item');\n return;\n }\n if (key.backspace || key.delete) {\n setFilter((f) => f.slice(0, -1));\n setSelected(0);\n return;\n }\n if (input && !key.ctrl && !key.meta) {\n // Keep printable codepoints (>= 0x20, != 0x7F DEL). Removing the old\n // `input.length === 1` cap so IME multi-char commits like '考研' (length 2)\n // and pasted runs reach the filter. filterRegistry already matches the\n // dictionary `name` field, which is Chinese for ~75% of the registry —\n // the bug was that the keystroke never made it to filter state.\n const cleaned = [...input]\n .filter((c) => {\n const cp = c.codePointAt(0)!;\n return cp >= 0x20 && cp !== 0x7f;\n })\n .join('');\n if (cleaned.length === 0) return;\n setFilter((f) => f + cleaned);\n setSelected(0);\n }\n });\n\n if (loading) {\n return (\n <Box alignItems=\"center\" justifyContent=\"center\" width=\"100%\" height=\"100%\">\n <Text color={PALETTE.muted}>{t.dict.loading}</Text>\n </Box>\n );\n }\n\n const itemPanelItems: ActionItem[] = current\n ? [\n {\n label: t.dict.action.setDefault,\n run: () => void doSetDefault(current.entry.id, params?.pickerMode !== undefined),\n },\n {\n label: t.dict.action.practice,\n run: () => void goPractice(current.entry.id),\n },\n {\n label: t.dict.action.delete,\n disabled: !current.local,\n run: () => doDelete(current.entry.id),\n },\n { label: t.common.cancel, run: () => setPanel(null) },\n ]\n : [];\n\n const morePanelItems: ActionItem[] = [\n {\n label: t.dict.command.pull,\n disabled: !current,\n run: () => current && doPull(current.entry.id),\n },\n {\n label: t.dict.command.import,\n disabled: true,\n run: () => undefined,\n },\n { label: t.dict.command.refreshList, run: () => doRefreshList() },\n { label: t.common.cancel, run: () => setPanel(null) },\n ];\n\n if (panel === 'item' && current) {\n return (\n <Box alignItems=\"center\" justifyContent=\"center\" width=\"100%\" height=\"100%\">\n <ActionPanel\n title={`${t.dict.action.title} · ${current.entry.name}`}\n items={itemPanelItems}\n onClose={() => setPanel(null)}\n />\n </Box>\n );\n }\n if (panel === 'more') {\n return (\n <Box alignItems=\"center\" justifyContent=\"center\" width=\"100%\" height=\"100%\">\n <ActionPanel\n title={t.dict.command.title}\n items={morePanelItems}\n onClose={() => setPanel(null)}\n />\n </Box>\n );\n }\n\n return (\n <Box flexDirection=\"column\" paddingX={2} paddingY={1} width=\"100%\" height=\"100%\">\n <Box>\n <Text bold color={PALETTE.accent}>\n {t.dict.title}\n </Text>\n <Box flexGrow={1} />\n <Text color={PALETTE.muted}>\n {filter ? `${t.dict.filterPlaceholder}: ${filter}_` : `${t.dict.filterPlaceholder}_`}\n </Text>\n <Text color={PALETTE.muted}>\n {' '}\n {t.dict.entries(filtered.length)}\n </Text>\n </Box>\n\n <Box marginTop={1} flexGrow={1}>\n <Box flexDirection=\"column\" width=\"75%\" paddingRight={1}>\n {filtered.slice(start, end).map((row, vi) => {\n const i = start + vi;\n const active = i === safeSelected;\n const isDefault = cfg.defaultDict === row.entry.id;\n return (\n <Box key={row.entry.id}>\n <Box width={2}>\n <Text color={active ? PALETTE.accent : PALETTE.muted}>{active ? '▌ ' : ' '}</Text>\n </Box>\n <Box width={2}>\n <Text color={row.local ? PALETTE.accent : PALETTE.muted}>\n {row.local ? '●' : '○'}\n </Text>\n </Box>\n <Box width={2}>\n <Text color={isDefault ? PALETTE.success : PALETTE.muted}>\n {isDefault ? '★' : ' '}\n </Text>\n </Box>\n <Box flexGrow={1}>\n <Text bold={active} color={active ? PALETTE.text : PALETTE.muted} wrap=\"truncate\">\n {row.entry.name}\n </Text>\n </Box>\n <Box width={6}>\n <Text color={PALETTE.muted}>{String(row.entry.length).padStart(5)}</Text>\n </Box>\n </Box>\n );\n })}\n </Box>\n\n <Box flexDirection=\"column\" width=\"25%\" paddingLeft={1}>\n {current && (\n <>\n <Text bold color={PALETTE.text} wrap=\"wrap\">\n {current.entry.name}\n </Text>\n <Text color={PALETTE.muted}>{current.entry.id}</Text>\n <Box marginTop={1}>\n <Text color={PALETTE.muted} wrap=\"wrap\">\n {current.entry.language} · {current.entry.category}\n </Text>\n </Box>\n <Box marginTop={1}>\n <Text color={PALETTE.muted}>{t.dict.wordsLabel(current.entry.length)}</Text>\n </Box>\n {current.entry.description && (\n <Box marginTop={1}>\n <Text color={PALETTE.primary} wrap=\"wrap\">{current.entry.description}</Text>\n </Box>\n )}\n {current.entry.tags.length > 0 && (\n <Box marginTop={1}>\n <Text color={PALETTE.muted} wrap=\"wrap\">{t.dict.tagsLabel(current.entry.tags.join(', '))}</Text>\n </Box>\n )}\n <Box marginTop={1}>\n <Text color={current.local ? PALETTE.accent : PALETTE.muted}>\n {current.local ? t.dict.local : t.dict.notLocal}\n </Text>\n </Box>\n {cfg.defaultDict === current.entry.id && (\n <Box>\n <Text color={PALETTE.success}>{t.dict.defaultMark}</Text>\n </Box>\n )}\n </>\n )}\n </Box>\n </Box>\n\n {pending && (\n <Box marginTop={1}>\n {pending.kind === 'pulling' && (\n <Text color={PALETTE.warning}>{t.dict.pulling(pending.id)}</Text>\n )}\n {pending.kind === 'removing' && (\n <Text color={PALETTE.warning}>{t.dict.removing(pending.id)}</Text>\n )}\n {pending.kind === 'refreshing' && (\n <Text color={PALETTE.warning}>{t.dict.command.refreshList}…</Text>\n )}\n {pending.kind === 'error' && (\n <Text color={PALETTE.error}>{t.dict.errorOn(pending.id, pending.msg)}</Text>\n )}\n </Box>\n )}\n\n <Box marginTop={1}>\n <Text color={PALETTE.muted}>{t.dict.footer}</Text>\n </Box>\n </Box>\n );\n}\n","import { useState } from 'react';\nimport { Box, Text, useInput } from 'ink';\nimport { PALETTE } from './BigWord.js';\n\nexport type ActionItem = {\n key?: string;\n label: string;\n disabled?: boolean;\n run: () => void | Promise<void>;\n};\n\ntype Props = {\n title: string;\n items: ActionItem[];\n onClose: () => void;\n};\n\nexport function ActionPanel({ title, items, onClose }: Props) {\n const enabledIndices = items\n .map((it, i) => (it.disabled ? -1 : i))\n .filter((i) => i >= 0);\n const initial = enabledIndices[0] ?? 0;\n const [selected, setSelected] = useState(initial);\n\n useInput((input, key) => {\n if (key.escape) {\n onClose();\n return;\n }\n if (key.upArrow) {\n const cur = enabledIndices.indexOf(selected);\n const next = enabledIndices[(cur - 1 + enabledIndices.length) % enabledIndices.length];\n if (next !== undefined) setSelected(next);\n return;\n }\n if (key.downArrow) {\n const cur = enabledIndices.indexOf(selected);\n const next = enabledIndices[(cur + 1) % enabledIndices.length];\n if (next !== undefined) setSelected(next);\n return;\n }\n if (key.return) {\n const item = items[selected];\n if (item && !item.disabled) {\n void item.run();\n }\n return;\n }\n for (let i = 0; i < items.length; i++) {\n const it = items[i]!;\n if (it.disabled) continue;\n if (it.key && input === it.key) {\n void it.run();\n return;\n }\n }\n });\n\n const maxLabel = Math.max(...items.map((it) => it.label.length));\n const width = Math.max(maxLabel + 8, title.length + 4, 24);\n\n return (\n <Box\n flexDirection=\"column\"\n borderStyle=\"round\"\n borderColor={PALETTE.accent}\n paddingX={2}\n paddingY={1}\n width={width}\n >\n <Box marginBottom={1}>\n <Text bold color={PALETTE.accent}>\n {title}\n </Text>\n </Box>\n {items.map((it, i) => {\n const active = i === selected;\n const color = it.disabled\n ? PALETTE.muted\n : active\n ? PALETTE.text\n : PALETTE.muted;\n return (\n <Box key={i}>\n <Text color={active ? PALETTE.accent : PALETTE.muted}>\n {active ? '▌ ' : ' '}\n </Text>\n <Text bold={active} color={color}>\n {it.label}\n </Text>\n </Box>\n );\n })}\n <Box marginTop={1}>\n <Text color={PALETTE.muted}>↑/↓ · Enter · Esc</Text>\n </Box>\n </Box>\n );\n}\n"],"mappings":"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","names":["useEffect","useMemo","useState","Box","Text","useInput","useStdout","useState","Box","Text","useInput","jsx","jsxs","ActionPanel","title","items","onClose","enabledIndices","it","i","initial","selected","setSelected","useState","useInput","input","key","cur","next","item","maxLabel","width","Box","PALETTE","Text","active","color","Fragment","jsx","jsxs","DictBrowser","params","nav","useNav","cfg","setCfg","useAppState","t","useStrings","stdout","useStdout","rows","setRows","useState","loading","setLoading","selected","setSelected","filter","setFilter","pending","setPending","tick","setTick","panel","setPanel","refresh","reg","loadRegistry","flagged","e","isLocallyAvailable","useEffect","filtered","useMemo","r","filterRegistry","safeSelected","current","rowsTotal","visibleH","half","start","end","goPractice","id","chapterIndex","resumeChapterFor","loadSessions","doSetDefault","navigate","doDelete","removeDictionary","n","err","doPull","pullDictionary","doRefreshList","useInput","input","key","i","f","cleaned","c","cp","Box","Text","PALETTE","itemPanelItems","morePanelItems","ActionPanel","row","vi","active","isDefault"]}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{b as p,d as a,f as i}from"./chunk-
|
|
2
|
-
//# sourceMappingURL=HelpScreen-
|
|
1
|
+
import{b as p,d as a,f as i}from"./chunk-FQ2MEK7M.js";import{Box as l,Text as n,useInput as x}from"ink";import{jsx as o,jsxs as c}from"react/jsx-runtime";function f(){let m=p(),t=a();x((s,r)=>{r.escape&&m.back()});let e=t.help.keys,u=[{title:t.help.sections.global,keys:[e.helpScreen,e.quit]},{title:t.help.sections.main,keys:[e.navigate,e.select,e.letterJump,e.helpScreen]},{title:t.help.sections.practice,keys:[e.pause,e.skip,e.replay,e.resume,e.nextChapter,e.reviewMistakes,e.stealthToggle,e.backMenu]},{title:t.help.sections.dict,keys:[e.navigate,e.filter,e.itemActions,e.moreActions,e.backScreen]},{title:t.help.sections.config,keys:[e.navigate,e.select,e.backMenu]},{title:t.help.sections.stats,keys:[e.cycleWindow,e.backMenu]},{title:t.help.sections.word,keys:[e.filter,e.navigate,e.backMenu]}];return c(l,{flexDirection:"column",paddingX:2,paddingY:1,width:"100%",height:"100%",children:[c(l,{children:[o(n,{bold:!0,color:i.accent,children:t.help.title}),o(n,{color:i.muted,children:" \xB7 "}),o(n,{color:i.muted,children:t.help.subtitle})]}),o(l,{marginTop:1,flexDirection:"column",flexGrow:1,children:u.map(s=>c(l,{flexDirection:"column",marginTop:1,children:[o(n,{bold:!0,color:i.text,children:s.title}),s.keys.map((r,k)=>o(l,{children:c(n,{color:i.muted,children:[" \xB7 ",r]})},k))]},s.title))}),o(l,{marginTop:1,children:o(n,{color:i.muted,children:t.help.footer})})]})}export{f as HelpScreen};
|
|
2
|
+
//# sourceMappingURL=HelpScreen-Y3TWEEWZ.js.map
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{b as ht,e as bt,h as Tt}from"./chunk-6ROGUGNX.js";import{a as ut,b as lt,c as dt,d as mt,e as ft,f as pt}from"./chunk-BIBS2Q3E.js";import{e as st}from"./chunk-DURXS5MX.js";import{b as Y}from"./chunk-CQRKGMPU.js";import"./chunk-7LTZGB7F.js";import{a as St}from"./chunk-HJIGZU3E.js";import{a as V,b as yt,c as wt}from"./chunk-G3DQB7FI.js";import{b as gt,e as L}from"./chunk-QG7ZTS2G.js";import{b as X,d as W,f as i,g as xt}from"./chunk-UWTJMVJ6.js";import"./chunk-NA5UNUVL.js";import"./chunk-E6BBQALJ.js";import{useState as j,useEffect as H,useRef as tt}from"react";import{Box as f,Text as g,useApp as de,useInput as O}from"ink";function It(t,e=Math.random){let r=[...t];for(let n=r.length-1;n>0;n--){let c=Math.floor(e()*(n+1)),d=r[n];r[n]=r[c],r[c]=d}return r}function kt(t){let e=t>>>0;return()=>{e=e+1831565813>>>0;let r=Math.imul(e^e>>>15,1|e);return r=r+Math.imul(r^r>>>7,61|r)^r,((r^r>>>14)>>>0)/4294967296}}function Ct(t,e){if(e<=0)throw new Error("chapterSize must be positive");let r=[];for(let n=0;n<t.length;n+=e)r.push(t.slice(n,n+e));return r}function Mt(t,e,r){if(e==="random"){let n=r===void 0?Math.random:kt(r);return It(t,n)}return t}function _(t){return{target:t,typed:"",errorsThisWord:0}}function Wt(t,e){switch(e.type){case"reset":return{state:{...t,typed:""},effect:"none"};case"backspace":return t.typed.length===0?{state:t,effect:"none"}:{state:{...t,typed:t.typed.slice(0,-1)},effect:"none"};case"char":{let r=t.typed+e.ch,n=[...t.target].slice(0,[...r].length).join("");return r===n?r.length===t.target.length?{state:{...t,typed:r},effect:"correct"}:{state:{...t,typed:r},effect:"progress"}:{state:{...t,typed:"",errorsThisWord:t.errorsThisWord+1},effect:"wrong"}}}}function J(t,e=Date.now()){return t.length===0?{startedAt:e,results:[],current:null,finishedAt:e,playlist:t}:{startedAt:e,results:[],current:{wordIndex:0,wordStartedAt:e,input:_(t[0].name)},finishedAt:null,playlist:t}}function Q(t,e,r=Date.now()){if(!t.current)return{session:t,effect:"none"};let{state:n,effect:c}=Wt(t.current.input,e);if(c==="correct"){let d={word:n.target,errors:n.errorsThisWord,durationMs:r-t.current.wordStartedAt},a=t.current.wordIndex+1,s=[...t.results,d];return a>=t.playlist.length?{session:{...t,results:s,current:null,finishedAt:r},effect:c}:{session:{...t,results:s,current:{wordIndex:a,wordStartedAt:r,input:_(t.playlist[a].name)}},effect:c}}return{session:{...t,current:{...t.current,input:n}},effect:c}}function vt(t,e=Date.now()){if(!t.current)return{session:t,effect:"none"};let r={word:t.current.input.target,errors:0,durationMs:e-t.current.wordStartedAt,skipped:!0},n=t.current.wordIndex+1,c=[...t.results,r];return n>=t.playlist.length?{session:{...t,results:c,current:null,finishedAt:e},effect:"skipped"}:{session:{...t,results:c,current:{wordIndex:n,wordStartedAt:e,input:_(t.playlist[n].name)}},effect:"skipped"}}function Z(t){let e=t.results.reduce((c,d)=>c+d.errors,0),r=(t.finishedAt??Date.now())-t.startedAt,n={};for(let c of t.results)c.errors>0&&(n[c.word]=(n[c.word]??0)+c.errors);return{wordCount:t.results.length,errors:e,durationMs:r,perWordErrors:n}}import{useEffect as Et,useReducer as Qt,useRef as Zt,useState as te}from"react";import{useInput as ee,useApp as re}from"ink";function ne(t,e){if(e.type==="start")return{session:J(e.playlist,e.now),lastEffect:null,effectSeq:0};if(e.type==="skip"){let r=vt(t.session,e.now);return{session:r.session,lastEffect:r.effect,effectSeq:t.effectSeq+1}}if(e.type==="event"){if(e.key.backspace||e.key.delete){let c=Q(t.session,{type:"backspace"},e.now);return{session:c.session,lastEffect:c.effect,effectSeq:t.effectSeq+1}}if(e.input.length===0)return t;let r=t.session,n=t.lastEffect;for(let c of e.input){let d=Q(r,{type:"char",ch:c},e.now);if(r=d.session,n=d.effect,r.finishedAt!==null)break}return{session:r,lastEffect:n,effectSeq:t.effectSeq+1}}return t}function oe(t){let e=[...t];if(e.some(n=>n.codePointAt(0)>=128))return{kind:"ime",cleaned:""};let r=e.filter(n=>{let c=n.codePointAt(0);return c>=32&&c<=126}).join("");return{kind:r.length>0?"valid":"noise",cleaned:r}}function Bt({playlist:t,onComplete:e,onTab:r,onEscape:n,onSkip:c,onImeBlock:d,onValidInput:a,enabled:s=!0}){let[l,y]=Qt(ne,void 0,()=>({session:J(t,Date.now()),lastEffect:null,effectSeq:0})),M=Zt(!1),[T,I]=te(0),{exit:w}=re();return ee((p,x)=>{if(x.ctrl&&p==="c"){w();return}if(x.ctrl&&p==="n"){c?.(),y({type:"skip",now:Date.now()});return}if(x.escape){n?.();return}if(x.tab){r?.();return}if(x.upArrow||x.downArrow||x.leftArrow||x.rightArrow||x.return||x.ctrl||x.meta)return;let{kind:v,cleaned:N}=oe(p);if(v==="ime"){d?.();return}v!=="noise"&&(a?.(),y({type:"event",input:N,key:x,now:Date.now()}))},{isActive:s}),Et(()=>{l.session.finishedAt!==null&&!M.current&&(M.current=!0,e(l.session))},[l.session,e]),Et(()=>{if(l.session.finishedAt!==null)return;let p=setInterval(()=>I(x=>x+1),1e3);return()=>clearInterval(p)},[l.session.finishedAt]),{session:l.session,lastEffect:l.lastEffect,effectSeq:l.effectSeq,tick:T}}import{useEffect as ce,useRef as ie}from"react";function At(t){let e=ie(!1);return ce(()=>{e.current||(e.current=!0,ut(!t.enabled).catch(()=>{}))},[t.enabled]),{keystroke:()=>t.enabled&<(),correct:()=>t.enabled&&dt(),wrong:()=>t.enabled&&mt(),pronounce:r=>{t.enabled&&t.autoplayPronunciation&&ft(r,t.accent,t.pronunciationRate,t.pronunciationSource)},prefetch:r=>{t.enabled&&pt(r,t.accent,t.pronunciationSource)}}}import{useCallback as ae}from"react";function Pt(t){return ae(async e=>{let r={ts:new Date().toISOString(),dictId:t.dictId,chapter:t.chapterIndex,mode:t.mode,wordCount:e.wordCount,errors:e.errors,durationMs:e.durationMs,perWordErrors:e.perWordErrors};await St(r),Tt({dictId:t.dictId,chapterIndex:t.chapterIndex,mode:t.mode,wordCount:e.wordCount,errors:e.errors,durationMs:e.durationMs,perWordErrors:e.perWordErrors});let n=Object.entries(e.perWordErrors).filter(([,d])=>d>0);if(n.length===0)return;let c=await V();for(let[d,a]of n)c=wt(c,d,t.dictId,a);await yt(c)},[t.dictId,t.chapterIndex,t.mode])}import{Box as R,Text as m,useStdout as se}from"ink";import{Fragment as le,jsx as u,jsxs as B}from"react/jsx-runtime";var $t=28;function Rt(t){let e=Math.floor(t/1e3),r=Math.floor(e/60),n=e%60;return`${r}:${String(n).padStart(2,"0")}`}function ue(){let{stdout:t}=se(),e=t?.columns??80;return Math.max(20,e-$t)}function E({left:t,right:e}){let r=ue();return B(R,{children:[u(R,{width:r,children:t}),u(R,{width:$t,justifyContent:"flex-end",children:e})]})}function Nt(t){let e=W(),r=[...t.target],n=[...t.typed],c=B(R,{children:[r.map((I,w)=>{let p=w<n.length,x=t.hideTarget&&!p?"_":p?n[w]:I,v=t.error?i.error:p?i.accent:i.muted;return u(m,{bold:!0,color:v,children:x},w)}),t.phonetic&&B(le,{children:[u(m,{children:" "}),u(m,{italic:!0,dimColor:!0,color:i.muted,children:t.phonetic})]})]}),d=t.translation.length>0?u(m,{color:i.primary,children:t.translation[0]}):u(m,{children:" "}),a=t.info,s=Number.isInteger(a.accPct)?`${a.accPct}`:a.accPct.toFixed(1),l=!t.imeBlocked&&t.audioWarning!==null,y=t.imeBlocked?u(m,{color:i.warning,children:e.practice.imeWarningShort}):l?u(m,{color:i.warning,children:e.practice.audioWarningShort}):a.visible?u(m,{color:i.muted,children:`${a.dictName} \xB7 ${a.chapterLabel}`}):u(m,{children:" "}),M=t.imeBlocked||l?u(m,{children:" "}):a.visible?u(m,{color:i.muted,children:`${a.completed}/${a.total} \xB7 ${a.wpm}wpm \xB7 ${s}%`}):u(m,{children:" "}),T=t.imeBlocked||l?u(m,{children:" "}):a.visible?u(m,{color:i.muted,children:Rt(a.elapsedMs)}):u(m,{children:" "});return B(R,{flexDirection:"column",children:[u(E,{left:c,right:y}),u(E,{left:u(m,{children:" "}),right:M}),u(E,{left:d,right:T})]})}function Dt(){let t=W();return B(R,{flexDirection:"column",children:[u(E,{left:u(m,{color:i.warning,children:t.stealth.paused}),right:u(m,{color:i.muted,children:t.stealth.pausedHintRight})}),u(E,{left:u(m,{children:" "}),right:B(m,{color:i.muted,children:["Esc ",t.common.back]})}),u(E,{left:u(m,{children:" "}),right:u(m,{children:" "})})]})}function Lt(t){let e=W(),r=Number.isInteger(t.accPct)?`${t.accPct}`:t.accPct.toFixed(1),n=`${e.stealth.chapterDone} \xB7 ${t.wordCount}w \xB7 ${t.wpm}wpm \xB7 ${r}% \xB7 ${Rt(t.durationMs)}`;return B(R,{flexDirection:"column",children:[u(E,{left:u(m,{color:i.success,children:n}),right:u(m,{color:i.muted,children:e.stealth.nextHintRight})}),u(E,{left:u(m,{children:" "}),right:B(m,{color:i.muted,children:["Esc ",e.common.back]})}),u(E,{left:u(m,{children:" "}),right:u(m,{children:" "})})]})}import{jsx as o,jsxs as C}from"react/jsx-runtime";function ar({params:t}){let{dictId:e,chapterIndex:r,mode:n}=t,{cfg:c}=Y(),d=W(),[a,s]=j("loading"),[l,y]=j(null),[M,T]=j(null);return H(()=>{let I=!1;return s("loading"),y(null),T(null),(async()=>{try{let w=await st(e);if(I)return;if(n==="review"){let N=await V();if(I)return;let q=w.filter(F=>N[F.name]?.count).slice(0,c.chapterSize);if(q.length===0){T(d.practice.errors.noMistakes),s("error");return}y({playlist:q,totalChapters:1}),s("typing");return}let p=Ct(w,c.chapterSize);if(p.length===0){T(d.practice.errors.dictEmpty(e)),s("error");return}let x=Math.max(0,Math.min(p.length-1,r)),v=Mt(p[x],n);y({playlist:v,totalChapters:p.length}),s("typing")}catch(w){if(I)return;T(w.message),s("error")}})(),()=>{I=!0}},[e,r,n,c.chapterSize,d]),a==="loading"?o(be,{text:d.practice.loading,color:i.muted}):a==="error"?o(ge,{msg:M??d.practice.errors.unknown}):l?o(me,{params:t,loaded:l,phase:a,setPhase:s},`${e}-${r}-${n}-${t.stealth?"s":"n"}`):null}function me({params:t,loaded:e,phase:r,setPhase:n}){let{dictId:c,chapterIndex:d,mode:a}=t,s=t.stealth===!0,{cfg:l}=Y(),y=X(),{exit:M}=de(),T=()=>y.stack.length>1?y.back():M(),I=Pt({dictId:c,chapterIndex:d,mode:a}),w=gt(c),p=At({enabled:!s&&l.sounds.master,accent:l.accent,autoplayPronunciation:!s&&l.autoplayPronunciation,pronunciationRate:l.sounds.pronunciationRate,pronunciationSource:l.sounds.pronunciationSource}),x=ht(),v=l.sounds.master?x.warning:null,N=tt(!1),q=tt(0),F=tt(-1),[Ot,rt]=j(!1),[nt,Ft]=j(null),[ot,ct]=j(!1);H(()=>{if(nt===null)return;let h=setTimeout(()=>rt(!1),2e3);return()=>clearTimeout(h)},[nt]);let{session:S,lastEffect:A,effectSeq:K,tick:Vt}=Bt({playlist:e.playlist,enabled:r==="typing",onComplete:h=>{N.current||(N.current=!0,n("summary"),Promise.resolve(I(Z(h))).catch(b=>{console.error("Failed to persist session:",b)}))},onEscape:()=>n(r==="paused"?"typing":"paused"),onTab:s?void 0:()=>{let h=S.current?e.playlist[S.current.wordIndex]:void 0;h&&p.pronounce(h.name)},onImeBlock:()=>ct(!0),onValidInput:()=>ct(!1)});H(()=>{s||K!==q.current&&(q.current=K,A!==null&&(A==="wrong"&&l.sounds.feedback&&p.wrong(),A==="progress"&&l.sounds.keystroke&&p.keystroke(),A==="correct"&&(l.sounds.feedback&&p.correct(),l.sounds.keystroke&&p.keystroke())))},[s,K,A,p,l.sounds.feedback,l.sounds.keystroke]),H(()=>{if(s)return;let h=S.current?.wordIndex??-1;if(h===-1||h===F.current)return;F.current=h;let b=e.playlist[h],$=e.playlist[h+1];b&&l.autoplayPronunciation&&p.pronounce(b.name),$&&p.prefetch($.name)},[s,S.current?.wordIndex,p,l.autoplayPronunciation,e.playlist]),O((h,b)=>{if(b.tab){rt(!0),Ft(Date.now());return}},{isActive:s&&r==="typing"}),O((h,b)=>{if(b.return){n("typing");return}if(b.escape){T();return}},{isActive:r==="paused"}),O((h,b)=>{b.ctrl&&h==="c"&&(bt(!0),M())},{isActive:s&&r==="paused"}),O((h,b)=>{if(b.escape){T();return}if(b.return){let $=d+1;a==="loop"?y.replace({name:"practice",params:{dictId:c,chapterIndex:d,mode:a,stealth:t.stealth}}):a==="review"||$>=e.totalChapters?T():y.replace({name:"practice",params:{dictId:c,chapterIndex:$,mode:a,stealth:t.stealth}});return}if(h==="m"){y.replace({name:"practice",params:{dictId:c,chapterIndex:0,mode:"review",stealth:t.stealth}});return}},{isActive:r==="summary"});let P=S.results.length,_t=S.results.reduce((h,b)=>h+b.errors,0),z=Date.now()-S.startedAt,it=z/6e4,at=it>0?Math.round(P/it*10)/10:0,k=r==="summary"?Z(S):null;if(s){if(r==="paused")return o(Dt,{});if(r==="summary"&&k){let D=k.durationMs/6e4,Ut=D>0?Math.round(k.wordCount/D*10)/10:0,Xt=Object.keys(k.perWordErrors).length,Yt=k.wordCount===0?1:Math.max(0,(k.wordCount-Xt)/k.wordCount),Jt=Math.round(Yt*1e3)/10;return o(Lt,{wordCount:k.wordCount,errors:k.errors,durationMs:k.durationMs,wpm:Ut,accPct:Jt})}let h=S.current?e.playlist[S.current.wordIndex]:e.playlist[e.playlist.length-1],b=S.current?.input??{target:"",typed:"",errorsThisWord:0},$=new Set(S.results.filter(D=>D.errors>0).map(D=>D.word)).size,Ht=P===0?1:Math.max(0,(P-$)/P),Kt=Math.round(Ht*1e3)/10,zt=a==="review"?"review":`ch ${d+1}/${e.totalChapters}`;return o(Nt,{target:h?.name??"",typed:b.typed,hideTarget:a==="dictation",phonetic:jt(h,l.accent),translation:h?.trans??[],error:A==="wrong",imeBlocked:ot,audioWarning:v,info:{visible:Ot,dictName:L(w,24),chapterLabel:zt,completed:P,total:e.playlist.length,wpm:at,accPct:Kt,elapsedMs:z}})}if(r==="paused")return o(he,{dictName:w,chapterIndex:d,totalChapters:e.totalChapters,mode:a,completed:P,total:e.playlist.length});if(r==="summary"&&k)return o(ye,{dictName:w,chapterIndex:d,totalChapters:e.totalChapters,mode:a,summary:k});let U=S.current?e.playlist[S.current.wordIndex]:e.playlist[e.playlist.length-1],Gt=S.current?.input??{target:"",typed:"",errorsThisWord:0};return o(fe,{dictName:w,chapterIndex:d,totalChapters:e.totalChapters,mode:a,accent:l.accent,completed:P,total:e.playlist.length,errors:_t,wpm:at,elapsedMs:z,target:U?.name??"",typed:Gt.typed,flashError:A==="wrong",hideTarget:a==="dictation",phonetic:jt(U,l.accent),translation:U?.trans??[],imeBlocked:ot,audioWarning:v})}function jt(t,e){return t?(e==="us"?t.usphone:t.ukphone)??null:null}function et(t){let e=Math.floor(t/1e3),r=Math.floor(e/60),n=e%60;return`${r}:${String(n).padStart(2,"0")}`}function fe(t){let e=W(),r=t.total===0?0:t.completed/t.total;return C(f,{flexDirection:"column",paddingX:2,paddingY:1,width:"100%",height:"100%",children:[o(pe,{dictName:t.dictName,chapterIndex:t.chapterIndex,totalChapters:t.totalChapters,mode:t.mode,accent:t.accent,completed:t.completed,total:t.total,elapsedMs:t.elapsedMs}),C(f,{flexGrow:1,flexDirection:"column",alignItems:"center",justifyContent:"center",children:[o(xt,{target:t.target,typed:t.typed,error:t.flashError,hideTarget:t.hideTarget}),t.phonetic&&o(f,{marginTop:1,children:o(g,{italic:!0,dimColor:!0,color:i.muted,children:t.phonetic})}),t.translation.length>0&&o(f,{marginTop:1,flexDirection:"column",alignItems:"center",children:t.translation.slice(0,2).map((n,c)=>o(g,{color:i.primary,children:n},c))}),t.imeBlocked&&o(f,{marginTop:1,children:o(g,{color:i.warning,children:e.practice.imeWarning})}),!t.imeBlocked&&t.audioWarning&&o(f,{marginTop:1,children:o(g,{color:i.warning,children:t.audioWarning})})]}),C(f,{flexDirection:"column",children:[o(qt,{frac:r}),o(f,{justifyContent:"center",marginTop:1,children:C(g,{color:i.muted,children:[t.completed,"/",t.total," \xB7 ",et(t.elapsedMs)," \xB7 ",t.wpm," ",e.practice.statCards.wpm," \xB7 ",t.errors," ",e.practice.statCards.errors]})}),o(f,{justifyContent:"center",marginTop:1,children:o(g,{color:i.muted,children:e.practice.footers.typing})})]})]})}function pe(t){let e=W(),r=e.practice.modes[t.mode],n=e.practice.accents[t.accent],c=L(t.dictName,20),d=t.mode==="review"?`${c} \xB7 ${e.practice.reviewLabel} \xB7 ${n}`:`${c} \xB7 ${e.practice.chapterLabel(t.chapterIndex+1,t.totalChapters)} \xB7 ${r} \xB7 ${n}`,a=`${t.completed}/${t.total} \xB7 ${et(t.elapsedMs)}`;return C(f,{children:[o(g,{color:i.muted,children:d}),o(f,{flexGrow:1}),o(g,{color:i.muted,children:a})]})}function qt({frac:t}){let e=process.stdout.columns??80,r=Math.max(20,Math.min(72,e-16)),n=Math.round(r*Math.max(0,Math.min(1,t))),c=r-n;return C(f,{justifyContent:"center",children:[o(g,{color:i.accent,children:"\u2501".repeat(n)}),o(g,{color:i.muted,children:"\u2500".repeat(c)})]})}function he(t){let e=W(),r=t.total===0?0:t.completed/t.total,n=t.mode==="review"?`${L(t.dictName,20)} \xB7 ${e.practice.reviewLabel}`:`${L(t.dictName,20)} \xB7 ${e.practice.pause.chapter(t.chapterIndex+1,t.totalChapters)}`;return C(f,{flexDirection:"column",alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:[o(g,{bold:!0,color:i.warning,children:e.practice.pause.title}),o(f,{marginTop:1,children:o(g,{color:i.muted,children:n})}),o(f,{marginTop:2,children:o(qt,{frac:r})}),o(f,{marginTop:1,children:o(g,{color:i.muted,children:e.practice.pause.progress(t.completed,t.total)})}),o(f,{marginTop:2,children:o(g,{color:i.muted,children:e.practice.pause.hint})})]})}function ge({msg:t}){let e=W();return C(f,{flexDirection:"column",alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:[o(g,{color:i.error,children:t}),o(f,{marginTop:2,children:C(g,{color:i.muted,children:["Esc ",e.common.back]})}),o(xe,{})]})}function xe(){let t=X();return O((e,r)=>{r.escape&&t.back()}),null}function be({text:t,color:e}){return o(f,{alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:o(g,{color:e,children:t})})}function ye(t){let{summary:e}=t,r=e.durationMs/6e4,n=r>0?Math.round(e.wordCount/r*10)/10:0,c=Object.keys(e.perWordErrors).length,d=e.wordCount===0?1:Math.max(0,(e.wordCount-c)/e.wordCount),a=Math.round(d*1e3)/10,s=W(),l=s.practice.modes[t.mode],y=L(t.dictName,20),M=t.mode==="review"?`${y} \xB7 ${s.practice.reviewLabel}`:`${y} \xB7 ${s.practice.chapterLabel(t.chapterIndex+1,t.totalChapters)} \xB7 ${l}`,I=`Enter ${t.mode==="loop"?s.practice.summary.loopAgain:t.mode==="review"||t.chapterIndex+1>=t.totalChapters?s.practice.summary.backMenu:s.practice.summary.nextChapter} \xB7 m ${s.practice.summary.reviewMistakes} \xB7 Esc ${s.practice.summary.backMenu}`;return C(f,{flexDirection:"column",alignItems:"center",justifyContent:"center",paddingY:1,width:"100%",height:"100%",children:[o(g,{bold:!0,color:i.success,children:s.practice.chapterComplete}),o(f,{marginTop:1,children:o(g,{color:i.muted,children:M})}),C(f,{marginTop:3,flexDirection:"row",justifyContent:"center",children:[o(G,{label:s.practice.statCards.words,value:String(e.wordCount),color:i.text}),o(G,{label:s.practice.statCards.errors,value:String(e.errors),color:e.errors>0?i.error:i.muted}),o(G,{label:s.practice.statCards.wpm,value:String(n),color:i.accent}),o(G,{label:s.practice.statCards.accuracy,value:`${a}%`,color:i.accent})]}),o(f,{marginTop:2,children:o(g,{color:i.muted,children:s.practice.statCards.elapsed(et(e.durationMs))})}),o(f,{flexGrow:1}),o(f,{marginTop:2,children:o(g,{color:i.muted,children:I})})]})}function G({label:t,value:e,color:r}){return C(f,{flexDirection:"column",alignItems:"center",marginX:3,children:[o(g,{bold:!0,color:r,children:e}),o(g,{color:i.muted,children:t})]})}export{ar as PracticeScreen};
|
|
2
|
-
//# sourceMappingURL=PracticeScreen-
|
|
1
|
+
import{b as ht,e as bt,h as Tt}from"./chunk-6ROGUGNX.js";import{a as ut,b as lt,c as dt,d as mt,e as ft,f as pt}from"./chunk-BIBS2Q3E.js";import{e as st}from"./chunk-DURXS5MX.js";import{b as Y}from"./chunk-CQRKGMPU.js";import"./chunk-7LTZGB7F.js";import{a as V,b as wt,c as St}from"./chunk-G3DQB7FI.js";import{a as yt}from"./chunk-GULN5HRV.js";import{b as gt,e as L}from"./chunk-QG7ZTS2G.js";import{b as X,d as W,f as i,g as xt}from"./chunk-FQ2MEK7M.js";import"./chunk-NA5UNUVL.js";import"./chunk-E6BBQALJ.js";import{useState as j,useEffect as H,useRef as tt}from"react";import{Box as f,Text as g,useApp as de,useInput as O}from"ink";function It(t,e=Math.random){let r=[...t];for(let n=r.length-1;n>0;n--){let c=Math.floor(e()*(n+1)),d=r[n];r[n]=r[c],r[c]=d}return r}function kt(t){let e=t>>>0;return()=>{e=e+1831565813>>>0;let r=Math.imul(e^e>>>15,1|e);return r=r+Math.imul(r^r>>>7,61|r)^r,((r^r>>>14)>>>0)/4294967296}}function Ct(t,e){if(e<=0)throw new Error("chapterSize must be positive");let r=[];for(let n=0;n<t.length;n+=e)r.push(t.slice(n,n+e));return r}function Mt(t,e,r){if(e==="random"){let n=r===void 0?Math.random:kt(r);return It(t,n)}return t}function _(t){return{target:t,typed:"",errorsThisWord:0}}function Wt(t,e){switch(e.type){case"reset":return{state:{...t,typed:""},effect:"none"};case"backspace":return t.typed.length===0?{state:t,effect:"none"}:{state:{...t,typed:t.typed.slice(0,-1)},effect:"none"};case"char":{let r=t.typed+e.ch,n=[...t.target].slice(0,[...r].length).join("");return r===n?r.length===t.target.length?{state:{...t,typed:r},effect:"correct"}:{state:{...t,typed:r},effect:"progress"}:{state:{...t,typed:"",errorsThisWord:t.errorsThisWord+1},effect:"wrong"}}}}function J(t,e=Date.now()){return t.length===0?{startedAt:e,results:[],current:null,finishedAt:e,playlist:t}:{startedAt:e,results:[],current:{wordIndex:0,wordStartedAt:e,input:_(t[0].name)},finishedAt:null,playlist:t}}function Q(t,e,r=Date.now()){if(!t.current)return{session:t,effect:"none"};let{state:n,effect:c}=Wt(t.current.input,e);if(c==="correct"){let d={word:n.target,errors:n.errorsThisWord,durationMs:r-t.current.wordStartedAt},a=t.current.wordIndex+1,s=[...t.results,d];return a>=t.playlist.length?{session:{...t,results:s,current:null,finishedAt:r},effect:c}:{session:{...t,results:s,current:{wordIndex:a,wordStartedAt:r,input:_(t.playlist[a].name)}},effect:c}}return{session:{...t,current:{...t.current,input:n}},effect:c}}function vt(t,e=Date.now()){if(!t.current)return{session:t,effect:"none"};let r={word:t.current.input.target,errors:0,durationMs:e-t.current.wordStartedAt,skipped:!0},n=t.current.wordIndex+1,c=[...t.results,r];return n>=t.playlist.length?{session:{...t,results:c,current:null,finishedAt:e},effect:"skipped"}:{session:{...t,results:c,current:{wordIndex:n,wordStartedAt:e,input:_(t.playlist[n].name)}},effect:"skipped"}}function Z(t){let e=t.results.reduce((c,d)=>c+d.errors,0),r=(t.finishedAt??Date.now())-t.startedAt,n={};for(let c of t.results)c.errors>0&&(n[c.word]=(n[c.word]??0)+c.errors);return{wordCount:t.results.length,errors:e,durationMs:r,perWordErrors:n}}import{useEffect as Et,useReducer as Qt,useRef as Zt,useState as te}from"react";import{useInput as ee,useApp as re}from"ink";function ne(t,e){if(e.type==="start")return{session:J(e.playlist,e.now),lastEffect:null,effectSeq:0};if(e.type==="skip"){let r=vt(t.session,e.now);return{session:r.session,lastEffect:r.effect,effectSeq:t.effectSeq+1}}if(e.type==="event"){if(e.key.backspace||e.key.delete){let c=Q(t.session,{type:"backspace"},e.now);return{session:c.session,lastEffect:c.effect,effectSeq:t.effectSeq+1}}if(e.input.length===0)return t;let r=t.session,n=t.lastEffect;for(let c of e.input){let d=Q(r,{type:"char",ch:c},e.now);if(r=d.session,n=d.effect,r.finishedAt!==null)break}return{session:r,lastEffect:n,effectSeq:t.effectSeq+1}}return t}function oe(t){let e=[...t];if(e.some(n=>n.codePointAt(0)>=128))return{kind:"ime",cleaned:""};let r=e.filter(n=>{let c=n.codePointAt(0);return c>=32&&c<=126}).join("");return{kind:r.length>0?"valid":"noise",cleaned:r}}function Bt({playlist:t,onComplete:e,onTab:r,onEscape:n,onSkip:c,onImeBlock:d,onValidInput:a,enabled:s=!0}){let[l,y]=Qt(ne,void 0,()=>({session:J(t,Date.now()),lastEffect:null,effectSeq:0})),M=Zt(!1),[T,I]=te(0),{exit:w}=re();return ee((p,x)=>{if(x.ctrl&&p==="c"){w();return}if(x.ctrl&&p==="n"){c?.(),y({type:"skip",now:Date.now()});return}if(x.escape){n?.();return}if(x.tab){r?.();return}if(x.upArrow||x.downArrow||x.leftArrow||x.rightArrow||x.return||x.ctrl||x.meta)return;let{kind:v,cleaned:N}=oe(p);if(v==="ime"){d?.();return}v!=="noise"&&(a?.(),y({type:"event",input:N,key:x,now:Date.now()}))},{isActive:s}),Et(()=>{l.session.finishedAt!==null&&!M.current&&(M.current=!0,e(l.session))},[l.session,e]),Et(()=>{if(l.session.finishedAt!==null)return;let p=setInterval(()=>I(x=>x+1),1e3);return()=>clearInterval(p)},[l.session.finishedAt]),{session:l.session,lastEffect:l.lastEffect,effectSeq:l.effectSeq,tick:T}}import{useEffect as ce,useRef as ie}from"react";function At(t){let e=ie(!1);return ce(()=>{e.current||(e.current=!0,ut(!t.enabled).catch(()=>{}))},[t.enabled]),{keystroke:()=>t.enabled&<(),correct:()=>t.enabled&&dt(),wrong:()=>t.enabled&&mt(),pronounce:r=>{t.enabled&&t.autoplayPronunciation&&ft(r,t.accent,t.pronunciationRate,t.pronunciationSource)},prefetch:r=>{t.enabled&&pt(r,t.accent,t.pronunciationSource)}}}import{useCallback as ae}from"react";function Pt(t){return ae(async e=>{let r={ts:new Date().toISOString(),dictId:t.dictId,chapter:t.chapterIndex,mode:t.mode,wordCount:e.wordCount,errors:e.errors,durationMs:e.durationMs,perWordErrors:e.perWordErrors};await yt(r),Tt({dictId:t.dictId,chapterIndex:t.chapterIndex,mode:t.mode,wordCount:e.wordCount,errors:e.errors,durationMs:e.durationMs,perWordErrors:e.perWordErrors});let n=Object.entries(e.perWordErrors).filter(([,d])=>d>0);if(n.length===0)return;let c=await V();for(let[d,a]of n)c=St(c,d,t.dictId,a);await wt(c)},[t.dictId,t.chapterIndex,t.mode])}import{Box as R,Text as m,useStdout as se}from"ink";import{Fragment as le,jsx as u,jsxs as B}from"react/jsx-runtime";var $t=28;function Rt(t){let e=Math.floor(t/1e3),r=Math.floor(e/60),n=e%60;return`${r}:${String(n).padStart(2,"0")}`}function ue(){let{stdout:t}=se(),e=t?.columns??80;return Math.max(20,e-$t)}function E({left:t,right:e}){let r=ue();return B(R,{children:[u(R,{width:r,children:t}),u(R,{width:$t,justifyContent:"flex-end",children:e})]})}function Nt(t){let e=W(),r=[...t.target],n=[...t.typed],c=B(R,{children:[r.map((I,w)=>{let p=w<n.length,x=t.hideTarget&&!p?"_":p?n[w]:I,v=t.error?i.error:p?i.accent:i.muted;return u(m,{bold:!0,color:v,children:x},w)}),t.phonetic&&B(le,{children:[u(m,{children:" "}),u(m,{italic:!0,dimColor:!0,color:i.muted,children:t.phonetic})]})]}),d=t.translation.length>0?u(m,{color:i.primary,children:t.translation[0]}):u(m,{children:" "}),a=t.info,s=Number.isInteger(a.accPct)?`${a.accPct}`:a.accPct.toFixed(1),l=!t.imeBlocked&&t.audioWarning!==null,y=t.imeBlocked?u(m,{color:i.warning,children:e.practice.imeWarningShort}):l?u(m,{color:i.warning,children:e.practice.audioWarningShort}):a.visible?u(m,{color:i.muted,children:`${a.dictName} \xB7 ${a.chapterLabel}`}):u(m,{children:" "}),M=t.imeBlocked||l?u(m,{children:" "}):a.visible?u(m,{color:i.muted,children:`${a.completed}/${a.total} \xB7 ${a.wpm}wpm \xB7 ${s}%`}):u(m,{children:" "}),T=t.imeBlocked||l?u(m,{children:" "}):a.visible?u(m,{color:i.muted,children:Rt(a.elapsedMs)}):u(m,{children:" "});return B(R,{flexDirection:"column",children:[u(E,{left:c,right:y}),u(E,{left:u(m,{children:" "}),right:M}),u(E,{left:d,right:T})]})}function Dt(){let t=W();return B(R,{flexDirection:"column",children:[u(E,{left:u(m,{color:i.warning,children:t.stealth.paused}),right:u(m,{color:i.muted,children:t.stealth.pausedHintRight})}),u(E,{left:u(m,{children:" "}),right:B(m,{color:i.muted,children:["Esc ",t.common.back]})}),u(E,{left:u(m,{children:" "}),right:u(m,{children:" "})})]})}function Lt(t){let e=W(),r=Number.isInteger(t.accPct)?`${t.accPct}`:t.accPct.toFixed(1),n=`${e.stealth.chapterDone} \xB7 ${t.wordCount}w \xB7 ${t.wpm}wpm \xB7 ${r}% \xB7 ${Rt(t.durationMs)}`;return B(R,{flexDirection:"column",children:[u(E,{left:u(m,{color:i.success,children:n}),right:u(m,{color:i.muted,children:e.stealth.nextHintRight})}),u(E,{left:u(m,{children:" "}),right:B(m,{color:i.muted,children:["Esc ",e.common.back]})}),u(E,{left:u(m,{children:" "}),right:u(m,{children:" "})})]})}import{jsx as o,jsxs as C}from"react/jsx-runtime";function ar({params:t}){let{dictId:e,chapterIndex:r,mode:n}=t,{cfg:c}=Y(),d=W(),[a,s]=j("loading"),[l,y]=j(null),[M,T]=j(null);return H(()=>{let I=!1;return s("loading"),y(null),T(null),(async()=>{try{let w=await st(e);if(I)return;if(n==="review"){let N=await V();if(I)return;let q=w.filter(F=>N[F.name]?.count).slice(0,c.chapterSize);if(q.length===0){T(d.practice.errors.noMistakes),s("error");return}y({playlist:q,totalChapters:1}),s("typing");return}let p=Ct(w,c.chapterSize);if(p.length===0){T(d.practice.errors.dictEmpty(e)),s("error");return}let x=Math.max(0,Math.min(p.length-1,r)),v=Mt(p[x],n);y({playlist:v,totalChapters:p.length}),s("typing")}catch(w){if(I)return;T(w.message),s("error")}})(),()=>{I=!0}},[e,r,n,c.chapterSize,d]),a==="loading"?o(be,{text:d.practice.loading,color:i.muted}):a==="error"?o(ge,{msg:M??d.practice.errors.unknown}):l?o(me,{params:t,loaded:l,phase:a,setPhase:s},`${e}-${r}-${n}-${t.stealth?"s":"n"}`):null}function me({params:t,loaded:e,phase:r,setPhase:n}){let{dictId:c,chapterIndex:d,mode:a}=t,s=t.stealth===!0,{cfg:l}=Y(),y=X(),{exit:M}=de(),T=()=>y.stack.length>1?y.back():M(),I=Pt({dictId:c,chapterIndex:d,mode:a}),w=gt(c),p=At({enabled:!s&&l.sounds.master,accent:l.accent,autoplayPronunciation:!s&&l.autoplayPronunciation,pronunciationRate:l.sounds.pronunciationRate,pronunciationSource:l.sounds.pronunciationSource}),x=ht(),v=l.sounds.master?x.warning:null,N=tt(!1),q=tt(0),F=tt(-1),[Ot,rt]=j(!1),[nt,Ft]=j(null),[ot,ct]=j(!1);H(()=>{if(nt===null)return;let h=setTimeout(()=>rt(!1),2e3);return()=>clearTimeout(h)},[nt]);let{session:S,lastEffect:A,effectSeq:K,tick:Vt}=Bt({playlist:e.playlist,enabled:r==="typing",onComplete:h=>{N.current||(N.current=!0,n("summary"),Promise.resolve(I(Z(h))).catch(b=>{console.error("Failed to persist session:",b)}))},onEscape:()=>n(r==="paused"?"typing":"paused"),onTab:s?void 0:()=>{let h=S.current?e.playlist[S.current.wordIndex]:void 0;h&&p.pronounce(h.name)},onImeBlock:()=>ct(!0),onValidInput:()=>ct(!1)});H(()=>{s||K!==q.current&&(q.current=K,A!==null&&(A==="wrong"&&l.sounds.feedback&&p.wrong(),A==="progress"&&l.sounds.keystroke&&p.keystroke(),A==="correct"&&(l.sounds.feedback&&p.correct(),l.sounds.keystroke&&p.keystroke())))},[s,K,A,p,l.sounds.feedback,l.sounds.keystroke]),H(()=>{if(s)return;let h=S.current?.wordIndex??-1;if(h===-1||h===F.current)return;F.current=h;let b=e.playlist[h],$=e.playlist[h+1];b&&l.autoplayPronunciation&&p.pronounce(b.name),$&&p.prefetch($.name)},[s,S.current?.wordIndex,p,l.autoplayPronunciation,e.playlist]),O((h,b)=>{if(b.tab){rt(!0),Ft(Date.now());return}},{isActive:s&&r==="typing"}),O((h,b)=>{if(b.return){n("typing");return}if(b.escape){T();return}},{isActive:r==="paused"}),O((h,b)=>{b.ctrl&&h==="c"&&(bt(!0),M())},{isActive:s&&r==="paused"}),O((h,b)=>{if(b.escape){T();return}if(b.return){let $=d+1;a==="loop"?y.replace({name:"practice",params:{dictId:c,chapterIndex:d,mode:a,stealth:t.stealth}}):a==="review"||$>=e.totalChapters?T():y.replace({name:"practice",params:{dictId:c,chapterIndex:$,mode:a,stealth:t.stealth}});return}if(h==="m"){y.replace({name:"practice",params:{dictId:c,chapterIndex:0,mode:"review",stealth:t.stealth}});return}},{isActive:r==="summary"});let P=S.results.length,_t=S.results.reduce((h,b)=>h+b.errors,0),z=Date.now()-S.startedAt,it=z/6e4,at=it>0?Math.round(P/it*10)/10:0,k=r==="summary"?Z(S):null;if(s){if(r==="paused")return o(Dt,{});if(r==="summary"&&k){let D=k.durationMs/6e4,Ut=D>0?Math.round(k.wordCount/D*10)/10:0,Xt=Object.keys(k.perWordErrors).length,Yt=k.wordCount===0?1:Math.max(0,(k.wordCount-Xt)/k.wordCount),Jt=Math.round(Yt*1e3)/10;return o(Lt,{wordCount:k.wordCount,errors:k.errors,durationMs:k.durationMs,wpm:Ut,accPct:Jt})}let h=S.current?e.playlist[S.current.wordIndex]:e.playlist[e.playlist.length-1],b=S.current?.input??{target:"",typed:"",errorsThisWord:0},$=new Set(S.results.filter(D=>D.errors>0).map(D=>D.word)).size,Ht=P===0?1:Math.max(0,(P-$)/P),Kt=Math.round(Ht*1e3)/10,zt=a==="review"?"review":`ch ${d+1}/${e.totalChapters}`;return o(Nt,{target:h?.name??"",typed:b.typed,hideTarget:a==="dictation",phonetic:jt(h,l.accent),translation:h?.trans??[],error:A==="wrong",imeBlocked:ot,audioWarning:v,info:{visible:Ot,dictName:L(w,24),chapterLabel:zt,completed:P,total:e.playlist.length,wpm:at,accPct:Kt,elapsedMs:z}})}if(r==="paused")return o(he,{dictName:w,chapterIndex:d,totalChapters:e.totalChapters,mode:a,completed:P,total:e.playlist.length});if(r==="summary"&&k)return o(ye,{dictName:w,chapterIndex:d,totalChapters:e.totalChapters,mode:a,summary:k});let U=S.current?e.playlist[S.current.wordIndex]:e.playlist[e.playlist.length-1],Gt=S.current?.input??{target:"",typed:"",errorsThisWord:0};return o(fe,{dictName:w,chapterIndex:d,totalChapters:e.totalChapters,mode:a,accent:l.accent,completed:P,total:e.playlist.length,errors:_t,wpm:at,elapsedMs:z,target:U?.name??"",typed:Gt.typed,flashError:A==="wrong",hideTarget:a==="dictation",phonetic:jt(U,l.accent),translation:U?.trans??[],imeBlocked:ot,audioWarning:v})}function jt(t,e){return t?(e==="us"?t.usphone:t.ukphone)??null:null}function et(t){let e=Math.floor(t/1e3),r=Math.floor(e/60),n=e%60;return`${r}:${String(n).padStart(2,"0")}`}function fe(t){let e=W(),r=t.total===0?0:t.completed/t.total;return C(f,{flexDirection:"column",paddingX:2,paddingY:1,width:"100%",height:"100%",children:[o(pe,{dictName:t.dictName,chapterIndex:t.chapterIndex,totalChapters:t.totalChapters,mode:t.mode,accent:t.accent,completed:t.completed,total:t.total,elapsedMs:t.elapsedMs}),C(f,{flexGrow:1,flexDirection:"column",alignItems:"center",justifyContent:"center",children:[o(xt,{target:t.target,typed:t.typed,error:t.flashError,hideTarget:t.hideTarget}),t.phonetic&&o(f,{marginTop:1,children:o(g,{italic:!0,dimColor:!0,color:i.muted,children:t.phonetic})}),t.translation.length>0&&o(f,{marginTop:1,flexDirection:"column",alignItems:"center",children:t.translation.slice(0,2).map((n,c)=>o(g,{color:i.primary,children:n},c))}),t.imeBlocked&&o(f,{marginTop:1,children:o(g,{color:i.warning,children:e.practice.imeWarning})}),!t.imeBlocked&&t.audioWarning&&o(f,{marginTop:1,children:o(g,{color:i.warning,children:t.audioWarning})})]}),C(f,{flexDirection:"column",children:[o(qt,{frac:r}),o(f,{justifyContent:"center",marginTop:1,children:C(g,{color:i.muted,children:[t.completed,"/",t.total," \xB7 ",et(t.elapsedMs)," \xB7 ",t.wpm," ",e.practice.statCards.wpm," \xB7 ",t.errors," ",e.practice.statCards.errors]})}),o(f,{justifyContent:"center",marginTop:1,children:o(g,{color:i.muted,children:e.practice.footers.typing})})]})]})}function pe(t){let e=W(),r=e.practice.modes[t.mode],n=e.practice.accents[t.accent],c=L(t.dictName,20),d=t.mode==="review"?`${c} \xB7 ${e.practice.reviewLabel} \xB7 ${n}`:`${c} \xB7 ${e.practice.chapterLabel(t.chapterIndex+1,t.totalChapters)} \xB7 ${r} \xB7 ${n}`,a=`${t.completed}/${t.total} \xB7 ${et(t.elapsedMs)}`;return C(f,{children:[o(g,{color:i.muted,children:d}),o(f,{flexGrow:1}),o(g,{color:i.muted,children:a})]})}function qt({frac:t}){let e=process.stdout.columns??80,r=Math.max(20,Math.min(72,e-16)),n=Math.round(r*Math.max(0,Math.min(1,t))),c=r-n;return C(f,{justifyContent:"center",children:[o(g,{color:i.accent,children:"\u2501".repeat(n)}),o(g,{color:i.muted,children:"\u2500".repeat(c)})]})}function he(t){let e=W(),r=t.total===0?0:t.completed/t.total,n=t.mode==="review"?`${L(t.dictName,20)} \xB7 ${e.practice.reviewLabel}`:`${L(t.dictName,20)} \xB7 ${e.practice.pause.chapter(t.chapterIndex+1,t.totalChapters)}`;return C(f,{flexDirection:"column",alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:[o(g,{bold:!0,color:i.warning,children:e.practice.pause.title}),o(f,{marginTop:1,children:o(g,{color:i.muted,children:n})}),o(f,{marginTop:2,children:o(qt,{frac:r})}),o(f,{marginTop:1,children:o(g,{color:i.muted,children:e.practice.pause.progress(t.completed,t.total)})}),o(f,{marginTop:2,children:o(g,{color:i.muted,children:e.practice.pause.hint})})]})}function ge({msg:t}){let e=W();return C(f,{flexDirection:"column",alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:[o(g,{color:i.error,children:t}),o(f,{marginTop:2,children:C(g,{color:i.muted,children:["Esc ",e.common.back]})}),o(xe,{})]})}function xe(){let t=X();return O((e,r)=>{r.escape&&t.back()}),null}function be({text:t,color:e}){return o(f,{alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:o(g,{color:e,children:t})})}function ye(t){let{summary:e}=t,r=e.durationMs/6e4,n=r>0?Math.round(e.wordCount/r*10)/10:0,c=Object.keys(e.perWordErrors).length,d=e.wordCount===0?1:Math.max(0,(e.wordCount-c)/e.wordCount),a=Math.round(d*1e3)/10,s=W(),l=s.practice.modes[t.mode],y=L(t.dictName,20),M=t.mode==="review"?`${y} \xB7 ${s.practice.reviewLabel}`:`${y} \xB7 ${s.practice.chapterLabel(t.chapterIndex+1,t.totalChapters)} \xB7 ${l}`,I=`Enter ${t.mode==="loop"?s.practice.summary.loopAgain:t.mode==="review"||t.chapterIndex+1>=t.totalChapters?s.practice.summary.backMenu:s.practice.summary.nextChapter} \xB7 m ${s.practice.summary.reviewMistakes} \xB7 Esc ${s.practice.summary.backMenu}`;return C(f,{flexDirection:"column",alignItems:"center",justifyContent:"center",paddingY:1,width:"100%",height:"100%",children:[o(g,{bold:!0,color:i.success,children:s.practice.chapterComplete}),o(f,{marginTop:1,children:o(g,{color:i.muted,children:M})}),C(f,{marginTop:3,flexDirection:"row",justifyContent:"center",children:[o(G,{label:s.practice.statCards.words,value:String(e.wordCount),color:i.text}),o(G,{label:s.practice.statCards.errors,value:String(e.errors),color:e.errors>0?i.error:i.muted}),o(G,{label:s.practice.statCards.wpm,value:String(n),color:i.accent}),o(G,{label:s.practice.statCards.accuracy,value:`${a}%`,color:i.accent})]}),o(f,{marginTop:2,children:o(g,{color:i.muted,children:s.practice.statCards.elapsed(et(e.durationMs))})}),o(f,{flexGrow:1}),o(f,{marginTop:2,children:o(g,{color:i.muted,children:I})})]})}function G({label:t,value:e,color:r}){return C(f,{flexDirection:"column",alignItems:"center",marginX:3,children:[o(g,{bold:!0,color:r,children:e}),o(g,{color:i.muted,children:t})]})}export{ar as PracticeScreen};
|
|
2
|
+
//# sourceMappingURL=PracticeScreen-PWXXG33U.js.map
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{
|
|
2
|
-
//# sourceMappingURL=StatsViewer-
|
|
1
|
+
import{a as j,d as F}from"./chunk-G3DQB7FI.js";import{b as N,c as $,d as O,f as V,h as Y}from"./chunk-GULN5HRV.js";import{b as B,d as k,e as y}from"./chunk-QG7ZTS2G.js";import{b as R,d as C,f as o}from"./chunk-FQ2MEK7M.js";import"./chunk-NA5UNUVL.js";import"./chunk-E6BBQALJ.js";import{useEffect as G,useState as D}from"react";import{Box as i,Text as r,useInput as U,useStdout as q}from"ink";import{jsx as e,jsxs as c}from"react/jsx-runtime";var W=[7,14,30,90];function it(){let s=R(),t=C(),[n,u]=D(null),[x,g]=D(null),[d,b]=D(1);if(G(()=>{(async()=>{let[l,a]=await Promise.all([N(),j()]);u(l),g(a)})()},[]),U((l,a)=>{if(a.escape){s.back();return}a.rightArrow&&b(M=>(M+1)%W.length),a.leftArrow&&b(M=>(M-1+W.length)%W.length)}),!n||!x)return e(i,{alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:e(r,{color:o.muted,children:t.stats.loading})});if(n.length===0)return c(i,{flexDirection:"column",alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:[e(r,{color:o.muted,children:t.stats.none}),e(i,{marginTop:1,children:e(r,{color:o.muted,children:t.stats.nonePractice})}),e(i,{marginTop:2,children:c(r,{color:o.muted,children:["Esc ",t.common.back]})})]});let m=W[d],S=V(n),f=n.reduce((l,a)=>l+a.wordCount,0),w=n.reduce((l,a)=>l+a.errors,0),T=n.reduce((l,a)=>l+a.durationMs,0),v=n.reduce((l,a)=>l+(a.wordCount-Object.keys(a.perWordErrors).length),0),H=T>0?Math.round(f/(T/6e4)*10)/10:0,X=f===0?1:v/f,_=n.slice(-5).reverse(),I=F(x,8),p=Y(n,m),A=p.peakWpmLifetime>0?Math.min(1,p.avgWpm/p.peakWpmLifetime):0,L=Math.min(1,p.avgAccPct/100),P=m>0?Math.min(1,p.sessionsInWindow/m):0;return c(i,{flexDirection:"column",paddingX:2,paddingY:1,width:"100%",height:"100%",children:[e(r,{bold:!0,color:o.accent,children:t.stats.title}),c(i,{marginTop:1,flexDirection:"column",children:[e(r,{color:o.muted,children:t.stats.lifetime}),c(i,{marginTop:1,children:[e(h,{label:t.stats.sessions,value:String(n.length)}),e(h,{label:t.stats.words,value:String(f)}),e(h,{label:t.stats.errors,value:String(w)}),e(h,{label:t.stats.wpm,value:String(H),accent:!0}),e(h,{label:t.stats.accuracy,value:`${Math.round(X*1e3)/10}%`,accent:!0}),e(h,{label:t.stats.streak,value:`${S}d`,accent:!0})]})]}),c(i,{marginTop:2,flexDirection:"column",children:[e(r,{color:o.muted,children:t.stats.last(m)}),c(i,{marginTop:1,flexDirection:"column",borderStyle:"round",borderColor:o.muted,paddingX:2,paddingY:0,children:[e(E,{label:t.stats.bars.speed,frac:A,valueText:`${p.avgWpm} wpm`,pct:Math.round(A*100)}),e(E,{label:t.stats.bars.accuracy,frac:L,valueText:`${p.avgAccPct}%`,pct:Math.round(L*100)}),e(E,{label:t.stats.bars.sessions,frac:P,valueText:`${p.sessionsInWindow} / ${m}`,pct:Math.round(P*100)})]})]}),c(i,{marginTop:2,flexDirection:"column",children:[e(r,{color:o.muted,children:t.stats.recent}),_.map((l,a)=>e(z,{session:l,units:t.stats.recentUnits},a))]}),I.length>0&&c(i,{marginTop:2,flexDirection:"column",children:[e(r,{color:o.muted,children:t.stats.topMistakes}),I.map(([l,a])=>e(J,{word:l,count:a.count,dictIds:a.dictIds,multiSuffix:t.stats.multiDictSuffix},l))]}),e(i,{flexGrow:1}),e(i,{marginTop:1,children:e(r,{color:o.muted,children:t.stats.footer})})]})}function z({session:s,units:t}){let n=B(s.dictId),u=y(n,14);return c(i,{children:[c(r,{color:o.muted,children:[" ",s.ts.replace("T"," ").slice(0,16)," "]}),e(r,{color:o.text,children:u.padEnd(14)}),c(r,{color:o.muted,children:[" ","ch",String(s.chapter+1).padStart(3)," ",s.mode.padEnd(9)," ",String(s.wordCount).padStart(3),t.words," ",s.errors,t.errors," ",$(s),t.wpm," ",Math.round(O(s)*1e3)/10,"%"]})]})}function J({word:s,count:t,dictIds:n,multiSuffix:u}){let x=n[0]??"",g=B(x),d=n.length>1?u(n.length-1):"";return c(i,{children:[c(r,{color:o.error,children:[" ",String(t).padStart(3)," "]}),e(r,{color:o.text,children:s.padEnd(20)}),c(r,{color:o.muted,children:[y(g,20),d]})]})}function h({label:s,value:t,accent:n=!1}){return c(i,{marginRight:3,children:[c(r,{color:o.muted,children:[s," "]}),e(r,{bold:!0,color:n?o.accent:o.text,children:t})]})}function K(s){return{barWidth:Math.max(0,Math.min(60,s-11-24)),labelWidth:9,valueWidth:8,pctWidth:5,marginLeft:2}}function E({label:s,frac:t,valueText:n,pct:u}){let{stdout:x}=q(),g=x?.columns??80,d=K(g),b=Math.max(0,Math.min(1,t)),m=Math.round(d.barWidth*b),S=d.barWidth-m,f=k(s,d.labelWidth),w=k(n,d.valueWidth),T=String(u).padStart(d.pctWidth-1," ")+"%",v=" ".repeat(d.marginLeft);return c(i,{children:[e(r,{color:o.muted,children:f}),e(r,{color:o.accent,children:"\u2501".repeat(m)}),e(r,{color:o.muted,children:"\u2500".repeat(S)}),c(r,{color:o.text,children:[v,w]}),e(r,{color:o.muted,children:T})]})}export{it as StatsViewer,K as calcMetricBarLayout};
|
|
2
|
+
//# sourceMappingURL=StatsViewer-XARAMVZW.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/ui/screens/StatsViewer.tsx"],"sourcesContent":["import { useEffect, useState } from 'react';\nimport { Box, Text, useInput, useStdout } from 'ink';\nimport { useNav } from '../nav.js';\nimport { useStrings } from '../../i18n/context.js';\nimport { useDictName } from '../registry-context.js';\nimport { PALETTE } from '../components/BigWord.js';\nimport {\n loadSessions,\n computeWPM,\n accuracy,\n windowAggregate,\n dailyStreak,\n type SessionRecord,\n} from '../../domain/stats.js';\nimport { loadMistakes, topN, type MistakeBook } from '../../domain/mistakes.js';\nimport { truncateName } from '../../util/dict-name.js';\nimport { pad } from '../../util/text.js';\n\nconst DAY_WINDOWS = [7, 14, 30, 90];\n\nexport function StatsViewer() {\n const nav = useNav();\n const t = useStrings();\n const [sessions, setSessions] = useState<SessionRecord[] | null>(null);\n const [book, setBook] = useState<MistakeBook | null>(null);\n const [windowIdx, setWindowIdx] = useState(1);\n\n useEffect(() => {\n void (async () => {\n const [s, b] = await Promise.all([loadSessions(), loadMistakes()]);\n setSessions(s);\n setBook(b);\n })();\n }, []);\n\n useInput((_input, key) => {\n if (key.escape) {\n nav.back();\n return;\n }\n if (key.rightArrow) setWindowIdx((i) => (i + 1) % DAY_WINDOWS.length);\n if (key.leftArrow) setWindowIdx((i) => (i - 1 + DAY_WINDOWS.length) % DAY_WINDOWS.length);\n });\n\n if (!sessions || !book) {\n return (\n <Box alignItems=\"center\" justifyContent=\"center\" width=\"100%\" height=\"100%\">\n <Text color={PALETTE.muted}>{t.stats.loading}</Text>\n </Box>\n );\n }\n\n if (sessions.length === 0) {\n return (\n <Box flexDirection=\"column\" alignItems=\"center\" justifyContent=\"center\" width=\"100%\" height=\"100%\">\n <Text color={PALETTE.muted}>{t.stats.none}</Text>\n <Box marginTop={1}>\n <Text color={PALETTE.muted}>{t.stats.nonePractice}</Text>\n </Box>\n <Box marginTop={2}>\n <Text color={PALETTE.muted}>Esc {t.common.back}</Text>\n </Box>\n </Box>\n );\n }\n\n const days = DAY_WINDOWS[windowIdx]!;\n const streak = dailyStreak(sessions);\n const totalWords = sessions.reduce((a, s) => a + s.wordCount, 0);\n const totalErrors = sessions.reduce((a, s) => a + s.errors, 0);\n const totalMs = sessions.reduce((a, s) => a + s.durationMs, 0);\n const firstTryWords = sessions.reduce(\n (a, s) => a + (s.wordCount - Object.keys(s.perWordErrors).length),\n 0,\n );\n const overallWpm = totalMs > 0 ? Math.round((totalWords / (totalMs / 60000)) * 10) / 10 : 0;\n const overallAcc = totalWords === 0 ? 1 : firstTryWords / totalWords;\n\n const recent = sessions.slice(-5).reverse();\n const top = topN(book, 8);\n\n const agg = windowAggregate(sessions, days);\n const speedPct = agg.peakWpmLifetime > 0 ? Math.min(1, agg.avgWpm / agg.peakWpmLifetime) : 0;\n const accPctFrac = Math.min(1, agg.avgAccPct / 100);\n const sessionsPctFrac = days > 0 ? Math.min(1, agg.sessionsInWindow / days) : 0;\n\n return (\n <Box flexDirection=\"column\" paddingX={2} paddingY={1} width=\"100%\" height=\"100%\">\n <Text bold color={PALETTE.accent}>\n {t.stats.title}\n </Text>\n\n <Box marginTop={1} flexDirection=\"column\">\n <Text color={PALETTE.muted}>{t.stats.lifetime}</Text>\n <Box marginTop={1}>\n <Stat label={t.stats.sessions} value={String(sessions.length)} />\n <Stat label={t.stats.words} value={String(totalWords)} />\n <Stat label={t.stats.errors} value={String(totalErrors)} />\n <Stat label={t.stats.wpm} value={String(overallWpm)} accent />\n <Stat label={t.stats.accuracy} value={`${Math.round(overallAcc * 1000) / 10}%`} accent />\n <Stat label={t.stats.streak} value={`${streak}d`} accent />\n </Box>\n </Box>\n\n <Box marginTop={2} flexDirection=\"column\">\n <Text color={PALETTE.muted}>{t.stats.last(days)}</Text>\n <Box marginTop={1} flexDirection=\"column\" borderStyle=\"round\" borderColor={PALETTE.muted} paddingX={2} paddingY={1}>\n <MetricBar\n label={t.stats.bars.speed}\n frac={speedPct}\n valueText={`${agg.avgWpm} wpm`}\n pct={Math.round(speedPct * 100)}\n />\n <MetricBar\n label={t.stats.bars.accuracy}\n frac={accPctFrac}\n valueText={`${agg.avgAccPct}%`}\n pct={Math.round(accPctFrac * 100)}\n />\n <MetricBar\n label={t.stats.bars.sessions}\n frac={sessionsPctFrac}\n valueText={`${agg.sessionsInWindow} / ${days}`}\n pct={Math.round(sessionsPctFrac * 100)}\n />\n </Box>\n </Box>\n\n <Box marginTop={2} flexDirection=\"column\">\n <Text color={PALETTE.muted}>{t.stats.recent}</Text>\n {recent.map((s, i) => (\n <RecentRow key={i} session={s} units={t.stats.recentUnits} />\n ))}\n </Box>\n\n {top.length > 0 && (\n <Box marginTop={2} flexDirection=\"column\">\n <Text color={PALETTE.muted}>{t.stats.topMistakes}</Text>\n {top.map(([word, entry]) => (\n <MistakeRow\n key={word}\n word={word}\n count={entry.count}\n dictIds={entry.dictIds}\n multiSuffix={t.stats.multiDictSuffix}\n />\n ))}\n </Box>\n )}\n\n <Box flexGrow={1} />\n <Box marginTop={1}>\n <Text color={PALETTE.muted}>{t.stats.footer}</Text>\n </Box>\n </Box>\n );\n}\n\nfunction RecentRow({\n session,\n units,\n}: {\n session: SessionRecord;\n units: { words: string; errors: string; wpm: string };\n}) {\n const name = useDictName(session.dictId);\n const display = truncateName(name, 14);\n return (\n <Box>\n <Text color={PALETTE.muted}> {session.ts.replace('T', ' ').slice(0, 16)} </Text>\n <Text color={PALETTE.text}>{display.padEnd(14)}</Text>\n <Text color={PALETTE.muted}>\n {' '}ch{String(session.chapter + 1).padStart(3)} {session.mode.padEnd(9)} {String(session.wordCount).padStart(3)}{units.words} {session.errors}{units.errors} {computeWPM(session)}{units.wpm} {Math.round(accuracy(session) * 1000) / 10}%\n </Text>\n </Box>\n );\n}\n\nfunction MistakeRow({\n word,\n count,\n dictIds,\n multiSuffix,\n}: {\n word: string;\n count: number;\n dictIds: string[];\n multiSuffix: (n: number) => string;\n}) {\n const firstId = dictIds[0] ?? '';\n const firstName = useDictName(firstId);\n const suffix = dictIds.length > 1 ? multiSuffix(dictIds.length - 1) : '';\n return (\n <Box>\n <Text color={PALETTE.error}> {String(count).padStart(3)} </Text>\n <Text color={PALETTE.text}>{word.padEnd(20)}</Text>\n <Text color={PALETTE.muted}>\n {truncateName(firstName, 20)}\n {suffix}\n </Text>\n </Box>\n );\n}\n\nfunction Stat({ label, value, accent = false }: { label: string; value: string; accent?: boolean }) {\n return (\n <Box marginRight={3}>\n <Text color={PALETTE.muted}>{label} </Text>\n <Text bold color={accent ? PALETTE.accent : PALETTE.text}>\n {value}\n </Text>\n </Box>\n );\n}\n\n// Pure layout calculation — extracted for testing without an Ink renderer.\n// OVERHEAD = outer paddingX(4) + inner border(2) + inner paddingX(4) + 1 col slack = 11.\n// The +1 slack guards against off-by-one in Ink/terminal cell rounding —\n// content was previously sized exactly to the inner width and any rounding\n// would wrap the bar onto a second line.\n// Fixed widths sum to 24; bar takes whatever cols remain, clamped 0..60.\nexport function calcMetricBarLayout(cols: number): {\n barWidth: number;\n labelWidth: number;\n valueWidth: number;\n pctWidth: number;\n marginLeft: number;\n} {\n const OVERHEAD = 11;\n const labelWidth = 9;\n const valueWidth = 8;\n const pctWidth = 5;\n const marginLeft = 2;\n const fixed = labelWidth + marginLeft + valueWidth + pctWidth;\n const barWidth = Math.max(0, Math.min(60, cols - OVERHEAD - fixed));\n return { barWidth, labelWidth, valueWidth, pctWidth, marginLeft };\n}\n\n// MetricBar uses pure inline <Text> siblings with CJK-aware pad() padding\n// rather than nested <Box width=N> wrappers. Mixed Box+Text flex children\n// can wrap unpredictably at terminal-width boundaries; inline text widths\n// are deterministic.\nfunction MetricBar({\n label,\n frac,\n valueText,\n pct,\n}: {\n label: string;\n frac: number;\n valueText: string;\n pct: number;\n}) {\n const { stdout } = useStdout();\n const cols = stdout?.columns ?? 80;\n const layout = calcMetricBarLayout(cols);\n const clamped = Math.max(0, Math.min(1, frac));\n const filled = Math.round(layout.barWidth * clamped);\n const empty = layout.barWidth - filled;\n const labelStr = pad(label, layout.labelWidth);\n const valueStr = pad(valueText, layout.valueWidth);\n const pctStr = String(pct).padStart(layout.pctWidth - 1, ' ') + '%';\n const margin = ' '.repeat(layout.marginLeft);\n return (\n <Box>\n <Text color={PALETTE.muted}>{labelStr}</Text>\n <Text color={PALETTE.accent}>{'━'.repeat(filled)}</Text>\n <Text color={PALETTE.muted}>{'─'.repeat(empty)}</Text>\n <Text color={PALETTE.text}>{margin}{valueStr}</Text>\n <Text color={PALETTE.muted}>{pctStr}</Text>\n </Box>\n );\n}\n"],"mappings":"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","names":["useEffect","useState","Box","Text","useInput","useStdout","jsx","jsxs","DAY_WINDOWS","StatsViewer","nav","useNav","useStrings","sessions","setSessions","useState","book","setBook","windowIdx","setWindowIdx","useEffect","s","b","loadSessions","loadMistakes","useInput","_input","key","i","Box","Text","PALETTE","days","streak","dailyStreak","totalWords","a","totalErrors","totalMs","firstTryWords","overallWpm","overallAcc","recent","top","topN","agg","windowAggregate","speedPct","accPctFrac","sessionsPctFrac","Stat","MetricBar","RecentRow","word","entry","MistakeRow","session","units","name","useDictName","display","truncateName","computeWPM","accuracy","count","dictIds","multiSuffix","firstId","firstName","suffix","label","value","accent","calcMetricBarLayout","cols","frac","valueText","pct","stdout","useStdout","layout","clamped","filled","empty","labelStr","pad","valueStr","pctStr","margin"]}
|
|
1
|
+
{"version":3,"sources":["../src/ui/screens/StatsViewer.tsx"],"sourcesContent":["import { useEffect, useState } from 'react';\nimport { Box, Text, useInput, useStdout } from 'ink';\nimport { useNav } from '../nav.js';\nimport { useStrings } from '../../i18n/context.js';\nimport { useDictName } from '../registry-context.js';\nimport { PALETTE } from '../components/BigWord.js';\nimport {\n loadSessions,\n computeWPM,\n accuracy,\n windowAggregate,\n dailyStreak,\n type SessionRecord,\n} from '../../domain/stats.js';\nimport { loadMistakes, topN, type MistakeBook } from '../../domain/mistakes.js';\nimport { truncateName } from '../../util/dict-name.js';\nimport { pad } from '../../util/text.js';\n\nconst DAY_WINDOWS = [7, 14, 30, 90];\n\nexport function StatsViewer() {\n const nav = useNav();\n const t = useStrings();\n const [sessions, setSessions] = useState<SessionRecord[] | null>(null);\n const [book, setBook] = useState<MistakeBook | null>(null);\n const [windowIdx, setWindowIdx] = useState(1);\n\n useEffect(() => {\n void (async () => {\n const [s, b] = await Promise.all([loadSessions(), loadMistakes()]);\n setSessions(s);\n setBook(b);\n })();\n }, []);\n\n useInput((_input, key) => {\n if (key.escape) {\n nav.back();\n return;\n }\n if (key.rightArrow) setWindowIdx((i) => (i + 1) % DAY_WINDOWS.length);\n if (key.leftArrow) setWindowIdx((i) => (i - 1 + DAY_WINDOWS.length) % DAY_WINDOWS.length);\n });\n\n if (!sessions || !book) {\n return (\n <Box alignItems=\"center\" justifyContent=\"center\" width=\"100%\" height=\"100%\">\n <Text color={PALETTE.muted}>{t.stats.loading}</Text>\n </Box>\n );\n }\n\n if (sessions.length === 0) {\n return (\n <Box flexDirection=\"column\" alignItems=\"center\" justifyContent=\"center\" width=\"100%\" height=\"100%\">\n <Text color={PALETTE.muted}>{t.stats.none}</Text>\n <Box marginTop={1}>\n <Text color={PALETTE.muted}>{t.stats.nonePractice}</Text>\n </Box>\n <Box marginTop={2}>\n <Text color={PALETTE.muted}>Esc {t.common.back}</Text>\n </Box>\n </Box>\n );\n }\n\n const days = DAY_WINDOWS[windowIdx]!;\n const streak = dailyStreak(sessions);\n const totalWords = sessions.reduce((a, s) => a + s.wordCount, 0);\n const totalErrors = sessions.reduce((a, s) => a + s.errors, 0);\n const totalMs = sessions.reduce((a, s) => a + s.durationMs, 0);\n const firstTryWords = sessions.reduce(\n (a, s) => a + (s.wordCount - Object.keys(s.perWordErrors).length),\n 0,\n );\n const overallWpm = totalMs > 0 ? Math.round((totalWords / (totalMs / 60000)) * 10) / 10 : 0;\n const overallAcc = totalWords === 0 ? 1 : firstTryWords / totalWords;\n\n const recent = sessions.slice(-5).reverse();\n const top = topN(book, 8);\n\n const agg = windowAggregate(sessions, days);\n const speedPct = agg.peakWpmLifetime > 0 ? Math.min(1, agg.avgWpm / agg.peakWpmLifetime) : 0;\n const accPctFrac = Math.min(1, agg.avgAccPct / 100);\n const sessionsPctFrac = days > 0 ? Math.min(1, agg.sessionsInWindow / days) : 0;\n\n return (\n <Box flexDirection=\"column\" paddingX={2} paddingY={1} width=\"100%\" height=\"100%\">\n <Text bold color={PALETTE.accent}>\n {t.stats.title}\n </Text>\n\n <Box marginTop={1} flexDirection=\"column\">\n <Text color={PALETTE.muted}>{t.stats.lifetime}</Text>\n <Box marginTop={1}>\n <Stat label={t.stats.sessions} value={String(sessions.length)} />\n <Stat label={t.stats.words} value={String(totalWords)} />\n <Stat label={t.stats.errors} value={String(totalErrors)} />\n <Stat label={t.stats.wpm} value={String(overallWpm)} accent />\n <Stat label={t.stats.accuracy} value={`${Math.round(overallAcc * 1000) / 10}%`} accent />\n <Stat label={t.stats.streak} value={`${streak}d`} accent />\n </Box>\n </Box>\n\n <Box marginTop={2} flexDirection=\"column\">\n <Text color={PALETTE.muted}>{t.stats.last(days)}</Text>\n <Box marginTop={1} flexDirection=\"column\" borderStyle=\"round\" borderColor={PALETTE.muted} paddingX={2} paddingY={0}>\n <MetricBar\n label={t.stats.bars.speed}\n frac={speedPct}\n valueText={`${agg.avgWpm} wpm`}\n pct={Math.round(speedPct * 100)}\n />\n <MetricBar\n label={t.stats.bars.accuracy}\n frac={accPctFrac}\n valueText={`${agg.avgAccPct}%`}\n pct={Math.round(accPctFrac * 100)}\n />\n <MetricBar\n label={t.stats.bars.sessions}\n frac={sessionsPctFrac}\n valueText={`${agg.sessionsInWindow} / ${days}`}\n pct={Math.round(sessionsPctFrac * 100)}\n />\n </Box>\n </Box>\n\n <Box marginTop={2} flexDirection=\"column\">\n <Text color={PALETTE.muted}>{t.stats.recent}</Text>\n {recent.map((s, i) => (\n <RecentRow key={i} session={s} units={t.stats.recentUnits} />\n ))}\n </Box>\n\n {top.length > 0 && (\n <Box marginTop={2} flexDirection=\"column\">\n <Text color={PALETTE.muted}>{t.stats.topMistakes}</Text>\n {top.map(([word, entry]) => (\n <MistakeRow\n key={word}\n word={word}\n count={entry.count}\n dictIds={entry.dictIds}\n multiSuffix={t.stats.multiDictSuffix}\n />\n ))}\n </Box>\n )}\n\n <Box flexGrow={1} />\n <Box marginTop={1}>\n <Text color={PALETTE.muted}>{t.stats.footer}</Text>\n </Box>\n </Box>\n );\n}\n\nfunction RecentRow({\n session,\n units,\n}: {\n session: SessionRecord;\n units: { words: string; errors: string; wpm: string };\n}) {\n const name = useDictName(session.dictId);\n const display = truncateName(name, 14);\n return (\n <Box>\n <Text color={PALETTE.muted}> {session.ts.replace('T', ' ').slice(0, 16)} </Text>\n <Text color={PALETTE.text}>{display.padEnd(14)}</Text>\n <Text color={PALETTE.muted}>\n {' '}ch{String(session.chapter + 1).padStart(3)} {session.mode.padEnd(9)} {String(session.wordCount).padStart(3)}{units.words} {session.errors}{units.errors} {computeWPM(session)}{units.wpm} {Math.round(accuracy(session) * 1000) / 10}%\n </Text>\n </Box>\n );\n}\n\nfunction MistakeRow({\n word,\n count,\n dictIds,\n multiSuffix,\n}: {\n word: string;\n count: number;\n dictIds: string[];\n multiSuffix: (n: number) => string;\n}) {\n const firstId = dictIds[0] ?? '';\n const firstName = useDictName(firstId);\n const suffix = dictIds.length > 1 ? multiSuffix(dictIds.length - 1) : '';\n return (\n <Box>\n <Text color={PALETTE.error}> {String(count).padStart(3)} </Text>\n <Text color={PALETTE.text}>{word.padEnd(20)}</Text>\n <Text color={PALETTE.muted}>\n {truncateName(firstName, 20)}\n {suffix}\n </Text>\n </Box>\n );\n}\n\nfunction Stat({ label, value, accent = false }: { label: string; value: string; accent?: boolean }) {\n return (\n <Box marginRight={3}>\n <Text color={PALETTE.muted}>{label} </Text>\n <Text bold color={accent ? PALETTE.accent : PALETTE.text}>\n {value}\n </Text>\n </Box>\n );\n}\n\n// Pure layout calculation — extracted for testing without an Ink renderer.\n// OVERHEAD = outer paddingX(4) + inner border(2) + inner paddingX(4) + 1 col slack = 11.\n// The +1 slack guards against off-by-one in Ink/terminal cell rounding —\n// content was previously sized exactly to the inner width and any rounding\n// would wrap the bar onto a second line.\n// Fixed widths sum to 24; bar takes whatever cols remain, clamped 0..60.\nexport function calcMetricBarLayout(cols: number): {\n barWidth: number;\n labelWidth: number;\n valueWidth: number;\n pctWidth: number;\n marginLeft: number;\n} {\n const OVERHEAD = 11;\n const labelWidth = 9;\n const valueWidth = 8;\n const pctWidth = 5;\n const marginLeft = 2;\n const fixed = labelWidth + marginLeft + valueWidth + pctWidth;\n const barWidth = Math.max(0, Math.min(60, cols - OVERHEAD - fixed));\n return { barWidth, labelWidth, valueWidth, pctWidth, marginLeft };\n}\n\n// MetricBar uses pure inline <Text> siblings with CJK-aware pad() padding\n// rather than nested <Box width=N> wrappers. Mixed Box+Text flex children\n// can wrap unpredictably at terminal-width boundaries; inline text widths\n// are deterministic.\nfunction MetricBar({\n label,\n frac,\n valueText,\n pct,\n}: {\n label: string;\n frac: number;\n valueText: string;\n pct: number;\n}) {\n const { stdout } = useStdout();\n const cols = stdout?.columns ?? 80;\n const layout = calcMetricBarLayout(cols);\n const clamped = Math.max(0, Math.min(1, frac));\n const filled = Math.round(layout.barWidth * clamped);\n const empty = layout.barWidth - filled;\n const labelStr = pad(label, layout.labelWidth);\n const valueStr = pad(valueText, layout.valueWidth);\n const pctStr = String(pct).padStart(layout.pctWidth - 1, ' ') + '%';\n const margin = ' '.repeat(layout.marginLeft);\n return (\n <Box>\n <Text color={PALETTE.muted}>{labelStr}</Text>\n <Text color={PALETTE.accent}>{'━'.repeat(filled)}</Text>\n <Text color={PALETTE.muted}>{'─'.repeat(empty)}</Text>\n <Text color={PALETTE.text}>{margin}{valueStr}</Text>\n <Text color={PALETTE.muted}>{pctStr}</Text>\n </Box>\n );\n}\n"],"mappings":"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","names":["useEffect","useState","Box","Text","useInput","useStdout","jsx","jsxs","DAY_WINDOWS","StatsViewer","nav","useNav","useStrings","sessions","setSessions","useState","book","setBook","windowIdx","setWindowIdx","useEffect","s","b","loadSessions","loadMistakes","useInput","_input","key","i","Box","Text","PALETTE","days","streak","dailyStreak","totalWords","a","totalErrors","totalMs","firstTryWords","overallWpm","overallAcc","recent","top","topN","agg","windowAggregate","speedPct","accPctFrac","sessionsPctFrac","Stat","MetricBar","RecentRow","word","entry","MistakeRow","session","units","name","useDictName","display","truncateName","computeWPM","accuracy","count","dictIds","multiSuffix","firstId","firstName","suffix","label","value","accent","calcMetricBarLayout","cols","frac","valueText","pct","stdout","useStdout","layout","clamped","filled","empty","labelStr","pad","valueStr","pctStr","margin"]}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{c as b}from"./chunk-DURXS5MX.js";import"./chunk-7LTZGB7F.js";import{a as M}from"./chunk-G3DQB7FI.js";import{b as B,e as h}from"./chunk-QG7ZTS2G.js";import{b as L,d as g,f as t}from"./chunk-
|
|
2
|
-
//# sourceMappingURL=WordLookup-
|
|
1
|
+
import{c as b}from"./chunk-DURXS5MX.js";import"./chunk-7LTZGB7F.js";import{a as M}from"./chunk-G3DQB7FI.js";import{b as B,e as h}from"./chunk-QG7ZTS2G.js";import{b as L,d as g,f as t}from"./chunk-FQ2MEK7M.js";import"./chunk-NA5UNUVL.js";import{a as I}from"./chunk-E6BBQALJ.js";import{useEffect as E,useState as u}from"react";import{Box as n,Text as r,useInput as S}from"ink";import{readdir as $}from"fs/promises";import{Fragment as G,jsx as o,jsxs as d}from"react/jsx-runtime";async function N(){try{return(await $(I.dictsDir)).filter(e=>e.endsWith(".json")&&!e.endsWith(".meta.json")).map(e=>e.replace(/\.json$/,""))}catch{return[]}}function O(){let i=L(),e=g(),[m,x]=u(""),[l,w]=u([]),[W,C]=u({}),[H,j]=u(!0),[k,f]=u(0);E(()=>{(async()=>{let a=await N(),c=[];for(let s of a){let y=await b(s);if(y)for(let A of y)c.push({dictId:s,word:A})}w(c),C(await M()),j(!1)})()},[]);let T=m.toLowerCase().trim(),p=T?l.filter(a=>a.word.name.toLowerCase().includes(T)).slice(0,50):[];if(S((a,c)=>{if(c.escape){i.back();return}if(c.upArrow){f(s=>Math.max(0,s-1));return}if(c.downArrow){f(s=>Math.min(p.length-1,s+1));return}if(c.backspace||c.delete){x(s=>s.slice(0,-1)),f(0);return}a&&!c.ctrl&&!c.meta&&a.trim().length>0&&(x(s=>s+a),f(0))}),H)return o(n,{alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:o(r,{color:t.muted,children:e.word.indexing})});if(l.length===0)return d(n,{flexDirection:"column",alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:[o(r,{color:t.muted,children:e.word.none}),o(n,{marginTop:1,children:o(r,{color:t.muted,children:e.word.pullFirst})}),o(n,{marginTop:2,children:d(r,{color:t.muted,children:["[Esc] ",e.common.back]})})]});let D=p[k];return d(n,{flexDirection:"column",paddingX:2,paddingY:1,width:"100%",height:"100%",children:[d(n,{children:[o(r,{bold:!0,color:t.accent,children:e.word.title}),o(n,{flexGrow:1}),o(r,{color:t.muted,children:e.word.countAcross(l.length)})]}),d(n,{marginTop:1,children:[o(r,{color:t.muted,children:"> "}),o(r,{color:t.text,children:m}),o(r,{color:t.accent,children:"_"})]}),d(n,{marginTop:1,flexGrow:1,children:[d(n,{flexDirection:"column",width:"40%",children:[p.map((a,c)=>o(q,{hit:a,active:c===k},`${a.dictId}-${a.word.name}-${c}`)),p.length===0&&T&&o(r,{color:t.muted,children:e.word.noMatches(m)})]}),o(n,{flexDirection:"column",width:"60%",paddingLeft:2,children:D&&o(v,{hit:D,book:W})})]}),o(n,{marginTop:1,children:o(r,{color:t.muted,children:e.word.footer})})]})}function q({hit:i,active:e}){let m=B(i.dictId);return d(n,{children:[o(r,{color:e?t.accent:t.muted,children:e?"\u258C ":" "}),o(r,{bold:e,color:e?t.text:t.muted,children:i.word.name.padEnd(20)}),o(r,{color:t.muted,children:h(m,18)})]})}function v({hit:i,book:e}){let m=g(),x=B(i.dictId);return d(G,{children:[o(r,{bold:!0,color:t.text,children:i.word.name}),d(n,{marginTop:1,children:[i.word.usphone&&d(r,{dimColor:!0,color:t.muted,children:["US ",i.word.usphone," "]}),i.word.ukphone&&d(r,{dimColor:!0,color:t.muted,children:["UK ",i.word.ukphone]})]}),o(n,{marginTop:1,flexDirection:"column",children:(i.word.trans??[]).map((l,w)=>d(r,{color:t.primary,children:["\xB7 ",l]},w))}),o(n,{marginTop:1,children:o(r,{color:t.muted,children:m.word.inDict(h(x,22))})}),e[i.word.name]&&o(n,{marginTop:1,children:o(r,{color:t.error,children:m.word.mistakes(e[i.word.name].count,e[i.word.name].lastSeen.slice(0,10))})})]})}export{O as WordLookup};
|
|
2
|
+
//# sourceMappingURL=WordLookup-LKBEPDTB.js.map
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
import{createContext as k,useContext as E,useState as v,useCallback as g}from"react";import{jsx as L}from"react/jsx-runtime";var h=k(null);function N({initial:t,children:e}){let[n,r]=v([t]),s=g(i=>{r(o=>[...o,i])},[]),a=g(i=>{r(o=>o.length===0?[i]:[...o.slice(0,-1),i])},[]),u=g(()=>{r(i=>i.length>1?i.slice(0,-1):i)},[]),c=g(i=>{r([i])},[]),l=n[n.length-1];return L(h.Provider,{value:{current:l,stack:n,navigate:s,replace:a,back:u,reset:c},children:e})}function D(){let t=E(h);if(!t)throw new Error("useNav must be used inside NavProvider");return t}import{createContext as x,useContext as S,useMemo as $}from"react";var d={app:{title:"qwerty",subtitle:"typing practice for the terminal"},common:{back:"back",quit:"quit",on:"on",off:"off",cancel:"cancel"},mainMenu:{items:{practiceLabel:"Practice",practiceHintWith:t=>`start ${t}`,practiceHintNone:"pick a dictionary",dictLabel:"Dictionaries",dictHint:"browse, pull, set default",wordLabel:"Word lookup",wordHint:"search local dicts",statsLabel:"Stats",statsHint:"history & trends",configLabel:"Config",configHint:"edit preferences",stealthLabel:"Stealth",stealthHint:"quiet practice mode",quitLabel:"Quit",quitHint:"Esc or Ctrl+C also exits"},hint:"\u2191/\u2193 navigate \xB7 Enter select \xB7 letters jump",helpHint:"? help"},dict:{title:"Dictionaries",loading:"loading dictionaries\u2026",entries:t=>`${t} entries`,filterPlaceholder:"type to filter",local:"local \u2713",notLocal:"not local",defaultMark:"default \u2605",tagsLabel:t=>`tags: ${t}`,wordsLabel:t=>`${t} words`,pulling:t=>`pulling ${t}\u2026`,removing:t=>`removing ${t}\u2026`,errorOn:(t,e)=>`error on ${t}: ${e}`,footer:"\u2191/\u2193 select \xB7 Enter actions \xB7 Ctrl+K more \xB7 Esc back",action:{title:"current dictionary",setDefault:"set as default",practice:"practice now",delete:"delete local"},command:{title:"more actions",pull:"pull selected",import:"import .json",refreshList:"update dictionary list"}},config:{title:"Config",fields:{defaultDict:"default dict",defaultMode:"default mode",accent:"accent",mirror:"dict mirror",chapterSize:"chapter size",autoplayPronunciation:"autoplay pronunciation",soundsMaster:"sounds master",soundsKeystroke:"sounds keystroke",soundsFeedback:"sounds feedback",soundsPronunciationRate:"pronunciation rate",soundsPronunciationSource:"pronunciation source",language:"language",stealth:"stealth mode"},enumValues:{stealth:{off:"off",menu:"show in menu",default:"default practice"},soundsPronunciationSource:{youdao:"Youdao",dictapi:"Wiktionary"},defaultMode:{order:"order",dictation:"dictation",review:"review",random:"random",loop:"loop"}},hints:{editing:"type to edit \xB7 Enter save \xB7 Esc cancel",bool:"space toggle \xB7 \u2191/\u2193 move \xB7 Esc back",enum:"\u2190/\u2192 cycle \xB7 \u2191/\u2193 move \xB7 Esc back",dictRef:"Enter pick dict \xB7 \u2191/\u2193 move \xB7 Esc back",stringOrInt:"Enter edit \xB7 \u2191/\u2193 move \xB7 Esc back"}},stats:{title:"Stats \xB7 overview",loading:"loading stats\u2026",none:"No practice history yet.",nonePractice:"Run a practice session first.",lifetime:"lifetime",sessions:"sessions",words:"words",errors:"errors",wpm:"wpm",accuracy:"accuracy",streak:"streak",last:t=>`last ${t} days (\u2190/\u2192 cycle window)`,cycleWindow:"\u2190/\u2192 cycle window \xB7 Esc back",recent:"recent sessions",topMistakes:"top mistakes",footer:"\u2190/\u2192 cycle window \xB7 Esc back",maxLabel:"max",recentUnits:{words:"w",errors:"err",wpm:"wpm"},multiDictSuffix:t=>` +${t} more`,bars:{speed:"speed",accuracy:"accuracy",sessions:"sessions"}},word:{title:"Word lookup",indexing:"indexing local dictionaries\u2026",none:"No local dictionaries.",pullFirst:"Pull one in Dictionaries first.",countAcross:t=>`${t} words across local dicts`,noMatches:t=>`no matches for "${t}"`,inDict:t=>`in: ${t}`,mistakes:(t,e)=>`mistakes: ${t} (last ${e})`,footer:"type to filter \xB7 \u2191/\u2193 select \xB7 Esc back"},practice:{loading:"loading\u2026",paused:"PAUSED",chapterComplete:"CHAPTER COMPLETE",chapterLabel:(t,e)=>`chapter ${t}/${e}`,reviewLabel:"review",statusBar:{mode:"mode",accent:"accent"},modes:{order:"order",dictation:"dictation",review:"review",random:"random",loop:"loop"},accents:{us:"us",uk:"uk"},statCards:{words:"words",errors:"errors",wpm:"wpm",accuracy:"accuracy",elapsed:t=>`elapsed ${t}`},pause:{title:"PAUSED",chapter:(t,e)=>`chapter ${t}/${e}`,progress:(t,e)=>`${t}/${e}`,hint:"Enter resume \xB7 Esc back to menu"},summary:{loopAgain:"again",nextChapter:"next chapter",reviewMistakes:"review mistakes",backMenu:"back to menu"},footers:{typing:"Ctrl+N skip \xB7 Esc pause \xB7 Tab replay"},errors:{noMistakes:"No mistakes to review yet. Practice some chapters first.",dictEmpty:t=>`Dictionary ${t} is empty.`,unknown:"Unknown error"},imeWarning:"! Non-English input detected \u2014 switch your IME to English.",imeWarningShort:"! IME",audioWarningShort:"! audio"},audio:{noPlayer:"! No audio player found on PATH (looked for afplay/ffplay/mpg123/paplay/aplay/powershell). Sounds disabled."},report:{title:"Session summary",duration:"duration",practiced:"practiced",chapters:"chapters",words:"words",accuracy:"accuracy",wpm:"wpm",newMistakes:"new mistakes",farewell:"see you next time.",notPracticed:"no practice this run"},help:{title:"Help",subtitle:"all shortcuts",sections:{main:"main menu",practice:"practice",dict:"dictionaries",config:"config",stats:"stats",word:"word lookup",global:"global"},keys:{navigate:"\u2191/\u2193 navigate items",select:"Enter confirm / continue",letterJump:"letter jump to menu item",pause:"Esc pause practice",skip:"Ctrl+N skip current word (neutral)",replay:"Tab replay pronunciation",resume:"Enter resume from pause",backMenu:"Esc back to previous screen",backScreen:"Esc close panel or back",nextChapter:"Enter next chapter",reviewMistakes:"m review mistakes",filter:"type to filter list",itemActions:"Enter open actions panel",moreActions:"Ctrl+K more actions panel",cycleWindow:"\u2190/\u2192 cycle day window",stealthToggle:"Ctrl+I toggle stealth info row",helpScreen:"? open this help screen",quit:"Ctrl+C quit immediately"},footer:"Esc back"},stealth:{paused:"paused",chapterDone:"chapter done",resumeHint:"Enter resume \xB7 Esc menu",nextHint:"Enter next \xB7 Esc menu",pausedHintRight:"Enter resume",nextHintRight:"Enter next",infoChipLabel:"info",infoFmt:(t,e,n,r,s,a)=>`${t} \xB7 ${e} \xB7 ${n}/${r} \xB7 ${s} wpm \xB7 ${a}%`}},p={app:{title:"qwerty",subtitle:"\u7EC8\u7AEF\u952E\u76D8\u7EC3\u4E60"},common:{back:"\u8FD4\u56DE",quit:"\u9000\u51FA",on:"\u5F00",off:"\u5173",cancel:"\u53D6\u6D88"},mainMenu:{items:{practiceLabel:"\u7EC3\u4E60",practiceHintWith:t=>`\u5F00\u59CB ${t}`,practiceHintNone:"\u8BF7\u5148\u9009\u8BCD\u5178",dictLabel:"\u8BCD\u5178",dictHint:"\u6D4F\u89C8\u3001\u4E0B\u8F7D\u3001\u8BBE\u4E3A\u9ED8\u8BA4",wordLabel:"\u67E5\u8BCD",wordHint:"\u5728\u672C\u5730\u8BCD\u5178\u4E2D\u641C\u7D22",statsLabel:"\u7EDF\u8BA1",statsHint:"\u5386\u53F2\u4E0E\u8D8B\u52BF",configLabel:"\u8BBE\u7F6E",configHint:"\u4FEE\u6539\u504F\u597D",stealthLabel:"\u795E\u9690",stealthHint:"\u795E\u9690\u7EC3\u4E60\u6A21\u5F0F",quitLabel:"\u9000\u51FA",quitHint:"Esc \u6216 Ctrl+C \u9000\u51FA"},hint:"\u2191/\u2193 \u79FB\u52A8 \xB7 Enter \u786E\u8BA4 \xB7 \u5B57\u6BCD\u76F4\u8FBE",helpHint:"? \u5E2E\u52A9"},dict:{title:"\u8BCD\u5178",loading:"\u52A0\u8F7D\u8BCD\u5178\u4E2D\u2026",entries:t=>`${t} \u90E8\u8BCD\u5178`,filterPlaceholder:"\u8F93\u5165\u8FC7\u6EE4",local:"\u5DF2\u4E0B\u8F7D \u2713",notLocal:"\u672A\u4E0B\u8F7D",defaultMark:"\u9ED8\u8BA4 \u2605",tagsLabel:t=>`\u6807\u7B7E:${t}`,wordsLabel:t=>`${t} \u8BCD`,pulling:t=>`\u62C9\u53D6 ${t} \u4E2D\u2026`,removing:t=>`\u5220\u9664 ${t} \u4E2D\u2026`,errorOn:(t,e)=>`${t} \u51FA\u9519:${e}`,footer:"\u2191/\u2193 \u9009\u62E9 \xB7 Enter \u64CD\u4F5C \xB7 Ctrl+K \u66F4\u591A \xB7 Esc \u8FD4\u56DE",action:{title:"\u5F53\u524D\u8BCD\u5178",setDefault:"\u8BBE\u4E3A\u9ED8\u8BA4",practice:"\u7ACB\u5373\u7EC3\u4E60",delete:"\u5220\u9664\u672C\u5730"},command:{title:"\u66F4\u591A\u529F\u80FD",pull:"\u62C9\u53D6\u9009\u4E2D",import:"\u5BFC\u5165 .json",refreshList:"\u66F4\u65B0\u8BCD\u5178\u5217\u8868"}},config:{title:"\u8BBE\u7F6E",fields:{defaultDict:"\u9ED8\u8BA4\u8BCD\u5178",defaultMode:"\u9ED8\u8BA4\u6A21\u5F0F",accent:"\u53D1\u97F3",mirror:"\u8BCD\u5178\u955C\u50CF\u6E90",chapterSize:"\u7AE0\u8282\u5355\u8BCD\u6570",autoplayPronunciation:"\u81EA\u52A8\u64AD\u653E\u53D1\u97F3",soundsMaster:"\u97F3\u6548\u603B\u5F00\u5173",soundsKeystroke:"\u6309\u952E\u97F3",soundsFeedback:"\u53CD\u9988\u97F3",soundsPronunciationRate:"\u53D1\u97F3\u901F\u7387",soundsPronunciationSource:"\u53D1\u97F3\u97F3\u6E90",language:"\u8BED\u8A00",stealth:"\u795E\u9690\u6A21\u5F0F"},enumValues:{stealth:{off:"\u5173\u95ED",menu:"\u4E3B\u83DC\u5355\u663E\u793A",default:"\u9ED8\u8BA4\u7EC3\u4E60\u6A21\u5F0F"},soundsPronunciationSource:{youdao:"\u6709\u9053\u8BCD\u5178",dictapi:"\u7EF4\u57FA\u8BCD\u5178"},defaultMode:{order:"\u987A\u5E8F",dictation:"\u9ED8\u5199",review:"\u590D\u4E60",random:"\u4E71\u5E8F",loop:"\u5FAA\u73AF"}},hints:{editing:"\u8F93\u5165\u4FEE\u6539 \xB7 Enter \u4FDD\u5B58 \xB7 Esc \u53D6\u6D88",bool:"\u7A7A\u683C\u5207\u6362 \xB7 \u2191/\u2193 \u79FB\u52A8 \xB7 Esc \u8FD4\u56DE",enum:"\u2190/\u2192 \u5207\u6362 \xB7 \u2191/\u2193 \u79FB\u52A8 \xB7 Esc \u8FD4\u56DE",dictRef:"Enter \u9009\u8BCD\u5178 \xB7 \u2191/\u2193 \u79FB\u52A8 \xB7 Esc \u8FD4\u56DE",stringOrInt:"Enter \u7F16\u8F91 \xB7 \u2191/\u2193 \u79FB\u52A8 \xB7 Esc \u8FD4\u56DE"}},stats:{title:"\u7EDF\u8BA1 \xB7 \u6982\u89C8",loading:"\u52A0\u8F7D\u7EDF\u8BA1\u4E2D\u2026",none:"\u8FD8\u6CA1\u6709\u7EC3\u4E60\u8BB0\u5F55\u3002",nonePractice:"\u5148\u6765\u4E00\u6B21\u7EC3\u4E60\u5427\u3002",lifetime:"\u7D2F\u8BA1",sessions:"\u4F1A\u8BDD",words:"\u8BCD\u6570",errors:"\u9519\u8BEF",wpm:"\u901F\u5EA6",accuracy:"\u51C6\u786E\u7387",streak:"\u8FDE\u7EED\u5929\u6570",last:t=>`\u6700\u8FD1 ${t} \u5929 (\u2190/\u2192 \u5207\u6362\u7A97\u53E3)`,cycleWindow:"\u2190/\u2192 \u5207\u6362\u7A97\u53E3 \xB7 Esc \u8FD4\u56DE",recent:"\u6700\u8FD1\u4F1A\u8BDD",topMistakes:"\u9AD8\u9891\u9519\u8BCD",footer:"\u2190/\u2192 \u5207\u6362\u7A97\u53E3 \xB7 Esc \u8FD4\u56DE",maxLabel:"\u6700\u5927",recentUnits:{words:"\u8BCD",errors:"\u9519",wpm:"\u901F"},multiDictSuffix:t=>` \u7B49 ${t} \u90E8`,bars:{speed:"\u901F\u5EA6",accuracy:"\u51C6\u786E\u7387",sessions:"\u4F1A\u8BDD"}},word:{title:"\u67E5\u8BCD",indexing:"\u7D22\u5F15\u672C\u5730\u8BCD\u5178\u4E2D\u2026",none:"\u6CA1\u6709\u672C\u5730\u8BCD\u5178\u3002",pullFirst:"\u5148\u5728\u300C\u8BCD\u5178\u300D\u4E2D\u62C9\u53D6\u4E00\u90E8\u3002",countAcross:t=>`\u672C\u5730\u8BCD\u5178\u5171 ${t} \u8BCD`,noMatches:t=>`\u6CA1\u6709\u5339\u914D\u300C${t}\u300D\u7684\u8BCD`,inDict:t=>`\u6765\u6E90:${t}`,mistakes:(t,e)=>`\u9519\u8FC7 ${t} \u6B21 (\u6700\u8FD1 ${e})`,footer:"\u8F93\u5165\u8FC7\u6EE4 \xB7 \u2191/\u2193 \u9009\u62E9 \xB7 Esc \u8FD4\u56DE"},practice:{loading:"\u52A0\u8F7D\u4E2D\u2026",paused:"\u5DF2\u6682\u505C",chapterComplete:"\u672C\u7AE0\u5B8C\u6210",chapterLabel:(t,e)=>`\u7B2C ${t}/${e} \u7AE0`,reviewLabel:"\u590D\u4E60",statusBar:{mode:"\u6A21\u5F0F",accent:"\u53D1\u97F3"},modes:{order:"\u987A\u5E8F",dictation:"\u9ED8\u5199",review:"\u590D\u4E60",random:"\u4E71\u5E8F",loop:"\u5FAA\u73AF"},accents:{us:"\u7F8E",uk:"\u82F1"},statCards:{words:"\u8BCD\u6570",errors:"\u9519\u8BEF",wpm:"\u901F\u5EA6",accuracy:"\u51C6\u786E\u7387",elapsed:t=>`\u8017\u65F6 ${t}`},pause:{title:"\u5DF2\u6682\u505C",chapter:(t,e)=>`\u7B2C ${t}/${e} \u7AE0`,progress:(t,e)=>`${t}/${e}`,hint:"Enter \u7EE7\u7EED \xB7 Esc \u8FD4\u56DE\u83DC\u5355"},summary:{loopAgain:"\u518D\u6765\u4E00\u904D",nextChapter:"\u4E0B\u4E00\u7AE0",reviewMistakes:"\u590D\u4E60\u9519\u8BCD",backMenu:"\u8FD4\u56DE\u83DC\u5355"},footers:{typing:"Ctrl+N \u8DF3\u8FC7 \xB7 Esc \u6682\u505C \xB7 Tab \u91CD\u64AD"},errors:{noMistakes:"\u9519\u8BCD\u672C\u662F\u7A7A\u7684\u3002\u5148\u7EC3\u4E60\u51E0\u7AE0\u5427\u3002",dictEmpty:t=>`\u8BCD\u5178 ${t} \u662F\u7A7A\u7684\u3002`,unknown:"\u672A\u77E5\u9519\u8BEF"},imeWarning:"! \u68C0\u6D4B\u5230\u4E2D\u6587/\u65E5\u6587/\u97E9\u6587\u8F93\u5165\uFF0C\u8BF7\u5207\u6362\u5230\u82F1\u6587\u8F93\u5165\u6CD5",imeWarningShort:"! \u4E2D\u6587\u8F93\u5165",audioWarningShort:"! \u65E0\u97F3\u6548"},audio:{noPlayer:"! \u672A\u5728 PATH \u4E2D\u627E\u5230\u97F3\u9891\u64AD\u653E\u5668(\u5C1D\u8BD5 afplay/ffplay/mpg123/paplay/aplay/powershell)\u3002\u97F3\u6548\u5DF2\u7981\u7528\u3002"},report:{title:"\u672C\u6B21\u4F1A\u8BDD",duration:"\u603B\u65F6\u957F",practiced:"\u7EC3\u4E60\u7528\u65F6",chapters:"\u5B8C\u6210\u7AE0\u8282",words:"\u8BCD\u6570",accuracy:"\u51C6\u786E\u7387",wpm:"\u901F\u5EA6",newMistakes:"\u65B0\u9519\u8BCD",farewell:"\u4E0B\u6B21\u89C1\u3002",notPracticed:"\u672C\u6B21\u672A\u7EC3\u4E60"},help:{title:"\u5E2E\u52A9",subtitle:"\u5168\u90E8\u5FEB\u6377\u952E",sections:{main:"\u4E3B\u83DC\u5355",practice:"\u7EC3\u4E60",dict:"\u8BCD\u5178",config:"\u8BBE\u7F6E",stats:"\u7EDF\u8BA1",word:"\u67E5\u8BCD",global:"\u5168\u5C40"},keys:{navigate:"\u2191/\u2193 \u79FB\u52A8\u9009\u9879",select:"Enter \u786E\u8BA4 / \u7EE7\u7EED",letterJump:"\u5B57\u6BCD\u952E \u76F4\u8FBE\u83DC\u5355\u9879",pause:"Esc \u6682\u505C\u7EC3\u4E60",skip:"Ctrl+N \u8DF3\u8FC7\u5F53\u524D\u8BCD(\u4E0D\u8BA1\u9519)",replay:"Tab \u91CD\u64AD\u53D1\u97F3",resume:"Enter \u7EE7\u7EED\u7EC3\u4E60",backMenu:"Esc \u8FD4\u56DE\u4E0A\u4E00\u5C4F",backScreen:"Esc \u5173\u95ED\u9762\u677F / \u8FD4\u56DE",nextChapter:"Enter \u4E0B\u4E00\u7AE0",reviewMistakes:"m \u590D\u4E60\u9519\u8BCD",filter:"\u8F93\u5165 \u8FC7\u6EE4\u5217\u8868",itemActions:"Enter \u5F39\u51FA\u52A8\u4F5C\u9762\u677F",moreActions:"Ctrl+K \u5F39\u51FA\u66F4\u591A\u529F\u80FD",cycleWindow:"\u2190/\u2192 \u5207\u6362\u65E5\u7A97\u53E3",stealthToggle:"Ctrl+I \u5207\u6362\u795E\u9690\u4FE1\u606F\u884C",helpScreen:"? \u6253\u5F00\u672C\u5E2E\u52A9\u9875",quit:"Ctrl+C \u7ACB\u5373\u9000\u51FA"},footer:"Esc \u8FD4\u56DE"},stealth:{paused:"paused",chapterDone:"chapter done",resumeHint:"Enter resume \xB7 Esc menu",nextHint:"Enter next \xB7 Esc menu",pausedHintRight:"Enter \u7EE7\u7EED",nextHintRight:"Enter \u4E0B\u4E00\u7AE0",infoChipLabel:"\u4FE1\u606F",infoFmt:(t,e,n,r,s,a)=>`${t} \xB7 ${e} \xB7 ${n}/${r} \xB7 ${s} wpm \xB7 ${a}%`}};function b(t){if(!t)return null;let e=t.toLowerCase();return e.startsWith("zh")?"zh":e.startsWith("en")?"en":null}function m(t){if(t==="zh"||t==="en")return t;let e=process.env.LC_ALL||process.env.LC_MESSAGES||process.env.LANG||process.env.LANGUAGE,n=b(e);if(n)return n;try{let r=Intl.DateTimeFormat().resolvedOptions().locale,s=b(r);if(s)return s}catch{}return"en"}import{jsx as P}from"react/jsx-runtime";var w=x(null);function I({pref:t,children:e}){let n=$(()=>{let r=m(t);return{lang:r,t:r==="zh"?p:d}},[t]);return P(w.Provider,{value:n,children:e})}function O(){let t=S(w);if(!t)throw new Error("useStrings must be used inside StringsProvider");return t.t}function K(t){let e=m(t);return{lang:e,t:e==="zh"?p:d}}import{Box as C,Text as M}from"ink";import{jsx as y}from"react/jsx-runtime";var f={accent:"#5eead4",muted:"#6b7280",text:"#e5e7eb",primary:"#7dcfff",success:"#86efac",warning:"#fbbf24",error:"#f87171"};function G({target:t,typed:e,error:n=!1,hideTarget:r=!1}){let s=[...t],a=[...e];return y(C,{paddingY:4,justifyContent:"center",children:s.map((u,c)=>{let l=c<a.length,i=r&&!l?"_":l?a[c]:u,o=n?f.error:l?f.accent:f.muted;return y(M,{bold:!0,color:o,children:i},c)})})}export{N as a,D as b,I as c,O as d,K as e,f,G as g};
|
|
2
|
+
//# sourceMappingURL=chunk-FQ2MEK7M.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/ui/nav.tsx","../src/i18n/context.tsx","../src/i18n/strings.ts","../src/i18n/locale.ts","../src/ui/components/BigWord.tsx"],"sourcesContent":["import { createContext, useContext, useState, useCallback, type ReactNode } from 'react';\n\nexport type ScreenName =\n | 'main'\n | 'practice'\n | 'dict'\n | 'config'\n | 'stats'\n | 'word'\n | 'help';\n\nexport type PracticeParams = {\n dictId: string;\n chapterIndex: number;\n mode: 'order' | 'dictation' | 'review' | 'random' | 'loop';\n stealth?: boolean;\n};\n\nexport type DictParams = {\n pickerMode?: 'set-default' | 'choose-then-practice';\n};\n\nexport type ScreenFrame =\n | { name: 'main' }\n | { name: 'practice'; params: PracticeParams }\n | { name: 'dict'; params?: DictParams }\n | { name: 'config' }\n | { name: 'stats' }\n | { name: 'word' }\n | { name: 'help' };\n\ntype NavContextValue = {\n current: ScreenFrame;\n stack: ScreenFrame[];\n navigate: (frame: ScreenFrame) => void;\n replace: (frame: ScreenFrame) => void;\n back: () => void;\n reset: (frame: ScreenFrame) => void;\n};\n\nconst NavContext = createContext<NavContextValue | null>(null);\n\nexport function NavProvider({ initial, children }: { initial: ScreenFrame; children: ReactNode }) {\n const [stack, setStack] = useState<ScreenFrame[]>([initial]);\n\n const navigate = useCallback((frame: ScreenFrame) => {\n setStack((s) => [...s, frame]);\n }, []);\n const replace = useCallback((frame: ScreenFrame) => {\n setStack((s) => (s.length === 0 ? [frame] : [...s.slice(0, -1), frame]));\n }, []);\n const back = useCallback(() => {\n setStack((s) => (s.length > 1 ? s.slice(0, -1) : s));\n }, []);\n const reset = useCallback((frame: ScreenFrame) => {\n setStack([frame]);\n }, []);\n\n const current = stack[stack.length - 1]!;\n return (\n <NavContext.Provider value={{ current, stack, navigate, replace, back, reset }}>\n {children}\n </NavContext.Provider>\n );\n}\n\nexport function useNav(): NavContextValue {\n const ctx = useContext(NavContext);\n if (!ctx) throw new Error('useNav must be used inside NavProvider');\n return ctx;\n}\n","import { createContext, useContext, useMemo, type ReactNode } from 'react';\nimport { en, zh, type Strings } from './strings.js';\nimport { detectLocale, type Lang, type LangPref } from './locale.js';\n\nconst StringsContext = createContext<{ lang: Lang; t: Strings } | null>(null);\n\nexport function StringsProvider({\n pref,\n children,\n}: {\n pref: LangPref;\n children: ReactNode;\n}) {\n const value = useMemo(() => {\n const lang = detectLocale(pref);\n return { lang, t: lang === 'zh' ? zh : en };\n }, [pref]);\n return <StringsContext.Provider value={value}>{children}</StringsContext.Provider>;\n}\n\nexport function useStrings(): Strings {\n const ctx = useContext(StringsContext);\n if (!ctx) throw new Error('useStrings must be used inside StringsProvider');\n return ctx.t;\n}\n\nexport function useLang(): Lang {\n const ctx = useContext(StringsContext);\n if (!ctx) throw new Error('useLang must be used inside StringsProvider');\n return ctx.lang;\n}\n\nexport function pickStrings(pref: LangPref): { lang: Lang; t: Strings } {\n const lang = detectLocale(pref);\n return { lang, t: lang === 'zh' ? zh : en };\n}\n","// Centralized string table for the qwerty-cli TUI.\n//\n// Print-only subcommands (qwerty config list, qwerty dict list, etc.) keep\n// English output regardless of locale so shell scripts stay predictable.\n// Only the interactive TUI is localized via this table.\n\nexport type Strings = {\n app: {\n title: string;\n subtitle: string;\n };\n common: {\n back: string;\n quit: string;\n on: string;\n off: string;\n cancel: string;\n };\n mainMenu: {\n items: {\n practiceLabel: string;\n practiceHintWith: (name: string) => string;\n practiceHintNone: string;\n dictLabel: string;\n dictHint: string;\n wordLabel: string;\n wordHint: string;\n statsLabel: string;\n statsHint: string;\n configLabel: string;\n configHint: string;\n stealthLabel: string;\n stealthHint: string;\n quitLabel: string;\n quitHint: string;\n };\n hint: string;\n helpHint: string;\n };\n dict: {\n title: string;\n loading: string;\n entries: (n: number) => string;\n filterPlaceholder: string;\n local: string;\n notLocal: string;\n defaultMark: string;\n tagsLabel: (tags: string) => string;\n wordsLabel: (n: number) => string;\n pulling: (id: string) => string;\n removing: (id: string) => string;\n errorOn: (id: string, msg: string) => string;\n footer: string;\n action: {\n title: string;\n setDefault: string;\n practice: string;\n delete: string;\n };\n command: {\n title: string;\n pull: string;\n import: string;\n refreshList: string;\n };\n };\n config: {\n title: string;\n fields: {\n defaultDict: string;\n defaultMode: string;\n accent: string;\n mirror: string;\n chapterSize: string;\n autoplayPronunciation: string;\n soundsMaster: string;\n soundsKeystroke: string;\n soundsFeedback: string;\n soundsPronunciationRate: string;\n soundsPronunciationSource: string;\n language: string;\n stealth: string;\n };\n enumValues: {\n stealth: { off: string; menu: string; default: string };\n soundsPronunciationSource: { youdao: string; dictapi: string };\n defaultMode: { order: string; dictation: string; review: string; random: string; loop: string };\n };\n hints: {\n editing: string;\n bool: string;\n enum: string;\n dictRef: string;\n stringOrInt: string;\n };\n };\n stats: {\n title: string;\n loading: string;\n none: string;\n nonePractice: string;\n lifetime: string;\n sessions: string;\n words: string;\n errors: string;\n wpm: string;\n accuracy: string;\n streak: string;\n last: (n: number) => string;\n cycleWindow: string;\n recent: string;\n topMistakes: string;\n footer: string;\n maxLabel: string;\n recentUnits: { words: string; errors: string; wpm: string };\n multiDictSuffix: (n: number) => string;\n bars: { speed: string; accuracy: string; sessions: string };\n };\n word: {\n title: string;\n indexing: string;\n none: string;\n pullFirst: string;\n countAcross: (n: number) => string;\n noMatches: (q: string) => string;\n inDict: (name: string) => string;\n mistakes: (n: number, date: string) => string;\n footer: string;\n };\n practice: {\n loading: string;\n paused: string;\n chapterComplete: string;\n chapterLabel: (c: number, t: number) => string;\n reviewLabel: string;\n statusBar: {\n mode: string;\n accent: string;\n };\n modes: {\n order: string;\n dictation: string;\n review: string;\n random: string;\n loop: string;\n };\n accents: {\n us: string;\n uk: string;\n };\n statCards: {\n words: string;\n errors: string;\n wpm: string;\n accuracy: string;\n elapsed: (t: string) => string;\n };\n pause: {\n title: string;\n chapter: (c: number, t: number) => string;\n progress: (completed: number, total: number) => string;\n hint: string;\n };\n summary: {\n loopAgain: string;\n nextChapter: string;\n reviewMistakes: string;\n backMenu: string;\n };\n footers: {\n typing: string;\n };\n errors: {\n noMistakes: string;\n dictEmpty: (id: string) => string;\n unknown: string;\n };\n imeWarning: string;\n imeWarningShort: string;\n audioWarningShort: string;\n };\n audio: {\n noPlayer: string;\n };\n report: {\n title: string;\n duration: string;\n practiced: string;\n chapters: string;\n words: string;\n accuracy: string;\n wpm: string;\n newMistakes: string;\n farewell: string;\n notPracticed: string;\n };\n help: {\n title: string;\n subtitle: string;\n sections: {\n main: string;\n practice: string;\n dict: string;\n config: string;\n stats: string;\n word: string;\n global: string;\n };\n keys: {\n navigate: string;\n select: string;\n letterJump: string;\n pause: string;\n skip: string;\n replay: string;\n resume: string;\n backMenu: string;\n backScreen: string;\n nextChapter: string;\n reviewMistakes: string;\n filter: string;\n itemActions: string;\n moreActions: string;\n cycleWindow: string;\n stealthToggle: string;\n helpScreen: string;\n quit: string;\n };\n footer: string;\n };\n stealth: {\n paused: string;\n chapterDone: string;\n resumeHint: string;\n nextHint: string;\n pausedHintRight: string;\n nextHintRight: string;\n infoChipLabel: string;\n infoFmt: (\n dict: string,\n chapter: string,\n completed: number,\n total: number,\n wpm: number,\n accPct: number,\n ) => string;\n };\n};\n\nexport const en: Strings = {\n app: {\n title: 'qwerty',\n subtitle: 'typing practice for the terminal',\n },\n common: {\n back: 'back',\n quit: 'quit',\n on: 'on',\n off: 'off',\n cancel: 'cancel',\n },\n mainMenu: {\n items: {\n practiceLabel: 'Practice',\n practiceHintWith: (name) => `start ${name}`,\n practiceHintNone: 'pick a dictionary',\n dictLabel: 'Dictionaries',\n dictHint: 'browse, pull, set default',\n wordLabel: 'Word lookup',\n wordHint: 'search local dicts',\n statsLabel: 'Stats',\n statsHint: 'history & trends',\n configLabel: 'Config',\n configHint: 'edit preferences',\n stealthLabel: 'Stealth',\n stealthHint: 'quiet practice mode',\n quitLabel: 'Quit',\n quitHint: 'Esc or Ctrl+C also exits',\n },\n hint: '↑/↓ navigate · Enter select · letters jump',\n helpHint: '? help',\n },\n dict: {\n title: 'Dictionaries',\n loading: 'loading dictionaries…',\n entries: (n) => `${n} entries`,\n filterPlaceholder: 'type to filter',\n local: 'local ✓',\n notLocal: 'not local',\n defaultMark: 'default ★',\n tagsLabel: (tags) => `tags: ${tags}`,\n wordsLabel: (n) => `${n} words`,\n pulling: (id) => `pulling ${id}…`,\n removing: (id) => `removing ${id}…`,\n errorOn: (id, msg) => `error on ${id}: ${msg}`,\n footer: '↑/↓ select · Enter actions · Ctrl+K more · Esc back',\n action: {\n title: 'current dictionary',\n setDefault: 'set as default',\n practice: 'practice now',\n delete: 'delete local',\n },\n command: {\n title: 'more actions',\n pull: 'pull selected',\n import: 'import .json',\n refreshList: 'update dictionary list',\n },\n },\n config: {\n title: 'Config',\n fields: {\n defaultDict: 'default dict',\n defaultMode: 'default mode',\n accent: 'accent',\n mirror: 'dict mirror',\n chapterSize: 'chapter size',\n autoplayPronunciation: 'autoplay pronunciation',\n soundsMaster: 'sounds master',\n soundsKeystroke: 'sounds keystroke',\n soundsFeedback: 'sounds feedback',\n soundsPronunciationRate: 'pronunciation rate',\n soundsPronunciationSource: 'pronunciation source',\n language: 'language',\n stealth: 'stealth mode',\n },\n enumValues: {\n stealth: { off: 'off', menu: 'show in menu', default: 'default practice' },\n soundsPronunciationSource: { youdao: 'Youdao', dictapi: 'Wiktionary' },\n defaultMode: { order: 'order', dictation: 'dictation', review: 'review', random: 'random', loop: 'loop' },\n },\n hints: {\n editing: 'type to edit · Enter save · Esc cancel',\n bool: 'space toggle · ↑/↓ move · Esc back',\n enum: '←/→ cycle · ↑/↓ move · Esc back',\n dictRef: 'Enter pick dict · ↑/↓ move · Esc back',\n stringOrInt: 'Enter edit · ↑/↓ move · Esc back',\n },\n },\n stats: {\n title: 'Stats · overview',\n loading: 'loading stats…',\n none: 'No practice history yet.',\n nonePractice: 'Run a practice session first.',\n lifetime: 'lifetime',\n sessions: 'sessions',\n words: 'words',\n errors: 'errors',\n wpm: 'wpm',\n accuracy: 'accuracy',\n streak: 'streak',\n last: (n) => `last ${n} days (←/→ cycle window)`,\n cycleWindow: '←/→ cycle window · Esc back',\n recent: 'recent sessions',\n topMistakes: 'top mistakes',\n footer: '←/→ cycle window · Esc back',\n maxLabel: 'max',\n recentUnits: { words: 'w', errors: 'err', wpm: 'wpm' },\n multiDictSuffix: (n) => ` +${n} more`,\n bars: { speed: 'speed', accuracy: 'accuracy', sessions: 'sessions' },\n },\n word: {\n title: 'Word lookup',\n indexing: 'indexing local dictionaries…',\n none: 'No local dictionaries.',\n pullFirst: 'Pull one in Dictionaries first.',\n countAcross: (n) => `${n} words across local dicts`,\n noMatches: (q) => `no matches for \"${q}\"`,\n inDict: (name) => `in: ${name}`,\n mistakes: (n, date) => `mistakes: ${n} (last ${date})`,\n footer: 'type to filter · ↑/↓ select · Esc back',\n },\n practice: {\n loading: 'loading…',\n paused: 'PAUSED',\n chapterComplete: 'CHAPTER COMPLETE',\n chapterLabel: (c, t) => `chapter ${c}/${t}`,\n reviewLabel: 'review',\n statusBar: {\n mode: 'mode',\n accent: 'accent',\n },\n modes: {\n order: 'order',\n dictation: 'dictation',\n review: 'review',\n random: 'random',\n loop: 'loop',\n },\n accents: {\n us: 'us',\n uk: 'uk',\n },\n statCards: {\n words: 'words',\n errors: 'errors',\n wpm: 'wpm',\n accuracy: 'accuracy',\n elapsed: (t) => `elapsed ${t}`,\n },\n pause: {\n title: 'PAUSED',\n chapter: (c, t) => `chapter ${c}/${t}`,\n progress: (completed, total) => `${completed}/${total}`,\n hint: 'Enter resume · Esc back to menu',\n },\n summary: {\n loopAgain: 'again',\n nextChapter: 'next chapter',\n reviewMistakes: 'review mistakes',\n backMenu: 'back to menu',\n },\n footers: {\n typing: 'Ctrl+N skip · Esc pause · Tab replay',\n },\n errors: {\n noMistakes: 'No mistakes to review yet. Practice some chapters first.',\n dictEmpty: (id) => `Dictionary ${id} is empty.`,\n unknown: 'Unknown error',\n },\n imeWarning: '! Non-English input detected — switch your IME to English.',\n imeWarningShort: '! IME',\n audioWarningShort: '! audio',\n },\n audio: {\n noPlayer: '! No audio player found on PATH (looked for afplay/ffplay/mpg123/paplay/aplay/powershell). Sounds disabled.',\n },\n report: {\n title: 'Session summary',\n duration: 'duration',\n practiced: 'practiced',\n chapters: 'chapters',\n words: 'words',\n accuracy: 'accuracy',\n wpm: 'wpm',\n newMistakes: 'new mistakes',\n farewell: 'see you next time.',\n notPracticed: 'no practice this run',\n },\n help: {\n title: 'Help',\n subtitle: 'all shortcuts',\n sections: {\n main: 'main menu',\n practice: 'practice',\n dict: 'dictionaries',\n config: 'config',\n stats: 'stats',\n word: 'word lookup',\n global: 'global',\n },\n keys: {\n navigate: '↑/↓ navigate items',\n select: 'Enter confirm / continue',\n letterJump: 'letter jump to menu item',\n pause: 'Esc pause practice',\n skip: 'Ctrl+N skip current word (neutral)',\n replay: 'Tab replay pronunciation',\n resume: 'Enter resume from pause',\n backMenu: 'Esc back to previous screen',\n backScreen: 'Esc close panel or back',\n nextChapter: 'Enter next chapter',\n reviewMistakes: 'm review mistakes',\n filter: 'type to filter list',\n itemActions: 'Enter open actions panel',\n moreActions: 'Ctrl+K more actions panel',\n cycleWindow: '←/→ cycle day window',\n stealthToggle: 'Ctrl+I toggle stealth info row',\n helpScreen: '? open this help screen',\n quit: 'Ctrl+C quit immediately',\n },\n footer: 'Esc back',\n },\n stealth: {\n paused: 'paused',\n chapterDone: 'chapter done',\n resumeHint: 'Enter resume · Esc menu',\n nextHint: 'Enter next · Esc menu',\n pausedHintRight: 'Enter resume',\n nextHintRight: 'Enter next',\n infoChipLabel: 'info',\n infoFmt: (dict, chapter, completed, total, wpm, accPct) =>\n `${dict} · ${chapter} · ${completed}/${total} · ${wpm} wpm · ${accPct}%`,\n },\n};\n\nexport const zh: Strings = {\n app: {\n title: 'qwerty',\n subtitle: '终端键盘练习',\n },\n common: {\n back: '返回',\n quit: '退出',\n on: '开',\n off: '关',\n cancel: '取消',\n },\n mainMenu: {\n items: {\n practiceLabel: '练习',\n practiceHintWith: (name) => `开始 ${name}`,\n practiceHintNone: '请先选词典',\n dictLabel: '词典',\n dictHint: '浏览、下载、设为默认',\n wordLabel: '查词',\n wordHint: '在本地词典中搜索',\n statsLabel: '统计',\n statsHint: '历史与趋势',\n configLabel: '设置',\n configHint: '修改偏好',\n stealthLabel: '神隐',\n stealthHint: '神隐练习模式',\n quitLabel: '退出',\n quitHint: 'Esc 或 Ctrl+C 退出',\n },\n hint: '↑/↓ 移动 · Enter 确认 · 字母直达',\n helpHint: '? 帮助',\n },\n dict: {\n title: '词典',\n loading: '加载词典中…',\n entries: (n) => `${n} 部词典`,\n filterPlaceholder: '输入过滤',\n local: '已下载 ✓',\n notLocal: '未下载',\n defaultMark: '默认 ★',\n tagsLabel: (tags) => `标签:${tags}`,\n wordsLabel: (n) => `${n} 词`,\n pulling: (id) => `拉取 ${id} 中…`,\n removing: (id) => `删除 ${id} 中…`,\n errorOn: (id, msg) => `${id} 出错:${msg}`,\n footer: '↑/↓ 选择 · Enter 操作 · Ctrl+K 更多 · Esc 返回',\n action: {\n title: '当前词典',\n setDefault: '设为默认',\n practice: '立即练习',\n delete: '删除本地',\n },\n command: {\n title: '更多功能',\n pull: '拉取选中',\n import: '导入 .json',\n refreshList: '更新词典列表',\n },\n },\n config: {\n title: '设置',\n fields: {\n defaultDict: '默认词典',\n defaultMode: '默认模式',\n accent: '发音',\n mirror: '词典镜像源',\n chapterSize: '章节单词数',\n autoplayPronunciation: '自动播放发音',\n soundsMaster: '音效总开关',\n soundsKeystroke: '按键音',\n soundsFeedback: '反馈音',\n soundsPronunciationRate: '发音速率',\n soundsPronunciationSource: '发音音源',\n language: '语言',\n stealth: '神隐模式',\n },\n enumValues: {\n stealth: { off: '关闭', menu: '主菜单显示', default: '默认练习模式' },\n soundsPronunciationSource: { youdao: '有道词典', dictapi: '维基词典' },\n defaultMode: { order: '顺序', dictation: '默写', review: '复习', random: '乱序', loop: '循环' },\n },\n hints: {\n editing: '输入修改 · Enter 保存 · Esc 取消',\n bool: '空格切换 · ↑/↓ 移动 · Esc 返回',\n enum: '←/→ 切换 · ↑/↓ 移动 · Esc 返回',\n dictRef: 'Enter 选词典 · ↑/↓ 移动 · Esc 返回',\n stringOrInt: 'Enter 编辑 · ↑/↓ 移动 · Esc 返回',\n },\n },\n stats: {\n title: '统计 · 概览',\n loading: '加载统计中…',\n none: '还没有练习记录。',\n nonePractice: '先来一次练习吧。',\n lifetime: '累计',\n sessions: '会话',\n words: '词数',\n errors: '错误',\n wpm: '速度',\n accuracy: '准确率',\n streak: '连续天数',\n last: (n) => `最近 ${n} 天 (←/→ 切换窗口)`,\n cycleWindow: '←/→ 切换窗口 · Esc 返回',\n recent: '最近会话',\n topMistakes: '高频错词',\n footer: '←/→ 切换窗口 · Esc 返回',\n maxLabel: '最大',\n recentUnits: { words: '词', errors: '错', wpm: '速' },\n multiDictSuffix: (n) => ` 等 ${n} 部`,\n bars: { speed: '速度', accuracy: '准确率', sessions: '会话' },\n },\n word: {\n title: '查词',\n indexing: '索引本地词典中…',\n none: '没有本地词典。',\n pullFirst: '先在「词典」中拉取一部。',\n countAcross: (n) => `本地词典共 ${n} 词`,\n noMatches: (q) => `没有匹配「${q}」的词`,\n inDict: (name) => `来源:${name}`,\n mistakes: (n, date) => `错过 ${n} 次 (最近 ${date})`,\n footer: '输入过滤 · ↑/↓ 选择 · Esc 返回',\n },\n practice: {\n loading: '加载中…',\n paused: '已暂停',\n chapterComplete: '本章完成',\n chapterLabel: (c, t) => `第 ${c}/${t} 章`,\n reviewLabel: '复习',\n statusBar: {\n mode: '模式',\n accent: '发音',\n },\n modes: {\n order: '顺序',\n dictation: '默写',\n review: '复习',\n random: '乱序',\n loop: '循环',\n },\n accents: {\n us: '美',\n uk: '英',\n },\n statCards: {\n words: '词数',\n errors: '错误',\n wpm: '速度',\n accuracy: '准确率',\n elapsed: (t) => `耗时 ${t}`,\n },\n pause: {\n title: '已暂停',\n chapter: (c, t) => `第 ${c}/${t} 章`,\n progress: (completed, total) => `${completed}/${total}`,\n hint: 'Enter 继续 · Esc 返回菜单',\n },\n summary: {\n loopAgain: '再来一遍',\n nextChapter: '下一章',\n reviewMistakes: '复习错词',\n backMenu: '返回菜单',\n },\n footers: {\n typing: 'Ctrl+N 跳过 · Esc 暂停 · Tab 重播',\n },\n errors: {\n noMistakes: '错词本是空的。先练习几章吧。',\n dictEmpty: (id) => `词典 ${id} 是空的。`,\n unknown: '未知错误',\n },\n imeWarning: '! 检测到中文/日文/韩文输入,请切换到英文输入法',\n imeWarningShort: '! 中文输入',\n audioWarningShort: '! 无音效',\n },\n audio: {\n noPlayer: '! 未在 PATH 中找到音频播放器(尝试 afplay/ffplay/mpg123/paplay/aplay/powershell)。音效已禁用。',\n },\n report: {\n title: '本次会话',\n duration: '总时长',\n practiced: '练习用时',\n chapters: '完成章节',\n words: '词数',\n accuracy: '准确率',\n wpm: '速度',\n newMistakes: '新错词',\n farewell: '下次见。',\n notPracticed: '本次未练习',\n },\n help: {\n title: '帮助',\n subtitle: '全部快捷键',\n sections: {\n main: '主菜单',\n practice: '练习',\n dict: '词典',\n config: '设置',\n stats: '统计',\n word: '查词',\n global: '全局',\n },\n keys: {\n navigate: '↑/↓ 移动选项',\n select: 'Enter 确认 / 继续',\n letterJump: '字母键 直达菜单项',\n pause: 'Esc 暂停练习',\n skip: 'Ctrl+N 跳过当前词(不计错)',\n replay: 'Tab 重播发音',\n resume: 'Enter 继续练习',\n backMenu: 'Esc 返回上一屏',\n backScreen: 'Esc 关闭面板 / 返回',\n nextChapter: 'Enter 下一章',\n reviewMistakes: 'm 复习错词',\n filter: '输入 过滤列表',\n itemActions: 'Enter 弹出动作面板',\n moreActions: 'Ctrl+K 弹出更多功能',\n cycleWindow: '←/→ 切换日窗口',\n stealthToggle: 'Ctrl+I 切换神隐信息行',\n helpScreen: '? 打开本帮助页',\n quit: 'Ctrl+C 立即退出',\n },\n footer: 'Esc 返回',\n },\n stealth: {\n paused: 'paused',\n chapterDone: 'chapter done',\n resumeHint: 'Enter resume · Esc menu',\n nextHint: 'Enter next · Esc menu',\n pausedHintRight: 'Enter 继续',\n nextHintRight: 'Enter 下一章',\n infoChipLabel: '信息',\n infoFmt: (dict, chapter, completed, total, wpm, accPct) =>\n `${dict} · ${chapter} · ${completed}/${total} · ${wpm} wpm · ${accPct}%`,\n },\n};\n","export type Lang = 'zh' | 'en';\nexport type LangPref = 'auto' | Lang;\n\nfunction pickFromString(s: string | undefined): Lang | null {\n if (!s) return null;\n const lower = s.toLowerCase();\n if (lower.startsWith('zh')) return 'zh';\n if (lower.startsWith('en')) return 'en';\n return null;\n}\n\nexport function detectLocale(pref: LangPref): Lang {\n if (pref === 'zh' || pref === 'en') return pref;\n const env =\n process.env.LC_ALL ||\n process.env.LC_MESSAGES ||\n process.env.LANG ||\n process.env.LANGUAGE;\n const fromEnv = pickFromString(env);\n if (fromEnv) return fromEnv;\n try {\n const intlLocale = Intl.DateTimeFormat().resolvedOptions().locale;\n const fromIntl = pickFromString(intlLocale);\n if (fromIntl) return fromIntl;\n } catch {\n // ignore\n }\n return 'en';\n}\n","import { Box, Text } from 'ink';\n\nexport const PALETTE = {\n accent: '#5eead4',\n muted: '#6b7280',\n text: '#e5e7eb',\n primary: '#7dcfff',\n success: '#86efac',\n warning: '#fbbf24',\n error: '#f87171',\n} as const;\n\ntype Props = {\n target: string;\n typed: string;\n error?: boolean;\n hideTarget?: boolean;\n};\n\nexport function BigWord({ target, typed, error = false, hideTarget = false }: Props) {\n const chars = [...target];\n const typedChars = [...typed];\n\n return (\n <Box paddingY={4} justifyContent=\"center\">\n {chars.map((ch, i) => {\n const isTyped = i < typedChars.length;\n const display = hideTarget && !isTyped ? '_' : isTyped ? typedChars[i]! : ch;\n const color = error\n ? PALETTE.error\n : isTyped\n ? PALETTE.accent\n : PALETTE.muted;\n return (\n <Text key={i} bold color={color}>\n {display}\n </Text>\n );\n })}\n </Box>\n );\n}\n"],"mappings":"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","names":["createContext","useContext","useState","useCallback","jsx","NavContext","NavProvider","initial","children","stack","setStack","navigate","frame","s","replace","back","reset","current","useNav","ctx","createContext","useContext","useMemo","en","name","n","tags","id","msg","q","date","c","t","completed","total","dict","chapter","wpm","accPct","zh","pickFromString","s","lower","detectLocale","pref","env","fromEnv","intlLocale","fromIntl","jsx","StringsContext","createContext","StringsProvider","pref","children","value","useMemo","lang","detectLocale","zh","en","useStrings","ctx","useContext","pickStrings","pref","lang","detectLocale","zh","en","Box","Text","jsx","PALETTE","BigWord","target","typed","error","hideTarget","chars","typedChars","ch","i","isTyped","display","color"]}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
import{a as l,g as h,h as S}from"./chunk-E6BBQALJ.js";import{z as c}from"zod";var x=c.object({ts:c.string(),dictId:c.string(),chapter:c.number().int().nonnegative(),mode:c.string(),wordCount:c.number().int().nonnegative(),errors:c.number().int().nonnegative(),durationMs:c.number().int().nonnegative(),perWordErrors:c.record(c.string(),c.number().int().nonnegative()).default({})});async function k(e){await h(l.stats,e)}async function C(){return(await S(l.stats)).map(t=>x.safeParse(t)).filter(t=>t.success).map(t=>t.data)}function b(e){if(e.durationMs===0)return 0;let t=e.durationMs/6e4;return Math.round(e.wordCount/t*10)/10}function y(e){if(e.wordCount===0)return 1;let t=Object.keys(e.perWordErrors).length;return Math.max(0,Math.min(1,(e.wordCount-t)/e.wordCount))}function R(e,t){let o=null;for(let r of e)r.dictId===t&&(o=r.chapter);return o===null?0:o+1}function v(e,t=new Date){if(e.length===0)return 0;let o=new Set;for(let n of e)o.add(n.ts.slice(0,10));let r=0,i=new Date(t);for(;;){let n=i.toISOString().slice(0,10);if(!o.has(n))break;r++,i.setUTCDate(i.getUTCDate()-1)}return r}var w=["\u2581","\u2582","\u2583","\u2584","\u2585","\u2586","\u2587","\u2588"];function I(e){if(e.length===0)return"";let t=Math.max(...e,1),o=Math.min(...e,0),r=Math.max(1,t-o);return e.map(i=>{let n=Math.floor((i-o)/r*(w.length-1));return w[Math.max(0,Math.min(w.length-1,n))]}).join("")}function P(e,t,o=new Date){if(e.length===0||t<=0)return{avgWpm:0,peakWpmLifetime:0,avgAccPct:0,sessionsInWindow:0,days:t};let r=new Date(o);r.setUTCDate(r.getUTCDate()-(t-1));let i=r.toISOString().slice(0,10),n=0,m=0,u=0,s=0,p=0;for(let a of e){let f=b(a);f>p&&(p=f),!(a.ts.slice(0,10)<i)&&(s++,n+=a.wordCount,m+=f*a.wordCount,u+=y(a)*a.wordCount)}if(n===0)return{avgWpm:0,peakWpmLifetime:p,avgAccPct:0,sessionsInWindow:s,days:t};let g=Math.round(m/n*10)/10,d=Math.round(u/n*1e3)/10;return{avgWpm:g,peakWpmLifetime:p,avgAccPct:d,sessionsInWindow:s,days:t}}function A(e,t,o=new Date){let r=[],i=new Map;for(let m of e){let u=m.ts.slice(0,10),s=i.get(u)??[];s.push(m),i.set(u,s)}let n=new Date(o);n.setUTCDate(n.getUTCDate()-(t-1));for(let m=0;m<t;m++){let u=n.toISOString().slice(0,10),s=i.get(u)??[];if(s.length===0)r.push({date:u,wpm:0,accuracy:0,sessions:0});else{let p=s.reduce((d,a)=>d+b(a),0)/s.length,g=s.reduce((d,a)=>d+y(a),0)/s.length;r.push({date:u,wpm:p,accuracy:g,sessions:s.length})}n.setUTCDate(n.getUTCDate()+1)}return r}export{k as a,C as b,b as c,y as d,R as e,v as f,I as g,P as h,A as i};
|
|
2
|
+
//# sourceMappingURL=chunk-GULN5HRV.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/domain/stats.ts"],"sourcesContent":["import { z } from 'zod';\nimport { paths } from '../infra/paths.js';\nimport { appendJsonl, readJsonl } from '../infra/fs-store.js';\n\nexport const SessionRecordSchema = z.object({\n ts: z.string(),\n dictId: z.string(),\n chapter: z.number().int().nonnegative(),\n mode: z.string(),\n wordCount: z.number().int().nonnegative(),\n errors: z.number().int().nonnegative(),\n durationMs: z.number().int().nonnegative(),\n perWordErrors: z.record(z.string(), z.number().int().nonnegative()).default({}),\n});\n\nexport type SessionRecord = z.infer<typeof SessionRecordSchema>;\n\nexport async function appendSession(record: SessionRecord): Promise<void> {\n await appendJsonl(paths.stats, record);\n}\n\nexport async function loadSessions(): Promise<SessionRecord[]> {\n const rows = await readJsonl<unknown>(paths.stats);\n return rows\n .map((r) => SessionRecordSchema.safeParse(r))\n .filter((r): r is { success: true; data: SessionRecord } => r.success)\n .map((r) => r.data);\n}\n\nexport function computeWPM(record: SessionRecord): number {\n if (record.durationMs === 0) return 0;\n const minutes = record.durationMs / 60000;\n return Math.round((record.wordCount / minutes) * 10) / 10;\n}\n\nexport function accuracy(record: SessionRecord): number {\n // Word-level: fraction of words finished without any mistake (first-try rate).\n if (record.wordCount === 0) return 1;\n const wordsWithErrors = Object.keys(record.perWordErrors).length;\n return Math.max(0, Math.min(1, (record.wordCount - wordsWithErrors) / record.wordCount));\n}\n\n// Resume target for a fresh entry into practice for dictId.\n// stats.jsonl is append-only in time order, so the last matching row is the\n// most recently practiced chapter; advance one to land on the next.\n// Returns 0 if the user has never practiced this dict.\n// PracticeScreen clamps to chapters.length - 1, so overshoot at the end of a\n// dict is harmless.\nexport function resumeChapterFor(sessions: SessionRecord[], dictId: string): number {\n let last: number | null = null;\n for (const s of sessions) if (s.dictId === dictId) last = s.chapter;\n return last === null ? 0 : last + 1;\n}\n\nexport function dailyStreak(sessions: SessionRecord[], now = new Date()): number {\n if (sessions.length === 0) return 0;\n const days = new Set<string>();\n for (const s of sessions) days.add(s.ts.slice(0, 10));\n let streak = 0;\n const cur = new Date(now);\n while (true) {\n const key = cur.toISOString().slice(0, 10);\n if (!days.has(key)) break;\n streak++;\n cur.setUTCDate(cur.getUTCDate() - 1);\n }\n return streak;\n}\n\nconst SPARK = ['▁', '▂', '▃', '▄', '▅', '▆', '▇', '█'];\n\nexport function sparkline(values: number[]): string {\n if (values.length === 0) return '';\n const max = Math.max(...values, 1);\n const min = Math.min(...values, 0);\n const range = Math.max(1, max - min);\n return values\n .map((v) => {\n const idx = Math.floor(((v - min) / range) * (SPARK.length - 1));\n return SPARK[Math.max(0, Math.min(SPARK.length - 1, idx))]!;\n })\n .join('');\n}\n\nexport type DailyMetric = 'wpm' | 'accuracy' | 'sessions';\n\nexport function dailyValues(\n sessions: SessionRecord[],\n days: number,\n metric: DailyMetric,\n now = new Date(),\n): number[] {\n const buckets = dailyBuckets(sessions, days, now);\n if (metric === 'wpm') return buckets.map((b) => b.wpm);\n if (metric === 'accuracy') return buckets.map((b) => b.accuracy * 100);\n return buckets.map((b) => b.sessions);\n}\n\nexport type WindowAggregate = {\n avgWpm: number;\n peakWpmLifetime: number;\n avgAccPct: number;\n sessionsInWindow: number;\n days: number;\n};\n\nexport function windowAggregate(\n sessions: SessionRecord[],\n days: number,\n now = new Date(),\n): WindowAggregate {\n if (sessions.length === 0 || days <= 0) {\n return { avgWpm: 0, peakWpmLifetime: 0, avgAccPct: 0, sessionsInWindow: 0, days };\n }\n const cutoff = new Date(now);\n cutoff.setUTCDate(cutoff.getUTCDate() - (days - 1));\n const cutoffKey = cutoff.toISOString().slice(0, 10);\n\n let wordsInWindow = 0;\n let wpmWeightedSum = 0;\n let accWeightedSum = 0;\n let sessionsInWindow = 0;\n let peakWpmLifetime = 0;\n for (const s of sessions) {\n const wpm = computeWPM(s);\n if (wpm > peakWpmLifetime) peakWpmLifetime = wpm;\n if (s.ts.slice(0, 10) < cutoffKey) continue;\n sessionsInWindow++;\n wordsInWindow += s.wordCount;\n wpmWeightedSum += wpm * s.wordCount;\n accWeightedSum += accuracy(s) * s.wordCount;\n }\n if (wordsInWindow === 0) {\n return { avgWpm: 0, peakWpmLifetime, avgAccPct: 0, sessionsInWindow, days };\n }\n const avgWpm = Math.round((wpmWeightedSum / wordsInWindow) * 10) / 10;\n const avgAccPct = Math.round((accWeightedSum / wordsInWindow) * 1000) / 10;\n return { avgWpm, peakWpmLifetime, avgAccPct, sessionsInWindow, days };\n}\n\nexport type DailyBucket = { date: string; wpm: number; accuracy: number; sessions: number };\n\nexport function dailyBuckets(sessions: SessionRecord[], days: number, now = new Date()): DailyBucket[] {\n const out: DailyBucket[] = [];\n const byDay = new Map<string, SessionRecord[]>();\n for (const s of sessions) {\n const key = s.ts.slice(0, 10);\n const arr = byDay.get(key) ?? [];\n arr.push(s);\n byDay.set(key, arr);\n }\n const cur = new Date(now);\n cur.setUTCDate(cur.getUTCDate() - (days - 1));\n for (let i = 0; i < days; i++) {\n const key = cur.toISOString().slice(0, 10);\n const todays = byDay.get(key) ?? [];\n if (todays.length === 0) {\n out.push({ date: key, wpm: 0, accuracy: 0, sessions: 0 });\n } else {\n const wpm = todays.reduce((a, s) => a + computeWPM(s), 0) / todays.length;\n const acc = todays.reduce((a, s) => a + accuracy(s), 0) / todays.length;\n out.push({ date: key, wpm, accuracy: acc, sessions: todays.length });\n }\n cur.setUTCDate(cur.getUTCDate() + 1);\n }\n return out;\n}\n"],"mappings":"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","names":["z","SessionRecordSchema","z","appendSession","record","appendJsonl","paths","loadSessions","readJsonl","r","computeWPM","minutes","accuracy","wordsWithErrors","resumeChapterFor","sessions","dictId","last","s","dailyStreak","now","days","streak","cur","key","SPARK","sparkline","values","max","min","range","v","idx","windowAggregate","sessions","days","now","cutoff","cutoffKey","wordsInWindow","wpmWeightedSum","accWeightedSum","sessionsInWindow","peakWpmLifetime","s","wpm","computeWPM","accuracy","avgWpm","avgAccPct","dailyBuckets","out","byDay","key","arr","cur","i","todays","a","acc"]}
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
import{a as L,b as R,c as I}from"./chunk-6ROGUGNX.js";import{a as C,b as k}from"./chunk-CQRKGMPU.js";import{b as $,e as z}from"./chunk-GULN5HRV.js";import{a as B,b as D,c as w,e as H}from"./chunk-QG7ZTS2G.js";import{a as P,b as M,c as E,d as N,f as a}from"./chunk-FQ2MEK7M.js";var _="\x1B[?1049h\x1B[?25l\x1B[2J\x1B[H",F="\x1B[?25h\x1B[?1049l",v=!1,W=!1;function O(){if(W)return;W=!0;let e=()=>{if(v){try{process.stdout.write(F)}catch{}v=!1}};process.once("exit",e),process.once("SIGINT",()=>{e(),process.exit(130)}),process.once("SIGTERM",()=>{e(),process.exit(143)})}function Y(){v||process.stdout.isTTY&&process.env.QWERTY_NO_ALTSCREEN!=="1"&&(O(),process.stdout.write(_),v=!0)}function A(){if(v){try{process.stdout.write(F)}catch{}v=!1}}import{Suspense as ne,lazy as T,useRef as ie}from"react";import{Box as se,Text as ae,useApp as ce,useInput as ue}from"ink";import{useEffect as Q,useState as X}from"react";import{Box as U,useStdout as Z}from"ink";import{jsx as ee}from"react/jsx-runtime";function j({children:e}){let{stdout:t}=Z(),[s,r]=X(()=>({rows:t?.rows??24,cols:t?.columns??80}));return Q(()=>{Y();let n=()=>{r({rows:process.stdout.rows??24,cols:process.stdout.columns??80})};return process.stdout.on("resize",n),()=>{process.stdout.off("resize",n),A()}},[]),ee(U,{width:s.cols,height:s.rows,flexDirection:"column",children:e})}import{useState as te}from"react";import{Box as y,Text as f,useApp as re,useInput as oe}from"ink";import{jsx as g,jsxs as x}from"react/jsx-runtime";function q({cfg:e}){let[t,s]=te(0),{exit:r}=re(),n=M(),h=R(),p=N(),l=D(e.defaultDict),o=p.mainMenu.items,S=async c=>{if(!e.defaultDict){n.navigate({name:"dict",params:{pickerMode:"choose-then-practice"}});return}let m=z(await $(),e.defaultDict);if(c){I({type:"stealth",dictId:e.defaultDict,chapterIndex:m,mode:e.defaultMode}),r();return}n.navigate({name:"practice",params:{dictId:e.defaultDict,chapterIndex:m,mode:e.defaultMode,stealth:c}})},d=[{key:"p",label:o.practiceLabel,hint:e.defaultDict?o.practiceHintWith(H(l,24)):o.practiceHintNone,run:()=>{S(e.stealth==="default")}}];(e.stealth==="menu"||e.stealth==="default")&&d.push({key:"b",label:o.stealthLabel,hint:o.stealthHint,run:()=>{S(!0)}}),d.push({key:"d",label:o.dictLabel,hint:o.dictHint,run:()=>n.navigate({name:"dict"})},{key:"w",label:o.wordLabel,hint:o.wordHint,run:()=>n.navigate({name:"word"})},{key:"s",label:o.statsLabel,hint:o.statsHint,run:()=>n.navigate({name:"stats"})},{key:"c",label:o.configLabel,hint:o.configHint,run:()=>n.navigate({name:"config"})},{key:"q",label:o.quitLabel,hint:o.quitHint,run:()=>r()});let J=Math.max(...d.map(c=>w(c.label)))+4;return oe((c,m)=>{if(m.escape){r();return}if(m.upArrow&&s(u=>(u-1+d.length)%d.length),m.downArrow&&s(u=>(u+1)%d.length),m.return){d[t].run();return}if(c==="?"){n.navigate({name:"help"});return}for(let u of d)if(c===u.key){u.run();return}}),x(y,{flexDirection:"column",paddingX:2,paddingY:1,width:"100%",children:[e.stealth!=="default"&&x(y,{children:[g(f,{bold:!0,color:a.accent,children:p.app.title}),x(f,{color:a.muted,children:[" \xB7 ",p.app.subtitle]})]}),g(y,{marginTop:e.stealth==="default"?0:2,flexDirection:"column",children:d.map((c,m)=>{let u=m===t,K=" ".repeat(Math.max(0,J-w(c.label)));return x(y,{children:[g(f,{color:u?a.accent:a.muted,children:u?"\u258C ":" "}),x(f,{color:u?a.accent:a.muted,children:["[",c.key,"]"]}),g(f,{children:" "}),x(f,{bold:u,color:u?a.text:a.muted,children:[c.label,K]}),g(f,{color:a.muted,children:c.hint})]},c.key)})}),g(y,{marginTop:2,children:x(f,{color:a.muted,children:[p.mainMenu.hint," \xB7 ",p.mainMenu.helpHint]})}),h.warning&&g(y,{marginTop:1,children:g(f,{color:a.warning,children:p.audio.noPlayer})})]})}import{jsx as i}from"react/jsx-runtime";var pe=T(()=>import("./PracticeScreen-PWXXG33U.js").then(e=>({default:e.PracticeScreen}))),le=T(()=>import("./DictBrowser-ZQ773KQV.js").then(e=>({default:e.DictBrowser}))),de=T(()=>import("./ConfigEditor-ZJ52VG3I.js").then(e=>({default:e.ConfigEditor}))),me=T(()=>import("./StatsViewer-XARAMVZW.js").then(e=>({default:e.StatsViewer}))),fe=T(()=>import("./WordLookup-LKBEPDTB.js").then(e=>({default:e.WordLookup}))),he=T(()=>import("./HelpScreen-Y3TWEEWZ.js").then(e=>({default:e.HelpScreen})));function ge(){return i(se,{alignItems:"center",justifyContent:"center",width:"100%",height:"100%",children:i(ae,{color:a.muted,children:"\u2026"})})}function Xe({initial:e,initialCfg:t,inline:s=!1}){return i(C,{initialCfg:t,children:i(xe,{children:i(B,{children:i(L,{disabled:!t.sounds.master,children:i(P,{initial:e,children:s?i(V,{inline:!0}):i(j,{children:i(V,{})})})})})})})}function xe({children:e}){let{cfg:t}=k();return i(E,{pref:t.language,children:e})}function be(e){if(e.name==="practice"){let t=e.params;return`practice:${t.dictId}:${t.chapterIndex}:${t.mode}:${t.stealth?"s":"n"}`}return e.name}function V({inline:e=!1}){let t=M(),{cfg:s}=k(),{exit:r}=ce(),n=ie(null);ue((o,S)=>{S.ctrl&&o==="c"&&r()});let h=t.current,p=be(h);n.current!==p&&(!e&&process.stdout.isTTY&&process.stdout.write("\x1B[2J\x1B[H"),n.current=p);let l=(()=>{switch(h.name){case"main":return i(q,{cfg:s});case"practice":return i(pe,{params:h.params});case"dict":return i(le,{params:h.params});case"config":return i(de,{});case"stats":return i(me,{});case"word":return i(fe,{});case"help":return i(he,{})}})();return i(ne,{fallback:i(ge,{}),children:l})}import b from"chalk";import Se from"boxen";function nt(){A()}function G(e,t){let s=Math.floor(e/1e3),r=Math.floor(s/60),n=s%60;return t==="zh"?r===0?`${n} \u79D2`:`${r} \u5206 ${n} \u79D2`:r===0?`${n}s`:`${r}m ${n}s`}function we(e){return e>=90?b.green:e<75?b.dim:t=>t}function ve(e,t,s){let r=[],n=[t.report.duration];e.chaptersCompleted===0?n.push(t.report.notPracticed):(n.push(t.report.practiced,t.report.chapters,t.report.words,t.report.accuracy,t.report.wpm),e.newMistakeWords>0&&n.push(t.report.newMistakes));let h=Math.max(...n.map(w)),p=o=>o+" ".repeat(Math.max(0,h-w(o))),l=(o,S)=>`${b.dim(p(o))} ${S}`;if(r.push(l(t.report.duration,G(e.totalDurationMs,s))),e.chaptersCompleted===0)r.push(b.dim(t.report.notPracticed));else{r.push(l(t.report.practiced,G(e.practiceMs,s))),r.push(l(t.report.chapters,String(e.chaptersCompleted))),r.push(l(t.report.words,String(e.wordCount)));let o=Math.round(e.accuracy*1e3)/10;r.push(l(t.report.accuracy,we(o)(`${o}%`))),r.push(l(t.report.wpm,String(e.wpm))),e.newMistakeWords>0&&r.push(l(t.report.newMistakes,b.red(String(e.newMistakeWords))))}return r.push(""),r.push(b.dim.italic(t.report.farewell)),r}function it(e,t,s){if(e.startedAt===null&&e.chaptersCompleted===0)return;let r=ve(e,t,s).join(`
|
|
2
|
+
`);console.log(Se(r,{title:b.bold.cyan(t.report.title),titleAlignment:"left",borderStyle:"round",borderColor:"gray",padding:{top:0,bottom:0,left:3,right:3},margin:{top:1,bottom:1,left:2,right:0}}))}export{Y as a,Xe as b,nt as c,it as d};
|
|
3
|
+
//# sourceMappingURL=chunk-TCYEMBFW.js.map
|