Made default runner and dxvk auto-installation

- Wine-GE runners family moved to the top of the runners list
- now installed runner will be automatically selected
- launcher controls were slightly moved down
- fixed double resolved unpack directory logging in AbstractInstaller
- fixed `Prefix.create()` method work
- added `State.update()` method logging
- now settings button will be hidden when user are doing something
- added `InstallWine` and `InstallDXVK` scripts; added related launcher states
This commit is contained in:
Observer KRypt0n_ 2021-12-29 21:34:23 +02:00
parent 2a4dcc03e3
commit 81676f4b7c
No known key found for this signature in database
GPG key ID: DC5D4EC1303465DA
12 changed files with 323 additions and 123 deletions

View file

@ -169,11 +169,11 @@ This is our current roadmap goals. You can find older ones [here](ROADMAP.md)
* <s>Splash screen</s> * <s>Splash screen</s>
* <s>Theming system</s> * <s>Theming system</s>
* <s>Game pre-installation</s> * <s>Game pre-installation</s>
* Launcher auto-updates * <s>Default runner and DXVK auto-installation</s>
* Statistics window * Statistics window
* Chengelog window
* Default runner auto-installation
* Ability to change the temp directory where the launcher should download some files * Ability to change the temp directory where the launcher should download some files
* Launcher auto-updates
* Changelog window
### Features ### Features

View file

