2018-12-30 23:16:09 +01:00
import Debug from '../conf/Debug' ;
import currentBrowser from '../conf/BrowserDetect' ;
import ExtensionConf from '../conf/ExtensionConf' ;
2019-01-20 23:01:45 +01:00
import ExtensionMode from '../../common/enums/extension-mode.enum' ;
2019-01-02 20:36:00 +01:00
import ObjectCopy from '../lib/ObjectCopy' ;
2019-02-13 23:58:19 +01:00
import Stretch from '../../common/enums/stretch.enum' ;
import VideoAlignment from '../../common/enums/video-alignment.enum' ;
2019-07-05 23:45:29 +02:00
import ExtensionConfPatch from '../conf/ExtConfPatches' ;
2018-12-30 23:16:09 +01:00
2019-06-03 00:37:19 +02:00
2018-08-05 23:48:56 +02:00
class Settings {
2018-08-30 00:56:15 +02:00
constructor ( activeSettings , updateCallback ) {
2018-08-22 22:34:07 +02:00
this . active = activeSettings ? activeSettings : undefined ;
2018-08-05 23:48:56 +02:00
this . default = ExtensionConf ;
2018-08-21 23:48:47 +02:00
this . useSync = false ;
2018-08-22 22:34:07 +02:00
this . version = undefined ;
2018-08-30 00:56:15 +02:00
this . updateCallback = updateCallback ;
2018-08-23 01:04:37 +02:00
const ths = this ;
2018-12-30 23:16:09 +01:00
if ( currentBrowser . firefox ) {
2018-08-23 01:04:37 +02:00
browser . storage . onChanged . addListener ( ( changes , area ) => {
2018-08-30 00:56:15 +02:00
if ( Debug . debug && Debug . debugStorage ) {
2018-08-29 00:41:26 +02:00
console . log ( "[Settings::<storage/on change>] Settings have been changed outside of here. Updating active settings. Changes:" , changes , "storage area:" , area ) ;
if ( changes [ 'uwSettings' ] && changes [ 'uwSettings' ] . newValue ) {
console . log ( "[Settings::<storage/on change>] new settings object:" , JSON . parse ( changes . uwSettings . newValue ) ) ;
}
}
2018-08-23 01:04:37 +02:00
if ( changes [ 'uwSettings' ] && changes [ 'uwSettings' ] . newValue ) {
2018-09-17 00:39:32 +02:00
ths . setActive ( JSON . parse ( changes . uwSettings . newValue ) ) ;
2018-08-23 01:04:37 +02:00
}
2018-08-30 00:56:15 +02:00
if ( this . updateCallback ) {
try {
2018-12-30 23:16:09 +01:00
updateCallback ( ths ) ;
2018-08-30 00:56:15 +02:00
} catch ( e ) {
console . log ( "[Settings] CALLING UPDATE CALLBACK FAILED." )
}
}
2018-08-23 01:04:37 +02:00
} ) ;
2018-12-30 23:16:09 +01:00
} else if ( currentBrowser . chrome ) {
2018-08-29 00:41:26 +02:00
chrome . storage . onChanged . addListener ( ( changes , area ) => {
2018-08-30 00:56:15 +02:00
if ( Debug . debug && Debug . debugStorage ) {
2018-08-29 00:41:26 +02:00
console . log ( "[Settings::<storage/on change>] Settings have been changed outside of here. Updating active settings. Changes:" , changes , "storage area:" , area ) ;
if ( changes [ 'uwSettings' ] && changes [ 'uwSettings' ] . newValue ) {
console . log ( "[Settings::<storage/on change>] new settings object:" , JSON . parse ( changes . uwSettings . newValue ) ) ;
}
}
if ( changes [ 'uwSettings' ] && changes [ 'uwSettings' ] . newValue ) {
2018-09-17 00:39:32 +02:00
ths . setActive ( JSON . parse ( changes . uwSettings . newValue ) ) ;
2018-08-29 00:41:26 +02:00
}
2018-08-30 00:56:15 +02:00
if ( this . updateCallback ) {
try {
2018-12-30 23:16:09 +01:00
updateCallback ( ths ) ;
2018-08-30 00:56:15 +02:00
} catch ( e ) {
console . log ( "[Settings] CALLING UPDATE CALLBACK FAILED." )
}
}
} ) ;
2018-08-23 01:04:37 +02:00
}
2018-08-05 23:48:56 +02:00
}
async init ( ) {
2018-08-21 23:48:47 +02:00
const settings = await this . get ( ) ;
2018-08-05 23:48:56 +02:00
if ( Debug . debug ) {
2018-08-21 23:48:47 +02:00
console . log ( "[Settings::init] Configuration fetched from storage:" , settings ) ;
2018-11-18 03:29:16 +01:00
if ( Debug . flushStoredSettings ) {
console . log ( "%c[Settings::init] Debug.flushStoredSettings is true. Using default settings" , "background: #d00; color: #ffd" ) ;
Debug . flushStoredSettings = false ; // don't do it again this session
2018-12-30 23:16:09 +01:00
this . setDefaultSettings ( ) ;
2018-11-18 03:29:16 +01:00
this . active = this . getDefaultSettings ( ) ;
return this . active ;
}
2018-08-05 23:48:56 +02:00
}
// if there's no settings saved, return default settings.
2018-08-07 23:31:28 +02:00
if ( ! settings || ( Object . keys ( settings ) . length === 0 && settings . constructor === Object ) ) {
2018-08-05 23:48:56 +02:00
this . setDefaultSettings ( ) ;
this . active = this . getDefaultSettings ( ) ;
return this . active ;
}
2019-06-05 23:36:47 +02:00
// if last saved settings was for version prior to 4.x, we reset settings to default
// it's not like people will notice cos that version didn't preserve settings at all
if ( settings . version && ! settings . version . startsWith ( '4' ) ) {
this . setDefaultSettings ( ) ;
this . active = this . getDefaultSettings ( ) ;
return this . active ;
}
2018-08-05 23:48:56 +02:00
// if there's settings, set saved object as active settings
this . active = settings ;
// check if extension has been updated. If not, return settings as they were retreived
2018-12-30 23:16:09 +01:00
if ( currentBrowser . firefox ) {
2018-08-22 22:34:07 +02:00
this . version = browser . runtime . getManifest ( ) . version ;
2018-12-30 23:16:09 +01:00
} else if ( currentBrowser . chrome ) {
2018-08-22 22:34:07 +02:00
this . version = chrome . runtime . getManifest ( ) . version ;
2018-12-30 23:16:09 +01:00
} else if ( currentBrowser . edge ) {
2018-08-22 22:34:07 +02:00
this . version = browser . runtime . getManifest ( ) . version ;
2018-08-07 23:31:28 +02:00
}
2018-08-22 22:34:07 +02:00
if ( settings . version === this . version ) {
2018-08-05 23:48:56 +02:00
if ( Debug . debug ) {
2018-08-22 22:34:07 +02:00
console . log ( "[Settings::init] extension was saved with current version of ultrawidify (" , this . version , "). Returning object as-is." ) ;
2018-08-05 23:48:56 +02:00
}
2019-07-05 23:45:29 +02:00
2018-08-05 23:48:56 +02:00
return this . active ;
}
// if extension has been updated, update existing settings with any options added in the
// new version. In addition to that, we remove old keys that are no longer used.
2019-06-05 23:36:47 +02:00
const patched = ObjectCopy . addNew ( settings , this . default ) ;
if ( Debug . debug ) {
console . log ( "[Settings.init] Results from ObjectCopy.addNew()?" , patched , "\n\nSettings from storage" , settings , "\ndefault?" , this . default , ) ;
}
if ( patched ) {
this . active = patched ;
} else {
2018-08-22 22:34:07 +02:00
this . active = JSON . parse ( JSON . stringify ( this . default ) ) ;
2019-06-05 23:36:47 +02:00
}
2018-08-22 22:34:07 +02:00
2019-07-05 23:45:29 +02:00
// in case settings in previous version contained a fucky wucky, we overwrite existing settings with a patch
ObjectCopy . overwrite ( this . active , ExtensionConfPatch [ '4.2.0' ] ) ;
2019-07-07 15:12:15 +02:00
// set 'whatsNewChecked' flag to false when updating
this . active . whatsNewChecked = false ;
2018-08-05 23:48:56 +02:00
this . set ( this . active ) ;
return this . active ;
}
async get ( ) {
2019-05-10 19:21:17 +02:00
let ret ;
2018-12-30 23:16:09 +01:00
if ( currentBrowser . firefox ) {
2019-05-10 19:21:17 +02:00
ret = await browser . storage . local . get ( 'uwSettings' ) ;
2018-12-30 23:16:09 +01:00
} else if ( currentBrowser . chrome ) {
2019-05-10 19:21:17 +02:00
ret = await new Promise ( ( resolve , reject ) => {
2019-05-03 00:58:06 +02:00
chrome . storage . local . get ( 'uwSettings' , ( res ) => resolve ( res ) ) ;
2018-08-29 00:41:26 +02:00
} ) ;
2018-12-30 23:16:09 +01:00
} else if ( currentBrowser . edge ) {
2019-05-10 19:21:17 +02:00
ret = await new Promise ( ( resolve , reject ) => {
2019-05-03 00:58:06 +02:00
browser . storage . local . get ( 'uwSettings' , ( res ) => resolve ( res ) ) ;
2018-09-25 23:37:08 +02:00
} ) ;
2019-05-10 19:21:17 +02:00
}
2019-06-08 03:45:35 +02:00
if ( Debug . debug && Debug . debugStorage ) {
try {
console . log ( "[Settings::get] Got settings:" , JSON . parse ( ret . uwSettings ) ) ;
} catch ( e ) {
console . log ( "[Settings::get] No settings." )
}
}
2019-05-10 19:21:17 +02:00
try {
return JSON . parse ( ret . uwSettings ) ;
} catch ( e ) {
return undefined ;
2018-08-05 23:48:56 +02:00
}
}
async set ( extensionConf ) {
2019-06-08 03:45:35 +02:00
if ( Debug . debug && Debug . debugStorage ) {
2018-08-23 01:04:37 +02:00
console . log ( "[Settings::set] setting new settings:" , extensionConf )
}
2018-12-30 23:16:09 +01:00
if ( currentBrowser . firefox || currentBrowser . edge ) {
2018-08-22 22:34:07 +02:00
extensionConf . version = this . version ;
2019-05-03 00:58:06 +02:00
return this . useSync ? browser . storage . local . set ( { 'uwSettings' : JSON . stringify ( extensionConf ) } ) : browser . storage . local . set ( { 'uwSettings' : JSON . stringify ( extensionConf ) } ) ;
2018-12-30 23:16:09 +01:00
} else if ( currentBrowser . chrome ) {
2019-05-03 00:58:06 +02:00
return chrome . storage . local . set ( { 'uwSettings' : JSON . stringify ( extensionConf ) } ) ;
2018-08-05 23:48:56 +02:00
}
}
2018-08-07 23:31:28 +02:00
async setActive ( activeSettings ) {
this . active = activeSettings ;
}
async setProp ( prop , value ) {
this . active [ prop ] = value ;
}
async save ( ) {
2019-06-08 03:45:35 +02:00
if ( Debug . debug && Debug . storage ) {
2018-08-23 01:04:37 +02:00
console . log ( "[Settings::save] Saving active settings:" , this . active ) ;
}
2018-08-07 23:31:28 +02:00
this . set ( this . active ) ;
}
2019-04-12 00:49:56 +02:00
async rollback ( ) {
this . active = await this . get ( ) ;
}
2018-08-05 23:48:56 +02:00
getDefaultSettings ( ) {
return JSON . parse ( JSON . stringify ( this . default ) ) ;
}
setDefaultSettings ( ) {
this . set ( this . default ) ;
}
2018-09-17 00:39:32 +02:00
2018-08-05 23:48:56 +02:00
// -----------------------------------------
// Nastavitve za posamezno stran
// Config for a given page:
//
// <hostname> : {
// status: <option> // should extension work on this site?
// arStatus: <option> // should we do autodetection on this site?
// statusEmbedded: <option> // reserved for future... maybe
// }
//
// Veljavne vrednosti za možnosti
// Valid values for options:
//
// status, arStatus, statusEmbedded:
//
// * enabled — always allow
2018-11-02 02:52:01 +01:00
// * basic — only allow fullscreen
2018-08-05 23:48:56 +02:00
// * default — allow if default is to allow, block if default is to block
// * disabled — never allow
2018-09-17 00:39:32 +02:00
2018-11-18 03:29:16 +01:00
getActionsForSite ( site ) {
if ( ! site ) {
return this . active . actions ;
}
if ( this . active . sites [ site ] && this . active . sites [ site ] . actions && this . active . sites [ site ] . actions . length > 0 ) {
return this . active . sites [ site ] . actions ;
}
return this . active . actions ;
}
2018-11-02 02:52:01 +01:00
getExtensionMode ( site ) {
if ( ! site ) {
site = window . location . hostname ;
if ( ! site ) {
2019-06-10 23:45:15 +02:00
if ( Debug . debug ) {
console . log ( "[Settings::canStartExtension] window.location.hostname is null or undefined:" , window . location . hostname )
console . log ( "active settings:" , this . active )
}
2018-11-02 02:52:01 +01:00
return ExtensionMode . Disabled ;
}
}
try {
// if site-specific settings don't exist for the site, we use default mode:
if ( ! this . active . sites [ site ] ) {
2019-06-05 23:36:47 +02:00
return this . getExtensionMode ( '@global' ) ;
2018-11-02 02:52:01 +01:00
}
2019-01-20 23:36:08 +01:00
if ( this . active . sites [ site ] . mode === ExtensionMode . Enabled ) {
2019-01-20 22:59:06 +01:00
return ExtensionMode . Enabled ;
2019-01-20 23:36:08 +01:00
} else if ( this . active . sites [ site ] . mode === ExtensionMode . Basic ) {
2019-06-05 23:36:47 +02:00
return ExtensionMode . Basic ;
} else if ( this . active . sites [ site ] . mode === ExtensionMode . Default && site !== '@global' ) {
return this . getExtensionMode ( '@global' ) ;
2018-11-02 02:52:01 +01:00
} else {
return ExtensionMode . Disabled ;
}
} catch ( e ) {
if ( Debug . debug ) {
2019-06-05 23:36:47 +02:00
console . log ( "[Settings.js::canStartExtension] Something went wrong — are settings defined/has init() been called?\n\nerror:" , e , "\n\nSettings object:" , this )
2018-11-02 02:52:01 +01:00
}
return ExtensionMode . Disabled ;
}
}
2018-08-05 23:48:56 +02:00
canStartExtension ( site ) {
// returns 'true' if extension can be started on a given site. Returns false if we shouldn't run.
if ( ! site ) {
site = window . location . hostname ;
if ( ! site ) {
2018-08-07 23:31:28 +02:00
console . log ( "[Settings::canStartExtension] window.location.hostname is null or undefined:" , window . location . hostname )
console . log ( "active settings:" , this . active )
2018-08-05 23:48:56 +02:00
return false ;
}
}
2018-11-02 02:52:01 +01:00
// if (Debug.debug) {
// // let's just temporarily disable debugging while recursively calling
// // this function to get extension status on current site without duplo
// // console logs (and without endless recursion)
// Debug.debug = false;
// const cse = this.canStartExtension(site);
// Debug.debug = true;
// }
2018-08-07 23:31:28 +02:00
try {
2018-08-05 23:48:56 +02:00
// if site is not defined, we use default mode:
2019-07-03 22:35:17 +02:00
if ( ! this . active . sites [ site ] || this . active . sites [ site ] . mode === ExtensionMode . Default ) {
2019-06-10 23:45:15 +02:00
return this . active . sites [ '@global' ] . mode === ExtensionMode . Enabled ;
}
2018-08-05 23:48:56 +02:00
2019-06-15 22:58:19 +02:00
if ( this . active . sites [ '@global' ] . mode === ExtensionMode . Enabled ) {
2019-06-10 23:45:15 +02:00
return this . active . sites [ site ] . mode !== ExtensionMode . Disabled ;
} else if ( this . active . sites [ '@global' ] . mode === ExtensionMode . Whitelist ) {
return this . active . sites [ site ] . mode === ExtensionMode . Enabled ;
} else {
return false ;
}
2019-06-02 23:54:22 +02:00
} catch ( e ) {
2018-08-22 22:34:07 +02:00
if ( Debug . debug ) {
console . log ( "[Settings.js::canStartExtension] Something went wrong — are settings defined/has init() been called?\nSettings object:" , this )
}
return false ;
}
2018-08-05 23:48:56 +02:00
}
2019-06-02 23:54:32 +02:00
keyboardShortcutsEnabled ( site ) {
if ( ! site ) {
site = window . location . hostname ;
}
if ( ! site ) {
return false ;
}
try {
if ( ! this . active . sites [ site ]
|| this . active . sites [ site ] . keyboardShortcutsEnabled === undefined
|| this . active . sites [ site ] . keyboardShortcutsEnabled === ExtensionMode . Default ) {
return this . keyboardShortcutsEnabled ( '@global' ) ;
} else {
return this . active . sites [ site ] . keyboardShortcutsEnabled === ExtensionMode . Enabled ;
}
} catch ( e ) {
if ( Debug . debug ) {
console . error ( "[Settings.js::keyboardDisabled] something went wrong:" , e ) ;
}
return false ;
}
}
2018-09-16 14:14:16 +02:00
extensionEnabled ( ) {
2019-01-20 23:36:08 +01:00
return this . active . sites [ '@global' ] !== ExtensionMode . Disabled
2018-09-16 14:14:16 +02:00
}
extensionEnabledForSite ( site ) {
return this . canStartExtension ( site ) ;
}
2018-08-05 23:48:56 +02:00
canStartAutoAr ( site ) {
2019-05-26 02:53:29 +02:00
// 'site' argument is only ever used when calling this function recursively for debugging
2018-08-05 23:48:56 +02:00
if ( ! site ) {
2019-02-15 00:00:22 +01:00
site = window . location . host ;
2018-08-05 23:48:56 +02:00
if ( ! site ) {
return false ;
}
}
if ( Debug . debug ) {
// let's just temporarily disable debugging while recursively calling
// this function to get extension status on current site without duplo
// console logs (and without endless recursion)
Debug . debug = false ;
const csar = this . canStartAutoAr ( site ) ;
Debug . debug = true ;
2019-02-15 00:00:22 +01:00
console . log ( "[Settings::canStartAutoAr] ----------------\nCAN WE START AUTOAR ON SITE" , site ,
"?\n\nsettings.active.sites[site]=" , this . active . sites [ site ] , "settings.active.sites[@global]=" , this . active . sites [ '@global' ] ,
"\nAutoar mode (global)?" , this . active . sites [ '@global' ] . autoar ,
` \n Autoar mode ( ${ site } ) ` , this . active . sites [ site ] ? this . active . sites [ site ] . autoar : '<not defined>' ,
"\nCan autoar be started?" , csar
2018-08-05 23:48:56 +02:00
) ;
}
// if site is not defined, we use default mode:
if ( ! this . active . sites [ site ] ) {
2019-02-15 00:00:22 +01:00
return this . active . sites [ '@global' ] . autoar === ExtensionMode . Enabled ;
2018-08-05 23:48:56 +02:00
}
2019-02-15 00:00:22 +01:00
if ( this . active . sites [ '@global' ] . autoar === ExtensionMode . Enabled ) {
2019-01-20 23:36:08 +01:00
return this . active . sites [ site ] . autoar !== ExtensionMode . Disabled ;
} else if ( this . active . sites [ '@global' ] . autoar === ExtensionMode . Whitelist ) {
2019-02-15 00:00:22 +01:00
console . log ( "canStartAutoAr — can(not) start csar because extension is in whitelist mode, and this site is (not) equal to" , ExtensionMode . Enabled )
return this . active . sites [ site ] . autoar === ExtensionMode . Enabled ;
2018-08-05 23:48:56 +02:00
} else {
2019-02-15 00:00:22 +01:00
console . log ( "canStartAutoAr — cannot start csar because extension is globally disabled" )
2018-08-05 23:48:56 +02:00
return false ;
}
}
2018-09-17 00:39:32 +02:00
2019-02-13 23:58:19 +01:00
getDefaultOption ( option ) {
const allDefault = {
mode : ExtensionMode . Default ,
autoar : ExtensionMode . Default ,
autoarFallback : ExtensionMode . Default ,
stretch : Stretch . Default ,
videoAlignment : VideoAlignment . Default ,
} ;
if ( ! option || allDefault [ option ] === undefined ) {
return allDefault ;
}
return allDefault [ option ] ;
}
2018-09-17 00:39:32 +02:00
getDefaultAr ( site ) {
2019-02-15 20:40:56 +01:00
// site = this.getSiteSettings(site);
2018-09-17 00:39:32 +02:00
2019-02-15 20:40:56 +01:00
// if (site.defaultAr) {
// return site.defaultAr;
// }
2018-12-03 00:31:28 +01:00
return this . active . miscSettings . defaultAr ;
2018-09-17 00:39:32 +02:00
}
2018-09-18 00:40:05 +02:00
getDefaultStretchMode ( site ) {
2019-02-19 21:10:49 +01:00
if ( site && this . active . sites [ site ] && this . active . sites [ site ] . stretch !== Stretch . Default ) {
2019-02-15 20:40:56 +01:00
return this . active . sites [ site ] . stretch ;
2018-09-17 00:39:32 +02:00
}
2019-02-15 20:40:56 +01:00
2019-01-20 23:05:04 +01:00
return this . active . sites [ '@global' ] . stretch ;
2018-09-17 00:39:32 +02:00
}
getDefaultVideoAlignment ( site ) {
2019-02-15 20:40:56 +01:00
if ( site && this . active . sites [ site ] && this . active . sites [ site ] . videoAlignment !== VideoAlignment . Default ) {
return this . active . sites [ site ] . videoAlignment ;
2018-09-17 00:39:32 +02:00
}
2019-02-15 20:40:56 +01:00
2019-01-20 23:05:04 +01:00
return this . active . sites [ '@global' ] . videoAlignment ;
2018-09-17 00:39:32 +02:00
}
2018-08-05 23:48:56 +02:00
}
2018-12-30 23:16:09 +01:00
2018-12-31 01:03:07 +01:00
export default Settings ;