Several changes

- fixed `runners.json` `Wine-6.16-GE-1` URI
- added recommendable only checkboxes for DXVK and runners lists
- fixed empty config property creation
- added runners installation, deletion and selection abilities
- `Runners.current` field was reworked as a `Runners.current()` method to be able
  to change selected runner according to the config file
This commit is contained in:
Observer KRypt0n_ 2021-12-27 18:39:40 +02:00
parent d26d9578d0
commit 082b596421
No known key found for this signature in database
GPG key ID: DC5D4EC1303465DA
13 changed files with 142 additions and 32 deletions

View file

@ -51,7 +51,11 @@ settings:
# Runners # Runners
runners: runners:
title: Wine version title: Wine version
items:
recommended: Show recommendable only
# DXVKs # DXVKs
dxvks: dxvks:
title: DXVK title: DXVK
items:
recommended: Show recommendable only

View file

@ -52,7 +52,11 @@ settings:
# Runners # Runners
runners: runners:
title: Версия Wine title: Версия Wine
items:
recommended: Показывать только рекомендуемое
# DXVKs # DXVKs
dxvks: dxvks:
title: DXVK title: DXVK
items:
recommended: Показывать только рекомендуемое

View file

@ -215,7 +215,7 @@
"family": "Wine-GE", "family": "Wine-GE",
"name": "lutris-ge-6.16-1-x86_64", "name": "lutris-ge-6.16-1-x86_64",
"title": "Wine-6.16-GE-1", "title": "Wine-6.16-GE-1",
"uri": "https://github.com/GloriousEggroll/wine-ge-custom/releases/download/6.16-GE-1/wine-lutris-ge-6.16-1-x86_64.tar.xz", "uri": "https://github.com/GloriousEggroll/wine-ge-custom/releases/download/6.16-GE-1/lutris-ge-6.16-1-x86_64.tar.xz",
"files": { "files": {
"wine": "bin/wine64", "wine": "bin/wine64",
"wineserver": "bin/wineserver", "wineserver": "bin/wineserver",

View file

@ -6,6 +6,8 @@
export let prop: string = ''; export let prop: string = '';
export let lang: string = ''; export let lang: string = '';
export let valueChanged: (value: boolean) => void = () => {};
import Checkmark from '../assets/svgs/checkmark.svg'; import Checkmark from '../assets/svgs/checkmark.svg';
import Configs from '../ts/Configs'; import Configs from '../ts/Configs';
@ -16,7 +18,11 @@
{ {
active = !active; active = !active;
Configs.set(prop, active); if (prop)
Configs.set(prop, active);
if (valueChanged)
valueChanged(active);
} }
</script> </script>

View file

@ -1,6 +1,8 @@
<script lang="ts"> <script lang="ts">
import { _ } from 'svelte-i18n'; import { _ } from 'svelte-i18n';
export let recommendable = true;
import DXVK from '../ts/core/DXVK'; import DXVK from '../ts/core/DXVK';
import type { import type {
@ -27,7 +29,7 @@
<div class="list"> <div class="list">
{#each dxvks as dxvk} {#each dxvks as dxvk}
<div class="list-item" class:list-item-downloaded={dxvkInstalled(dxvk)} class:list-item-active={dxvk.version === selectedVersion}> <div class="list-item" class:list-item-downloaded={dxvkInstalled(dxvk)} class:list-item-active={dxvk.version === selectedVersion} class:list-item-hidden={recommendable && !dxvk.recommended}>
{ dxvk.version } { dxvk.version }
<div> <div>

View file

@ -1,25 +1,63 @@
<script lang="ts"> <script lang="ts">
import { _ } from 'svelte-i18n'; import { _ } from 'svelte-i18n';
export let recommendable = true;
import Runners from '../ts/core/Runners'; import Runners from '../ts/core/Runners';
import type { Runner, RunnerFamily } from '../ts/types/Runners'; import type { Runner, RunnerFamily } from '../ts/types/Runners';
let runners: RunnerFamily[] = [], selectedVersion;
Runners.list().then((list) => runners = list);
Runners.current.then((current) => selectedVersion = current?.name);
import Delete from '../assets/images/delete.png'; import Delete from '../assets/images/delete.png';
import Download from '../assets/images/download.png'; import Download from '../assets/images/download.png';
const runnerInstalled = (runner: Runner): boolean => { let runners: RunnerFamily[] = [],
for (const family of runners) installedRunners = {},
for (const wine of family.runners) disabledRunners = {},
if (wine.name === runner.name) selectedVersion;
return wine.installed;
return false; Runners.list().then((list) => {
runners = list;
for (const family of runners)
for (const runner of family.runners)
{
installedRunners[runner.name] = runner.installed;
disabledRunners[runner.name] = false;
}
});
Runners.current().then((current) => selectedVersion = current?.name);
let progress = {};
const downloadRunner = (runner: Runner) => {
Runners.download(runner).then((stream) => {
stream?.downloadStart(() => disabledRunners[runner.name] = true);
stream?.downloadProgress((current, total) => {
progress[runner.name] = `${Math.round(current / total * 100)}%`;
});
stream?.unpackProgress((current, total) => {
progress[runner.name] = `${Math.round(current / total * 100)}%`;
});
stream?.unpackFinish(() => {
installedRunners[runner.name] = true;
disabledRunners[runner.name] = false;
progress[runner.name] = undefined;
});
});
};
const deleteRunner = (runner: Runner) => {
disabledRunners[runner.name] = true;
Runners.delete(runner).then(() => {
installedRunners[runner.name] = false;
disabledRunners[runner.name] = false;
});
}; };
</script> </script>
@ -28,17 +66,32 @@
<h2>{ family.title }</h2> <h2>{ family.title }</h2>
{#each family.runners as runner} {#each family.runners as runner}
<div class="list-item" class:list-item-downloaded={runnerInstalled(runner)} class:list-item-active={runner.name === selectedVersion}> <div
class="list-item"
class:list-item-downloaded={installedRunners[runner.name]}
class:list-item-active={runner.name === selectedVersion}
class:list-item-hidden={recommendable && !runner.recommended}
class:list-item-downloading={progress[runner.name] !== undefined}
class:list-item-disabled={disabledRunners[runner.name]}
on:click={() => {
if (installedRunners[runner.name])
{
selectedVersion = runner.name;
Runners.current(runner);
}
}}
>
{ runner.title } { runner.title }
<div> <div>
<span></span> <span>{progress[runner.name] ?? ''}</span>
<!-- svelte-ignore a11y-missing-attribute --> <!-- svelte-ignore a11y-missing-attribute -->
<img class="item-delete" src={Delete}> <img class="item-delete" src={Delete} on:click={() => deleteRunner(runner)}>
<!-- svelte-ignore a11y-missing-attribute --> <!-- svelte-ignore a11y-missing-attribute -->
<img class="item-download" src={Download}> <img class="item-download" src={Download} on:click={() => downloadRunner(runner)}>
</div> </div>
</div> </div>
{/each} {/each}

View file

@ -63,6 +63,9 @@
img img
filter: invert(70%) sepia(13%) saturate(241%) hue-rotate(196deg) brightness(97%) contrast(91%) filter: invert(70%) sepia(13%) saturate(241%) hue-rotate(196deg) brightness(97%) contrast(91%)
.list-item-hidden
display: none
.list-item-downloading .list-item-downloading
img img
display: none !important display: none !important

View file

@ -87,6 +87,9 @@
selectedItem = visibleElement.getAttribute('id'); selectedItem = visibleElement.getAttribute('id');
}; };
let dxvkRecommendable = true,
runnersRecommendable = true;
// Do some stuff when all the content will be loaded // Do some stuff when all the content will be loaded
onMount(() => { onMount(() => {
Window.current.show(); Window.current.show();
@ -143,13 +146,19 @@
<div class="settings-item" id="runners"> <div class="settings-item" id="runners">
<h1>{$_('settings.runners.title')}</h1> <h1>{$_('settings.runners.title')}</h1>
<RunnerSelectionList /> <Checkbox lang="settings.runners.items.recommended" valueChanged={(value) => runnersRecommendable = value} />
<RunnerSelectionList recommendable={runnersRecommendable} />
</div> </div>
<div class="settings-item" id="dxvks"> <div class="settings-item" id="dxvks">
<h1>{$_('settings.dxvks.title')}</h1> <h1>{$_('settings.dxvks.title')}</h1>
<DXVKSelectionList /> <Checkbox lang="settings.runners.items.recommended" valueChanged={(value) => dxvkRecommendable = value} />
<br><br>
<DXVKSelectionList recommendable={dxvkRecommendable} />
</div> </div>
</div> </div>
{/if} {/if}

View file

@ -86,7 +86,7 @@ export default class Voice
/** /**
* Get updated voice data from the specified version to the latest * Get updated voice data from the specified version to the latest
* *
* @returns Latest voice pack if current is out of date * @returns null if the difference can't be calculated
* @returns Error object if company's servers are unreachable or they responded with an error * @returns Error object if company's servers are unreachable or they responded with an error
*/ */
public static getDiff(version: string): Promise<VoicePack[]|null> public static getDiff(version: string): Promise<VoicePack[]|null>

View file

@ -72,7 +72,7 @@ export default class Prefix
]; ];
return new Promise((resolve) => { return new Promise((resolve) => {
Runners.current.then((runner) => { Runners.current().then((runner) => {
if (runner === null) if (runner === null)
resolve(false); resolve(false);

View file

@ -23,16 +23,29 @@ class Runners
{ {
/** /**
* Get the current using runner according to the config file * Get the current using runner according to the config file
* or set the new one
*/ */
public static get current(): Promise<Runner|null> public static current(runner: Runner|Runner['name']|null = null): Promise<Runner|null>
{ {
return new Promise((resolve) => { return new Promise(async (resolve) => {
Configs.get('runner').then((runner) => { if (runner === null)
if (typeof runner === 'string') {
Runners.get(runner).then((runner) => resolve(runner)); Configs.get('runner').then((runner) => {
if (typeof runner === 'string')
Runners.get(runner).then((runner) => resolve(runner));
else resolve(null); else resolve(null);
}); });
}
else
{
Configs.set('runner', typeof runner === 'string' ?
runner : runner.name);
resolve(typeof runner === 'string' ?
await this.get(runner) : runner);
}
}); });
} }
@ -121,6 +134,22 @@ class Runners
else resolve(new Stream(runner)); else resolve(new Stream(runner));
}); });
} }
/**
* Delete specified runner
*/
public static delete(runner: Runner|Runner['name']): Promise<void>
{
return new Promise(async (resolve) => {
const name = typeof runner !== 'string' ?
runner.name : runner;
Process.run(`rm -rf '${Process.addSlashes(await constants.paths.runnersDir + '/' + name)}'`)
.then((process) => {
process.finish(() => resolve());
});
});
}
} }
export default Runners; export default Runners;

View file

@ -22,7 +22,7 @@ type PromiseOptions = {
}; };
/** /**
* Make a promise from the provided function(s) and run it/them * Make a promise from the provided function(s) and run it(them)
*/ */
export default function promisify(callback: callback|Promise<any>|PromiseOptions): Promise<any> export default function promisify(callback: callback|Promise<any>|PromiseOptions): Promise<any>
{ {

View file

@ -12,7 +12,7 @@ export default (): Promise<void> => {
*/ */
let wineExeutable = 'wine'; let wineExeutable = 'wine';
const runner = await Runners.current; const runner = await Runners.current();
if (runner !== null) if (runner !== null)
{ {