2018-08-13 10:34:00 +00:00
'use strict' ;
var me = { } ;
var phraseBox = { } ;
2019-02-28 07:28:17 +00:00
if ( tmplInits === undefined ) var tmplInits = { } ;
2018-08-13 10:34:00 +00:00
var tmplPhrases = [ ] ; // [key] array of phrases indexed by order of use
2019-03-21 22:59:41 +00:00
var hooks = { // Shorten this list by binding the hooks just in time?
2018-08-13 10:34:00 +00:00
"pre_iffe" : [ ] ,
"pre_init" : [ ] ,
"start_init" : [ ] ,
2019-04-25 06:02:51 +00:00
"almost_end_init" : [ ] ,
2018-08-13 10:34:00 +00:00
"end_init" : [ ] ,
2018-11-30 03:02:20 +00:00
"after_phrases" : [ ] ,
2018-09-20 10:45:33 +00:00
"after_add_alert" : [ ] ,
"after_update_alert_list" : [ ] ,
2018-12-27 05:42:41 +00:00
"open_edit" : [ ] ,
"close_edit" : [ ] ,
2019-03-04 07:47:38 +00:00
"edit_item_pre_bind" : [ ] ,
2019-03-21 22:59:41 +00:00
"analytics_loaded" : [ ] ,
2018-08-13 10:34:00 +00:00
} ;
var ranInitHooks = { }
function runHook ( name , ... args ) {
if ( ! ( name in hooks ) ) {
console . log ( "Couldn't find hook '" + name + "'" ) ;
return ;
}
console . log ( "Running hook '" + name + "'" ) ;
let hook = hooks [ name ] ;
2019-04-27 06:32:26 +00:00
let ret ;
for ( const index in hook ) ret = hook [ index ] ( ... args ) ;
return ret ;
2018-08-13 10:34:00 +00:00
}
function addHook ( name , callback ) {
2019-04-25 06:02:51 +00:00
if ( hooks [ name ] === undefined ) hooks [ name ] = [ ] ;
2018-08-13 10:34:00 +00:00
hooks [ name ] . push ( callback ) ;
}
// InitHooks are slightly special, as if they are run, then any adds after the initial run will run immediately, this is to deal with the async nature of script loads
2018-11-30 03:02:20 +00:00
function runInitHook ( name , ... args ) {
2019-04-27 06:32:26 +00:00
let ret = runHook ( name , ... args ) ;
2018-08-13 10:34:00 +00:00
ranInitHooks [ name ] = true ;
2019-04-27 06:32:26 +00:00
return ret ;
2018-08-13 10:34:00 +00:00
}
function addInitHook ( name , callback ) {
addHook ( name , callback ) ;
2019-04-27 06:32:26 +00:00
if ( name in ranInitHooks ) callback ( ) ;
2018-08-13 10:34:00 +00:00
}
// Temporary hack for templates
function len ( item ) {
return item . length ;
}
2019-02-28 07:28:17 +00:00
function asyncGetScript ( source ) {
2018-08-13 10:34:00 +00:00
return new Promise ( ( resolve , reject ) => {
let script = document . createElement ( 'script' ) ;
script . async = true ;
2019-02-28 07:28:17 +00:00
const onloadHandler = ( e , isAbort ) => {
2018-08-13 10:34:00 +00:00
if ( isAbort || ! script . readyState || /loaded|complete/ . test ( script . readyState ) ) {
script . onload = null ;
script . onreadystatechange = null ;
script = undefined ;
2019-02-28 07:28:17 +00:00
isAbort ? reject ( e ) : resolve ( ) ;
2018-08-13 10:34:00 +00:00
}
}
2019-02-28 07:28:17 +00:00
script . onerror = ( e ) => {
reject ( e ) ;
2018-08-13 10:34:00 +00:00
} ;
2019-02-28 07:28:17 +00:00
script . onload = onloadHandler ;
script . onreadystatechange = onloadHandler ;
2018-08-13 10:34:00 +00:00
script . src = source ;
const prior = document . getElementsByTagName ( 'script' ) [ 0 ] ;
prior . parentNode . insertBefore ( script , prior ) ;
} ) ;
2019-02-28 07:28:17 +00:00
}
2018-08-13 10:34:00 +00:00
2019-02-28 07:28:17 +00:00
function notifyOnScript ( source ) {
2019-08-14 10:39:04 +00:00
source = "/s/" + source ;
2019-02-28 07:28:17 +00:00
return new Promise ( ( resolve , reject ) => {
2019-08-14 10:39:04 +00:00
let ss = source . replace ( "/s/" , "" ) ;
2019-03-16 11:31:10 +00:00
try {
let ssp = ss . charAt ( 0 ) . toUpperCase ( ) + ss . slice ( 1 )
console . log ( "ssp:" , ssp )
if ( window [ ssp ] ) {
resolve ( ) ;
return ;
}
} catch ( e ) { }
console . log ( "source:" , source )
2019-02-28 07:28:17 +00:00
let script = document . querySelectorAll ( '[src^="' + source + '"]' ) [ 0 ] ;
2019-03-16 11:31:10 +00:00
console . log ( "script:" , script ) ;
2019-02-28 07:28:17 +00:00
if ( script === undefined ) {
reject ( "no script found" ) ;
return ;
}
2019-03-16 11:31:10 +00:00
const onloadHandler = ( e ) => {
script . onload = null ;
script . onreadystatechange = null ;
resolve ( ) ;
2019-02-28 07:28:17 +00:00
}
script . onerror = ( e ) => {
reject ( e ) ;
} ;
script . onload = onloadHandler ;
script . onreadystatechange = onloadHandler ;
} ) ;
}
function notifyOnScriptW ( name , complete , success ) {
notifyOnScript ( name )
. then ( ( ) => {
console . log ( "Loaded " + name + ".js" ) ;
complete ( ) ;
if ( success !== undefined ) success ( ) ;
} ) . catch ( ( e ) => {
console . log ( "Unable to get script name '" + name + "'" ) ;
console . log ( "e: " , e ) ;
console . trace ( ) ;
2019-03-16 11:31:10 +00:00
complete ( e ) ;
2019-02-28 07:28:17 +00:00
} ) ;
}
// TODO: Send data at load time so we don't have to rely on a fallback template here
function loadScript ( name , callback , fail ) {
let fname = name ;
let value = "; " + document . cookie ;
let parts = value . split ( "; current_theme=" ) ;
if ( parts . length == 2 ) fname += "_" + parts . pop ( ) . split ( ";" ) . shift ( ) ;
2019-08-14 10:39:04 +00:00
let url = "/s/" + fname + ".js"
let iurl = "/s/" + name + ".js"
2018-08-13 10:34:00 +00:00
asyncGetScript ( url )
. then ( callback )
2019-02-28 07:28:17 +00:00
. catch ( ( e ) => {
2018-08-13 10:34:00 +00:00
console . log ( "Unable to get script '" + url + "'" ) ;
2019-02-28 07:28:17 +00:00
if ( fname != name ) {
asyncGetScript ( iurl )
. then ( callback )
. catch ( ( e ) => {
console . log ( "Unable to get script '" + iurl + "'" ) ;
console . log ( "e: " , e ) ;
console . trace ( ) ;
} ) ;
}
console . log ( "e: " , e ) ;
2018-08-13 10:34:00 +00:00
console . trace ( ) ;
2019-02-28 07:28:17 +00:00
fail ( e ) ;
2018-08-13 10:34:00 +00:00
} ) ;
}
2019-02-10 05:52:26 +00:00
/ *
function loadTmpl ( name , callback ) {
2019-08-14 10:39:04 +00:00
let url = "/s/" + name
2019-02-10 05:52:26 +00:00
let worker = new Worker ( url ) ;
}
* /
2018-08-13 10:34:00 +00:00
function DoNothingButPassBack ( item ) {
return item ;
}
2019-02-10 05:52:26 +00:00
function RelativeTime ( date ) {
return date ;
}
2019-04-27 06:32:26 +00:00
function initPhrases ( loggedIn , panel = false ) {
2019-02-28 07:28:17 +00:00
console . log ( "in initPhrases" )
2019-03-16 11:31:10 +00:00
console . log ( "tmlInits:" , tmplInits )
2019-04-15 01:54:13 +00:00
let e = "" ;
2019-05-06 04:04:00 +00:00
if ( loggedIn && ! panel ) e = ",status,topic_list,topic" ;
2019-04-27 06:32:26 +00:00
else if ( panel ) e = ",analytics,panel" ; // TODO: Request phrases for just one section of the control panel?
2019-04-28 02:57:25 +00:00
else e = ",status,topic_list" ;
fetchPhrases ( "alerts,paginator" + e ) // TODO: Break this up?
2019-01-21 12:27:59 +00:00
}
function fetchPhrases ( plist ) {
2019-07-11 21:38:06 +00:00
fetch ( "/api/phrases/?q=" + plist , { cache : "no-cache" } )
2018-08-13 10:34:00 +00:00
. then ( ( resp ) => resp . json ( ) )
. then ( ( data ) => {
console . log ( "loaded phrase endpoint data" ) ;
console . log ( "data:" , data ) ;
Object . keys ( tmplInits ) . forEach ( ( key ) => {
let phrases = [ ] ;
let tmplInit = tmplInits [ key ] ;
2019-01-21 12:27:59 +00:00
for ( let phraseName of tmplInit ) phrases . push ( data [ phraseName ] ) ;
2018-08-13 10:34:00 +00:00
console . log ( "Adding phrases" ) ;
console . log ( "key:" , key ) ;
console . log ( "phrases:" , phrases ) ;
tmplPhrases [ key ] = phrases ;
} ) ;
let prefixes = { } ;
Object . keys ( data ) . forEach ( ( key ) => {
let prefix = key . split ( "." ) [ 0 ] ;
2019-01-21 12:27:59 +00:00
if ( prefixes [ prefix ] === undefined ) prefixes [ prefix ] = { } ;
2018-08-13 10:34:00 +00:00
prefixes [ prefix ] [ key ] = data [ key ] ;
} ) ;
Object . keys ( prefixes ) . forEach ( ( prefix ) => {
console . log ( "adding phrase prefix '" + prefix + "' to box" ) ;
phraseBox [ prefix ] = prefixes [ prefix ] ;
} ) ;
2018-11-30 03:02:20 +00:00
runInitHook ( "after_phrases" ) ;
2018-08-13 10:34:00 +00:00
} ) ;
}
( ( ) => {
runInitHook ( "pre_iife" ) ;
2019-04-15 01:54:13 +00:00
let loggedIn = document . head . querySelector ( "[property='x-loggedin']" ) . content == "true" ;
2019-04-27 06:32:26 +00:00
let panel = window . location . pathname . startsWith ( "/panel/" ) ;
2019-04-15 01:54:13 +00:00
2019-04-27 06:32:26 +00:00
let toLoad = 1 ;
// TODO: Shunt this into loggedIn if there aren't any search and filter widgets?
let q = ( f ) => {
toLoad -- ;
if ( toLoad === 0 ) initPhrases ( loggedIn , panel ) ;
if ( f ) throw ( "template function not found" ) ;
} ;
if ( ! panel ) {
toLoad += 2 ;
2019-04-20 02:22:57 +00:00
if ( loggedIn ) {
2019-06-16 07:26:31 +00:00
toLoad += 3 ;
2019-04-20 02:22:57 +00:00
notifyOnScriptW ( "template_topic_c_edit_post" , ( ) => q ( ! Template _topic _c _edit _post ) ) ;
notifyOnScriptW ( "template_topic_c_attach_item" , ( ) => q ( ! Template _topic _c _attach _item ) ) ;
2019-06-16 07:26:31 +00:00
notifyOnScriptW ( "template_topic_c_poll_input" , ( ) => q ( ! Template _topic _c _poll _input ) ) ;
2019-04-20 02:22:57 +00:00
}
notifyOnScriptW ( "template_topics_topic" , ( ) => q ( ! Template _topics _topic ) ) ;
notifyOnScriptW ( "template_paginator" , ( ) => q ( ! Template _paginator ) ) ;
2019-04-15 01:54:13 +00:00
}
2019-04-27 06:32:26 +00:00
notifyOnScriptW ( "template_notice" , ( ) => q ( ! Template _notice ) ) ;
2019-02-10 05:52:26 +00:00
2019-04-15 01:54:13 +00:00
if ( loggedIn ) {
2018-08-13 12:01:27 +00:00
fetch ( "/api/me/" )
2018-08-13 10:34:00 +00:00
. then ( ( resp ) => resp . json ( ) )
. then ( ( data ) => {
console . log ( "loaded me endpoint data" ) ;
console . log ( "data:" , data ) ;
me = data ;
runInitHook ( "pre_init" ) ;
} ) ;
2018-08-13 12:01:27 +00:00
} else {
2019-08-31 22:59:00 +00:00
me = { User : { ID : 0 , S : "" } , Site : { "MaxRequestSize" : 0 } } ;
2018-08-13 12:01:27 +00:00
runInitHook ( "pre_init" ) ;
2018-08-13 10:34:00 +00:00
}
} ) ( ) ;