2012-04-22 12:45:45 +02:00
/ * *
2016-07-11 11:58:15 +02:00
* PrivateBin
2012-04-23 16:30:02 +02:00
*
2012-04-30 22:58:08 +02:00
* a zero - knowledge paste bin
*
2017-01-14 15:29:12 +01:00
* @ see { @ link https : //github.com/PrivateBin/PrivateBin}
* @ copyright 2012 Sébastien SAUVAGE ( { @ link http : //sebsauvage.net})
* @ license { @ link https : //www.opensource.org/licenses/zlib-license.php The zlib/libpng License}
2018-08-11 19:29:58 +02:00
* @ version 1.2 . 1
2017-01-14 15:29:12 +01:00
* @ name PrivateBin
* @ namespace
2012-04-22 12:45:45 +02:00
* /
2012-04-21 21:59:45 +02:00
2016-07-11 15:47:42 +02:00
/** global: Base64 */
2017-11-22 07:03:29 +01:00
/** global: DOMPurify */
2016-07-11 15:47:42 +02:00
/** global: FileReader */
/** global: RawDeflate */
/** global: history */
/** global: navigator */
/** global: prettyPrint */
/** global: prettyPrintOne */
/** global: showdown */
2017-11-26 15:59:12 +01:00
/** global: kjua */
2015-09-06 13:07:46 +02:00
2017-02-14 22:21:55 +01:00
// main application start, called when DOM is fully loaded
jQuery ( document ) . ready ( function ( ) {
2018-02-21 22:51:31 +01:00
'use strict' ;
2017-02-14 22:21:55 +01:00
// run main controller
$ . PrivateBin . Controller . init ( ) ;
} ) ;
2018-10-20 19:53:21 +02:00
jQuery . PrivateBin = ( function ( $ , RawDeflate ) {
2017-02-12 18:08:08 +01:00
'use strict' ;
2018-12-27 21:32:13 +01:00
/ * *
* zlib library interface
*
* @ private
* /
let z ;
2015-09-05 17:12:11 +02:00
/ * *
2017-02-14 22:21:55 +01:00
* static Helper methods
2017-01-14 15:29:12 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name Helper
2017-01-14 15:29:12 +01:00
* @ class
2015-09-05 17:12:11 +02:00
* /
2018-12-29 18:40:59 +01:00
const Helper = ( function ( ) {
const me = { } ;
2017-02-08 20:12:22 +01:00
2017-04-11 16:34:13 +02:00
/ * *
2017-04-13 10:46:09 +02:00
* blacklist of UserAgents ( parts ) known to belong to a bot
2017-04-11 16:34:13 +02:00
*
* @ private
* @ enum { Object }
* @ readonly
* /
2018-12-29 18:40:59 +01:00
const BadBotUA = [
2017-04-11 16:34:13 +02:00
'Bot' ,
'bot'
] ;
2017-02-08 20:12:22 +01:00
/ * *
* cache for script location
*
2017-03-13 20:24:18 +01:00
* @ name Helper . baseUri
2017-02-08 20:12:22 +01:00
* @ private
* @ enum { string | null }
* /
2018-12-29 18:40:59 +01:00
let baseUri = null ;
2017-02-08 20:12:22 +01:00
2015-09-05 17:12:11 +02:00
/ * *
2017-01-14 15:29:12 +01:00
* converts a duration ( in seconds ) into human friendly approximation
2015-09-05 17:12:11 +02:00
*
2017-02-14 22:21:55 +01:00
* @ name Helper . secondsToHuman
2017-01-14 15:29:12 +01:00
* @ function
* @ param { number } seconds
* @ return { Array }
2015-09-05 17:12:11 +02:00
* /
2017-02-08 20:12:22 +01:00
me . secondsToHuman = function ( seconds )
2015-09-05 17:12:11 +02:00
{
2018-12-29 18:40:59 +01:00
let v ;
2015-09-05 17:12:11 +02:00
if ( seconds < 60 )
{
2016-07-11 15:47:42 +02:00
v = Math . floor ( seconds ) ;
2015-09-06 15:54:43 +02:00
return [ v , 'second' ] ;
2015-09-05 17:12:11 +02:00
}
if ( seconds < 60 * 60 )
{
2016-07-11 15:47:42 +02:00
v = Math . floor ( seconds / 60 ) ;
2015-09-06 15:54:43 +02:00
return [ v , 'minute' ] ;
2015-09-05 17:12:11 +02:00
}
if ( seconds < 60 * 60 * 24 )
{
2016-07-11 15:47:42 +02:00
v = Math . floor ( seconds / ( 60 * 60 ) ) ;
2015-09-06 15:54:43 +02:00
return [ v , 'hour' ] ;
2015-09-05 17:12:11 +02:00
}
// If less than 2 months, display in days:
if ( seconds < 60 * 60 * 24 * 60 )
{
2016-07-11 15:47:42 +02:00
v = Math . floor ( seconds / ( 60 * 60 * 24 ) ) ;
2015-09-06 15:54:43 +02:00
return [ v , 'day' ] ;
2015-09-05 17:12:11 +02:00
}
2016-07-11 15:47:42 +02:00
v = Math . floor ( seconds / ( 60 * 60 * 24 * 30 ) ) ;
2015-09-06 15:54:43 +02:00
return [ v , 'month' ] ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 20:12:22 +01:00
2015-09-05 17:12:11 +02:00
/ * *
2017-01-14 15:29:12 +01:00
* text range selection
2015-09-05 17:12:11 +02:00
*
2017-01-14 15:29:12 +01:00
* @ see { @ link https : //stackoverflow.com/questions/985272/jquery-selecting-text-in-an-element-akin-to-highlighting-with-your-mouse}
2017-02-14 22:21:55 +01:00
* @ name Helper . selectText
2017-01-14 15:29:12 +01:00
* @ function
2017-02-08 20:12:22 +01:00
* @ param { HTMLElement } element
2015-09-05 17:12:11 +02:00
* /
2017-02-08 20:12:22 +01:00
me . selectText = function ( element )
2015-09-05 17:12:11 +02:00
{
2018-12-29 18:40:59 +01:00
let range , selection ;
2015-09-05 17:12:11 +02:00
// MS
2017-02-14 22:21:55 +01:00
if ( document . body . createTextRange ) {
2017-02-08 20:12:22 +01:00
range = document . body . createTextRange ( ) ;
range . moveToElementText ( element ) ;
2015-09-05 17:12:11 +02:00
range . select ( ) ;
2017-11-13 21:57:49 +01:00
} else if ( window . getSelection ) {
2015-09-05 17:12:11 +02:00
selection = window . getSelection ( ) ;
2017-02-08 20:12:22 +01:00
range = document . createRange ( ) ;
range . selectNodeContents ( element ) ;
2015-09-05 17:12:11 +02:00
selection . removeAllRanges ( ) ;
selection . addRange ( range ) ;
}
2018-01-06 09:26:10 +01:00
} ;
2015-09-12 17:33:16 +02:00
2015-09-05 17:12:11 +02:00
/ * *
2017-01-14 15:29:12 +01:00
* convert URLs to clickable links .
2015-09-05 17:12:11 +02:00
* URLs to handle :
2017-01-14 15:29:12 +01:00
* < pre >
2015-09-05 17:12:11 +02:00
* magnet : ? xt . 1 = urn : sha1 : YNCKHTQCWBTRNJIV4WNAE52SJUQCZO5C & xt . 2 = urn : sha1 : TXGCZQTH26NL6OUQAJJPFALHG2LTGBC7
2017-04-11 16:34:13 +02:00
* https : //example.com:8800/zero/?6f09182b8ea51997#WtLEUO5Epj9UHAV9JFs+6pUQZp13TuspAUjnF+iM+dM=
2017-01-14 15:29:12 +01:00
* http : //user:example.com@localhost:8800/zero/?6f09182b8ea51997#WtLEUO5Epj9UHAV9JFs+6pUQZp13TuspAUjnF+iM+dM=
* < / p r e >
2015-09-05 17:12:11 +02:00
*
2017-02-14 22:21:55 +01:00
* @ name Helper . urls2links
2017-01-14 15:29:12 +01:00
* @ function
2017-11-22 22:27:38 +01:00
* @ param { string } html
* @ return { string }
2015-09-05 17:12:11 +02:00
* /
2017-11-22 22:27:38 +01:00
me . urls2links = function ( html )
2015-09-05 17:12:11 +02:00
{
2017-11-22 22:27:38 +01:00
return html . replace (
2018-08-11 07:33:33 +02:00
/(((https?|ftp):\/\/[\w?!=&.\/-;#@~%+*-]+(?![\w\s?!&.\/;#~%"=-]*>))|((magnet):[\w?=&.\/-;#@~%+*-]+))/ig ,
2017-11-22 22:27:38 +01:00
'<a href="$1" rel="nofollow">$1</a>'
2015-09-05 17:12:11 +02:00
) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-06 13:07:46 +02:00
/ * *
* minimal sprintf emulation for % s and % d formats
*
2017-02-11 19:34:51 +01:00
* Note that this function needs the parameters in the same order as the
* format strings appear in the string , contrary to the original .
*
2017-01-14 15:29:12 +01:00
* @ see { @ link https : //stackoverflow.com/questions/610406/javascript-equivalent-to-printf-string-format#4795914}
2017-02-14 22:21:55 +01:00
* @ name Helper . sprintf
2017-01-14 15:29:12 +01:00
* @ function
* @ param { string } format
* @ param { ... * } args - one or multiple parameters injected into format string
* @ return { string }
2015-09-06 13:07:46 +02:00
* /
2017-02-08 20:12:22 +01:00
me . sprintf = function ( )
2015-09-06 13:07:46 +02:00
{
2018-12-29 18:40:59 +01:00
const args = Array . prototype . slice . call ( arguments ) ;
let format = args [ 0 ] ,
2015-09-06 13:07:46 +02:00
i = 1 ;
2017-02-11 19:34:51 +01:00
return format . replace ( /%(s|d)/g , function ( m ) {
2015-09-06 13:07:46 +02:00
// m is the matched format, e.g. %s, %d
2018-12-29 18:40:59 +01:00
let val = args [ i ] ;
2017-02-11 19:34:51 +01:00
// A switch statement so that the formatter can be extended.
switch ( m )
{
case '%d' :
val = parseFloat ( val ) ;
if ( isNaN ( val ) ) {
val = 0 ;
}
break ;
default :
// Default is %s
2015-09-06 13:07:46 +02:00
}
2017-02-11 19:34:51 +01:00
++ i ;
2015-09-06 13:07:46 +02:00
return val ;
} ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-21 22:43:00 +02:00
2015-09-19 11:21:13 +02:00
/ * *
* get value of cookie , if it was set , empty string otherwise
*
2017-01-14 15:29:12 +01:00
* @ see { @ link http : //www.w3schools.com/js/js_cookies.asp}
2017-02-14 22:21:55 +01:00
* @ name Helper . getCookie
2017-01-14 15:29:12 +01:00
* @ function
2017-02-12 17:11:21 +01:00
* @ param { string } cname - may not be empty
2017-01-14 15:29:12 +01:00
* @ return { string }
2015-09-19 11:21:13 +02:00
* /
2017-02-08 20:12:22 +01:00
me . getCookie = function ( cname ) {
2018-12-29 18:40:59 +01:00
const name = cname + '=' ,
ca = document . cookie . split ( ';' ) ;
for ( let i = 0 ; i < ca . length ; ++ i ) {
let c = ca [ i ] ;
2017-02-05 14:47:03 +01:00
while ( c . charAt ( 0 ) === ' ' )
{
c = c . substring ( 1 ) ;
}
2016-07-11 15:47:42 +02:00
if ( c . indexOf ( name ) === 0 )
{
return c . substring ( name . length , c . length ) ;
}
2015-09-19 11:21:13 +02:00
}
return '' ;
2018-01-06 09:26:10 +01:00
} ;
2016-07-19 16:12:11 +02:00
2017-02-05 14:47:03 +01:00
/ * *
2017-02-14 22:21:55 +01:00
* get the current location ( without search or hash part of the URL ) ,
2017-04-11 16:34:13 +02:00
* eg . https : //example.com/path/?aaaa#bbbb --> https://example.com/path/
2017-02-05 14:47:03 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name Helper . baseUri
2017-02-05 14:47:03 +01:00
* @ function
2017-02-14 22:21:55 +01:00
* @ return { string }
2017-02-05 14:47:03 +01:00
* /
2017-02-14 22:21:55 +01:00
me . baseUri = function ( )
2017-02-05 14:47:03 +01:00
{
2017-02-08 20:12:22 +01:00
// check for cached version
2017-02-14 22:21:55 +01:00
if ( baseUri !== null ) {
return baseUri ;
2017-02-05 14:47:03 +01:00
}
2018-08-04 17:25:59 +02:00
baseUri = window . location . origin + window . location . pathname ;
2017-02-14 22:21:55 +01:00
return baseUri ;
2018-01-06 09:26:10 +01:00
} ;
2016-07-19 16:12:11 +02:00
/ * *
2017-03-05 12:11:55 +01:00
* resets state , used for unit testing
2017-01-14 15:29:12 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name Helper . reset
2017-03-05 12:11:55 +01:00
* @ function
2016-07-19 16:12:11 +02:00
* /
2017-03-05 12:11:55 +01:00
me . reset = function ( )
{
baseUri = null ;
2018-01-06 09:26:10 +01:00
} ;
2017-03-05 12:11:55 +01:00
2017-04-11 16:34:13 +02:00
/ * *
* checks whether this is a bot we dislike
*
* @ name Helper . isBadBot
* @ function
* @ return { bool }
* /
me . isBadBot = function ( ) {
// check whether a bot user agent part can be found in the current
// user agent
2018-12-29 18:40:59 +01:00
for ( let i = 0 ; i < BadBotUA . length ; ++ i ) {
2017-04-11 16:34:13 +02:00
if ( navigator . userAgent . indexOf ( BadBotUA ) >= 0 ) {
return true ;
}
}
return false ;
}
2017-02-08 20:12:22 +01:00
return me ;
2017-03-05 12:11:55 +01:00
} ) ( ) ;
2015-09-05 17:12:11 +02:00
2015-09-06 13:07:46 +02:00
/ * *
2017-02-14 22:21:55 +01:00
* internationalization module
2017-01-14 15:29:12 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name I18n
2017-01-14 15:29:12 +01:00
* @ class
2015-09-06 13:07:46 +02:00
* /
2018-12-29 18:40:59 +01:00
const I18n = ( function ( ) {
const me = { } ;
2017-02-08 20:12:22 +01:00
2017-02-14 22:21:55 +01:00
/ * *
* const for string of loaded language
*
2017-03-13 20:24:18 +01:00
* @ name I18n . languageLoadedEvent
2017-02-14 22:21:55 +01:00
* @ private
* @ prop { string }
* @ readonly
* /
2018-12-29 18:40:59 +01:00
const languageLoadedEvent = 'languageLoaded' ;
2017-02-14 22:21:55 +01:00
2015-09-06 15:54:43 +02:00
/ * *
* supported languages , minus the built in 'en'
2017-01-14 15:29:12 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name I18n . supportedLanguages
2017-02-08 20:12:22 +01:00
* @ private
2017-01-14 15:29:12 +01:00
* @ prop { string [ ] }
* @ readonly
2015-09-06 15:54:43 +02:00
* /
2018-12-29 18:40:59 +01:00
const supportedLanguages = [ 'de' , 'es' , 'fr' , 'it' , 'hu' , 'no' , 'nl' , 'pl' , 'pt' , 'oc' , 'ru' , 'sl' , 'zh' ] ;
2017-02-08 20:12:22 +01:00
/ * *
* built in language
*
2017-03-13 20:24:18 +01:00
* @ name I18n . language
2017-02-08 20:12:22 +01:00
* @ private
2017-02-14 22:21:55 +01:00
* @ prop { string | null }
2017-02-08 20:12:22 +01:00
* /
2018-12-29 18:40:59 +01:00
let language = null ;
2017-02-08 20:12:22 +01:00
/ * *
* translation cache
*
2017-03-13 20:24:18 +01:00
* @ name I18n . translations
2017-02-08 20:12:22 +01:00
* @ private
* @ enum { Object }
* /
2018-12-29 18:40:59 +01:00
let translations = { } ;
2015-09-06 15:54:43 +02:00
2015-09-06 13:07:46 +02:00
/ * *
2017-03-13 20:24:18 +01:00
* translate a string , alias for I18n . translate
2015-09-06 13:07:46 +02:00
*
2017-02-14 22:21:55 +01:00
* @ name I18n . _
2017-01-14 15:29:12 +01:00
* @ function
2017-02-14 22:21:55 +01:00
* @ param { jQuery } $element - optional
2017-01-14 15:29:12 +01:00
* @ param { string } messageId
* @ param { ... * } args - one or multiple parameters injected into placeholders
* @ return { string }
2015-09-06 13:07:46 +02:00
* /
2017-02-08 20:12:22 +01:00
me . _ = function ( )
2015-09-06 13:07:46 +02:00
{
2017-02-14 22:21:55 +01:00
return me . translate . apply ( this , arguments ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-06 13:07:46 +02:00
/ * *
* translate a string
*
2017-02-14 22:21:55 +01:00
* Optionally pass a jQuery element as the first parameter , to automatically
* let the text of this element be replaced . In case the ( asynchronously
* loaded ) language is not downloadet yet , this will make sure the string
* is replaced when it is actually loaded .
* So for easy translations passing the jQuery object to apply it to is
* more save , especially when they are loaded in the beginning .
*
* @ name I18n . translate
2017-01-14 15:29:12 +01:00
* @ function
2017-02-14 22:21:55 +01:00
* @ param { jQuery } $element - optional
2017-01-14 15:29:12 +01:00
* @ param { string } messageId
* @ param { ... * } args - one or multiple parameters injected into placeholders
* @ return { string }
2015-09-06 13:07:46 +02:00
* /
2017-02-08 20:12:22 +01:00
me . translate = function ( )
2015-09-06 13:07:46 +02:00
{
2017-02-14 22:21:55 +01:00
// convert parameters to array
2018-12-29 18:40:59 +01:00
let args = Array . prototype . slice . call ( arguments ) ,
2017-02-14 22:21:55 +01:00
messageId ,
$element = null ;
// parse arguments
if ( args [ 0 ] instanceof jQuery ) {
// optional jQuery element as first parameter
$element = args [ 0 ] ;
args . shift ( ) ;
2016-07-11 15:47:42 +02:00
}
2017-02-14 22:21:55 +01:00
// extract messageId from arguments
2018-12-29 18:40:59 +01:00
let usesPlurals = $ . isArray ( args [ 0 ] ) ;
2017-02-14 22:21:55 +01:00
if ( usesPlurals ) {
2015-09-06 15:54:43 +02:00
// use the first plural form as messageId, otherwise the singular
2018-02-21 22:51:31 +01:00
messageId = args [ 0 ] . length > 1 ? args [ 0 ] [ 1 ] : args [ 0 ] [ 0 ] ;
2017-02-14 22:21:55 +01:00
} else {
2015-09-06 15:54:43 +02:00
messageId = args [ 0 ] ;
}
2017-02-14 22:21:55 +01:00
if ( messageId . length === 0 ) {
2016-07-11 15:47:42 +02:00
return messageId ;
}
2017-02-14 22:21:55 +01:00
// if no translation string cannot be found (in translations object)
2017-03-12 17:08:12 +01:00
if ( ! translations . hasOwnProperty ( messageId ) || language === null ) {
2017-02-14 22:21:55 +01:00
// if language is still loading and we have an elemt assigned
if ( language === null && $element !== null ) {
// handle the error by attaching the language loaded event
2018-12-29 18:40:59 +01:00
let orgArguments = arguments ;
2017-02-14 22:21:55 +01:00
$ ( document ) . on ( languageLoadedEvent , function ( ) {
2017-02-15 22:59:55 +01:00
// log to show that the previous error could be mitigated
2017-10-30 07:04:59 +01:00
console . warn ( 'Fix missing translation of \'' + messageId + '\' with now loaded language ' + language ) ;
2017-02-14 22:21:55 +01:00
// re-execute this function
me . translate . apply ( this , orgArguments ) ;
} ) ;
// and fall back to English for now until the real language
// file is loaded
}
2017-04-11 22:21:30 +02:00
// for all other languages than English for which this behaviour
2017-02-14 22:21:55 +01:00
// is expected as it is built-in, log error
2017-03-12 17:08:12 +01:00
if ( language !== null && language !== 'en' ) {
2017-02-14 22:21:55 +01:00
console . error ( 'Missing translation for: \'' + messageId + '\' in language ' + language ) ;
// fallback to English
2016-07-11 15:47:42 +02:00
}
2017-02-14 22:21:55 +01:00
// save English translation (should be the same on both sides)
2017-02-08 20:12:22 +01:00
translations [ messageId ] = args [ 0 ] ;
2015-09-06 15:54:43 +02:00
}
2017-02-14 22:21:55 +01:00
// lookup plural translation
if ( usesPlurals && $ . isArray ( translations [ messageId ] ) ) {
2018-12-29 18:40:59 +01:00
let n = parseInt ( args [ 1 ] || 1 , 10 ) ,
2017-02-08 20:12:22 +01:00
key = me . getPluralForm ( n ) ,
maxKey = translations [ messageId ] . length - 1 ;
2017-02-14 22:21:55 +01:00
if ( key > maxKey ) {
2016-07-11 15:47:42 +02:00
key = maxKey ;
}
2017-02-08 20:12:22 +01:00
args [ 0 ] = translations [ messageId ] [ key ] ;
2015-09-06 15:54:43 +02:00
args [ 1 ] = n ;
2017-02-14 22:21:55 +01:00
} else {
// lookup singular translation
2017-02-08 20:12:22 +01:00
args [ 0 ] = translations [ messageId ] ;
2015-09-06 15:54:43 +02:00
}
2017-02-14 22:21:55 +01:00
// format string
2018-12-29 18:40:59 +01:00
let output = Helper . sprintf . apply ( this , args ) ;
2017-02-14 22:21:55 +01:00
// if $element is given, apply text to element
if ( $element !== null ) {
2017-03-12 17:08:12 +01:00
// get last text node of element
2018-12-29 18:40:59 +01:00
let content = $element . contents ( ) ;
2017-03-12 17:08:12 +01:00
if ( content . length > 1 ) {
content [ content . length - 1 ] . nodeValue = ' ' + output ;
} else {
$element . text ( output ) ;
}
2015-09-06 13:07:46 +02:00
}
2017-02-14 22:21:55 +01:00
return output ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-06 13:07:46 +02:00
2015-09-08 20:48:18 +02:00
/ * *
* per language functions to use to determine the plural form
*
2017-01-14 15:29:12 +01:00
* @ see { @ link http : //localization-guide.readthedocs.org/en/latest/l10n/pluralforms.html}
2017-02-14 22:21:55 +01:00
* @ name I18n . getPluralForm
2017-01-14 15:29:12 +01:00
* @ function
2017-02-15 22:59:55 +01:00
* @ param { int } n
* @ return { int } array key
2015-09-08 20:48:18 +02:00
* /
2017-02-08 20:12:22 +01:00
me . getPluralForm = function ( n ) {
switch ( language )
2015-09-08 20:48:18 +02:00
{
case 'fr' :
2017-01-08 07:56:56 +01:00
case 'oc' :
2016-04-26 20:21:30 +02:00
case 'zh' :
2018-02-21 22:51:31 +01:00
return n > 1 ? 1 : 0 ;
2015-09-08 20:48:18 +02:00
case 'pl' :
2018-02-21 22:51:31 +01:00
return n === 1 ? 0 : ( n % 10 >= 2 && n % 10 <= 4 && ( n % 100 < 10 || n % 100 >= 20 ) ? 1 : 2 ) ;
2016-12-16 10:21:15 +01:00
case 'ru' :
2018-02-21 22:51:31 +01:00
return n % 10 === 1 && n % 100 !== 11 ? 0 : ( n % 10 >= 2 && n % 10 <= 4 && ( n % 100 < 10 || n % 100 >= 20 ) ? 1 : 2 ) ;
2017-01-01 14:35:39 +01:00
case 'sl' :
2018-02-21 22:51:31 +01:00
return n % 100 === 1 ? 1 : ( n % 100 === 2 ? 2 : ( n % 100 === 3 || n % 100 === 4 ? 3 : 0 ) ) ;
2018-05-31 21:04:35 +02:00
// de, en, es, hu, it, nl, no, pt
2015-09-08 20:48:18 +02:00
default :
2018-02-21 22:51:31 +01:00
return n !== 1 ? 1 : 0 ;
2015-09-08 20:48:18 +02:00
}
2018-01-06 09:26:10 +01:00
} ;
2015-09-08 20:48:18 +02:00
2015-09-06 15:54:43 +02:00
/ * *
2017-02-14 22:21:55 +01:00
* load translations into cache
2015-09-06 15:54:43 +02:00
*
2017-02-14 22:21:55 +01:00
* @ name I18n . loadTranslations
2017-01-14 15:29:12 +01:00
* @ function
2015-09-06 15:54:43 +02:00
* /
2017-02-08 20:12:22 +01:00
me . loadTranslations = function ( )
2015-09-06 13:07:46 +02:00
{
2018-12-29 18:40:59 +01:00
let newLanguage = Helper . getCookie ( 'lang' ) ;
2017-02-08 13:20:51 +01:00
2017-02-08 20:12:22 +01:00
// auto-select language based on browser settings
2017-02-14 22:21:55 +01:00
if ( newLanguage . length === 0 ) {
2018-04-07 04:53:00 +02:00
newLanguage = ( navigator . language || navigator . userLanguage || 'en' ) . substring ( 0 , 2 ) ;
2017-02-05 14:47:03 +01:00
}
2015-09-06 13:07:46 +02:00
2017-02-14 22:21:55 +01:00
// if language is already used skip update
2017-02-12 18:08:08 +01:00
if ( newLanguage === language ) {
2017-02-08 20:12:22 +01:00
return ;
2015-09-08 20:48:18 +02:00
}
2015-09-06 15:54:43 +02:00
2017-02-14 22:21:55 +01:00
// if language is built-in (English) skip update
if ( newLanguage === 'en' ) {
language = 'en' ;
return ;
2015-09-08 20:48:18 +02:00
}
2015-09-06 13:07:46 +02:00
2017-02-08 20:12:22 +01:00
// if language is not supported, show error
2017-02-12 18:08:08 +01:00
if ( supportedLanguages . indexOf ( newLanguage ) === - 1 ) {
2017-02-08 20:12:22 +01:00
console . error ( 'Language \'%s\' is not supported. Translation failed, fallback to English.' , newLanguage ) ;
2017-02-14 22:21:55 +01:00
language = 'en' ;
return ;
2017-02-08 20:12:22 +01:00
}
2015-09-06 15:54:43 +02:00
2017-02-14 22:21:55 +01:00
// load strings from JSON
2017-02-08 20:12:22 +01:00
$ . getJSON ( 'i18n/' + newLanguage + '.json' , function ( data ) {
language = newLanguage ;
translations = data ;
2017-02-14 22:21:55 +01:00
$ ( document ) . triggerHandler ( languageLoadedEvent ) ;
2017-02-08 20:12:22 +01:00
} ) . fail ( function ( data , textStatus , errorMsg ) {
console . error ( 'Language \'%s\' could not be loaded (%s: %s). Translation failed, fallback to English.' , newLanguage , textStatus , errorMsg ) ;
2017-02-14 22:21:55 +01:00
language = 'en' ;
2017-02-08 20:12:22 +01:00
} ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-06 13:07:46 +02:00
2017-03-25 09:41:24 +01:00
/ * *
* resets state , used for unit testing
*
* @ name I18n . reset
* @ function
* /
2017-03-26 09:24:42 +02:00
me . reset = function ( mockLanguage , mockTranslations )
2017-03-25 09:41:24 +01:00
{
2017-03-26 09:24:42 +02:00
language = mockLanguage || null ;
translations = mockTranslations || { } ;
2018-01-06 09:26:10 +01:00
} ;
2017-03-25 09:41:24 +01:00
2017-02-08 20:12:22 +01:00
return me ;
2017-03-26 09:24:42 +02:00
} ) ( ) ;
2015-09-06 13:07:46 +02:00
2015-09-05 17:12:11 +02:00
/ * *
2017-02-12 21:13:04 +01:00
* handles everything related to en / decryption
2017-01-14 15:29:12 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name CryptTool
2017-01-14 15:29:12 +01:00
* @ class
2015-09-05 17:12:11 +02:00
* /
2018-12-29 18:40:59 +01:00
const CryptTool = ( function ( ) {
const me = { } ;
2017-02-08 20:12:22 +01:00
2018-09-01 19:42:22 +02:00
/ * *
* convert UTF - 8 string stored in a DOMString to a standard UTF - 16 DOMString
*
* Iterates over the bytes of the message , converting them all hexadecimal
* percent encoded representations , then URI decodes them all
*
2018-12-29 18:40:59 +01:00
* @ name CryptTool . utf8To16
2018-09-01 19:42:22 +02:00
* @ function
* @ private
* @ param { string } message UTF - 8 string
* @ return { string } UTF - 16 string
* /
2018-12-29 18:40:59 +01:00
function utf8To16 ( message )
2018-09-01 19:42:22 +02:00
{
return decodeURIComponent (
message . split ( '' ) . map (
function ( character )
{
return '%' + ( '00' + character . charCodeAt ( 0 ) . toString ( 16 ) ) . slice ( - 2 ) ;
}
) . join ( '' )
) ;
}
2015-09-05 17:12:11 +02:00
/ * *
2018-08-05 08:56:03 +02:00
* convert DOMString ( UTF - 16 ) to a UTF - 8 string stored in a DOMString
*
* URI encodes the message , then finds the percent encoded characters
* and transforms these hexadecimal representation back into bytes
*
2018-12-29 18:40:59 +01:00
* @ name CryptTool . utf16To8
2018-08-05 08:56:03 +02:00
* @ function
* @ private
* @ param { string } message UTF - 16 string
* @ return { string } UTF - 8 string
* /
2018-12-29 18:40:59 +01:00
function utf16To8 ( message )
2018-08-05 08:56:03 +02:00
{
return encodeURIComponent ( message ) . replace (
/%([0-9A-F]{2})/g ,
function ( match , hexCharacter )
{
return String . fromCharCode ( '0x' + hexCharacter ) ;
}
) ;
}
/ * *
2018-09-01 19:42:22 +02:00
* convert ArrayBuffer into a UTF - 8 string
2018-08-05 08:56:03 +02:00
*
2018-09-01 19:42:22 +02:00
* Iterates over the bytes of the array , catenating them into a string
2018-08-05 08:56:03 +02:00
*
2018-12-29 18:40:59 +01:00
* @ name CryptTool . arraybufferToString
2018-09-01 19:42:22 +02:00
* @ function
* @ private
* @ param { ArrayBuffer } messageArray
* @ return { string } message
* /
2018-12-29 18:40:59 +01:00
function arraybufferToString ( messageArray )
2018-09-01 19:42:22 +02:00
{
2018-12-29 18:40:59 +01:00
const array = new Uint8Array ( messageArray ) ;
let message = '' ,
i = 0 ;
while ( i < array . length ) {
message += String . fromCharCode ( array [ i ++ ] ) ;
2018-09-01 19:42:22 +02:00
}
return message ;
}
/ * *
* convert UTF - 8 string into a Uint8Array
*
* Iterates over the bytes of the message , writing them to the array
*
2018-12-29 18:40:59 +01:00
* @ name CryptTool . stringToArraybuffer
2018-08-05 08:56:03 +02:00
* @ function
* @ private
* @ param { string } message UTF - 8 string
2018-09-01 19:42:22 +02:00
* @ return { Uint8Array } array
2018-08-05 08:56:03 +02:00
* /
2018-12-29 18:40:59 +01:00
function stringToArraybuffer ( message )
2018-08-05 08:56:03 +02:00
{
2018-12-29 18:40:59 +01:00
const messageArray = new Uint8Array ( message . length ) ;
for ( let i = 0 ; i < message . length ; ++ i ) {
messageArray [ i ] = message . charCodeAt ( i ) ;
2018-09-01 19:42:22 +02:00
}
return messageArray ;
2018-08-05 08:56:03 +02:00
}
/ * *
2018-12-27 21:32:13 +01:00
* compress a string ( deflate compression ) , returns buffer
2015-09-05 17:12:11 +02:00
*
2017-03-13 20:24:18 +01:00
* @ name CryptTool . compress
2018-12-27 21:32:13 +01:00
* @ async
2017-01-14 15:29:12 +01:00
* @ function
2017-02-12 18:08:08 +01:00
* @ private
2017-01-14 15:29:12 +01:00
* @ param { string } message
2018-12-27 21:32:13 +01:00
* @ param { string } mode
* @ return { ArrayBuffer } data
2015-09-05 17:12:11 +02:00
* /
2018-12-27 21:32:13 +01:00
async function compress ( message , mode )
2015-09-05 17:12:11 +02:00
{
2018-12-29 18:40:59 +01:00
message = stringToArraybuffer (
utf16To8 ( message )
2018-12-27 21:32:13 +01:00
) ;
if ( mode === 'zlib' ) {
return z . deflate ( message ) . buffer ;
2018-08-05 08:56:03 +02:00
}
2018-12-27 21:32:13 +01:00
return message ;
2017-02-12 18:08:08 +01:00
}
2015-09-05 17:12:11 +02:00
/ * *
2018-12-27 21:32:13 +01:00
* decompress potentially base64 encoded , deflate compressed buffer , returns string
2015-09-05 17:12:11 +02:00
*
2017-03-13 20:24:18 +01:00
* @ name CryptTool . decompress
2018-12-27 21:32:13 +01:00
* @ async
2017-01-14 15:29:12 +01:00
* @ function
2017-02-12 18:08:08 +01:00
* @ private
2018-12-27 21:32:13 +01:00
* @ param { ArrayBuffer } data
* @ param { string } mode
2017-01-14 15:29:12 +01:00
* @ return { string } message
2015-09-05 17:12:11 +02:00
* /
2018-12-27 21:32:13 +01:00
async function decompress ( data , mode )
2015-09-05 17:12:11 +02:00
{
2018-12-27 21:32:13 +01:00
if ( mode === 'zlib' || mode === 'none' ) {
if ( mode === 'zlib' ) {
2018-12-29 18:40:59 +01:00
data = z . inflate (
new Uint8Array ( data )
) . buffer ;
2018-12-27 21:32:13 +01:00
}
2018-12-29 18:40:59 +01:00
return utf8To16 (
arraybufferToString ( data )
2018-12-27 21:32:13 +01:00
) ;
}
2018-08-05 08:56:03 +02:00
// detect presence of Base64.js, indicating legacy ZeroBin paste
if ( typeof Base64 === 'undefined' ) {
2018-12-29 18:40:59 +01:00
return utf8To16 (
2018-12-27 21:32:13 +01:00
RawDeflate . inflate (
2018-12-29 18:40:59 +01:00
utf8To16 (
2018-12-27 21:32:13 +01:00
atob (
2018-12-29 18:40:59 +01:00
arraybufferToString ( data )
2018-12-27 21:32:13 +01:00
)
)
)
) ;
2018-08-05 08:56:03 +02:00
} else {
2018-12-27 21:32:13 +01:00
return Base64 . btou (
RawDeflate . inflate (
Base64 . fromBase64 (
2018-12-29 18:40:59 +01:00
arraybufferToString ( data )
2018-12-27 21:32:13 +01:00
)
)
) ;
2018-08-05 08:56:03 +02:00
}
2017-02-12 18:08:08 +01:00
}
2015-09-05 17:12:11 +02:00
2018-09-01 19:42:22 +02:00
/ * *
* returns specified number of random bytes
*
* @ name CryptTool . getRandomBytes
* @ function
* @ private
* @ param { int } length number of random bytes to fetch
* @ throws { string }
* @ return { string } random bytes
* /
function getRandomBytes ( length )
{
if (
typeof window !== 'undefined' &&
typeof Uint8Array !== 'undefined' &&
String . fromCodePoint &&
(
typeof window . crypto !== 'undefined' ||
typeof window . msCrypto !== 'undefined'
)
) {
// modern browser environment
2018-12-29 18:40:59 +01:00
let bytes = '' ;
const byteArray = new Uint8Array ( length ) ,
crypto = window . crypto || window . msCrypto ;
2018-09-01 19:42:22 +02:00
crypto . getRandomValues ( byteArray ) ;
2018-12-29 18:40:59 +01:00
for ( let i = 0 ; i < length ; ++ i ) {
2018-09-01 19:42:22 +02:00
bytes += String . fromCharCode ( byteArray [ i ] ) ;
}
return bytes ;
} else {
// legacy browser or unsupported environment
throw 'No supported crypto API detected, you may read pastes and comments, but can\'t create pastes or add new comments.' ;
}
2018-10-08 20:36:50 +02:00
}
2018-09-01 19:42:22 +02:00
2015-09-05 17:12:11 +02:00
/ * *
2018-10-20 22:34:36 +02:00
* derive cryptographic key from key string and password
2015-09-05 17:12:11 +02:00
*
2018-10-20 22:34:36 +02:00
* @ name CryptTool . deriveKey
2018-10-20 17:57:21 +02:00
* @ async
2017-01-14 15:29:12 +01:00
* @ function
2018-10-20 22:34:36 +02:00
* @ private
2017-01-14 15:29:12 +01:00
* @ param { string } key
* @ param { string } password
2018-12-25 17:34:39 +01:00
* @ param { array } spec cryptographic specification
2018-10-20 22:34:36 +02:00
* @ return { CryptoKey } derived key
2015-09-05 17:12:11 +02:00
* /
2018-12-25 17:34:39 +01:00
async function deriveKey ( key , password , spec )
2015-09-05 17:12:11 +02:00
{
2018-12-29 18:40:59 +01:00
let keyArray = stringToArraybuffer ( key ) ;
2018-12-28 05:49:34 +01:00
if ( password . length > 0 ) {
// version 1 pastes did append the passwords SHA-256 hash in hex
if ( spec [ 7 ] === 'rawdeflate' ) {
let passwordBuffer = await window . crypto . subtle . digest (
{ name : 'SHA-256' } ,
2018-12-29 18:40:59 +01:00
stringToArraybuffer (
utf16To8 ( password )
)
2018-12-28 05:49:34 +01:00
) ;
password = Array . prototype . map . call (
2018-12-29 18:40:59 +01:00
new Uint8Array ( passwordBuffer ) ,
x => ( '00' + x . toString ( 16 ) ) . slice ( - 2 )
2018-12-28 05:49:34 +01:00
) . join ( '' ) ;
}
2018-12-29 18:40:59 +01:00
let passwordArray = stringToArraybuffer ( password ) ,
2018-12-25 17:34:39 +01:00
newKeyArray = new Uint8Array ( keyArray . length + passwordArray . length ) ;
newKeyArray . set ( keyArray , 0 ) ;
newKeyArray . set ( passwordArray , keyArray . length ) ;
keyArray = newKeyArray ;
2015-09-05 17:12:11 +02:00
}
2018-09-01 19:42:22 +02:00
// import raw key
2018-10-20 19:53:21 +02:00
const importedKey = await window . crypto . subtle . importKey (
2018-09-01 19:42:22 +02:00
'raw' , // only 'raw' is allowed
2018-10-20 19:53:21 +02:00
keyArray ,
2018-09-01 19:42:22 +02:00
{ name : 'PBKDF2' } , // we use PBKDF2 for key derivation
false , // the key may not be exported
2018-10-08 20:36:50 +02:00
[ 'deriveKey' ] // we may only use it for key derivation
2018-10-20 22:05:35 +02:00
) ;
2018-09-01 19:42:22 +02:00
// derive a stronger key for use with AES
2018-12-25 17:34:39 +01:00
return window . crypto . subtle . deriveKey (
2018-09-01 19:42:22 +02:00
{
name : 'PBKDF2' , // we use PBKDF2 for key derivation
2018-12-29 18:40:59 +01:00
salt : stringToArraybuffer ( spec [ 1 ] ) , // salt used in HMAC
2018-12-25 17:34:39 +01:00
iterations : spec [ 2 ] , // amount of iterations to apply
2018-10-08 20:36:50 +02:00
hash : { name : 'SHA-256' } // can be "SHA-1", "SHA-256", "SHA-384" or "SHA-512"
2018-09-01 19:42:22 +02:00
} ,
importedKey ,
{
2018-12-25 17:34:39 +01:00
name : 'AES-' + spec [ 6 ] . toUpperCase ( ) , // can be any supported AES algorithm ("AES-CTR", "AES-CBC", "AES-CMAC", "AES-GCM", "AES-CFB", "AES-KW", "ECDH", "DH" or "HMAC")
length : spec [ 3 ] // can be 128, 192 or 256
2018-09-01 19:42:22 +02:00
} ,
false , // the key may not be exported
2018-12-28 05:49:34 +01:00
[ 'encrypt' , 'decrypt' ] // we may only use it for en- and decryption
2018-10-20 22:05:35 +02:00
) ;
2018-10-20 22:34:36 +02:00
}
2018-10-20 23:08:13 +02:00
/ * *
2018-12-25 17:34:39 +01:00
* gets crypto settings from specification and authenticated data
2018-10-20 23:08:13 +02:00
*
* @ name CryptTool . cryptoSettings
* @ function
* @ private
2018-12-25 17:34:39 +01:00
* @ param { string } adata authenticated data
* @ param { array } spec cryptographic specification
2018-10-20 23:08:13 +02:00
* @ return { object } crypto settings
* /
2018-12-25 17:34:39 +01:00
function cryptoSettings ( adata , spec )
2018-10-20 23:08:13 +02:00
{
return {
2018-12-25 17:34:39 +01:00
name : 'AES-' + spec [ 6 ] . toUpperCase ( ) , // can be any supported AES algorithm ("AES-CTR", "AES-CBC", "AES-CMAC", "AES-GCM", "AES-CFB", "AES-KW", "ECDH", "DH" or "HMAC")
2018-12-29 18:40:59 +01:00
iv : stringToArraybuffer ( spec [ 0 ] ) , // the initialization vector you used to encrypt
additionalData : stringToArraybuffer ( adata ) , // the addtional data you used during encryption (if any)
2018-12-25 17:34:39 +01:00
tagLength : spec [ 4 ] // the length of the tag you used to encrypt (if any)
2018-10-20 23:08:13 +02:00
} ;
}
2018-10-20 22:34:36 +02:00
/ * *
* compress , then encrypt message with given key and password
*
* @ name CryptTool . cipher
* @ async
* @ function
* @ param { string } key
* @ param { string } password
* @ param { string } message
2018-12-25 17:34:39 +01:00
* @ param { array } adata
2018-12-27 21:32:13 +01:00
* @ return { array } encrypted message in base64 encoding & adata containing encryption spec
2018-12-25 17:34:39 +01:00
* /
me . cipher = async function ( key , password , message , adata )
{
// AES in Galois Counter Mode, keysize 256 bit,
// authentication tag 128 bit, 10000 iterations in key derivation
const spec = [
getRandomBytes ( 16 ) , // initialization vector
getRandomBytes ( 8 ) , // salt
2018-12-25 20:19:57 +01:00
100000 , // iterations
2018-12-25 17:34:39 +01:00
256 , // key size
128 , // tag size
'aes' , // algorithm
'gcm' , // algorithm mode
2018-12-27 21:32:13 +01:00
'zlib' // compression
] , encodedSpec = [ ] ;
for ( let i = 0 ; i < spec . length ; ++ i ) {
encodedSpec [ i ] = i < 2 ? btoa ( spec [ i ] ) : spec [ i ] ;
}
2018-12-25 17:34:39 +01:00
if ( adata . length === 0 ) {
// comment
adata = encodedSpec ;
} else if ( adata [ 0 ] === null ) {
// paste
adata [ 0 ] = encodedSpec ;
}
2018-09-01 19:42:22 +02:00
// finally, encrypt message
2018-12-25 17:34:39 +01:00
return [
btoa (
2018-12-29 18:40:59 +01:00
arraybufferToString (
2018-12-25 17:34:39 +01:00
await window . crypto . subtle . encrypt (
cryptoSettings ( JSON . stringify ( adata ) , spec ) ,
await deriveKey ( key , password , spec ) ,
2018-12-27 21:32:13 +01:00
await compress ( message , spec [ 7 ] )
2018-12-25 17:34:39 +01:00
)
)
) ,
adata
] ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
/ * *
2017-01-14 15:29:12 +01:00
* decrypt message with key , then decompress
2015-09-05 17:12:11 +02:00
*
2017-02-14 22:21:55 +01:00
* @ name CryptTool . decipher
2018-10-20 17:57:21 +02:00
* @ async
2017-01-14 15:29:12 +01:00
* @ function
* @ param { string } key
* @ param { string } password
2018-12-25 17:34:39 +01:00
* @ param { string | object } data encrypted message
2018-01-06 13:32:07 +01:00
* @ return { string } decrypted message , empty if decryption failed
2015-09-05 17:12:11 +02:00
* /
2018-09-01 19:42:22 +02:00
me . decipher = async function ( key , password , data )
2015-09-05 17:12:11 +02:00
{
2018-12-28 05:49:34 +01:00
let adataString , encodedSpec , cipherMessage ;
2018-12-25 17:34:39 +01:00
if ( data instanceof Array ) {
// version 2
adataString = JSON . stringify ( data [ 1 ] ) ;
encodedSpec = ( data [ 1 ] [ 0 ] instanceof Array ? data [ 1 ] [ 0 ] : data [ 1 ] ) ;
cipherMessage = data [ 0 ] ;
} else if ( typeof data === 'string' ) {
// version 1
let object = JSON . parse ( data ) ;
adataString = atob ( object . adata ) ;
encodedSpec = [
object . iv ,
object . salt ,
object . iter ,
object . ks ,
object . ts ,
object . cipher ,
object . mode ,
'rawdeflate'
] ;
cipherMessage = object . ct ;
} else {
throw 'unsupported message format' ;
}
let spec = encodedSpec , plainText = '' ;
spec [ 0 ] = atob ( spec [ 0 ] ) ;
spec [ 1 ] = atob ( spec [ 1 ] ) ;
2018-09-01 19:42:22 +02:00
try {
2018-12-27 21:32:13 +01:00
return await decompress (
2018-12-25 17:34:39 +01:00
await window . crypto . subtle . decrypt (
cryptoSettings ( adataString , spec ) ,
await deriveKey ( key , password , spec ) ,
2018-12-29 18:40:59 +01:00
stringToArraybuffer (
atob ( cipherMessage )
)
2018-12-27 21:32:13 +01:00
) ,
encodedSpec [ 7 ]
2018-10-20 22:05:35 +02:00
) ;
2018-09-01 19:42:22 +02:00
} catch ( err ) {
return '' ;
2015-08-31 21:14:12 +02:00
}
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 13:20:51 +01:00
2017-02-13 21:12:00 +01:00
/ * *
* returns a random symmetric key
*
2018-08-04 22:30:01 +02:00
* generates 256 bit long keys ( 8 Bits * 32 ) for AES with 256 bit long blocks
*
2017-02-14 22:21:55 +01:00
* @ name CryptTool . getSymmetricKey
2017-02-13 21:12:00 +01:00
* @ function
2018-08-04 22:30:01 +02:00
* @ throws { string }
* @ return { string } base64 encoded key
2017-02-13 21:12:00 +01:00
* /
2017-03-25 00:58:59 +01:00
me . getSymmetricKey = function ( )
2017-02-13 21:12:00 +01:00
{
2018-12-29 18:40:59 +01:00
return getRandomBytes ( 32 ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
2017-02-08 20:12:22 +01:00
return me ;
2017-02-12 18:08:08 +01:00
} ) ( ) ;
2015-09-05 17:12:11 +02:00
2017-01-14 15:29:12 +01:00
/ * *
2017-02-15 22:59:55 +01:00
* ( Model ) Data source ( aka MVC )
2017-01-14 15:29:12 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name Model
2017-01-14 15:29:12 +01:00
* @ class
* /
2018-12-29 18:40:59 +01:00
const Model = ( function ( ) {
const me = { } ;
2017-02-12 21:13:04 +01:00
2018-12-29 18:40:59 +01:00
let id = null ,
pasteData = null ,
symmetricKey = null ,
2017-02-17 20:46:10 +01:00
$templates ;
2017-02-12 21:13:04 +01:00
2015-09-27 20:34:39 +02:00
/ * *
2017-02-14 22:21:55 +01:00
* returns the expiration set in the HTML
2017-01-14 15:29:12 +01:00
*
2017-02-15 22:59:55 +01:00
* @ name Model . getExpirationDefault
2017-02-14 22:21:55 +01:00
* @ function
* @ return string
2015-09-27 20:34:39 +02:00
* /
2017-02-14 22:21:55 +01:00
me . getExpirationDefault = function ( )
{
return $ ( '#pasteExpiration' ) . val ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-27 20:34:39 +02:00
2016-08-18 15:09:58 +02:00
/ * *
2017-02-14 22:21:55 +01:00
* returns the format set in the HTML
2017-01-14 15:29:12 +01:00
*
2017-02-15 22:59:55 +01:00
* @ name Model . getFormatDefault
2017-02-14 22:21:55 +01:00
* @ function
* @ return string
2016-08-18 15:09:58 +02:00
* /
2017-02-14 22:21:55 +01:00
me . getFormatDefault = function ( )
{
return $ ( '#pasteFormatter' ) . val ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2016-08-18 15:09:58 +02:00
/ * *
2018-04-30 20:01:38 +02:00
* returns the paste data ( including the cipher data )
2017-01-14 15:29:12 +01:00
*
2017-04-11 16:34:13 +02:00
* @ name Model . getPasteData
2017-01-14 15:29:12 +01:00
* @ function
2017-04-11 16:34:13 +02:00
* @ param { function } callback ( optional ) Called when data is available
* @ param { function } useCache ( optional ) Whether to use the cache or
* force a data reload . Default : true
2017-02-12 21:13:04 +01:00
* @ return string
2015-09-05 17:12:11 +02:00
* /
2017-04-11 16:34:13 +02:00
me . getPasteData = function ( callback , useCache )
2015-09-05 17:12:11 +02:00
{
2017-04-11 16:34:13 +02:00
// use cache if possible/allowed
if ( useCache !== false && pasteData !== null ) {
//execute callback
if ( typeof callback === 'function' ) {
return callback ( pasteData ) ;
}
// alternatively just using inline
return pasteData ;
}
// reload data
2018-12-25 17:34:39 +01:00
ServerInteraction . prepare ( ) ;
ServerInteraction . setUrl ( Helper . baseUri ( ) + '?' + me . getPasteId ( ) ) ;
2017-04-11 16:34:13 +02:00
2018-12-25 17:34:39 +01:00
ServerInteraction . setFailure ( function ( status , data ) {
2017-04-11 16:34:13 +02:00
// revert loading status…
Alert . hideLoading ( ) ;
TopNav . showViewButtons ( ) ;
// show error message
2018-12-25 17:34:39 +01:00
Alert . showError ( ServerInteraction . parseUploadError ( status , data , 'get paste data' ) ) ;
2018-05-22 11:43:44 +02:00
} ) ;
2018-12-25 17:34:39 +01:00
ServerInteraction . setSuccess ( function ( status , data ) {
2017-04-11 16:34:13 +02:00
pasteData = data ;
if ( typeof callback === 'function' ) {
return callback ( data ) ;
}
2018-05-22 11:43:44 +02:00
} ) ;
2018-12-25 17:34:39 +01:00
ServerInteraction . run ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
2015-09-12 17:33:16 +02:00
/ * *
2017-02-14 22:21:55 +01:00
* get the pastes unique identifier from the URL ,
2017-04-11 16:34:13 +02:00
* eg . https : //example.com/path/?c05354954c49a487#dfdsdgdgdfgdf returns c05354954c49a487
2015-09-12 17:33:16 +02:00
*
2017-02-15 22:59:55 +01:00
* @ name Model . getPasteId
2017-01-14 15:29:12 +01:00
* @ function
2017-02-14 22:21:55 +01:00
* @ return { string } unique identifier
2017-02-17 20:46:10 +01:00
* @ throws { string }
2015-09-12 17:33:16 +02:00
* /
2017-02-14 22:21:55 +01:00
me . getPasteId = function ( )
2015-09-12 17:33:16 +02:00
{
2017-02-14 22:21:55 +01:00
if ( id === null ) {
2018-05-22 11:41:35 +02:00
// Attention: This also returns the delete token inside of the ID, if it is specified
2017-02-14 22:21:55 +01:00
id = window . location . search . substring ( 1 ) ;
2017-02-17 20:46:10 +01:00
if ( id === '' ) {
throw 'no paste id given' ;
}
2015-09-12 17:33:16 +02:00
}
2017-02-14 22:21:55 +01:00
return id ;
2018-05-22 11:41:35 +02:00
}
/ * *
2018-12-29 18:40:59 +01:00
* returns true , when the URL has a delete token and the current call was used for deleting a paste .
2018-05-22 11:41:35 +02:00
*
* @ name Model . hasDeleteToken
* @ function
* @ return { bool }
* /
me . hasDeleteToken = function ( )
{
return window . location . search . indexOf ( 'deletetoken' ) !== - 1 ;
}
2015-09-12 17:33:16 +02:00
2015-09-05 17:12:11 +02:00
/ * *
2017-02-14 22:21:55 +01:00
* return the deciphering key stored in anchor part of the URL
2015-09-05 17:12:11 +02:00
*
2017-02-15 22:59:55 +01:00
* @ name Model . getPasteKey
2017-01-14 15:29:12 +01:00
* @ function
2017-02-17 20:46:10 +01:00
* @ return { string | null } key
* @ throws { string }
2015-09-05 17:12:11 +02:00
* /
2017-02-14 22:21:55 +01:00
me . getPasteKey = function ( )
2015-09-05 17:12:11 +02:00
{
2017-02-14 22:21:55 +01:00
if ( symmetricKey === null ) {
2018-12-29 18:40:59 +01:00
let newKey = window . location . hash . substring ( 1 ) ;
if ( newKey === '' ) {
2017-02-17 20:46:10 +01:00
throw 'no encryption key given' ;
}
2017-02-14 22:21:55 +01:00
// Some web 2.0 services and redirectors add data AFTER the anchor
// (such as &utm_source=...). We will strip any additional data.
2018-12-29 18:40:59 +01:00
let ampersandPos = newKey . indexOf ( '&' ) ;
2017-02-14 22:21:55 +01:00
if ( ampersandPos > - 1 )
2015-09-05 17:12:11 +02:00
{
2018-12-29 18:40:59 +01:00
newKey = newKey . substring ( 0 , ampersandPos ) ;
2017-02-14 22:21:55 +01:00
}
2018-12-29 18:40:59 +01:00
symmetricKey = atob ( newKey ) ;
2017-02-14 22:21:55 +01:00
}
2015-09-18 12:33:10 +02:00
2017-02-14 22:21:55 +01:00
return symmetricKey ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
/ * *
2017-02-17 20:46:10 +01:00
* returns a jQuery copy of the HTML template
2015-09-05 17:12:11 +02:00
*
2017-02-17 20:46:10 +01:00
* @ name Model . getTemplate
2017-01-14 15:29:12 +01:00
* @ function
2017-02-17 20:46:10 +01:00
* @ param { string } name - the name of the template
* @ return { jQuery }
2015-09-05 17:12:11 +02:00
* /
2017-02-17 20:46:10 +01:00
me . getTemplate = function ( name )
2015-09-05 17:12:11 +02:00
{
2017-02-17 20:46:10 +01:00
// find template
2018-12-29 18:40:59 +01:00
let $element = $templates . find ( '#' + name + 'template' ) . clone ( true ) ;
2017-02-17 20:46:10 +01:00
// change ID to avoid collisions (one ID should really be unique)
return $element . prop ( 'id' , name ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
/ * *
2017-02-25 09:35:55 +01:00
* resets state , used for unit testing
2015-09-05 17:12:11 +02:00
*
2017-02-25 09:35:55 +01:00
* @ name Model . reset
2017-01-14 15:29:12 +01:00
* @ function
2015-09-05 17:12:11 +02:00
* /
2017-02-25 09:35:55 +01:00
me . reset = function ( )
2015-09-05 17:12:11 +02:00
{
2017-04-11 16:34:13 +02:00
pasteData = $templates = id = symmetricKey = null ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
/ * *
2017-02-12 21:13:04 +01:00
* init navigation manager
2015-09-05 17:12:11 +02:00
*
2017-02-12 21:13:04 +01:00
* preloads jQuery elements
*
2017-02-15 22:59:55 +01:00
* @ name Model . init
2017-01-14 15:29:12 +01:00
* @ function
2015-09-05 17:12:11 +02:00
* /
2017-02-12 21:13:04 +01:00
me . init = function ( )
2015-09-01 22:33:07 +02:00
{
2017-02-17 20:46:10 +01:00
$templates = $ ( '#templates' ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
2017-02-12 21:13:04 +01:00
return me ;
2017-02-25 09:35:55 +01:00
} ) ( ) ;
2017-02-06 20:16:03 +01:00
2017-02-08 20:11:04 +01:00
/ * *
2017-02-14 22:21:55 +01:00
* Helper functions for user interface
*
* everything directly UI - related , which fits nowhere else
2017-02-08 20:11:04 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name UiHelper
2017-02-08 20:12:22 +01:00
* @ class
2017-02-08 20:11:04 +01:00
* /
2018-12-29 18:40:59 +01:00
const UiHelper = ( function ( ) {
const me = { } ;
2015-09-16 22:51:48 +02:00
/ * *
2017-02-13 11:35:04 +01:00
* handle history ( pop ) state changes
*
* currently this does only handle redirects to the home page .
2015-09-16 22:51:48 +02:00
*
2017-03-13 20:24:18 +01:00
* @ name UiHelper . historyChange
2017-02-14 22:21:55 +01:00
* @ private
2017-01-14 15:29:12 +01:00
* @ function
2017-02-13 11:35:04 +01:00
* @ param { Event } event
2015-09-16 22:51:48 +02:00
* /
2017-02-14 22:21:55 +01:00
function historyChange ( event )
2015-09-16 22:51:48 +02:00
{
2018-12-29 18:40:59 +01:00
let currentLocation = Helper . baseUri ( ) ;
2017-02-13 11:35:04 +01:00
if ( event . originalEvent . state === null && // no state object passed
2017-10-22 09:56:44 +02:00
event . target . location . href === currentLocation && // target location is home page
2017-02-13 11:35:04 +01:00
window . location . href === currentLocation // and we are not already on the home page
) {
// redirect to home page
window . location . href = currentLocation ;
2015-09-16 22:51:48 +02:00
}
2017-02-25 09:35:55 +01:00
}
2012-04-21 21:59:45 +02:00
2016-01-31 09:56:06 +01:00
/ * *
2017-02-13 11:35:04 +01:00
* reload the page
2016-01-31 09:56:06 +01:00
*
2017-02-14 22:21:55 +01:00
* This takes the user to the PrivateBin homepage .
2017-02-13 11:35:04 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name UiHelper . reloadHome
2017-01-14 15:29:12 +01:00
* @ function
2016-01-31 09:56:06 +01:00
* /
2017-02-14 22:21:55 +01:00
me . reloadHome = function ( )
2016-01-31 09:56:06 +01:00
{
2017-02-14 22:21:55 +01:00
window . location . href = Helper . baseUri ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 13:20:51 +01:00
2015-09-05 17:12:11 +02:00
/ * *
2017-02-17 20:46:10 +01:00
* checks whether the element is currently visible in the viewport ( so
* the user can actually see it )
2017-01-14 15:29:12 +01:00
*
2017-03-13 20:24:18 +01:00
* @ see { @ link https : //stackoverflow.com/a/40658647}
2017-02-17 20:46:10 +01:00
* @ name UiHelper . isVisible
2017-01-14 15:29:12 +01:00
* @ function
2017-02-17 20:46:10 +01:00
* @ param { jQuery } $element The link hash to move to .
2015-09-05 17:12:11 +02:00
* /
2017-02-17 20:46:10 +01:00
me . isVisible = function ( $element )
2015-09-05 17:12:11 +02:00
{
2018-12-29 18:40:59 +01:00
let elementTop = $element . offset ( ) . top ,
viewportTop = $ ( window ) . scrollTop ( ) ,
viewportBottom = viewportTop + $ ( window ) . height ( ) ;
2018-02-21 22:51:31 +01:00
return elementTop > viewportTop && elementTop < viewportBottom ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
2017-02-05 22:09:46 +01:00
/ * *
2017-02-17 20:46:10 +01:00
* scrolls to a specific element
2017-02-05 22:09:46 +01:00
*
2017-03-13 20:24:18 +01:00
* @ see { @ link https : //stackoverflow.com/questions/4198041/jquery-smooth-scroll-to-an-anchor#answer-12714767}
2017-02-17 20:46:10 +01:00
* @ name UiHelper . scrollTo
2017-02-05 22:09:46 +01:00
* @ function
2017-02-17 20:46:10 +01:00
* @ param { jQuery } $element The link hash to move to .
* @ param { ( number | string ) } animationDuration passed to jQuery . animate , when set to 0 the animation is skipped
* @ param { string } animationEffect passed to jQuery . animate
* @ param { function } finishedCallback function to call after animation finished
2017-02-05 22:09:46 +01:00
* /
2017-02-17 20:46:10 +01:00
me . scrollTo = function ( $element , animationDuration , animationEffect , finishedCallback )
2017-02-05 22:09:46 +01:00
{
2018-12-29 18:40:59 +01:00
let $body = $ ( 'html, body' ) ,
2017-02-17 20:46:10 +01:00
margin = 50 ,
2018-12-29 18:40:59 +01:00
callbackCalled = false ,
dest = 0 ;
2017-02-17 20:46:10 +01:00
2018-12-29 18:40:59 +01:00
// calculate destination place
2017-02-17 20:46:10 +01:00
// if it would scroll out of the screen at the bottom only scroll it as
// far as the screen can go
if ( $element . offset ( ) . top > $ ( document ) . height ( ) - $ ( window ) . height ( ) ) {
dest = $ ( document ) . height ( ) - $ ( window ) . height ( ) ;
} else {
dest = $element . offset ( ) . top - margin ;
}
// skip animation if duration is set to 0
if ( animationDuration === 0 ) {
window . scrollTo ( 0 , dest ) ;
} else {
// stop previous animation
$body . stop ( ) ;
// scroll to destination
$body . animate ( {
scrollTop : dest
} , animationDuration , animationEffect ) ;
}
// as we have finished we can enable scrolling again
$body . queue ( function ( next ) {
if ( ! callbackCalled ) {
// call user function if needed
if ( typeof finishedCallback !== 'undefined' ) {
finishedCallback ( ) ;
}
2017-02-05 22:09:46 +01:00
2017-02-17 20:46:10 +01:00
// prevent calling this function twice
callbackCalled = true ;
}
next ( ) ;
} ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-05 22:09:46 +01:00
2017-10-22 09:56:44 +02:00
/ * *
* trigger a history ( pop ) state change
*
* used to test the UiHelper . historyChange private function
*
* @ name UiHelper . mockHistoryChange
* @ function
2017-10-22 10:39:18 +02:00
* @ param { string } state ( optional ) state to mock
2017-10-22 09:56:44 +02:00
* /
2017-10-22 10:39:18 +02:00
me . mockHistoryChange = function ( state )
2017-10-22 09:56:44 +02:00
{
2017-10-22 10:39:18 +02:00
if ( typeof state === 'undefined' ) {
state = null ;
}
historyChange ( $ . Event ( 'popstate' , { originalEvent : new PopStateEvent ( 'popstate' , { state : state } ) , target : window } ) ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-05 22:09:46 +01:00
2017-02-06 22:39:45 +01:00
/ * *
2017-02-14 22:21:55 +01:00
* initialize
2017-02-06 22:39:45 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name UiHelper . init
2017-02-06 22:39:45 +01:00
* @ function
* /
2017-02-13 11:35:04 +01:00
me . init = function ( )
2017-02-06 22:39:45 +01:00
{
2017-02-14 22:21:55 +01:00
// update link to home page
$ ( '.reloadlink' ) . prop ( 'href' , Helper . baseUri ( ) ) ;
2017-02-13 11:35:04 +01:00
2017-02-14 22:21:55 +01:00
$ ( window ) . on ( 'popstate' , historyChange ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-12 17:33:16 +02:00
2017-02-13 11:35:04 +01:00
return me ;
2017-10-22 13:39:23 +02:00
} ) ( ) ;
2017-02-06 22:39:45 +01:00
2017-02-13 11:35:04 +01:00
/ * *
2017-02-14 22:21:55 +01:00
* Alert / error manager
2017-02-13 11:35:04 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name Alert
2017-02-13 11:35:04 +01:00
* @ class
* /
2018-12-29 18:40:59 +01:00
const Alert = ( function ( ) {
const me = { } ;
2017-02-13 11:35:04 +01:00
2018-12-29 18:40:59 +01:00
let $errorMessage ,
2017-02-17 22:46:18 +01:00
$loadingIndicator ,
2017-03-12 14:16:08 +01:00
$statusMessage ,
2018-12-29 18:40:59 +01:00
$remainingTime ,
currentIcon ,
customHandler ;
2017-02-15 22:59:55 +01:00
2018-12-29 18:40:59 +01:00
const alertType = [
'loading' , // not in bootstrap CSS, but using a plausible value here
'info' , // status icon
2017-02-17 20:46:10 +01:00
'warning' , // not used yet
2018-12-29 18:40:59 +01:00
'danger' // error icon
2017-02-17 20:46:10 +01:00
] ;
2015-09-05 17:12:11 +02:00
/ * *
2017-02-15 22:59:55 +01:00
* forwards a request to the i18n module and shows the element
2016-07-11 11:09:41 +02:00
*
2017-03-13 20:24:18 +01:00
* @ name Alert . handleNotification
2017-02-15 22:59:55 +01:00
* @ private
2017-01-14 15:29:12 +01:00
* @ function
2017-02-15 22:59:55 +01:00
* @ param { int } id - id of notification
* @ param { jQuery } $element - jQuery object
* @ param { string | array } args
* @ param { string | null } icon - optional , icon
2015-09-05 17:12:11 +02:00
* /
2017-02-15 22:59:55 +01:00
function handleNotification ( id , $element , args , icon )
{
2017-02-17 20:46:10 +01:00
// basic parsing/conversion of parameters
if ( typeof icon === 'undefined' ) {
icon = null ;
}
if ( typeof args === 'undefined' ) {
args = null ;
} else if ( typeof args === 'string' ) {
// convert string to array if needed
args = [ args ] ;
}
2017-03-12 14:16:08 +01:00
// pass to custom handler if defined
2017-02-17 20:46:10 +01:00
if ( typeof customHandler === 'function' ) {
2018-12-29 18:40:59 +01:00
let handlerResult = customHandler ( alertType [ id ] , $element , args , icon ) ;
2017-02-17 20:46:10 +01:00
if ( handlerResult === true ) {
2018-01-02 15:38:37 +01:00
// if it returns true, skip own handler
2017-02-17 20:46:10 +01:00
return ;
2016-07-11 11:09:41 +02:00
}
2017-02-17 20:46:10 +01:00
if ( handlerResult instanceof jQuery ) {
// continue processing with new element
$element = handlerResult ;
icon = null ; // icons not supported in this case
}
}
2017-02-15 22:59:55 +01:00
// handle icon
2017-02-17 20:46:10 +01:00
if ( icon !== null && // icon was passed
2017-02-15 22:59:55 +01:00
icon !== currentIcon [ id ] // and it differs from current icon
) {
2018-12-29 18:40:59 +01:00
let $glyphIcon = $element . find ( ':first' ) ;
2017-02-15 22:59:55 +01:00
// remove (previous) icon
$glyphIcon . removeClass ( currentIcon [ id ] ) ;
// any other thing as a string (e.g. 'null') (only) removes the icon
if ( typeof icon === 'string' ) {
// set new icon
currentIcon [ id ] = 'glyphicon-' + icon ;
$glyphIcon . addClass ( currentIcon [ id ] ) ;
2016-07-11 11:09:41 +02:00
}
2017-02-15 22:59:55 +01:00
}
2016-07-11 11:09:41 +02:00
2017-02-15 22:59:55 +01:00
// show text
2017-02-17 20:46:10 +01:00
if ( args !== null ) {
2017-03-12 17:08:12 +01:00
// add jQuery object to it as first parameter
args . unshift ( $element ) ;
// pass it to I18n
I18n . _ . apply ( this , args ) ;
2015-09-05 17:12:11 +02:00
}
2017-02-15 22:59:55 +01:00
// show notification
$element . removeClass ( 'hidden' ) ;
}
2015-09-05 17:12:11 +02:00
/ * *
2017-02-13 21:12:00 +01:00
* display a status message
2017-02-08 20:12:22 +01:00
*
2017-02-15 22:59:55 +01:00
* This automatically passes the text to I18n for translation .
2017-01-14 15:29:12 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name Alert . showStatus
2017-01-14 15:29:12 +01:00
* @ function
2017-02-15 22:59:55 +01:00
* @ param { string | array } message string , use an array for % s / % d options
* @ param { string | null } icon optional , the icon to show ,
* default : leave previous icon
2015-09-05 17:12:11 +02:00
* /
2018-01-06 13:32:07 +01:00
me . showStatus = function ( message , icon )
2015-09-05 17:12:11 +02:00
{
2017-02-15 22:59:55 +01:00
handleNotification ( 1 , $statusMessage , message , icon ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
2016-08-11 11:31:34 +02:00
/ * *
2017-02-15 22:59:55 +01:00
* display an error message
2017-01-14 15:29:12 +01:00
*
2017-02-15 22:59:55 +01:00
* This automatically passes the text to I18n for translation .
*
* @ name Alert . showError
2017-01-14 15:29:12 +01:00
* @ function
2017-02-15 22:59:55 +01:00
* @ param { string | array } message string , use an array for % s / % d options
* @ param { string | null } icon optional , the icon to show , default :
* leave previous icon
2016-08-11 11:31:34 +02:00
* /
2018-01-06 13:32:07 +01:00
me . showError = function ( message , icon )
2016-08-11 11:31:34 +02:00
{
2017-02-15 22:59:55 +01:00
handleNotification ( 3 , $errorMessage , message , icon ) ;
2018-01-06 09:26:10 +01:00
} ;
2016-08-11 11:31:34 +02:00
2016-08-18 15:09:58 +02:00
/ * *
2017-03-12 14:16:08 +01:00
* display remaining message
*
* This automatically passes the text to I18n for translation .
2016-08-18 15:09:58 +02:00
*
2017-03-12 14:16:08 +01:00
* @ name Alert . showRemaining
2017-01-14 15:29:12 +01:00
* @ function
2017-03-12 14:16:08 +01:00
* @ param { string | array } message string , use an array for % s / % d options
2016-08-18 15:09:58 +02:00
* /
2017-03-12 14:16:08 +01:00
me . showRemaining = function ( message )
2016-08-18 15:09:58 +02:00
{
2017-03-12 14:16:08 +01:00
handleNotification ( 1 , $remainingTime , message ) ;
2018-01-06 09:26:10 +01:00
} ;
2016-08-18 15:09:58 +02:00
2015-09-05 17:12:11 +02:00
/ * *
2017-02-15 22:59:55 +01:00
* shows a loading message , optionally with a percentage
2015-09-05 17:12:11 +02:00
*
2017-02-15 22:59:55 +01:00
* This automatically passes all texts to the i10s module .
2017-02-06 20:39:52 +01:00
*
2017-02-15 22:59:55 +01:00
* @ name Alert . showLoading
2017-01-14 15:29:12 +01:00
* @ function
2017-02-15 22:59:55 +01:00
* @ param { string | array | null } message optional , use an array for % s / % d options , default : 'Loading…'
* @ param { string | null } icon optional , the icon to show , default : leave previous icon
2015-09-05 17:12:11 +02:00
* /
2018-01-06 13:32:07 +01:00
me . showLoading = function ( message , icon )
2015-09-05 17:12:11 +02:00
{
2017-02-15 22:59:55 +01:00
// default message text
if ( typeof message === 'undefined' ) {
message = 'Loading…' ;
}
handleNotification ( 0 , $loadingIndicator , message , icon ) ;
2017-02-17 20:46:10 +01:00
// show loading status (cursor)
$ ( 'body' ) . addClass ( 'loading' ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
/ * *
2017-02-15 22:59:55 +01:00
* hides the loading message
2015-09-05 17:12:11 +02:00
*
2017-02-15 22:59:55 +01:00
* @ name Alert . hideLoading
2017-01-14 15:29:12 +01:00
* @ function
2015-09-05 17:12:11 +02:00
* /
2017-02-15 22:59:55 +01:00
me . hideLoading = function ( )
2015-09-05 17:12:11 +02:00
{
2017-02-15 22:59:55 +01:00
$loadingIndicator . addClass ( 'hidden' ) ;
2017-02-17 20:46:10 +01:00
// hide loading cursor
$ ( 'body' ) . removeClass ( 'loading' ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
/ * *
2017-02-15 22:59:55 +01:00
* hides any status / error messages
*
* This does not include the loading message .
2015-09-05 17:12:11 +02:00
*
2017-02-15 22:59:55 +01:00
* @ name Alert . hideMessages
2017-01-14 15:29:12 +01:00
* @ function
2015-09-05 17:12:11 +02:00
* /
2017-02-15 22:59:55 +01:00
me . hideMessages = function ( )
2015-09-05 17:12:11 +02:00
{
2017-02-17 20:46:10 +01:00
// also possible: $('.statusmessage').addClass('hidden');
2017-02-15 22:59:55 +01:00
$statusMessage . addClass ( 'hidden' ) ;
$errorMessage . addClass ( 'hidden' ) ;
2018-01-06 09:26:10 +01:00
} ;
2012-04-21 21:59:45 +02:00
2016-08-09 14:46:32 +02:00
/ * *
2017-02-17 20:46:10 +01:00
* set a custom handler , which gets all notifications .
*
* This handler gets the following arguments :
* alertType ( see array ) , $element , args , icon
* If it returns true , the own processing will be stopped so the message
* will not be displayed . Otherwise it will continue .
* As an aditional feature it can return q jQuery element , which will
* then be used to add the message there . Icons are not supported in
* that case and will be ignored .
* Pass 'null' to reset / delete the custom handler .
* Note that there is no notification when a message is supposed to get
* hidden .
*
* @ name Alert . setCustomHandler
2017-01-14 15:29:12 +01:00
* @ function
2017-02-17 20:46:10 +01:00
* @ param { function | null } newHandler
2016-08-09 14:46:32 +02:00
* /
2017-02-17 20:46:10 +01:00
me . setCustomHandler = function ( newHandler )
2016-08-09 14:46:32 +02:00
{
2017-02-17 20:46:10 +01:00
customHandler = newHandler ;
2018-01-06 09:26:10 +01:00
} ;
2016-08-09 14:46:32 +02:00
/ * *
2017-02-14 22:21:55 +01:00
* init status manager
2017-02-13 11:35:04 +01:00
*
2017-02-14 22:21:55 +01:00
* preloads jQuery elements
2016-08-09 14:46:32 +02:00
*
2017-02-14 22:21:55 +01:00
* @ name Alert . init
2017-01-14 15:29:12 +01:00
* @ function
2016-08-09 14:46:32 +02:00
* /
2017-02-14 22:21:55 +01:00
me . init = function ( )
2016-08-09 14:46:32 +02:00
{
2017-02-17 20:46:10 +01:00
// hide "no javascript" error message
2017-02-14 22:21:55 +01:00
$ ( '#noscript' ) . hide ( ) ;
2016-08-11 11:40:37 +02:00
2017-02-17 20:46:10 +01:00
// not a reset, but first set of the elements
2017-02-14 22:21:55 +01:00
$errorMessage = $ ( '#errormessage' ) ;
2017-02-15 22:59:55 +01:00
$loadingIndicator = $ ( '#loadingindicator' ) ;
2017-02-17 22:46:18 +01:00
$statusMessage = $ ( '#status' ) ;
2017-03-12 14:16:08 +01:00
$remainingTime = $ ( '#remainingtime' ) ;
2017-10-23 21:33:07 +02:00
currentIcon = [
'glyphicon-time' , // loading icon
'glyphicon-info-sign' , // status icon
'' , // reserved for warning, not used yet
'glyphicon-alert' // error icon
] ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 20:12:22 +01:00
2017-02-14 22:21:55 +01:00
return me ;
2017-03-25 00:58:59 +01:00
} ) ( ) ;
2017-02-14 22:21:55 +01:00
/ * *
* handles paste status / result
*
2017-03-13 20:24:18 +01:00
* @ name PasteStatus
2017-02-14 22:21:55 +01:00
* @ class
* /
2018-12-29 18:40:59 +01:00
const PasteStatus = ( function ( ) {
const me = { } ;
2017-02-14 22:21:55 +01:00
2018-12-29 18:40:59 +01:00
let $pasteSuccess ,
2017-02-14 22:21:55 +01:00
$pasteUrl ,
2017-02-17 22:46:18 +01:00
$remainingTime ,
$shortenButton ;
2016-08-09 14:46:32 +02:00
/ * *
2017-02-13 21:12:00 +01:00
* forward to URL shortener
2016-08-09 14:46:32 +02:00
*
2017-03-13 20:24:18 +01:00
* @ name PasteStatus . sendToShortener
2017-02-13 21:12:00 +01:00
* @ private
2017-01-14 15:29:12 +01:00
* @ function
2016-08-09 14:46:32 +02:00
* /
2018-01-06 13:32:07 +01:00
function sendToShortener ( )
2016-08-09 14:46:32 +02:00
{
2018-02-21 22:51:31 +01:00
window . location . href = $shortenButton . data ( 'shortener' ) +
encodeURIComponent ( $pasteUrl . attr ( 'href' ) ) ;
2017-02-13 21:12:00 +01:00
}
/ * *
2017-02-14 22:21:55 +01:00
* Forces opening the paste if the link does not do this automatically .
*
* This is necessary as browsers will not reload the page when it is
* already loaded ( which is fake as it is set via history . pushState ( ) ) .
2017-02-13 21:12:00 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name PasteStatus . pasteLinkClick
2017-02-14 22:21:55 +01:00
* @ function
* /
2018-01-06 13:32:07 +01:00
function pasteLinkClick ( )
2017-02-14 22:21:55 +01:00
{
// check if location is (already) shown in URL bar
if ( window . location . href === $pasteUrl . attr ( 'href' ) ) {
// if so we need to load link by reloading the current site
window . location . reload ( true ) ;
}
}
/ * *
* creates a notification after a successfull paste upload
2017-02-13 21:12:00 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name PasteStatus . createPasteNotification
2017-02-13 21:12:00 +01:00
* @ function
* @ param { string } url
* @ param { string } deleteUrl
* /
me . createPasteNotification = function ( url , deleteUrl )
2017-02-12 18:08:08 +01:00
{
2017-02-25 09:35:55 +01:00
$ ( '#pastelink' ) . html (
2017-02-14 22:21:55 +01:00
I18n . _ (
2017-02-13 21:12:00 +01:00
'Your paste is <a id="pasteurl" href="%s">%s</a> <span id="copyhint">(Hit [Ctrl]+[c] to copy)</span>' ,
url , url
)
) ;
// save newly created element
$pasteUrl = $ ( '#pasteurl' ) ;
// and add click event
$pasteUrl . click ( pasteLinkClick ) ;
// shorten button
2017-02-14 22:21:55 +01:00
$ ( '#deletelink' ) . html ( '<a href="' + deleteUrl + '">' + I18n . _ ( 'Delete data' ) + '</a>' ) ;
2017-02-13 11:35:04 +01:00
2017-02-13 21:12:00 +01:00
// show result
$pasteSuccess . removeClass ( 'hidden' ) ;
// we pre-select the link so that the user only has to [Ctrl]+[c] the link
2017-02-14 22:21:55 +01:00
Helper . selectText ( $pasteUrl [ 0 ] ) ;
2018-01-06 09:26:10 +01:00
} ;
2012-04-21 21:59:45 +02:00
2017-02-13 21:12:00 +01:00
/ * *
2017-02-14 22:21:55 +01:00
* shows the remaining time
2017-02-13 21:12:00 +01:00
*
2017-02-15 22:59:55 +01:00
* @ name PasteStatus . showRemainingTime
2017-02-13 21:12:00 +01:00
* @ function
2017-02-14 22:21:55 +01:00
* @ param { object } pasteMetaData
2017-02-13 21:12:00 +01:00
* /
2017-02-14 22:21:55 +01:00
me . showRemainingTime = function ( pasteMetaData )
2017-02-13 21:12:00 +01:00
{
2017-02-14 22:21:55 +01:00
if ( pasteMetaData . burnafterreading ) {
// display paste "for your eyes only" if it is deleted
2017-04-13 10:46:09 +02:00
// the paste has been deleted when the JSON with the ciphertext
2017-04-11 16:34:13 +02:00
// has been downloaded
2017-02-14 22:21:55 +01:00
2018-10-08 20:36:50 +02:00
Alert . showRemaining ( 'FOR YOUR EYES ONLY. Don\'t close this window, this message can\'t be displayed again.' ) ;
2017-02-14 22:21:55 +01:00
$remainingTime . addClass ( 'foryoureyesonly' ) ;
// discourage cloning (it cannot really be prevented)
TopNav . hideCloneButton ( ) ;
} else if ( pasteMetaData . expire _date ) {
// display paste expiration
2018-12-29 18:40:59 +01:00
let expiration = Helper . secondsToHuman ( pasteMetaData . remaining _time ) ,
2017-02-14 22:21:55 +01:00
expirationLabel = [
'This document will expire in %d ' + expiration [ 1 ] + '.' ,
'This document will expire in %d ' + expiration [ 1 ] + 's.'
] ;
2017-03-12 17:08:12 +01:00
Alert . showRemaining ( [ expirationLabel , expiration [ 0 ] ] ) ;
2017-11-14 06:52:12 +01:00
$remainingTime . removeClass ( 'foryoureyesonly' ) ;
2017-02-14 22:21:55 +01:00
} else {
// never expires
return ;
2017-02-13 21:12:00 +01:00
}
2017-02-14 22:21:55 +01:00
// in the end, display notification
$remainingTime . removeClass ( 'hidden' ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
2017-02-15 22:59:55 +01:00
/ * *
* hides the remaining time and successful upload notification
*
2018-03-01 06:43:30 +01:00
* @ name PasteStatus . hideMessages
2017-02-15 22:59:55 +01:00
* @ function
* /
me . hideMessages = function ( )
{
$remainingTime . addClass ( 'hidden' ) ;
$pasteSuccess . addClass ( 'hidden' ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-15 22:59:55 +01:00
2017-02-08 20:12:22 +01:00
/ * *
2017-02-13 11:35:04 +01:00
* init status manager
2017-02-08 20:12:22 +01:00
*
2017-02-13 11:35:04 +01:00
* preloads jQuery elements
*
2017-03-13 20:24:18 +01:00
* @ name PasteStatus . init
2017-02-08 20:12:22 +01:00
* @ function
* /
2017-02-12 18:08:08 +01:00
me . init = function ( )
2017-02-08 20:12:22 +01:00
{
2017-11-16 08:57:08 +01:00
$pasteSuccess = $ ( '#pastesuccess' ) ;
2017-02-14 22:21:55 +01:00
// $pasteUrl is saved in me.createPasteNotification() after creation
$remainingTime = $ ( '#remainingtime' ) ;
2017-02-17 22:46:18 +01:00
$shortenButton = $ ( '#shortenbutton' ) ;
2017-02-08 20:12:22 +01:00
2017-02-13 21:12:00 +01:00
// bind elements
$shortenButton . click ( sendToShortener ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 13:20:51 +01:00
2017-02-12 18:08:08 +01:00
return me ;
2017-11-13 21:57:49 +01:00
} ) ( ) ;
2017-02-12 18:08:08 +01:00
/ * *
2017-02-14 22:21:55 +01:00
* password prompt
2017-02-12 18:08:08 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name Prompt
2017-02-12 18:08:08 +01:00
* @ class
* /
2018-12-29 18:40:59 +01:00
const Prompt = ( function ( ) {
const me = { } ;
2017-02-12 18:08:08 +01:00
2018-12-29 18:40:59 +01:00
let $passwordDecrypt ,
2017-02-13 11:35:04 +01:00
$passwordForm ,
2018-12-29 18:40:59 +01:00
$passwordModal ,
password = '' ;
2017-02-14 22:21:55 +01:00
2017-03-13 20:24:18 +01:00
/ * *
* submit a password in the modal dialog
*
* @ name Prompt . submitPasswordModal
* @ private
* @ function
* @ param { Event } event
* /
function submitPasswordModal ( event )
{
event . preventDefault ( ) ;
// get input
password = $passwordDecrypt . val ( ) ;
// hide modal
$passwordModal . modal ( 'hide' ) ;
PasteDecrypter . run ( ) ;
}
2017-02-08 20:12:22 +01:00
/ * *
2017-02-13 11:35:04 +01:00
* ask the user for the password and set it
2017-02-08 20:12:22 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name Prompt . requestPassword
2017-02-08 20:12:22 +01:00
* @ function
* /
2017-02-13 11:35:04 +01:00
me . requestPassword = function ( )
2017-02-08 20:12:22 +01:00
{
2017-02-17 20:46:10 +01:00
// show new bootstrap method (if available)
if ( $passwordModal . length !== 0 ) {
$passwordModal . modal ( {
backdrop : 'static' ,
keyboard : false
} ) ;
return ;
}
2017-02-15 22:59:55 +01:00
2017-02-17 20:46:10 +01:00
// fallback to old method for page template
2018-07-21 08:44:04 +02:00
password = prompt ( I18n . _ ( 'Please enter the password for this paste:' ) , '' ) ;
if ( password === null ) {
2017-02-17 20:46:10 +01:00
throw 'password prompt canceled' ;
}
if ( password . length === 0 ) {
2017-11-20 08:49:25 +01:00
// recurse…
2017-02-17 20:46:10 +01:00
return me . requestPassword ( ) ;
}
2018-07-21 08:44:04 +02:00
PasteDecrypter . run ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
/ * *
2017-04-11 16:34:13 +02:00
* get the cached password
2017-02-14 22:21:55 +01:00
*
2017-02-15 22:59:55 +01:00
* If you do not get a password with this function
* ( returns an empty string ) , use requestPassword .
2017-02-14 22:21:55 +01:00
*
* @ name Prompt . getPassword
* @ function
2017-02-15 22:59:55 +01:00
* @ return { string }
2017-02-14 22:21:55 +01:00
* /
me . getPassword = function ( )
{
return password ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-05 22:09:46 +01:00
2017-04-11 22:21:30 +02:00
/ * *
* resets the password to an empty string
*
* @ name Prompt . reset
* @ function
* /
me . reset = function ( )
{
// reset internal
password = '' ;
// and also reset UI
$passwordDecrypt . val ( '' ) ;
}
2017-02-08 20:12:22 +01:00
/ * *
2017-02-13 11:35:04 +01:00
* init status manager
2017-02-08 20:12:22 +01:00
*
2017-02-13 11:35:04 +01:00
* preloads jQuery elements
*
2017-03-13 20:24:18 +01:00
* @ name Prompt . init
2017-02-08 20:12:22 +01:00
* @ function
* /
2017-02-13 11:35:04 +01:00
me . init = function ( )
2017-02-08 20:12:22 +01:00
{
2017-02-13 11:35:04 +01:00
$passwordDecrypt = $ ( '#passworddecrypt' ) ;
2017-02-17 22:46:18 +01:00
$passwordForm = $ ( '#passwordform' ) ;
$passwordModal = $ ( '#passwordmodal' ) ;
2017-02-08 20:12:22 +01:00
2017-02-13 11:35:04 +01:00
// bind events
2017-02-08 20:12:22 +01:00
2017-02-13 11:35:04 +01:00
// focus password input when it is shown
2017-02-17 20:46:10 +01:00
$passwordModal . on ( 'shown.bs.Model' , function ( ) {
2017-02-13 11:35:04 +01:00
$passwordDecrypt . focus ( ) ;
} ) ;
2017-02-15 22:59:55 +01:00
// handle Model password submission
2017-02-17 20:46:10 +01:00
$passwordForm . submit ( submitPasswordModal ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
2017-02-12 18:08:08 +01:00
return me ;
2017-03-25 00:58:59 +01:00
} ) ( ) ;
2017-02-12 18:08:08 +01:00
/ * *
2017-02-14 22:21:55 +01:00
* Manage paste / message input , and preview tab
*
* Note that the actual preview is handled by PasteViewer .
2017-02-12 18:08:08 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name Editor
2017-02-12 18:08:08 +01:00
* @ class
* /
2018-12-29 18:40:59 +01:00
const Editor = ( function ( ) {
const me = { } ;
2017-02-12 18:08:08 +01:00
2018-12-29 18:40:59 +01:00
let $editorTabs ,
2017-02-13 11:35:04 +01:00
$messageEdit ,
$messagePreview ,
2018-12-29 18:40:59 +01:00
$message ,
isPreview = false ;
2016-08-09 14:46:32 +02:00
2015-10-15 22:06:01 +02:00
/ * *
2017-01-14 15:29:12 +01:00
* support input of tab character
2015-10-15 22:06:01 +02:00
*
2017-02-14 22:21:55 +01:00
* @ name Editor . supportTabs
2017-01-14 15:29:12 +01:00
* @ function
* @ param { Event } event
2017-02-15 22:59:55 +01:00
* @ this $message ( but not used , so it is jQuery - free , possibly faster )
2015-10-15 22:06:01 +02:00
* /
2017-02-13 11:35:04 +01:00
function supportTabs ( event )
2015-10-15 22:06:01 +02:00
{
2018-12-29 18:40:59 +01:00
const keyCode = event . keyCode || event . which ;
2015-10-15 22:06:01 +02:00
// tab was pressed
2017-02-15 22:59:55 +01:00
if ( keyCode === 9 ) {
2015-10-15 22:06:01 +02:00
// get caret position & selection
2018-12-29 18:40:59 +01:00
const val = this . value ,
start = this . selectionStart ,
end = this . selectionEnd ;
2015-10-15 22:06:01 +02:00
// set textarea value to: text before caret + tab + text after caret
this . value = val . substring ( 0 , start ) + '\t' + val . substring ( end ) ;
// put caret at right position again
this . selectionStart = this . selectionEnd = start + 1 ;
2017-02-15 22:59:55 +01:00
// prevent the textarea to lose focus
event . preventDefault ( ) ;
2015-10-15 22:06:01 +02:00
}
2017-02-13 11:35:04 +01:00
}
2015-10-15 22:06:01 +02:00
2016-07-11 11:09:41 +02:00
/ * *
2017-02-14 22:21:55 +01:00
* view the Editor tab
2016-07-11 11:09:41 +02:00
*
2017-02-14 22:21:55 +01:00
* @ name Editor . viewEditor
2017-01-14 15:29:12 +01:00
* @ function
2017-02-13 11:35:04 +01:00
* @ param { Event } event - optional
2016-07-11 11:09:41 +02:00
* /
2017-02-13 11:35:04 +01:00
function viewEditor ( event )
2016-07-11 11:09:41 +02:00
{
2017-02-13 11:35:04 +01:00
// toggle buttons
$messageEdit . addClass ( 'active' ) ;
$messagePreview . removeClass ( 'active' ) ;
2017-02-14 22:21:55 +01:00
PasteViewer . hide ( ) ;
2017-02-13 11:35:04 +01:00
// reshow input
$message . removeClass ( 'hidden' ) ;
me . focusInput ( ) ;
// finish
isPreview = false ;
2017-02-14 22:21:55 +01:00
// prevent jumping of page to top
if ( typeof event !== 'undefined' ) {
event . preventDefault ( ) ;
}
2017-02-13 11:35:04 +01:00
}
2016-07-11 11:09:41 +02:00
/ * *
2017-01-14 15:29:12 +01:00
* view the preview tab
2016-07-11 11:09:41 +02:00
*
2017-02-14 22:21:55 +01:00
* @ name Editor . viewPreview
2017-01-14 15:29:12 +01:00
* @ function
* @ param { Event } event
2016-07-11 11:09:41 +02:00
* /
2017-02-13 11:35:04 +01:00
function viewPreview ( event )
2016-07-11 11:09:41 +02:00
{
2017-02-13 11:35:04 +01:00
// toggle buttons
$messageEdit . removeClass ( 'active' ) ;
$messagePreview . addClass ( 'active' ) ;
// hide input as now preview is shown
$message . addClass ( 'hidden' ) ;
// show preview
2017-02-14 22:21:55 +01:00
PasteViewer . setText ( $message . val ( ) ) ;
2017-05-15 22:05:52 +02:00
if ( AttachmentViewer . hasAttachmentData ( ) ) {
2018-12-29 18:40:59 +01:00
let attachmentData = AttachmentViewer . getAttachmentData ( ) || AttachmentViewer . getAttachmentLink ( ) . attr ( 'href' ) ;
2017-05-15 22:05:52 +02:00
AttachmentViewer . handleAttachmentPreview ( AttachmentViewer . getAttachmentPreview ( ) , attachmentData ) ;
2017-05-13 21:27:41 +02:00
}
2017-02-14 22:21:55 +01:00
PasteViewer . run ( ) ;
2017-02-13 11:35:04 +01:00
// finish
isPreview = true ;
2017-02-14 22:21:55 +01:00
// prevent jumping of page to top
if ( typeof event !== 'undefined' ) {
event . preventDefault ( ) ;
}
2017-02-13 11:35:04 +01:00
}
2016-07-11 11:09:41 +02:00
2017-02-05 14:47:03 +01:00
/ * *
2017-02-13 11:35:04 +01:00
* get the state of the preview
2017-02-05 14:47:03 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name Editor . isPreview
2017-02-05 14:47:03 +01:00
* @ function
* /
2017-02-13 11:35:04 +01:00
me . isPreview = function ( )
2017-02-05 14:47:03 +01:00
{
2017-02-13 11:35:04 +01:00
return isPreview ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-05 14:47:03 +01:00
2017-02-05 21:22:09 +01:00
/ * *
2017-02-14 22:21:55 +01:00
* reset the Editor view
2017-02-05 21:22:09 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name Editor . resetInput
2017-02-05 21:22:09 +01:00
* @ function
* /
2017-02-13 11:35:04 +01:00
me . resetInput = function ( )
2017-02-05 21:22:09 +01:00
{
2017-02-13 11:35:04 +01:00
// go back to input
if ( isPreview ) {
viewEditor ( ) ;
2017-02-05 21:22:09 +01:00
}
2017-02-13 11:35:04 +01:00
// clear content
$message . val ( '' ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-05 21:22:09 +01:00
2015-09-05 17:12:11 +02:00
/ * *
2017-02-14 22:21:55 +01:00
* shows the Editor
2017-01-14 15:29:12 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name Editor . show
2017-01-14 15:29:12 +01:00
* @ function
2015-09-05 17:12:11 +02:00
* /
2017-02-13 11:35:04 +01:00
me . show = function ( )
2015-09-05 17:12:11 +02:00
{
2017-02-13 11:35:04 +01:00
$message . removeClass ( 'hidden' ) ;
$editorTabs . removeClass ( 'hidden' ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
2015-09-16 22:51:48 +02:00
/ * *
2017-02-14 22:21:55 +01:00
* hides the Editor
2017-01-14 15:29:12 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name Editor . reset
2017-01-14 15:29:12 +01:00
* @ function
2015-09-16 22:51:48 +02:00
* /
2017-02-13 11:35:04 +01:00
me . hide = function ( )
2015-09-16 22:51:48 +02:00
{
2017-02-13 11:35:04 +01:00
$message . addClass ( 'hidden' ) ;
$editorTabs . addClass ( 'hidden' ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-16 22:51:48 +02:00
2016-11-13 18:12:10 +01:00
/ * *
2017-02-13 11:35:04 +01:00
* focuses the message input
2017-01-14 15:29:12 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name Editor . focusInput
2017-01-14 15:29:12 +01:00
* @ function
2016-11-13 18:12:10 +01:00
* /
2017-02-13 11:35:04 +01:00
me . focusInput = function ( )
2016-11-13 18:12:10 +01:00
{
2017-02-13 11:35:04 +01:00
$message . focus ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2016-11-13 18:12:10 +01:00
/ * *
2017-02-15 22:59:55 +01:00
* sets a new text
2016-11-13 18:12:10 +01:00
*
2017-02-15 22:59:55 +01:00
* @ name Editor . setText
2017-01-14 15:29:12 +01:00
* @ function
2017-02-15 22:59:55 +01:00
* @ param { string } newText
2016-11-13 18:12:10 +01:00
* /
2017-02-15 22:59:55 +01:00
me . setText = function ( newText )
2016-11-13 18:12:10 +01:00
{
2017-02-15 22:59:55 +01:00
$message . val ( newText ) ;
2018-01-06 09:26:10 +01:00
} ;
2016-11-13 18:12:10 +01:00
2015-09-05 17:12:11 +02:00
/ * *
2017-02-13 11:35:04 +01:00
* returns the current text
2015-09-05 17:12:11 +02:00
*
2017-02-14 22:21:55 +01:00
* @ name Editor . getText
2017-01-14 15:29:12 +01:00
* @ function
2017-02-13 11:35:04 +01:00
* @ return { string }
2015-09-05 17:12:11 +02:00
* /
2017-02-13 11:35:04 +01:00
me . getText = function ( )
2015-09-05 17:12:11 +02:00
{
2018-01-06 10:57:54 +01:00
return $message . val ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2015-09-05 17:12:11 +02:00
/ * *
2017-02-12 18:08:08 +01:00
* init status manager
2015-09-05 17:12:11 +02:00
*
2017-02-12 18:08:08 +01:00
* preloads jQuery elements
*
2017-02-14 22:21:55 +01:00
* @ name Editor . init
2017-01-14 15:29:12 +01:00
* @ function
2015-09-05 17:12:11 +02:00
* /
2017-02-12 18:08:08 +01:00
me . init = function ( )
2015-09-05 17:12:11 +02:00
{
2017-02-13 11:35:04 +01:00
$editorTabs = $ ( '#editorTabs' ) ;
2017-02-17 22:46:18 +01:00
$message = $ ( '#message' ) ;
2017-02-12 18:08:08 +01:00
2017-02-13 11:35:04 +01:00
// bind events
$message . keydown ( supportTabs ) ;
2017-02-12 21:13:04 +01:00
2017-02-13 11:35:04 +01:00
// bind click events to tab switchers (a), but save parent of them
// (li)
$messageEdit = $ ( '#messageedit' ) . click ( viewEditor ) . parent ( ) ;
$messagePreview = $ ( '#messagepreview' ) . click ( viewPreview ) . parent ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2016-07-11 11:09:41 +02:00
2017-02-12 18:08:08 +01:00
return me ;
2017-03-25 00:58:59 +01:00
} ) ( ) ;
2017-02-12 18:08:08 +01:00
/ * *
2017-02-14 22:21:55 +01:00
* ( view ) Parse and show paste .
2017-02-12 18:08:08 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name PasteViewer
2017-02-12 18:08:08 +01:00
* @ class
* /
2018-12-29 18:40:59 +01:00
const PasteViewer = ( function ( ) {
const me = { } ;
2017-02-12 18:08:08 +01:00
2018-12-29 18:40:59 +01:00
let $placeholder ,
2017-02-13 11:35:04 +01:00
$prettyMessage ,
2017-02-17 22:46:18 +01:00
$prettyPrint ,
2018-12-29 18:40:59 +01:00
$plainText ,
text ,
2017-02-13 11:35:04 +01:00
format = 'plaintext' ,
isDisplayed = false ,
isChanged = true ; // by default true as nothing was parsed yet
2012-04-23 16:30:02 +02:00
2015-09-05 17:12:11 +02:00
/ * *
2017-02-13 11:35:04 +01:00
* apply the set format on paste and displays it
2017-01-14 15:29:12 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name PasteViewer . parsePaste
2017-02-13 11:35:04 +01:00
* @ private
2017-01-14 15:29:12 +01:00
* @ function
2015-09-05 17:12:11 +02:00
* /
2017-02-13 11:35:04 +01:00
function parsePaste ( )
2015-09-05 17:12:11 +02:00
{
2017-02-13 11:35:04 +01:00
// skip parsing if no text is given
if ( text === '' ) {
return ;
2017-02-12 18:08:08 +01:00
}
2016-08-09 14:46:32 +02:00
2018-01-03 21:18:33 +01:00
// escape HTML entities, link URLs, sanitize
2018-12-29 18:40:59 +01:00
const escapedLinkedText = Helper . urls2links (
2018-01-03 21:18:33 +01:00
$ ( '<div />' ) . text ( text ) . html ( )
2018-12-29 18:40:59 +01:00
) ,
sanitizedLinkedText = DOMPurify . sanitize ( escapedLinkedText ) ;
2017-11-22 22:27:38 +01:00
$plainText . html ( sanitizedLinkedText ) ;
$prettyPrint . html ( sanitizedLinkedText ) ;
2016-08-09 14:46:32 +02:00
2017-02-13 11:35:04 +01:00
switch ( format ) {
case 'markdown' :
2018-12-29 18:40:59 +01:00
const converter = new showdown . Converter ( {
2017-02-13 11:35:04 +01:00
strikethrough : true ,
tables : true ,
2018-07-01 20:22:21 +02:00
tablesHeaderId : true ,
simplifiedAutoLink : true ,
excludeTrailingPunctuationFromURLs : true
2017-02-13 11:35:04 +01:00
} ) ;
2017-11-21 21:22:51 +01:00
// let showdown convert the HTML and sanitize HTML *afterwards*!
2017-02-14 22:21:55 +01:00
$plainText . html (
2018-12-29 18:40:59 +01:00
DOMPurify . sanitize (
converter . makeHtml ( text )
)
2017-02-13 11:35:04 +01:00
) ;
// add table classes from bootstrap css
2017-02-14 22:21:55 +01:00
$plainText . find ( 'table' ) . addClass ( 'table-condensed table-bordered' ) ;
2017-02-13 11:35:04 +01:00
break ;
case 'syntaxhighlighting' :
2017-11-22 22:27:38 +01:00
// yes, this is really needed to initialize the environment
2017-02-13 11:35:04 +01:00
if ( typeof prettyPrint === 'function' )
{
prettyPrint ( ) ;
}
2016-11-13 18:12:10 +01:00
2017-02-13 11:35:04 +01:00
$prettyPrint . html (
2017-11-22 22:27:38 +01:00
DOMPurify . sanitize (
2018-01-03 21:18:33 +01:00
prettyPrintOne ( escapedLinkedText , null , true )
2017-02-13 11:35:04 +01:00
)
) ;
// fall through, as the rest is the same
default : // = 'plaintext'
$prettyPrint . css ( 'white-space' , 'pre-wrap' ) ;
$prettyPrint . css ( 'word-break' , 'normal' ) ;
$prettyPrint . removeClass ( 'prettyprint' ) ;
}
}
2015-09-05 17:12:11 +02:00
/ * *
2017-02-13 11:35:04 +01:00
* displays the paste
2017-01-14 15:29:12 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name PasteViewer . showPaste
2017-02-13 11:35:04 +01:00
* @ private
2017-01-14 15:29:12 +01:00
* @ function
2015-09-05 17:12:11 +02:00
* /
2017-02-13 11:35:04 +01:00
function showPaste ( )
2015-09-05 17:12:11 +02:00
{
2017-02-13 11:35:04 +01:00
// instead of "nothing" better display a placeholder
if ( text === '' ) {
2018-01-06 09:26:10 +01:00
$placeholder . removeClass ( 'hidden' ) ;
2015-09-05 17:12:11 +02:00
return ;
}
2017-02-13 11:35:04 +01:00
// otherwise hide the placeholder
2018-01-06 09:26:10 +01:00
$placeholder . addClass ( 'hidden' ) ;
2012-04-23 16:30:02 +02:00
2017-02-13 11:35:04 +01:00
switch ( format ) {
case 'markdown' :
2017-02-14 22:21:55 +01:00
$plainText . removeClass ( 'hidden' ) ;
2017-02-13 11:35:04 +01:00
$prettyMessage . addClass ( 'hidden' ) ;
break ;
default :
2017-02-14 22:21:55 +01:00
$plainText . addClass ( 'hidden' ) ;
2017-02-13 11:35:04 +01:00
$prettyMessage . removeClass ( 'hidden' ) ;
break ;
2015-09-05 17:12:11 +02:00
}
2017-02-13 11:35:04 +01:00
}
2017-02-08 20:12:22 +01:00
/ * *
2017-02-13 11:35:04 +01:00
* sets the format in which the text is shown
2017-02-08 20:12:22 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name PasteViewer . setFormat
2017-02-08 20:12:22 +01:00
* @ function
2017-11-21 10:53:33 +01:00
* @ param { string } newFormat the new format
2017-02-08 20:12:22 +01:00
* /
2017-02-13 11:35:04 +01:00
me . setFormat = function ( newFormat )
2017-02-08 20:12:22 +01:00
{
2017-02-15 22:59:55 +01:00
// skip if there is no update
if ( format === newFormat ) {
return ;
2015-09-05 17:12:11 +02:00
}
2017-02-15 22:59:55 +01:00
2017-11-21 10:53:33 +01:00
// needs to update display too, if we switch from or to Markdown
2017-02-15 22:59:55 +01:00
if ( format === 'markdown' || newFormat === 'markdown' ) {
isDisplayed = false ;
2015-09-05 17:12:11 +02:00
}
2017-02-15 22:59:55 +01:00
format = newFormat ;
isChanged = true ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 20:11:04 +01:00
2017-02-08 20:12:22 +01:00
/ * *
2017-02-13 11:35:04 +01:00
* returns the current format
2017-02-08 20:12:22 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name PasteViewer . getFormat
2017-02-08 20:12:22 +01:00
* @ function
2017-02-13 11:35:04 +01:00
* @ return { string }
2017-02-08 20:12:22 +01:00
* /
2017-02-13 11:35:04 +01:00
me . getFormat = function ( )
2017-02-08 20:12:22 +01:00
{
2017-02-13 11:35:04 +01:00
return format ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 20:11:04 +01:00
2017-02-14 22:21:55 +01:00
/ * *
* returns whether the current view is pretty printed
*
* @ name PasteViewer . isPrettyPrinted
* @ function
* @ return { bool }
* /
me . isPrettyPrinted = function ( )
{
return $prettyPrint . hasClass ( 'prettyprinted' ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
2017-02-08 20:12:22 +01:00
/ * *
2017-02-13 11:35:04 +01:00
* sets the text to show
2017-02-08 20:12:22 +01:00
*
2017-02-15 22:59:55 +01:00
* @ name PasteViewer . setText
2017-02-08 20:12:22 +01:00
* @ function
2017-02-14 22:21:55 +01:00
* @ param { string } newText the text to show
2017-02-08 20:12:22 +01:00
* /
2017-02-13 11:35:04 +01:00
me . setText = function ( newText )
2017-02-08 20:12:22 +01:00
{
2017-02-13 11:35:04 +01:00
if ( text !== newText ) {
text = newText ;
isChanged = true ;
}
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 20:11:04 +01:00
2017-02-15 22:59:55 +01:00
/ * *
* gets the current cached text
*
* @ name PasteViewer . getText
* @ function
* @ return { string }
* /
2017-03-25 00:58:59 +01:00
me . getText = function ( )
2017-02-15 22:59:55 +01:00
{
return text ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-15 22:59:55 +01:00
2017-02-08 20:12:22 +01:00
/ * *
2017-02-13 11:35:04 +01:00
* show / update the parsed text ( preview )
2017-02-08 20:12:22 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name PasteViewer . run
2017-02-08 20:12:22 +01:00
* @ function
* /
2017-02-14 22:21:55 +01:00
me . run = function ( )
2017-02-08 20:12:22 +01:00
{
2017-02-13 11:35:04 +01:00
if ( isChanged ) {
parsePaste ( ) ;
isChanged = false ;
}
if ( ! isDisplayed ) {
showPaste ( ) ;
isDisplayed = true ;
}
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 20:11:04 +01:00
2017-02-08 20:12:22 +01:00
/ * *
2017-02-13 11:35:04 +01:00
* hide parsed text ( preview )
2017-02-08 20:12:22 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name PasteViewer . hide
2017-02-08 20:12:22 +01:00
* @ function
* /
2017-02-13 11:35:04 +01:00
me . hide = function ( )
2017-02-08 20:12:22 +01:00
{
2017-02-13 11:35:04 +01:00
if ( ! isDisplayed ) {
2018-12-25 17:34:39 +01:00
return ;
2017-02-13 11:35:04 +01:00
}
2017-02-14 22:21:55 +01:00
$plainText . addClass ( 'hidden' ) ;
2017-02-13 11:35:04 +01:00
$prettyMessage . addClass ( 'hidden' ) ;
$placeholder . addClass ( 'hidden' ) ;
2017-04-02 18:58:11 +02:00
AttachmentViewer . hideAttachmentPreview ( ) ;
2017-02-13 11:35:04 +01:00
isDisplayed = false ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 20:11:04 +01:00
2017-02-08 20:12:22 +01:00
/ * *
2017-02-12 18:08:08 +01:00
* init status manager
2017-02-08 20:12:22 +01:00
*
2017-02-12 18:08:08 +01:00
* preloads jQuery elements
*
2017-03-13 20:24:18 +01:00
* @ name PasteViewer . init
2017-02-08 20:12:22 +01:00
* @ function
* /
2017-02-12 18:08:08 +01:00
me . init = function ( )
2017-02-08 20:12:22 +01:00
{
2017-02-13 11:35:04 +01:00
$placeholder = $ ( '#placeholder' ) ;
2017-02-17 22:46:18 +01:00
$plainText = $ ( '#plaintext' ) ;
2017-02-13 11:35:04 +01:00
$prettyMessage = $ ( '#prettymessage' ) ;
$prettyPrint = $ ( '#prettyprint' ) ;
2017-02-12 18:08:08 +01:00
2017-02-13 11:35:04 +01:00
// check requirements
if ( typeof prettyPrintOne !== 'function' ) {
2017-02-15 22:59:55 +01:00
Alert . showError ( [
'The library %s is not available. This may cause display errors.' ,
'pretty print'
] ) ;
2017-02-13 11:35:04 +01:00
}
if ( typeof showdown !== 'object' ) {
2017-02-15 22:59:55 +01:00
Alert . showError ( [
'The library %s is not available. This may cause display errors.' ,
'showdown'
] ) ;
2017-02-13 11:35:04 +01:00
}
// get default option from template/HTML or fall back to set value
2017-02-15 22:59:55 +01:00
format = Model . getFormatDefault ( ) || format ;
2017-11-21 10:53:33 +01:00
text = '' ;
isDisplayed = false ;
isChanged = true ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
return me ;
2017-03-25 00:58:59 +01:00
} ) ( ) ;
2017-02-14 22:21:55 +01:00
/ * *
* ( view ) Show attachment and preview if possible
*
2017-03-13 20:24:18 +01:00
* @ name AttachmentViewer
2017-02-14 22:21:55 +01:00
* @ class
* /
2018-12-29 18:40:59 +01:00
const AttachmentViewer = ( function ( ) {
const me = { } ;
2017-02-14 22:21:55 +01:00
2018-12-29 18:40:59 +01:00
let $attachmentLink ,
$attachmentPreview ,
$attachment ,
attachmentData ,
file ,
$fileInput ,
$dragAndDropFileName ,
attachmentHasPreview = false ;
2017-02-14 22:21:55 +01:00
/ * *
* sets the attachment but does not yet show it
*
* @ name AttachmentViewer . setAttachment
* @ function
* @ param { string } attachmentData - base64 - encoded data of file
* @ param { string } fileName - optional , file name
* /
me . setAttachment = function ( attachmentData , fileName )
{
2018-04-09 00:36:55 +02:00
// IE does not support setting a data URI on an a element
// Convert dataURI to a Blob and use msSaveBlob to download
if ( window . Blob && navigator . msSaveBlob ) {
2018-04-09 17:57:58 +02:00
$attachmentLink . off ( 'click' ) . on ( 'click' , function ( ) {
2018-04-09 06:44:37 +02:00
// data URI format: data:[<mediaType>][;base64],<data>
2018-04-07 08:29:36 +02:00
2018-04-09 06:44:37 +02:00
// position in data URI string of where data begins
2018-12-29 18:40:59 +01:00
const base64Start = attachmentData . indexOf ( ',' ) + 1 ;
2018-04-09 06:44:37 +02:00
// position in data URI string of where mediaType ends
2018-12-29 18:40:59 +01:00
const mediaTypeEnd = attachmentData . indexOf ( ';' ) ;
2018-04-09 00:36:55 +02:00
2018-04-09 06:44:37 +02:00
// extract mediaType
2018-12-29 18:40:59 +01:00
const mediaType = attachmentData . substring ( 5 , mediaTypeEnd ) ;
2018-04-09 06:44:37 +02:00
// extract data and convert to binary
2018-12-29 18:40:59 +01:00
const decodedData = atob ( attachmentData . substring ( base64Start ) ) ;
2018-04-07 08:29:36 +02:00
2018-04-09 06:44:37 +02:00
// Transform into a Blob
2018-12-29 18:40:59 +01:00
const buf = new Uint8Array ( decodedData . length ) ;
for ( let i = 0 ; i < decodedData . length ; ++ i ) {
2018-04-09 06:44:37 +02:00
buf [ i ] = decodedData . charCodeAt ( i ) ;
}
2018-12-29 18:40:59 +01:00
const blob = new window . Blob ( [ buf ] , { type : mediaType } ) ;
2018-04-09 06:44:37 +02:00
navigator . msSaveBlob ( blob , fileName ) ;
2018-04-09 00:36:55 +02:00
} ) ;
2018-04-07 08:29:36 +02:00
} else {
$attachmentLink . attr ( 'href' , attachmentData ) ;
}
2017-02-14 22:21:55 +01:00
if ( typeof fileName !== 'undefined' ) {
$attachmentLink . attr ( 'download' , fileName ) ;
}
2017-05-15 22:05:52 +02:00
me . handleAttachmentPreview ( $attachmentPreview , attachmentData ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
/ * *
* displays the attachment
*
* @ name AttachmentViewer . showAttachment
* @ function
* /
me . showAttachment = function ( )
{
$attachment . removeClass ( 'hidden' ) ;
if ( attachmentHasPreview ) {
$attachmentPreview . removeClass ( 'hidden' ) ;
}
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
/ * *
2017-02-15 22:59:55 +01:00
* removes the attachment
*
2017-08-12 13:26:43 +02:00
* This automatically hides the attachment containers too , to
2017-02-15 22:59:55 +01:00
* prevent an inconsistent display .
2017-02-14 22:21:55 +01:00
*
2017-02-15 22:59:55 +01:00
* @ name AttachmentViewer . removeAttachment
2017-02-14 22:21:55 +01:00
* @ function
* /
me . removeAttachment = function ( )
{
2017-05-15 22:05:52 +02:00
if ( ! $attachment . length ) {
2017-05-13 21:27:41 +02:00
return ;
}
2017-02-15 22:59:55 +01:00
me . hideAttachment ( ) ;
me . hideAttachmentPreview ( ) ;
2017-08-12 13:26:43 +02:00
$attachmentLink . removeAttr ( 'href' ) ;
$attachmentLink . removeAttr ( 'download' ) ;
2018-04-09 17:57:58 +02:00
$attachmentLink . off ( 'click' ) ;
2017-02-15 22:59:55 +01:00
$attachmentPreview . html ( '' ) ;
2017-04-02 18:58:11 +02:00
2018-05-21 19:32:01 +02:00
AttachmentViewer . removeAttachmentData ( ) ;
} ;
/ * *
* removes the attachment data
*
* This removes the data , which would be uploaded otherwise .
*
* @ name AttachmentViewer . removeAttachmentData
* @ function
* /
me . removeAttachmentData = function ( )
{
2017-05-15 22:05:52 +02:00
file = undefined ;
attachmentData = undefined ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
2018-05-22 00:41:02 +02:00
/ * *
* Cleares the drag & drop data .
*
* @ name AttachmentViewer . clearDragAndDrop
* @ function
* /
me . clearDragAndDrop = function ( )
{
$dragAndDropFileName . text ( '' ) ;
} ;
2017-02-15 22:59:55 +01:00
/ * *
* hides the attachment
*
2017-03-13 20:24:18 +01:00
* This will not hide the preview ( see AttachmentViewer . hideAttachmentPreview
* for that ) nor will it hide the attachment link if it was moved somewhere
* else ( see AttachmentViewer . moveAttachmentTo ) .
2017-02-15 22:59:55 +01:00
*
* @ name AttachmentViewer . hideAttachment
* @ function
* /
me . hideAttachment = function ( )
{
$attachment . addClass ( 'hidden' ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
2017-02-15 22:59:55 +01:00
/ * *
* hides the attachment preview
*
* @ name AttachmentViewer . hideAttachmentPreview
* @ function
* /
me . hideAttachmentPreview = function ( )
{
2017-05-15 22:05:52 +02:00
if ( $attachmentPreview ) {
$attachmentPreview . addClass ( 'hidden' ) ;
2017-05-13 21:27:41 +02:00
}
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
/ * *
2018-05-21 19:32:01 +02:00
* checks if there is an attachment displayed
2017-02-14 22:21:55 +01:00
*
* @ name AttachmentViewer . hasAttachment
* @ function
* /
me . hasAttachment = function ( )
{
2017-05-15 22:05:52 +02:00
if ( ! $attachment . length ) {
2017-05-13 21:27:41 +02:00
return false ;
}
2018-12-29 18:40:59 +01:00
const link = $attachmentLink . prop ( 'href' ) ;
2017-05-15 22:05:52 +02:00
return ( typeof link !== 'undefined' && link !== '' ) ;
} ;
/ * *
2018-05-21 19:32:01 +02:00
* checks if there is attachment data ( for preview ! ) available
*
* It returns true , when there is data that needs to be encrypted .
2017-05-15 22:05:52 +02:00
*
* @ name AttachmentViewer . hasAttachmentData
* @ function
* /
me . hasAttachmentData = function ( )
{
if ( $attachment . length ) {
return true ;
}
return false ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
/ * *
* return the attachment
*
* @ name AttachmentViewer . getAttachment
* @ function
* @ returns { array }
* /
me . getAttachment = function ( )
{
return [
2017-02-15 22:59:55 +01:00
$attachmentLink . prop ( 'href' ) ,
$attachmentLink . prop ( 'download' )
2017-02-14 22:21:55 +01:00
] ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
2017-02-15 22:59:55 +01:00
/ * *
* moves the attachment link to another element
*
* It is advisable to hide the attachment afterwards ( AttachmentViewer . hideAttachment )
*
2017-03-13 20:24:18 +01:00
* @ name AttachmentViewer . moveAttachmentTo
2017-02-15 22:59:55 +01:00
* @ function
* @ param { jQuery } $element - the wrapper / container element where this should be moved to
* @ param { string } label - the text to show ( % s will be replaced with the file name ) , will automatically be translated
* /
me . moveAttachmentTo = function ( $element , label )
{
// move elemement to new place
$attachmentLink . appendTo ( $element ) ;
// update text
I18n . _ ( $attachmentLink , label , $attachmentLink . attr ( 'download' ) ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-15 22:59:55 +01:00
2017-02-14 22:21:55 +01:00
/ * *
2018-07-22 10:24:39 +02:00
* read file data as data URL using the FileReader API
2017-02-14 22:21:55 +01:00
*
2018-04-29 11:57:03 +02:00
* @ name AttachmentViewer . readFileData
2018-05-22 00:43:24 +02:00
* @ private
2017-02-14 22:21:55 +01:00
* @ function
2018-07-22 10:24:39 +02:00
* @ param { object } loadedFile ( optional ) loaded file object
2018-04-29 11:57:03 +02:00
* @ see { @ link https : //developer.mozilla.org/en-US/docs/Web/API/FileReader#readAsDataURL()}
2017-02-14 22:21:55 +01:00
* /
2018-05-22 10:19:53 +02:00
function readFileData ( loadedFile ) {
2017-05-13 19:46:22 +02:00
if ( typeof FileReader === 'undefined' ) {
// revert loading status…
2017-05-15 22:05:52 +02:00
me . hideAttachment ( ) ;
me . hideAttachmentPreview ( ) ;
Alert . showError ( 'Your browser does not support uploading encrypted files. Please use a newer browser.' ) ;
2017-05-13 19:46:22 +02:00
return ;
}
2017-02-14 22:21:55 +01:00
2018-12-29 18:40:59 +01:00
const fileReader = new FileReader ( ) ;
2017-05-20 16:04:10 +02:00
if ( loadedFile === undefined ) {
loadedFile = $fileInput [ 0 ] . files [ 0 ] ;
$dragAndDropFileName . text ( '' ) ;
2017-05-13 19:46:22 +02:00
} else {
2017-05-20 16:04:10 +02:00
$dragAndDropFileName . text ( loadedFile . name ) ;
2017-05-13 19:46:22 +02:00
}
2017-02-14 22:21:55 +01:00
2017-05-20 16:04:10 +02:00
file = loadedFile ;
2017-02-14 22:21:55 +01:00
2017-05-13 19:46:22 +02:00
fileReader . onload = function ( event ) {
2018-12-29 18:40:59 +01:00
const dataURL = event . target . result ;
2017-05-15 22:05:52 +02:00
attachmentData = dataURL ;
2017-05-13 19:46:22 +02:00
if ( Editor . isPreview ( ) ) {
2017-05-15 22:05:52 +02:00
me . handleAttachmentPreview ( $attachmentPreview , dataURL ) ;
2018-04-29 11:57:03 +02:00
$attachmentPreview . removeClass ( 'hidden' ) ;
2017-05-13 19:46:22 +02:00
}
} ;
2017-05-20 16:04:10 +02:00
fileReader . readAsDataURL ( loadedFile ) ;
2018-05-22 10:19:53 +02:00
}
2017-05-13 19:46:22 +02:00
/ * *
2018-04-29 11:57:03 +02:00
* handle the preview of files that can either be an image , video , audio or pdf element
*
* @ name AttachmentViewer . handleAttachmentPreview
* @ function
2018-07-22 10:24:39 +02:00
* @ argument { jQuery } $targetElement element where the preview should be appended
* @ argument { string } file as a data URL
2017-05-13 19:46:22 +02:00
* /
2017-05-15 22:05:52 +02:00
me . handleAttachmentPreview = function ( $targetElement , data ) {
2017-05-13 19:46:22 +02:00
if ( data ) {
2018-04-29 11:57:03 +02:00
// source: https://developer.mozilla.org/en-US/docs/Web/API/FileReader#readAsDataURL()
2018-12-29 18:40:59 +01:00
const mimeType = data . slice (
2018-04-29 11:57:03 +02:00
data . indexOf ( 'data:' ) + 5 ,
data . indexOf ( ';base64,' )
) ;
2017-05-13 19:46:22 +02:00
2018-04-29 11:57:03 +02:00
attachmentHasPreview = true ;
2017-05-13 19:46:22 +02:00
if ( mimeType . match ( /image\//i ) ) {
2018-04-29 11:57:03 +02:00
$targetElement . html (
$ ( document . createElement ( 'img' ) )
. attr ( 'src' , data )
. attr ( 'class' , 'img-thumbnail' )
) ;
2017-05-13 19:46:22 +02:00
} else if ( mimeType . match ( /video\//i ) ) {
2018-04-29 11:57:03 +02:00
$targetElement . html (
$ ( document . createElement ( 'video' ) )
. attr ( 'controls' , 'true' )
. attr ( 'autoplay' , 'true' )
. attr ( 'class' , 'img-thumbnail' )
. append ( $ ( document . createElement ( 'source' ) )
. attr ( 'type' , mimeType )
. attr ( 'src' , data ) )
) ;
2017-05-13 19:46:22 +02:00
} else if ( mimeType . match ( /audio\//i ) ) {
2018-04-29 11:57:03 +02:00
$targetElement . html (
$ ( document . createElement ( 'audio' ) )
. attr ( 'controls' , 'true' )
. attr ( 'autoplay' , 'true' )
. append ( $ ( document . createElement ( 'source' ) )
. attr ( 'type' , mimeType )
. attr ( 'src' , data ) )
) ;
2017-05-13 19:46:22 +02:00
} else if ( mimeType . match ( /\/pdf/i ) ) {
2018-04-29 11:57:03 +02:00
// PDFs are only displayed if the filesize is smaller than about 1MB (after base64 encoding).
// Bigger filesizes currently cause crashes in various browsers.
// See also: https://code.google.com/p/chromium/issues/detail?id=69227
// Firefox crashes with files that are about 1.5MB
// The performance with 1MB files is bearable
if ( data . length > 1398488 ) {
2018-05-22 00:41:02 +02:00
Alert . showError ( 'File too large, to display a preview. Please download the attachment.' ) ; //TODO: is this error really neccessary?
2018-04-29 11:57:03 +02:00
return ;
}
2017-05-13 19:46:22 +02:00
2018-04-29 11:57:03 +02:00
// Fallback for browsers, that don't support the vh unit
2018-12-29 18:40:59 +01:00
const clientHeight = $ ( window ) . height ( ) ;
2017-05-13 19:46:22 +02:00
2018-04-29 11:57:03 +02:00
$targetElement . html (
$ ( document . createElement ( 'embed' ) )
. attr ( 'src' , data )
. attr ( 'type' , 'application/pdf' )
. attr ( 'class' , 'pdfPreview' )
. css ( 'height' , clientHeight )
2017-05-13 19:46:22 +02:00
) ;
2018-04-29 11:57:03 +02:00
} else {
attachmentHasPreview = false ;
}
2017-05-15 22:05:52 +02:00
}
2017-05-13 19:46:22 +02:00
} ;
/ * *
2018-04-29 11:57:03 +02:00
* attaches the file attachment drag & drop handler to the page
*
* @ name AttachmentViewer . addDragDropHandler
2018-05-22 00:43:24 +02:00
* @ private
2018-04-29 11:57:03 +02:00
* @ function
2017-05-13 19:46:22 +02:00
* /
2018-05-22 10:19:53 +02:00
function addDragDropHandler ( ) {
2017-05-15 22:05:52 +02:00
if ( typeof $fileInput === 'undefined' || $fileInput . length === 0 ) {
2017-05-13 19:46:22 +02:00
return ;
}
2018-12-29 18:40:59 +01:00
const ignoreDragDrop = function ( event ) {
2017-05-13 19:46:22 +02:00
event . stopPropagation ( ) ;
event . preventDefault ( ) ;
} ;
2018-12-29 18:40:59 +01:00
const drop = function ( event ) {
const evt = event . originalEvent ;
2017-05-15 22:05:52 +02:00
evt . stopPropagation ( ) ;
evt . preventDefault ( ) ;
2017-05-13 19:46:22 +02:00
2017-05-15 22:05:52 +02:00
if ( $fileInput ) {
2018-12-29 18:40:59 +01:00
const file = evt . dataTransfer . files [ 0 ] ;
2017-05-13 19:46:22 +02:00
//Clear the file input:
2017-05-15 22:05:52 +02:00
$fileInput . wrap ( '<form>' ) . closest ( 'form' ) . get ( 0 ) . reset ( ) ;
$fileInput . unwrap ( ) ;
2017-05-13 19:46:22 +02:00
//Only works in Chrome:
//fileInput[0].files = e.dataTransfer.files;
2018-05-22 00:43:24 +02:00
readFileData ( file ) ;
2017-05-13 19:46:22 +02:00
}
} ;
2017-05-15 22:05:52 +02:00
$ ( document ) . on ( 'drop' , drop ) ;
$ ( document ) . on ( 'dragenter' , ignoreDragDrop ) ;
$ ( document ) . on ( 'dragover' , ignoreDragDrop ) ;
2018-05-22 00:41:02 +02:00
$fileInput . on ( 'change' , function ( ) {
2018-05-22 00:43:24 +02:00
readFileData ( ) ;
2017-05-13 19:46:22 +02:00
} ) ;
2018-05-22 10:19:53 +02:00
}
2017-05-13 19:46:22 +02:00
2017-05-13 21:43:32 +02:00
/ * *
2018-04-29 11:57:03 +02:00
* attaches the clipboard attachment handler to the page
*
* @ name AttachmentViewer . addClipboardEventHandler
2018-05-22 00:43:24 +02:00
* @ private
2018-04-29 11:57:03 +02:00
* @ function
2017-05-13 21:43:32 +02:00
* /
2018-05-22 10:19:53 +02:00
function addClipboardEventHandler ( ) {
$ ( document ) . on ( 'paste' , function ( event ) {
2018-12-29 18:40:59 +01:00
const items = ( event . clipboardData || event . originalEvent . clipboardData ) . items ;
for ( let i = 0 ; i < items . length ; ++ i ) {
2018-08-04 22:30:01 +02:00
if ( items [ i ] . kind === 'file' ) {
//Clear the file input:
$fileInput . wrap ( '<form>' ) . closest ( 'form' ) . get ( 0 ) . reset ( ) ;
$fileInput . unwrap ( ) ;
readFileData ( items [ i ] . getAsFile ( ) ) ;
2018-05-22 10:19:53 +02:00
}
}
} ) ;
}
2017-05-15 22:05:52 +02:00
2018-04-29 11:57:03 +02:00
/ * *
* getter for attachment data
*
* @ name AttachmentViewer . getAttachmentData
* @ function
* @ return { jQuery }
* /
2017-05-15 22:05:52 +02:00
me . getAttachmentData = function ( ) {
return attachmentData ;
} ;
2018-04-29 11:57:03 +02:00
/ * *
* getter for attachment link
*
* @ name AttachmentViewer . getAttachmentLink
* @ function
* @ return { jQuery }
* /
2017-05-15 22:05:52 +02:00
me . getAttachmentLink = function ( ) {
return $attachmentLink ;
} ;
2018-04-29 11:57:03 +02:00
/ * *
* getter for attachment preview
*
* @ name AttachmentViewer . getAttachmentPreview
* @ function
* @ return { jQuery }
* /
2017-05-15 22:05:52 +02:00
me . getAttachmentPreview = function ( ) {
return $attachmentPreview ;
} ;
2018-04-29 11:57:03 +02:00
/ * *
* getter for file data , returns the file contents
*
* @ name AttachmentViewer . getFile
* @ function
* @ return { string }
* /
2017-05-15 22:05:52 +02:00
me . getFile = function ( ) {
return file ;
2017-05-13 21:43:32 +02:00
} ;
2017-02-14 22:21:55 +01:00
/ * *
* initiate
*
* preloads jQuery elements
*
* @ name AttachmentViewer . init
* @ function
* /
me . init = function ( )
{
$attachment = $ ( '#attachment' ) ;
2017-05-15 22:05:52 +02:00
if ( $attachment . length ) {
$attachmentLink = $ ( '#attachment a' ) ;
$attachmentPreview = $ ( '#attachmentPreview' ) ;
$dragAndDropFileName = $ ( '#dragAndDropFileName' ) ;
2017-05-13 19:46:22 +02:00
2017-05-15 22:05:52 +02:00
$fileInput = $ ( '#file' ) ;
2018-05-22 00:43:24 +02:00
addDragDropHandler ( ) ;
addClipboardEventHandler ( ) ;
2017-05-13 21:27:41 +02:00
}
2017-02-25 09:35:55 +01:00
}
2017-02-14 22:21:55 +01:00
return me ;
2017-11-28 06:38:10 +01:00
} ) ( ) ;
2017-02-14 22:21:55 +01:00
/ * *
* ( view ) Shows discussion thread and handles replies
*
2017-03-13 20:24:18 +01:00
* @ name DiscussionViewer
2017-02-14 22:21:55 +01:00
* @ class
* /
2018-12-29 18:40:59 +01:00
const DiscussionViewer = ( function ( ) {
const me = { } ;
2017-02-14 22:21:55 +01:00
2018-12-29 18:40:59 +01:00
let $commentTail ,
2017-02-17 20:46:10 +01:00
$discussion ,
$reply ,
$replyMessage ,
$replyNickname ,
2017-02-17 22:46:18 +01:00
$replyStatus ,
2018-12-29 18:40:59 +01:00
$commentContainer ,
replyCommentId ;
2017-02-14 22:21:55 +01:00
/ * *
2017-02-17 20:46:10 +01:00
* initializes the templates
2017-02-14 22:21:55 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name DiscussionViewer . initTemplates
2017-02-17 20:46:10 +01:00
* @ private
2017-02-14 22:21:55 +01:00
* @ function
* /
2017-02-17 20:46:10 +01:00
function initTemplates ( )
2017-02-14 22:21:55 +01:00
{
2017-02-17 20:46:10 +01:00
$reply = Model . getTemplate ( 'reply' ) ;
$replyMessage = $reply . find ( '#replymessage' ) ;
$replyNickname = $reply . find ( '#nickname' ) ;
$replyStatus = $reply . find ( '#replystatus' ) ;
// cache jQuery elements
$commentTail = Model . getTemplate ( 'commenttail' ) ;
}
2017-02-14 22:21:55 +01:00
2017-03-13 20:24:18 +01:00
/ * *
* open the comment entry when clicking the "Reply" button of a comment
*
* @ name DiscussionViewer . openReply
* @ private
* @ function
* @ param { Event } event
* /
function openReply ( event )
{
2018-12-29 18:40:59 +01:00
const $source = $ ( event . target ) ;
2017-03-13 20:24:18 +01:00
// clear input
$replyMessage . val ( '' ) ;
$replyNickname . val ( '' ) ;
// get comment id from source element
replyCommentId = $source . parent ( ) . prop ( 'id' ) . split ( '_' ) [ 1 ] ;
// move to correct position
$source . after ( $reply ) ;
// show
$reply . removeClass ( 'hidden' ) ;
$replyMessage . focus ( ) ;
event . preventDefault ( ) ;
}
2017-02-14 22:21:55 +01:00
/ * *
2017-02-17 20:46:10 +01:00
* custom handler for displaying notifications in own status message area
2017-02-14 22:21:55 +01:00
*
2017-02-17 20:46:10 +01:00
* @ name DiscussionViewer . handleNotification
2017-02-14 22:21:55 +01:00
* @ function
2017-02-17 20:46:10 +01:00
* @ param { string } alertType
* @ return { bool | jQuery }
2017-02-14 22:21:55 +01:00
* /
2018-01-06 13:32:07 +01:00
me . handleNotification = function ( alertType )
2017-02-14 22:21:55 +01:00
{
2017-02-17 20:46:10 +01:00
// ignore loading messages
if ( alertType === 'loading' ) {
return false ;
}
2017-02-14 22:21:55 +01:00
2017-02-17 20:46:10 +01:00
if ( alertType === 'danger' ) {
$replyStatus . removeClass ( 'alert-info' ) ;
$replyStatus . addClass ( 'alert-danger' ) ;
$replyStatus . find ( ':first' ) . removeClass ( 'glyphicon-alert' ) ;
$replyStatus . find ( ':first' ) . addClass ( 'glyphicon-info-sign' ) ;
} else {
$replyStatus . removeClass ( 'alert-danger' ) ;
$replyStatus . addClass ( 'alert-info' ) ;
$replyStatus . find ( ':first' ) . removeClass ( 'glyphicon-info-sign' ) ;
$replyStatus . find ( ':first' ) . addClass ( 'glyphicon-alert' ) ;
}
return $replyStatus ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
2017-02-17 20:46:10 +01:00
/ * *
* adds another comment
*
* @ name DiscussionViewer . addComment
* @ function
* @ param { object } comment
* @ param { string } commentText
2018-01-02 11:42:03 +01:00
* @ param { string } nickname
2017-02-17 20:46:10 +01:00
* /
2018-01-02 11:42:03 +01:00
me . addComment = function ( comment , commentText , nickname )
2017-02-17 20:46:10 +01:00
{
if ( commentText === '' ) {
commentText = 'comment decryption failed' ;
}
// create new comment based on template
2018-12-29 18:40:59 +01:00
const $commentEntry = Model . getTemplate ( 'comment' ) ;
2017-02-17 20:46:10 +01:00
$commentEntry . prop ( 'id' , 'comment_' + comment . id ) ;
2018-12-29 18:40:59 +01:00
const $commentEntryData = $commentEntry . find ( 'div.commentdata' ) ;
2017-02-17 20:46:10 +01:00
// set & parse text
2017-11-22 22:27:38 +01:00
$commentEntryData . html (
DOMPurify . sanitize (
2018-01-01 10:25:07 +01:00
Helper . urls2links ( commentText )
2017-11-22 22:27:38 +01:00
)
) ;
2017-02-17 20:46:10 +01:00
// set nickname
2017-03-12 17:08:12 +01:00
if ( nickname . length > 0 ) {
2017-02-17 20:46:10 +01:00
$commentEntry . find ( 'span.nickname' ) . text ( nickname ) ;
} else {
2017-03-12 17:08:12 +01:00
$commentEntry . find ( 'span.nickname' ) . html ( '<i></i>' ) ;
I18n . _ ( $commentEntry . find ( 'span.nickname i' ) , 'Anonymous' ) ;
2017-02-17 20:46:10 +01:00
}
// set date
$commentEntry . find ( 'span.commentdate' )
. text ( ' (' + ( new Date ( comment . meta . postdate * 1000 ) . toLocaleString ( ) ) + ')' )
. attr ( 'title' , 'CommentID: ' + comment . id ) ;
// if an avatar is available, display it
if ( comment . meta . vizhash ) {
$commentEntry . find ( 'span.nickname' )
2017-03-12 17:08:12 +01:00
. before (
'<img src="' + comment . meta . vizhash + '" class="vizhash" /> '
) ;
$ ( document ) . on ( 'languageLoaded' , function ( ) {
$commentEntry . find ( 'img.vizhash' )
. prop ( 'title' , I18n . _ ( 'Avatar generated from IP address' ) ) ;
} ) ;
2017-02-17 20:46:10 +01:00
}
2018-01-02 11:42:03 +01:00
// starting point (default value/fallback)
2018-12-29 18:40:59 +01:00
let $place = $commentContainer ;
2018-01-02 11:42:03 +01:00
// if parent comment exists
2018-12-29 18:40:59 +01:00
const $parentComment = $ ( '#comment_' + comment . parentid ) ;
2018-01-02 11:42:03 +01:00
if ( $parentComment . length ) {
// use parent as position for new comment, so it is shifted
// to the right
$place = $parentComment ;
}
2017-02-17 20:46:10 +01:00
// finally append comment
$place . append ( $commentEntry ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-17 20:46:10 +01:00
/ * *
* finishes the discussion area after last comment
*
* @ name DiscussionViewer . finishDiscussion
* @ function
* /
me . finishDiscussion = function ( )
{
// add 'add new comment' area
$commentContainer . append ( $commentTail ) ;
// show discussions
$discussion . removeClass ( 'hidden' ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-17 20:46:10 +01:00
/ * *
* removes the old discussion and prepares everything for creating a new
* one .
*
2017-12-18 14:47:17 +01:00
* @ name DiscussionViewer . prepareNewDiscussion
2017-02-17 20:46:10 +01:00
* @ function
* /
2017-12-18 14:47:17 +01:00
me . prepareNewDiscussion = function ( )
2017-02-17 20:46:10 +01:00
{
$commentContainer . html ( '' ) ;
$discussion . addClass ( 'hidden' ) ;
// (re-)init templates
initTemplates ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-17 20:46:10 +01:00
/ * *
2017-12-18 14:47:17 +01:00
* returns the users message from the reply form
2017-02-17 20:46:10 +01:00
*
2017-12-18 14:47:17 +01:00
* @ name DiscussionViewer . getReplyMessage
2017-02-17 20:46:10 +01:00
* @ function
2017-12-18 14:47:17 +01:00
* @ return { String }
2017-02-17 20:46:10 +01:00
* /
2017-12-18 14:47:17 +01:00
me . getReplyMessage = function ( )
2017-02-17 20:46:10 +01:00
{
2017-12-18 14:47:17 +01:00
return $replyMessage . val ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-12-18 14:47:17 +01:00
/ * *
* returns the users nickname ( if any ) from the reply form
*
* @ name DiscussionViewer . getReplyNickname
* @ function
* @ return { String }
* /
me . getReplyNickname = function ( )
{
return $replyNickname . val ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-12-18 14:47:17 +01:00
/ * *
* returns the id of the parent comment the user is replying to
*
* @ name DiscussionViewer . getReplyCommentId
* @ function
* @ return { int | undefined }
* /
me . getReplyCommentId = function ( )
{
return replyCommentId ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-17 20:46:10 +01:00
/ * *
* highlights a specific comment and scrolls to it if necessary
*
* @ name DiscussionViewer . highlightComment
* @ function
* @ param { string } commentId
* @ param { bool } fadeOut - whether to fade out the comment
* /
me . highlightComment = function ( commentId , fadeOut )
{
2018-12-29 18:40:59 +01:00
const $comment = $ ( '#comment_' + commentId ) ;
2017-02-17 20:46:10 +01:00
// in case comment does not exist, cancel
if ( $comment . length === 0 ) {
return ;
}
2018-07-01 08:59:55 +02:00
$comment . addClass ( 'highlight' ) ;
2018-12-29 18:40:59 +01:00
const highlightComment = function ( ) {
2017-02-17 20:46:10 +01:00
if ( fadeOut === true ) {
setTimeout ( function ( ) {
$comment . removeClass ( 'highlight' ) ;
} , 300 ) ;
}
2018-01-06 10:57:54 +01:00
} ;
2017-02-17 20:46:10 +01:00
if ( UiHelper . isVisible ( $comment ) ) {
return highlightComment ( ) ;
}
UiHelper . scrollTo ( $comment , 100 , 'swing' , highlightComment ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
/ * *
* initiate
*
* preloads jQuery elements
*
2017-02-15 22:59:55 +01:00
* @ name DiscussionViewer . init
2017-02-14 22:21:55 +01:00
* @ function
* /
me . init = function ( )
{
2017-02-17 20:46:10 +01:00
// bind events to templates (so they are later cloned)
$ ( '#commenttailtemplate, #commenttemplate' ) . find ( 'button' ) . on ( 'click' , openReply ) ;
$ ( '#replytemplate' ) . find ( 'button' ) . on ( 'click' , PasteEncrypter . sendComment ) ;
$commentContainer = $ ( '#commentcontainer' ) ;
2017-02-14 22:21:55 +01:00
$discussion = $ ( '#discussion' ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 20:11:04 +01:00
2017-02-12 18:08:08 +01:00
return me ;
2017-12-18 14:47:17 +01:00
} ) ( ) ;
2017-02-12 18:08:08 +01:00
/ * *
* Manage top ( navigation ) bar
*
2017-03-13 20:24:18 +01:00
* @ name TopNav
* @ param { object } window
* @ param { object } document
2017-02-12 18:08:08 +01:00
* @ class
* /
2018-12-29 18:40:59 +01:00
const TopNav = ( function ( window , document ) {
const me = { } ;
2017-02-12 21:13:04 +01:00
2018-12-29 18:40:59 +01:00
let createButtonsDisplayed = false ,
viewButtonsDisplayed = false ,
$attach ,
2017-02-12 18:08:08 +01:00
$burnAfterReading ,
$burnAfterReadingOption ,
$cloneButton ,
2017-02-15 22:59:55 +01:00
$customAttachment ,
2017-02-12 18:08:08 +01:00
$expiration ,
$fileRemoveButton ,
2017-02-17 22:59:16 +01:00
$fileWrap ,
2017-02-12 18:08:08 +01:00
$formatter ,
$newButton ,
$openDiscussion ,
2017-02-17 22:46:18 +01:00
$openDiscussionOption ,
2017-02-12 21:13:04 +01:00
$password ,
2017-02-13 21:12:00 +01:00
$passwordInput ,
2017-02-12 18:08:08 +01:00
$rawTextButton ,
2017-12-25 14:59:15 +01:00
$qrCodeLink ,
2017-04-11 22:21:30 +02:00
$sendButton ,
2018-12-29 18:40:59 +01:00
$retryButton ,
pasteExpiration = '1week' ,
2017-04-11 22:21:30 +02:00
retryButtonCallback ;
2017-02-13 11:35:04 +01:00
2017-02-12 18:08:08 +01:00
/ * *
2017-02-13 11:35:04 +01:00
* set the expiration on bootstrap templates in dropdown
2017-02-12 18:08:08 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . updateExpiration
2017-03-13 20:24:18 +01:00
* @ private
2017-02-12 18:08:08 +01:00
* @ function
* @ param { Event } event
* /
2017-02-13 11:35:04 +01:00
function updateExpiration ( event )
2017-02-12 18:08:08 +01:00
{
2017-02-13 11:35:04 +01:00
// get selected option
2018-12-29 18:40:59 +01:00
const target = $ ( event . target ) ;
2017-02-13 11:35:04 +01:00
// update dropdown display and save new expiration time
2017-02-12 18:08:08 +01:00
$ ( '#pasteExpirationDisplay' ) . text ( target . text ( ) ) ;
2017-02-13 11:35:04 +01:00
pasteExpiration = target . data ( 'expiration' ) ;
event . preventDefault ( ) ;
2017-02-12 18:08:08 +01:00
}
/ * *
2017-02-13 11:35:04 +01:00
* set the format on bootstrap templates in dropdown
2017-02-12 18:08:08 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . updateFormat
2017-03-13 20:24:18 +01:00
* @ private
2017-02-12 18:08:08 +01:00
* @ function
* @ param { Event } event
* /
2017-02-13 11:35:04 +01:00
function updateFormat ( event )
2017-02-12 18:08:08 +01:00
{
2017-02-13 11:35:04 +01:00
// get selected option
2018-12-29 18:40:59 +01:00
const $target = $ ( event . target ) ;
2017-02-12 18:08:08 +01:00
2017-02-13 11:35:04 +01:00
// update dropdown display and save new format
2018-12-29 18:40:59 +01:00
const newFormat = $target . data ( 'format' ) ;
2017-02-13 11:35:04 +01:00
$ ( '#pasteFormatterDisplay' ) . text ( $target . text ( ) ) ;
2017-02-14 22:21:55 +01:00
PasteViewer . setFormat ( newFormat ) ;
2017-02-13 11:35:04 +01:00
// update preview
2017-02-14 22:21:55 +01:00
if ( Editor . isPreview ( ) ) {
PasteViewer . run ( ) ;
2017-02-12 18:08:08 +01:00
}
2017-02-13 11:35:04 +01:00
2017-02-12 18:08:08 +01:00
event . preventDefault ( ) ;
2017-02-13 11:35:04 +01:00
}
2017-02-08 20:12:22 +01:00
/ * *
2017-02-12 18:08:08 +01:00
* when "burn after reading" is checked , disable discussion
2017-02-08 20:12:22 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . changeBurnAfterReading
2017-03-13 20:24:18 +01:00
* @ private
2017-02-08 20:12:22 +01:00
* @ function
* /
2017-02-12 18:08:08 +01:00
function changeBurnAfterReading ( )
2017-02-08 20:11:04 +01:00
{
2017-02-13 11:35:04 +01:00
if ( $burnAfterReading . is ( ':checked' ) ) {
$openDiscussionOption . addClass ( 'buttondisabled' ) ;
$openDiscussion . prop ( 'checked' , false ) ;
// if button is actually disabled, force-enable it and uncheck other button
$burnAfterReadingOption . removeClass ( 'buttondisabled' ) ;
} else {
$openDiscussionOption . removeClass ( 'buttondisabled' ) ;
2017-02-08 20:12:22 +01:00
}
2017-02-12 18:08:08 +01:00
}
2015-09-05 17:12:11 +02:00
2017-02-08 20:12:22 +01:00
/ * *
2017-02-12 18:08:08 +01:00
* when discussion is checked , disable "burn after reading"
2017-02-08 20:12:22 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . changeOpenDiscussion
2017-03-13 20:24:18 +01:00
* @ private
2017-02-08 20:12:22 +01:00
* @ function
* /
2017-02-13 11:35:04 +01:00
function changeOpenDiscussion ( )
2015-09-05 17:12:11 +02:00
{
2017-02-13 11:35:04 +01:00
if ( $openDiscussion . is ( ':checked' ) ) {
2017-02-12 18:08:08 +01:00
$burnAfterReadingOption . addClass ( 'buttondisabled' ) ;
2017-02-13 11:35:04 +01:00
$burnAfterReading . prop ( 'checked' , false ) ;
// if button is actually disabled, force-enable it and uncheck other button
$openDiscussionOption . removeClass ( 'buttondisabled' ) ;
} else {
2017-02-12 18:08:08 +01:00
$burnAfterReadingOption . removeClass ( 'buttondisabled' ) ;
2017-02-08 20:12:22 +01:00
}
2017-02-12 18:08:08 +01:00
}
/ * *
* return raw text
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . rawText
2017-03-13 20:24:18 +01:00
* @ private
2017-02-12 18:08:08 +01:00
* @ function
* /
2018-01-06 13:32:07 +01:00
function rawText ( )
2017-02-12 18:08:08 +01:00
{
2017-02-17 22:26:39 +01:00
TopNav . hideAllButtons ( ) ;
2018-01-06 13:32:07 +01:00
Alert . showLoading ( 'Showing raw text…' , 'time' ) ;
2018-12-29 18:40:59 +01:00
let paste = PasteViewer . getText ( ) ;
2017-02-15 22:59:55 +01:00
// push a new state to allow back navigation with browser back button
2017-02-12 18:08:08 +01:00
history . pushState (
2017-02-15 22:59:55 +01:00
{ type : 'raw' } ,
document . title ,
// recreate paste URL
Helper . baseUri ( ) + '?' + Model . getPasteId ( ) + '#' +
2018-12-29 18:40:59 +01:00
btoa ( Model . getPasteKey ( ) )
2017-02-12 18:08:08 +01:00
) ;
2017-02-15 22:59:55 +01:00
2017-02-12 18:08:08 +01:00
// we use text/html instead of text/plain to avoid a bug when
// reloading the raw text view (it reverts to type text/html)
2018-12-29 18:40:59 +01:00
const $head = $ ( 'head' ) . children ( ) . not ( 'noscript, script, link[type="text/css"]' ) ,
newDoc = document . open ( 'text/html' , 'replace' ) ;
2017-02-17 22:26:39 +01:00
newDoc . write ( '<!DOCTYPE html><html><head>' ) ;
2018-12-29 18:40:59 +01:00
for ( let i = 0 ; i < $head . length ; ++ i ) {
2017-02-17 22:26:39 +01:00
newDoc . write ( $head [ i ] . outerHTML ) ;
}
2018-09-02 09:14:36 +02:00
newDoc . write ( '</head><body><pre>' + DOMPurify . sanitize ( $ ( '<div />' ) . text ( paste ) . html ( ) ) + '</pre></body></html>' ) ;
2017-02-12 18:08:08 +01:00
newDoc . close ( ) ;
}
/ * *
2017-02-14 22:21:55 +01:00
* saves the language in a cookie and reloads the page
2017-02-12 18:08:08 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . setLanguage
2017-03-13 20:24:18 +01:00
* @ private
2017-02-12 18:08:08 +01:00
* @ function
* @ param { Event } event
* /
function setLanguage ( event )
{
document . cookie = 'lang=' + $ ( event . target ) . data ( 'lang' ) ;
2017-02-14 22:21:55 +01:00
UiHelper . reloadHome ( ) ;
2017-02-12 18:08:08 +01:00
}
2017-02-15 22:59:55 +01:00
/ * *
* hides all messages and creates a new paste
*
2017-03-13 20:24:18 +01:00
* @ name TopNav . clickNewPaste
2017-02-15 22:59:55 +01:00
* @ private
* @ function
* /
2018-01-06 13:32:07 +01:00
function clickNewPaste ( )
2017-02-15 22:59:55 +01:00
{
Controller . hideStatusMessages ( ) ;
Controller . newPaste ( ) ;
}
2017-04-11 22:21:30 +02:00
/ * *
* retrys some callback registered before
*
* @ name TopNav . clickRetryButton
* @ private
* @ function
* @ param { Event } event
* /
function clickRetryButton ( event )
{
2017-04-11 22:36:25 +02:00
retryButtonCallback ( event ) ;
2017-04-11 22:21:30 +02:00
}
2017-02-15 22:59:55 +01:00
/ * *
* removes the existing attachment
*
2017-03-13 20:24:18 +01:00
* @ name TopNav . removeAttachment
2017-02-15 22:59:55 +01:00
* @ private
* @ function
* @ param { Event } event
* /
function removeAttachment ( event )
{
// if custom attachment is used, remove it first
if ( ! $customAttachment . hasClass ( 'hidden' ) ) {
AttachmentViewer . removeAttachment ( ) ;
$customAttachment . addClass ( 'hidden' ) ;
$fileWrap . removeClass ( 'hidden' ) ;
}
2018-05-21 19:32:01 +02:00
// in any case, remove saved attachment data
AttachmentViewer . removeAttachmentData ( ) ;
2018-05-22 00:41:02 +02:00
// hide UI for selected files
2017-02-15 22:59:55 +01:00
// our up-to-date jQuery can handle it :)
$fileWrap . find ( 'input' ) . val ( '' ) ;
2018-05-22 00:41:02 +02:00
AttachmentViewer . clearDragAndDrop ( ) ;
2017-02-15 22:59:55 +01:00
// pevent '#' from appearing in the URL
event . preventDefault ( ) ;
}
2017-02-08 20:12:22 +01:00
/ * *
2017-12-25 14:59:15 +01:00
* Shows the QR code of the current paste ( URL ) .
*
* @ name TopNav . displayQrCode
2018-03-03 07:55:27 +01:00
* @ private
2017-12-25 14:59:15 +01:00
* @ function
* /
2018-01-06 13:32:07 +01:00
function displayQrCode ( )
2017-12-25 14:59:15 +01:00
{
2018-12-29 18:40:59 +01:00
const qrCanvas = kjua ( {
2017-12-25 14:59:15 +01:00
render : 'canvas' ,
text : window . location . href
} ) ;
$ ( '#qrcode-display' ) . html ( qrCanvas ) ;
2018-01-06 09:58:19 +01:00
}
2017-12-25 14:59:15 +01:00
2017-02-08 20:12:22 +01:00
/ * *
2018-02-25 09:45:51 +01:00
* Shows all navigation elements for viewing an existing paste
2017-02-08 20:12:22 +01:00
*
2017-02-17 22:26:39 +01:00
* @ name TopNav . showViewButtons
2017-02-08 20:12:22 +01:00
* @ function
* /
2017-02-12 18:08:08 +01:00
me . showViewButtons = function ( )
2015-09-05 17:12:11 +02:00
{
2017-02-12 21:13:04 +01:00
if ( viewButtonsDisplayed ) {
return ;
}
2017-02-17 20:46:10 +01:00
$newButton . removeClass ( 'hidden' ) ;
2017-02-12 18:08:08 +01:00
$cloneButton . removeClass ( 'hidden' ) ;
$rawTextButton . removeClass ( 'hidden' ) ;
2017-12-25 14:59:15 +01:00
$qrCodeLink . removeClass ( 'hidden' ) ;
2017-02-12 21:13:04 +01:00
viewButtonsDisplayed = true ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 20:12:22 +01:00
2017-02-12 18:08:08 +01:00
/ * *
2018-02-25 09:45:51 +01:00
* Hides all navigation elements for viewing an existing paste
2017-02-12 18:08:08 +01:00
*
2017-02-17 22:26:39 +01:00
* @ name TopNav . hideViewButtons
2017-02-12 18:08:08 +01:00
* @ function
* /
me . hideViewButtons = function ( )
{
2017-02-12 21:13:04 +01:00
if ( ! viewButtonsDisplayed ) {
return ;
}
2017-02-12 18:08:08 +01:00
$cloneButton . addClass ( 'hidden' ) ;
2017-04-11 22:21:30 +02:00
$newButton . addClass ( 'hidden' ) ;
2017-02-12 18:08:08 +01:00
$rawTextButton . addClass ( 'hidden' ) ;
2017-12-25 14:59:15 +01:00
$qrCodeLink . addClass ( 'hidden' ) ;
2017-02-12 21:13:04 +01:00
viewButtonsDisplayed = false ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 20:12:22 +01:00
2017-02-17 22:26:39 +01:00
/ * *
* Hides all elements belonging to existing pastes
*
* @ name TopNav . hideAllButtons
* @ function
* /
me . hideAllButtons = function ( )
{
me . hideViewButtons ( ) ;
me . hideCreateButtons ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-17 22:26:39 +01:00
2017-02-12 18:08:08 +01:00
/ * *
* shows all elements needed when creating a new paste
*
2017-02-17 20:46:10 +01:00
* @ name TopNav . showCreateButtons
2017-02-12 18:08:08 +01:00
* @ function
* /
me . showCreateButtons = function ( )
{
2017-02-12 21:13:04 +01:00
if ( createButtonsDisplayed ) {
return ;
}
2017-04-11 22:21:30 +02:00
$attach . removeClass ( 'hidden' ) ;
$burnAfterReadingOption . removeClass ( 'hidden' ) ;
2017-02-12 18:08:08 +01:00
$expiration . removeClass ( 'hidden' ) ;
$formatter . removeClass ( 'hidden' ) ;
$newButton . removeClass ( 'hidden' ) ;
2017-04-11 22:21:30 +02:00
$openDiscussionOption . removeClass ( 'hidden' ) ;
2017-02-12 18:08:08 +01:00
$password . removeClass ( 'hidden' ) ;
2017-04-11 22:21:30 +02:00
$sendButton . removeClass ( 'hidden' ) ;
2017-02-12 21:13:04 +01:00
createButtonsDisplayed = true ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 20:12:22 +01:00
2017-02-12 18:08:08 +01:00
/ * *
* shows all elements needed when creating a new paste
*
2017-02-17 20:46:10 +01:00
* @ name TopNav . hideCreateButtons
2017-02-12 18:08:08 +01:00
* @ function
* /
me . hideCreateButtons = function ( )
{
2017-02-12 21:13:04 +01:00
if ( ! createButtonsDisplayed ) {
return ;
}
2017-02-14 22:21:55 +01:00
$newButton . addClass ( 'hidden' ) ;
2017-02-12 18:08:08 +01:00
$sendButton . addClass ( 'hidden' ) ;
$expiration . addClass ( 'hidden' ) ;
$formatter . addClass ( 'hidden' ) ;
$burnAfterReadingOption . addClass ( 'hidden' ) ;
2017-02-13 11:35:04 +01:00
$openDiscussionOption . addClass ( 'hidden' ) ;
2017-02-12 18:08:08 +01:00
$password . addClass ( 'hidden' ) ;
$attach . addClass ( 'hidden' ) ;
2017-02-12 21:13:04 +01:00
2017-02-14 22:21:55 +01:00
createButtonsDisplayed = false ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
/ * *
* only shows the "new paste" button
*
2017-02-17 20:46:10 +01:00
* @ name TopNav . showNewPasteButton
2017-02-14 22:21:55 +01:00
* @ function
* /
me . showNewPasteButton = function ( )
{
$newButton . removeClass ( 'hidden' ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
2017-04-11 22:21:30 +02:00
/ * *
* only shows the "retry" button
*
* @ name TopNav . showRetryButton
* @ function
* /
me . showRetryButton = function ( )
{
$retryButton . removeClass ( 'hidden' ) ;
}
/ * *
* hides the "retry" button
*
* @ name TopNav . hideRetryButton
* @ function
* /
me . hideRetryButton = function ( )
{
$retryButton . addClass ( 'hidden' ) ;
}
2017-02-14 22:21:55 +01:00
/ * *
* only hides the clone button
*
* @ name TopNav . hideCloneButton
* @ function
* /
me . hideCloneButton = function ( )
{
$cloneButton . addClass ( 'hidden' ) ;
2018-01-06 09:26:10 +01:00
} ;
2012-04-23 16:30:02 +02:00
2017-02-12 18:08:08 +01:00
/ * *
2017-02-14 22:21:55 +01:00
* only hides the raw text button
2017-02-12 18:08:08 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . hideRawButton
2017-02-12 18:08:08 +01:00
* @ function
* /
2017-02-14 22:21:55 +01:00
me . hideRawButton = function ( )
2017-02-12 18:08:08 +01:00
{
2017-02-14 22:21:55 +01:00
$rawTextButton . addClass ( 'hidden' ) ;
2018-01-06 09:26:10 +01:00
} ;
2012-04-23 16:30:02 +02:00
2017-02-08 20:12:22 +01:00
/ * *
2017-02-15 22:59:55 +01:00
* hides the file selector in attachment
2017-02-08 20:12:22 +01:00
*
2017-02-15 22:59:55 +01:00
* @ name TopNav . hideFileSelector
2017-02-08 20:12:22 +01:00
* @ function
* /
2017-02-15 22:59:55 +01:00
me . hideFileSelector = function ( )
2017-02-08 20:11:04 +01:00
{
2017-02-15 22:59:55 +01:00
$fileWrap . addClass ( 'hidden' ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-12 18:08:08 +01:00
2017-02-15 22:59:55 +01:00
2017-02-12 18:08:08 +01:00
/ * *
2017-02-15 22:59:55 +01:00
* shows the custom attachment
2017-02-12 18:08:08 +01:00
*
2017-02-15 22:59:55 +01:00
* @ name TopNav . showCustomAttachment
2017-02-12 18:08:08 +01:00
* @ function
* /
2017-02-15 22:59:55 +01:00
me . showCustomAttachment = function ( )
2017-02-12 18:08:08 +01:00
{
2017-02-15 22:59:55 +01:00
$customAttachment . removeClass ( 'hidden' ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
/ * *
2018-03-04 11:47:58 +01:00
* collapses the navigation bar , only if expanded
2017-02-13 21:12:00 +01:00
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . collapseBar
2017-02-13 21:12:00 +01:00
* @ function
* /
me . collapseBar = function ( )
{
2018-03-04 13:19:49 +01:00
if ( $ ( '#navbar' ) . attr ( 'aria-expanded' ) === 'true' ) {
2018-03-03 07:55:27 +01:00
$ ( '.navbar-toggle' ) . click ( ) ;
2017-02-13 21:12:00 +01:00
}
2018-01-06 09:26:10 +01:00
} ;
2017-02-08 20:12:22 +01:00
2017-02-13 11:35:04 +01:00
/ * *
* returns the currently set expiration time
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . getExpiration
2017-02-13 11:35:04 +01:00
* @ function
* @ return { int }
* /
me . getExpiration = function ( )
{
return pasteExpiration ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 11:35:04 +01:00
2017-02-13 21:12:00 +01:00
/ * *
* returns the currently selected file ( s )
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . getFileList
2017-02-13 21:12:00 +01:00
* @ function
* @ return { FileList | null }
* /
me . getFileList = function ( )
{
2018-12-29 18:40:59 +01:00
const $file = $ ( '#file' ) ;
2017-02-13 21:12:00 +01:00
// if no file given, return null
if ( ! $file . length || ! $file [ 0 ] . files . length ) {
return null ;
}
2018-01-06 13:32:07 +01:00
// ensure the selected file is still accessible
2017-02-13 21:12:00 +01:00
if ( ! ( $file [ 0 ] . files && $file [ 0 ] . files [ 0 ] ) ) {
return null ;
}
return $file [ 0 ] . files ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
/ * *
* returns the state of the burn after reading checkbox
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . getExpiration
2017-02-13 21:12:00 +01:00
* @ function
* @ return { bool }
* /
me . getBurnAfterReading = function ( )
{
return $burnAfterReading . is ( ':checked' ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
/ * *
* returns the state of the discussion checkbox
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . getOpenDiscussion
2017-02-13 21:12:00 +01:00
* @ function
* @ return { bool }
* /
me . getOpenDiscussion = function ( )
{
return $openDiscussion . is ( ':checked' ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
/ * *
* returns the entered password
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . getPassword
2017-02-13 21:12:00 +01:00
* @ function
* @ return { string }
* /
me . getPassword = function ( )
{
return $passwordInput . val ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
2017-02-15 22:59:55 +01:00
/ * *
* returns the element where custom attachments can be placed
*
* Used by AttachmentViewer when an attachment is cloned here .
*
* @ name TopNav . getCustomAttachment
* @ function
* @ return { jQuery }
* /
me . getCustomAttachment = function ( )
{
return $customAttachment ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-15 22:59:55 +01:00
2017-04-11 22:21:30 +02:00
/ * *
* Set a function to call when the retry button is clicked .
*
* @ name TopNav . setRetryCallback
* @ function
* @ param { function } callback
* /
me . setRetryCallback = function ( callback )
{
retryButtonCallback = callback ;
}
2017-02-12 18:08:08 +01:00
/ * *
* init navigation manager
*
* preloads jQuery elements
*
2017-02-14 22:21:55 +01:00
* @ name TopNav . init
2017-02-12 18:08:08 +01:00
* @ function
* /
me . init = function ( )
{
2017-02-08 20:12:22 +01:00
$attach = $ ( '#attach' ) ;
$burnAfterReading = $ ( '#burnafterreading' ) ;
$burnAfterReadingOption = $ ( '#burnafterreadingoption' ) ;
$cloneButton = $ ( '#clonebutton' ) ;
2017-02-15 22:59:55 +01:00
$customAttachment = $ ( '#customattachment' ) ;
2017-02-08 20:12:22 +01:00
$expiration = $ ( '#expiration' ) ;
$fileRemoveButton = $ ( '#fileremovebutton' ) ;
2017-02-17 22:46:18 +01:00
$fileWrap = $ ( '#filewrap' ) ;
2017-02-08 20:12:22 +01:00
$formatter = $ ( '#formatter' ) ;
$newButton = $ ( '#newbutton' ) ;
$openDiscussion = $ ( '#opendiscussion' ) ;
2017-02-17 22:46:18 +01:00
$openDiscussionOption = $ ( '#opendiscussionoption' ) ;
2017-02-12 21:13:04 +01:00
$password = $ ( '#password' ) ;
2017-02-13 21:12:00 +01:00
$passwordInput = $ ( '#passwordinput' ) ;
2017-02-08 20:12:22 +01:00
$rawTextButton = $ ( '#rawtextbutton' ) ;
2017-04-11 22:21:30 +02:00
$retryButton = $ ( '#retrybutton' ) ;
2017-02-08 20:12:22 +01:00
$sendButton = $ ( '#sendbutton' ) ;
2017-12-25 14:59:15 +01:00
$qrCodeLink = $ ( '#qrcodelink' ) ;
2013-11-01 01:15:14 +01:00
2017-02-12 18:08:08 +01:00
// bootstrap template drop down
2017-02-17 20:46:10 +01:00
$ ( '#language ul.dropdown-menu li a' ) . click ( setLanguage ) ;
2017-02-12 18:08:08 +01:00
// page template drop down
2017-02-17 20:46:10 +01:00
$ ( '#language select option' ) . click ( setLanguage ) ;
2017-02-12 18:08:08 +01:00
// bind events
$burnAfterReading . change ( changeBurnAfterReading ) ;
2017-02-13 11:35:04 +01:00
$openDiscussionOption . change ( changeOpenDiscussion ) ;
2017-02-15 22:59:55 +01:00
$newButton . click ( clickNewPaste ) ;
2017-03-13 20:24:18 +01:00
$sendButton . click ( PasteEncrypter . sendPaste ) ;
2017-02-14 22:21:55 +01:00
$cloneButton . click ( Controller . clonePaste ) ;
2017-02-12 21:13:04 +01:00
$rawTextButton . click ( rawText ) ;
2017-04-11 22:21:30 +02:00
$retryButton . click ( clickRetryButton ) ;
2017-02-15 22:59:55 +01:00
$fileRemoveButton . click ( removeAttachment ) ;
2017-12-25 14:59:15 +01:00
$qrCodeLink . click ( displayQrCode ) ;
2017-02-12 18:08:08 +01:00
2017-02-13 11:35:04 +01:00
// bootstrap template drop downs
$ ( 'ul.dropdown-menu li a' , $ ( '#expiration' ) . parent ( ) ) . click ( updateExpiration ) ;
$ ( 'ul.dropdown-menu li a' , $ ( '#formatter' ) . parent ( ) ) . click ( updateFormat ) ;
2017-02-12 18:08:08 +01:00
// initiate default state of checkboxes
changeBurnAfterReading ( ) ;
2017-02-13 11:35:04 +01:00
changeOpenDiscussion ( ) ;
// get default value from template or fall back to set value
2017-02-15 22:59:55 +01:00
pasteExpiration = Model . getExpirationDefault ( ) || pasteExpiration ;
2018-03-01 06:43:30 +01:00
createButtonsDisplayed = false ;
viewButtonsDisplayed = false ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-12 18:08:08 +01:00
return me ;
} ) ( window , document ) ;
/ * *
2017-02-13 21:12:00 +01:00
* Responsible for AJAX requests , transparently handles encryption …
2017-02-12 18:08:08 +01:00
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction
2017-02-12 18:08:08 +01:00
* @ class
* /
2018-12-29 18:40:59 +01:00
const ServerInteraction = ( function ( ) {
const me = { } ;
2017-02-12 18:08:08 +01:00
2018-12-29 18:40:59 +01:00
let successFunc = null ,
2017-02-14 22:21:55 +01:00
failureFunc = null ,
2018-12-25 17:34:39 +01:00
symmetricKey = null ,
2017-02-14 22:21:55 +01:00
url ,
data ,
2017-02-13 21:12:00 +01:00
password ;
2017-02-14 22:21:55 +01:00
/ * *
* public variable ( 'constant' ) for errors to prevent magic numbers
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . error
2017-02-14 22:21:55 +01:00
* @ readonly
* @ enum { Object }
* /
2017-02-13 21:12:00 +01:00
me . error = {
okay : 0 ,
custom : 1 ,
unknown : 2 ,
serverError : 3
} ;
2017-02-12 18:08:08 +01:00
/ * *
2017-02-13 21:12:00 +01:00
* ajaxHeaders to send in AJAX requests
2017-02-12 18:08:08 +01:00
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . ajaxHeaders
2017-02-12 18:08:08 +01:00
* @ private
2017-02-13 21:12:00 +01:00
* @ readonly
2017-02-12 18:08:08 +01:00
* @ enum { Object }
* /
2018-12-29 18:40:59 +01:00
const ajaxHeaders = { 'X-Requested-With' : 'JSONHttpRequest' } ;
2017-02-12 18:08:08 +01:00
/ * *
2017-02-13 21:12:00 +01:00
* called after successful upload
2017-02-12 18:08:08 +01:00
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . success
2017-02-17 20:46:10 +01:00
* @ private
2017-02-13 21:12:00 +01:00
* @ function
* @ param { int } status
2017-03-25 00:58:59 +01:00
* @ param { int } result - optional
2017-02-12 18:08:08 +01:00
* /
2017-02-13 21:12:00 +01:00
function success ( status , result )
{
// add useful data to result
2017-02-17 20:46:10 +01:00
result . encryptionKey = symmetricKey ;
2017-02-13 21:12:00 +01:00
result . requestData = data ;
if ( successFunc !== null ) {
successFunc ( status , result ) ;
}
}
2017-02-12 18:08:08 +01:00
/ * *
2017-02-13 21:12:00 +01:00
* called after a upload failure
2017-02-12 18:08:08 +01:00
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . fail
2017-02-17 20:46:10 +01:00
* @ private
2017-02-13 21:12:00 +01:00
* @ function
* @ param { int } status - internal code
2017-03-25 00:58:59 +01:00
* @ param { int } result - original error code
2017-02-12 18:08:08 +01:00
* /
2017-02-13 21:12:00 +01:00
function fail ( status , result )
{
if ( failureFunc !== null ) {
failureFunc ( status , result ) ;
}
}
/ * *
* actually uploads the data
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . run
2017-02-13 21:12:00 +01:00
* @ function
* /
2017-02-14 22:21:55 +01:00
me . run = function ( )
2017-02-13 21:12:00 +01:00
{
$ . ajax ( {
2018-12-25 17:34:39 +01:00
type : data ? 'POST' : 'GET' ,
2017-02-13 21:12:00 +01:00
url : url ,
data : data ,
dataType : 'json' ,
headers : ajaxHeaders ,
success : function ( result ) {
if ( result . status === 0 ) {
success ( 0 , result ) ;
} else if ( result . status === 1 ) {
fail ( 1 , result ) ;
} else {
fail ( 2 , result ) ;
}
}
} )
. fail ( function ( jqXHR , textStatus , errorThrown ) {
console . error ( textStatus , errorThrown ) ;
fail ( 3 , jqXHR ) ;
} ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
/ * *
* set success function
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . setUrl
2017-02-14 22:21:55 +01:00
* @ function
2017-03-25 00:58:59 +01:00
* @ param { function } newUrl
2017-02-14 22:21:55 +01:00
* /
me . setUrl = function ( newUrl )
{
url = newUrl ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
2017-02-17 20:46:10 +01:00
/ * *
* sets the password to use ( first value ) and optionally also the
2018-12-25 17:34:39 +01:00
* encryption key ( not recommended , it is automatically generated ) .
2017-02-17 20:46:10 +01:00
*
* Note : Call this after prepare ( ) as prepare ( ) resets these values .
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . setCryptValues
2017-02-17 20:46:10 +01:00
* @ function
* @ param { string } newPassword
* @ param { string } newKey - optional
* /
me . setCryptParameters = function ( newPassword , newKey )
{
password = newPassword ;
if ( typeof newKey !== 'undefined' ) {
symmetricKey = newKey ;
}
2018-01-06 09:26:10 +01:00
} ;
2017-02-17 20:46:10 +01:00
2017-02-14 22:21:55 +01:00
/ * *
* set success function
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . setSuccess
2017-02-13 21:12:00 +01:00
* @ function
* @ param { function } func
* /
me . setSuccess = function ( func )
{
successFunc = func ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
/ * *
* set failure function
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . setFailure
2017-02-13 21:12:00 +01:00
* @ function
* @ param { function } func
* /
me . setFailure = function ( func )
{
failureFunc = func ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
/ * *
* prepares a new upload
*
2017-02-17 20:46:10 +01:00
* Call this when doing a new upload to reset any data from potential
* previous uploads . Must be called before any other method of this
* module .
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . prepare
2017-02-13 21:12:00 +01:00
* @ function
* @ return { object }
* /
2017-02-17 20:46:10 +01:00
me . prepare = function ( )
2017-02-13 21:12:00 +01:00
{
2017-02-14 22:21:55 +01:00
// entropy should already be checked!
2017-02-13 21:12:00 +01:00
2017-02-17 20:46:10 +01:00
// reset password
password = '' ;
// reset key, so it a new one is generated when it is used
symmetricKey = null ;
2017-02-13 21:12:00 +01:00
// reset data
2017-02-14 22:21:55 +01:00
successFunc = null ;
failureFunc = null ;
2017-02-17 20:46:10 +01:00
url = Helper . baseUri ( ) ;
2017-02-13 21:12:00 +01:00
data = { } ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
/ * *
* encrypts and sets the data
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . setCipherMessage
2018-10-08 21:03:10 +02:00
* @ async
2017-02-13 21:12:00 +01:00
* @ function
2018-12-25 17:34:39 +01:00
* @ param { object } cipherMessage
2017-02-13 21:12:00 +01:00
* /
2018-12-25 17:34:39 +01:00
me . setCipherMessage = async function ( cipherMessage )
2017-02-13 21:12:00 +01:00
{
2018-12-25 17:34:39 +01:00
if (
symmetricKey === null ||
( typeof symmetricKey === 'string' && symmetricKey === '' )
) {
symmetricKey = CryptTool . getSymmetricKey ( ) ;
}
if ( ! data . hasOwnProperty ( 'adata' ) ) {
data [ 'adata' ] = [ ] ;
}
let cipherResult = await CryptTool . cipher ( symmetricKey , password , JSON . stringify ( cipherMessage ) , data [ 'adata' ] ) ;
data [ 'v' ] = 2 ;
data [ 'ct' ] = cipherResult [ 0 ] ;
data [ 'adata' ] = cipherResult [ 1 ] ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
/ * *
* set the additional metadata to send unencrypted
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . setUnencryptedData
2017-02-13 21:12:00 +01:00
* @ function
* @ param { string } index
* @ param { mixed } element
* /
me . setUnencryptedData = function ( index , element )
{
data [ index ] = element ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
/ * *
2018-12-25 17:34:39 +01:00
* Helper , which parses shows a general error message based on the result of the ServerInteraction
2017-02-13 21:12:00 +01:00
*
2018-12-25 17:34:39 +01:00
* @ name ServerInteraction . parseUploadError
2017-02-17 20:46:10 +01:00
* @ function
* @ param { int } status
* @ param { object } data
* @ param { string } doThisThing - a human description of the action , which was tried
* @ return { array }
* /
me . parseUploadError = function ( status , data , doThisThing ) {
2018-12-29 18:40:59 +01:00
let errorArray ;
2017-02-17 20:46:10 +01:00
switch ( status ) {
2018-01-06 09:26:10 +01:00
case me . error . custom :
2017-02-17 20:46:10 +01:00
errorArray = [ 'Could not ' + doThisThing + ': %s' , data . message ] ;
break ;
2018-01-06 09:26:10 +01:00
case me . error . unknown :
2017-02-17 20:46:10 +01:00
errorArray = [ 'Could not ' + doThisThing + ': %s' , I18n . _ ( 'unknown status' ) ] ;
break ;
2018-01-06 09:26:10 +01:00
case me . error . serverError :
2017-03-25 00:58:59 +01:00
errorArray = [ 'Could not ' + doThisThing + ': %s' , I18n . _ ( 'server error or not responding' ) ] ;
break ;
2017-02-17 20:46:10 +01:00
default :
errorArray = [ 'Could not ' + doThisThing + ': %s' , I18n . _ ( 'unknown error' ) ] ;
break ;
}
return errorArray ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-17 20:46:10 +01:00
2017-02-13 21:12:00 +01:00
return me ;
} ) ( ) ;
/ * *
2017-02-14 22:21:55 +01:00
* ( controller ) Responsible for encrypting paste and sending it to server .
2017-02-13 21:12:00 +01:00
*
2018-12-25 17:34:39 +01:00
* Does upload , encryption is done transparently by ServerInteraction .
2017-02-17 20:46:10 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name PasteEncrypter
2017-02-13 21:12:00 +01:00
* @ class
* /
2018-12-29 18:40:59 +01:00
const PasteEncrypter = ( function ( ) {
const me = { } ;
2017-02-13 21:12:00 +01:00
2018-12-29 18:40:59 +01:00
let requirementsChecked = false ;
2017-02-14 22:21:55 +01:00
2017-02-13 21:12:00 +01:00
/ * *
2017-02-17 20:46:10 +01:00
* called after successful paste upload
2017-02-13 21:12:00 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name PasteEncrypter . showCreatedPaste
2017-02-14 22:21:55 +01:00
* @ private
2017-02-13 21:12:00 +01:00
* @ function
* @ param { int } status
2017-02-17 20:46:10 +01:00
* @ param { object } data
2017-02-13 21:12:00 +01:00
* /
function showCreatedPaste ( status , data ) {
2017-02-15 22:59:55 +01:00
Alert . hideLoading ( ) ;
2017-02-14 22:21:55 +01:00
Alert . hideMessages ( ) ;
2017-02-13 21:12:00 +01:00
// show notification
2018-12-29 18:40:59 +01:00
const baseUri = Helper . baseUri ( ) + '?' ,
url = baseUri + data . id + '#' + btoa ( data . encryptionKey ) ,
deleteUrl = baseUri + 'pasteid=' + data . id + '&deletetoken=' + data . deletetoken ;
2018-01-06 09:26:10 +01:00
PasteStatus . createPasteNotification ( url , deleteUrl ) ;
2017-02-13 21:12:00 +01:00
// show new URL in browser bar
history . pushState ( { type : 'newpaste' } , document . title , url ) ;
2017-02-14 22:21:55 +01:00
TopNav . showViewButtons ( ) ;
TopNav . hideRawButton ( ) ;
Editor . hide ( ) ;
2017-02-13 21:12:00 +01:00
// parse and show text
2017-03-13 20:24:18 +01:00
// (preparation already done in me.sendPaste())
2017-02-14 22:21:55 +01:00
PasteViewer . run ( ) ;
}
2017-02-17 20:46:10 +01:00
/ * *
* called after successful comment upload
*
2017-03-13 20:24:18 +01:00
* @ name PasteEncrypter . showUploadedComment
2017-02-17 20:46:10 +01:00
* @ private
* @ function
* @ param { int } status
* @ param { object } data
* /
function showUploadedComment ( status , data ) {
// show success message
2018-01-06 13:32:07 +01:00
Alert . showStatus ( 'Comment posted.' ) ;
2017-02-17 20:46:10 +01:00
// reload paste
Controller . refreshPaste ( function ( ) {
// highlight sent comment
DiscussionViewer . highlightComment ( data . id , true ) ;
// reset error handler
Alert . setCustomHandler ( null ) ;
} ) ;
}
2017-02-12 18:08:08 +01:00
/ * *
* send a reply in a discussion
*
2017-02-14 22:21:55 +01:00
* @ name PasteEncrypter . sendComment
2018-10-20 17:57:21 +02:00
* @ async
2017-02-12 18:08:08 +01:00
* @ function
* /
2018-10-08 21:03:10 +02:00
me . sendComment = async function ( )
2017-02-12 18:08:08 +01:00
{
2017-02-17 20:46:10 +01:00
Alert . hideMessages ( ) ;
Alert . setCustomHandler ( DiscussionViewer . handleNotification ) ;
// UI loading state
2017-02-17 22:26:39 +01:00
TopNav . hideAllButtons ( ) ;
2018-01-06 13:32:07 +01:00
Alert . showLoading ( 'Sending comment…' , 'cloud-upload' ) ;
2017-02-17 20:46:10 +01:00
2017-12-18 14:47:17 +01:00
// get data
2018-12-29 18:40:59 +01:00
const plainText = DiscussionViewer . getReplyMessage ( ) ,
nickname = DiscussionViewer . getReplyNickname ( ) ,
parentid = DiscussionViewer . getReplyCommentId ( ) ;
2017-02-17 20:46:10 +01:00
// do not send if there is no data
if ( plainText . length === 0 ) {
// revert loading status…
Alert . hideLoading ( ) ;
Alert . setCustomHandler ( null ) ;
TopNav . showViewButtons ( ) ;
2017-02-12 18:08:08 +01:00
return ;
}
2018-12-25 17:34:39 +01:00
// prepare server interaction
ServerInteraction . prepare ( ) ;
ServerInteraction . setCryptParameters ( Prompt . getPassword ( ) , Model . getPasteKey ( ) ) ;
2017-02-17 20:46:10 +01:00
// set success/fail functions
2018-12-25 17:34:39 +01:00
ServerInteraction . setSuccess ( showUploadedComment ) ;
ServerInteraction . setFailure ( function ( status , data ) {
2017-02-17 20:46:10 +01:00
// revert loading status…
Alert . hideLoading ( ) ;
TopNav . showViewButtons ( ) ;
2018-10-20 11:40:37 +02:00
// …show error message…
2018-01-06 13:32:07 +01:00
Alert . showError (
2018-12-25 17:34:39 +01:00
ServerInteraction . parseUploadError ( status , data , 'post comment' )
2018-01-06 13:32:07 +01:00
) ;
2017-02-17 21:48:21 +01:00
2018-10-20 11:40:37 +02:00
// …and reset error handler
2017-02-17 21:48:21 +01:00
Alert . setCustomHandler ( null ) ;
2017-02-12 18:08:08 +01:00
} ) ;
2017-02-17 20:46:10 +01:00
// fill it with unencrypted params
2018-12-25 17:34:39 +01:00
ServerInteraction . setUnencryptedData ( 'pasteid' , Model . getPasteId ( ) ) ;
2017-02-17 20:46:10 +01:00
if ( typeof parentid === 'undefined' ) {
// if parent id is not set, this is the top-most comment, so use
2018-01-06 13:32:07 +01:00
// paste id as parent, as the root element of the discussion tree
2018-12-25 17:34:39 +01:00
ServerInteraction . setUnencryptedData ( 'parentid' , Model . getPasteId ( ) ) ;
2017-02-17 20:46:10 +01:00
} else {
2018-12-25 17:34:39 +01:00
ServerInteraction . setUnencryptedData ( 'parentid' , parentid ) ;
2017-02-17 20:46:10 +01:00
}
2018-12-25 17:34:39 +01:00
// prepare cypher message
let cipherMessage = {
'comment' : plainText
} ;
2018-10-20 11:40:37 +02:00
if ( nickname . length > 0 ) {
2018-12-25 17:34:39 +01:00
cipherMessage [ 'nickname' ] = nickname ;
2018-08-04 22:30:01 +02:00
}
2017-02-17 20:46:10 +01:00
2018-12-25 17:34:39 +01:00
await ServerInteraction . setCipherMessage ( cipherMessage ) . catch ( Alert . showError ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-12 18:08:08 +01:00
/ * *
2017-02-13 21:12:00 +01:00
* sends a new paste to server
2017-02-12 18:08:08 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name PasteEncrypter . sendPaste
2018-10-20 17:57:21 +02:00
* @ async
2017-02-12 18:08:08 +01:00
* @ function
* /
2018-10-08 21:03:10 +02:00
me . sendPaste = async function ( )
2017-02-12 18:08:08 +01:00
{
2017-02-15 22:59:55 +01:00
// hide previous (error) messages
2017-02-17 20:46:10 +01:00
Controller . hideStatusMessages ( ) ;
2017-02-15 22:59:55 +01:00
2017-02-13 21:12:00 +01:00
// UI loading state
2017-02-17 22:26:39 +01:00
TopNav . hideAllButtons ( ) ;
2018-01-06 13:32:07 +01:00
Alert . showLoading ( 'Sending paste…' , 'cloud-upload' ) ;
2017-02-14 22:21:55 +01:00
TopNav . collapseBar ( ) ;
2017-02-12 18:08:08 +01:00
2017-02-13 21:12:00 +01:00
// get data
2018-12-29 18:40:59 +01:00
const plainText = Editor . getText ( ) ,
format = PasteViewer . getFormat ( ) ,
// the methods may return different values if no files are attached (null, undefined or false)
files = TopNav . getFileList ( ) || AttachmentViewer . getFile ( ) || AttachmentViewer . hasAttachment ( ) ;
2017-02-13 21:12:00 +01:00
// do not send if there is no data
2018-05-21 19:32:01 +02:00
if ( plainText . length === 0 && ! files ) {
2017-02-13 21:12:00 +01:00
// revert loading status…
2017-02-15 22:59:55 +01:00
Alert . hideLoading ( ) ;
2017-02-14 22:21:55 +01:00
TopNav . showCreateButtons ( ) ;
2017-02-12 18:08:08 +01:00
return ;
}
2018-12-25 17:34:39 +01:00
// prepare server interaction
ServerInteraction . prepare ( ) ;
ServerInteraction . setCryptParameters ( TopNav . getPassword ( ) ) ;
2017-02-13 21:12:00 +01:00
// set success/fail functions
2018-12-25 17:34:39 +01:00
ServerInteraction . setSuccess ( showCreatedPaste ) ;
ServerInteraction . setFailure ( function ( status , data ) {
2017-02-12 18:08:08 +01:00
// revert loading status…
2017-02-15 22:59:55 +01:00
Alert . hideLoading ( ) ;
2017-02-14 22:21:55 +01:00
TopNav . showCreateButtons ( ) ;
2017-02-13 21:12:00 +01:00
// show error message
2018-01-06 13:32:07 +01:00
Alert . showError (
2018-12-25 17:34:39 +01:00
ServerInteraction . parseUploadError ( status , data , 'create paste' )
2018-01-06 13:32:07 +01:00
) ;
2017-02-12 18:08:08 +01:00
} ) ;
2017-02-13 21:12:00 +01:00
// fill it with unencrypted submitted options
2018-12-25 17:34:39 +01:00
ServerInteraction . setUnencryptedData ( 'adata' , [
null , format ,
TopNav . getOpenDiscussion ( ) ? 1 : 0 ,
TopNav . getBurnAfterReading ( ) ? 1 : 0
] ) ;
ServerInteraction . setUnencryptedData ( 'meta' , { 'expire' : TopNav . getExpiration ( ) } ) ;
2017-02-12 18:08:08 +01:00
2017-02-13 21:12:00 +01:00
// prepare PasteViewer for later preview
2017-02-14 22:21:55 +01:00
PasteViewer . setText ( plainText ) ;
PasteViewer . setFormat ( format ) ;
2018-12-25 17:34:39 +01:00
// prepare cypher message
let file = AttachmentViewer . getAttachmentData ( ) ,
cipherMessage = {
'paste' : plainText
} ;
if ( typeof file !== 'undefined' && file !== null ) {
cipherMessage [ 'attachment' ] = file ;
cipherMessage [ 'attachment_name' ] = AttachmentViewer . getFile ( ) . name ;
} else if ( AttachmentViewer . hasAttachment ( ) ) {
// fall back to cloned part
let attachment = AttachmentViewer . getAttachment ( ) ;
cipherMessage [ 'attachment' ] = attachment [ 0 ] ;
cipherMessage [ 'attachment_name' ] = attachment [ 1 ] ;
}
2018-10-08 21:03:10 +02:00
2018-12-25 17:34:39 +01:00
// encrypt message
await ServerInteraction . setCipherMessage ( cipherMessage ) . catch ( Alert . showError ) ;
2018-10-08 21:03:10 +02:00
// send data
2018-12-25 17:34:39 +01:00
ServerInteraction . run ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-14 22:21:55 +01:00
return me ;
} ) ( ) ;
/ * *
* ( controller ) Responsible for decrypting cipherdata and passing data to view .
*
2017-02-17 20:46:10 +01:00
* Only decryption , no download .
*
2017-03-13 20:24:18 +01:00
* @ name PasteDecrypter
2017-02-14 22:21:55 +01:00
* @ class
* /
2018-12-29 18:40:59 +01:00
const PasteDecrypter = ( function ( ) {
const me = { } ;
2017-02-14 22:21:55 +01:00
/ * *
2017-04-11 16:34:13 +02:00
* decrypt data or prompts for password in case of failure
2017-02-14 22:21:55 +01:00
*
2017-03-13 20:24:18 +01:00
* @ name PasteDecrypter . decryptOrPromptPassword
2017-02-14 22:21:55 +01:00
* @ private
2018-10-20 09:56:05 +02:00
* @ async
2017-02-14 22:21:55 +01:00
* @ function
2017-03-13 20:24:18 +01:00
* @ param { string } key
* @ param { string } password - optional , may be an empty string
* @ param { string } cipherdata
2017-02-17 20:46:10 +01:00
* @ throws { string }
2017-03-13 20:24:18 +01:00
* @ return { false | string } false , when unsuccessful or string ( decrypted data )
2017-02-14 22:21:55 +01:00
* /
2018-10-20 09:56:05 +02:00
async function decryptOrPromptPassword ( key , password , cipherdata )
2017-02-14 22:21:55 +01:00
{
// try decryption without password
2018-10-20 17:57:21 +02:00
const plaindata = await CryptTool . decipher ( key , password , cipherdata ) ;
2017-02-14 22:21:55 +01:00
// if it fails, request password
2017-02-17 20:46:10 +01:00
if ( plaindata . length === 0 && password . length === 0 ) {
2017-04-11 16:34:13 +02:00
// show prompt
Prompt . requestPassword ( ) ;
// Thus, we cannot do anything yet, we need to wait for the user
// input.
return false ;
2017-02-14 22:21:55 +01:00
}
2017-02-17 20:46:10 +01:00
// if all tries failed, we can only return an error
if ( plaindata . length === 0 ) {
throw 'failed to decipher data' ;
}
return plaindata ;
}
/ * *
* decrypt the actual paste text
*
2017-04-11 16:34:13 +02:00
* @ name PasteDecrypter . decryptPaste
2017-02-17 20:46:10 +01:00
* @ private
2018-10-20 12:40:08 +02:00
* @ async
2017-02-17 20:46:10 +01:00
* @ function
2017-03-13 20:24:18 +01:00
* @ param { object } paste - paste data in object form
* @ param { string } key
* @ param { string } password
2017-02-17 20:46:10 +01:00
* @ throws { string }
2018-10-20 12:40:08 +02:00
* @ return { Promise }
2017-02-17 20:46:10 +01:00
* /
2018-12-29 18:40:59 +01:00
async function decryptPaste ( paste , key , password )
2017-02-17 20:46:10 +01:00
{
2018-12-29 18:40:59 +01:00
const pastePlain = await decryptOrPromptPassword (
key , password ,
paste . hasOwnProperty ( 'ct' ) ? paste . ct : paste . data
) ;
if ( pastePlain === false ) {
if ( password . length === 0 ) {
throw 'waiting on user to provide a password' ;
} else {
displayDecryptionError ( 'failed to decipher paste text: Incorrect password?' ) ;
throw 'waiting on user to provide correct password' ;
2017-02-17 20:46:10 +01:00
}
2018-12-29 18:40:59 +01:00
}
2017-02-14 22:21:55 +01:00
2018-12-29 18:40:59 +01:00
if ( paste . hasOwnProperty ( 'attachment' ) && paste . hasOwnProperty ( 'attachmentname' ) ) {
// version 1 paste
Promise . all ( [
CryptTool . decipher ( key , password , paste . attachment ) ,
CryptTool . decipher ( key , password , paste . attachmentname )
] ) . then ( ( attachment ) => {
AttachmentViewer . setAttachment ( attachment [ 0 ] , attachment [ 1 ] ) ;
} ) ;
PasteViewer . setFormat ( paste . meta . formatter ) ;
PasteViewer . setText ( pastePlain ) ;
} else {
// version 2 paste
const pasteMessage = JSON . parse ( pastePlain ) ;
if ( pasteMessage . hasOwnProperty ( 'attachment' ) && pasteMessage . hasOwnProperty ( 'attachment_name' ) ) {
AttachmentViewer . setAttachment ( pasteMessage . attachment , pasteMessage . attachment _name ) ;
2017-02-14 22:21:55 +01:00
}
2018-12-29 18:40:59 +01:00
PasteViewer . setFormat ( paste . adata [ 1 ] ) ;
PasteViewer . setText ( pasteMessage . paste ) ;
}
PasteViewer . run ( ) ;
AttachmentViewer . showAttachment ( ) ;
2017-02-17 20:46:10 +01:00
}
/ * *
* decrypts all comments and shows them
*
2017-03-13 20:24:18 +01:00
* @ name PasteDecrypter . decryptComments
2017-02-17 20:46:10 +01:00
* @ private
2018-10-20 12:40:08 +02:00
* @ async
2017-02-17 20:46:10 +01:00
* @ function
2017-03-13 20:24:18 +01:00
* @ param { object } paste - paste data in object form
* @ param { string } key
* @ param { string } password
2018-10-20 12:40:08 +02:00
* @ return { Promise }
2017-02-17 20:46:10 +01:00
* /
2018-10-20 12:40:08 +02:00
async function decryptComments ( paste , key , password )
2017-02-17 20:46:10 +01:00
{
2018-10-20 10:20:32 +02:00
// remove potential previous discussion
2017-12-18 14:47:17 +01:00
DiscussionViewer . prepareNewDiscussion ( ) ;
2017-02-17 20:46:10 +01:00
2018-12-29 18:40:59 +01:00
const commentDecryptionPromises = [ ] ;
2017-02-17 20:46:10 +01:00
// iterate over comments
2018-10-20 17:57:21 +02:00
for ( let i = 0 ; i < paste . comments . length ; ++ i ) {
2018-12-29 18:40:59 +01:00
if ( paste . comments [ i ] . hasOwnProperty ( 'v' ) && paste . comments [ i ] . v === 2 ) {
// version 2 comment
commentDecryptionPromises . push (
CryptTool . decipher ( key , password , paste . comments [ i ] . ct )
. then ( ( commentJson ) => {
const commentMessage = JSON . parse ( commentJson ) ;
return [
commentMessage . hasOwnProperty ( 'comment' ) ? commentMessage . comment : '' ,
commentMessage . hasOwnProperty ( 'nickname' ) ? commentMessage . nickname : ''
] ;
} )
) ;
} else {
// version 1 comment
commentDecryptionPromises . push (
Promise . all ( [
CryptTool . decipher ( key , password , paste . comments [ i ] . data ) ,
paste . comments [ i ] . meta . nickname ?
CryptTool . decipher ( key , password , paste . comments [ i ] . meta . nickname ) :
Promise . resolve ( '' )
] )
) ;
}
2017-02-17 20:46:10 +01:00
}
2018-10-20 12:40:08 +02:00
return Promise . all ( commentDecryptionPromises ) . then ( ( plaintexts ) => {
2018-10-20 17:57:21 +02:00
for ( let i = 0 ; i < paste . comments . length ; ++ i ) {
2018-10-20 13:54:17 +02:00
if ( plaintexts [ i ] [ 0 ] . length === 0 ) {
2018-10-20 11:40:37 +02:00
continue ;
}
2018-10-20 17:57:21 +02:00
const comment = paste . comments [ i ] ;
2018-10-20 10:20:32 +02:00
DiscussionViewer . addComment (
comment ,
2018-10-20 13:54:17 +02:00
plaintexts [ i ] [ 0 ] ,
plaintexts [ i ] [ 1 ]
2018-10-20 10:20:32 +02:00
) ;
}
DiscussionViewer . finishDiscussion ( ) ;
} ) ;
2017-02-14 22:21:55 +01:00
}
2018-10-20 11:40:37 +02:00
/ * *
* displays and logs decryption errors
*
* @ name PasteDecrypter . displayDecryptionError
* @ private
* @ function
* @ param { string } message
* /
function displayDecryptionError ( message )
{
Alert . hideLoading ( ) ;
// log detailed error, but display generic translation
console . error ( message ) ;
Alert . showError ( 'Could not decrypt data. Did you enter a wrong password? Retry with the button at the top.' ) ;
// reset password, so it can be re-entered
Prompt . reset ( ) ;
TopNav . showRetryButton ( ) ;
}
2017-02-14 22:21:55 +01:00
/ * *
* show decrypted text in the display area , including discussion ( if open )
*
* @ name PasteDecrypter . run
* @ function
* @ param { Object } [ paste ] - ( optional ) object including comments to display ( items = array with keys ( 'data' , 'meta' ) )
* /
me . run = function ( paste )
{
2017-03-12 16:06:17 +01:00
Alert . hideMessages ( ) ;
2018-01-06 13:32:07 +01:00
Alert . showLoading ( 'Decrypting paste…' , 'cloud-download' ) ;
2017-02-14 22:21:55 +01:00
if ( typeof paste === 'undefined' ) {
2017-04-11 16:34:13 +02:00
// get cipher data and wait until it is available
Model . getPasteData ( me . run ) ;
return ;
2017-02-14 22:21:55 +01:00
}
2018-10-20 11:40:37 +02:00
let key = Model . getPasteKey ( ) ,
password = Prompt . getPassword ( ) ,
2018-10-20 17:57:21 +02:00
decryptionPromises = [ ] ;
2017-02-14 22:21:55 +01:00
2018-10-20 11:40:37 +02:00
TopNav . setRetryCallback ( function ( ) {
TopNav . hideRetryButton ( ) ;
me . run ( paste ) ;
} ) ;
2018-12-29 18:40:59 +01:00
// decrypt paste & attachments
decryptionPromises . push ( decryptPaste ( paste , key , password ) )
2017-03-13 19:30:44 +01:00
2018-10-20 11:40:37 +02:00
// if the discussion is opened on this paste, display it
if ( paste . meta . opendiscussion ) {
2018-10-20 17:57:21 +02:00
decryptionPromises . push ( decryptComments ( paste , key , password ) ) ;
2018-10-20 11:40:37 +02:00
}
2017-03-13 19:30:44 +01:00
2018-10-20 13:54:17 +02:00
// shows the remaining time (until) deletion
PasteStatus . showRemainingTime ( paste . meta ) ;
2018-12-29 18:40:59 +01:00
Promise . all ( decryptionPromises )
. then ( ( ) => {
Alert . hideLoading ( ) ;
TopNav . showViewButtons ( ) ;
} )
. catch ( ( err ) => {
// wait for the user to type in the password,
// then PasteDecrypter.run will be called again
} ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-13 21:12:00 +01:00
2017-02-14 22:21:55 +01:00
return me ;
} ) ( ) ;
2013-11-01 01:15:14 +01:00
2017-01-29 14:32:55 +01:00
/ * *
2017-02-14 22:21:55 +01:00
* ( controller ) main PrivateBin logic
*
2017-03-13 20:24:18 +01:00
* @ name Controller
2017-02-14 22:21:55 +01:00
* @ param { object } window
* @ param { object } document
* @ class
2017-01-29 14:32:55 +01:00
* /
2018-12-29 18:40:59 +01:00
const Controller = ( function ( window , document ) {
const me = { } ;
2017-02-14 22:21:55 +01:00
2017-02-15 22:59:55 +01:00
/ * *
* hides all status messages no matter which module showed them
*
* @ name Controller . hideStatusMessages
* @ function
* /
me . hideStatusMessages = function ( )
{
PasteStatus . hideMessages ( ) ;
Alert . hideMessages ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-15 22:59:55 +01:00
2017-02-12 21:13:04 +01:00
/ * *
* creates a new paste
*
2017-02-14 22:21:55 +01:00
* @ name Controller . newPaste
2017-02-12 21:13:04 +01:00
* @ function
* /
me . newPaste = function ( )
{
2017-02-17 20:46:10 +01:00
// Important: This *must not* run Alert.hideMessages() as previous
// errors from viewing a paste should be shown.
2017-02-17 22:26:39 +01:00
TopNav . hideAllButtons ( ) ;
2018-01-06 13:32:07 +01:00
Alert . showLoading ( 'Preparing new paste…' , 'time' ) ;
2017-02-15 22:59:55 +01:00
2017-02-17 20:46:10 +01:00
PasteStatus . hideMessages ( ) ;
2017-02-14 22:21:55 +01:00
PasteViewer . hide ( ) ;
Editor . resetInput ( ) ;
Editor . show ( ) ;
Editor . focusInput ( ) ;
2017-05-13 21:27:41 +02:00
AttachmentViewer . removeAttachment ( ) ;
2017-02-15 22:59:55 +01:00
TopNav . showCreateButtons ( ) ;
Alert . hideLoading ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-12 21:13:04 +01:00
2017-04-11 16:34:13 +02:00
/ * *
* shows how we much we love bots that execute JS ; )
*
* @ name Controller . showBadBotMessage
* @ function
* /
me . showBadBotMessage = function ( )
{
TopNav . hideAllButtons ( ) ;
Alert . showError ( 'I love you too, bot…' ) ;
}
2017-02-17 20:46:10 +01:00
/ * *
* shows the loaded paste
*
* @ name Controller . showPaste
* @ function
* /
me . showPaste = function ( )
{
try {
Model . getPasteKey ( ) ;
} catch ( err ) {
console . error ( err ) ;
// missing decryption key (or paste ID) in URL?
if ( window . location . hash . length === 0 ) {
Alert . showError ( 'Cannot decrypt paste: Decryption key missing in URL (Did you use a redirector or an URL shortener which strips part of the URL?)' ) ;
return ;
}
}
// show proper elements on screen
PasteDecrypter . run ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-17 20:46:10 +01:00
/ * *
* refreshes the loaded paste to show potential new data
*
* @ name Controller . refreshPaste
* @ function
2017-03-13 20:24:18 +01:00
* @ param { function } callback
2017-02-17 20:46:10 +01:00
* /
me . refreshPaste = function ( callback )
{
// save window position to restore it later
2018-12-29 18:40:59 +01:00
const orgPosition = $ ( window ) . scrollTop ( ) ;
2017-02-17 20:46:10 +01:00
2017-04-11 16:34:13 +02:00
Model . getPasteData ( function ( data ) {
2018-12-25 17:34:39 +01:00
ServerInteraction . prepare ( ) ;
ServerInteraction . setUrl ( Helper . baseUri ( ) + '?' + Model . getPasteId ( ) ) ;
2017-02-17 20:46:10 +01:00
2018-12-25 17:34:39 +01:00
ServerInteraction . setFailure ( function ( status , data ) {
2018-04-30 20:01:38 +02:00
// revert loading status…
Alert . hideLoading ( ) ;
TopNav . showViewButtons ( ) ;
2017-02-17 20:46:10 +01:00
2018-04-30 20:01:38 +02:00
// show error message
Alert . showError (
2018-12-25 17:34:39 +01:00
ServerInteraction . parseUploadError ( status , data , 'refresh display' )
2018-04-30 20:01:38 +02:00
) ;
} ) ;
2018-12-25 17:34:39 +01:00
ServerInteraction . setSuccess ( function ( status , data ) {
2018-04-30 20:01:38 +02:00
PasteDecrypter . run ( data ) ;
2017-02-17 20:46:10 +01:00
2018-04-30 20:01:38 +02:00
// restore position
window . scrollTo ( 0 , orgPosition ) ;
2017-04-11 16:34:13 +02:00
2018-04-30 20:01:38 +02:00
// NOTE: could create problems as callback may be called
// asyncronously if PasteDecrypter e.g. needs to wait for a
// password being entered
callback ( ) ;
} ) ;
2018-12-25 17:34:39 +01:00
ServerInteraction . run ( ) ;
2017-04-11 16:34:13 +02:00
} , false ) ; // this false is important as it circumvents the cache
2017-02-25 09:35:55 +01:00
}
2017-02-17 20:46:10 +01:00
2017-02-12 18:08:08 +01:00
/ * *
* clone the current paste
*
2017-02-14 22:21:55 +01:00
* @ name Controller . clonePaste
2017-02-12 18:08:08 +01:00
* @ function
* /
2018-01-06 13:32:07 +01:00
me . clonePaste = function ( )
2017-02-12 18:08:08 +01:00
{
2017-02-15 22:59:55 +01:00
TopNav . collapseBar ( ) ;
2017-02-17 22:26:39 +01:00
TopNav . hideAllButtons ( ) ;
2017-02-15 22:59:55 +01:00
// hide messages from previous paste
me . hideStatusMessages ( ) ;
2017-02-12 18:08:08 +01:00
// erase the id and the key in url
2017-02-15 22:59:55 +01:00
history . pushState ( { type : 'clone' } , document . title , Helper . baseUri ( ) ) ;
2017-02-12 18:08:08 +01:00
2017-02-15 22:59:55 +01:00
if ( AttachmentViewer . hasAttachment ( ) ) {
AttachmentViewer . moveAttachmentTo (
TopNav . getCustomAttachment ( ) ,
'Cloned: \'%s\''
) ;
TopNav . hideFileSelector ( ) ;
AttachmentViewer . hideAttachment ( ) ;
// NOTE: it also looks nice without removing the attachment
// but for a consistent display we remove it…
AttachmentViewer . hideAttachmentPreview ( ) ;
TopNav . showCustomAttachment ( ) ;
// show another status message to make the user aware that the
// file was cloned too!
Alert . showStatus (
[
'The cloned file \'%s\' was attached to this paste.' ,
AttachmentViewer . getAttachment ( ) [ 1 ]
2018-01-06 13:32:07 +01:00
] ,
'copy'
) ;
2017-02-15 22:59:55 +01:00
}
2018-01-06 10:57:54 +01:00
Editor . setText ( PasteViewer . getText ( ) ) ;
2017-02-15 22:59:55 +01:00
PasteViewer . hide ( ) ;
Editor . show ( ) ;
TopNav . showCreateButtons ( ) ;
2018-01-06 09:26:10 +01:00
} ;
2017-02-12 18:08:08 +01:00
/ * *
* application start
*
2017-02-14 22:21:55 +01:00
* @ name Controller . init
2017-02-12 18:08:08 +01:00
* @ function
* /
2018-12-27 21:32:13 +01:00
me . init = async function ( )
2017-02-12 18:08:08 +01:00
{
// first load translations
2017-02-14 22:21:55 +01:00
I18n . loadTranslations ( ) ;
2017-02-12 18:08:08 +01:00
2018-01-01 10:25:07 +01:00
DOMPurify . setConfig ( { SAFE _FOR _JQUERY : true } ) ;
2017-02-12 21:13:04 +01:00
// initialize other modules/"classes"
2017-02-14 22:21:55 +01:00
Alert . init ( ) ;
2017-02-15 22:59:55 +01:00
Model . init ( ) ;
2017-02-17 22:46:18 +01:00
AttachmentViewer . init ( ) ;
DiscussionViewer . init ( ) ;
2017-02-14 22:21:55 +01:00
Editor . init ( ) ;
PasteStatus . init ( ) ;
PasteViewer . init ( ) ;
Prompt . init ( ) ;
2017-02-17 22:46:18 +01:00
TopNav . init ( ) ;
UiHelper . init ( ) ;
2018-12-27 21:32:13 +01:00
z = ( await zlib ) ;
2017-02-08 20:11:04 +01:00
2017-04-11 16:34:13 +02:00
// check whether existing paste needs to be shown
try {
Model . getPasteId ( ) ;
} catch ( e ) {
// otherwise create a new paste
return me . newPaste ( ) ;
}
2018-05-22 11:41:35 +02:00
// if delete token is passed (i.e. paste has been deleted by this access)
// there is no more stuf we need to do
if ( Model . hasDeleteToken ( ) ) {
return ;
}
2017-04-11 16:34:13 +02:00
// prevent bots from viewing a paste and potentially deleting data
// when burn-after-reading is set
// see https://github.com/elrido/ZeroBin/issues/11
if ( Helper . isBadBot ( ) ) {
return me . showBadBotMessage ( ) ;
2017-02-13 21:12:00 +01:00
}
2018-04-30 20:01:38 +02:00
// display an existing paste
2017-04-11 16:34:13 +02:00
return me . showPaste ( ) ;
2017-02-25 09:35:55 +01:00
}
2017-02-08 20:12:22 +01:00
return me ;
} ) ( window , document ) ;
2017-01-29 14:32:55 +01:00
2017-01-22 10:42:11 +01:00
return {
2017-02-14 22:21:55 +01:00
Helper : Helper ,
I18n : I18n ,
CryptTool : CryptTool ,
2017-02-17 22:46:18 +01:00
Model : Model ,
UiHelper : UiHelper ,
2017-02-14 22:21:55 +01:00
Alert : Alert ,
2017-02-17 22:46:18 +01:00
PasteStatus : PasteStatus ,
Prompt : Prompt ,
Editor : Editor ,
2017-02-17 20:46:10 +01:00
PasteViewer : PasteViewer ,
2017-02-17 22:46:18 +01:00
AttachmentViewer : AttachmentViewer ,
DiscussionViewer : DiscussionViewer ,
TopNav : TopNav ,
2018-12-25 17:34:39 +01:00
ServerInteraction : ServerInteraction ,
2017-02-17 22:46:18 +01:00
PasteEncrypter : PasteEncrypter ,
PasteDecrypter : PasteDecrypter ,
Controller : Controller
2017-01-22 10:42:11 +01:00
} ;
2018-10-20 19:53:21 +02:00
} ) ( jQuery , RawDeflate ) ;