mirror of
https://github.com/an-anime-team/an-anime-game-launcher.git
synced 2025-04-04 07:45:14 +03:00
2.2.0-beta4
- added Marie Piontek as a project contributor - added launcher icon - made background picture caching - made automatical patch downloading - added social media buttons from the official launcher Thanks Maroxy for all these enhancements + pull request #6
This commit is contained in:
parent
2015b831b0
commit
9801cf90df
9 changed files with 72 additions and 68 deletions
12
README.md
12
README.md
|
@ -8,7 +8,7 @@
|
||||||
|
|
||||||
| Game version | Launcher version | Patch version |
|
| Game version | Launcher version | Patch version |
|
||||||
| :---: | :---: | :---: |
|
| :---: | :---: | :---: |
|
||||||
| 2.2.0 | 2.2.0-beta3 ⚠️ | 2.2.0-stable ✅ |
|
| 2.2.0 | 2.2.0-beta4 ⚠️ | 2.2.0-stable ✅ |
|
||||||
|
|
||||||
Download from [Releases](https://notabug.org/nobody/an-anime-game-launcher/releases)
|
Download from [Releases](https://notabug.org/nobody/an-anime-game-launcher/releases)
|
||||||
|
|
||||||
|
@ -41,13 +41,17 @@ npm start
|
||||||
|
|
||||||
To 2.2.0-release1
|
To 2.2.0-release1
|
||||||
|
|
||||||
* <s>Fix AppImage builds</s>
|
* <s>Fix AppImage builds</s> *(beta 3)*
|
||||||
* <s>Parse background banners from the game's API ([issue #1](https://notabug.org/nobody/an-anime-game-launcher/issues/1), [pull request #2](https://notabug.org/nobody/an-anime-game-launcher/pulls/2))</s>
|
* <s>Parse background banners from the game's API ([issue #1](https://notabug.org/nobody/an-anime-game-launcher/issues/1), [pull request #2](https://notabug.org/nobody/an-anime-game-launcher/pulls/2))</s> *(beta 3)*
|
||||||
|
* <s>Update launcher logo</s> *(beta 4)*
|
||||||
|
* <s>Cache launcher background picture ([pull request #6](https://notabug.org/nobody/an-anime-game-launcher/pulls/6))</s> *(beta 4)*
|
||||||
* Make Proton-GE default compatibility tool and fix game input issues
|
* Make Proton-GE default compatibility tool and fix game input issues
|
||||||
* Add additional telemetry checking
|
* Add additional telemetry checking
|
||||||
* Add preferences menu
|
* Add preferences menu
|
||||||
* Add launcher updates notifications
|
* Add launcher updates notifications
|
||||||
* Cache launcher background picture
|
* Make automatical patch state parsing
|
||||||
|
|
||||||
|
And don't forget to change the patch's URI when it will be changed
|
||||||
|
|
||||||
<br>
|
<br>
|
||||||
|
|
||||||
|
|
2
entry.js
2
entry.js
|
@ -33,9 +33,9 @@ function createWindow ()
|
||||||
// open URLs in Browser instead of an pop-up in electron app.
|
// open URLs in Browser instead of an pop-up in electron app.
|
||||||
mainWindow.webContents.setWindowOpenHandler(({ url }) => {
|
mainWindow.webContents.setWindowOpenHandler(({ url }) => {
|
||||||
shell.openExternal(url);
|
shell.openExternal(url);
|
||||||
|
|
||||||
return { action: 'deny' };
|
return { action: 'deny' };
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
// mainWindow.webContents.openDevTools();
|
// mainWindow.webContents.openDevTools();
|
||||||
}
|
}
|
||||||
|
|
10
package.json
10
package.json
|
@ -1,9 +1,11 @@
|
||||||
{
|
{
|
||||||
"name": "an-anime-game-linux-launcher",
|
"name": "an-anime-game-linux-launcher",
|
||||||
"version": "2.2.0-beta3",
|
"version": "2.2.0-beta4",
|
||||||
"description": "An Anime Game Linux Launcher",
|
"description": "An Anime Game Linux Launcher",
|
||||||
"author": "Nikita Podvirnyy (https://notabug.org/nobody)",
|
"author": "Nikita Podvirnyy <suimin.tu.mu.ga.mi@gmail.com>",
|
||||||
"contributors": ["Marie Piontek <marie@mxy.gg> (https://marie.omg.lol)"],
|
"contributors": [
|
||||||
|
"Marie Piontek <marie@mxy.gg> (https://marie.omg.lol)"
|
||||||
|
],
|
||||||
"license": "GPL-3.0",
|
"license": "GPL-3.0",
|
||||||
"main": "entry.js",
|
"main": "entry.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
@ -23,7 +25,7 @@
|
||||||
"entry.js"
|
"entry.js"
|
||||||
],
|
],
|
||||||
"linux": {
|
"linux": {
|
||||||
"icon": "public/images/icon512.png",
|
"icon": "public/images/icon256.png",
|
||||||
"category": "Utility",
|
"category": "Utility",
|
||||||
"target": [
|
"target": [
|
||||||
{
|
{
|
||||||
|
|
BIN
public/images/icon256.png
Normal file
BIN
public/images/icon256.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 76 KiB |
Binary file not shown.
Before Width: | Height: | Size: 26 KiB |
Binary file not shown.
Before Width: | Height: | Size: 2.8 KiB After Width: | Height: | Size: 20 KiB |
27
src/ts/GIJSON.d.ts
vendored
27
src/ts/GIJSON.d.ts
vendored
|
@ -1,4 +1,5 @@
|
||||||
interface VoicePack {
|
interface VoicePack
|
||||||
|
{
|
||||||
language: string;
|
language: string;
|
||||||
name: string;
|
name: string;
|
||||||
path: string;
|
path: string;
|
||||||
|
@ -6,7 +7,8 @@ interface VoicePack {
|
||||||
md5: string;
|
md5: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Latest {
|
interface Latest
|
||||||
|
{
|
||||||
name: string;
|
name: string;
|
||||||
version: string;
|
version: string;
|
||||||
path: string;
|
path: string;
|
||||||
|
@ -18,7 +20,8 @@ interface Latest {
|
||||||
segments: any[];
|
segments: any[];
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Diff {
|
interface Diff
|
||||||
|
{
|
||||||
name: string;
|
name: string;
|
||||||
version: string;
|
version: string;
|
||||||
path: string;
|
path: string;
|
||||||
|
@ -28,12 +31,14 @@ interface Diff {
|
||||||
voice_packs: VoicePack[];
|
voice_packs: VoicePack[];
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Game {
|
interface Game
|
||||||
|
{
|
||||||
latest: Latest;
|
latest: Latest;
|
||||||
diffs: Diff[];
|
diffs: Diff[];
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Plugins {
|
interface Plugins
|
||||||
|
{
|
||||||
name: string;
|
name: string;
|
||||||
version: string;
|
version: string;
|
||||||
path: string;
|
path: string;
|
||||||
|
@ -42,17 +47,20 @@ interface Plugins {
|
||||||
entry: string;
|
entry: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Plugin {
|
interface Plugin
|
||||||
|
{
|
||||||
plugins: Plugins[];
|
plugins: Plugins[];
|
||||||
version: string;
|
version: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface DeprecatedPackage {
|
interface DeprecatedPackage
|
||||||
|
{
|
||||||
name: string;
|
name: string;
|
||||||
md5: string;
|
md5: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Data {
|
interface Data
|
||||||
|
{
|
||||||
game: Game;
|
game: Game;
|
||||||
plugin: Plugin;
|
plugin: Plugin;
|
||||||
web_url: string;
|
web_url: string;
|
||||||
|
@ -62,7 +70,8 @@ interface Data {
|
||||||
sdk?: any;
|
sdk?: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default interface GIJSON {
|
export default interface GIJSON
|
||||||
|
{
|
||||||
retcode: number;
|
retcode: number;
|
||||||
message: string;
|
message: string;
|
||||||
data: Data;
|
data: Data;
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import GIJSON from "./GIJSON";
|
import GIJSON from './GIJSON';
|
||||||
|
|
||||||
const https = require('https');
|
const https = require('https');
|
||||||
const fs = require('fs');
|
const fs = require('fs');
|
||||||
|
@ -13,7 +13,7 @@ type Config = {
|
||||||
},
|
},
|
||||||
background: {
|
background: {
|
||||||
time: string|null,
|
time: string|null,
|
||||||
name: string|null
|
file: string|null
|
||||||
},
|
},
|
||||||
version: string|null,
|
version: string|null,
|
||||||
patch: {
|
patch: {
|
||||||
|
@ -32,12 +32,13 @@ export class Genshinlib
|
||||||
public static readonly launcherDir: string = path.join(os.homedir(), 'genshin-impact-launcher');
|
public static readonly launcherDir: string = path.join(os.homedir(), 'genshin-impact-launcher');
|
||||||
public static readonly launcherJson: string = path.join(this.launcherDir, 'launcher.json');
|
public static readonly launcherJson: string = path.join(this.launcherDir, 'launcher.json');
|
||||||
|
|
||||||
public static readonly TMPpatchDir: string = path.join(this.launcherDir, 'gi-on-linux');
|
public static readonly tmpPatchDir: string = path.join(this.launcherDir, 'gi-on-linux');
|
||||||
|
|
||||||
public static readonly prefixDir: string = path.join(this.launcherDir, 'game');
|
public static readonly prefixDir: string = path.join(this.launcherDir, 'game');
|
||||||
public static readonly gameDir: string = path.join(this.prefixDir, 'drive_c', 'Program Files', 'Genshin Impact');
|
public static readonly gameDir: string = path.join(this.prefixDir, 'drive_c', 'Program Files', 'Genshin Impact');
|
||||||
|
|
||||||
protected static uri: string = 'https://sdk-os-static.mihoyo.com/hk4e_global/mdk/launcher/api/resource?key=gcStgarh&launcher_id=10';
|
protected static versionsUri: string = 'https://sdk-os-static.mihoyo.com/hk4e_global/mdk/launcher/api/resource?key=gcStgarh&launcher_id=10';
|
||||||
|
protected static backgroundUri: string = 'https://sdk-os-static.mihoyo.com/hk4e_global/mdk/launcher/api/content?filter_adv=true&launcher_id=10&language=';
|
||||||
|
|
||||||
public static get version(): Config['version']
|
public static get version(): Config['version']
|
||||||
{
|
{
|
||||||
|
@ -78,7 +79,7 @@ export class Genshinlib
|
||||||
public static async getData (): Promise<any>
|
public static async getData (): Promise<any>
|
||||||
{
|
{
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
https.get(this.uri, (response: any) => {
|
https.get(this.versionsUri, (response: any) => {
|
||||||
let data = '';
|
let data = '';
|
||||||
|
|
||||||
response.on('data', (chunk: any) => data += chunk);
|
response.on('data', (chunk: any) => data += chunk);
|
||||||
|
@ -100,36 +101,36 @@ export class Genshinlib
|
||||||
|
|
||||||
if (!this.getConfig().background.time || new Date(new Date().setHours(0,0,0,0)).setDate(new Date(new Date().setHours(0,0,0,0)).getDate()).toString() >= this.getConfig().background.time!)
|
if (!this.getConfig().background.time || new Date(new Date().setHours(0,0,0,0)).setDate(new Date(new Date().setHours(0,0,0,0)).getDate()).toString() >= this.getConfig().background.time!)
|
||||||
{
|
{
|
||||||
await fetch(`https://sdk-os-static.mihoyo.com/hk4e_global/mdk/launcher/api/content?filter_adv=true&launcher_id=10&language=${this.lang.launcher}`)
|
await fetch(this.backgroundUri + this.lang.launcher)
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(async resdone => {
|
.then(async resdone => {
|
||||||
let oldbg = this.getConfig().background.name;
|
let oldbg = this.getConfig().background.file;
|
||||||
|
|
||||||
this.setConfig({
|
this.setConfig({
|
||||||
...this.getConfig(),
|
...this.getConfig(),
|
||||||
background: {
|
background: {
|
||||||
time: new Date(new Date().setHours(0,0,0,0)).setDate(new Date(new Date().setHours(0,0,0,0)).getDate() + 7).toString(),
|
time: new Date(new Date().setHours(0,0,0,0)).setDate(new Date(new Date().setHours(0,0,0,0)).getDate() + 7).toString(),
|
||||||
name: resdone.data.adv.background.replace(/.*\//, '')
|
file: resdone.data.adv.background.replace(/.*\//, '')
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
if (fs.existsSync(path.join(this.launcherDir, this.getConfig().background.name)))
|
if (fs.existsSync(path.join(this.launcherDir, this.getConfig().background.file)))
|
||||||
{
|
bg = path.join(this.launcherDir, this.getConfig().background.file);
|
||||||
bg = path.join(this.launcherDir, this.getConfig().background.name);
|
|
||||||
}
|
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
await this.downloadFile(resdone.data.adv.background, path.join(this.launcherDir, this.getConfig().background.name), (current: number, total: number, difference: number) => null).then(() => {
|
await this.downloadFile(resdone.data.adv.background, path.join(this.launcherDir, this.getConfig().background.file), (current: number, total: number, difference: number) => null).then(() => {
|
||||||
!oldbg ? console.log('No Old Background Found.') : fs.unlinkSync(path.join(this.launcherDir, oldbg));
|
!oldbg ?
|
||||||
bg = path.join(this.launcherDir, this.getConfig().background.name);
|
console.log('No old background found') :
|
||||||
|
fs.unlinkSync(path.join(this.launcherDir, oldbg));
|
||||||
|
|
||||||
|
bg = path.join(this.launcherDir, this.getConfig().background.file);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
else bg = path.join(this.launcherDir, this.getConfig().background.file);
|
||||||
bg = path.join(this.launcherDir, this.getConfig().background.name);
|
|
||||||
};
|
|
||||||
|
|
||||||
return bg;
|
return bg;
|
||||||
}
|
}
|
||||||
|
@ -155,9 +156,7 @@ export class Genshinlib
|
||||||
});
|
});
|
||||||
|
|
||||||
response.on('end', () => resolve());
|
response.on('end', () => resolve());
|
||||||
}).on('error', (err: Error) => {
|
}).on('error', (err: Error) => reject(err));
|
||||||
reject(err);
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -177,15 +176,13 @@ export class Genshinlib
|
||||||
line = line.slice(0, -1);
|
line = line.slice(0, -1);
|
||||||
|
|
||||||
let matches = /^(\d+) [a-zA-Z\:]+[ ]+(\d+)[ ]+[0-9\-]+% [0-9\-]+ [0-9\:]+ [a-f0-9]{8} (.+)/.exec(line);
|
let matches = /^(\d+) [a-zA-Z\:]+[ ]+(\d+)[ ]+[0-9\-]+% [0-9\-]+ [0-9\:]+ [a-f0-9]{8} (.+)/.exec(line);
|
||||||
if (matches) {
|
|
||||||
|
if (matches)
|
||||||
return {
|
return {
|
||||||
path: matches[3],
|
path: matches[3],
|
||||||
|
|
||||||
compressedSize: parseInt(matches[2]),
|
compressedSize: parseInt(matches[2]),
|
||||||
|
|
||||||
uncompressedSize: parseInt(matches[1])
|
uncompressedSize: parseInt(matches[1])
|
||||||
};
|
};
|
||||||
};
|
|
||||||
});
|
});
|
||||||
|
|
||||||
let total = fs.statSync(zipPath)['size'], current = 0;
|
let total = fs.statSync(zipPath)['size'], current = 0;
|
||||||
|
@ -268,21 +265,23 @@ export class Genshinlib
|
||||||
return fs.existsSync(path.join(prefixPath, 'drive_c'));
|
return fs.existsSync(path.join(prefixPath, 'drive_c'));
|
||||||
}
|
}
|
||||||
|
|
||||||
public static patchGame (version: string, onFinish: () => void, onData: (data: string) => void) {
|
public static patchGame (version: string, onFinish: () => void, onData: (data: string) => void)
|
||||||
|
{
|
||||||
this.downloadFile('https://notabug.org/Krock/GI-on-Linux/archive/master.zip', path.join(this.launcherDir, 'krock.zip'), (current: number, total: number, difference: number) => null).then(() => {
|
this.downloadFile('https://notabug.org/Krock/GI-on-Linux/archive/master.zip', path.join(this.launcherDir, 'krock.zip'), (current: number, total: number, difference: number) => null).then(() => {
|
||||||
this.unzip(path.join(this.launcherDir, 'krock.zip'), this.launcherDir, (current: number, total: number, difference: number) => null).then(() => {
|
this.unzip(path.join(this.launcherDir, 'krock.zip'), this.launcherDir, (current: number, total: number, difference: number) => null).then(() => {
|
||||||
// Delete zip file and assign patch directory.
|
// Delete zip file and assign patch directory.
|
||||||
fs.unlinkSync(path.join(this.launcherDir, 'krock.zip'));
|
fs.unlinkSync(path.join(this.launcherDir, 'krock.zip'));
|
||||||
let patchdir: string = path.join(this.TMPpatchDir, version.replace(/\./g, ''));
|
|
||||||
|
let patchDir: string = path.join(this.tmpPatchDir, version.replace(/\./g, ''));
|
||||||
|
|
||||||
// Patch out the testing phase content from the shell files if active and make sure the shell files are executable.
|
// Patch out the testing phase content from the shell files if active and make sure the shell files are executable.
|
||||||
exec(`cd ${patchdir} && sed -i '/^echo "If you would like to test this patch, modify this script and remove the line below this one."/,+5d' patch.sh`);
|
exec(`cd ${patchDir} && sed -i '/^echo "If you would like to test this patch, modify this script and remove the line below this one."/,+5d' patch.sh`);
|
||||||
exec(`cd ${patchdir} && sed -i '/^echo " necessary afterwards (Friday?). If that's the case, comment the line below."/,+2d' patch_anti_logincrash.sh`);
|
exec(`cd ${patchDir} && sed -i '/^echo " necessary afterwards (Friday?). If that's the case, comment the line below."/,+2d' patch_anti_logincrash.sh`);
|
||||||
exec(`chmod +x ${path.join(patchdir, 'patch.sh')}`);
|
exec(`chmod +x ${path.join(patchDir, 'patch.sh')}`);
|
||||||
exec(`chmod +x ${path.join(patchdir, 'patch_anti_logincrash.sh')}`);
|
exec(`chmod +x ${path.join(patchDir, 'patch_anti_logincrash.sh')}`);
|
||||||
|
|
||||||
// Execute the patch file with "yes yes" in the beginning to agree to the choices.
|
// Execute the patch file with "yes yes" in the beginning to agree to the choices.
|
||||||
let patcherProcess = exec(`yes yes | ${path.join(patchdir, 'patch.sh')}`, {
|
let patcherProcess = exec(`yes yes | ${path.join(patchDir, 'patch.sh')}`, {
|
||||||
cwd: this.gameDir,
|
cwd: this.gameDir,
|
||||||
env: {
|
env: {
|
||||||
...process.env,
|
...process.env,
|
||||||
|
@ -294,7 +293,7 @@ export class Genshinlib
|
||||||
|
|
||||||
patcherProcess.on('close', () => {
|
patcherProcess.on('close', () => {
|
||||||
// Execute the patch file with "yes" in the beginning to agree to the choice.
|
// Execute the patch file with "yes" in the beginning to agree to the choice.
|
||||||
let patcherAntiCrashProcess = exec(`yes | ${path.join(patchdir, 'patch_anti_logincrash.sh')}`, {
|
let patcherAntiCrashProcess = exec(`yes | ${path.join(patchDir, 'patch_anti_logincrash.sh')}`, {
|
||||||
cwd: this.gameDir,
|
cwd: this.gameDir,
|
||||||
env: {
|
env: {
|
||||||
...process.env,
|
...process.env,
|
||||||
|
@ -305,7 +304,7 @@ export class Genshinlib
|
||||||
patcherAntiCrashProcess.stdout.on('data', (data: string) => onData(data));
|
patcherAntiCrashProcess.stdout.on('data', (data: string) => onData(data));
|
||||||
|
|
||||||
patcherAntiCrashProcess.on('close', () => {
|
patcherAntiCrashProcess.on('close', () => {
|
||||||
fs.rmSync(this.TMPpatchDir, { recursive: true });
|
fs.rmSync(this.tmpPatchDir, { recursive: true });
|
||||||
|
|
||||||
onFinish();
|
onFinish();
|
||||||
});
|
});
|
||||||
|
@ -313,8 +312,8 @@ export class Genshinlib
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
/*
|
|
||||||
public static applyPatch (onFinish: () => void, onData: (data: string) => void)
|
/*public static applyPatch (onFinish: () => void, onData: (data: string) => void)
|
||||||
{
|
{
|
||||||
let patcherProcess = spawn('bash', [Genshinlib.patchSh], {
|
let patcherProcess = spawn('bash', [Genshinlib.patchSh], {
|
||||||
cwd: Genshinlib.gameDir,
|
cwd: Genshinlib.gameDir,
|
||||||
|
|
|
@ -17,23 +17,13 @@ $(() => {
|
||||||
|
|
||||||
Genshinlib.getBackgroundUri().then(uri => $('body').css('background-image', `url(${ uri })`));
|
Genshinlib.getBackgroundUri().then(uri => $('body').css('background-image', `url(${ uri })`));
|
||||||
|
|
||||||
// Don't really think we need this, but maybe in future~~
|
|
||||||
|
|
||||||
fetch(`https://genshin.mihoyo.com/launcher/10/${ Genshinlib.getConfig().lang.launcher }?api_url=https%3A%2F%2Fapi-os-takumi.mihoyo.com%2Fhk4e_global&prev=false`)
|
fetch(`https://genshin.mihoyo.com/launcher/10/${ Genshinlib.getConfig().lang.launcher }?api_url=https%3A%2F%2Fapi-os-takumi.mihoyo.com%2Fhk4e_global&prev=false`)
|
||||||
.then(res => res.text())
|
.then(res => res.text())
|
||||||
.then(body => {
|
.then(body => {
|
||||||
// Get the element which should have the launchers content.
|
$(body).find('#__layout').appendTo('#launchcontent');
|
||||||
let container = document.getElementById('launchcontent');
|
|
||||||
|
|
||||||
// Parse the HTML and append the HTML from the site to the div.
|
$('#launchcontent .home__main .home-swiper-wrap').remove();
|
||||||
let parser = new DOMParser();
|
$('#launchcontent .home__main .home-news').remove();
|
||||||
let converted = parser.parseFromString(body, 'text/html');
|
|
||||||
container!.appendChild(converted.getElementById('__layout')!);
|
|
||||||
|
|
||||||
// Get the home container to remove the functions that don't work currently.
|
|
||||||
let home = container!.getElementsByClassName('home__main')[0];
|
|
||||||
home.removeChild(home.getElementsByClassName('home-swiper-wrap')[0]);
|
|
||||||
home.removeChild(home.getElementsByClassName('home-news')[0]);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
Genshinlib.getData().then(data => {
|
Genshinlib.getData().then(data => {
|
||||||
|
|
Loading…
Add table
Reference in a new issue