2017-08-18 12:16:56 +00:00
'use strict' ;
2018-02-22 02:27:17 +00:00
var formVars = { } ;
2018-06-24 13:49:29 +00:00
var tmplInits = { } ;
2018-06-30 03:40:50 +00:00
var tmplPhrases = [ ] ; // [key] array of phrases indexed by order of use
var phraseBox = { } ;
2017-06-10 07:58:15 +00:00
var alertList = [ ] ;
var alertCount = 0 ;
2018-06-30 03:40:50 +00:00
var moreTopicCount = 0 ;
2017-06-12 09:03:14 +00:00
var conn ;
2017-10-30 09:57:08 +00:00
var selectedTopics = [ ] ;
var attachItemCallback = function ( ) { }
2018-02-22 02:27:17 +00:00
var hooks = {
"start_init" : [ ] ,
"end_init" : [ ] ,
} ;
2017-10-30 09:57:08 +00:00
2018-01-14 12:03:20 +00:00
// Topic move
var forumToMoveTo = 0 ;
2018-02-22 02:27:17 +00:00
function runHook ( name , ... args ) {
if ( ! ( name in hooks ) ) return ;
let hook = hooks [ name ] ;
for ( const callback in hook ) {
callback ( ... args ) ;
}
}
2017-10-30 09:57:08 +00:00
// TODO: Write a friendlier error handler which uses a .notice or something, we could have a specialised one for alerts
function ajaxError ( xhr , status , errstr ) {
console . log ( "The AJAX request failed" ) ;
console . log ( "xhr" , xhr ) ;
console . log ( "status" , status ) ;
console . log ( "errstr" , errstr ) ;
if ( status == "parsererror" ) {
console . log ( "The server didn't respond with a valid JSON response" ) ;
}
console . trace ( ) ;
}
2017-01-31 05:13:38 +00:00
2017-10-30 09:57:08 +00:00
function postLink ( event )
2016-12-02 07:38:54 +00:00
{
event . preventDefault ( ) ;
2017-10-30 09:57:08 +00:00
let formAction = $ ( event . target ) . closest ( 'a' ) . attr ( "href" ) ;
//console.log("Form Action: " + formAction);
$ . ajax ( { url : formAction , type : "POST" , dataType : "json" , error : ajaxError , data : { js : "1" } } ) ;
2016-12-02 07:38:54 +00:00
}
2017-10-30 09:57:08 +00:00
function bindToAlerts ( ) {
2017-08-18 12:16:56 +00:00
$ ( ".alertItem.withAvatar a" ) . click ( function ( event ) {
event . stopPropagation ( ) ;
2017-10-30 09:57:08 +00:00
$ . ajax ( { url : "/api/?action=set&module=dismiss-alert" , type : "POST" , dataType : "json" , error : ajaxError , data : { asid : $ ( this ) . attr ( "data-asid" ) } } ) ;
2017-08-18 12:16:56 +00:00
} ) ;
}
2018-05-14 08:56:56 +00:00
var alertsInitted = false ;
2017-09-10 16:57:22 +00:00
// TODO: Add the ability for users to dismiss alerts
2017-11-06 16:24:45 +00:00
function loadAlerts ( menuAlerts )
2017-03-03 16:28:49 +00:00
{
2018-05-14 08:56:56 +00:00
if ( ! alertsInitted ) return ;
2017-11-06 16:24:45 +00:00
var alertListNode = menuAlerts . getElementsByClassName ( "alertList" ) [ 0 ] ;
var alertCounterNode = menuAlerts . getElementsByClassName ( "alert_counter" ) [ 0 ] ;
2017-08-17 11:13:49 +00:00
alertCounterNode . textContent = "0" ;
2017-03-03 16:28:49 +00:00
$ . ajax ( {
2017-11-06 16:24:45 +00:00
type : 'get' ,
dataType : 'json' ,
url : '/api/?action=get&module=alerts' ,
2018-05-14 08:56:56 +00:00
success : ( data ) => {
2017-11-06 16:24:45 +00:00
if ( "errmsg" in data ) {
alertListNode . innerHTML = "<div class='alertItem'>" + data . errmsg + "</div>" ;
return ;
}
2017-05-29 14:52:37 +00:00
2017-11-06 16:24:45 +00:00
var alist = "" ;
for ( var i in data . msgs ) {
var msg = data . msgs [ i ] ;
var mmsg = msg . msg ;
if ( "sub" in msg ) {
for ( var i = 0 ; i < msg . sub . length ; i ++ ) {
mmsg = mmsg . replace ( "\{" + i + "\}" , msg . sub [ i ] ) ;
//console.log("Sub #" + i + ":",msg.sub[i]);
2017-03-03 16:28:49 +00:00
}
}
2018-06-24 13:49:29 +00:00
let aItem = Template _alert ( {
2018-05-14 08:56:56 +00:00
ASID : msg . asid || 0 ,
Path : msg . path ,
Avatar : msg . avatar || "" ,
Message : mmsg
} )
2018-06-24 13:49:29 +00:00
alist += aItem ;
alertList . push ( aItem ) ;
2017-11-06 16:24:45 +00:00
//console.log(msg);
//console.log(mmsg);
}
2017-08-18 12:16:56 +00:00
2017-11-06 16:24:45 +00:00
if ( alist == "" ) alist = "<div class='alertItem'>You don't have any alerts</div>" ;
alertListNode . innerHTML = alist ;
if ( data . msgCount != 0 && data . msgCount != undefined ) {
alertCounterNode . textContent = data . msgCount ;
menuAlerts . classList . add ( "has_alerts" ) ;
} else {
menuAlerts . classList . remove ( "has_alerts" ) ;
2017-03-03 16:28:49 +00:00
}
2017-11-06 16:24:45 +00:00
alertCount = data . msgCount ;
bindToAlerts ( ) ;
} ,
2018-05-14 08:56:56 +00:00
error : ( magic , theStatus , error ) => {
2017-11-06 16:24:45 +00:00
let errtxt
try {
var data = JSON . parse ( magic . responseText ) ;
if ( "errmsg" in data ) errtxt = data . errmsg ;
else errtxt = "Unable to get the alerts" ;
} catch ( err ) {
errtxt = "Unable to get the alerts" ;
console . log ( magic . responseText ) ;
console . log ( err ) ;
}
console . log ( "error" , error ) ;
alertListNode . innerHTML = "<div class='alertItem'>" + errtxt + "</div>" ;
}
} ) ;
2017-03-03 16:28:49 +00:00
}
2017-08-15 13:47:56 +00:00
function SplitN ( data , ch , n ) {
var out = [ ] ;
Added Quick Topic.
Added Attachments.
Added Attachment Media Embeds.
Renamed a load of *Store and *Cache methods to reduce the amount of unneccesary typing.
Added petabytes as a unit and cleaned up a few of the friendly units.
Refactored the username change logic to make it easier to maintain.
Refactored the avatar change logic to make it easier to maintain.
Shadow now uses CSS Variables for most of it's colours. We have plans to transpile this to support older browsers later on!
Snuck some CSS Variables into Tempra Conflux.
Added the GroupCache interface to MemoryGroupStore.
Added the Length method to MemoryGroupStore.
Added support for a site short name.
Added the UploadFiles permission.
Renamed more functions.
Fixed the background for the left gutter on the postbit for Tempra Simple and Shadow.
Added support for if statements operating on int8, int16, int32, int32, int64, uint, uint8, uint16, uint32, uint64, float32, and float64 for the template compiler.
Added support for if statements operating on slices and maps for the template compiler.
Fixed a security exploit in reply editing.
Fixed a bug in the URL detector in the parser where it couldn't find URLs with non-standard ports.
Fixed buttons having blue outlines on focus on Shadow.
Refactored the topic creation logic to make it easier to maintain.
Made a few responsive fixes, but there's still more to do in the following commits!
2017-10-05 10:20:28 +00:00
if ( data . length === 0 ) return out ;
2017-08-15 13:47:56 +00:00
var lastIndex = 0 ;
var j = 0 ;
var lastN = 1 ;
Added Quick Topic.
Added Attachments.
Added Attachment Media Embeds.
Renamed a load of *Store and *Cache methods to reduce the amount of unneccesary typing.
Added petabytes as a unit and cleaned up a few of the friendly units.
Refactored the username change logic to make it easier to maintain.
Refactored the avatar change logic to make it easier to maintain.
Shadow now uses CSS Variables for most of it's colours. We have plans to transpile this to support older browsers later on!
Snuck some CSS Variables into Tempra Conflux.
Added the GroupCache interface to MemoryGroupStore.
Added the Length method to MemoryGroupStore.
Added support for a site short name.
Added the UploadFiles permission.
Renamed more functions.
Fixed the background for the left gutter on the postbit for Tempra Simple and Shadow.
Added support for if statements operating on int8, int16, int32, int32, int64, uint, uint8, uint16, uint32, uint64, float32, and float64 for the template compiler.
Added support for if statements operating on slices and maps for the template compiler.
Fixed a security exploit in reply editing.
Fixed a bug in the URL detector in the parser where it couldn't find URLs with non-standard ports.
Fixed buttons having blue outlines on focus on Shadow.
Refactored the topic creation logic to make it easier to maintain.
Made a few responsive fixes, but there's still more to do in the following commits!
2017-10-05 10:20:28 +00:00
for ( let i = 0 ; i < data . length ; i ++ ) {
if ( data [ i ] === ch ) {
2017-08-15 13:47:56 +00:00
out [ j ++ ] = data . substring ( lastIndex , i ) ;
lastIndex = i ;
Added Quick Topic.
Added Attachments.
Added Attachment Media Embeds.
Renamed a load of *Store and *Cache methods to reduce the amount of unneccesary typing.
Added petabytes as a unit and cleaned up a few of the friendly units.
Refactored the username change logic to make it easier to maintain.
Refactored the avatar change logic to make it easier to maintain.
Shadow now uses CSS Variables for most of it's colours. We have plans to transpile this to support older browsers later on!
Snuck some CSS Variables into Tempra Conflux.
Added the GroupCache interface to MemoryGroupStore.
Added the Length method to MemoryGroupStore.
Added support for a site short name.
Added the UploadFiles permission.
Renamed more functions.
Fixed the background for the left gutter on the postbit for Tempra Simple and Shadow.
Added support for if statements operating on int8, int16, int32, int32, int64, uint, uint8, uint16, uint32, uint64, float32, and float64 for the template compiler.
Added support for if statements operating on slices and maps for the template compiler.
Fixed a security exploit in reply editing.
Fixed a bug in the URL detector in the parser where it couldn't find URLs with non-standard ports.
Fixed buttons having blue outlines on focus on Shadow.
Refactored the topic creation logic to make it easier to maintain.
Made a few responsive fixes, but there's still more to do in the following commits!
2017-10-05 10:20:28 +00:00
if ( lastN === n ) break ;
2017-08-15 13:47:56 +00:00
lastN ++ ;
2017-05-11 13:04:43 +00:00
}
}
2017-08-15 13:47:56 +00:00
if ( data . length > lastIndex ) out [ out . length - 1 ] += data . substring ( lastIndex ) ;
return out ;
}
2017-05-29 14:52:37 +00:00
2018-06-24 13:49:29 +00:00
function wsAlertEvent ( data ) {
var msg = data . msg ;
if ( "sub" in data ) {
for ( var i = 0 ; i < data . sub . length ; i ++ ) {
msg = msg . replace ( "\{" + i + "\}" , data . sub [ i ] ) ;
}
}
let aItem = Template _alert ( {
ASID : data . asid || 0 ,
Path : data . path ,
Avatar : data . avatar || "" ,
Message : msg
} )
alertList . push ( aItem ) ;
if ( alertList . length > 8 ) alertList . shift ( ) ;
//console.log("post alertList",alertList);
alertCount ++ ;
var alist = "" ;
for ( var i = 0 ; i < alertList . length ; i ++ ) alist += alertList [ i ] ;
//console.log(alist);
// TODO: Add support for other alert feeds like PM Alerts
var generalAlerts = document . getElementById ( "general_alerts" ) ;
var alertListNode = generalAlerts . getElementsByClassName ( "alertList" ) [ 0 ] ;
var alertCounterNode = generalAlerts . getElementsByClassName ( "alert_counter" ) [ 0 ] ;
alertListNode . innerHTML = alist ;
alertCounterNode . textContent = alertCount ;
// TODO: Add some sort of notification queue to avoid flooding the end-user with notices?
// TODO: Use the site name instead of "Something Happened"
if ( Notification . permission === "granted" ) {
var n = new Notification ( "Something Happened" , {
body : msg ,
icon : data . avatar ,
} ) ;
setTimeout ( n . close . bind ( n ) , 8000 ) ;
}
bindToAlerts ( ) ;
}
2017-11-06 16:24:45 +00:00
function runWebSockets ( ) {
2018-06-24 13:49:29 +00:00
if ( window . location . protocol == "https:" ) {
2017-11-06 16:24:45 +00:00
conn = new WebSocket ( "wss://" + document . location . host + "/ws/" ) ;
2018-06-24 13:49:29 +00:00
} else conn = new WebSocket ( "ws://" + document . location . host + "/ws/" ) ;
2017-06-10 07:58:15 +00:00
2018-06-24 13:49:29 +00:00
conn . onopen = ( ) => {
2017-11-06 16:24:45 +00:00
console . log ( "The WebSockets connection was opened" ) ;
conn . send ( "page " + document . location . pathname + '\r' ) ;
// TODO: Don't ask again, if it's denied. We could have a setting in the UCP which automatically requests this when someone flips desktop notifications on
2018-06-24 13:49:29 +00:00
if ( loggedIn ) {
Notification . requestPermission ( ) ;
}
2017-11-06 16:24:45 +00:00
}
2018-06-24 13:49:29 +00:00
conn . onclose = ( ) => {
2017-11-06 16:24:45 +00:00
conn = false ;
console . log ( "The WebSockets connection was closed" ) ;
}
2018-06-24 13:49:29 +00:00
conn . onmessage = ( event ) => {
2018-03-31 05:25:27 +00:00
//console.log("WSMessage:", event.data);
2017-11-06 16:24:45 +00:00
if ( event . data [ 0 ] == "{" ) {
2018-06-24 13:49:29 +00:00
console . log ( "json message" ) ;
let data = "" ;
2017-11-06 16:24:45 +00:00
try {
2018-06-24 13:49:29 +00:00
data = JSON . parse ( event . data ) ;
2017-11-06 16:24:45 +00:00
} catch ( err ) {
console . log ( err ) ;
2018-06-24 13:49:29 +00:00
return ;
2017-11-06 16:24:45 +00:00
}
2017-08-18 12:16:56 +00:00
2017-11-06 16:24:45 +00:00
if ( "msg" in data ) {
2018-06-30 03:40:50 +00:00
// TODO: Fix the data race where the alert template hasn't been loaded yet
2018-06-24 13:49:29 +00:00
wsAlertEvent ( data ) ;
} else if ( "Topics" in data ) {
console . log ( "topic in data" ) ;
console . log ( "data:" , data ) ;
let topic = data . Topics [ 0 ] ;
if ( topic === undefined ) {
console . log ( "empty topic list" ) ;
return ;
2017-06-10 07:58:15 +00:00
}
2018-06-30 03:40:50 +00:00
// TODO: Fix the data race where the function hasn't been loaded yet
2018-06-24 13:49:29 +00:00
let renTopic = Template _topics _topic ( topic ) ;
2018-06-30 03:40:50 +00:00
$ ( ".topic_row[data-tid='" + topic . ID + "']" ) . addClass ( "ajax_topic_dupe" ) ;
2018-06-24 13:49:29 +00:00
let node = $ ( renTopic ) ;
2018-06-30 03:40:50 +00:00
node . addClass ( "new_item hide_ajax_topic" ) ;
2018-06-24 13:49:29 +00:00
console . log ( "Prepending to topic list" ) ;
$ ( ".topic_list" ) . prepend ( node ) ;
2018-06-30 03:40:50 +00:00
moreTopicCount ++ ;
let moreTopicBlocks = document . getElementsByClassName ( "more_topic_block_initial" ) ;
for ( let i = 0 ; i < moreTopicBlocks . length ; i ++ ) {
let moreTopicBlock = moreTopicBlocks [ i ] ;
moreTopicBlock . classList . remove ( "more_topic_block_initial" ) ;
moreTopicBlock . classList . add ( "more_topic_block_active" ) ;
console . log ( "phraseBox:" , phraseBox ) ;
let msgBox = moreTopicBlock . getElementsByClassName ( "more_topics" ) [ 0 ] ;
msgBox . innerText = phraseBox [ "topic_list" ] [ "topic_list.changed_topics" ] . replace ( "%d" , moreTopicCount ) ;
}
2018-06-24 13:49:29 +00:00
} else {
console . log ( "unknown message" ) ;
console . log ( data ) ;
2017-06-10 07:58:15 +00:00
}
2017-11-06 16:24:45 +00:00
}
2017-06-10 07:58:15 +00:00
2017-11-06 16:24:45 +00:00
var messages = event . data . split ( '\r' ) ;
for ( var i = 0 ; i < messages . length ; i ++ ) {
2018-03-31 05:25:27 +00:00
let message = messages [ i ] ;
//console.log("Message: ",message);
if ( message . startsWith ( "set " ) ) {
//msgblocks = message.split(' ',3);
let msgblocks = SplitN ( message , " " , 3 ) ;
2017-11-06 16:24:45 +00:00
if ( msgblocks . length < 3 ) continue ;
document . querySelector ( msgblocks [ 1 ] ) . innerHTML = msgblocks [ 2 ] ;
2018-03-31 05:25:27 +00:00
} else if ( message . startsWith ( "set-class " ) ) {
let msgblocks = SplitN ( message , " " , 3 ) ;
2017-11-06 16:24:45 +00:00
if ( msgblocks . length < 3 ) continue ;
document . querySelector ( msgblocks [ 1 ] ) . className = msgblocks [ 2 ] ;
2017-05-11 13:04:43 +00:00
}
}
}
2017-11-06 16:24:45 +00:00
}
2018-06-24 13:49:29 +00:00
// Temporary hack for templates
function len ( item ) {
return item . length ;
}
2018-05-15 05:59:52 +00:00
function loadScript ( name , callback ) {
let url = "//" + siteURL + "/static/" + name
$ . getScript ( url )
. done ( callback )
. fail ( ( e , xhr , settings , ex ) => {
console . log ( "Unable to get script '" + url + "'" ) ;
console . log ( "e: " , e ) ;
console . log ( "xhr: " , xhr ) ;
console . log ( "settings: " , settings ) ;
console . log ( "ex: " , ex ) ;
console . trace ( ) ;
} ) ;
}
2018-06-24 13:49:29 +00:00
function DoNothingButPassBack ( item ) {
return item ;
}
function fetchPhrases ( ) {
2018-06-30 03:40:50 +00:00
fetch ( "//" + siteURL + "/api/phrases/?query=status,topic_list" )
2018-06-24 13:49:29 +00:00
. then ( ( resp ) => resp . json ( ) )
. then ( ( data ) => {
2018-06-30 03:40:50 +00:00
console . log ( "loaded phrase endpoint data" ) ;
console . log ( "data:" , data ) ;
2018-06-24 13:49:29 +00:00
Object . keys ( tmplInits ) . forEach ( ( key ) => {
let phrases = [ ] ;
let tmplInit = tmplInits [ key ] ;
for ( let phraseName of tmplInit ) {
phrases . push ( data [ phraseName ] ) ;
}
console . log ( "Adding phrases" ) ;
console . log ( "key:" , key ) ;
console . log ( "phrases:" , phrases ) ;
tmplPhrases [ key ] = phrases ;
2018-06-30 03:40:50 +00:00
} ) ;
let prefixes = { } ;
Object . keys ( data ) . forEach ( ( key ) => {
let prefix = key . split ( "." ) [ 0 ] ;
if ( prefixes [ prefix ] === undefined ) {
prefixes [ prefix ] = { } ;
}
prefixes [ prefix ] [ key ] = data [ key ] ;
} ) ;
Object . keys ( prefixes ) . forEach ( ( prefix ) => {
console . log ( "adding phrase prefix '" + prefix + "' to box" ) ;
phraseBox [ prefix ] = prefixes [ prefix ] ;
} ) ;
2018-06-24 13:49:29 +00:00
} ) ;
}
2017-11-06 16:24:45 +00:00
$ ( document ) . ready ( function ( ) {
2018-02-22 02:27:17 +00:00
runHook ( "start_init" ) ;
2018-06-24 13:49:29 +00:00
if ( loggedIn ) {
let toLoad = 1 ;
loadScript ( "template_topics_topic.js" , ( ) => {
console . log ( "Loaded template_topics_topic.js" ) ;
toLoad -- ;
if ( toLoad === 0 ) fetchPhrases ( ) ;
} ) ;
}
// We can only get away with this because template_alert has no phrases, otherwise it too would have to be part of the "dance", I miss Go concurrency :(
loadScript ( "template_alert.js" , ( ) => {
2018-05-14 08:56:56 +00:00
console . log ( "Loaded template_alert.js" ) ;
alertsInitted = true ;
var alertMenuList = document . getElementsByClassName ( "menu_alerts" ) ;
for ( var i = 0 ; i < alertMenuList . length ; i ++ ) {
loadAlerts ( alertMenuList [ i ] ) ;
}
2018-06-24 13:49:29 +00:00
} ) ;
2017-11-06 16:24:45 +00:00
if ( window [ "WebSocket" ] ) runWebSockets ( ) ;
2017-06-05 11:57:27 +00:00
else conn = false ;
2017-05-29 14:52:37 +00:00
2018-06-30 03:40:50 +00:00
$ ( ".more_topics" ) . click ( ( event ) => {
event . preventDefault ( ) ;
let moreTopicBlocks = document . getElementsByClassName ( "more_topic_block_active" ) ;
for ( let i = 0 ; i < moreTopicBlocks . length ; i ++ ) {
let moreTopicBlock = moreTopicBlocks [ i ] ;
moreTopicBlock . classList . remove ( "more_topic_block_active" ) ;
moreTopicBlock . classList . add ( "more_topic_block_initial" ) ;
}
$ ( ".ajax_topic_dupe" ) . fadeOut ( "slow" , function ( ) {
$ ( this ) . remove ( ) ;
} ) ;
$ ( ".hide_ajax_topic" ) . removeClass ( "hide_ajax_topic" ) ; // TODO: Do Fade
moreTopicCount = 0 ;
} )
2018-03-31 05:25:27 +00:00
$ ( ".add_like" ) . click ( function ( event ) {
event . preventDefault ( ) ;
let likeButton = this ;
let target = this . closest ( "a" ) . getAttribute ( "href" ) ;
console . log ( "target: " , target ) ;
likeButton . classList . remove ( "add_like" ) ;
likeButton . classList . add ( "remove_like" ) ;
let controls = likeButton . closest ( ".controls" ) ;
let hadLikes = controls . classList . contains ( "has_likes" ) ;
if ( ! hadLikes ) controls . classList . add ( "has_likes" ) ;
let likeCountNode = controls . getElementsByClassName ( "like_count" ) [ 0 ] ;
console . log ( "likeCountNode" , likeCountNode ) ;
likeCountNode . innerHTML = parseInt ( likeCountNode . innerHTML ) + 1 ;
$ . ajax ( {
url : target ,
type : "POST" ,
dataType : "json" ,
data : { isJs : 1 } ,
error : ajaxError ,
success : function ( data , status , xhr ) {
if ( "success" in data ) {
if ( data [ "success" ] == "1" ) {
return ;
}
}
// addNotice("Failed to add a like: {err}")
likeButton . classList . add ( "add_like" ) ;
likeButton . classList . remove ( "remove_like" ) ;
if ( ! hadLikes ) controls . classList . remove ( "has_likes" ) ;
likeCountNode . innerHTML = parseInt ( likeCountNode . innerHTML ) - 1 ;
console . log ( "data" , data ) ;
console . log ( "status" , status ) ;
console . log ( "xhr" , xhr ) ;
}
} ) ;
} ) ;
$ ( ".open_edit" ) . click ( ( event ) => {
2016-12-02 07:38:54 +00:00
event . preventDefault ( ) ;
$ ( ".hide_on_edit" ) . hide ( ) ;
$ ( ".show_on_edit" ) . show ( ) ;
} ) ;
2017-05-29 14:52:37 +00:00
2017-01-31 05:13:38 +00:00
$ ( ".topic_item .submit_edit" ) . click ( function ( event ) {
2016-12-02 07:38:54 +00:00
event . preventDefault ( ) ;
2018-03-31 05:25:27 +00:00
let topicNameInput = $ ( ".topic_name_input" ) . val ( ) ;
$ ( ".topic_name" ) . html ( topicNameInput ) ;
$ ( ".topic_name" ) . attr ( topicNameInput ) ;
let topicContentInput = $ ( '.topic_content_input' ) . val ( ) ;
$ ( ".topic_content" ) . html ( topicContentInput . replace ( /(\n)+/g , "<br />" ) ) ;
let topicStatusInput = $ ( '.topic_status_input' ) . val ( ) ;
$ ( ".topic_status_e:not(.open_edit)" ) . html ( topicStatusInput ) ;
2017-05-29 14:52:37 +00:00
2016-12-02 07:38:54 +00:00
$ ( ".hide_on_edit" ) . show ( ) ;
$ ( ".show_on_edit" ) . hide ( ) ;
2017-05-29 14:52:37 +00:00
2017-10-16 07:32:58 +00:00
let formAction = this . form . getAttribute ( "action" ) ;
2017-10-21 00:27:47 +00:00
//console.log("New Topic Name: ", topicNameInput);
//console.log("New Topic Status: ", topicStatusInput);
//console.log("New Topic Content: ", topicContentInput);
//console.log("Form Action: ", formAction);
2016-12-02 07:38:54 +00:00
$ . ajax ( {
2017-10-16 07:32:58 +00:00
url : formAction ,
2017-05-29 14:52:37 +00:00
type : "POST" ,
dataType : "json" ,
2017-10-30 09:57:08 +00:00
error : ajaxError ,
2016-12-02 07:38:54 +00:00
data : {
2017-10-16 07:32:58 +00:00
topic _name : topicNameInput ,
topic _status : topicStatusInput ,
topic _content : topicContentInput ,
2016-12-02 07:38:54 +00:00
topic _js : 1
2017-05-29 14:52:37 +00:00
}
2016-12-02 07:38:54 +00:00
} ) ;
} ) ;
2017-05-29 14:52:37 +00:00
2018-01-28 14:30:24 +00:00
$ ( ".delete_item" ) . click ( function ( event ) {
2017-10-30 09:57:08 +00:00
postLink ( event ) ;
2017-10-16 07:32:58 +00:00
$ ( this ) . closest ( '.deletable_block' ) . remove ( ) ;
2016-12-02 07:38:54 +00:00
} ) ;
2017-05-29 14:52:37 +00:00
2018-01-08 08:53:51 +00:00
$ ( ".edit_item" ) . click ( function ( event ) {
2016-12-02 07:38:54 +00:00
event . preventDefault ( ) ;
2017-10-16 07:32:58 +00:00
let blockParent = $ ( this ) . closest ( '.editable_parent' ) ;
let block = blockParent . find ( '.editable_block' ) . eq ( 0 ) ;
2016-12-03 04:50:35 +00:00
block . html ( "<textarea style='width: 99%;' name='edit_item'>" + block . html ( ) + "</textarea><br /><a href='" + $ ( this ) . closest ( 'a' ) . attr ( "href" ) + "'><button class='submit_edit' type='submit'>Update</button></a>" ) ;
2017-05-29 14:52:37 +00:00
2018-01-08 08:53:51 +00:00
$ ( ".submit_edit" ) . click ( function ( event ) {
2016-12-02 07:38:54 +00:00
event . preventDefault ( ) ;
2017-10-16 07:32:58 +00:00
let blockParent = $ ( this ) . closest ( '.editable_parent' ) ;
let block = blockParent . find ( '.editable_block' ) . eq ( 0 ) ;
let newContent = block . find ( 'textarea' ) . eq ( 0 ) . val ( ) ;
2016-12-02 07:38:54 +00:00
block . html ( newContent ) ;
2017-05-29 14:52:37 +00:00
2017-10-16 07:32:58 +00:00
var formAction = $ ( this ) . closest ( 'a' ) . attr ( "href" ) ;
2017-10-21 00:27:47 +00:00
//console.log("Form Action:",formAction);
2017-10-30 09:57:08 +00:00
$ . ajax ( { url : formAction , type : "POST" , error : ajaxError , dataType : "json" , data : { isJs : "1" , edit _item : newContent }
2016-12-02 07:38:54 +00:00
} ) ;
} ) ;
} ) ;
2017-05-29 14:52:37 +00:00
2018-03-31 05:25:27 +00:00
$ ( ".edit_field" ) . click ( function ( event ) {
2016-12-06 10:26:48 +00:00
event . preventDefault ( ) ;
2017-10-16 07:32:58 +00:00
let blockParent = $ ( this ) . closest ( '.editable_parent' ) ;
let block = blockParent . find ( '.editable_block' ) . eq ( 0 ) ;
2016-12-06 10:26:48 +00:00
block . html ( "<input name='edit_field' value='" + block . text ( ) + "' type='text'/><a href='" + $ ( this ) . closest ( 'a' ) . attr ( "href" ) + "'><button class='submit_edit' type='submit'>Update</button></a>" ) ;
2017-05-29 14:52:37 +00:00
2017-10-21 00:27:47 +00:00
$ ( ".submit_edit" ) . click ( function ( event ) {
2016-12-06 10:26:48 +00:00
event . preventDefault ( ) ;
2017-10-16 07:32:58 +00:00
let blockParent = $ ( this ) . closest ( '.editable_parent' ) ;
let block = blockParent . find ( '.editable_block' ) . eq ( 0 ) ;
let newContent = block . find ( 'input' ) . eq ( 0 ) . val ( ) ;
2016-12-06 10:26:48 +00:00
block . html ( newContent ) ;
2017-05-29 14:52:37 +00:00
2017-10-16 07:32:58 +00:00
let formAction = $ ( this ) . closest ( 'a' ) . attr ( "href" ) ;
2017-10-21 00:27:47 +00:00
//console.log("Form Action:", formAction);
2016-12-06 10:26:48 +00:00
$ . ajax ( {
2017-10-16 07:32:58 +00:00
url : formAction + "?session=" + session ,
2016-12-06 10:26:48 +00:00
type : "POST" ,
dataType : "json" ,
2017-10-30 09:57:08 +00:00
error : ajaxError ,
2017-10-21 00:27:47 +00:00
data : { isJs : "1" , edit _item : newContent }
2016-12-06 10:26:48 +00:00
} ) ;
} ) ;
} ) ;
2017-05-29 14:52:37 +00:00
2017-01-31 05:13:38 +00:00
$ ( ".edit_fields" ) . click ( function ( event )
{
event . preventDefault ( ) ;
2017-08-27 09:33:45 +00:00
if ( $ ( this ) . find ( "input" ) . length !== 0 ) return ;
2017-05-29 14:52:37 +00:00
//console.log("clicked .edit_fields");
2017-10-30 09:57:08 +00:00
var blockParent = $ ( this ) . closest ( '.editable_parent' ) ;
//console.log(blockParent);
blockParent . find ( '.hide_on_edit' ) . hide ( ) ;
blockParent . find ( '.show_on_edit' ) . show ( ) ;
blockParent . find ( '.editable_block' ) . show ( ) ;
blockParent . find ( '.editable_block' ) . each ( function ( ) {
var fieldName = this . getAttribute ( "data-field" ) ;
var fieldType = this . getAttribute ( "data-type" ) ;
2018-01-28 14:30:24 +00:00
if ( fieldType == "list" ) {
2017-10-30 09:57:08 +00:00
var fieldValue = this . getAttribute ( "data-value" ) ;
2018-02-22 02:27:17 +00:00
if ( fieldName in formVars ) var it = formVars [ fieldName ] ;
2017-01-31 05:13:38 +00:00
else var it = [ 'No' , 'Yes' ] ;
var itLen = it . length ;
var out = "" ;
2017-10-30 09:57:08 +00:00
//console.log("Field Name:",fieldName);
//console.log("Field Type:",fieldType);
//console.log("Field Value:",fieldValue);
2017-06-05 11:57:27 +00:00
for ( var i = 0 ; i < itLen ; i ++ ) {
2017-09-25 00:48:35 +00:00
var sel = "" ;
2017-10-30 09:57:08 +00:00
if ( fieldValue == i || fieldValue == it [ i ] ) {
2017-05-29 14:52:37 +00:00
sel = "selected " ;
2017-10-30 09:57:08 +00:00
this . classList . remove ( fieldName + '_' + it [ i ] ) ;
2017-05-29 14:52:37 +00:00
this . innerHTML = "" ;
2017-09-25 00:48:35 +00:00
}
2017-01-31 05:13:38 +00:00
out += "<option " + sel + "value='" + i + "'>" + it [ i ] + "</option>" ;
}
2017-10-30 09:57:08 +00:00
this . innerHTML = "<select data-field='" + fieldName + "' name='" + fieldName + "'>" + out + "</select>" ;
2017-01-31 05:13:38 +00:00
}
2017-10-30 09:57:08 +00:00
else if ( fieldType == "hidden" ) { }
else this . innerHTML = "<input name='" + fieldName + "' value='" + this . textContent + "' type='text'/>" ;
2017-01-31 05:13:38 +00:00
} ) ;
2017-05-29 14:52:37 +00:00
// Remove any handlers already attached to the submitter
$ ( ".submit_edit" ) . unbind ( "click" ) ;
2017-01-31 05:13:38 +00:00
$ ( ".submit_edit" ) . click ( function ( event )
{
event . preventDefault ( ) ;
2017-05-29 14:52:37 +00:00
//console.log("running .submit_edit event");
2017-10-30 09:57:08 +00:00
var outData = { isJs : "1" }
var blockParent = $ ( this ) . closest ( '.editable_parent' ) ;
blockParent . find ( '.editable_block' ) . each ( function ( ) {
var fieldName = this . getAttribute ( "data-field" ) ;
var fieldType = this . getAttribute ( "data-type" ) ;
if ( fieldType == "list" ) {
2017-05-29 14:52:37 +00:00
var newContent = $ ( this ) . find ( 'select :selected' ) . text ( ) ;
2017-10-30 09:57:08 +00:00
this . classList . add ( fieldName + '_' + newContent ) ;
2017-05-29 14:52:37 +00:00
this . innerHTML = "" ;
2017-10-30 09:57:08 +00:00
} else if ( fieldType == "hidden" ) {
2017-06-05 11:57:27 +00:00
var newContent = $ ( this ) . val ( ) ;
2017-05-29 14:52:37 +00:00
} else {
var newContent = $ ( this ) . find ( 'input' ) . eq ( 0 ) . val ( ) ;
this . innerHTML = newContent ;
}
this . setAttribute ( "data-value" , newContent ) ;
2017-10-30 09:57:08 +00:00
outData [ fieldName ] = newContent ;
2017-01-31 05:13:38 +00:00
} ) ;
2017-05-29 14:52:37 +00:00
2017-10-30 09:57:08 +00:00
var formAction = $ ( this ) . closest ( 'a' ) . attr ( "href" ) ;
//console.log("Form Action:", formAction);
//console.log(outData);
$ . ajax ( { url : formAction + "?session=" + session , type : "POST" , dataType : "json" , data : outData , error : ajaxError } ) ;
blockParent . find ( '.hide_on_edit' ) . show ( ) ;
blockParent . find ( '.show_on_edit' ) . hide ( ) ;
2017-01-31 05:13:38 +00:00
} ) ;
} ) ;
2017-05-29 14:52:37 +00:00
2017-08-18 12:16:56 +00:00
// This one's for Tempra Conflux
2017-09-10 16:57:22 +00:00
// TODO: We might want to use pure JS here
2017-02-16 06:47:55 +00:00
$ ( ".ip_item" ) . each ( function ( ) {
var ip = this . textContent ;
2017-01-17 07:55:46 +00:00
if ( ip . length > 10 ) {
2017-02-16 06:47:55 +00:00
this . innerHTML = "Show IP" ;
2017-10-30 09:57:08 +00:00
this . onclick = function ( event ) {
2017-01-17 07:55:46 +00:00
event . preventDefault ( ) ;
2017-02-16 06:47:55 +00:00
this . textContent = ip ;
} ;
2017-01-17 07:55:46 +00:00
}
} ) ;
2017-05-29 14:52:37 +00:00
2018-03-31 05:25:27 +00:00
$ ( this ) . click ( ( ) => {
2017-03-03 16:28:49 +00:00
$ ( ".selectedAlert" ) . removeClass ( "selectedAlert" ) ;
2017-06-05 11:57:27 +00:00
$ ( "#back" ) . removeClass ( "alertActive" ) ;
} ) ;
$ ( ".alert_bell" ) . click ( function ( ) {
2017-10-30 09:57:08 +00:00
var menuAlerts = $ ( this ) . parent ( ) ;
if ( menuAlerts . hasClass ( "selectedAlert" ) ) {
2017-06-05 11:57:27 +00:00
event . stopPropagation ( ) ;
2017-10-30 09:57:08 +00:00
menuAlerts . removeClass ( "selectedAlert" ) ;
2017-06-05 11:57:27 +00:00
$ ( "#back" ) . removeClass ( "alertActive" ) ;
}
2017-03-03 16:28:49 +00:00
} ) ;
2017-05-29 14:52:37 +00:00
2017-03-01 11:36:50 +00:00
$ ( ".menu_alerts" ) . click ( function ( event ) {
2017-03-03 16:28:49 +00:00
event . stopPropagation ( ) ;
2017-03-01 11:36:50 +00:00
if ( $ ( this ) . hasClass ( "selectedAlert" ) ) return ;
2017-10-30 09:57:08 +00:00
if ( ! conn ) loadAlerts ( this ) ;
2017-06-05 11:57:27 +00:00
this . className += " selectedAlert" ;
document . getElementById ( "back" ) . className += " alertActive"
2017-03-01 11:36:50 +00:00
} ) ;
2017-05-29 14:52:37 +00:00
2018-03-31 05:25:27 +00:00
$ ( "input,textarea,select,option" ) . keyup ( event => event . stopPropagation ( ) )
2017-06-16 10:41:30 +00:00
2018-02-05 10:29:13 +00:00
$ ( ".create_topic_link" ) . click ( ( event ) => {
Added Quick Topic.
Added Attachments.
Added Attachment Media Embeds.
Renamed a load of *Store and *Cache methods to reduce the amount of unneccesary typing.
Added petabytes as a unit and cleaned up a few of the friendly units.
Refactored the username change logic to make it easier to maintain.
Refactored the avatar change logic to make it easier to maintain.
Shadow now uses CSS Variables for most of it's colours. We have plans to transpile this to support older browsers later on!
Snuck some CSS Variables into Tempra Conflux.
Added the GroupCache interface to MemoryGroupStore.
Added the Length method to MemoryGroupStore.
Added support for a site short name.
Added the UploadFiles permission.
Renamed more functions.
Fixed the background for the left gutter on the postbit for Tempra Simple and Shadow.
Added support for if statements operating on int8, int16, int32, int32, int64, uint, uint8, uint16, uint32, uint64, float32, and float64 for the template compiler.
Added support for if statements operating on slices and maps for the template compiler.
Fixed a security exploit in reply editing.
Fixed a bug in the URL detector in the parser where it couldn't find URLs with non-standard ports.
Fixed buttons having blue outlines on focus on Shadow.
Refactored the topic creation logic to make it easier to maintain.
Made a few responsive fixes, but there's still more to do in the following commits!
2017-10-05 10:20:28 +00:00
event . preventDefault ( ) ;
$ ( ".topic_create_form" ) . show ( ) ;
} ) ;
2018-02-05 10:29:13 +00:00
$ ( ".topic_create_form .close_form" ) . click ( ( event ) => {
Added Quick Topic.
Added Attachments.
Added Attachment Media Embeds.
Renamed a load of *Store and *Cache methods to reduce the amount of unneccesary typing.
Added petabytes as a unit and cleaned up a few of the friendly units.
Refactored the username change logic to make it easier to maintain.
Refactored the avatar change logic to make it easier to maintain.
Shadow now uses CSS Variables for most of it's colours. We have plans to transpile this to support older browsers later on!
Snuck some CSS Variables into Tempra Conflux.
Added the GroupCache interface to MemoryGroupStore.
Added the Length method to MemoryGroupStore.
Added support for a site short name.
Added the UploadFiles permission.
Renamed more functions.
Fixed the background for the left gutter on the postbit for Tempra Simple and Shadow.
Added support for if statements operating on int8, int16, int32, int32, int64, uint, uint8, uint16, uint32, uint64, float32, and float64 for the template compiler.
Added support for if statements operating on slices and maps for the template compiler.
Fixed a security exploit in reply editing.
Fixed a bug in the URL detector in the parser where it couldn't find URLs with non-standard ports.
Fixed buttons having blue outlines on focus on Shadow.
Refactored the topic creation logic to make it easier to maintain.
Made a few responsive fixes, but there's still more to do in the following commits!
2017-10-05 10:20:28 +00:00
event . preventDefault ( ) ;
$ ( ".topic_create_form" ) . hide ( ) ;
} ) ;
function uploadFileHandler ( ) {
var fileList = this . files ;
// Truncate the number of files to 5
let files = [ ] ;
for ( var i = 0 ; i < fileList . length && i < 5 ; i ++ )
files [ i ] = fileList [ i ] ;
// Iterate over the files
2018-06-17 07:28:18 +00:00
let totalSize = 0 ;
Added Quick Topic.
Added Attachments.
Added Attachment Media Embeds.
Renamed a load of *Store and *Cache methods to reduce the amount of unneccesary typing.
Added petabytes as a unit and cleaned up a few of the friendly units.
Refactored the username change logic to make it easier to maintain.
Refactored the avatar change logic to make it easier to maintain.
Shadow now uses CSS Variables for most of it's colours. We have plans to transpile this to support older browsers later on!
Snuck some CSS Variables into Tempra Conflux.
Added the GroupCache interface to MemoryGroupStore.
Added the Length method to MemoryGroupStore.
Added support for a site short name.
Added the UploadFiles permission.
Renamed more functions.
Fixed the background for the left gutter on the postbit for Tempra Simple and Shadow.
Added support for if statements operating on int8, int16, int32, int32, int64, uint, uint8, uint16, uint32, uint64, float32, and float64 for the template compiler.
Added support for if statements operating on slices and maps for the template compiler.
Fixed a security exploit in reply editing.
Fixed a bug in the URL detector in the parser where it couldn't find URLs with non-standard ports.
Fixed buttons having blue outlines on focus on Shadow.
Refactored the topic creation logic to make it easier to maintain.
Made a few responsive fixes, but there's still more to do in the following commits!
2017-10-05 10:20:28 +00:00
for ( let i = 0 ; i < files . length ; i ++ ) {
console . log ( "files[" + i + "]" , files [ i ] ) ;
2018-06-17 07:28:18 +00:00
totalSize += files [ i ] [ "size" ] ;
Added Quick Topic.
Added Attachments.
Added Attachment Media Embeds.
Renamed a load of *Store and *Cache methods to reduce the amount of unneccesary typing.
Added petabytes as a unit and cleaned up a few of the friendly units.
Refactored the username change logic to make it easier to maintain.
Refactored the avatar change logic to make it easier to maintain.
Shadow now uses CSS Variables for most of it's colours. We have plans to transpile this to support older browsers later on!
Snuck some CSS Variables into Tempra Conflux.
Added the GroupCache interface to MemoryGroupStore.
Added the Length method to MemoryGroupStore.
Added support for a site short name.
Added the UploadFiles permission.
Renamed more functions.
Fixed the background for the left gutter on the postbit for Tempra Simple and Shadow.
Added support for if statements operating on int8, int16, int32, int32, int64, uint, uint8, uint16, uint32, uint64, float32, and float64 for the template compiler.
Added support for if statements operating on slices and maps for the template compiler.
Fixed a security exploit in reply editing.
Fixed a bug in the URL detector in the parser where it couldn't find URLs with non-standard ports.
Fixed buttons having blue outlines on focus on Shadow.
Refactored the topic creation logic to make it easier to maintain.
Made a few responsive fixes, but there's still more to do in the following commits!
2017-10-05 10:20:28 +00:00
let reader = new FileReader ( ) ;
reader . onload = function ( e ) {
var fileDock = document . getElementById ( "upload_file_dock" ) ;
var fileItem = document . createElement ( "label" ) ;
console . log ( "fileItem" , fileItem ) ;
if ( ! files [ i ] [ "name" ] . indexOf ( '.' > - 1 ) ) {
// TODO: Surely, there's a prettier and more elegant way of doing this?
alert ( "This file doesn't have an extension" ) ;
return ;
}
var ext = files [ i ] [ "name" ] . split ( '.' ) . pop ( ) ;
fileItem . innerText = "." + ext ;
fileItem . className = "formbutton uploadItem" ;
fileItem . style . backgroundImage = "url(" + e . target . result + ")" ;
fileDock . appendChild ( fileItem ) ;
let reader = new FileReader ( ) ;
reader . onload = function ( e ) {
2018-06-30 03:40:50 +00:00
crypto . subtle . digest ( 'SHA-256' , e . target . result )
. then ( function ( hash ) {
const hashArray = Array . from ( new Uint8Array ( hash ) )
return hashArray . map ( b => ( '00' + b . toString ( 16 ) ) . slice ( - 2 ) ) . join ( '' )
} ) . then ( function ( hash ) {
console . log ( "hash" , hash ) ;
let content = document . getElementById ( "input_content" )
console . log ( "content.value" , content . value ) ;
let attachItem ;
if ( content . value == "" ) attachItem = "//" + siteURL + "/attachs/" + hash + "." + ext ;
else attachItem = "\r\n//" + siteURL + "/attachs/" + hash + "." + ext ;
content . value = content . value + attachItem ;
console . log ( "content.value" , content . value ) ;
2017-10-30 09:57:08 +00:00
2018-06-30 03:40:50 +00:00
// For custom / third party text editors
attachItemCallback ( attachItem ) ;
} ) ;
Added Quick Topic.
Added Attachments.
Added Attachment Media Embeds.
Renamed a load of *Store and *Cache methods to reduce the amount of unneccesary typing.
Added petabytes as a unit and cleaned up a few of the friendly units.
Refactored the username change logic to make it easier to maintain.
Refactored the avatar change logic to make it easier to maintain.
Shadow now uses CSS Variables for most of it's colours. We have plans to transpile this to support older browsers later on!
Snuck some CSS Variables into Tempra Conflux.
Added the GroupCache interface to MemoryGroupStore.
Added the Length method to MemoryGroupStore.
Added support for a site short name.
Added the UploadFiles permission.
Renamed more functions.
Fixed the background for the left gutter on the postbit for Tempra Simple and Shadow.
Added support for if statements operating on int8, int16, int32, int32, int64, uint, uint8, uint16, uint32, uint64, float32, and float64 for the template compiler.
Added support for if statements operating on slices and maps for the template compiler.
Fixed a security exploit in reply editing.
Fixed a bug in the URL detector in the parser where it couldn't find URLs with non-standard ports.
Fixed buttons having blue outlines on focus on Shadow.
Refactored the topic creation logic to make it easier to maintain.
Made a few responsive fixes, but there's still more to do in the following commits!
2017-10-05 10:20:28 +00:00
}
reader . readAsArrayBuffer ( files [ i ] ) ;
}
reader . readAsDataURL ( files [ i ] ) ;
}
2018-06-17 07:28:18 +00:00
if ( totalSize > maxRequestSize ) {
2018-06-30 03:40:50 +00:00
// TODO: Use a notice instead
2018-06-17 07:28:18 +00:00
alert ( "You can't upload this much data at once, max: " + maxRequestSize ) ;
}
Added Quick Topic.
Added Attachments.
Added Attachment Media Embeds.
Renamed a load of *Store and *Cache methods to reduce the amount of unneccesary typing.
Added petabytes as a unit and cleaned up a few of the friendly units.
Refactored the username change logic to make it easier to maintain.
Refactored the avatar change logic to make it easier to maintain.
Shadow now uses CSS Variables for most of it's colours. We have plans to transpile this to support older browsers later on!
Snuck some CSS Variables into Tempra Conflux.
Added the GroupCache interface to MemoryGroupStore.
Added the Length method to MemoryGroupStore.
Added support for a site short name.
Added the UploadFiles permission.
Renamed more functions.
Fixed the background for the left gutter on the postbit for Tempra Simple and Shadow.
Added support for if statements operating on int8, int16, int32, int32, int64, uint, uint8, uint16, uint32, uint64, float32, and float64 for the template compiler.
Added support for if statements operating on slices and maps for the template compiler.
Fixed a security exploit in reply editing.
Fixed a bug in the URL detector in the parser where it couldn't find URLs with non-standard ports.
Fixed buttons having blue outlines on focus on Shadow.
Refactored the topic creation logic to make it easier to maintain.
Made a few responsive fixes, but there's still more to do in the following commits!
2017-10-05 10:20:28 +00:00
}
2017-10-12 03:24:14 +00:00
var uploadFiles = document . getElementById ( "upload_files" ) ;
Added Quick Topic.
Added Attachments.
Added Attachment Media Embeds.
Renamed a load of *Store and *Cache methods to reduce the amount of unneccesary typing.
Added petabytes as a unit and cleaned up a few of the friendly units.
Refactored the username change logic to make it easier to maintain.
Refactored the avatar change logic to make it easier to maintain.
Shadow now uses CSS Variables for most of it's colours. We have plans to transpile this to support older browsers later on!
Snuck some CSS Variables into Tempra Conflux.
Added the GroupCache interface to MemoryGroupStore.
Added the Length method to MemoryGroupStore.
Added support for a site short name.
Added the UploadFiles permission.
Renamed more functions.
Fixed the background for the left gutter on the postbit for Tempra Simple and Shadow.
Added support for if statements operating on int8, int16, int32, int32, int64, uint, uint8, uint16, uint32, uint64, float32, and float64 for the template compiler.
Added support for if statements operating on slices and maps for the template compiler.
Fixed a security exploit in reply editing.
Fixed a bug in the URL detector in the parser where it couldn't find URLs with non-standard ports.
Fixed buttons having blue outlines on focus on Shadow.
Refactored the topic creation logic to make it easier to maintain.
Made a few responsive fixes, but there's still more to do in the following commits!
2017-10-05 10:20:28 +00:00
if ( uploadFiles != null ) {
uploadFiles . addEventListener ( "change" , uploadFileHandler , false ) ;
}
2017-10-30 09:57:08 +00:00
2018-05-27 09:36:35 +00:00
$ ( ".moderate_link" ) . click ( ( event ) => {
2017-10-30 09:57:08 +00:00
event . preventDefault ( ) ;
$ ( ".pre_opt" ) . removeClass ( "auto_hide" ) ;
2018-05-27 09:36:35 +00:00
$ ( ".moderate_link" ) . addClass ( "moderate_open" ) ;
2017-10-30 09:57:08 +00:00
$ ( ".topic_row" ) . each ( function ( ) {
$ ( this ) . click ( function ( ) {
selectedTopics . push ( parseInt ( $ ( this ) . attr ( "data-tid" ) , 10 ) ) ;
if ( selectedTopics . length == 1 ) {
$ ( ".mod_floater_head span" ) . html ( "What do you want to do with this topic?" ) ;
} else {
$ ( ".mod_floater_head span" ) . html ( "What do you want to do with these " + selectedTopics . length + " topics?" ) ;
}
$ ( this ) . addClass ( "topic_selected" ) ;
$ ( ".mod_floater" ) . removeClass ( "auto_hide" ) ;
} ) ;
} ) ;
2018-01-14 12:03:20 +00:00
2018-01-15 08:24:18 +00:00
let bulkActionSender = function ( action , selectedTopics , fragBit ) {
let url = "/topic/" + action + "/submit/" + fragBit + "?session=" + session ;
2018-01-14 12:03:20 +00:00
$ . ajax ( {
url : url ,
type : "POST" ,
data : JSON . stringify ( selectedTopics ) ,
contentType : "application/json" ,
error : ajaxError ,
2018-02-05 10:29:13 +00:00
success : ( ) => {
2018-01-14 12:03:20 +00:00
window . location . reload ( ) ;
}
} ) ;
} ;
2017-10-30 09:57:08 +00:00
$ ( ".mod_floater_submit" ) . click ( function ( event ) {
event . preventDefault ( ) ;
let selectNode = this . form . querySelector ( ".mod_floater_options" ) ;
let optionNode = selectNode . options [ selectNode . selectedIndex ] ;
let action = optionNode . getAttribute ( "val" ) ;
2018-02-05 10:29:13 +00:00
//console.log("action", action);
2018-01-11 08:03:17 +00:00
// Handle these specially
switch ( action ) {
case "move" :
console . log ( "move action" ) ;
2018-01-14 12:03:20 +00:00
let modTopicMover = $ ( "#mod_topic_mover" ) ;
2018-01-11 08:03:17 +00:00
$ ( "#mod_topic_mover" ) . removeClass ( "auto_hide" ) ;
2018-01-14 12:03:20 +00:00
$ ( "#mod_topic_mover .pane_row" ) . click ( function ( ) {
modTopicMover . find ( ".pane_row" ) . removeClass ( "pane_selected" ) ;
let fid = this . getAttribute ( "data-fid" ) ;
if ( fid == null ) {
return ;
}
this . classList . add ( "pane_selected" ) ;
console . log ( "fid: " + fid ) ;
2018-01-15 08:24:18 +00:00
forumToMoveTo = fid ;
$ ( "#mover_submit" ) . click ( function ( event ) {
event . preventDefault ( ) ;
bulkActionSender ( "move" , selectedTopics , forumToMoveTo ) ;
} ) ;
2018-01-14 12:03:20 +00:00
} ) ;
2018-01-11 08:03:17 +00:00
return ;
}
2017-10-30 09:57:08 +00:00
2018-01-15 08:24:18 +00:00
bulkActionSender ( action , selectedTopics , "" ) ;
2017-10-30 09:57:08 +00:00
} ) ;
} ) ;
Added Quick Topic.
Added Attachments.
Added Attachment Media Embeds.
Renamed a load of *Store and *Cache methods to reduce the amount of unneccesary typing.
Added petabytes as a unit and cleaned up a few of the friendly units.
Refactored the username change logic to make it easier to maintain.
Refactored the avatar change logic to make it easier to maintain.
Shadow now uses CSS Variables for most of it's colours. We have plans to transpile this to support older browsers later on!
Snuck some CSS Variables into Tempra Conflux.
Added the GroupCache interface to MemoryGroupStore.
Added the Length method to MemoryGroupStore.
Added support for a site short name.
Added the UploadFiles permission.
Renamed more functions.
Fixed the background for the left gutter on the postbit for Tempra Simple and Shadow.
Added support for if statements operating on int8, int16, int32, int32, int64, uint, uint8, uint16, uint32, uint64, float32, and float64 for the template compiler.
Added support for if statements operating on slices and maps for the template compiler.
Fixed a security exploit in reply editing.
Fixed a bug in the URL detector in the parser where it couldn't find URLs with non-standard ports.
Fixed buttons having blue outlines on focus on Shadow.
Refactored the topic creation logic to make it easier to maintain.
Made a few responsive fixes, but there's still more to do in the following commits!
2017-10-05 10:20:28 +00:00
2017-09-10 16:57:22 +00:00
$ ( "#themeSelectorSelect" ) . change ( function ( ) {
console . log ( "Changing the theme to " + this . options [ this . selectedIndex ] . getAttribute ( "val" ) ) ;
$ . ajax ( {
url : this . form . getAttribute ( "action" ) + "?session=" + session ,
type : "POST" ,
dataType : "json" ,
data : { "newTheme" : this . options [ this . selectedIndex ] . getAttribute ( "val" ) , isJs : "1" } ,
2017-10-30 09:57:08 +00:00
error : ajaxError ,
2017-09-10 16:57:22 +00:00
success : function ( data , status , xhr ) {
console . log ( "Theme successfully switched" ) ;
2017-10-30 09:57:08 +00:00
console . log ( "data" , data ) ;
console . log ( "status" , status ) ;
console . log ( "xhr" , xhr ) ;
2017-09-10 16:57:22 +00:00
window . location . reload ( ) ;
}
} ) ;
} ) ;
2018-01-08 08:53:51 +00:00
// The time range selector for the time graphs in the Control Panel
$ ( ".timeRangeSelector" ) . change ( function ( ) {
console . log ( "Changed the time range to " + this . options [ this . selectedIndex ] . getAttribute ( "val" ) ) ;
window . location = this . form . getAttribute ( "action" ) + "?timeRange=" + this . options [ this . selectedIndex ] . getAttribute ( "val" ) ; // Do a redirect as a form submission refuses to work properly
} ) ;
2018-01-10 03:32:48 +00:00
$ ( ".unix_to_24_hour_time" ) . each ( function ( ) {
let unixTime = this . innerText ;
let date = new Date ( unixTime * 1000 ) ;
console . log ( "date: " , date ) ;
let minutes = "0" + date . getMinutes ( ) ;
let formattedTime = date . getHours ( ) + ":" + minutes . substr ( - 2 ) ;
console . log ( "formattedTime:" , formattedTime ) ;
this . innerText = formattedTime ;
} ) ;
2017-06-16 10:41:30 +00:00
this . onkeyup = function ( event ) {
2017-02-16 06:47:55 +00:00
if ( event . which == 37 ) this . querySelectorAll ( "#prevFloat a" ) [ 0 ] . click ( ) ;
if ( event . which == 39 ) this . querySelectorAll ( "#nextFloat a" ) [ 0 ] . click ( ) ;
} ;
2018-01-08 08:53:51 +00:00
2018-01-26 05:53:34 +00:00
function addPollInput ( ) {
console . log ( "clicked on pollinputinput" ) ;
let dataPollInput = $ ( this ) . parent ( ) . attr ( "data-pollinput" ) ;
console . log ( "dataPollInput: " , dataPollInput ) ;
if ( dataPollInput == undefined ) return ;
if ( dataPollInput != ( pollInputIndex - 1 ) ) return ;
2018-02-03 05:47:14 +00:00
$ ( ".poll_content_row .formitem" ) . append ( "<div class='pollinput' data-pollinput='" + pollInputIndex + "'><input type='checkbox' disabled /><label class='pollinputlabel'></label><input form='quick_post_form' name='pollinputitem[" + pollInputIndex + "]' class='pollinputinput' type='text' placeholder='Add new poll option' /></div>" ) ;
2018-01-26 05:53:34 +00:00
pollInputIndex ++ ;
console . log ( "new pollInputIndex: " , pollInputIndex ) ;
$ ( ".pollinputinput" ) . off ( "click" ) ;
$ ( ".pollinputinput" ) . click ( addPollInput ) ;
}
var pollInputIndex = 1 ;
2018-02-05 10:29:13 +00:00
$ ( "#add_poll_button" ) . click ( ( event ) => {
2018-01-25 04:57:33 +00:00
event . preventDefault ( ) ;
$ ( ".poll_content_row" ) . removeClass ( "auto_hide" ) ;
2018-01-26 05:53:34 +00:00
$ ( "#has_poll_input" ) . val ( "1" ) ;
$ ( ".pollinputinput" ) . click ( addPollInput ) ;
2018-01-25 04:57:33 +00:00
} ) ;
2018-01-28 14:30:24 +00:00
2018-02-03 05:47:14 +00:00
//id="poll_results_{{.Poll.ID}}" class="poll_results auto_hide"
2018-01-28 14:30:24 +00:00
$ ( ".poll_results_button" ) . click ( function ( ) {
let pollID = $ ( this ) . attr ( "data-poll-id" ) ;
$ ( "#poll_results_" + pollID + " .user_content" ) . html ( "<div id='poll_results_chart_" + pollID + "'></div>" ) ;
2018-02-03 05:47:14 +00:00
$ ( "#poll_results_" + pollID ) . removeClass ( "auto_hide" ) ;
2018-01-28 14:30:24 +00:00
fetch ( "/poll/results/" + pollID , {
credentials : 'same-origin'
} ) . then ( ( response ) => response . text ( ) ) . catch ( ( error ) => console . error ( "Error:" , error ) ) . then ( ( rawData ) => {
// TODO: Make sure the received data is actually a list of integers
let data = JSON . parse ( rawData ) ;
console . log ( "rawData: " , rawData ) ;
console . log ( "series: " , data ) ;
Chartist . Pie ( '#poll_results_chart_' + pollID , {
series : data ,
} , {
2018-02-03 05:47:14 +00:00
height : '120px' ,
2018-01-28 14:30:24 +00:00
} ) ;
} )
} ) ;
2018-02-22 02:27:17 +00:00
runHook ( "end_init" ) ;
2017-05-29 14:52:37 +00:00
} ) ;