fchat-rising/electron/Index.vue

418 lines
17 KiB
Vue
Raw Normal View History

2017-09-02 01:50:31 +00:00
<template>
2020-04-03 23:48:39 +00:00
<div @mouseover="onMouseOver" id="page" style="position:relative;padding:5px 10px 10px" :class="getThemeClass()" @auxclick.prevent>
2017-09-02 01:50:31 +00:00
<div v-html="styling"></div>
<div v-if="!characters" style="display:flex; align-items:center; justify-content:center; height: 100%;">
<div class="card bg-light" style="width: 400px;">
2020-06-30 21:51:06 +00:00
<div class="initializer" v-show="showSpinner">
<div class="title">
Getting ready, please wait...
</div>
<i class="fas fa-circle-notch fa-spin search-spinner"></i>
</div>
2019-09-17 17:14:14 +00:00
<h3 class="card-header" style="margin-top:0;display:flex">
{{l('title')}}
2020-06-30 21:51:06 +00:00
2019-09-17 17:14:14 +00:00
<a href="#" @click.prevent="showLogs()" class="btn" style="flex:1;text-align:right">
<span class="fa fa-file-alt"></span> <span class="btn-text">{{l('logs.title')}}</span>
</a>
</h3>
<div class="card-body">
<div class="alert alert-danger" v-show="error">
{{error}}
</div>
<div class="form-group">
<label class="control-label" for="account">{{l('login.account')}}</label>
2019-09-17 17:14:14 +00:00
<input class="form-control" id="account" v-model="settings.account" @keypress.enter="login()"
:disabled="loggingIn"/>
</div>
<div class="form-group">
<label class="control-label" for="password">{{l('login.password')}}</label>
2019-09-17 17:14:14 +00:00
<input class="form-control" type="password" id="password" v-model="password" @keypress.enter="login()"
:disabled="loggingIn"/>
</div>
<div class="form-group" v-show="showAdvanced">
<label class="control-label" for="host">{{l('login.host')}}</label>
<div class="input-group">
2019-01-03 17:38:17 +00:00
<input class="form-control" id="host" v-model="settings.host" @keypress.enter="login()" :disabled="loggingIn"/>
<div class="input-group-append">
2019-09-17 17:14:14 +00:00
<button class="btn btn-outline-secondary" @click="resetHost()"><span class="fas fa-undo-alt"></span>
</button>
</div>
</div>
</div>
<div class="form-group">
<label for="advanced"><input type="checkbox" id="advanced" v-model="showAdvanced"/> {{l('login.advanced')}}</label>
</div>
<div class="form-group">
<label for="save"><input type="checkbox" id="save" v-model="saveLogin"/> {{l('login.save')}}</label>
</div>
<div class="form-group" style="margin:0;text-align:right">
<button class="btn btn-primary" @click="login" :disabled="loggingIn">
{{l(loggingIn ? 'login.working' : 'login.submit')}}
</button>
</div>
2017-09-02 01:50:31 +00:00
</div>
</div>
</div>
<chat v-else :ownCharacters="characters" :defaultCharacter="defaultCharacter" ref="chat"></chat>
<div ref="linkPreview" class="link-preview"></div>
<modal :action="l('importer.importing')" ref="importModal" :buttons="false">
2018-03-28 13:51:05 +00:00
<span style="white-space:pre-wrap">{{l('importer.importingNote')}}</span>
2017-09-02 01:50:31 +00:00
<div class="progress" style="margin-top:5px">
<div class="progress-bar" :style="{width: importProgress * 100 + '%'}"></div>
</div>
</modal>
<modal :buttons="false" ref="profileViewer" dialogClass="profile-viewer">
2019-09-24 19:53:43 +00:00
<character-page :authenticated="true" :oldApi="true" :name="profileName" :image-preview="true" ref="characterPage"></character-page>
<template slot="title">
{{profileName}}
<a class="btn" @click="openProfileInBrowser"><i class="fa fa-external-link-alt"/></a>
<a class="btn" @click="openConversation"><i class="fa fa-comment"></i></a>
2019-09-24 19:53:43 +00:00
<a class="btn" @click="reloadCharacter"><i class="fa fa-sync" /></a>
<i class="fas fa-circle-notch fa-spin profileRefreshSpinner" v-show="isRefreshingProfile()"></i>
</template>
</modal>
2018-03-28 13:51:05 +00:00
<modal :action="l('fixLogs.action')" ref="fixLogsModal" @submit="fixLogs" buttonClass="btn-danger">
<span style="white-space:pre-wrap">{{l('fixLogs.text')}}</span>
<div class="form-group">
<label class="control-label">{{l('fixLogs.character')}}</label>
<select id="import" class="form-control" v-model="fixCharacter">
<option v-for="character in fixCharacters" :value="character">{{character}}</option>
</select>
</div>
</modal>
2019-09-17 17:14:14 +00:00
<logs ref="logsDialog"></logs>
2017-09-02 01:50:31 +00:00
</div>
</template>
<script lang="ts">
2019-01-03 17:38:17 +00:00
import {Component, Hook} from '@f-list/vue-ts';
2017-09-02 01:50:31 +00:00
import Axios from 'axios';
import * as electron from 'electron';
import log from 'electron-log'; //tslint:disable-line:match-default-export-name
2017-09-02 01:50:31 +00:00
import * as fs from 'fs';
import * as path from 'path';
import * as qs from 'querystring';
2020-04-19 17:06:50 +00:00
import Raven from 'raven-js';
2017-09-02 01:50:31 +00:00
import {promisify} from 'util';
2018-01-06 16:14:21 +00:00
import Vue from 'vue';
2017-09-02 01:50:31 +00:00
import Chat from '../chat/Chat.vue';
import {getKey, Settings} from '../chat/common';
2020-06-30 21:51:06 +00:00
import core /*, { init as initCore }*/ from '../chat/core';
2017-09-02 01:50:31 +00:00
import l from '../chat/localize';
2019-09-17 17:14:14 +00:00
import Logs from '../chat/Logs.vue';
2017-09-02 01:50:31 +00:00
import Socket from '../chat/WebSocket';
import Modal from '../components/Modal.vue';
2019-09-17 17:14:14 +00:00
import {SimpleCharacter} from '../interfaces';
import {Keys} from '../keys';
// import { BetterSqliteStore } from '../learn/store/better-sqlite3';
// import { Sqlite3Store } from '../learn/store/sqlite3';
import CharacterPage from '../site/character_page/character_page.vue';
import {defaultHost, GeneralSettings, nativeRequire} from './common';
2020-06-30 21:51:06 +00:00
import { fixLogs /*SettingsStore, Logs as FSLogs*/ } from './filesystem';
2017-09-02 01:50:31 +00:00
import * as SlimcatImporter from './importer';
2020-06-30 21:51:06 +00:00
// import Connection from '../fchat/connection';
// import Notifications from './notifications';
2017-09-02 01:50:31 +00:00
2018-01-06 16:14:21 +00:00
const webContents = electron.remote.getCurrentWebContents();
const parent = electron.remote.getCurrentWindow().webContents;
2017-09-02 01:50:31 +00:00
2019-10-27 18:58:41 +00:00
// Allow requests to imgur.com
const session = electron.remote.session;
/* tslint:disable:no-unsafe-any no-any no-unnecessary-type-assertion */
session!.defaultSession!.webRequest!.onBeforeSendHeaders(
{
urls: [
2020-03-14 21:24:49 +00:00
'https://(api|i).imgur.com/.*',
'http://(api|i).imgur.com/.*'
2019-10-27 18:58:41 +00:00
]
},
(details: any, callback: any) => {
details.requestHeaders['Origin'] = null;
details.headers['Origin'] = null;
callback({requestHeaders: details.requestHeaders});
}
);
log.info('About to load keytar');
2019-07-06 16:49:19 +00:00
/* tslint:disable: no-any no-unsafe-any */ //because this is hacky
2017-09-02 01:50:31 +00:00
const keyStore = nativeRequire<{
getPassword(account: string): Promise<string>
setPassword(account: string, password: string): Promise<void>
deletePassword(account: string): Promise<void>
[key: string]: (...args: any[]) => Promise<any>
}>('keytar/build/Release/keytar.node');
for(const key in keyStore) keyStore[key] = promisify(<(...args: any[]) => any>keyStore[key].bind(keyStore, 'fchat'));
//tslint:enable
log.info('Loaded keytar.');
2017-09-02 01:50:31 +00:00
@Component({
2019-09-17 17:14:14 +00:00
components: {chat: Chat, modal: Modal, characterPage: CharacterPage, logs: Logs}
2017-09-02 01:50:31 +00:00
})
export default class Index extends Vue {
showAdvanced = false;
saveLogin = false;
loggingIn = false;
password = '';
2019-09-17 17:14:14 +00:00
character?: string;
characters?: SimpleCharacter[];
2017-09-02 01:50:31 +00:00
error = '';
2019-09-17 17:14:14 +00:00
defaultCharacter?: number;
2017-09-02 01:50:31 +00:00
l = l;
settings!: GeneralSettings;
2017-09-02 01:50:31 +00:00
importProgress = 0;
profileName = '';
2019-07-08 23:27:14 +00:00
adName = '';
2018-03-28 13:51:05 +00:00
fixCharacters: ReadonlyArray<string> = [];
fixCharacter = '';
2017-09-02 01:50:31 +00:00
2020-06-30 21:51:06 +00:00
showSpinner = true;
2019-01-03 17:38:17 +00:00
@Hook('created')
2020-06-30 21:51:06 +00:00
async created(): Promise<void> {
// tslint:disable-next-line no-floating-promises
await core.cache.start(this.settings);
// await this.prepper;
this.showSpinner = false;
2018-01-06 16:14:21 +00:00
if(this.settings.account.length > 0) this.saveLogin = true;
keyStore.getPassword(this.settings.account)
.then((value: string) => this.password = value, (err: Error) => this.error = err.message);
2017-09-02 01:50:31 +00:00
2018-01-06 16:14:21 +00:00
Vue.set(core.state, 'generalSettings', this.settings);
2017-09-02 01:50:31 +00:00
electron.ipcRenderer.on('settings',
(_: Event, settings: GeneralSettings) => core.state.generalSettings = this.settings = settings);
2019-07-08 23:27:14 +00:00
electron.ipcRenderer.on('open-profile', (_: Event, name: string) => {
2018-01-06 16:14:21 +00:00
const profileViewer = <Modal>this.$refs['profileViewer'];
this.profileName = name;
profileViewer.show();
});
2019-07-08 23:27:14 +00:00
2018-03-28 13:51:05 +00:00
electron.ipcRenderer.on('fix-logs', async() => {
2019-09-17 17:14:14 +00:00
this.fixCharacters = await core.settingsStore.getAvailableCharacters();
2018-03-28 13:51:05 +00:00
this.fixCharacter = this.fixCharacters[0];
(<Modal>this.$refs['fixLogsModal']).show();
});
window.addEventListener('keydown', (e) => {
if(getKey(e) === Keys.Tab && e.ctrlKey && !e.altKey && !e.shiftKey)
parent.send('switch-tab', this.character);
2018-01-06 16:14:21 +00:00
});
2019-06-07 20:40:17 +00:00
/*if (process.env.NODE_ENV !== 'production') {
2019-06-07 20:40:17 +00:00
const dt = require('@vue/devtools');
dt.connect();
}*/
2019-06-07 20:40:17 +00:00
2017-09-02 01:50:31 +00:00
}
async login(): Promise<void> {
if(this.loggingIn) return;
this.loggingIn = true;
try {
2018-01-06 16:14:21 +00:00
if(!this.saveLogin) await keyStore.deletePassword(this.settings.account);
2020-06-29 22:01:06 +00:00
2018-01-06 16:14:21 +00:00
const data = <{ticket?: string, error: string, characters: {[key: string]: number}, default_character: number}>
(await Axios.post('https://www.f-list.net/json/getApiTicket.php', qs.stringify({
account: this.settings.account, password: this.password, no_friends: true, no_bookmarks: true,
new_character_list: true
}))).data;
2017-09-02 01:50:31 +00:00
if(data.error !== '') {
this.error = data.error;
return;
}
2018-03-28 13:51:05 +00:00
if(this.saveLogin) {
electron.ipcRenderer.send('save-login', this.settings.account, this.settings.host);
await keyStore.setPassword(this.settings.account, this.password);
}
2018-01-06 16:14:21 +00:00
Socket.host = this.settings.host;
2019-09-17 17:14:14 +00:00
core.connection.onEvent('connecting', async() => {
2018-01-06 16:14:21 +00:00
if(!electron.ipcRenderer.sendSync('connect', core.connection.character) && process.env.NODE_ENV === 'production') {
alert(l('login.alreadyLoggedIn'));
return core.connection.close();
}
2018-04-11 19:17:58 +00:00
parent.send('connect', webContents.id, core.connection.character);
2019-09-17 17:14:14 +00:00
this.character = core.connection.character;
2018-01-06 16:14:21 +00:00
if((await core.settingsStore.get('settings')) === undefined &&
SlimcatImporter.canImportCharacter(core.connection.character)) {
if(!confirm(l('importer.importGeneral'))) return core.settingsStore.set('settings', new Settings());
2017-09-02 01:50:31 +00:00
(<Modal>this.$refs['importModal']).show(true);
await SlimcatImporter.importCharacter(core.connection.character, (progress) => this.importProgress = progress);
(<Modal>this.$refs['importModal']).hide();
}
});
2019-09-17 17:14:14 +00:00
core.connection.onEvent('connected', () => {
2018-01-06 16:14:21 +00:00
core.watch(() => core.conversations.hasNew, (newValue) => parent.send('has-new', webContents.id, newValue));
2017-09-02 01:50:31 +00:00
Raven.setUserContext({username: core.connection.character});
});
2019-09-17 17:14:14 +00:00
core.connection.onEvent('closed', () => {
if(this.character === undefined) return;
2019-01-03 17:38:17 +00:00
electron.ipcRenderer.send('disconnect', this.character);
2018-01-06 16:14:21 +00:00
this.character = undefined;
parent.send('disconnect', webContents.id);
2017-09-02 01:50:31 +00:00
Raven.setUserContext();
});
2019-09-17 17:14:14 +00:00
core.connection.setCredentials(this.settings.account, this.password);
this.characters = Object.keys(data.characters).map((name) => ({name, id: data.characters[name], deleted: false}))
.sort((x, y) => x.name.localeCompare(y.name));
this.defaultCharacter = data.default_character;
2017-09-02 01:50:31 +00:00
} catch(e) {
this.error = l('login.error');
if(process.env.NODE_ENV !== 'production') throw e;
} finally {
this.loggingIn = false;
}
}
2018-03-28 13:51:05 +00:00
fixLogs(): void {
if(!electron.ipcRenderer.sendSync('connect', this.fixCharacter)) return alert(l('login.alreadyLoggedIn'));
try {
fixLogs(this.fixCharacter);
alert(l('fixLogs.success'));
} catch(e) {
alert(l('fixLogs.error'));
throw e;
} finally {
electron.ipcRenderer.send('disconnect', this.fixCharacter);
}
}
resetHost(): void {
this.settings.host = defaultHost;
}
2017-09-02 01:50:31 +00:00
onMouseOver(e: MouseEvent): void {
const preview = (<HTMLDivElement>this.$refs.linkPreview);
if((<HTMLElement>e.target).tagName === 'A') {
const target = <HTMLAnchorElement>e.target;
if(target.hostname !== '') {
//tslint:disable-next-line:prefer-template
preview.className = 'link-preview ' +
(e.clientX < window.innerWidth / 2 && e.clientY > window.innerHeight - 150 ? ' right' : '');
preview.textContent = target.href;
preview.style.display = 'block';
return;
}
}
preview.textContent = '';
preview.style.display = 'none';
}
2019-09-17 17:14:14 +00:00
async openProfileInBrowser(): Promise<void> {
2020-03-15 14:02:31 +00:00
await electron.remote.shell.openExternal(`https://www.f-list.net/c/${this.profileName}`);
2019-06-07 21:45:34 +00:00
2019-07-06 16:49:19 +00:00
// tslint:disable-next-line: no-any no-unsafe-any
2019-06-07 21:45:34 +00:00
(this.$refs.profileViewer as any).hide();
}
2019-06-07 21:45:34 +00:00
openConversation(): void {
//this.
// this.profileName
const character = core.characters.get(this.profileName);
const conversation = core.conversations.getPrivate(character);
conversation.show();
2019-07-06 16:49:19 +00:00
// tslint:disable-next-line: no-any no-unsafe-any
2019-06-07 21:45:34 +00:00
(this.$refs.profileViewer as any).hide();
}
2019-09-24 19:53:43 +00:00
isRefreshingProfile(): boolean {
const cp = this.$refs.characterPage as CharacterPage;
return cp && cp.refreshing;
}
2019-09-24 19:53:43 +00:00
reloadCharacter(): void {
// tslint:disable-next-line: no-any no-unsafe-any
(this.$refs.characterPage as any).reload();
}
2020-04-03 23:48:39 +00:00
getThemeClass(): string {
return `theme-${this.settings.theme}`;
}
2017-09-02 01:50:31 +00:00
get styling(): string {
try {
2020-03-15 14:02:31 +00:00
return `<style>${fs.readFileSync(path.join(__dirname, `themes/${this.settings.theme}.css`), 'utf8').toString()}</style>`;
2017-09-02 01:50:31 +00:00
} catch(e) {
2018-01-06 16:14:21 +00:00
if((<Error & {code: string}>e).code === 'ENOENT' && this.settings.theme !== 'default') {
this.settings.theme = 'default';
2017-09-02 01:50:31 +00:00
return this.styling;
}
throw e;
}
}
2019-09-17 17:14:14 +00:00
showLogs(): void {
(<Logs>this.$refs['logsDialog']).show();
}
2017-09-02 01:50:31 +00:00
}
</script>
2020-06-30 21:51:06 +00:00
<style lang="scss">
2017-09-02 01:50:31 +00:00
html, body, #page {
height: 100%;
}
2018-08-10 16:59:37 +00:00
2018-08-18 19:37:53 +00:00
a[href^="#"]:not([draggable]) {
2018-08-10 16:59:37 +00:00
-webkit-user-drag: none;
-webkit-app-region: no-drag;
}
.profileRefreshSpinner {
font-size: 12pt;
opacity: 0.5;
}
2020-06-30 21:51:06 +00:00
.initializer {
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
display: flex;
align-items: center;
justify-content: center;
transition: all 0.25s;
backdrop-filter: blur(3px) grayscale(35%);
i {
font-size: 130pt;
top: 50%;
right: 50%;
transform: translate(-50%, -50%);
width: fit-content;
}
.title {
position: absolute;
top: 0;
background: rgba(19, 19, 19, 0.6);
width: 100%;
text-align: center;
padding-top: 20px;
padding-bottom: 20px;
font-weight: bold;
}
}
</style>