@ -1,4 +1,69 @@
[ [
{
"title": "Wine-GE",
"runners": [
{
"family": "Wine-GE",
"name": "lutris-ge-6.21-1-x86_64",
"title": "Wine-6.21-GE-1",
"uri": "https://github.com/GloriousEggroll/wine-ge-custom/releases/download/6.21-GE-1/wine-lutris-ge-6.21-1-x86_64.tar.xz",
"files": {
"wine": "bin/wine64",
"wineserver": "bin/wineserver",
"winecfg": "lib64/wine/x86_64-windows/winecfg.exe"
},
"recommended": true
},
{
"family": "Wine-GE",
"name": "lutris-ge-6.20-1-x86_64",
"title": "Wine-6.20-GE-1",
"uri": "https://github.com/GloriousEggroll/wine-ge-custom/releases/download/6.20-GE-1/wine-lutris-ge-6.20-1-x86_64.tar.xz",
"files": {
"wine": "bin/wine64",
"wineserver": "bin/wineserver",
"winecfg": "lib64/wine/x86_64-windows/winecfg.exe"
},
"recommended": true
},
{
"family": "Wine-GE",
"name": "lutris-ge-6.19-1-x86_64",
"title": "Wine-6.19-GE-1",
"uri": "https://github.com/GloriousEggroll/wine-ge-custom/releases/download/6.19-GE-1/wine-lutris-ge-6.19-1-x86_64.tar.xz",
"files": {
"wine": "bin/wine64",
"wineserver": "bin/wineserver",
"winecfg": "lib64/wine/x86_64-windows/winecfg.exe"
},
"recommended": true
},
{
"family": "Wine-GE",
"name": "lutris-ge-6.18-1-x86_64",
"title": "Wine-6.18-GE-1",
"uri": "https://github.com/GloriousEggroll/wine-ge-custom/releases/download/6.18-GE-1/wine-lutris-ge-6.18-1-x86_64.tar.xz",
"files": {
"wine": "bin/wine64",
"wineserver": "bin/wineserver",
"winecfg": "lib64/wine/x86_64-windows/winecfg.exe"
},
"recommended": true
},
{
"family": "Wine-GE",
"name": "lutris-ge-6.16-1-x86_64",
"title": "Wine-6.16-GE-1",
"uri": "https://github.com/GloriousEggroll/wine-ge-custom/releases/download/6.16-GE-1/lutris-ge-6.16-1-x86_64.tar.xz",
"files": {
"wine": "bin/wine64",
"wineserver": "bin/wineserver",
"winecfg": "lib64/wine/x86_64-windows/winecfg.exe"
},
"recommended": true
}
]
},
{ {
"title": "Lutris", "title": "Lutris",
"runners": [ "runners": [
@ -159,70 +224,5 @@
"recommended": false "recommended": false
} }
] ]
},
{
"title": "Wine-GE",
"runners": [
{
"family": "Wine-GE",
"name": "lutris-ge-6.21-1-x86_64",
"title": "Wine-6.21-GE-1",
"uri": "https://github.com/GloriousEggroll/wine-ge-custom/releases/download/6.21-GE-1/wine-lutris-ge-6.21-1-x86_64.tar.xz",
"files": {
"wine": "bin/wine64",
"wineserver": "bin/wineserver",
"winecfg": "lib64/wine/x86_64-windows/winecfg.exe"
},
"recommended": true
},
{
"family": "Wine-GE",
"name": "lutris-ge-6.20-1-x86_64",
"title": "Wine-6.20-GE-1",
"uri": "https://github.com/GloriousEggroll/wine-ge-custom/releases/download/6.20-GE-1/wine-lutris-ge-6.20-1-x86_64.tar.xz",
"files": {
"wine": "bin/wine64",
"wineserver": "bin/wineserver",
"winecfg": "lib64/wine/x86_64-windows/winecfg.exe"
},
"recommended": true
},
{
"family": "Wine-GE",
"name": "lutris-ge-6.19-1-x86_64",
"title": "Wine-6.19-GE-1",
"uri": "https://github.com/GloriousEggroll/wine-ge-custom/releases/download/6.19-GE-1/wine-lutris-ge-6.19-1-x86_64.tar.xz",
"files": {
"wine": "bin/wine64",
"wineserver": "bin/wineserver",
"winecfg": "lib64/wine/x86_64-windows/winecfg.exe"
},
"recommended": false
},
{
"family": "Wine-GE",
"name": "lutris-ge-6.18-1-x86_64",
"title": "Wine-6.18-GE-1",
"uri": "https://github.com/GloriousEggroll/wine-ge-custom/releases/download/6.18-GE-1/wine-lutris-ge-6.18-1-x86_64.tar.xz",
"files": {
"wine": "bin/wine64",
"wineserver": "bin/wineserver",
"winecfg": "lib64/wine/x86_64-windows/winecfg.exe"
},
"recommended": false
},
{
"family": "Wine-GE",
"name": "lutris-ge-6.16-1-x86_64",
"title": "Wine-6.16-GE-1",
"uri": "https://github.com/GloriousEggroll/wine-ge-custom/releases/download/6.16-GE-1/lutris-ge-6.16-1-x86_64.tar.xz",
"files": {
"wine": "bin/wine64",
"wineserver": "bin/wineserver",
"winecfg": "lib64/wine/x86_64-windows/winecfg.exe"
},
"recommended": false
}
]
} }
] ]

View file

@ -47,6 +47,10 @@
disabledRunners[runner.name] = false; disabledRunners[runner.name] = false;
progress[runner.name] = undefined; progress[runner.name] = undefined;
selectedVersion = runner.name;
Runners.current(runner);
}); });
}); });
}; };

View file

@ -41,7 +41,7 @@
height: 64px height: 64px
right: 128px right: 128px
bottom: 64px bottom: 48px
font-size: 22px font-size: 22px
@ -53,7 +53,7 @@
height: 52px height: 52px
right: 386px right: 386px
bottom: 70px bottom: 54px
border-radius: 32px border-radius: 32px
@ -98,7 +98,7 @@
position: absolute position: absolute
left: 48px left: 48px
bottom: 116px bottom: 100px
color: white color: white
font-size: 18px font-size: 18px
@ -119,7 +119,7 @@
padding: 0 padding: 0
left: 48px left: 48px
bottom: 68px bottom: 52px
width: 720px width: 720px
height: 36px height: 36px

View file

