Compare commits
15 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 3ce15ed794 | |||
| bd310d8305 | |||
| a22897e19e | |||
| 6ec3e66e9c | |||
| defde986a4 | |||
| bb3752b8ae | |||
| be04f333b0 | |||
| fc25246594 | |||
| d7e104c17a | |||
| c197445f35 | |||
| 57ae54d716 | |||
| 82a3717689 | |||
| d4c93af2c2 | |||
| b4dd706bff | |||
| 797682eae7 |
23
LICENSE
Normal file
23
LICENSE
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2026 gkucmierz
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
|
||||||
|
See README.md for project description.
|
||||||
10
README.md
10
README.md
@@ -1,3 +1,11 @@
|
|||||||
# Nonograms
|
# Nonograms
|
||||||
|
|
||||||
Link do aplikacji: https://nonograms.7u.pl
|
## English Description
|
||||||
|
|
||||||
|
Nonograms is a modern, fast, and accessible logic puzzle game (also known as Picross or Griddlers). Solve pixel-art puzzles by marking cells according to numeric clues for rows and columns. The app features:
|
||||||
|
- Clean UX with keyboard and touch support
|
||||||
|
- Multiple languages and PWA support (installable on desktop and mobile)
|
||||||
|
- Difficulty simulation and guide to learn solving strategies
|
||||||
|
- Shareable puzzles and persistent progress
|
||||||
|
|
||||||
|
Play online at https://nonograms.7u.pl or install as a PWA for an app-like experience.
|
||||||
|
|||||||
28
check_i18n.cjs
Normal file
28
check_i18n.cjs
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
const fs = require('fs');
|
||||||
|
|
||||||
|
const fileContent = fs.readFileSync('src/composables/useI18n.js', 'utf8');
|
||||||
|
|
||||||
|
const match = fileContent.match(/const messages = ({[\s\S]*?});/);
|
||||||
|
if (!match) {
|
||||||
|
console.error('Could not find messages object');
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
const messagesStr = match[1];
|
||||||
|
const messages = eval(`(${messagesStr})`);
|
||||||
|
|
||||||
|
const enKeys = Object.keys(messages.en);
|
||||||
|
const languages = Object.keys(messages);
|
||||||
|
|
||||||
|
const missing = {};
|
||||||
|
|
||||||
|
languages.forEach(lang => {
|
||||||
|
if (lang === 'en') return;
|
||||||
|
const langKeys = Object.keys(messages[lang]);
|
||||||
|
const missingKeys = enKeys.filter(k => !langKeys.includes(k));
|
||||||
|
if (missingKeys.length > 0) {
|
||||||
|
missing[lang] = missingKeys;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
console.log(JSON.stringify(missing, null, 2));
|
||||||
@@ -2,7 +2,9 @@
|
|||||||
<html lang="pl">
|
<html lang="pl">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8" />
|
<meta charset="UTF-8" />
|
||||||
<link rel="icon" type="image/svg+xml" href="/vite.svg" />
|
<link rel="icon" type="image/svg+xml" href="/nonograms.svg" />
|
||||||
|
<link rel="apple-touch-icon" href="/nonograms.svg" />
|
||||||
|
<link rel="mask-icon" href="/nonograms.svg" color="#00f2fe" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
<title>Nonograms Pro - Vue 3 SOLID</title>
|
<title>Nonograms Pro - Vue 3 SOLID</title>
|
||||||
</head>
|
</head>
|
||||||
@@ -10,4 +12,4 @@
|
|||||||
<div id="app"></div>
|
<div id="app"></div>
|
||||||
<script type="module" src="/src/main.js"></script>
|
<script type="module" src="/src/main.js"></script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|||||||
4
package-lock.json
generated
4
package-lock.json
generated
@@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "vue-nonograms-solid",
|
"name": "vue-nonograms-solid",
|
||||||
"version": "1.9.6",
|
"version": "1.9.13",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "vue-nonograms-solid",
|
"name": "vue-nonograms-solid",
|
||||||
"version": "1.9.6",
|
"version": "1.9.13",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"fireworks-js": "^2.10.8",
|
"fireworks-js": "^2.10.8",
|
||||||
"flag-icons": "^7.5.0",
|
"flag-icons": "^7.5.0",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "vue-nonograms-solid",
|
"name": "vue-nonograms-solid",
|
||||||
"version": "1.9.6",
|
"version": "1.9.13",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
|
|||||||
28
public/nonograms.svg
Normal file
28
public/nonograms.svg
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="192" height="192" viewBox="0 0 192 192">
|
||||||
|
<rect width="192" height="192" fill="#0b0f1f"/>
|
||||||
|
<g transform="translate(24,24)">
|
||||||
|
<rect x="0" y="0" width="144" height="144" rx="16" fill="#121639" stroke="#00f2fe" stroke-width="4"/>
|
||||||
|
<g stroke="#00f2fe" stroke-width="2">
|
||||||
|
<line x1="24" y1="0" x2="24" y2="144"/>
|
||||||
|
<line x1="48" y1="0" x2="48" y2="144"/>
|
||||||
|
<line x1="72" y1="0" x2="72" y2="144"/>
|
||||||
|
<line x1="96" y1="0" x2="96" y2="144"/>
|
||||||
|
<line x1="120" y1="0" x2="120" y2="144"/>
|
||||||
|
<line x1="0" y1="24" x2="144" y2="24"/>
|
||||||
|
<line x1="0" y1="48" x2="144" y2="48"/>
|
||||||
|
<line x1="0" y1="72" x2="144" y2="72"/>
|
||||||
|
<line x1="0" y1="96" x2="144" y2="96"/>
|
||||||
|
<line x1="0" y1="120" x2="144" y2="120"/>
|
||||||
|
</g>
|
||||||
|
<g fill="#00f2fe">
|
||||||
|
<rect x="6" y="6" width="18" height="18" rx="3"/>
|
||||||
|
<rect x="54" y="30" width="18" height="18" rx="3"/>
|
||||||
|
<rect x="102" y="78" width="18" height="18" rx="3"/>
|
||||||
|
<rect x="30" y="126" width="18" height="18" rx="3"/>
|
||||||
|
</g>
|
||||||
|
<g fill="#ffffff">
|
||||||
|
<path d="M36 40 h16 v64 h-16 z"/>
|
||||||
|
<path d="M52 40 h16 l32 48 v-48 h16 v64 h-16 l-32 -48 v48 h-16 z"/>
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 1.2 KiB |
49
scripts/fill_language_labels.cjs
Normal file
49
scripts/fill_language_labels.cjs
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
const fs = require('fs');
|
||||||
|
const path = 'src/composables/useI18n.js';
|
||||||
|
let content = fs.readFileSync(path, 'utf8');
|
||||||
|
|
||||||
|
const messagesMatch = content.match(/const messages = ({[\s\S]*?});/);
|
||||||
|
if (!messagesMatch) {
|
||||||
|
console.error('Could not find messages object');
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
const messages = eval(`(${messagesMatch[1]})`);
|
||||||
|
const en = messages.en || {};
|
||||||
|
const allLangKeys = Object.keys(en).filter((k) => k.startsWith('language.'));
|
||||||
|
|
||||||
|
function injectMissingLanguageLabels(localeCode) {
|
||||||
|
const blockStartRegex = new RegExp(`\\s{2}['\"]?${localeCode}['\"]?\\s*:\\s*\\{`);
|
||||||
|
const startIndex = content.search(blockStartRegex);
|
||||||
|
if (startIndex === -1) return;
|
||||||
|
const braceStart = content.indexOf('{', startIndex);
|
||||||
|
let i = braceStart + 1;
|
||||||
|
let depth = 1;
|
||||||
|
while (i < content.length && depth > 0) {
|
||||||
|
if (content[i] === '{') depth++;
|
||||||
|
else if (content[i] === '}') depth--;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
const block = content.slice(braceStart + 1, i - 1);
|
||||||
|
let updated = block;
|
||||||
|
let addedAny = false;
|
||||||
|
allLangKeys.forEach((key) => {
|
||||||
|
const keyRegex = new RegExp(`(['\"])${key}\\1\\s*:\\s*(['\"]).*?\\2`);
|
||||||
|
if (!keyRegex.test(updated)) {
|
||||||
|
const value = en[key];
|
||||||
|
updated = updated.trim().endsWith(',') ? updated + `\n '${key}': '${value}'` : updated + `,\n '${key}': '${value}'`;
|
||||||
|
addedAny = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
if (addedAny) {
|
||||||
|
content = content.slice(0, braceStart + 1) + updated + content.slice(i - 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Object.keys(messages).forEach((locale) => {
|
||||||
|
if (locale === 'en') return;
|
||||||
|
injectMissingLanguageLabels(locale);
|
||||||
|
});
|
||||||
|
|
||||||
|
fs.writeFileSync(path, content);
|
||||||
|
console.log('Filled missing language.* labels for all locales.');
|
||||||
514
scripts/translate_simulation.cjs
Normal file
514
scripts/translate_simulation.cjs
Normal file
@@ -0,0 +1,514 @@
|
|||||||
|
const fs = require('fs');
|
||||||
|
const path = 'src/composables/useI18n.js';
|
||||||
|
let content = fs.readFileSync(path, 'utf8');
|
||||||
|
|
||||||
|
const keys = {
|
||||||
|
'custom.showMap': {
|
||||||
|
es: 'Mostrar mapa de dificultad',
|
||||||
|
fr: 'Afficher la carte de difficulté',
|
||||||
|
de: 'Schwierigkeitskarte anzeigen',
|
||||||
|
it: 'Mostra mappa difficoltà',
|
||||||
|
pt: 'Mostrar mapa de dificuldade',
|
||||||
|
'pt-br': 'Mostrar mapa de dificuldade',
|
||||||
|
ru: 'Показать карту сложности',
|
||||||
|
zh: '显示难度地图',
|
||||||
|
ja: '難易度マップを表示',
|
||||||
|
ko: '난이도 맵 표시',
|
||||||
|
tr: 'Zorluk haritasını göster',
|
||||||
|
uk: 'Показати карту складності',
|
||||||
|
cs: 'Zobrazit mapu obtížnosti',
|
||||||
|
sk: 'Zobraziť mapu náročnosti',
|
||||||
|
hu: 'Nehézségi térkép megjelenítése',
|
||||||
|
ro: 'Arată harta dificultății',
|
||||||
|
bg: 'Покажи картата на трудността',
|
||||||
|
el: 'Εμφάνιση χάρτη δυσκολίας',
|
||||||
|
sr: 'Прикажи карту тежине',
|
||||||
|
hr: 'Prikaži kartu težine',
|
||||||
|
sl: 'Prikaži zemljevid težavnosti',
|
||||||
|
lt: 'Rodyti sudėtingumo žemėlapį',
|
||||||
|
lv: 'Rādīt grūtības karti',
|
||||||
|
et: 'Näita raskusaste kaarti',
|
||||||
|
nl: 'Moeilijkheidskaart weergeven',
|
||||||
|
sv: 'Visa svårighetskarta',
|
||||||
|
da: 'Vis sværhedsgradskort',
|
||||||
|
fi: 'Näytä vaikeuskartta',
|
||||||
|
no: 'Vis vanskelighetskart',
|
||||||
|
ar: 'إظهار خريطة الصعوبة',
|
||||||
|
hi: 'कठिनाई मानचित्र दिखाएँ',
|
||||||
|
bn: 'কঠিনতার মানচিত্র দেখান'
|
||||||
|
},
|
||||||
|
'custom.hideMap': {
|
||||||
|
es: 'Ocultar mapa de dificultad',
|
||||||
|
fr: 'Masquer la carte de difficulté',
|
||||||
|
de: 'Schwierigkeitskarte ausblenden',
|
||||||
|
it: 'Nascondi mappa difficoltà',
|
||||||
|
pt: 'Ocultar mapa de dificuldade',
|
||||||
|
'pt-br': 'Ocultar mapa de dificuldade',
|
||||||
|
ru: 'Скрыть карту сложности',
|
||||||
|
zh: '隐藏难度地图',
|
||||||
|
ja: '難易度マップを非表示',
|
||||||
|
ko: '난이도 맵 숨기기',
|
||||||
|
tr: 'Zorluk haritasını gizle',
|
||||||
|
uk: 'Приховати карту складності',
|
||||||
|
cs: 'Skrýt mapu obtížnosti',
|
||||||
|
sk: 'Skryť mapu náročnosti',
|
||||||
|
hu: 'Nehézségi térkép elrejtése',
|
||||||
|
ro: 'Ascunde harta dificultății',
|
||||||
|
bg: 'Скрий картата на трудността',
|
||||||
|
el: 'Απόκρυψη χάρτη δυσκολίας',
|
||||||
|
sr: 'Сакриј карту тежине',
|
||||||
|
hr: 'Sakrij kartu težine',
|
||||||
|
sl: 'Skrij zemljevid težavnosti',
|
||||||
|
lt: 'Slėpti sudėtingumo žemėlapį',
|
||||||
|
lv: 'Paslēpt grūtības karti',
|
||||||
|
et: 'Peida raskusaste kaart',
|
||||||
|
nl: 'Moeilijkheidskaart verbergen',
|
||||||
|
sv: 'Dölj svårighetskarta',
|
||||||
|
da: 'Skjul sværhedsgradskort',
|
||||||
|
fi: 'Piilota vaikeuskartta',
|
||||||
|
no: 'Skjul vanskelighetskart',
|
||||||
|
ar: 'إخفاء خريطة الصعوبة',
|
||||||
|
hi: 'कठिनाई मानचित्र छुपाएँ',
|
||||||
|
bn: 'কঠিনতার মানচিত্র লুকান'
|
||||||
|
},
|
||||||
|
'simulation.title': {
|
||||||
|
es: 'Simulación de Dificultad',
|
||||||
|
fr: 'Simulation de difficulté',
|
||||||
|
de: 'Schwierigkeitssimulation',
|
||||||
|
it: 'Simulazione della difficoltà',
|
||||||
|
pt: 'Simulação de Dificuldade',
|
||||||
|
'pt-br': 'Simulação de Dificuldade',
|
||||||
|
ru: 'Симуляция сложности',
|
||||||
|
zh: '难度模拟',
|
||||||
|
ja: '難易度シミュレーション',
|
||||||
|
ko: '난이도 시뮬레이션',
|
||||||
|
tr: 'Zorluk simülasyonu',
|
||||||
|
uk: 'Симуляція складності',
|
||||||
|
cs: 'Simulace obtížnosti',
|
||||||
|
sk: 'Simulácia náročnosti',
|
||||||
|
hu: 'Nehézség szimuláció',
|
||||||
|
ro: 'Simulare de dificultate',
|
||||||
|
bg: 'Симулиране на трудност',
|
||||||
|
el: 'Προσομοίωση δυσκολίας',
|
||||||
|
sr: 'Симулација тежине',
|
||||||
|
hr: 'Simulacija težine',
|
||||||
|
sl: 'Simulacija težavnosti',
|
||||||
|
lt: 'Sudėtingumo simuliacija',
|
||||||
|
lv: 'Grūtības simulācija',
|
||||||
|
et: 'Raskusastme simulatsioon',
|
||||||
|
nl: 'Moeilijkheidssimulatie',
|
||||||
|
sv: 'Svårighetssimulering',
|
||||||
|
da: 'Sværhedsgradssimulering',
|
||||||
|
fi: 'Vaikeussimulointi',
|
||||||
|
no: 'Vanskelighetssimulering',
|
||||||
|
ar: 'محاكاة الصعوبة',
|
||||||
|
hi: 'कठिनाई सिमुलेशन',
|
||||||
|
bn: 'কঠিনতা সিমুলেশন'
|
||||||
|
},
|
||||||
|
'simulation.status.ready': {
|
||||||
|
es: 'Listo',
|
||||||
|
fr: 'Prêt',
|
||||||
|
de: 'Bereit',
|
||||||
|
it: 'Pronto',
|
||||||
|
pt: 'Pronto',
|
||||||
|
'pt-br': 'Pronto',
|
||||||
|
ru: 'Готово',
|
||||||
|
zh: '就绪',
|
||||||
|
ja: '準備完了',
|
||||||
|
ko: '준비됨',
|
||||||
|
tr: 'Hazır',
|
||||||
|
uk: 'Готово',
|
||||||
|
cs: 'Připraveno',
|
||||||
|
sk: 'Pripravené',
|
||||||
|
hu: 'Kész',
|
||||||
|
ro: 'Gata',
|
||||||
|
bg: 'Готово',
|
||||||
|
el: 'Έτοιμο',
|
||||||
|
sr: 'Спремно',
|
||||||
|
hr: 'Spremno',
|
||||||
|
sl: 'Pripravljeno',
|
||||||
|
lt: 'Paruošta',
|
||||||
|
lv: 'Gatavs',
|
||||||
|
et: 'Valmis',
|
||||||
|
nl: 'Gereed',
|
||||||
|
sv: 'Klar',
|
||||||
|
da: 'Klar',
|
||||||
|
fi: 'Valmis',
|
||||||
|
no: 'Klar',
|
||||||
|
ar: 'جاهز',
|
||||||
|
hi: 'तैयार',
|
||||||
|
bn: 'প্রস্তুত'
|
||||||
|
},
|
||||||
|
'simulation.status.stopped': {
|
||||||
|
es: 'Detenido',
|
||||||
|
fr: 'Arrêté',
|
||||||
|
de: 'Gestoppt',
|
||||||
|
it: 'Arrestato',
|
||||||
|
pt: 'Parado',
|
||||||
|
'pt-br': 'Parado',
|
||||||
|
ru: 'Остановлено',
|
||||||
|
zh: '已停止',
|
||||||
|
ja: '停止',
|
||||||
|
ko: '중지됨',
|
||||||
|
tr: 'Durduruldu',
|
||||||
|
uk: 'Зупинено',
|
||||||
|
cs: 'Zastaveno',
|
||||||
|
sk: 'Zastavené',
|
||||||
|
hu: 'Leállítva',
|
||||||
|
ro: 'Oprit',
|
||||||
|
bg: 'Спряно',
|
||||||
|
el: 'Διακοπή',
|
||||||
|
sr: 'Заустављено',
|
||||||
|
hr: 'Zaustavljeno',
|
||||||
|
sl: 'Ustavljeno',
|
||||||
|
lt: 'Sustabdyta',
|
||||||
|
lv: 'Apturēts',
|
||||||
|
et: 'Peatatud',
|
||||||
|
nl: 'Gestopt',
|
||||||
|
sv: 'Stoppad',
|
||||||
|
da: 'Stoppet',
|
||||||
|
fi: 'Pysäytetty',
|
||||||
|
no: 'Stoppet',
|
||||||
|
ar: 'متوقف',
|
||||||
|
hi: 'रोका गया',
|
||||||
|
bn: 'বন্ধ'
|
||||||
|
},
|
||||||
|
'simulation.status.completed': {
|
||||||
|
es: 'Completado',
|
||||||
|
fr: 'Terminé',
|
||||||
|
de: 'Abgeschlossen',
|
||||||
|
it: 'Completato',
|
||||||
|
pt: 'Concluído',
|
||||||
|
'pt-br': 'Concluído',
|
||||||
|
ru: 'Завершено',
|
||||||
|
zh: '已完成',
|
||||||
|
ja: '完了',
|
||||||
|
ko: '완료됨',
|
||||||
|
tr: 'Tamamlandı',
|
||||||
|
uk: 'Завершено',
|
||||||
|
cs: 'Dokončeno',
|
||||||
|
sk: 'Dokončené',
|
||||||
|
hu: 'Befejezve',
|
||||||
|
ro: 'Finalizat',
|
||||||
|
bg: 'Завършено',
|
||||||
|
el: 'Ολοκληρώθηκε',
|
||||||
|
sr: 'Завршено',
|
||||||
|
hr: 'Dovršeno',
|
||||||
|
sl: 'Dokončano',
|
||||||
|
lt: 'Baigta',
|
||||||
|
lv: 'Pabeigts',
|
||||||
|
et: 'Lõpetatud',
|
||||||
|
nl: 'Voltooid',
|
||||||
|
sv: 'Slutförd',
|
||||||
|
da: 'Fuldført',
|
||||||
|
fi: 'Valmis',
|
||||||
|
no: 'Fullført',
|
||||||
|
ar: 'مكتمل',
|
||||||
|
hi: 'पूर्ण',
|
||||||
|
bn: 'সম্পন্ন'
|
||||||
|
},
|
||||||
|
'simulation.status.simulating': {
|
||||||
|
es: 'Simulando {size}x{size} @ {density}%',
|
||||||
|
fr: 'Simulation de {size}x{size} à {density}%',
|
||||||
|
de: 'Simuliere {size}x{size} @ {density}%',
|
||||||
|
it: 'Simulazione {size}x{size} @ {density}%',
|
||||||
|
pt: 'Simulando {size}x{size} @ {density}%',
|
||||||
|
'pt-br': 'Simulando {size}x{size} @ {density}%',
|
||||||
|
ru: 'Симуляция {size}x{size} @ {density}%',
|
||||||
|
zh: '正在模拟 {size}x{size} @ {density}%',
|
||||||
|
ja: '{size}x{size} @ {density}% をシミュレーション中',
|
||||||
|
ko: '{size}x{size} @ {density}% 시뮬레이션 중',
|
||||||
|
tr: '{size}x{size} @ {density}% simüle ediliyor',
|
||||||
|
uk: 'Симулювання {size}x{size} @ {density}%',
|
||||||
|
cs: 'Simulace {size}x{size} @ {density}%',
|
||||||
|
sk: 'Simulácia {size}x{size} @ {density}%',
|
||||||
|
hu: 'Szimulálás {size}x{size} @ {density}%',
|
||||||
|
ro: 'Simulare {size}x{size} @ {density}%',
|
||||||
|
bg: 'Симулиране {size}x{size} @ {density}%',
|
||||||
|
el: 'Προσομοίωση {size}x{size} @ {density}%',
|
||||||
|
sr: 'Симулирање {size}x{size} @ {density}%',
|
||||||
|
hr: 'Simulacija {size}x{size} @ {density}%',
|
||||||
|
sl: 'Simulacija {size}x{size} @ {density}%',
|
||||||
|
lt: 'Simuliuojama {size}x{size} @ {density}%',
|
||||||
|
lv: 'Simulācija {size}x{size} @ {density}%',
|
||||||
|
et: 'Simuleerimine {size}x{size} @ {density}%',
|
||||||
|
nl: 'Simuleren {size}x{size} @ {density}%',
|
||||||
|
sv: 'Simulerar {size}x{size} @ {density}%',
|
||||||
|
da: 'Simulerer {size}x{size} @ {density}%',
|
||||||
|
fi: 'Simulointi {size}x{size} @ {density}%',
|
||||||
|
no: 'Simulerer {size}x{size} @ {density}%',
|
||||||
|
ar: 'محاكاة {size}x{size} @ {density}%',
|
||||||
|
hi: '{size}x{size} @ {density}% का सिमुलेशन',
|
||||||
|
bn: '{size}x{size} @ {density}% সিমুলেট করা হচ্ছে'
|
||||||
|
},
|
||||||
|
'simulation.start': {
|
||||||
|
es: 'Iniciar simulación',
|
||||||
|
fr: 'Démarrer la simulation',
|
||||||
|
de: 'Simulation starten',
|
||||||
|
it: 'Avvia simulazione',
|
||||||
|
pt: 'Iniciar simulação',
|
||||||
|
'pt-br': 'Iniciar simulação',
|
||||||
|
ru: 'Начать симуляцию',
|
||||||
|
zh: '开始模拟',
|
||||||
|
ja: 'シミュレーション開始',
|
||||||
|
ko: '시뮬레이션 시작',
|
||||||
|
tr: 'Simülasyonu başlat',
|
||||||
|
uk: 'Почати симуляцію',
|
||||||
|
cs: 'Spustit simulaci',
|
||||||
|
sk: 'Spustiť simuláciu',
|
||||||
|
hu: 'Szimuláció indítása',
|
||||||
|
ro: 'Pornește simularea',
|
||||||
|
bg: 'Стартирай симулация',
|
||||||
|
el: 'Έναρξη προσομοίωσης',
|
||||||
|
sr: 'Покрени симулацију',
|
||||||
|
hr: 'Pokreni simulaciju',
|
||||||
|
sl: 'Zaženi simulacijo',
|
||||||
|
lt: 'Pradėti simuliaciją',
|
||||||
|
lv: 'Sākt simulāciju',
|
||||||
|
et: 'Alusta simulatsiooni',
|
||||||
|
nl: 'Simulatie starten',
|
||||||
|
sv: 'Starta simulering',
|
||||||
|
da: 'Start simulering',
|
||||||
|
fi: 'Aloita simulointi',
|
||||||
|
no: 'Start simulering',
|
||||||
|
ar: 'بدء المحاكاة',
|
||||||
|
hi: 'सिमुलेशन शुरू करें',
|
||||||
|
bn: 'সিমুলেশন শুরু'
|
||||||
|
},
|
||||||
|
'simulation.stop': {
|
||||||
|
es: 'Detener',
|
||||||
|
fr: 'Arrêter',
|
||||||
|
de: 'Stoppen',
|
||||||
|
it: 'Stop',
|
||||||
|
pt: 'Parar',
|
||||||
|
'pt-br': 'Parar',
|
||||||
|
ru: 'Стоп',
|
||||||
|
zh: '停止',
|
||||||
|
ja: '停止',
|
||||||
|
ko: '중지',
|
||||||
|
tr: 'Durdur',
|
||||||
|
uk: 'Зупинити',
|
||||||
|
cs: 'Zastavit',
|
||||||
|
sk: 'Zastaviť',
|
||||||
|
hu: 'Leállítás',
|
||||||
|
ro: 'Oprește',
|
||||||
|
bg: 'Спри',
|
||||||
|
el: 'Διακοπή',
|
||||||
|
sr: 'Заустави',
|
||||||
|
hr: 'Zaustavi',
|
||||||
|
sl: 'Ustavi',
|
||||||
|
lt: 'Stabdyti',
|
||||||
|
lv: 'Apturēt',
|
||||||
|
et: 'Peata',
|
||||||
|
nl: 'Stoppen',
|
||||||
|
sv: 'Stoppa',
|
||||||
|
da: 'Stop',
|
||||||
|
fi: 'Pysäytä',
|
||||||
|
no: 'Stopp',
|
||||||
|
ar: 'إيقاف',
|
||||||
|
hi: 'रोकें',
|
||||||
|
bn: 'বন্ধ করুন'
|
||||||
|
},
|
||||||
|
'simulation.table.size': {
|
||||||
|
es: 'Tamaño',
|
||||||
|
fr: 'Taille',
|
||||||
|
de: 'Größe',
|
||||||
|
it: 'Dimensione',
|
||||||
|
pt: 'Tamanho',
|
||||||
|
'pt-br': 'Tamanho',
|
||||||
|
ru: 'Размер',
|
||||||
|
zh: '大小',
|
||||||
|
ja: 'サイズ',
|
||||||
|
ko: '크기',
|
||||||
|
tr: 'Boyut',
|
||||||
|
uk: 'Розмір',
|
||||||
|
cs: 'Velikost',
|
||||||
|
sk: 'Veľkosť',
|
||||||
|
hu: 'Méret',
|
||||||
|
ro: 'Dimensiune',
|
||||||
|
bg: 'Размер',
|
||||||
|
el: 'Μέγεθος',
|
||||||
|
sr: 'Величина',
|
||||||
|
hr: 'Veličina',
|
||||||
|
sl: 'Velikost',
|
||||||
|
lt: 'Dydis',
|
||||||
|
lv: 'Izmērs',
|
||||||
|
et: 'Suurus',
|
||||||
|
nl: 'Grootte',
|
||||||
|
sv: 'Storlek',
|
||||||
|
da: 'Størrelse',
|
||||||
|
fi: 'Koko',
|
||||||
|
no: 'Størrelse',
|
||||||
|
ar: 'الحجم',
|
||||||
|
hi: 'आकार',
|
||||||
|
bn: 'আকার'
|
||||||
|
},
|
||||||
|
'simulation.table.density': {
|
||||||
|
es: 'Densidad',
|
||||||
|
fr: 'Densité',
|
||||||
|
de: 'Dichte',
|
||||||
|
it: 'Densità',
|
||||||
|
pt: 'Densidade',
|
||||||
|
'pt-br': 'Densidade',
|
||||||
|
ru: 'Плотность',
|
||||||
|
zh: '密度',
|
||||||
|
ja: '密度',
|
||||||
|
ko: '밀도',
|
||||||
|
tr: 'Yoğunluk',
|
||||||
|
uk: 'Щільність',
|
||||||
|
cs: 'Hustota',
|
||||||
|
sk: 'Hustota',
|
||||||
|
hu: 'Sűrűség',
|
||||||
|
ro: 'Densitate',
|
||||||
|
bg: 'Плътност',
|
||||||
|
el: 'Πυκνότητα',
|
||||||
|
sr: 'Густина',
|
||||||
|
hr: 'Gustoća',
|
||||||
|
sl: 'Gostota',
|
||||||
|
lt: 'Tankis',
|
||||||
|
lv: 'Blīvums',
|
||||||
|
et: 'Tihedus',
|
||||||
|
nl: 'Dichtheid',
|
||||||
|
sv: 'Densitet',
|
||||||
|
da: 'Densitet',
|
||||||
|
fi: 'Tiheys',
|
||||||
|
no: 'Tetthet',
|
||||||
|
ar: 'الكثافة',
|
||||||
|
hi: 'घनत्व',
|
||||||
|
bn: 'ঘনত্ব'
|
||||||
|
},
|
||||||
|
'simulation.table.solved': {
|
||||||
|
es: 'Resuelto (Lógica)',
|
||||||
|
fr: 'Résolu (Logique)',
|
||||||
|
de: 'Gelöst (Logik)',
|
||||||
|
it: 'Risolto (Logica)',
|
||||||
|
pt: 'Resolvido (Lógica)',
|
||||||
|
'pt-br': 'Resolvido (Lógica)',
|
||||||
|
ru: 'Решено (Логика)',
|
||||||
|
zh: '已解(逻辑)',
|
||||||
|
ja: '解決(ロジック)',
|
||||||
|
ko: '해결됨(논리)',
|
||||||
|
tr: 'Çözüldü (Mantık)',
|
||||||
|
uk: 'Розв’язано (Логіка)',
|
||||||
|
cs: 'Vyřešeno (Logika)',
|
||||||
|
sk: 'Vyriešené (Logika)',
|
||||||
|
hu: 'Megoldva (Logika)',
|
||||||
|
ro: 'Rezolvat (Logică)',
|
||||||
|
bg: 'Решено (Логика)',
|
||||||
|
el: 'Επιλύθηκε (Λογική)',
|
||||||
|
sr: 'Решено (Логика)',
|
||||||
|
hr: 'Riješeno (Logika)',
|
||||||
|
sl: 'Rešeno (Logika)',
|
||||||
|
lt: 'Išspręsta (Logika)',
|
||||||
|
lv: 'Atrisināts (Loģika)',
|
||||||
|
et: 'Lahendatud (Loogika)',
|
||||||
|
nl: 'Opgelost (Logica)',
|
||||||
|
sv: 'Löst (Logik)',
|
||||||
|
da: 'Løst (Logik)',
|
||||||
|
fi: 'Ratkaistu (Logiikka)',
|
||||||
|
no: 'Løst (Logikk)',
|
||||||
|
ar: 'تم الحل (منطق)',
|
||||||
|
hi: 'हल (तर्क)',
|
||||||
|
bn: 'সমাধান (লজিক)'
|
||||||
|
},
|
||||||
|
'simulation.empty': {
|
||||||
|
es: 'Pulsa Iniciar para ejecutar la simulación Monte Carlo',
|
||||||
|
fr: 'Appuyez sur Démarrer pour lancer la simulation Monte Carlo',
|
||||||
|
de: 'Drücke Start, um die Monte-Carlo-Simulation zu starten',
|
||||||
|
it: 'Premi Avvia per eseguire la simulazione Monte Carlo',
|
||||||
|
pt: 'Pressione Iniciar para executar a simulação de Monte Carlo',
|
||||||
|
'pt-br': 'Pressione Iniciar para executar a simulação de Monte Carlo',
|
||||||
|
ru: 'Нажмите «Старт», чтобы запустить моделирование Монте‑Карло',
|
||||||
|
zh: '点击开始运行蒙特卡罗模拟',
|
||||||
|
ja: 'Monte Carlo シミュレーションを実行するには開始を押してください',
|
||||||
|
ko: 'Monte Carlo 시뮬레이션을 실행하려면 시작을 누르세요',
|
||||||
|
tr: 'Monte Carlo simülasyonunu çalıştırmak için Başlat’a basın',
|
||||||
|
uk: 'Натисніть «Почати», щоб запустити симуляцію Монте‑Карло',
|
||||||
|
cs: 'Stiskněte Start pro spuštění simulace Monte Carlo',
|
||||||
|
sk: 'Stlačte Štart pre spustenie simulácie Monte Carlo',
|
||||||
|
hu: 'Nyomd meg a Startot a Monte Carlo szimulációhoz',
|
||||||
|
ro: 'Apasă Start pentru a rula simularea Monte Carlo',
|
||||||
|
bg: 'Натисни Старт, за да стартираш симулация Монте Карло',
|
||||||
|
el: 'Πατήστε Έναρξη για να τρέξετε προσομοίωση Monte Carlo',
|
||||||
|
sr: 'Притисни Старт да покренеш Монте Карло симулацију',
|
||||||
|
hr: 'Pritisni Start za pokretanje Monte Carlo simulacije',
|
||||||
|
sl: 'Pritisnite Start za zagon simulacije Monte Carlo',
|
||||||
|
lt: 'Paspauskite Start, kad paleistumėte Monte Karlo simuliaciją',
|
||||||
|
lv: 'Nospiediet Start, lai palaistu Monte Carlo simulāciju',
|
||||||
|
et: 'Vajuta Start, et käivitada Monte Carlo simulatsioon',
|
||||||
|
nl: 'Druk op Start om de Monte Carlo-simulatie te starten',
|
||||||
|
sv: 'Tryck Start för att köra Monte Carlo-simuleringen',
|
||||||
|
da: 'Tryk Start for at køre Monte Carlo-simuleringen',
|
||||||
|
fi: 'Paina Käynnistä aloittaaksesi Monte Carlo -simulaation',
|
||||||
|
no: 'Trykk Start for å kjøre Monte Carlo-simuleringen',
|
||||||
|
ar: 'اضغط ابدأ لتشغيل محاكاة مونتِ كارلو',
|
||||||
|
hi: 'मोंटे कार्लो सिमुलेशन चलाने के लिए स्टार्ट दबाएँ',
|
||||||
|
bn: 'মোন্টে কার্লো সিমুলেশন চালাতে স্টার্ট চাপুন'
|
||||||
|
},
|
||||||
|
'custom.simulationHelp': {
|
||||||
|
es: '¿Cómo se calcula?',
|
||||||
|
fr: 'Comment est-ce calculé ?',
|
||||||
|
de: 'Wie wird das berechnet?',
|
||||||
|
it: 'Come viene calcolato?',
|
||||||
|
pt: 'Como isso é calculado?',
|
||||||
|
'pt-br': 'Como isso é calculado?',
|
||||||
|
ru: 'Как это рассчитывается?',
|
||||||
|
zh: '这是如何计算的?',
|
||||||
|
ja: 'これはどのように計算されますか?',
|
||||||
|
ko: '이것은 어떻게 계산됩니까?',
|
||||||
|
tr: 'Bu nasıl hesaplanıyor?',
|
||||||
|
uk: 'Як це обчислюється?',
|
||||||
|
cs: 'Jak se to počítá?',
|
||||||
|
sk: 'Ako sa to počíta?',
|
||||||
|
hu: 'Hogyan számoljuk?',
|
||||||
|
ro: 'Cum este calculat?',
|
||||||
|
bg: 'Как се изчислява?',
|
||||||
|
el: 'Πώς υπολογίζεται;',
|
||||||
|
sr: 'Како се израчунава?',
|
||||||
|
hr: 'Kako se izračunava?',
|
||||||
|
sl: 'Kako je izračunano?',
|
||||||
|
lt: 'Kaip tai apskaičiuojama?',
|
||||||
|
lv: 'Kā tas tiek aprēķināts?',
|
||||||
|
et: 'Kuidas see arvutatakse?',
|
||||||
|
nl: 'Hoe wordt dit berekend?',
|
||||||
|
sv: 'Hur beräknas detta?',
|
||||||
|
da: 'Hvordan beregnes dette?',
|
||||||
|
fi: 'Miten tämä lasketaan?',
|
||||||
|
no: 'Hvordan beregnes dette?',
|
||||||
|
ar: 'كيف يتم احتساب ذلك؟',
|
||||||
|
hi: 'यह कैसे गणना किया जाता है?',
|
||||||
|
bn: 'এটি কীভাবে গণনা করা হয়?'
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function replaceInLanguage(lang, key, value) {
|
||||||
|
const langStart = new RegExp(`\\s{2}['\"]?${lang}['\"]?\\s*:\\s*\\{`);
|
||||||
|
const index = content.search(langStart);
|
||||||
|
if (index === -1) return;
|
||||||
|
const start = content.indexOf('{', index);
|
||||||
|
let depth = 1;
|
||||||
|
let i = start + 1;
|
||||||
|
while (i < content.length && depth > 0) {
|
||||||
|
if (content[i] === '{') depth++;
|
||||||
|
else if (content[i] === '}') depth--;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
const block = content.slice(start + 1, i - 1);
|
||||||
|
let newBlock;
|
||||||
|
const keyRegex = new RegExp(`(['\"])${key}\\1\\s*:\\s*(['\"]).*?\\2`);
|
||||||
|
if (keyRegex.test(block)) {
|
||||||
|
newBlock = block.replace(keyRegex, `'${key}': '${value}'`);
|
||||||
|
} else {
|
||||||
|
newBlock = block.trim().endsWith(',') ? block + `\n '${key}': '${value}'` : block + `,\n '${key}': '${value}'`;
|
||||||
|
}
|
||||||
|
content = content.slice(0, start + 1) + newBlock + content.slice(i - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
Object.entries(keys).forEach(([key, translations]) => {
|
||||||
|
Object.entries(translations).forEach(([lang, value]) => {
|
||||||
|
replaceInLanguage(lang, key, value);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
fs.writeFileSync(path, content);
|
||||||
|
console.log('Translations updated.');
|
||||||
19
src/App.vue
19
src/App.vue
@@ -28,6 +28,21 @@ const appVersion = __APP_VERSION__;
|
|||||||
let displayModeMedia = null;
|
let displayModeMedia = null;
|
||||||
let prefersColorSchemeMedia = null;
|
let prefersColorSchemeMedia = null;
|
||||||
|
|
||||||
|
const onKeyDownGlobal = (e) => {
|
||||||
|
if (e.key !== 'Escape') return;
|
||||||
|
if (showSimulation.value) {
|
||||||
|
showSimulation.value = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (showCustomModal.value) {
|
||||||
|
showCustomModal.value = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (store.isGameWon) {
|
||||||
|
store.closeWinModal();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const installLabel = computed(() => {
|
const installLabel = computed(() => {
|
||||||
return isCoarsePointer.value ? t('pwa.installMobile') : t('pwa.installDesktop');
|
return isCoarsePointer.value ? t('pwa.installMobile') : t('pwa.installDesktop');
|
||||||
});
|
});
|
||||||
@@ -114,6 +129,7 @@ onMounted(() => {
|
|||||||
} else if (displayModeMedia?.addListener) {
|
} else if (displayModeMedia?.addListener) {
|
||||||
displayModeMedia.addListener(updateStandalone);
|
displayModeMedia.addListener(updateStandalone);
|
||||||
}
|
}
|
||||||
|
window.addEventListener('keydown', onKeyDownGlobal);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -131,6 +147,7 @@ onUnmounted(() => {
|
|||||||
} else if (displayModeMedia?.removeListener) {
|
} else if (displayModeMedia?.removeListener) {
|
||||||
displayModeMedia.removeListener(updateStandalone);
|
displayModeMedia.removeListener(updateStandalone);
|
||||||
}
|
}
|
||||||
|
window.removeEventListener('keydown', onKeyDownGlobal);
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@@ -279,4 +296,4 @@ onUnmounted(() => {
|
|||||||
border-top: 1px solid var(--panel-border);
|
border-top: 1px solid var(--panel-border);
|
||||||
z-index: 90;
|
z-index: 90;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -32,7 +32,11 @@ defineProps({
|
|||||||
v-for="(group, index) in hints"
|
v-for="(group, index) in hints"
|
||||||
:key="index"
|
:key="index"
|
||||||
class="hint-group"
|
class="hint-group"
|
||||||
:class="{ 'is-active': index === activeIndex }"
|
:class="{
|
||||||
|
'is-active': index === activeIndex,
|
||||||
|
'guide-right': orientation === 'col' && (index + 1) % 5 === 0 && index !== size - 1,
|
||||||
|
'guide-bottom': orientation === 'row' && (index + 1) % 5 === 0 && index !== size - 1
|
||||||
|
}"
|
||||||
>
|
>
|
||||||
<span
|
<span
|
||||||
v-for="(num, idx) in group"
|
v-for="(num, idx) in group"
|
||||||
@@ -111,4 +115,12 @@ defineProps({
|
|||||||
border-color: rgba(79, 172, 254, 0.8);
|
border-color: rgba(79, 172, 254, 0.8);
|
||||||
box-shadow: 0 0 12px rgba(79, 172, 254, 0.35);
|
box-shadow: 0 0 12px rgba(79, 172, 254, 0.35);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Guide lines every 5 */
|
||||||
|
.hint-group.guide-right {
|
||||||
|
border-right: 2px solid rgba(0, 242, 255, 0.5);
|
||||||
|
}
|
||||||
|
.hint-group.guide-bottom {
|
||||||
|
border-bottom: 2px solid rgba(0, 242, 255, 0.5);
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, computed } from 'vue';
|
import { ref, computed, onMounted, onUnmounted } from 'vue';
|
||||||
import { generateRandomGrid, calculateHints } from '@/utils/puzzleUtils';
|
import { generateRandomGrid, calculateHints } from '@/utils/puzzleUtils';
|
||||||
import { solvePuzzle } from '@/utils/solver';
|
import { solvePuzzle } from '@/utils/solver';
|
||||||
import { useI18n } from '@/composables/useI18n';
|
import { useI18n } from '@/composables/useI18n';
|
||||||
@@ -21,6 +21,22 @@ const simulationSpeed = ref(1); // 1 = Normal, 2 = Fast (less render updates)
|
|||||||
|
|
||||||
let stopRequested = false;
|
let stopRequested = false;
|
||||||
|
|
||||||
|
const onKeyDown = (e) => {
|
||||||
|
if (e.key === 'Escape') {
|
||||||
|
e.stopImmediatePropagation?.();
|
||||||
|
e.preventDefault?.();
|
||||||
|
emit('close');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
window.addEventListener('keydown', onKeyDown);
|
||||||
|
});
|
||||||
|
|
||||||
|
onUnmounted(() => {
|
||||||
|
window.removeEventListener('keydown', onKeyDown);
|
||||||
|
});
|
||||||
|
|
||||||
const displayStatus = computed(() => {
|
const displayStatus = computed(() => {
|
||||||
if (!currentStatus.value) return t('simulation.status.ready');
|
if (!currentStatus.value) return t('simulation.status.ready');
|
||||||
return currentStatus.value;
|
return currentStatus.value;
|
||||||
|
|||||||
@@ -28,6 +28,8 @@ const handleClose = () => {
|
|||||||
|
|
||||||
const handleKeyDown = (e) => {
|
const handleKeyDown = (e) => {
|
||||||
if (e.key === 'Escape') {
|
if (e.key === 'Escape') {
|
||||||
|
e.stopImmediatePropagation?.();
|
||||||
|
e.preventDefault?.();
|
||||||
handleClose();
|
handleClose();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -9,8 +9,8 @@ export function useSolver() {
|
|||||||
const isPlaying = ref(false);
|
const isPlaying = ref(false);
|
||||||
const isProcessing = ref(false);
|
const isProcessing = ref(false);
|
||||||
const speedIndex = ref(0);
|
const speedIndex = ref(0);
|
||||||
const speeds = [1000, 500, 250, 125, 62];
|
const speeds = [1000, 500, 250, 125, 62, 31, 16];
|
||||||
const speedLabels = ['x1', 'x2', 'x4', 'x8', 'x16'];
|
const speedLabels = ['x1', 'x2', 'x4', 'x8', 'x16', 'x32', 'x64'];
|
||||||
const statusText = ref(t('guide.waiting'));
|
const statusText = ref(t('guide.waiting'));
|
||||||
|
|
||||||
let intervalId = null;
|
let intervalId = null;
|
||||||
|
|||||||
Reference in New Issue
Block a user