2017-09-02 01:50:31 +00:00
< template >
< div id = "page" style = "position: relative; padding: 10px;" v-if ="settings" >
< div v-html ="styling" > < / div >
< div v-if ="!characters" style="display:flex; align-items:center; justify-content:center; height: 100%;" >
2018-03-04 02:32:26 +00:00
< div class = "card bg-light" style = "width: 400px;" >
< h3 class = "card-header" style = "margin-top:0" > { { l ( 'title' ) } } < / 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 >
< 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 >
< input class = "form-control" type = "password" id = "password" v -model = " settings.password " @keypress.enter ="login" :disabled ="loggingIn" / >
< / div >
< div class = "form-group" v-show ="showAdvanced" >
< label class = "control-label" for = "host" > { { l ( 'login.host' ) } } < / label >
< input class = "form-control" id = "host" v -model = " settings.host " @keypress.enter ="login" :disabled ="loggingIn" / >
< / div >
< div class = "form-group" >
< label class = "control-label" for = "theme" > { { l ( 'settings.theme' ) } } < / label >
< select class = "form-control custom-select" id = "theme" v-model ="settings.theme" >
< option > default < / option >
< option > dark < / option >
< option > light < / option >
< / select >
< / 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 = "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 >
2018-03-04 02:32:26 +00:00
< modal :buttons ="false" ref = "profileViewer" dialogClass = "profile-viewer" >
2018-03-28 13:51:05 +00:00
< character -page :authenticated ="true" :oldApi ="true" :name ="profileName" > < / c h a r a c t e r - p a g e >
< template slot = "title" > { { profileName } } < a class = "btn" @click ="openProfileInBrowser" > < i class = "fa fa-external-link-alt" > < / i > < / a >
2018-03-04 02:32:26 +00:00
< / template >
2017-12-05 01:47:27 +00:00
< / modal >
2017-09-02 01:50:31 +00:00
< / div >
< / template >
< script lang = "ts" >
import Axios from 'axios' ;
import * as qs from 'qs' ;
import * as Raven from 'raven-js' ;
import Vue from 'vue' ;
import Component from 'vue-class-component' ;
import Chat from '../chat/Chat.vue' ;
import core , { init as initCore } from '../chat/core' ;
import l from '../chat/localize' ;
2017-12-05 01:47:27 +00:00
import { init as profileApiInit } from '../chat/profile_api' ;
2017-10-16 23:58:57 +00:00
import Socket from '../chat/WebSocket' ;
2017-09-02 01:50:31 +00:00
import Modal from '../components/Modal.vue' ;
2017-10-16 23:58:57 +00:00
import Connection from '../fchat/connection' ;
2017-12-05 01:47:27 +00:00
import CharacterPage from '../site/character_page/character_page.vue' ;
2018-03-04 02:32:26 +00:00
import { appVersion , GeneralSettings , getGeneralSettings , Logs , setGeneralSettings , SettingsStore } from './filesystem' ;
2017-09-02 01:50:31 +00:00
import Notifications from './notifications' ;
2018-01-06 16:14:21 +00:00
declare global {
interface Window {
NativeView : {
setTheme ( theme : string ) : void
} | undefined ;
}
2018-03-04 02:32:26 +00:00
const NativeBackground : {
start ( ) : void
stop ( ) : void
} ;
2018-01-06 16:14:21 +00:00
}
2018-03-04 02:32:26 +00:00
function confirmBack ( e : Event ) : void {
if ( ! confirm ( l ( 'chat.confirmLeave' ) ) ) e . preventDefault ( ) ;
2017-10-18 23:29:28 +00:00
}
2017-09-02 01:50:31 +00:00
@ Component ( {
2017-12-05 01:47:27 +00:00
components : { chat : Chat , modal : Modal , characterPage : CharacterPage }
2017-09-02 01:50:31 +00:00
} )
export default class Index extends Vue {
//tslint:disable:no-null-keyword
showAdvanced = false ;
saveLogin = false ;
loggingIn = false ;
characters : ReadonlyArray < string > | null = null ;
error = '' ;
defaultCharacter : string | null = null ;
settingsStore = new SettingsStore ( ) ;
l = l ;
settings : GeneralSettings | null = null ;
2017-12-05 01:47:27 +00:00
profileName = '' ;
2017-09-02 01:50:31 +00:00
async created ( ) : Promise < void > {
2018-03-04 02:32:26 +00:00
document . addEventListener ( 'open-profile' , ( e : Event ) => {
const profileViewer = < Modal > this . $refs [ 'profileViewer' ] ;
this . profileName = ( < Event & { detail : string } > e ) . detail ;
profileViewer . show ( ) ;
} ) ;
2017-09-02 01:50:31 +00:00
let settings = await getGeneralSettings ( ) ;
if ( settings === undefined ) settings = new GeneralSettings ( ) ;
2018-03-04 02:32:26 +00:00
if ( settings . version !== appVersion ) {
alert ( 'Your beta version of F-Chat 3.0 has been updated. If you are experiencing any issues after this update, please perform a full reinstall of the application. If the issue persists, please report it.' ) ;
settings . version = appVersion ;
await setGeneralSettings ( settings ) ;
}
2017-09-02 01:50:31 +00:00
if ( settings . account . length > 0 ) this . saveLogin = true ;
this . settings = settings ;
}
get styling ( ) : string {
2018-01-06 16:14:21 +00:00
if ( window . NativeView !== undefined ) window . NativeView . setTheme ( this . settings ! . theme ) ;
2018-03-04 02:32:26 +00:00
//tslint:disable-next-line:no-require-imports
return ` <style> ${ require ( '../scss/fa.scss' ) } ${ require ( ` ../scss/themes/chat/ ${ this . settings ! . theme } .scss ` ) } </style> ` ;
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
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 . settings ! . 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-01-06 16:14:21 +00:00
if ( this . saveLogin ) await setGeneralSettings ( this . settings ! ) ;
2017-10-16 23:58:57 +00:00
Socket . host = this . settings ! . host ;
2018-03-04 02:32:26 +00:00
const connection = new Connection ( 'F-Chat 3.0 (Mobile)' , appVersion , Socket ,
2018-01-06 16:14:21 +00:00
this . settings ! . account , this . settings ! . password ) ;
2017-10-16 23:58:57 +00:00
connection . onEvent ( 'connected' , ( ) => {
Raven . setUserContext ( { username : core . connection . character } ) ;
2017-10-18 23:29:28 +00:00
document . addEventListener ( 'backbutton' , confirmBack ) ;
2018-03-04 02:32:26 +00:00
NativeBackground . start ( ) ;
2017-10-16 23:58:57 +00:00
} ) ;
connection . onEvent ( 'closed' , ( ) => {
Raven . setUserContext ( ) ;
2017-10-18 23:29:28 +00:00
document . removeEventListener ( 'backbutton' , confirmBack ) ;
2018-03-04 02:32:26 +00:00
NativeBackground . stop ( ) ;
2017-10-16 23:58:57 +00:00
} ) ;
2017-09-02 01:50:31 +00:00
initCore ( connection , Logs , SettingsStore , Notifications ) ;
2018-01-06 16:14:21 +00:00
const charNames = Object . keys ( data . characters ) ;
this . characters = charNames . sort ( ) ;
for ( const character of charNames )
if ( data . characters [ character ] === data . default _character ) {
this . defaultCharacter = character ;
break ;
}
profileApiInit ( data . characters ) ;
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-04 02:32:26 +00:00
openProfileInBrowser ( ) : void {
window . open ( ` profile:// ${ this . profileName } ` ) ;
}
2017-09-02 01:50:31 +00:00
}
< / script >
< style >
html , body , # page {
height : 100 % ;
}
2018-03-04 02:32:26 +00:00
html , . modal {
padding : env ( safe - area - inset - top ) env ( safe - area - inset - right ) env ( safe - area - inset - bottom ) env ( safe - area - inset - left ) ;
}
2018-01-06 16:14:21 +00:00
< / style >