2018-06-20 06:15:10 -04:00
// Requirements
const os = require ( 'os' )
const semver = require ( 'semver' )
2018-06-11 22:11:05 -04:00
2018-07-22 11:40:15 -04:00
const { AssetGuard } = require ( './assets/js/assetguard' )
2018-08-07 04:16:15 -04:00
const DropinModUtil = require ( './assets/js/dropinmodutil' )
2018-07-22 11:40:15 -04:00
2018-06-04 16:28:17 -04:00
const settingsState = {
invalid : new Set ( )
}
2018-05-30 22:22:17 -04:00
/ * *
* General Settings Functions
* /
2018-07-22 13:31:15 -04:00
/ * *
2018-06-04 16:28:17 -04:00
* Bind value validators to the settings UI elements . These will
* validate against the criteria defined in the ConfigManager ( if
* and ) . If the value is invalid , the UI will reflect this and saving
* will be disabled until the value is corrected . This is an automated
* process . More complex UI may need to be bound separately .
* /
function initSettingsValidators ( ) {
const sEls = document . getElementById ( 'settingsContainer' ) . querySelectorAll ( '[cValue]' )
Array . from ( sEls ) . map ( ( v , index , arr ) => {
const vFn = ConfigManager [ 'validate' + v . getAttribute ( 'cValue' ) ]
if ( typeof vFn === 'function' ) {
if ( v . tagName === 'INPUT' ) {
if ( v . type === 'number' || v . type === 'text' ) {
v . addEventListener ( 'keyup' , ( e ) => {
const v = e . target
if ( ! vFn ( v . value ) ) {
settingsState . invalid . add ( v . id )
v . setAttribute ( 'error' , '' )
settingsSaveDisabled ( true )
} else {
if ( v . hasAttribute ( 'error' ) ) {
v . removeAttribute ( 'error' )
settingsState . invalid . delete ( v . id )
if ( settingsState . invalid . size === 0 ) {
settingsSaveDisabled ( false )
}
}
}
} )
}
}
}
} )
}
/ * *
* Load configuration values onto the UI . This is an automated process .
* /
function initSettingsValues ( ) {
const sEls = document . getElementById ( 'settingsContainer' ) . querySelectorAll ( '[cValue]' )
Array . from ( sEls ) . map ( ( v , index , arr ) => {
2018-07-29 14:42:11 +02:00
const cVal = v . getAttribute ( 'cValue' )
const gFn = ConfigManager [ 'get' + cVal ]
2018-06-04 16:28:17 -04:00
if ( typeof gFn === 'function' ) {
if ( v . tagName === 'INPUT' ) {
if ( v . type === 'number' || v . type === 'text' ) {
2018-07-22 13:31:15 -04:00
// Special Conditions
2018-06-14 03:49:55 -04:00
if ( cVal === 'JavaExecutable' ) {
populateJavaExecDetails ( v . value )
2018-06-14 06:09:09 -04:00
v . value = gFn ( )
} else if ( cVal === 'JVMOptions' ) {
v . value = gFn ( ) . join ( ' ' )
} else {
v . value = gFn ( )
2018-06-14 03:49:55 -04:00
}
2018-06-04 16:28:17 -04:00
} else if ( v . type === 'checkbox' ) {
v . checked = gFn ( )
}
2018-06-11 22:11:05 -04:00
} else if ( v . tagName === 'DIV' ) {
if ( v . classList . contains ( 'rangeSlider' ) ) {
2018-06-12 03:25:36 -04:00
// Special Conditions
if ( cVal === 'MinRAM' || cVal === 'MaxRAM' ) {
let val = gFn ( )
if ( val . endsWith ( 'M' ) ) {
val = Number ( val . substring ( 0 , val . length - 1 ) ) / 1000
} else {
val = Number . parseFloat ( val )
}
v . setAttribute ( 'value' , val )
} else {
v . setAttribute ( 'value' , Number . parseFloat ( gFn ( ) ) )
}
2018-06-11 22:11:05 -04:00
}
2018-06-04 16:28:17 -04:00
}
}
} )
}
/ * *
* Save the settings values .
* /
function saveSettingsValues ( ) {
const sEls = document . getElementById ( 'settingsContainer' ) . querySelectorAll ( '[cValue]' )
Array . from ( sEls ) . map ( ( v , index , arr ) => {
2018-07-29 14:42:11 +02:00
const cVal = v . getAttribute ( 'cValue' )
const sFn = ConfigManager [ 'set' + cVal ]
2018-06-04 16:28:17 -04:00
if ( typeof sFn === 'function' ) {
if ( v . tagName === 'INPUT' ) {
if ( v . type === 'number' || v . type === 'text' ) {
2018-06-14 06:09:09 -04:00
// Special Conditions
if ( cVal === 'JVMOptions' ) {
sFn ( v . value . split ( ' ' ) )
} else {
sFn ( v . value )
}
2018-06-04 16:28:17 -04:00
} else if ( v . type === 'checkbox' ) {
sFn ( v . checked )
2018-06-04 19:34:47 -04:00
// Special Conditions
if ( cVal === 'AllowPrerelease' ) {
changeAllowPrerelease ( v . checked )
}
2018-06-04 16:28:17 -04:00
}
2018-06-12 03:25:36 -04:00
} else if ( v . tagName === 'DIV' ) {
if ( v . classList . contains ( 'rangeSlider' ) ) {
// Special Conditions
if ( cVal === 'MinRAM' || cVal === 'MaxRAM' ) {
let val = Number ( v . getAttribute ( 'value' ) )
if ( val % 1 > 0 ) {
val = val * 1000 + 'M'
} else {
val = val + 'G'
}
sFn ( val )
} else {
sFn ( v . getAttribute ( 'value' ) )
}
}
2018-06-04 16:28:17 -04:00
}
}
} )
}
2018-06-21 12:38:21 -04:00
let selectedSettingsTab = 'settingsTabAccount'
2018-05-30 16:00:07 -04:00
2018-06-20 07:38:53 -04:00
/ * *
* Modify the settings container UI when the scroll threshold reaches
* a certain poin .
*
* @ param { UIEvent } e The scroll event .
* /
function settingsTabScrollListener ( e ) {
if ( e . target . scrollTop > Number . parseFloat ( getComputedStyle ( e . target . firstElementChild ) . marginTop ) ) {
document . getElementById ( 'settingsContainer' ) . setAttribute ( 'scrolled' , '' )
} else {
document . getElementById ( 'settingsContainer' ) . removeAttribute ( 'scrolled' )
}
}
2018-05-30 22:22:17 -04:00
/ * *
* Bind functionality for the settings navigation items .
* /
2018-05-30 16:00:07 -04:00
function setupSettingsTabs ( ) {
Array . from ( document . getElementsByClassName ( 'settingsNavItem' ) ) . map ( ( val ) => {
2018-06-20 07:38:53 -04:00
if ( val . hasAttribute ( 'rSc' ) ) {
2018-06-21 12:38:21 -04:00
val . onclick = ( ) => {
settingsNavItemListener ( val )
2018-05-30 16:00:07 -04:00
}
}
} )
}
2018-06-21 12:38:21 -04:00
/ * *
* Settings nav item onclick lisener . Function is exposed so that
* other UI elements can quickly toggle to a certain tab from other views .
*
* @ param { Element } ele The nav item which has been clicked .
* @ param { boolean } fade Optional . True to fade transition .
* /
function settingsNavItemListener ( ele , fade = true ) {
if ( ele . hasAttribute ( 'selected' ) ) {
return
}
const navItems = document . getElementsByClassName ( 'settingsNavItem' )
for ( let i = 0 ; i < navItems . length ; i ++ ) {
if ( navItems [ i ] . hasAttribute ( 'selected' ) ) {
navItems [ i ] . removeAttribute ( 'selected' )
}
}
ele . setAttribute ( 'selected' , '' )
let prevTab = selectedSettingsTab
selectedSettingsTab = ele . getAttribute ( 'rSc' )
document . getElementById ( prevTab ) . onscroll = null
document . getElementById ( selectedSettingsTab ) . onscroll = settingsTabScrollListener
if ( fade ) {
$ ( ` # ${ prevTab } ` ) . fadeOut ( 250 , ( ) => {
$ ( ` # ${ selectedSettingsTab } ` ) . fadeIn ( {
duration : 250 ,
start : ( ) => {
settingsTabScrollListener ( {
target : document . getElementById ( selectedSettingsTab )
} )
}
} )
} )
} else {
$ ( ` # ${ prevTab } ` ) . hide ( 0 , ( ) => {
$ ( ` # ${ selectedSettingsTab } ` ) . show ( {
duration : 0 ,
start : ( ) => {
settingsTabScrollListener ( {
target : document . getElementById ( selectedSettingsTab )
} )
}
} )
} )
}
}
2018-06-20 06:15:10 -04:00
const settingsNavDone = document . getElementById ( 'settingsNavDone' )
2018-06-04 16:28:17 -04:00
/ * *
* Set if the settings save ( done ) button is disabled .
*
* @ param { boolean } v True to disable , false to enable .
* /
function settingsSaveDisabled ( v ) {
settingsNavDone . disabled = v
}
2018-05-30 23:32:51 -04:00
/* Closes the settings view and saves all data. */
settingsNavDone . onclick = ( ) => {
2018-06-04 16:28:17 -04:00
saveSettingsValues ( )
2018-07-29 14:42:11 +02:00
saveModConfiguration ( )
2018-06-04 16:28:17 -04:00
ConfigManager . save ( )
2018-08-07 04:16:15 -04:00
saveDropinModConfiguration ( )
2018-05-30 23:32:51 -04:00
switchView ( getCurrentView ( ) , VIEWS . landing )
}
2018-05-30 22:22:17 -04:00
/ * *
* Account Management Tab
* /
// Bind the add account button.
2018-06-20 06:15:10 -04:00
document . getElementById ( 'settingsAddAccount' ) . onclick = ( e ) => {
2018-05-30 22:22:17 -04:00
switchView ( getCurrentView ( ) , VIEWS . login , 500 , 500 , ( ) => {
loginViewOnCancel = VIEWS . settings
loginViewOnSuccess = VIEWS . settings
loginCancelEnabled ( true )
} )
}
/ * *
* Bind functionality for the account selection buttons . If another account
* is selected , the UI of the previously selected account will be updated .
* /
function bindAuthAccountSelect ( ) {
Array . from ( document . getElementsByClassName ( 'settingsAuthAccountSelect' ) ) . map ( ( val ) => {
val . onclick = ( e ) => {
if ( val . hasAttribute ( 'selected' ) ) {
return
}
const selectBtns = document . getElementsByClassName ( 'settingsAuthAccountSelect' )
for ( let i = 0 ; i < selectBtns . length ; i ++ ) {
if ( selectBtns [ i ] . hasAttribute ( 'selected' ) ) {
selectBtns [ i ] . removeAttribute ( 'selected' )
selectBtns [ i ] . innerHTML = 'Select Account'
}
}
val . setAttribute ( 'selected' , '' )
val . innerHTML = 'Selected Account ✔'
2018-05-30 23:32:51 -04:00
setSelectedAccount ( val . closest ( '.settingsAuthAccount' ) . getAttribute ( 'uuid' ) )
2018-05-30 22:22:17 -04:00
}
} )
}
/ * *
* Bind functionality for the log out button . If the logged out account was
* the selected account , another account will be selected and the UI will
* be updated accordingly .
* /
function bindAuthAccountLogOut ( ) {
Array . from ( document . getElementsByClassName ( 'settingsAuthAccountLogOut' ) ) . map ( ( val ) => {
val . onclick = ( e ) => {
2018-05-30 23:32:51 -04:00
let isLastAccount = false
if ( Object . keys ( ConfigManager . getAuthAccounts ( ) ) . length === 1 ) {
isLastAccount = true
setOverlayContent (
'Warning<br>This is Your Last Account' ,
'In order to use the launcher you must be logged into at least one account. You will need to login again after.<br><br>Are you sure you want to log out?' ,
'I\'m Sure' ,
'Cancel'
)
setOverlayHandler ( ( ) => {
processLogOut ( val , isLastAccount )
switchView ( getCurrentView ( ) , VIEWS . login )
toggleOverlay ( false )
} )
setDismissHandler ( ( ) => {
toggleOverlay ( false )
} )
toggleOverlay ( true , true )
} else {
processLogOut ( val , isLastAccount )
}
}
} )
}
/ * *
* Process a log out .
*
* @ param { Element } val The log out button element .
* @ param { boolean } isLastAccount If this logout is on the last added account .
* /
function processLogOut ( val , isLastAccount ) {
const parent = val . closest ( '.settingsAuthAccount' )
const uuid = parent . getAttribute ( 'uuid' )
const prevSelAcc = ConfigManager . getSelectedAccount ( )
AuthManager . removeAccount ( uuid ) . then ( ( ) => {
if ( ! isLastAccount && uuid === prevSelAcc . uuid ) {
const selAcc = ConfigManager . getSelectedAccount ( )
refreshAuthAccountSelected ( selAcc . uuid )
updateSelectedAccount ( selAcc )
validateSelectedAccount ( )
2018-05-30 22:22:17 -04:00
}
} )
2018-05-30 23:32:51 -04:00
$ ( parent ) . fadeOut ( 250 , ( ) => {
parent . remove ( )
} )
2018-05-30 22:22:17 -04:00
}
/ * *
* Refreshes the status of the selected account on the auth account
* elements .
*
* @ param { string } uuid The UUID of the new selected account .
* /
function refreshAuthAccountSelected ( uuid ) {
Array . from ( document . getElementsByClassName ( 'settingsAuthAccount' ) ) . map ( ( val ) => {
const selBtn = val . getElementsByClassName ( 'settingsAuthAccountSelect' ) [ 0 ]
if ( uuid === val . getAttribute ( 'uuid' ) ) {
selBtn . setAttribute ( 'selected' , '' )
selBtn . innerHTML = 'Selected Account ✔'
} else {
if ( selBtn . hasAttribute ( 'selected' ) ) {
selBtn . removeAttribute ( 'selected' )
}
selBtn . innerHTML = 'Select Account'
}
} )
}
2018-06-20 06:15:10 -04:00
const settingsCurrentAccounts = document . getElementById ( 'settingsCurrentAccounts' )
2018-05-30 22:22:17 -04:00
/ * *
* Add auth account elements for each one stored in the authentication database .
* /
function populateAuthAccounts ( ) {
const authAccounts = ConfigManager . getAuthAccounts ( )
const authKeys = Object . keys ( authAccounts )
const selectedUUID = ConfigManager . getSelectedAccount ( ) . uuid
2018-07-22 13:31:15 -04:00
let authAccountStr = ''
2018-05-30 22:22:17 -04:00
authKeys . map ( ( val ) => {
const acc = authAccounts [ val ]
authAccountStr += ` <div class="settingsAuthAccount" uuid=" ${ acc . uuid } ">
< div class = "settingsAuthAccountLeft" >
< img class = "settingsAuthAccountImage" alt = "${acc.displayName}" src = "https://crafatar.com/renders/body/${acc.uuid}?scale=3&default=MHF_Steve&overlay" >
< / d i v >
< div class = "settingsAuthAccountRight" >
< div class = "settingsAuthAccountDetails" >
< div class = "settingsAuthAccountDetailPane" >
< div class = "settingsAuthAccountDetailTitle" > Username < / d i v >
< div class = "settingsAuthAccountDetailValue" > $ { acc . displayName } < / d i v >
< / d i v >
< div class = "settingsAuthAccountDetailPane" >
2018-07-02 02:08:48 -04:00
< div class = "settingsAuthAccountDetailTitle" > UUID < / d i v >
< div class = "settingsAuthAccountDetailValue" > $ { acc . uuid } < / d i v >
2018-05-30 22:22:17 -04:00
< / d i v >
< / d i v >
< div class = "settingsAuthAccountActions" >
< button class = "settingsAuthAccountSelect" $ { selectedUUID === acc . uuid ? 'selected>Selected Account ✔' : '>Select Account' } < / b u t t o n >
< div class = "settingsAuthAccountWrapper" >
< button class = "settingsAuthAccountLogOut" > Log Out < / b u t t o n >
< / d i v >
< / d i v >
< / d i v >
< / d i v > `
} )
settingsCurrentAccounts . innerHTML = authAccountStr
}
2018-06-14 06:09:09 -04:00
/ * *
* Prepare the accounts tab for display .
* /
2018-05-30 22:22:17 -04:00
function prepareAccountsTab ( ) {
populateAuthAccounts ( )
bindAuthAccountSelect ( )
bindAuthAccountLogOut ( )
}
2018-06-04 16:28:17 -04:00
/ * *
* Minecraft Tab
* /
2018-07-22 13:31:15 -04:00
/ * *
2018-06-04 16:28:17 -04:00
* Disable decimals , negative signs , and scientific notation .
* /
2018-07-22 13:31:15 -04:00
document . getElementById ( 'settingsGameWidth' ) . addEventListener ( 'keydown' , ( e ) => {
if ( /^[-.eE]$/ . test ( e . key ) ) {
2018-06-04 16:28:17 -04:00
e . preventDefault ( )
}
} )
2018-06-20 06:15:10 -04:00
document . getElementById ( 'settingsGameHeight' ) . addEventListener ( 'keydown' , ( e ) => {
2018-07-22 13:31:15 -04:00
if ( /^[-.eE]$/ . test ( e . key ) ) {
2018-06-04 16:28:17 -04:00
e . preventDefault ( )
}
} )
2018-07-24 01:14:26 -04:00
/ * *
* Mods Tab
* /
2018-07-29 14:42:11 +02:00
const settingsModsContainer = document . getElementById ( 'settingsModsContainer' )
2018-08-07 04:41:26 -04:00
/ * *
* Resolve and update the mods on the UI .
* /
2018-07-24 01:14:26 -04:00
function resolveModsForUI ( ) {
const serv = ConfigManager . getSelectedServer ( )
const distro = DistroManager . getDistribution ( )
const servConf = ConfigManager . getModConfiguration ( serv )
const modStr = parseModulesForUI ( distro . getServer ( serv ) . getModules ( ) , false , servConf . mods )
document . getElementById ( 'settingsReqModsContent' ) . innerHTML = modStr . reqMods
document . getElementById ( 'settingsOptModsContent' ) . innerHTML = modStr . optMods
}
2018-08-07 04:41:26 -04:00
/ * *
* Recursively build the mod UI elements .
*
* @ param { Object [ ] } mdls An array of modules to parse .
* @ param { boolean } submodules Whether or not we are parsing submodules .
* @ param { Object } servConf The server configuration object for this module level .
* /
2018-08-07 00:58:32 -04:00
function parseModulesForUI ( mdls , submodules , servConf ) {
2018-07-24 01:14:26 -04:00
let reqMods = ''
let optMods = ''
for ( const mdl of mdls ) {
if ( mdl . getType ( ) === DistroManager . Types . ForgeMod || mdl . getType ( ) === DistroManager . Types . LiteMod || mdl . getType ( ) === DistroManager . Types . LiteLoader ) {
if ( mdl . getRequired ( ) . isRequired ( ) ) {
2018-08-07 04:16:15 -04:00
reqMods += ` <div id=" ${ mdl . getVersionlessID ( ) } " class="settingsBaseMod settings ${ submodules ? 'Sub' : '' } Mod" enabled>
2018-07-24 01:14:26 -04:00
< div class = "settingsModContent" >
< div class = "settingsModMainWrapper" >
< div class = "settingsModStatus" > < / d i v >
< div class = "settingsModDetails" >
2018-07-24 02:25:39 -04:00
< span class = "settingsModName" > $ { mdl . getName ( ) } < / s p a n >
2018-07-24 01:14:26 -04:00
< span class = "settingsModVersion" > v$ { mdl . getVersion ( ) } < / s p a n >
< / d i v >
< / d i v >
< label class = "toggleSwitch" reqmod >
< input type = "checkbox" checked >
< span class = "toggleSwitchSlider" > < / s p a n >
< / l a b e l >
< / d i v >
$ { mdl . hasSubModules ( ) ? ` <div class="settingsSubModContainer">
2018-07-29 16:32:41 +02:00
$ { Object . values ( parseModulesForUI ( mdl . getSubModules ( ) , true , servConf [ mdl . getVersionlessID ( ) ] ) ) . join ( '' ) }
2018-07-24 01:14:26 -04:00
< / d i v > ` : ' ' }
< / d i v > `
} else {
const conf = servConf [ mdl . getVersionlessID ( ) ]
const val = typeof conf === 'object' ? conf . value : conf
2018-08-07 04:16:15 -04:00
optMods += ` <div id=" ${ mdl . getVersionlessID ( ) } " class="settingsBaseMod settings ${ submodules ? 'Sub' : '' } Mod" ${ val ? 'enabled' : '' } >
2018-07-24 01:14:26 -04:00
< div class = "settingsModContent" >
< div class = "settingsModMainWrapper" >
< div class = "settingsModStatus" > < / d i v >
< div class = "settingsModDetails" >
2018-07-24 02:25:39 -04:00
< span class = "settingsModName" > $ { mdl . getName ( ) } < / s p a n >
2018-07-24 01:14:26 -04:00
< span class = "settingsModVersion" > v$ { mdl . getVersion ( ) } < / s p a n >
< / d i v >
< / d i v >
< label class = "toggleSwitch" >
2018-07-29 14:42:11 +02:00
< input type = "checkbox" formod = "${mdl.getVersionlessID()}" $ { val ? 'checked' : '' } >
2018-07-24 01:14:26 -04:00
< span class = "toggleSwitchSlider" > < / s p a n >
< / l a b e l >
< / d i v >
$ { mdl . hasSubModules ( ) ? ` <div class="settingsSubModContainer">
2018-07-29 14:42:11 +02:00
$ { Object . values ( parseModulesForUI ( mdl . getSubModules ( ) , true , conf . mods ) ) . join ( '' ) }
2018-07-24 01:14:26 -04:00
< / d i v > ` : ' ' }
< / d i v > `
}
}
}
return {
reqMods ,
optMods
}
}
2018-07-29 14:42:11 +02:00
/ * *
* Bind functionality to mod config toggle switches . Switching the value
* will also switch the status color on the left of the mod UI .
* /
function bindModsToggleSwitch ( ) {
const sEls = settingsModsContainer . querySelectorAll ( '[formod]' )
Array . from ( sEls ) . map ( ( v , index , arr ) => {
v . onchange = ( ) => {
if ( v . checked ) {
document . getElementById ( v . getAttribute ( 'formod' ) ) . setAttribute ( 'enabled' , '' )
} else {
document . getElementById ( v . getAttribute ( 'formod' ) ) . removeAttribute ( 'enabled' )
}
}
} )
}
/ * *
* Save the mod configuration based on the UI values .
* /
function saveModConfiguration ( ) {
const serv = ConfigManager . getSelectedServer ( )
const modConf = ConfigManager . getModConfiguration ( serv )
modConf . mods = _saveModConfiguration ( modConf . mods )
ConfigManager . setModConfiguration ( serv , modConf )
}
/ * *
* Recursively save mod config with submods .
*
* @ param { Object } modConf Mod config object to save .
* /
function _saveModConfiguration ( modConf ) {
for ( m of Object . entries ( modConf ) ) {
2018-07-29 16:32:41 +02:00
const tSwitch = settingsModsContainer . querySelectorAll ( ` [formod=' ${ m [ 0 ] } '] ` )
2018-08-07 04:16:15 -04:00
if ( ! tSwitch [ 0 ] . hasAttribute ( 'dropin' ) ) {
if ( typeof m [ 1 ] === 'boolean' ) {
modConf [ m [ 0 ] ] = tSwitch [ 0 ] . checked
} else {
if ( m [ 1 ] != null ) {
if ( tSwitch . length > 0 ) {
modConf [ m [ 0 ] ] . value = tSwitch [ 0 ] . checked
}
modConf [ m [ 0 ] ] . mods = _saveModConfiguration ( modConf [ m [ 0 ] ] . mods )
2018-07-29 16:32:41 +02:00
}
2018-07-29 14:42:11 +02:00
}
}
}
return modConf
}
2018-08-07 04:41:26 -04:00
// Drop-in mod elements.
2018-08-07 00:58:32 -04:00
2018-08-07 04:16:15 -04:00
let CACHE _SETTINGS _MODS _DIR
let CACHE _DROPIN _MODS
2018-08-07 04:41:26 -04:00
/ * *
* Resolve any located drop - in mods for this server and
* populate the results onto the UI .
* /
2018-08-07 04:16:15 -04:00
function resolveDropinModsForUI ( ) {
const serv = DistroManager . getDistribution ( ) . getServer ( ConfigManager . getSelectedServer ( ) )
CACHE _SETTINGS _MODS _DIR = path . join ( ConfigManager . getInstanceDirectory ( ) , serv . getID ( ) , 'mods' )
CACHE _DROPIN _MODS = DropinModUtil . scanForDropinMods ( CACHE _SETTINGS _MODS _DIR , serv . getMinecraftVersion ( ) )
let dropinMods = ''
for ( dropin of CACHE _DROPIN _MODS ) {
dropinMods += ` <div id=" ${ dropin . fullName } " class="settingsBaseMod settingsDropinMod" ${ ! dropin . disabled ? 'enabled' : '' } >
< div class = "settingsModContent" >
< div class = "settingsModMainWrapper" >
< div class = "settingsModStatus" > < / d i v >
< div class = "settingsModDetails" >
< span class = "settingsModName" > $ { dropin . name } < / s p a n >
< div class = "settingsDropinRemoveWrapper" >
< button class = "settingsDropinRemoveButton" remmod = "${dropin.fullName}" > Remove < / b u t t o n >
< / d i v >
< / d i v >
< / d i v >
< label class = "toggleSwitch" >
< input type = "checkbox" formod = "${dropin.fullName}" dropin $ { ! dropin . disabled ? 'checked' : '' } >
< span class = "toggleSwitchSlider" > < / s p a n >
< / l a b e l >
< / d i v >
< / d i v > `
}
document . getElementById ( 'settingsDropinModsContent' ) . innerHTML = dropinMods
}
2018-08-07 04:41:26 -04:00
/ * *
* Bind the remove button for each loaded drop - in mod .
* /
2018-08-07 04:16:15 -04:00
function bindDropinModsRemoveButton ( ) {
const sEls = settingsModsContainer . querySelectorAll ( '[remmod]' )
Array . from ( sEls ) . map ( ( v , index , arr ) => {
v . onclick = ( ) => {
const fullName = v . getAttribute ( 'remmod' )
const res = DropinModUtil . deleteDropinMod ( CACHE _SETTINGS _MODS _DIR , fullName )
if ( res ) {
document . getElementById ( fullName ) . remove ( )
} else {
setOverlayContent (
` Failed to Delete<br>Drop-in Mod ${ fullName } ` ,
'Make sure the file is not in use and try again.' ,
'Okay'
)
setOverlayHandler ( null )
toggleOverlay ( true )
}
}
} )
}
2018-08-07 04:41:26 -04:00
/ * *
* Bind functionality to the file system button for the selected
* server configuration .
* /
2018-08-07 04:16:15 -04:00
function bindDropinModFileSystemButton ( ) {
const fsBtn = document . getElementById ( 'settingsDropinFileSystemButton' )
fsBtn . onclick = ( ) => {
shell . openItem ( CACHE _SETTINGS _MODS _DIR )
}
}
2018-08-07 04:41:26 -04:00
/ * *
* Save drop - in mod states . Enabling and disabling is just a matter
* of adding / removing the . disabled extension .
* /
2018-08-07 04:16:15 -04:00
function saveDropinModConfiguration ( ) {
for ( dropin of CACHE _DROPIN _MODS ) {
const dropinUI = document . getElementById ( dropin . fullName )
if ( dropinUI != null ) {
const dropinUIEnabled = dropinUI . hasAttribute ( 'enabled' )
if ( DropinModUtil . isDropinModEnabled ( dropin . fullName ) != dropinUIEnabled ) {
DropinModUtil . toggleDropinMod ( CACHE _SETTINGS _MODS _DIR , dropin . fullName , dropinUIEnabled ) . catch ( err => {
if ( ! isOverlayVisible ( ) ) {
setOverlayContent (
'Failed to Toggle<br>One or More Drop-in Mods' ,
err . message ,
'Okay'
)
setOverlayHandler ( null )
toggleOverlay ( true )
}
} )
}
}
}
}
2018-08-07 04:41:26 -04:00
// Refresh the drop-in mods when F5 is pressed.
// Only active on the mods tab.
2018-08-07 04:16:15 -04:00
document . addEventListener ( 'keydown' , ( e ) => {
if ( getCurrentView ( ) === VIEWS . settings && selectedSettingsTab === 'settingsTabMods' ) {
if ( e . key === 'F5' ) {
resolveDropinModsForUI ( )
bindDropinModsRemoveButton ( )
bindDropinModFileSystemButton ( )
bindModsToggleSwitch ( )
}
}
} )
2018-08-07 04:41:26 -04:00
// Server status bar functions.
/ * *
* Load the currently selected server information onto the mods tab .
* /
function loadSelectedServerOnModsTab ( ) {
const serv = DistroManager . getDistribution ( ) . getServer ( ConfigManager . getSelectedServer ( ) )
document . getElementById ( 'settingsSelServContent' ) . innerHTML = `
< img class = "serverListingImg" src = "${serv.getIcon()}" / >
< div class = "serverListingDetails" >
< span class = "serverListingName" > $ { serv . getName ( ) } < / s p a n >
< span class = "serverListingDescription" > $ { serv . getDescription ( ) } < / s p a n >
< div class = "serverListingInfo" >
< div class = "serverListingVersion" > $ { serv . getMinecraftVersion ( ) } < / d i v >
< div class = "serverListingRevision" > $ { serv . getVersion ( ) } < / d i v >
$ { serv . isMainServer ( ) ? ` <div class="serverListingStarWrapper">
< svg id = "Layer_1" viewBox = "0 0 107.45 104.74" width = "20px" height = "20px" >
< defs >
< style > . cls - 1 { fill : # fff ; } . cls - 2 { fill : none ; stroke : # fff ; stroke - miterlimit : 10 ; } < / s t y l e >
< / d e f s >
< path class = "cls-1" d = "M100.93,65.54C89,62,68.18,55.65,63.54,52.13c2.7-5.23,18.8-19.2,28-27.55C81.36,31.74,63.74,43.87,58.09,45.3c-2.41-5.37-3.61-26.52-4.37-39-.77,12.46-2,33.64-4.36,39-5.7-1.46-23.3-13.57-33.49-20.72,9.26,8.37,25.39,22.36,28,27.55C39.21,55.68,18.47,62,6.52,65.55c12.32-2,33.63-6.06,39.34-4.9-.16,5.87-8.41,26.16-13.11,37.69,6.1-10.89,16.52-30.16,21-33.9,4.5,3.79,14.93,23.09,21,34C70,86.84,61.73,66.48,61.59,60.65,67.36,59.49,88.64,63.52,100.93,65.54Z" / >
< circle class = "cls-2" cx = "53.73" cy = "53.9" r = "38" / >
< / s v g >
< span class = "serverListingStarTooltip" > Main Server < / s p a n >
< / d i v > ` : ' ' }
< / d i v >
< / d i v >
`
}
// Bind functionality to the server switch button.
document . getElementById ( 'settingsSwitchServerButton' ) . addEventListener ( 'click' , ( e ) => {
e . target . blur ( )
toggleServerSelection ( true )
} )
/ * *
* Function to refresh the mods tab whenever the selected
* server is changed .
* /
2018-08-07 00:58:32 -04:00
function animateModsTabRefresh ( ) {
$ ( '#settingsTabMods' ) . fadeOut ( 500 , ( ) => {
2018-08-07 04:16:15 -04:00
saveModConfiguration ( )
ConfigManager . save ( )
saveDropinModConfiguration ( )
2018-08-07 00:58:32 -04:00
prepareModsTab ( )
$ ( '#settingsTabMods' ) . fadeIn ( 500 )
} )
}
2018-07-24 01:14:26 -04:00
/ * *
2018-08-07 00:58:32 -04:00
* Prepare the Mods tab for display .
2018-07-24 01:14:26 -04:00
* /
2018-08-07 00:58:32 -04:00
function prepareModsTab ( first ) {
2018-07-24 01:14:26 -04:00
resolveModsForUI ( )
2018-08-07 04:16:15 -04:00
resolveDropinModsForUI ( )
bindDropinModsRemoveButton ( )
bindDropinModFileSystemButton ( )
2018-07-29 14:42:11 +02:00
bindModsToggleSwitch ( )
2018-08-07 00:58:32 -04:00
loadSelectedServerOnModsTab ( )
2018-07-24 01:14:26 -04:00
}
2018-06-11 22:11:05 -04:00
/ * *
* Java Tab
* /
2018-06-20 06:15:10 -04:00
// DOM Cache
const settingsMaxRAMRange = document . getElementById ( 'settingsMaxRAMRange' )
const settingsMinRAMRange = document . getElementById ( 'settingsMinRAMRange' )
const settingsMaxRAMLabel = document . getElementById ( 'settingsMaxRAMLabel' )
const settingsMinRAMLabel = document . getElementById ( 'settingsMinRAMLabel' )
const settingsMemoryTotal = document . getElementById ( 'settingsMemoryTotal' )
const settingsMemoryAvail = document . getElementById ( 'settingsMemoryAvail' )
const settingsJavaExecDetails = document . getElementById ( 'settingsJavaExecDetails' )
const settingsJavaExecVal = document . getElementById ( 'settingsJavaExecVal' )
const settingsJavaExecSel = document . getElementById ( 'settingsJavaExecSel' )
2018-06-14 06:09:09 -04:00
// Store maximum memory values.
const SETTINGS _MAX _MEMORY = ConfigManager . getAbsoluteMaxRAM ( )
const SETTINGS _MIN _MEMORY = ConfigManager . getAbsoluteMinRAM ( )
// Set the max and min values for the ranged sliders.
settingsMaxRAMRange . setAttribute ( 'max' , SETTINGS _MAX _MEMORY )
settingsMaxRAMRange . setAttribute ( 'min' , SETTINGS _MIN _MEMORY )
settingsMinRAMRange . setAttribute ( 'max' , SETTINGS _MAX _MEMORY )
settingsMinRAMRange . setAttribute ( 'min' , SETTINGS _MIN _MEMORY )
2018-06-11 22:11:05 -04:00
2018-06-14 06:09:09 -04:00
// Bind on change event for min memory container.
2018-06-11 22:11:05 -04:00
settingsMinRAMRange . onchange = ( e ) => {
2018-06-14 06:09:09 -04:00
// Current range values
2018-06-11 22:11:05 -04:00
const sMaxV = Number ( settingsMaxRAMRange . getAttribute ( 'value' ) )
const sMinV = Number ( settingsMinRAMRange . getAttribute ( 'value' ) )
2018-06-14 06:09:09 -04:00
// Get reference to range bar.
2018-06-12 03:48:36 -04:00
const bar = e . target . getElementsByClassName ( 'rangeSliderBar' ) [ 0 ]
2018-06-14 06:09:09 -04:00
// Calculate effective total memory.
2018-06-12 03:48:36 -04:00
const max = ( os . totalmem ( ) - 1000000000 ) / 1000000000
2018-06-14 06:09:09 -04:00
// Change range bar color based on the selected value.
2018-06-12 03:48:36 -04:00
if ( sMinV >= max / 2 ) {
bar . style . background = '#e86060'
} else if ( sMinV >= max / 4 ) {
bar . style . background = '#e8e18b'
} else {
bar . style . background = null
}
2018-06-14 06:09:09 -04:00
// Increase maximum memory if the minimum exceeds its value.
2018-06-11 22:11:05 -04:00
if ( sMaxV < sMinV ) {
const sliderMeta = calculateRangeSliderMeta ( settingsMaxRAMRange )
updateRangedSlider ( settingsMaxRAMRange , sMinV ,
2018-07-22 13:31:15 -04:00
( ( sMinV - sliderMeta . min ) / sliderMeta . step ) * sliderMeta . inc )
2018-06-12 03:25:36 -04:00
settingsMaxRAMLabel . innerHTML = sMinV . toFixed ( 1 ) + 'G'
2018-06-11 22:11:05 -04:00
}
2018-06-14 06:09:09 -04:00
// Update label
2018-06-12 03:25:36 -04:00
settingsMinRAMLabel . innerHTML = sMinV . toFixed ( 1 ) + 'G'
2018-06-11 22:11:05 -04:00
}
2018-06-12 03:25:36 -04:00
2018-06-14 06:09:09 -04:00
// Bind on change event for max memory container.
2018-06-11 22:11:05 -04:00
settingsMaxRAMRange . onchange = ( e ) => {
2018-06-14 06:09:09 -04:00
// Current range values
2018-06-11 22:11:05 -04:00
const sMaxV = Number ( settingsMaxRAMRange . getAttribute ( 'value' ) )
const sMinV = Number ( settingsMinRAMRange . getAttribute ( 'value' ) )
2018-06-14 06:09:09 -04:00
// Get reference to range bar.
2018-06-12 03:48:36 -04:00
const bar = e . target . getElementsByClassName ( 'rangeSliderBar' ) [ 0 ]
2018-06-14 06:09:09 -04:00
// Calculate effective total memory.
2018-06-12 03:48:36 -04:00
const max = ( os . totalmem ( ) - 1000000000 ) / 1000000000
2018-06-14 06:09:09 -04:00
// Change range bar color based on the selected value.
2018-06-12 03:48:36 -04:00
if ( sMaxV >= max / 2 ) {
bar . style . background = '#e86060'
} else if ( sMaxV >= max / 4 ) {
bar . style . background = '#e8e18b'
} else {
bar . style . background = null
}
2018-06-14 06:09:09 -04:00
// Decrease the minimum memory if the maximum value is less.
2018-06-11 22:11:05 -04:00
if ( sMaxV < sMinV ) {
2018-06-12 03:25:36 -04:00
const sliderMeta = calculateRangeSliderMeta ( settingsMaxRAMRange )
updateRangedSlider ( settingsMinRAMRange , sMaxV ,
2018-07-22 13:31:15 -04:00
( ( sMaxV - sliderMeta . min ) / sliderMeta . step ) * sliderMeta . inc )
2018-06-12 03:25:36 -04:00
settingsMinRAMLabel . innerHTML = sMaxV . toFixed ( 1 ) + 'G'
2018-06-11 22:11:05 -04:00
}
2018-06-12 03:25:36 -04:00
settingsMaxRAMLabel . innerHTML = sMaxV . toFixed ( 1 ) + 'G'
2018-06-11 22:11:05 -04:00
}
2018-06-14 06:09:09 -04:00
/ * *
* Calculate common values for a ranged slider .
*
* @ param { Element } v The range slider to calculate against .
* @ returns { Object } An object with meta values for the provided ranged slider .
* /
2018-06-11 22:11:05 -04:00
function calculateRangeSliderMeta ( v ) {
const val = {
max : Number ( v . getAttribute ( 'max' ) ) ,
min : Number ( v . getAttribute ( 'min' ) ) ,
step : Number ( v . getAttribute ( 'step' ) ) ,
}
2018-06-12 03:48:36 -04:00
val . ticks = ( val . max - val . min ) / val . step
2018-06-11 22:11:05 -04:00
val . inc = 100 / val . ticks
return val
}
2018-06-14 06:09:09 -04:00
/ * *
* Binds functionality to the ranged sliders . They ' re more than
* just divs now : ' ) .
* /
2018-06-11 22:11:05 -04:00
function bindRangeSlider ( ) {
Array . from ( document . getElementsByClassName ( 'rangeSlider' ) ) . map ( ( v ) => {
2018-06-14 06:09:09 -04:00
// Reference the track (thumb).
2018-06-11 22:11:05 -04:00
const track = v . getElementsByClassName ( 'rangeSliderTrack' ) [ 0 ]
2018-06-14 06:09:09 -04:00
// Set the initial slider value.
2018-06-11 22:11:05 -04:00
const value = v . getAttribute ( 'value' )
const sliderMeta = calculateRangeSliderMeta ( v )
2018-06-14 06:09:09 -04:00
updateRangedSlider ( v , value , ( ( value - sliderMeta . min ) / sliderMeta . step ) * sliderMeta . inc )
// The magic happens when we click on the track.
2018-06-11 22:11:05 -04:00
track . onmousedown = ( e ) => {
2018-06-14 06:09:09 -04:00
// Stop moving the track on mouse up.
2018-06-11 22:11:05 -04:00
document . onmouseup = ( e ) => {
document . onmousemove = null
document . onmouseup = null
}
2018-06-14 06:09:09 -04:00
// Move slider according to the mouse position.
2018-06-11 22:11:05 -04:00
document . onmousemove = ( e ) => {
2018-06-14 06:09:09 -04:00
// Distance from the beginning of the bar in pixels.
2018-06-11 22:11:05 -04:00
const diff = e . pageX - v . offsetLeft - track . offsetWidth / 2
2018-06-14 06:09:09 -04:00
// Don't move the track off the bar.
2018-06-11 22:11:05 -04:00
if ( diff >= 0 && diff <= v . offsetWidth - track . offsetWidth / 2 ) {
2018-06-14 06:09:09 -04:00
// Convert the difference to a percentage.
2018-06-11 22:11:05 -04:00
const perc = ( diff / v . offsetWidth ) * 100
2018-06-14 06:09:09 -04:00
// Calculate the percentage of the closest notch.
2018-06-11 22:11:05 -04:00
const notch = Number ( perc / sliderMeta . inc ) . toFixed ( 0 ) * sliderMeta . inc
2018-06-12 03:48:36 -04:00
2018-06-14 06:09:09 -04:00
// If we're close to that notch, stick to it.
2018-06-11 22:11:05 -04:00
if ( Math . abs ( perc - notch ) < sliderMeta . inc / 2 ) {
2018-06-12 03:48:36 -04:00
updateRangedSlider ( v , sliderMeta . min + ( sliderMeta . step * ( notch / sliderMeta . inc ) ) , notch )
2018-06-11 22:11:05 -04:00
}
}
}
}
} )
}
2018-06-14 06:09:09 -04:00
/ * *
* Update a ranged slider ' s value and position .
*
* @ param { Element } element The ranged slider to update .
* @ param { string | number } value The new value for the ranged slider .
* @ param { number } notch The notch that the slider should now be at .
* /
2018-06-11 22:11:05 -04:00
function updateRangedSlider ( element , value , notch ) {
const oldVal = element . getAttribute ( 'value' )
const bar = element . getElementsByClassName ( 'rangeSliderBar' ) [ 0 ]
const track = element . getElementsByClassName ( 'rangeSliderTrack' ) [ 0 ]
2018-06-14 06:09:09 -04:00
2018-06-11 22:11:05 -04:00
element . setAttribute ( 'value' , value )
2018-06-14 06:09:09 -04:00
2018-06-21 21:04:10 -04:00
if ( notch < 0 ) {
notch = 0
} else if ( notch > 100 ) {
notch = 100
}
2018-06-11 22:11:05 -04:00
const event = new MouseEvent ( 'change' , {
target : element ,
type : 'change' ,
bubbles : false ,
cancelable : true
} )
2018-06-14 06:09:09 -04:00
2018-06-11 22:11:05 -04:00
let cancelled = ! element . dispatchEvent ( event )
2018-06-14 06:09:09 -04:00
2018-06-11 22:11:05 -04:00
if ( ! cancelled ) {
track . style . left = notch + '%'
bar . style . width = notch + '%'
} else {
element . setAttribute ( 'value' , oldVal )
}
}
2018-06-14 06:09:09 -04:00
/ * *
* Display the total and available RAM .
* /
function populateMemoryStatus ( ) {
2018-06-12 03:25:36 -04:00
settingsMemoryTotal . innerHTML = Number ( ( os . totalmem ( ) - 1000000000 ) / 1000000000 ) . toFixed ( 1 ) + 'G'
settingsMemoryAvail . innerHTML = Number ( os . freemem ( ) / 1000000000 ) . toFixed ( 1 ) + 'G'
}
2018-06-14 06:09:09 -04:00
// Bind the executable file input to the display text input.
settingsJavaExecSel . onchange = ( e ) => {
settingsJavaExecVal . value = settingsJavaExecSel . files [ 0 ] . path
populateJavaExecDetails ( settingsJavaExecVal . value )
}
/ * *
* Validate the provided executable path and display the data on
* the UI .
*
* @ param { string } execPath The executable path to populate against .
* /
function populateJavaExecDetails ( execPath ) {
AssetGuard . _validateJavaBinary ( execPath ) . then ( v => {
if ( v . valid ) {
settingsJavaExecDetails . innerHTML = ` Selected: Java ${ v . version . major } Update ${ v . version . update } (x ${ v . arch } ) `
} else {
settingsJavaExecDetails . innerHTML = 'Invalid Selection'
}
} )
}
/ * *
* Prepare the Java tab for display .
* /
2018-06-11 22:11:05 -04:00
function prepareJavaTab ( ) {
bindRangeSlider ( )
2018-06-14 06:09:09 -04:00
populateMemoryStatus ( )
2018-06-11 22:11:05 -04:00
}
2018-06-20 06:15:10 -04:00
/ * *
* About Tab
* /
const settingsAboutCurrentVersionCheck = document . getElementById ( 'settingsAboutCurrentVersionCheck' )
const settingsAboutCurrentVersionTitle = document . getElementById ( 'settingsAboutCurrentVersionTitle' )
const settingsAboutCurrentVersionValue = document . getElementById ( 'settingsAboutCurrentVersionValue' )
const settingsChangelogTitle = document . getElementById ( 'settingsChangelogTitle' )
const settingsChangelogText = document . getElementById ( 'settingsChangelogText' )
const settingsChangelogButton = document . getElementById ( 'settingsChangelogButton' )
// Bind the devtools toggle button.
document . getElementById ( 'settingsAboutDevToolsButton' ) . onclick = ( e ) => {
let window = remote . getCurrentWindow ( )
window . toggleDevTools ( )
}
/ * *
* Retrieve the version information and display it on the UI .
* /
function populateVersionInformation ( ) {
const version = remote . app . getVersion ( )
settingsAboutCurrentVersionValue . innerHTML = version
const preRelComp = semver . prerelease ( version )
if ( preRelComp != null && preRelComp . length > 0 ) {
settingsAboutCurrentVersionTitle . innerHTML = 'Pre-release'
settingsAboutCurrentVersionTitle . style . color = '#ff886d'
settingsAboutCurrentVersionCheck . style . background = '#ff886d'
} else {
settingsAboutCurrentVersionTitle . innerHTML = 'Stable Release'
settingsAboutCurrentVersionTitle . style . color = null
settingsAboutCurrentVersionCheck . style . background = null
}
}
/ * *
* Fetches the GitHub atom release feed and parses it for the release notes
* of the current version . This value is displayed on the UI .
* /
function populateReleaseNotes ( ) {
$ . ajax ( {
url : 'https://github.com/WesterosCraftCode/ElectronLauncher/releases.atom' ,
success : ( data ) => {
const version = 'v' + remote . app . getVersion ( )
const entries = $ ( data ) . find ( 'entry' )
for ( let i = 0 ; i < entries . length ; i ++ ) {
const entry = $ ( entries [ i ] )
let id = entry . find ( 'id' ) . text ( )
id = id . substring ( id . lastIndexOf ( '/' ) + 1 )
if ( id === version ) {
settingsChangelogTitle . innerHTML = entry . find ( 'title' ) . text ( )
settingsChangelogText . innerHTML = entry . find ( 'content' ) . text ( )
settingsChangelogButton . href = entry . find ( 'link' ) . attr ( 'href' )
}
}
} ,
timeout : 2500
} ) . catch ( err => {
settingsChangelogText . innerHTML = 'Failed to load release notes.'
} )
}
/ * *
* Prepare account tab for display .
* /
function prepareAboutTab ( ) {
populateVersionInformation ( )
populateReleaseNotes ( )
}
2018-06-11 22:11:05 -04:00
2018-05-30 22:22:17 -04:00
/ * *
* Settings preparation functions .
* /
2018-07-22 13:31:15 -04:00
/ * *
2018-05-30 22:22:17 -04:00
* Prepare the entire settings UI .
2018-06-04 16:28:17 -04:00
*
* @ param { boolean } first Whether or not it is the first load .
2018-05-30 22:22:17 -04:00
* /
2018-06-04 16:28:17 -04:00
function prepareSettings ( first = false ) {
if ( first ) {
setupSettingsTabs ( )
initSettingsValidators ( )
2018-07-24 01:14:26 -04:00
} else {
prepareModsTab ( )
2018-06-04 16:28:17 -04:00
}
initSettingsValues ( )
2018-05-30 22:22:17 -04:00
prepareAccountsTab ( )
2018-06-11 22:11:05 -04:00
prepareJavaTab ( )
2018-06-20 06:15:10 -04:00
prepareAboutTab ( )
2018-05-30 22:22:17 -04:00
}
// Prepare the settings UI on startup.
2018-06-04 16:28:17 -04:00
prepareSettings ( true )