@ -114,15 +114,9 @@ export default abstract class Installer
if (this.onDownloadFinish) if (this.onDownloadFinish)
this.onDownloadFinish(); this.onDownloadFinish();
Promise.resolve(unpackDir)
.then((unpackDir) => {
if (shouldResolve)
debugThread.log(`Resolved unpack dir: ${unpackDir}`);
unpackArchive(); unpackArchive();
}); });
}); });
});
} }
else unpackArchive(); else unpackArchive();

View file

@ -104,7 +104,7 @@ export default class Prefix
this.getWinetricks().then(async (winetricks) => { this.getWinetricks().then(async (winetricks) => {
let installationProgress = 0; let installationProgress = 0;
const process = await Process.run(`./'${Process.addSlashes(winetricks)}' corefonts usetakefocus=n`, { const process = await Process.run(`'${Process.addSlashes(winetricks)}' corefonts usetakefocus=n`, {
env: { env: {
WINE: `${await constants.paths.runnersDir}/${runner.name}/${runner.files.wine}`, WINE: `${await constants.paths.runnersDir}/${runner.name}/${runner.files.wine}`,
WINESERVER: `${await constants.paths.runnersDir}/${runner.name}/${runner.files.wineserver}`, WINESERVER: `${await constants.paths.runnersDir}/${runner.name}/${runner.files.wineserver}`,

View file

@ -6,6 +6,9 @@ import Window from '../neutralino/Window';
import Game from '../Game'; import Game from '../Game';
import Patch from '../Patch'; import Patch from '../Patch';
import Voice from '../Voice'; import Voice from '../Voice';
import Runners from '../core/Runners';
import { DebugThread } from '../core/Debug';
import DXVK from '../core/DXVK';
declare const Neutralino; declare const Neutralino;
@ -15,15 +18,17 @@ export default class State
public launchButton: HTMLElement; public launchButton: HTMLElement;
public predownloadButton: HTMLElement; public predownloadButton: HTMLElement;
public settingsButton: HTMLElement;
protected _state: LauncherState = 'game-launch-available'; protected _state: LauncherState = 'game-launch-available';
protected events = { protected events = {
'runner-installation-required': import('./states/InstallWine'),
'dxvk-installation-required': import('./states/InstallDXVK'),
'game-launch-available': import('./states/Launch'), 'game-launch-available': import('./states/Launch'),
'game-installation-available': import('./states/Install'), 'game-installation-available': import('./states/Install'),
'game-update-available': import('./states/Install'), 'game-update-available': import('./states/Install'),
'game-voice-update-required': import('./states/InstallVoice'), 'game-voice-update-required': import('./states/InstallVoice'),
'test-patch-available': import('./states/ApplyPatch'), 'test-patch-available': import('./states/ApplyPatch'),
@ -36,16 +41,19 @@ export default class State
this.launchButton = <HTMLElement>document.getElementById('launch'); this.launchButton = <HTMLElement>document.getElementById('launch');
this.predownloadButton = <HTMLElement>document.getElementById('predownload'); this.predownloadButton = <HTMLElement>document.getElementById('predownload');
this.settingsButton = <HTMLElement>document.getElementById('settings');
this.launchButton.onclick = () => { this.launchButton.onclick = () => {
if (this.events[this._state]) if (this.events[this._state])
{ {
this.launchButton.style['display'] = 'none'; this.launchButton.style['display'] = 'none';
this.settingsButton.style['display'] = 'none';
this.events[this._state].then((event) => { this.events[this._state].then((event) => {
event.default(this.launcher).then(() => { event.default(this.launcher).then(() => {
this.update().then(() => { this.update().then(() => {
this.launchButton.style['display'] = 'block'; this.launchButton.style['display'] = 'block';
this.settingsButton.style['display'] = 'block';
}); });
}); });
}); });
@ -55,6 +63,7 @@ export default class State
this.predownloadButton.onclick = () => { this.predownloadButton.onclick = () => {
this.launchButton.style['display'] = 'none'; this.launchButton.style['display'] = 'none';
this.predownloadButton.style['display'] = 'none'; this.predownloadButton.style['display'] = 'none';
this.settingsButton.style['display'] = 'none';
const module = this._state === 'game-pre-installation-available' ? const module = this._state === 'game-pre-installation-available' ?
'Predownload' : 'PredownloadVoice'; 'Predownload' : 'PredownloadVoice';
@ -63,6 +72,7 @@ export default class State
module.default(this.launcher).then(() => { module.default(this.launcher).then(() => {
this.update().then(() => { this.update().then(() => {
this.launchButton.style['display'] = 'block'; this.launchButton.style['display'] = 'block';
this.settingsButton.style['display'] = 'block';
}); });
}); });
}); });
@ -95,6 +105,16 @@ export default class State
switch(state) switch(state)
{ {
case 'runner-installation-required':
this.launchButton.textContent = 'Install wine';
break;
case 'dxvk-installation-required':
this.launchButton.textContent = 'Install DXVK';
break;
case 'game-launch-available': case 'game-launch-available':
this.launchButton.textContent = 'Launch'; this.launchButton.textContent = 'Launch';
@ -148,9 +168,84 @@ export default class State
*/ */
public update(): Promise<string> public update(): Promise<string>
{ {
return new Promise(async (resolve) => { const debugThread = new DebugThread('State.update', 'Updating launcher state');
let state: LauncherState;
return new Promise(async (resolve) => {
let state: LauncherState|null = null;
const runner = await Runners.current();
const dxvk = await DXVK.current();
// Check if the wine is installed
if (runner === null)
{
debugThread.log('Runner is not specified');
state = 'runner-installation-required';
Runners.list().then((list) => {
for (const family of list)
for (const runner of family.runners)
if (runner.installed && runner.recommended)
{
debugThread.log(`Automatically selected runner ${runner.title} (${runner.name})`);
state = null;
Runners.current(runner).then(() => {
this.update().then(resolve);
});
return;
}
});
if (state !== null)
{
debugThread.log('No recommended runner installed');
this.set(state);
resolve(state);
}
}
// Check if the DXVK is installed
else if (dxvk === null)
{
debugThread.log('DXVK is not specified');
state = 'dxvk-installation-required';
DXVK.list().then((list) => {
for (const dxvk of list)
if (dxvk.installed && dxvk.recommended)
{
debugThread.log(`Automatically selected DXVK ${dxvk.version}`);
state = null;
DXVK.current(dxvk).then(() => {
this.update().then(resolve);
});
return;
}
});
if (state !== null)
{
debugThread.log('No recommended DXVK installed');
this.set(state);
resolve(state);
}
}
// Otherwise select some launcher state
else
{
const gameCurrent = await Game.current; const gameCurrent = await Game.current;
const gameLatest = await Game.getLatestData(); const gameLatest = await Game.getLatestData();
const patch = await Patch.latest; const patch = await Patch.latest;
@ -181,9 +276,12 @@ export default class State
else state = 'game-launch-available'; else state = 'game-launch-available';
debugThread.log(`Updated state: ${state}`);
this.set(state); this.set(state);
resolve(state); resolve(state);
}
}); });
} }
}; };

View file

@ -7,7 +7,7 @@ export default (launcher: Launcher): Promise<void> => {
return new Promise(async (resolve) => { return new Promise(async (resolve) => {
const prefixDir = await constants.paths.prefix.current; const prefixDir = await constants.paths.prefix.current;
Prefix.exists().then((exists) => { Prefix.exists(prefixDir).then((exists) => {
if (exists) if (exists)
resolve(); resolve();
@ -28,23 +28,12 @@ export default (launcher: Launcher): Promise<void> => {
Prefix.create(prefixDir, (output, current, total) => { Prefix.create(prefixDir, (output, current, total) => {
progressLabel = output; progressLabel = output;
if (progressLabel.length > 70) if (progressLabel.length > 80)
progressLabel = progressLabel.substring(0, 70) + '...'; progressLabel = progressLabel.substring(0, 80) + '...';
launcher.progressBar!.update(current, total, 1); launcher.progressBar!.update(current, total, 1);
}) })
.then((result) => { .then(() => resolve());
if (result === true)
resolve();
else
{
// TODO
console.error('There\'s no wine version installed to use to create the prefix');
resolve();
}
});
} }
}); });
}); });

View file

@ -0,0 +1,56 @@
import type Launcher from '../../Launcher';
import DXVK from '../../core/DXVK';
import constants from '../../Constants';
export default (launcher: Launcher): Promise<void> => {
return new Promise(async (resolve) => {
// Create prefix if it is not created
import('./CreatePrefix').then((module) => {
module.default(launcher).then(() => {
// And then download the DXVK
DXVK.download('1.9.2').then((stream) => {
launcher.progressBar?.init({
label: 'Downloading DXVK 1.9.2...',
showSpeed: true,
showEta: true,
showPercents: true,
showTotals: true
});
stream?.downloadStart(() => launcher.progressBar?.show());
stream?.downloadProgress((current: number, total: number, difference: number) => {
launcher.progressBar?.update(current, total, difference);
});
let unpacking = true;
stream?.unpackStart(() => {
launcher.progressBar?.init({
label: () => unpacking ? 'Unpacking DXVK 1.9.2...' : 'Applying DXVK 1.9.2...',
showSpeed: true,
showEta: true,
showPercents: true,
showTotals: true
});
});
stream?.unpackProgress((current: number, total: number, difference: number) => {
launcher.progressBar?.update(current, total, difference);
});
stream?.unpackFinish(async () => {
unpacking = true;
DXVK.apply(await constants.paths.prefix.current, '1.9.2').then(() => {
launcher.progressBar?.hide();
resolve();
});
});
});
});
});
});
};

View file

@ -0,0 +1,57 @@
import type Launcher from '../../Launcher';
import Runners from '../../core/Runners';
import DXVK from '../../core/DXVK';
export default (launcher: Launcher): Promise<void> => {
return new Promise(async (resolve) => {
Runners.download('lutris-ge-6.21-1-x86_64').then((stream) => {
launcher.progressBar?.init({
label: 'Downloading Wine-GE 6.21-1...',
showSpeed: true,
showEta: true,
showPercents: true,
showTotals: true
});
stream?.downloadStart(() => launcher.progressBar?.show());
stream?.downloadProgress((current: number, total: number, difference: number) => {
launcher.progressBar?.update(current, total, difference);
});
stream?.unpackStart(() => {
launcher.progressBar?.init({
label: 'Unpacking Wine-GE 6.21-1...',
showSpeed: true,
showEta: true,
showPercents: true,
showTotals: true
});
});
stream?.unpackProgress((current: number, total: number, difference: number) => {
launcher.progressBar?.update(current, total, difference);
});
stream?.unpackFinish(() => {
// Create prefix if it is not created
import('./CreatePrefix').then((module) => {
module.default(launcher).then(() => {
// Download DXVK if it wasn't downloaded
DXVK.current().then((dxvk) => {
if (dxvk === null)
{
import('./InstallDXVK').then((module) => {
module.default(launcher).then(() => resolve());
});
}
else resolve();
});
});
});
});
});
});
};

View file

@ -144,7 +144,7 @@ export default (): Promise<void> => {
Window.current.show(); Window.current.show();
// todo // TODO
resolve(); resolve();
}); });

View file

@ -13,6 +13,8 @@
*/ */
type LauncherState = type LauncherState =
| 'runner-installation-required'
| 'dxvk-installation-required'
| 'patch-unavailable' | 'patch-unavailable'
| 'test-patch-available' | 'test-patch-available'
| 'patch-available' | 'patch-available'