2013-12-30 12:42:36 -02:00
// interface_common.js
// 2013 Lucas Leal, Miguel Freitas
//
// Common interface functions to all pages, modal manipulation, button manipulation etc
// Profile, mentions and hashtag modal
// Post actions: submit, count characters
2016-01-31 11:23:07 +05:00
var twister = {
2019-02-02 14:09:26 +05:00
profiles : { } ,
avatars : { } ,
2016-03-21 00:55:32 +05:00
URIs : { } , // shortened URIs are cached here after fetching
2016-05-08 13:09:38 -03:00
torrentIds : { } , // auto-download torrentIds
2016-03-23 04:02:20 +05:00
focus : { } , // focused elements are counted here
2016-01-31 11:23:07 +05:00
html : {
detached : $ ( '<div>' ) , // here elements go to detach themself
2016-04-05 19:35:59 +05:00
blanka : $ ( '<a rel="nofollow noreferrer" target="_blank">' ) // to open stuff in new tab, see routeOnClick()
2016-01-31 11:23:07 +05:00
} ,
2016-02-03 12:02:47 +05:00
tmpl : { // templates pointers are stored here
root : $ ( '<div>' ) // templates should be detached from DOM and attached here; use extractTemplate()
} ,
2017-01-23 03:32:34 +05:00
modal : { } ,
2017-06-17 23:59:30 +05:00
res : { } , // reses for various reqs are cached here
2019-02-02 14:14:03 +05:00
twists : { } ,
2017-01-23 03:32:34 +05:00
var : {
2018-01-14 01:19:45 +05:00
dateFormatter : {
format : function ( req ) {
return req . toString ( ) . replace ( /GMT.*/g , '' ) ;
}
} ,
2019-02-02 12:57:30 +05:00
initializated : false ,
2019-01-24 15:58:37 +05:00
isCurrentInputSplittable : false ,
2017-02-02 14:17:10 +05:00
localAccounts : [ ] ,
2017-01-23 03:32:34 +05:00
updatesCheckClient : { }
}
2016-01-31 11:23:07 +05:00
} ;
2015-03-01 13:31:56 +05:00
var window _scrollY = 0 ;
2016-01-10 03:25:40 +05:00
var _watchHashChangeRelaxDontDoIt = window . location . hash === '' ? true : false ;
2016-01-31 11:23:07 +05:00
// FIXME so looks like it's wrapper over $; it's here to select and manipulate detached elements too
// and actually I'm talking about 'so called \'detached\'' elements which appended to twister.html.detached
// we may just append twister.html.detached to document instead and remove this weird shit (or I need to
// improve my google skills to find native jQuery way to dig through all detached elemets with one query)
2016-03-23 23:49:58 +05:00
function getElem ( req , searchInTmpl ) {
2016-01-31 11:23:07 +05:00
var elem = $ ( req ) ;
var h = twister . html . detached . find ( req ) ;
for ( var i = 0 ; i < h . length ; i ++ )
elem [ elem . length ++ ] = h [ i ] ;
2016-03-23 23:49:58 +05:00
if ( searchInTmpl ) {
h = twister . tmpl . root . find ( req ) ;
for ( var i = 0 ; i < h . length ; i ++ )
elem [ elem . length ++ ] = h [ i ] ;
}
2016-01-31 11:23:07 +05:00
return elem ;
}
2015-03-01 13:31:56 +05:00
2015-05-23 22:26:23 +05:00
function openModal ( modal ) {
2015-08-20 10:30:07 +05:00
if ( ! modal . classBase ) {
2015-05-23 22:26:23 +05:00
modal . classBase = '.modal-wrapper' ;
2014-01-08 00:47:13 -08:00
2015-08-20 10:30:07 +05:00
window _scrollY = window . pageYOffset ;
$ ( 'body' ) . css ( 'overflow' , 'hidden' ) ;
}
2016-01-31 11:23:07 +05:00
if ( modal . classBase !== '.prompt-wrapper' )
closeModal ( $ ( modal . classBase + ':not(#templates *)' ) , true ) ;
2013-12-30 12:42:36 -02:00
2015-05-23 22:26:23 +05:00
modal . self = $ ( '#templates ' + modal . classBase ) . clone ( true )
. addClass ( modal . classAdd ) ;
2015-01-22 15:59:00 +01:00
2017-02-02 14:17:10 +05:00
if ( modal . removeBlackout )
modal . self . find ( '.modal-blackout' ) . remove ( ) ;
2015-05-23 22:26:23 +05:00
if ( modal . title )
modal . self . find ( '.modal-header h3' ) . html ( modal . title ) ;
if ( modal . content )
modal . content = modal . self . find ( '.modal-content' )
. append ( modal . content ) ;
else
modal . content = modal . self . find ( '.modal-content' ) ;
2015-01-22 15:59:00 +01:00
2016-09-11 18:45:38 +05:00
if ( modal . warn && modal . warn . name && modal . warn . text ) {
var elem = twister . tmpl . modalComponentWarn . clone ( true )
2016-09-11 17:26:53 +05:00
. attr ( 'data-warn-name' , modal . warn . name )
. toggle ( ! $ . Options . get ( 'skipWarn' + modal . warn . name ) )
2016-09-11 17:15:58 +05:00
;
2016-09-11 18:45:38 +05:00
fillElemWithTxt ( elem . find ( '.text' ) , modal . warn . text , { markout : 'apply' } ) ;
elem . find ( '.options .never-again + span' ) . text ( polyglot . t ( 'do_not_show_it_again' ) ) ;
elem . insertBefore ( modal . content ) ;
}
2016-09-11 17:15:58 +05:00
2016-01-31 11:23:07 +05:00
modal . self . appendTo ( 'body' ) . fadeIn ( 'fast' ) ; // FIXME maybe it's better to append it to some container inside body
2015-01-22 15:59:00 +01:00
2015-06-12 03:34:59 +05:00
if ( modal . classBase === '.modal-wrapper' ) {
2016-01-31 11:23:07 +05:00
twister . modal [ window . location . hash ] = modal ;
modal . self . attr ( 'data-modal-id' , window . location . hash ) ;
modal . drapper = $ ( '<div>' ) . appendTo ( twister . html . detached ) ; // here modal goes instead detaching
2016-09-11 17:15:58 +05:00
modal . content . outerHeight ( modal . self . height ( ) - modal . self . find ( '.modal-header' ) . outerHeight ( )
2019-01-23 06:24:32 +05:00
- ( modal . warn && ! $ . Options . get ( 'skipWarn' + modal . warn . name ) ?
modal . self . find ( '.inline-warn' ) . outerHeight ( ) : 0 )
) ;
2015-06-12 03:34:59 +05:00
var windowHeight = $ ( window ) . height ( ) ;
if ( modal . self . outerHeight ( ) > windowHeight ) {
modal . content . outerHeight ( modal . content . outerHeight ( ) - modal . self . outerHeight ( ) + windowHeight ) ;
modal . self . outerHeight ( windowHeight ) ;
modal . self . css ( 'margin-top' , - windowHeight / 2 ) ;
}
}
2015-05-23 22:26:23 +05:00
return modal ;
}
2015-01-22 15:59:00 +01:00
2016-01-31 11:23:07 +05:00
function closeModal ( req , switchMode ) {
if ( typeof req === 'undefined' )
var elem = $ ( '.modal-wrapper:not(#templates *)' ) ; // select active modal(s)
else if ( req . jquery )
var elem = req ;
else if ( req . target )
var elem = getElem ( req . target ) ; // getElem() to search in minimized too
else if ( typeof req === 'string' || req . outerHTML )
var elem = getElem ( req ) ;
if ( ! elem || ! elem . length )
return ;
// we close all modals which are containing element(s)
elem . closest ( '.modal-wrapper:not(.closed)' ) . addClass ( 'closed' )
. fadeOut ( switchMode ? 10 : 'fast' , function ( ) {
var i = this . getAttribute ( 'data-modal-id' ) ;
if ( twister . modal [ i ] . minimized )
twister . modal [ i ] . btnResume . fadeOut ( 'fast' , function ( ) { this . remove ( ) ; } ) ;
else
this . remove ( ) ; // if it's minimized it will be removed with twister.modal[i].drapper
2015-08-20 10:30:07 +05:00
2016-11-24 15:54:33 +03:00
if ( typeof twister . modal [ i ] . onClose === 'function' )
2016-11-25 02:52:57 +05:00
twister . modal [ i ] . onClose ( twister . modal [ i ] . onCloseReq ) ;
2016-11-24 15:54:33 +03:00
2016-01-31 11:23:07 +05:00
twister . modal [ i ] . drapper . remove ( ) ;
twister . modal [ i ] = undefined ;
}
) ;
if ( ! switchMode ) {
if ( window . location . hash !== '' ) {
_watchHashChangeRelaxDontDoIt = true ;
window . location . hash = '#' ;
}
window . scroll ( window . pageXOffset , window _scrollY ) ;
$ ( 'body' ) . css ( {
'overflow' : 'auto' ,
'margin-right' : '0'
} ) ;
2016-01-10 03:25:40 +05:00
}
2015-01-22 15:59:00 +01:00
}
2016-01-31 11:23:07 +05:00
function closePrompt ( req ) {
if ( typeof req === 'undefined' )
var elem = $ ( '.prompt-wrapper:not(#templates *)' ) ;
else if ( req . jquery )
var elem = req ;
else if ( req . target )
var elem = $ ( req . target ) ;
else if ( typeof req === 'string' || req . outerHTML )
var elem = $ ( req ) ;
if ( ! elem || ! elem . length )
return ;
2015-09-04 03:21:33 +05:00
2016-01-31 11:23:07 +05:00
if ( typeof req . stopPropagation === 'function' ) {
req . preventDefault ( ) ;
req . stopPropagation ( ) ;
req = req . data ;
}
2015-01-22 15:59:00 +01:00
2016-01-31 11:23:07 +05:00
// we close all prompts which are containing element(s)
elem . closest ( '.prompt-wrapper:not(.closed)' ) . addClass ( 'closed' )
. fadeOut ( 'fast' , function ( ) { this . remove ( ) ; } ) ;
if ( req && typeof req . cbFunc === 'function' ) // FIXME maybe bind to ^ prompt fadeout function
req . cbFunc ( req . cbReq ) ;
2015-01-22 15:59:00 +01:00
}
2015-11-19 01:09:04 +05:00
function minimizeModal ( modal , switchMode ) {
2016-01-31 11:23:07 +05:00
2015-11-19 01:09:04 +05:00
function minimize ( modal , scroll ) {
2016-01-31 11:23:07 +05:00
var i = modal . attr ( 'data-modal-id' ) ;
2015-11-19 01:09:04 +05:00
2016-01-31 11:23:07 +05:00
modal . appendTo ( twister . modal [ i ] . drapper ) ;
twister . modal [ i ] . minimized = true ;
twister . modal [ i ] . scroll = scroll ;
twister . modal [ i ] . btnResume = $ ( '<li>' + modal . find ( '.modal-header h3' ) . text ( ) + '</li>' )
2016-02-04 00:03:45 +05:00
. on ( 'click' , { hashString : window . location . hash } , function ( event ) {
2019-01-23 06:24:32 +05:00
if ( ! event . button ) // only if left mouse (button is 0) or elem.trigger('click') (button is undefined)
2016-02-04 00:03:45 +05:00
resumeModal ( event ) ;
} )
2016-01-31 11:23:07 +05:00
. on ( 'mouseup' , { route : window . location . hash , blankOnly : true } , routeOnClick )
2015-11-19 01:09:04 +05:00
. appendTo ( $ ( '#modals-minimized' ) )
;
}
2016-01-31 11:23:07 +05:00
if ( modal . is ( '.closed' ) ) return ;
2015-11-19 01:09:04 +05:00
var scroll ; // MUST be setted before modal.detach(), modal.fadeOut() and so on
2016-02-16 04:17:33 +05:00
if ( modal . is ( '.profile-modal' ) ) {
2015-11-19 01:09:04 +05:00
if ( modal . find ( '.profile-card' ) . attr ( 'data-screen-name' ) [ 0 ] === '*' )
scroll = {
targetSelector : '.modal-content .members' ,
top : modal . find ( '.modal-content .members' ) . scrollTop ( )
} ;
else
scroll = {
targetSelector : '.modal-content .postboard-posts' ,
top : modal . find ( '.modal-content .postboard-posts' ) . scrollTop ( )
} ;
2016-02-16 04:17:33 +05:00
} else
scroll = {
targetSelector : '.modal-content' ,
top : modal . find ( '.modal-content' ) . scrollTop ( )
} ;
2015-11-19 01:09:04 +05:00
if ( switchMode )
minimize ( modal , scroll ) ;
else
modal . fadeOut ( 'fast' , function ( ) {
minimize ( modal , scroll ) ;
2016-01-10 03:25:40 +05:00
_watchHashChangeRelaxDontDoIt = true ;
2015-11-19 01:09:04 +05:00
window . location . hash = '#' ;
window . scroll ( window . pageXOffset , window _scrollY ) ;
$ ( 'body' ) . css ( {
'overflow' : 'auto' ,
'margin-right' : '0'
} ) ;
} ) ;
}
function resumeModal ( event ) {
2016-01-31 11:23:07 +05:00
$ ( event . target ) . fadeOut ( 'fast' , function ( ) { this . remove ( ) ; } ) ;
2015-11-19 01:09:04 +05:00
2016-01-31 11:23:07 +05:00
var modalActive = $ ( '.modal-wrapper:not(#templates *)' ) . not ( '.closed' ) ;
2015-11-19 01:09:04 +05:00
if ( modalActive . length )
minimizeModal ( modalActive , true ) ;
else {
window _scrollY = window . pageYOffset ;
$ ( 'body' ) . css ( 'overflow' , 'hidden' ) ;
}
2016-01-31 11:23:07 +05:00
var modal = twister . modal [ event . data . hashString ] ;
if ( modal . self . not ( '.closed' ) && modal . minimized ) {
modal . minimized = false ;
modal . btnResume = undefined ;
2016-01-10 03:25:40 +05:00
if ( window . location . hash !== event . data . hashString ) {
_watchHashChangeRelaxDontDoIt = true ;
window . location . hash = event . data . hashString ;
}
2015-11-19 01:09:04 +05:00
modal . self . prependTo ( 'body' ) . fadeIn ( 'fast' , function ( ) {
// TODO also need reset modal height here maybe and then compute new scroll
if ( modal . scroll )
modal . self . find ( $ ( modal . scroll . targetSelector ) . scrollTop ( modal . scroll . top ) ) ;
2015-12-03 03:31:49 +05:00
2016-01-31 11:23:07 +05:00
if ( typeof modal . onResume === 'function' )
modal . onResume ( modal . onResumeReq ) ;
2015-11-19 01:09:04 +05:00
} ) ;
}
}
2016-01-31 11:23:07 +05:00
function focusModalWithElement ( elem , cbFunc , cbReq ) {
2015-12-03 03:31:49 +05:00
if ( elem . jquery ? elem . is ( 'html *' ) : $ ( elem ) . is ( 'html *' ) ) {
2016-01-09 00:03:11 +05:00
if ( typeof cbFunc === 'function' )
2016-01-31 11:23:07 +05:00
cbFunc ( cbReq ) ;
2015-12-03 03:31:49 +05:00
return true ;
}
2016-01-31 11:23:07 +05:00
var i = getHashOfMinimizedModalWithElem ( elem ) ;
if ( i ) {
if ( typeof i === 'object' ) i = i [ 0 ] ; // several modals, but only one may be active currently
twister . modal [ i ] . onResume = cbFunc ;
twister . modal [ i ] . onResumeReq = cbReq ;
2019-01-23 06:24:32 +05:00
twister . modal [ i ] . btnResume . trigger ( 'click' ) ;
2015-12-03 03:31:49 +05:00
return true ;
}
return false ;
}
2016-01-31 11:23:07 +05:00
function getHashOfMinimizedModalWithElem ( req ) {
var hashes = [ ] ;
for ( var i in twister . modal )
if ( twister . modal [ i ] && twister . modal [ i ] . minimized && twister . modal [ i ] . drapper . find ( req ) . length )
hashes [ hashes . length ++ ] = i ;
2015-12-03 03:31:49 +05:00
2016-01-31 11:23:07 +05:00
return hashes . length > 1 ? hashes : hashes [ 0 ] ;
2015-12-03 03:31:49 +05:00
}
function isModalWithElemExists ( elem ) {
if ( elem . jquery ? elem . is ( 'html *' ) : $ ( elem ) . is ( 'html *' ) )
return true ;
else
return getHashOfMinimizedModalWithElem ( elem ) ? true : false ;
}
2016-01-31 11:23:07 +05:00
function confirmPopup ( req ) {
if ( ! req ) return ;
2015-09-04 03:21:33 +05:00
2016-01-31 11:23:07 +05:00
if ( typeof req . stopPropagation === 'function' ) {
req . preventDefault ( ) ;
req . stopPropagation ( ) ;
if ( req . data )
req = req . data ;
else
return ;
2015-11-19 23:29:24 +05:00
}
2015-09-04 03:21:33 +05:00
var modal = openModal ( {
classBase : '.prompt-wrapper' ,
2017-02-02 14:17:10 +05:00
classAdd : req . classAdd ? 'confirm-popup ' + req . classAdd : 'confirm-popup' ,
2015-09-04 03:21:33 +05:00
content : $ ( '#confirm-popup-template' ) . children ( ) . clone ( true ) ,
2017-02-02 14:17:10 +05:00
removeBlackout : ! req . addBlackout ,
2016-01-31 11:23:07 +05:00
title : req . txtTitle
2015-09-04 03:21:33 +05:00
} ) ;
2016-02-15 15:18:06 +05:00
if ( req . txtMessage ) {
if ( req . txtMessage . polyglot )
req . txtMessage = polyglot . t ( req . txtMessage . polyglot , req . txtMessage . polyglotReq ) ;
2016-02-22 17:09:01 +05:00
fillElemWithTxt ( modal . content . find ( '.message' ) , req . txtMessage , { markout : 'apply' } ) ;
2016-02-15 15:18:06 +05:00
}
2015-09-04 03:21:33 +05:00
var btn = modal . content . find ( '.confirm' ) ;
2015-11-02 02:15:58 +05:00
if ( req . removeConfirm )
btn . remove ( ) ;
else {
2016-01-31 11:23:07 +05:00
if ( req . txtConfirm )
btn . text ( req . txtConfirm ) ;
2015-11-02 02:15:58 +05:00
else
btn . text ( polyglot . t ( 'Confirm' ) ) ;
2016-01-31 11:23:07 +05:00
if ( req . cbConfirm )
btn . on ( 'click' , { cbFunc : req . cbConfirm , cbReq : req . cbConfirmReq } , closePrompt ) ;
else
2015-11-02 02:15:58 +05:00
btn . on ( 'click' , closePrompt ) ;
}
2015-09-04 03:21:33 +05:00
var btn = modal . content . find ( '.cancel' ) ;
2015-11-02 02:15:58 +05:00
if ( req . removeCancel )
btn . remove ( ) ;
else {
2016-01-31 11:23:07 +05:00
if ( req . txtCancel )
btn . text ( req . txtCancel ) ;
2015-11-02 02:15:58 +05:00
else
btn . text ( polyglot . t ( 'Cancel' ) ) ;
2016-01-31 11:23:07 +05:00
if ( req . cbCancel )
btn . on ( 'click' , { cbFunc : req . cbCancel , cbReq : req . cbCancelReq } , closePrompt ) ;
else
2015-11-02 02:15:58 +05:00
btn . on ( 'click' , closePrompt ) ;
}
var btn = modal . self . find ( '.prompt-close' ) ;
if ( req . removeClose )
btn . remove ( ) ;
else {
2016-01-31 11:23:07 +05:00
if ( req . cbClose ) {
if ( typeof req . cbClose === 'string' )
if ( req . cbClose === 'cbConfirm' ) {
req . cbClose = req . cbConfirm ;
req . cbCloseReq = req . cbConfirmReq ;
} else if ( req . cbClose === 'cbCancel' ) {
req . cbClose = req . cbCancel ;
req . cbCloseReq = req . cbCancelReq ;
2015-11-02 02:15:58 +05:00
}
2016-01-31 11:23:07 +05:00
btn . on ( 'click' , { cbFunc : req . cbClose , cbReq : req . cbCloseReq } , closePrompt ) ;
2015-11-02 02:15:58 +05:00
}
}
2017-02-02 14:17:10 +05:00
return modal ;
2015-09-04 03:21:33 +05:00
}
2016-01-31 11:23:07 +05:00
function alertPopup ( req ) {
if ( ! req ) return ;
2016-02-15 01:05:24 +05:00
if ( typeof req . stopPropagation === 'function' ) {
if ( typeof req . data !== 'object' ) return ;
if ( ! req . data . txtConfirm )
req . data . txtConfirm = polyglot . t ( 'btn_ok' ) ;
req . data . removeCancel = true ;
} else {
if ( ! req . txtConfirm )
req . txtConfirm = polyglot . t ( 'btn_ok' ) ;
req . removeCancel = true ;
}
2016-01-31 11:23:07 +05:00
2017-02-02 14:17:10 +05:00
return confirmPopup ( req ) ;
2016-01-31 11:23:07 +05:00
}
function checkNetworkStatusAndAskRedirect ( cbFunc , cbReq ) {
networkUpdate ( function ( req ) {
2015-05-23 22:26:23 +05:00
if ( ! twisterdConnectedAndUptodate ) {
2016-01-31 11:23:07 +05:00
confirmPopup ( {
txtMessage : polyglot . t ( 'confirm_switch_to_network' , { page : '/network.html' } ) ,
cbConfirm : $ . MAL . goNetwork
2015-11-02 04:33:51 +05:00
} ) ;
2015-05-23 22:26:23 +05:00
} else {
2016-01-31 11:23:07 +05:00
if ( req . cbFunc )
req . cbFunc ( req . cbReq ) ;
2015-05-23 22:26:23 +05:00
}
2016-01-31 11:23:07 +05:00
} , { cbFunc : cbFunc , cbReq : cbReq } ) ;
2013-12-30 12:42:36 -02:00
}
function timeGmtToText ( t ) {
2016-09-22 18:14:40 +03:00
if ( t == 0 ) return '-' ;
2013-12-30 12:42:36 -02:00
var d = new Date ( 0 ) ;
d . setUTCSeconds ( t ) ;
2018-01-14 01:19:45 +05:00
return twister . var . dateFormatter . format ( d ) ;
}
function setupTimeGmtToText ( lang ) {
if ( typeof window . Intl !== 'object' || typeof window . Intl . DateTimeFormat !== 'function' )
return ;
twister . var . dateFormatter = new Intl . DateTimeFormat (
knownLanguages . indexOf ( lang ) !== - 1 ? lang : undefined ,
{
hour12 : false ,
weekday : 'short' ,
year : 'numeric' ,
month : 'short' ,
day : 'numeric' ,
hour : 'numeric' ,
minute : 'numeric' ,
second : 'numeric'
}
) ;
2013-12-30 12:42:36 -02:00
}
2014-01-05 10:29:49 -06:00
function timeSincePost ( t ) {
var d = new Date ( 0 ) ;
d . setUTCSeconds ( t ) ;
var now = new Date ( ) ;
var t _delta = Math . ceil ( ( now - d ) / 1000 ) ;
2015-05-23 22:26:23 +05:00
var expression ;
if ( t _delta < 60 )
expression = polyglot . t ( 'seconds' , t _delta ) ;
else if ( t _delta < 3600 ) // 60 * 60
expression = polyglot . t ( 'minutes' , Math . floor ( t _delta / 60 ) ) ;
else if ( t _delta < 86400 ) // 24 * 60 * 60
expression = polyglot . t ( 'hours' , Math . floor ( t _delta / 3600 ) ) ; // 60 * 60
else
expression = polyglot . t ( 'days' , Math . floor ( t _delta / 86400 ) ) ; // 24 * 60 * 60
2013-12-30 12:42:36 -02:00
2015-05-23 22:26:23 +05:00
return polyglot . t ( 'time_ago' , { time : expression } ) ;
2013-12-30 12:42:36 -02:00
}
2019-02-02 14:14:03 +05:00
function openModalAccount ( ) {
if ( ! defaultScreenName ) {
alertPopup ( {
//txtTitle: polyglot.t(''), add some title (not 'error', please) or just KISS
txtMessage : polyglot . t ( 'username_undefined' )
} ) ;
history . back ( ) ;
return ;
}
var modal = openModal ( {
classAdd : 'account-modal' ,
content : twister . tmpl . accountMC . clone ( true ) ,
title : polyglot . t ( 'Edit profile' )
} ) ;
modal . content . find ( '.alias' ) . text ( '@' + defaultScreenName ) ;
loadProfileForEdit ( defaultScreenName , {
fullname : modal . content . find ( '.input-name' ) . attr ( 'placeholder' , polyglot . t ( 'Full name here' ) ) ,
bio : modal . content . find ( '.input-bio' ) . attr ( 'placeholder' , polyglot . t ( 'Describe yourself' ) ) ,
location : modal . content . find ( '.input-location' ) . attr ( 'placeholder' , polyglot . t ( 'Location' ) ) ,
url : modal . content . find ( '.input-url' ) . attr ( 'placeholder' , polyglot . t ( 'website' ) ) ,
tox : modal . content . find ( '.input-tox' ) . attr ( 'placeholder' , polyglot . t ( 'Tox address' ) ) ,
bitmessage : modal . content . find ( '.input-bitmessage' ) . attr ( 'placeholder' , polyglot . t ( 'Bitmessage address' ) )
} ) ;
loadAvatarForEdit ( defaultScreenName , modal . content . find ( '.avatar img' ) ) ;
dumpPubkey ( defaultScreenName , checkAccountRegistrationCB , {
peerAlias : defaultScreenName ,
cbFunc : checkAccountRegistrationCB ,
submitChangesElem : modal . content . find ( '.submit-changes' )
} ) ;
}
function checkAccountRegistrationCB ( req , res ) {
if ( res . length > 0 ) {
if ( followingUsers . indexOf ( 'twister' ) !== - 1 ) {
if ( ! isModalWithElemExists ( req . submitChangesElem ) )
return ;
req . submitChangesElem . attr ( 'data-blocked' , 'false' ) ;
if ( req . submitChangesElem . attr ( 'data-profile-changed' ) === 'true'
|| req . submitChangesElem . attr ( 'data-avatar-changed' ) === 'true' )
$ . MAL . enableButton ( req . submitChangesElem ) ;
} else {
follow ( 'twister' , true ,
function ( req ) {
// TODO add alertPopup() with welcome message
if ( ! isModalWithElemExists ( req ) )
return ;
req . attr ( 'data-blocked' , 'false' ) ;
if ( req . attr ( 'data-profile-changed' ) === 'true'
|| req . attr ( 'data-avatar-changed' ) === 'true' )
$ . MAL . enableButton ( req ) ;
} ,
req . submitChangesElem
) ;
}
} else {
if ( ! req . notYetAcceptedWarnDisplayed ) {
req . notYetAcceptedWarnDisplayed = true ;
alertPopup ( {
//txtTitle: polyglot.t(''), add some title or just KISS
txtMessage : polyglot . t ( 'user_not_yet_accepted' )
} ) ;
}
setTimeout ( dumpPubkey , 5000 , req . peerAlias , req . cbFunc , req ) ;
}
}
function handleAvatarFileSelect ( event , avatarElem ) {
if ( ! window . File || ! window . FileReader || ! window . FileList || ! window . Blob ) {
alert ( 'The File APIs are not fully supported in this browser.' ) ;
return ;
}
if ( ! event || ! event . target || ! event . target . files || ! event . target . files . length
|| ! avatarElem || ! avatarElem . length )
return ;
var file = event . target . files [ 0 ] ;
if ( ! file . type . startsWith ( 'image/' ) )
return ;
var reader = new FileReader ( ) ;
reader . data = avatarElem ;
reader . onload = function ( event ) {
var img = document . createElement ( 'img' ) ;
img . data = event . target . data ;
img . onload = function ( event ) {
var ratio = 64 / Math . max ( event . target . width , event . target . height ) ;
var canvas = document . createElement ( 'canvas' ) ;
canvas . width = Math . round ( event . target . width * ratio ) ;
canvas . height = Math . round ( event . target . height * ratio ) ;
canvas . getContext ( '2d' ) . drawImage ( event . target , 0 , 0 , canvas . width , canvas . height ) ;
var imgURL ;
for ( var quality = 1.0 ; ( ! imgURL || imgURL . length > 4096 ) && quality > 0.1 ; quality -= 0.01 )
imgURL = canvas . toDataURL ( 'image/jpeg' , quality ) ;
event . target . data . attr ( 'src' , imgURL ) ;
} ;
img . src = event . target . result ;
} ;
reader . readAsDataURL ( file ) ;
}
2017-02-02 14:17:10 +05:00
function openModalLogin ( ) {
var modal = openModal ( {
classAdd : 'login-modal' ,
content : twister . tmpl . loginMC . clone ( true ) ,
title : polyglot . t ( 'twister login' )
} ) ;
}
function handleClickAccountLoginLogin ( event ) {
loginToAccount ( $ ( event . target ) . closest ( '.module' ) . find ( 'select.local-usernames' ) . val ( ) ) ;
}
function handleInputAccountCreateSetReq ( event ) {
var container = $ ( event . target ) . closest ( '.module' ) ;
container . find ( '.availability' ) . text ( '' ) ;
$ . MAL . enableButton ( container . find ( '.check' ) ) ;
$ . MAL . disableButton ( container . find ( '.create' ) ) ;
}
function handleClickAccountCreateCheckReq ( event ) {
var container = $ ( event . target ) . closest ( '.module' ) ;
var peerAliasElem = container . find ( '.alias' ) ;
var peerAlias = peerAliasElem . val ( ) . toLowerCase ( ) ;
var availField = container . find ( '.availability' ) ;
peerAliasElem . val ( peerAlias ) ;
$ . MAL . disableButton ( container . find ( '.check' ) ) ;
if ( ! peerAlias . length )
return ;
if ( peerAlias . length > 16 ) {
availField . text ( polyglot . t ( 'Must be 16 characters or less.' ) ) ;
return ;
}
// check for non-alphabetic characters and space
if ( peerAlias . search ( /[^a-z0-9_]/ ) !== - 1 ) {
availField . text ( polyglot . t ( 'Only alphanumeric and underscore allowed.' ) ) ;
return ;
}
availField . text ( polyglot . t ( 'Checking...' ) ) ;
dumpPubkey ( peerAlias ,
function ( req , ret ) {
if ( ret ) {
req . container . find ( '.availability' ) . text ( polyglot . t ( 'Not available' ) ) ;
} else {
req . container . find ( '.availability' ) . text ( polyglot . t ( 'Available' ) ) ;
$ . MAL . enableButton ( req . container . find ( '.create' ) ) ;
}
} , { container : container }
) ;
}
function handleClickAccountCreateCreate ( event ) {
var container = $ ( event . target ) . closest ( '.module' ) ;
var peerAlias = container . find ( '.alias' ) . val ( ) . toLowerCase ( ) ;
if ( twister . var . localAccounts . indexOf ( peerAlias ) < 0 ) {
createAccount ( peerAlias ) ;
} else {
// user exists in wallet but transaction not sent
dumpPrivkey ( peerAlias ,
function ( req , ret ) {
$ . MAL . processCreateAccount ( req . peerAlias , ret ) ;
} , { peerAlias : peerAlias }
) ;
}
}
function handleInputAccountImportSetReq ( event ) {
var container = $ ( event . target ) . closest ( '.module' ) ;
if ( container . find ( '.secret-key' ) . val ( ) . length === 52
&& container . find ( '.alias' ) . val ( ) . toLowerCase ( ) . length )
$ . MAL . enableButton ( container . find ( '.import' ) ) ;
else
$ . MAL . disableButton ( container . find ( '.import' ) ) ;
}
function handleClickAccountImportImport ( event ) {
var container = $ ( event . target ) . closest ( '.module' ) ;
importAccount ( container . find ( '.alias' ) . val ( ) . toLowerCase ( ) , container . find ( '.secret-key' ) . val ( ) )
}
2015-09-04 03:21:33 +05:00
function openGroupProfileModalWithNameHandler ( groupAlias ) {
var modal = openModal ( {
classAdd : 'profile-modal' ,
content : $ ( '#group-profile-modal-template' ) . children ( ) . clone ( true ) ,
title : polyglot . t ( 'users_profile' , { username : '<span>' + groupAlias + '</span>' } )
} ) ;
modal . content . find ( '.profile-card' ) . attr ( 'data-screen-name' , groupAlias ) ;
groupMsgGetGroupInfo ( groupAlias ,
function ( req , ret ) {
if ( ret ) {
2015-10-24 01:20:41 +05:00
req . modal . content . find ( '.profile-bio .group-description' )
. val ( ret . description )
. attr ( 'val-origin' , ret . description )
;
2015-09-04 03:21:33 +05:00
if ( ret . members . indexOf ( defaultScreenName ) !== - 1 )
2019-01-23 06:24:32 +05:00
req . modal . content . find ( '.group-messages-control' ) . children ( 'button' ) . prop ( 'disabled' , false ) ;
2015-09-04 03:21:33 +05:00
var membersList = req . modal . content . find ( '.members' ) ;
var memberTemplate = $ ( '#group-profile-member-template' ) . children ( ) ;
for ( var i = 0 ; i < ret . members . length ; i ++ ) {
var item = memberTemplate . clone ( true ) . appendTo ( membersList ) ;
item . find ( '.twister-user-info' ) . attr ( 'data-screen-name' , ret . members [ i ] ) ;
item . find ( '.twister-user-name' ) . attr ( 'href' , $ . MAL . userUrl ( ret . members [ i ] ) ) ;
getAvatar ( ret . members [ i ] , item . find ( '.twister-user-photo' ) ) ;
getFullname ( ret . members [ i ] , item . find ( '.twister-user-full' ) ) ;
2015-10-27 21:31:04 +05:00
getBioToElem ( ret . members [ i ] , item . find ( '.bio' ) ) ;
2015-09-04 03:21:33 +05:00
}
2015-10-22 02:05:18 +05:00
elemFitNextIntoParentHeight ( req . modal . content . find ( '.profile-card' ) ) ;
2015-09-04 03:21:33 +05:00
}
} , { modal : modal }
) ;
2015-10-22 02:05:18 +05:00
elemFitNextIntoParentHeight ( modal . content . find ( '.profile-card' ) ) ;
2015-09-04 03:21:33 +05:00
}
2016-01-31 11:23:07 +05:00
function openUserProfileModalWithNameHandler ( peerAlias ) {
2015-05-23 22:26:23 +05:00
var content = $ ( '#profile-modal-template' ) . children ( ) . clone ( true ) ;
2013-12-30 12:42:36 -02:00
2016-01-31 11:23:07 +05:00
updateProfileData ( content , peerAlias ) ;
2015-05-23 22:26:23 +05:00
// FIXME following ctc could be part of updateProfileData() when mobile will be ready for this
content . find ( '.tox-ctc' ) . attr ( 'title' , polyglot . t ( 'Copy to clipboard' ) ) ;
content . find ( '.bitmessage-ctc' ) . attr ( 'title' , polyglot . t ( 'Copy to clipboard' ) ) ;
2013-12-30 12:42:36 -02:00
2016-07-31 02:37:52 +05:00
content . find ( '.open-followers' ) . on ( 'mouseup' , { route : '#followers?user=' + peerAlias } , routeOnClick ) ;
2015-06-12 03:34:59 +05:00
var modal = openModal ( {
2015-05-23 22:26:23 +05:00
classAdd : 'profile-modal' ,
content : content ,
2016-01-31 11:23:07 +05:00
title : polyglot . t ( 'users_profile' , { username : peerAlias } )
2015-06-12 03:34:59 +05:00
} ) ;
2014-10-16 10:47:08 +02:00
2016-01-31 11:23:07 +05:00
toggleFollowButton ( {
button : modal . content . find ( '.profile-card-buttons .follow' ) ,
peerAlias : peerAlias ,
toggleUnfollow : followingUsers . indexOf ( peerAlias ) !== - 1 ? true : false
} ) ;
2014-10-16 10:47:08 +02:00
2015-10-22 02:05:18 +05:00
elemFitNextIntoParentHeight ( modal . content . find ( '.profile-card' ) ) ;
2015-06-12 03:34:59 +05:00
var postboard = modal . content . find ( '.postboard' ) ;
2015-10-22 02:05:18 +05:00
postboard . find ( 'ol' ) . outerHeight ( postboard . actual ( 'height' )
- postboard . find ( 'h2' ) . actual ( 'outerHeight' , { includeMargin : true } ) ) ;
2013-12-30 12:42:36 -02:00
}
2015-05-23 22:26:23 +05:00
function openHashtagModalFromSearchHandler ( hashtag ) {
var modal = openModal ( {
classAdd : 'hashtag-modal' ,
content : $ ( '#hashtag-modal-template' ) . children ( ) . clone ( true ) ,
title : '#' + hashtag
2013-12-30 12:42:36 -02:00
} ) ;
2017-06-17 23:59:30 +05:00
var req = queryStart ( modal . content . find ( '.postboard-posts' ) , hashtag , 'hashtag' ) ;
modal . content . find ( '.postboard-news' ) . on ( 'click' , { req : req } , handleClickDisplayPendingTwists ) ;
2013-12-30 12:42:36 -02:00
}
2017-06-17 23:59:30 +05:00
function handleClickDisplayPendingTwists ( event ) {
if ( ! event || ! event . data || ! event . data . req )
return ;
2015-12-03 03:31:49 +05:00
2017-06-17 23:59:30 +05:00
$ ( event . target ) . hide ( ) ;
2015-12-03 03:31:49 +05:00
2017-06-17 23:59:30 +05:00
queryPendingDraw ( event . data . req ) ;
2015-12-03 03:31:49 +05:00
2017-06-17 23:59:30 +05:00
if ( typeof event . data . cbFunc === 'function' )
event . data . cbFunc ( event . data . cbReq ) ;
2013-12-30 12:42:36 -02:00
}
2016-09-06 17:49:29 +03:00
function openFavsModal ( event ) {
if ( event && typeof event . stopPropagation === 'function' ) {
event . preventDefault ( ) ;
event . stopPropagation ( ) ;
}
var userInfo = $ ( this ) . closest ( '[data-screen-name]' ) ;
var peerAlias = '' ;
if ( userInfo . length )
peerAlias = userInfo . attr ( 'data-screen-name' ) ;
else if ( defaultScreenName )
peerAlias = defaultScreenName ;
else {
alertPopup ( {
//txtTitle: polyglot.t(''), add some title (not 'error', please) or just KISS
txtMessage : polyglot . t ( 'No favs here because you are not logged in.' )
} ) ;
return ;
}
window . location . hash = '#favs?user=' + peerAlias ;
}
function openFavsModalHandler ( peerAlias ) {
var modal = openModal ( {
classAdd : 'hashtag-modal' ,
content : $ ( '#hashtag-modal-template' ) . children ( ) . clone ( true ) ,
title : polyglot . t ( 'users_favs' , { username : peerAlias } )
} ) ;
2017-06-17 23:59:30 +05:00
var req = queryStart ( modal . content . find ( '.postboard-posts' ) , peerAlias , 'fav' ) ;
modal . content . find ( '.postboard-news' ) . on ( 'click' , { req : req } , handleClickDisplayPendingTwists ) ;
2016-09-06 17:49:29 +03:00
}
2016-01-31 11:23:07 +05:00
function openMentionsModal ( event ) {
if ( event && typeof event . stopPropagation === 'function' ) {
event . preventDefault ( ) ;
event . stopPropagation ( ) ;
2015-02-16 12:32:12 +05:00
}
2013-12-30 12:42:36 -02:00
2015-05-23 22:26:23 +05:00
var userInfo = $ ( this ) . closest ( '[data-screen-name]' ) ;
if ( userInfo . length )
2016-01-31 11:23:07 +05:00
var peerAlias = userInfo . attr ( 'data-screen-name' ) ;
2015-05-23 22:26:23 +05:00
else if ( defaultScreenName )
2016-01-31 11:23:07 +05:00
var peerAlias = defaultScreenName ;
2015-05-23 22:26:23 +05:00
else {
2016-01-31 11:23:07 +05:00
alertPopup ( {
//txtTitle: polyglot.t(''), add some title (not 'error', please) or just KISS
txtMessage : polyglot . t ( 'No one can mention you because you are not logged in.' )
} ) ;
2015-05-19 23:33:59 +05:00
return ;
2014-03-29 18:16:24 +01:00
}
2013-12-30 12:42:36 -02:00
2016-01-31 11:23:07 +05:00
window . location . hash = '#mentions?user=' + peerAlias ;
2014-11-14 18:39:33 -02:00
}
2016-01-31 11:23:07 +05:00
function openMentionsModalHandler ( peerAlias ) {
2015-05-23 22:26:23 +05:00
var modal = openModal ( {
classAdd : 'hashtag-modal' ,
content : $ ( '#hashtag-modal-template' ) . children ( ) . clone ( true ) ,
2016-01-31 11:23:07 +05:00
title : polyglot . t ( 'users_mentions' , { username : peerAlias } )
2015-05-23 22:26:23 +05:00
} ) ;
2013-12-30 12:42:36 -02:00
2017-06-17 23:59:30 +05:00
var req = queryStart ( modal . content . find ( '.postboard-posts' ) , peerAlias , 'mention' ) ;
2017-07-14 00:56:30 +05:00
modal . content . find ( '.postboard-news' ) . on ( 'click' , { req : req } , handleClickDisplayPendingTwists ) ;
2013-12-30 12:42:36 -02:00
2017-07-14 00:56:30 +05:00
if ( peerAlias === defaultScreenName )
2017-07-03 22:36:30 +05:00
modal . content . on ( 'scroll' , handleMentionsModalScroll ) ;
2013-12-30 12:42:36 -02:00
}
2016-02-15 22:27:37 +05:00
function openFollowersModal ( peerAlias ) {
2016-09-11 19:42:36 +05:00
var followers , title , warn ;
2016-02-16 03:35:34 +05:00
if ( ! peerAlias || peerAlias === defaultScreenName ) {
if ( ! defaultScreenName ) {
alertPopup ( {
//txtTitle: polyglot.t(''), add some title (not 'error', please) or just KISS
txtMessage : polyglot . t ( 'You don\'t have any followers because you are not logged in.' )
} ) ;
history . back ( ) ;
return ;
}
title = polyglot . t ( 'Followers' ) ;
followers = twisterFollowingO . knownFollowers . slice ( ) ;
2016-09-11 19:42:36 +05:00
warn = {
name : 'FollowersNotAll' ,
text : '* ' + polyglot . t ( 'warn_followers_not_all' )
} ;
2016-02-16 03:35:34 +05:00
} else {
title = polyglot . t ( 'Followers_of' , { alias : peerAlias } ) ;
followers = whoFollows ( peerAlias ) ;
2016-09-11 19:42:36 +05:00
warn = {
name : 'FollowersNotAllOf' ,
text : polyglot . t ( 'warn_followers_not_all_of' , { alias : peerAlias } )
} ;
2016-02-15 22:27:37 +05:00
}
var modal = openModal ( {
classAdd : 'followers-modal' ,
content : twister . tmpl . followersList . clone ( true ) ,
2016-09-10 18:14:08 +03:00
title : title ,
2016-09-11 19:42:36 +05:00
warn : warn
2016-02-15 22:27:37 +05:00
} ) ;
2016-02-16 03:35:34 +05:00
appendFollowersToElem ( modal . content . find ( 'ol' ) , followers ) ;
2016-02-15 22:27:37 +05:00
}
2016-02-16 03:35:34 +05:00
function appendFollowersToElem ( list , followers ) {
for ( var i = 0 ; i < followers . length ; i ++ )
addPeerToFollowersList ( list , followers [ i ] ) ;
2016-02-15 22:27:37 +05:00
$ . MAL . listLoaded ( list ) ;
}
2016-02-15 23:53:12 +05:00
function addPeerToFollowersList ( list , peerAlias , isCheckNeeded ) {
if ( typeof list !== 'object' || ! list . length )
return ;
if ( isCheckNeeded && list . find ( 'li[data-peer-alias="' + peerAlias + '"]' ) . length )
return ;
2016-02-15 22:27:37 +05:00
var item = twister . tmpl . followersPeer . clone ( true ) . attr ( 'data-peer-alias' , peerAlias ) ;
item . find ( '.alias' ) . text ( '@' + peerAlias ) ;
item . find ( '.alias, .avatar, .name' ) . on ( 'mouseup' , { route : $ . MAL . userUrl ( peerAlias ) } , routeOnClick ) ;
getAvatar ( peerAlias , item . find ( '.avatar img' ) ) ;
getFullname ( peerAlias , item . find ( '.name' ) ) ;
getBioToElem ( peerAlias , item . find ( '.bio' ) ) ;
item . prependTo ( list ) ;
}
2016-01-31 11:23:07 +05:00
function openFollowingModal ( peerAlias ) {
if ( ! peerAlias || peerAlias === defaultScreenName ) {
if ( ! defaultScreenName ) {
alertPopup ( {
//txtTitle: polyglot.t(''), add some title (not 'error', please) or just KISS
txtMessage : polyglot . t ( 'You are not following anyone because you are not logged in.' )
} ) ;
history . back ( ) ;
return ;
}
2014-01-05 12:12:34 -06:00
2016-01-31 11:23:07 +05:00
var modal = openModal ( {
classAdd : 'following-own-modal' ,
content : twister . tmpl . followingList . clone ( true ) ,
title : polyglot . t ( 'Following' )
} ) ;
2016-02-15 22:56:05 +05:00
appendFollowingToElem ( modal . content . find ( '.following-list' ) ) ;
2016-01-31 11:23:07 +05:00
requestSwarmProgress ( ) ;
} else {
var modal = openModal ( {
classAdd : 'following-modal' ,
content : $ ( '#following-modal-template' ) . children ( ) . clone ( true ) ,
title : polyglot . t ( 'followed_by' , { username : peerAlias } )
} ) ;
modal . content . find ( '.following-screen-name b' ) . text ( peerAlias ) ;
loadFollowingIntoList ( peerAlias , modal . content . find ( 'ol' ) ) ;
}
}
2014-01-05 12:12:34 -06:00
2016-02-15 22:56:05 +05:00
function appendFollowingToElem ( list ) {
2016-01-31 11:23:07 +05:00
if ( followingEmptyOrMyself ( ) )
2016-02-15 22:56:05 +05:00
$ . MAL . warnFollowingNotAny ( closeModal , list ) ;
2016-01-31 11:23:07 +05:00
else
for ( var i = 0 ; i < followingUsers . length ; i ++ )
2016-02-15 22:56:05 +05:00
addPeerToFollowingList ( list , followingUsers [ i ] ) ;
2016-01-31 11:23:07 +05:00
2016-02-15 22:56:05 +05:00
$ . MAL . listLoaded ( list ) ;
2016-01-31 11:23:07 +05:00
}
2016-02-15 22:56:05 +05:00
function addPeerToFollowingList ( list , peerAlias ) {
var item = twister . tmpl . followingPeer . clone ( true ) . attr ( 'data-peer-alias' , peerAlias ) ;
2016-01-31 11:23:07 +05:00
item . find ( '.mini-profile-info' ) . attr ( 'data-screen-name' , peerAlias )
item . find ( '.following-screen-name' ) . text ( peerAlias ) ;
item . find ( 'a.open-profile-modal' ) . attr ( 'href' , $ . MAL . userUrl ( peerAlias ) ) ;
item . find ( '.direct-messages-with-user' ) . text ( polyglot . t ( 'send_DM' ) )
. on ( 'mouseup' , { route : $ . MAL . dmchatUrl ( peerAlias ) } , routeOnClick ) ;
item . find ( '.mentions-from-user' ) . text ( polyglot . t ( 'display_mentions' ) )
. on ( 'mouseup' , { route : $ . MAL . mentionsUrl ( peerAlias ) } , routeOnClick ) ;
getAvatar ( peerAlias , item . find ( '.mini-profile-photo' ) ) ;
getFullname ( peerAlias , item . find ( '.mini-profile-name' ) ) ;
2016-09-27 04:04:35 +05:00
getStatusTime ( peerAlias , item . find ( '.latest-activity .time' ) ) ;
2016-01-31 11:23:07 +05:00
if ( peerAlias === defaultScreenName )
2016-02-09 23:20:39 +05:00
item . find ( '.following-config' ) . hide ( ) ;
2016-01-31 11:23:07 +05:00
toggleFollowButton ( {
button : item . find ( '.follow' ) ,
peerAlias : peerAlias ,
toggleUnfollow : true
2015-05-23 22:26:23 +05:00
} ) ;
2016-01-31 11:23:07 +05:00
var elem = item . find ( '.public-following' ) . on ( 'click' , followingListPublicCheckbox ) ;
if ( isPublicFollowing ( peerAlias ) )
elem . text ( polyglot . t ( 'Public' ) ) ;
else
elem . text ( polyglot . t ( 'Private' ) ) . addClass ( 'private' ) ;
2016-02-15 22:56:05 +05:00
item . prependTo ( list ) ;
2014-05-05 22:28:15 +03:00
}
function fillWhoToFollowModal ( list , hlist , start ) {
2015-05-23 22:26:23 +05:00
for ( var i = 0 ; i < followingUsers . length && list . length < start + 20 ; i ++ ) {
if ( typeof twisterFollowingO . followingsFollowings [ followingUsers [ i ] ] !== 'undefined' ) {
for ( var j = 0 ; j < twisterFollowingO . followingsFollowings [ followingUsers [ i ] ] . following . length && list . length < start + 25 ; j ++ ) {
var utf = twisterFollowingO . followingsFollowings [ followingUsers [ i ] ] . following [ j ] ;
if ( followingUsers . indexOf ( utf ) < 0 && list . indexOf ( utf ) < 0 ) {
2014-05-09 12:27:48 +03:00
list . push ( utf ) ;
2014-05-05 22:28:15 +03:00
2016-11-19 14:16:33 +03:00
processWhoToFollowSuggestion ( hlist , utf , followingUsers [ i ] ) ;
2014-05-05 22:28:15 +03:00
}
}
}
}
2014-05-06 14:43:39 +03:00
2015-05-23 22:26:23 +05:00
if ( i >= followingUsers . length - 1 )
2014-05-06 14:43:39 +03:00
return false ;
2015-05-23 22:26:23 +05:00
2014-05-06 14:43:39 +03:00
// returns true, if there are more...
return true ;
2014-05-05 22:28:15 +03:00
}
2015-01-22 15:59:00 +01:00
function openWhoToFollowModal ( ) {
2015-05-23 22:26:23 +05:00
var modal = openModal ( {
classAdd : 'who-to-follow-modal' ,
title : polyglot . t ( 'Who to Follow' )
} ) ;
2014-05-05 22:28:15 +03:00
var tmplist = [ ] ;
2015-05-23 22:26:23 +05:00
var hlist = $ ( '<ol class="follow-suggestions"></ol>' )
. appendTo ( modal . content ) ;
2014-05-05 22:28:15 +03:00
2015-05-23 22:26:23 +05:00
modal . content . on ( 'scroll' , function ( ) {
if ( modal . content . scrollTop ( ) >= hlist . height ( ) - modal . content . height ( ) - 20 ) {
2016-11-19 14:16:33 +03:00
if ( ! fillWhoToFollowModal ( tmplist , modal . self , tmplist . length ) )
2015-05-23 22:26:23 +05:00
modal . content . off ( 'scroll' ) ;
2014-05-05 22:28:15 +03:00
}
} ) ;
2016-11-19 14:16:33 +03:00
fillWhoToFollowModal ( tmplist , modal . self , 0 ) ;
}
function openNewUsersModal ( ) {
var modal = openModal ( {
classAdd : 'new-users-modal' ,
2016-11-24 15:54:33 +03:00
title : polyglot . t ( 'New Users' ) ,
onClose : function ( ) {
NewUserSearch . isNewUserModalOpen = false ;
}
2016-11-19 14:16:33 +03:00
} ) ;
var hlist = $ ( '<ol class="follow-suggestions"></ol>' )
. appendTo ( modal . content ) ;
2016-11-19 18:31:47 +03:00
var count = 15 ;
2016-11-19 14:16:33 +03:00
modal . content . on ( 'scroll' , function ( ) {
if ( modal . content . scrollTop ( ) >= hlist . height ( ) - modal . content . height ( ) - 20 ) {
2016-11-24 15:54:33 +03:00
if ( newUsers . getLastNUsers ( 5 , count , modal . self ) )
2016-11-19 18:31:47 +03:00
count += 5 ;
2016-11-19 14:16:33 +03:00
}
} ) ;
2016-11-24 15:54:33 +03:00
NewUserSearch . isNewUserModalOpen = true ;
newUsers . getLastNUsers ( 15 , 0 , modal . self ) ;
2014-05-05 22:28:15 +03:00
}
2016-09-06 13:54:11 +05:00
function openModalUriShortener ( )
{
var modal = openModal ( {
classAdd : 'uri-shortener-modal' ,
content : twister . tmpl . uriShortenerMC . clone ( true ) ,
title : polyglot . t ( 'URI_shortener' )
} ) ;
modal . content . find ( '.uri-shortener-control .shorten-uri' ) . text ( polyglot . t ( 'shorten_URI' ) ) ;
modal . content . find ( '.uri-shortener-control .clear-cache' ) . text ( polyglot . t ( 'clear_cache' ) ) ;
var urisList = modal . content . find ( '.uris-list' ) ;
//var i = 0;
for ( var short in twister . URIs ) {
//i++;
var long = twister . URIs [ short ] instanceof Array ? twister . URIs [ short ] [ 0 ] : twister . URIs [ short ] ;
var item = twister . tmpl . uriShortenerUrisListItem . clone ( true ) ;
item . find ( '.short' ) . text ( short ) ;
item . find ( '.long' ) . text ( long ) . attr ( 'href' , long ) ;
item . appendTo ( urisList ) ;
}
//i + URIs are cached
}
2016-01-31 11:23:07 +05:00
function newConversationModal ( peerAlias , resource ) {
2015-05-23 22:26:23 +05:00
var content = $ ( '#hashtag-modal-template' ) . children ( ) . clone ( true ) ;
2015-05-19 23:33:59 +05:00
2016-01-31 11:23:07 +05:00
requestPost ( content . find ( '.postboard-posts' ) , peerAlias , resource ,
2015-05-23 22:26:23 +05:00
function ( args ) {
var postboard = args . content . find ( '.postboard-posts' ) ;
var postLi = postboard . children ( ) . first ( )
. css ( 'display' , 'none' ) ;
getTopPostOfConversation ( postLi , null , postboard ) ;
2015-12-03 03:31:49 +05:00
} , { content : content }
2015-05-23 22:26:23 +05:00
) ;
2015-01-23 09:08:52 +01:00
2015-05-23 22:26:23 +05:00
return content ;
2014-05-13 19:46:35 +03:00
}
2016-02-03 22:04:53 +05:00
function addToCommonDMsList ( list , targetAlias , message ) {
var item = twister . tmpl . commonDMsListItem . clone ( true )
. attr ( 'data-screen-name' , targetAlias )
. attr ( 'data-last_id' , message . id )
. attr ( 'data-time' , message . time )
;
item . find ( '.post-info-tag' ) . text ( '@' + targetAlias ) ;
item . find ( '.post-info-name' ) . attr ( 'href' , $ . MAL . userUrl ( targetAlias ) ) ;
2016-02-22 17:09:01 +05:00
fillElemWithTxt ( item . find ( '.post-text' ) , message . text ) ;
2016-02-03 22:04:53 +05:00
item . find ( '.post-info-time' )
. attr ( 'title' , timeSincePost ( message . time ) )
. find ( 'span:last' )
. text ( timeGmtToText ( message . time ) )
;
if ( targetAlias [ 0 ] === '*' ) {
2019-02-02 14:14:03 +05:00
getAvatar ( message . from , // it's impossible yet to get or to set an avatar of a group itself
item . find ( '.post-photo' ) . attr ( 'data-peer-alias' , message . from ) . find ( 'img' ) ) ;
2016-02-03 22:04:53 +05:00
getGroupChatName ( targetAlias , item . find ( 'a.post-info-name' ) ) ;
} else {
getAvatar ( targetAlias , item . find ( '.post-photo img' ) ) ;
getFullname ( targetAlias , item . find ( 'a.post-info-name' ) ) ;
}
2017-07-14 00:56:30 +05:00
if ( twister . DMs [ targetAlias ] . lengthNew > 0 )
2016-02-03 22:04:53 +05:00
item . addClass ( 'new' )
2017-07-14 00:56:30 +05:00
. find ( '.messages-qtd' ) . text ( twister . DMs [ targetAlias ] . lengthNew ) . show ( ) ;
2016-02-03 22:04:53 +05:00
var items = list . children ( ) ;
for ( var i = 0 ; i < items . length ; i ++ ) {
var elem = items . eq ( i ) ;
var time = elem . attr ( 'data-time' ) ;
if ( typeof time === 'undefined' || message . time > parseInt ( time ) ) {
elem . before ( item ) ;
break ;
}
}
if ( i === items . length ) // equals to !item.parent().length
list . append ( item ) ;
}
2016-03-23 23:51:12 +05:00
function handleClickOpenProfileModal ( event ) {
2016-03-25 04:06:15 +05:00
event . data = { route : $ ( this ) . attr ( 'href' ) } ;
2016-03-23 23:51:12 +05:00
routeOnClick ( event ) ;
}
2016-03-23 23:56:36 +05:00
function handleClickOpenConversation ( event ) {
2016-09-27 04:21:38 +05:00
var elem = $ ( event . target ) . closest ( event . data . feeder ) ;
if ( ! elem . length ) {
muteEvent ( event , true ) ;
return ;
}
2015-01-23 09:48:30 +01:00
2016-09-27 04:21:38 +05:00
var post = {
writer : elem . attr ( 'data-screen-name' ) ,
id : elem . attr ( 'data-id' )
} ;
if ( ! post . writer || ! post . id ) {
muteEvent ( event , true ) ;
return ;
}
2015-01-23 09:48:30 +01:00
2016-09-27 04:21:38 +05:00
event . data . route = '#conversation?post=' + post . writer + ':post' + post . id ;
2016-01-31 11:23:07 +05:00
routeOnClick ( event ) ;
2015-01-23 09:48:30 +01:00
}
2016-01-31 11:23:07 +05:00
function openConversationModal ( peerAlias , resource ) {
2015-05-23 22:26:23 +05:00
openModal ( {
classAdd : 'conversation-modal' ,
2016-01-31 11:23:07 +05:00
content : newConversationModal ( peerAlias , resource ) ,
title : polyglot . t ( 'conversation_title' , { username : peerAlias } )
2015-05-23 22:26:23 +05:00
} ) ;
2014-05-13 19:46:35 +03:00
}
2016-03-23 04:02:20 +05:00
function openRequestShortURIForm ( event ) {
if ( event ) muteEvent ( event ) ;
2016-03-22 01:54:56 +05:00
if ( ! defaultScreenName ) {
alertPopup ( {
//txtTitle: polyglot.t(''), add some title (not 'error', please) or just KISS
2016-03-22 23:12:34 +05:00
txtMessage : 'You can\'t shorten links because you are not logged in.'
2016-03-22 01:54:56 +05:00
} ) ;
2016-03-22 17:43:04 +05:00
return ;
2016-03-22 01:54:56 +05:00
}
2016-03-22 17:43:04 +05:00
if ( parseInt ( twisterVersion ) < 93500 ) {
alertPopup ( {
//txtTitle: polyglot.t(''), add some title (not 'error', please) or just KISS
2016-10-06 23:44:15 +05:00
txtMessage : 'You can\'t shorten links —\n'
+ polyglot . t ( 'daemon_is_obsolete' , { versionReq : '0.9.35' } )
2016-03-22 17:43:04 +05:00
} ) ;
return ;
}
2016-03-23 21:46:02 +05:00
var uri = prompt ( polyglot . t ( 'shorten_URI_enter_link' ) ) ; // FIXME replace native prompt
2016-03-23 04:02:20 +05:00
if ( event && event . data && typeof event . data . cbFunc === 'function' )
newShortURI ( uri , event . data . cbFunc , event . data . cbReq ) ;
else
newShortURI ( uri , showURIPair ) ;
}
function showURIPair ( uriLong , uriShort ) { // FIXME req
if ( uriShort )
alertPopup ( {
2017-02-02 20:15:30 +05:00
txtTitle : polyglot . t ( 'URI_shortener' ) ,
2016-03-23 04:02:20 +05:00
txtMessage : uriLong + ' — `' + uriShort + '`'
} ) ;
else
showURIShortenerErrorRPC ( uriShort ) ;
}
function showURIShortenerErrorRPC ( ret ) {
alertPopup ( {
2017-02-02 20:15:30 +05:00
txtTitle : polyglot . t ( 'URI_shortener' ) ,
2016-03-23 04:02:20 +05:00
txtMessage : 'something went wrong. RPC error message:\n' + ( ret && ret . message ? ret . message : ret )
} ) ;
2016-03-22 01:54:56 +05:00
}
2016-02-22 16:40:04 +05:00
function fillElemWithTxt ( elem , txt , htmlFormatMsgOpt ) {
var formatted = htmlFormatMsg ( txt , htmlFormatMsgOpt ) ;
elem . html ( formatted . html ) ;
2016-02-23 01:08:01 +05:00
elem . find ( 'a' ) . each ( function ( i , elem ) {
var href = elem . getAttribute ( 'href' ) ;
2016-03-20 22:44:57 +05:00
if ( elem . classList . contains ( 'link-shortened' ) ) {
2016-05-06 16:40:12 +05:00
$ ( elem ) . on ( 'click mouseup' , { href : href } ,
function ( event ) {
muteEvent ( event , true ) ;
fetchShortenedURI ( event . data . href ) ;
}
) ;
2016-03-20 22:44:57 +05:00
fetchShortenedURI ( href ) ;
} else if ( href && href [ 0 ] === '#' )
2016-02-23 01:08:01 +05:00
$ ( elem )
. on ( 'click' , { preventDefault : true } , muteEvent )
. on ( 'mouseup' , { route : href } , routeOnClick )
;
else
$ ( elem ) . on ( 'click mouseup' , muteEvent ) ;
} ) ;
2016-02-22 16:40:04 +05:00
return formatted ;
}
2016-05-05 09:14:35 +05:00
function fetchShortenedURI ( req , attemptCount ) {
2016-03-21 00:55:32 +05:00
if ( twister . URIs [ req ] ) {
applyShortenedURI ( req , twister . URIs [ req ] ) ;
return ;
}
2016-03-21 22:58:42 +05:00
decodeShortURI ( req ,
2016-03-21 00:55:32 +05:00
function ( req , ret ) {
2016-03-21 22:58:42 +05:00
if ( ret ) {
2016-05-05 09:14:35 +05:00
twister . URIs [ req . shortURI ] = ret ;
2016-03-20 18:44:28 -03:00
$ . localStorage . set ( 'twistaURIs' , twister . URIs ) ;
2016-05-05 09:14:35 +05:00
applyShortenedURI ( req . shortURI , ret ) ;
2016-03-20 18:44:28 -03:00
} else {
2016-05-05 09:14:35 +05:00
console . warn ( 'can\'t fetch URI "' + req . shortURI + '": null response' ) ;
if ( ( req . attemptCount ? ++ req . attemptCount : req . attemptCount = 1 ) < 3 ) // < $.Options.decodeShortURITriesMax
fetchShortenedURI ( req . shortURI , req . attemptCount ) ;
2016-03-20 18:44:28 -03:00
}
2016-05-05 09:32:45 +05:00
} , { shortURI : req , attemptCount : attemptCount }
2016-03-21 22:58:42 +05:00
) ;
2016-03-20 22:44:57 +05:00
}
2016-05-03 23:02:54 -03:00
function applyShortenedURI ( short , uriAndMimetype ) {
var long = ( uriAndMimetype instanceof Array ) ? uriAndMimetype [ 0 ] : uriAndMimetype ;
2016-05-08 13:09:38 -03:00
var mimetype = ( uriAndMimetype instanceof Array ) ? uriAndMimetype [ 1 ] : undefined ;
2016-11-25 03:12:07 +05:00
var elems = getElem ( '.link-shortened[href="' + short + '"]' ) ;
if ( isUriSuspicious ( long ) ) {
elems . replaceWith (
'…<br><b><i>' + polyglot . t ( 'busted_oh' ) + '</i> '
+ polyglot . t ( 'busted_avowal' ) + ':</b><br><samp>'
+ long
. replace ( /&(?!lt;|gt;)/g , '&' )
. replace ( /"/g , '"' )
. replace ( /'/g , ''' )
+ '</samp><br>…<br>'
) ;
return ;
}
elems
2016-03-21 00:55:32 +05:00
. attr ( 'href' , long )
. removeClass ( 'link-shortened' )
. off ( 'click mouseup' )
. on ( 'click mouseup' , muteEvent )
;
2016-11-25 03:12:07 +05:00
2016-04-03 15:44:30 +05:00
var cropped = ( /*$.Options.cropLongURIs &&*/ long . length > 23 ) ? long . slice ( 0 , 23 ) + '…' : undefined ;
2016-05-08 13:09:38 -03:00
for ( var i = 0 ; i < elems . length ; i ++ ) {
2016-03-21 00:55:32 +05:00
if ( elems [ i ] . text === short ) // there may be some other text, possibly formatted, so we check it
2016-04-03 15:44:30 +05:00
if ( cropped )
$ ( elems [ i ] )
. text ( cropped )
. on ( 'mouseover' , { uri : long } , function ( event ) { event . target . text = event . data . uri ; } )
. on ( 'mouseout' , { uri : cropped } , function ( event ) { event . target . text = event . data . uri ; } )
;
else
elems [ i ] . text = long ;
2016-05-08 13:09:38 -03:00
if ( long . lastIndexOf ( 'magnet:?xt=urn:btih:' ) === 0 ) {
var previewContainer = $ ( elems [ i ] ) . parents ( ".post-data" ) . find ( ".preview-container" ) ;
var fromUser = $ ( elems [ i ] ) . parents ( ".post-data" ) . attr ( "data-screen-name" ) ;
var isMedia = mimetype !== undefined &&
( mimetype . lastIndexOf ( 'video' ) === 0 ||
mimetype . lastIndexOf ( 'image' ) === 0 ||
mimetype . lastIndexOf ( 'audio' ) === 0 ) ;
if ( $ . Options . WebTorrent . val === 'enable' ) {
if ( $ . Options . WebTorrentAutoDownload . val === 'enable' &&
followingUsers . indexOf ( fromUser ) !== - 1 ) {
2016-05-08 23:28:32 -03:00
if ( ! ( long in twister . torrentIds ) ) {
twister . torrentIds [ long ] = true ;
$ . localStorage . set ( 'torrentIds' , twister . torrentIds ) ;
}
2016-05-08 13:09:38 -03:00
startTorrentDownloadAndPreview ( long , previewContainer , isMedia ) ;
} else {
// webtorrent enabled but no auto-download. provide a link to start manually.
var startTorrentLink = $ ( '<a href="#">Start WebTorrent download of this media</a>' ) ;
startTorrentLink . on ( 'click' , function ( event ) {
event . stopPropagation ( ) ;
startTorrentDownloadAndPreview ( long , previewContainer , isMedia )
} ) ;
previewContainer . append ( startTorrentLink ) ;
}
} else {
2016-07-31 02:37:52 +05:00
var enableWebTorrentWarning = $ ( '<span>' +
2016-05-08 23:57:30 -03:00
polyglot . t ( 'Enable WebTorrent support in options page to display this content' ) +
'</span>' ) ;
previewContainer . append ( enableWebTorrentWarning ) ;
2016-05-08 13:09:38 -03:00
}
}
}
}
function startTorrentDownloadAndPreview ( torrentId , previewContainer , isMedia ) {
2016-05-09 22:43:33 -03:00
if ( typeof WebTorrentClient !== 'undefined' ) {
_startTorrentDownloadAndPreview ( torrentId , previewContainer , isMedia ) ;
} else {
// delay execution until WebTorrent is loaded/initialized
setTimeout ( _startTorrentDownloadAndPreview , 1000 , torrentId , previewContainer , isMedia )
}
}
function _startTorrentDownloadAndPreview ( torrentId , previewContainer , isMedia ) {
2016-05-08 13:09:38 -03:00
var torrent = WebTorrentClient . get ( torrentId ) ;
2016-07-31 02:37:52 +05:00
if ( torrent === null )
2016-05-08 13:09:38 -03:00
torrent = WebTorrentClient . add ( torrentId ) ;
previewContainer . empty ( ) ;
var speedStatus = $ ( '<span class="post-text"/>' ) ;
previewContainer . append ( speedStatus ) ;
function updateSpeed ( ) {
var progress = ( 100 * torrent . progress ) . toFixed ( 1 )
speedStatus [ 0 ] . innerHTML =
'<b>Peers:</b> ' + torrent . numPeers + ' ' +
'<b>Progress:</b> ' + progress + '% ' +
'<b>Download:</b> ' + torrent . downloadSpeed + '/s ' +
'<b>Upload:</b> ' + torrent . uploadSpeed + '/s' ;
}
setInterval ( updateSpeed , 1000 ) ;
updateSpeed ( ) ;
if ( torrent . files . length ) {
webtorrentFilePreview ( torrent . files [ 0 ] , previewContainer , isMedia )
} else {
torrent . on ( 'metadata' , function ( ) {
webtorrentFilePreview ( torrent . files [ 0 ] , previewContainer , isMedia )
} ) ;
}
}
function webtorrentFilePreview ( file , previewContainer , isMedia ) {
2016-05-08 16:31:54 -03:00
if ( ! isMedia ) {
// try guessing by filename extension
2021-07-11 17:13:10 +03:00
isMedia = /^[^?]+\.(?:jpe?g|gif|png|webp|mp4|webm|mp3|ogg|wav|)$/i . test ( file . name )
2016-05-08 16:31:54 -03:00
}
2016-07-31 02:37:52 +05:00
2016-05-08 13:09:38 -03:00
if ( isMedia ) {
var imagePreview = $ ( '<div class="image-preview" />' ) ;
previewContainer . append ( imagePreview ) ;
file . appendTo ( imagePreview [ 0 ] , function ( err , elem ) {
2016-05-08 23:28:32 -03:00
if ( 'pause' in elem ) {
elem . pause ( ) ;
}
2016-05-08 13:09:38 -03:00
} ) ;
2016-09-03 17:25:40 +03:00
var $vid = imagePreview . find ( "video" ) ;
$vid . removeAttr ( "autoplay" ) ;
$vid . on ( 'click mouseup' , muteEvent ) ;
2016-05-08 13:09:38 -03:00
} else {
file . getBlobURL ( function ( err , url ) {
if ( err ) return console . error ( err )
var blobLink = $ ( '<a href="' + url + '" download="' + file . name + '">' +
'Download ' + file . name + '</a>' ) ;
previewContainer . append ( blobLink ) ;
} )
}
2016-03-21 00:55:32 +05:00
}
2016-01-31 11:23:07 +05:00
function routeOnClick ( event ) {
2016-02-22 16:02:12 +05:00
function routeNewTab ( event ) {
if ( event . target . href ) {
// we can't prevent hyperlink navigation on middle button in Firefox via event.preventDefault(), see https://bugzilla.mozilla.org/show_bug.cgi?id=1249970
// so we need to delete href to make element not hyperlink and add it again after 200 ms
setTimeout ( ( function ( ) { this . elem . setAttribute ( 'href' , this . href ) ; } )
. bind ( { elem : event . target , href : event . target . getAttribute ( 'href' ) } ) , 200 ) ;
event . target . removeAttribute ( 'href' ) ;
}
2019-01-23 06:24:32 +05:00
twister . html . blanka . attr ( 'href' , event . data . route ) [ 0 ] . trigger ( 'click' ) ; // opens .route in new tab
2016-02-22 16:02:12 +05:00
}
2016-01-31 11:23:07 +05:00
if ( ! event || ! event . data || ! event . data . route )
return ;
2016-07-31 03:12:16 +05:00
if ( event . button === 0 && window . getSelection ( ) . toString ( ) !== '' )
return ;
2016-01-31 11:23:07 +05:00
event . stopPropagation ( ) ;
event . preventDefault ( ) ;
if ( event . button === 0 && ! event . data . blankOnly ) // left mouse button
2016-02-04 00:03:45 +05:00
window . location = event . data . route ; // closes modal(s) in watchHashChange() and opens .route
else if ( event . button === 1 ) // middle mouse button
if ( event . data . blankOnly || event . metaKey || event . ctrlKey )
2016-02-22 16:02:12 +05:00
routeNewTab ( event ) ;
2016-02-04 00:03:45 +05:00
else {
var modal = $ ( event . target ) . closest ( '.modal-wrapper:not(.closed)' ) ;
2016-03-23 23:52:53 +05:00
if ( modal . length && window . location . hash !== event . data . route ) {
minimizeModal ( modal , true ) ; // yep, we minimize current modal before .route opening
2016-02-04 00:03:45 +05:00
window . location . hash = event . data . route ;
} else
2016-02-22 16:02:12 +05:00
routeNewTab ( event ) ;
2016-02-04 00:03:45 +05:00
}
2016-01-31 11:23:07 +05:00
}
function watchHashChange ( event ) {
if ( typeof event !== 'undefined' ) {
var prevurlsplit = event . oldURL . split ( '#' ) ;
2015-05-23 22:26:23 +05:00
var prevhashstring = prevurlsplit [ 1 ] ;
2015-01-24 09:44:48 +01:00
2015-11-19 01:09:04 +05:00
// FIXME need to move back button handling to special function and call it in openModal() and resumeModal()
2015-05-23 22:26:23 +05:00
var notFirstModalView = ( prevhashstring !== undefined && prevhashstring . length > 0 ) ;
var notNavigatedBackToFirstModalView = ( window . history . state == null ||
( window . history . state != null && window . history . state . showCloseButton !== false ) ) ;
2015-05-19 23:33:59 +05:00
2015-05-23 22:26:23 +05:00
if ( notFirstModalView && notNavigatedBackToFirstModalView ) {
$ ( '.modal-back' ) . css ( 'display' , 'inline' ) ;
2015-01-22 15:59:00 +01:00
} else {
2016-01-10 03:22:28 +05:00
window . history . replaceState ( { showCloseButton : false } , '' , window . location . pathname + window . location . hash ) ;
2015-05-23 22:26:23 +05:00
$ ( '.modal-back' ) . css ( 'display' , 'none' ) ;
2015-01-21 12:10:04 +01:00
}
}
2016-01-10 03:25:40 +05:00
if ( _watchHashChangeRelaxDontDoIt )
_watchHashChangeRelaxDontDoIt = false ;
2015-11-19 01:09:04 +05:00
else
loadModalFromHash ( ) ;
2015-01-22 21:22:23 +01:00
}
2015-05-23 22:26:23 +05:00
function loadModalFromHash ( ) {
2016-01-31 11:23:07 +05:00
var i = window . location . hash ;
if ( twister . modal [ i ] && twister . modal [ i ] . minimized ) {
2019-01-23 06:24:32 +05:00
// need to close active modal(s) before btnResume.trigger('click') or it will be minimized in resumeModal()
2015-11-19 22:22:32 +05:00
// e.g. for case when you click on profile link in some modal having this profile's modal minimized already
2016-02-16 04:46:47 +05:00
closeModal ( undefined , true ) ;
2019-01-23 06:24:32 +05:00
twister . modal [ i ] . btnResume . trigger ( 'click' ) ;
2015-11-19 01:09:04 +05:00
return ;
}
2015-05-23 22:26:23 +05:00
var hashstring = decodeURIComponent ( window . location . hash ) ;
2016-01-10 03:25:40 +05:00
if ( hashstring === '' ) {
2016-01-31 11:23:07 +05:00
closeModal ( ) ; // close active modal(s)
2016-01-10 03:25:40 +05:00
return ;
}
2014-10-17 00:17:40 +02:00
var hashdata = hashstring . split ( ':' ) ;
2015-05-23 22:26:23 +05:00
2019-02-02 12:57:30 +05:00
if ( ! twister . var . initializated ) {
setTimeout ( loadModalFromHash , 1000 ) ;
return ;
}
2016-02-16 03:35:34 +05:00
// FIXME rework hash scheme from '#following?user=twister' to something like '#/@twister/following'
2015-05-23 22:26:23 +05:00
if ( hashdata [ 0 ] !== '#web+twister' )
2016-09-06 17:49:29 +03:00
hashdata = hashstring . match ( /(hashtag|profile|mentions|directmessages|followers|following|conversation|favs)\?(?:group|user|hashtag|post)=(.+)/ ) ;
2014-10-16 16:57:25 +02:00
2015-05-23 22:26:23 +05:00
if ( hashdata && hashdata [ 1 ] !== undefined && hashdata [ 2 ] !== undefined ) {
if ( hashdata [ 1 ] === 'profile' )
2015-09-04 03:21:33 +05:00
if ( hashdata [ 2 ] [ 0 ] === '*' )
openGroupProfileModalWithNameHandler ( hashdata [ 2 ] ) ;
else
openUserProfileModalWithNameHandler ( hashdata [ 2 ] ) ;
2015-05-23 22:26:23 +05:00
else if ( hashdata [ 1 ] === 'hashtag' )
2014-11-14 17:21:05 -02:00
openHashtagModalFromSearchHandler ( hashdata [ 2 ] ) ;
2015-05-23 22:26:23 +05:00
else if ( hashdata [ 1 ] === 'mentions' )
2014-11-14 18:39:33 -02:00
openMentionsModalHandler ( hashdata [ 2 ] ) ;
2015-09-04 03:21:33 +05:00
else if ( hashdata [ 1 ] === 'directmessages' ) {
if ( hashdata [ 2 ] [ 0 ] === '*' )
openGroupMessagesModal ( hashdata [ 2 ] ) ;
else
openDmWithUserModal ( hashdata [ 2 ] ) ;
2016-02-16 03:35:34 +05:00
} else if ( hashdata [ 1 ] === 'followers' )
openFollowersModal ( hashdata [ 2 ] ) ;
else if ( hashdata [ 1 ] === 'following' )
2015-01-21 14:13:45 +01:00
openFollowingModal ( hashdata [ 2 ] ) ;
2015-05-23 22:26:23 +05:00
else if ( hashdata [ 1 ] === 'conversation' ) {
splithashdata2 = hashdata [ 2 ] . split ( ':' ) ;
openConversationModal ( splithashdata2 [ 0 ] , splithashdata2 [ 1 ] ) ;
2014-10-16 16:57:25 +02:00
}
2016-09-06 17:49:29 +03:00
else if ( hashdata [ 1 ] === 'favs' )
openFavsModalHandler ( hashdata [ 2 ] ) ;
2015-05-23 22:26:23 +05:00
} else if ( hashstring === '#directmessages' )
2016-02-03 22:04:53 +05:00
openCommonDMsModal ( ) ;
2016-02-15 22:27:37 +05:00
else if ( hashstring === '#followers' )
openFollowersModal ( ) ;
2016-01-31 11:23:07 +05:00
else if ( hashstring === '#following' )
openFollowingModal ( ) ;
2015-09-04 03:21:33 +05:00
else if ( hashstring === '#groupmessages' )
openGroupMessagesModal ( ) ;
else if ( hashstring === '#groupmessages+newgroup' )
openGroupMessagesNewGroupModal ( ) ;
else if ( hashstring === '#groupmessages+joingroup' )
openGroupMessagesJoinGroupModal ( ) ;
2019-02-02 14:14:03 +05:00
else if ( hashstring === '#/account' )
openModalAccount ( ) ;
2017-02-02 14:17:10 +05:00
else if ( hashstring === '#/login' )
openModalLogin ( ) ;
2015-05-23 22:26:23 +05:00
else if ( hashstring === '#whotofollow' )
2015-01-22 15:59:00 +01:00
openWhoToFollowModal ( ) ;
2016-09-06 13:54:11 +05:00
else if ( hashstring === '#/uri-shortener' )
openModalUriShortener ( ) ;
2016-11-19 14:16:33 +03:00
else if ( hashstring === '#newusers' )
openNewUsersModal ( ) ;
2015-01-22 21:22:23 +01:00
}
2014-10-16 10:47:08 +02:00
2015-05-23 22:26:23 +05:00
function initHashWatching ( ) {
// register custom protocol handler
if ( window . navigator && window . navigator . registerProtocolHandler &&
! _getResourceFromStorage ( 'twister_protocol_registered' ) ) {
window . navigator . registerProtocolHandler (
'web+twister' ,
window . location . protocol + '//' + window . location . host + '/home.html#%s' ,
'Twister'
) ;
_putResourceIntoStorage ( 'twister_protocol_registered' , true ) ;
2014-10-21 00:33:14 +02:00
}
2014-10-17 00:17:40 +02:00
2015-05-23 22:26:23 +05:00
// register hash spy and launch it once
2014-10-17 00:17:40 +02:00
window . addEventListener ( 'hashchange' , watchHashChange , false ) ;
2015-05-23 22:26:23 +05:00
setTimeout ( watchHashChange , 1000 ) ;
2014-10-17 00:17:40 +02:00
}
2015-08-20 17:19:54 +05:00
function reTwistPopup ( event , post , textArea ) {
event . stopPropagation ( ) ;
2015-05-23 22:26:23 +05:00
if ( ! defaultScreenName ) {
2016-01-31 11:23:07 +05:00
alertPopup ( {
//txtTitle: polyglot.t(''), add some title (not 'error', please) or just KISS
txtMessage : polyglot . t ( 'You have to log in to retransmit messages.' )
} ) ;
2015-05-19 23:33:59 +05:00
return ;
2014-03-29 18:16:24 +01:00
}
2014-10-16 10:47:08 +02:00
2015-08-20 17:19:54 +05:00
if ( typeof post === 'undefined' )
2019-01-23 06:24:32 +05:00
post = JSON . parse ( $ ( event . target ) . closest ( '.post-data' ) . attr ( 'data-userpost' ) ) ;
2015-08-20 17:19:54 +05:00
2015-08-12 22:10:42 +05:00
var modal = openModal ( {
2015-05-23 22:26:23 +05:00
classBase : '.prompt-wrapper' ,
classAdd : 'reTwist' ,
title : polyglot . t ( 'retransmit_this' )
2015-08-12 22:10:42 +05:00
} ) ;
modal . content
2015-08-20 17:19:54 +05:00
. append ( postToElem ( post , '' ) )
. append ( $ ( '#reTwist-modal-template' ) . children ( ) . clone ( true ) )
2015-08-12 22:10:42 +05:00
;
2015-08-20 17:19:54 +05:00
modal . content . find ( '.switch-mode' )
. text ( polyglot . t ( 'Switch to Reply' ) )
2016-01-31 11:23:07 +05:00
. on ( 'click' , { post : post } ,
function ( event ) {
var textArea = $ ( event . target ) . closest ( 'form' ) . find ( 'textarea' ) . detach ( ) ;
closePrompt ( event . target ) ;
replyInitPopup ( event , event . data . post , textArea ) ;
}
)
2015-08-20 17:19:54 +05:00
;
2015-08-12 22:10:42 +05:00
2015-08-20 17:19:54 +05:00
var replyArea = modal . content . find ( '.post-area .post-area-new' ) ;
if ( typeof textArea === 'undefined' ) {
textArea = replyArea . find ( 'textarea' ) ;
var textAreaPostInline = modal . content . find ( '.post .post-area-new textarea' ) ;
$ . each ( [ 'placeholder' , 'data-reply-to' ] , function ( i , attribute ) {
textArea . attr ( attribute , textAreaPostInline . attr ( attribute ) ) ;
} ) ;
} else {
replyArea . find ( 'textarea' ) . replaceWith ( textArea ) ;
2015-08-20 17:31:16 +05:00
if ( textArea . val ( ) ) {
2019-01-23 06:24:32 +05:00
textArea . trigger ( 'focus' ) ;
2015-08-20 17:31:16 +05:00
replyArea . addClass ( 'open' ) ;
}
2015-08-20 17:19:54 +05:00
}
2015-08-12 22:10:42 +05:00
replyArea . find ( '.post-submit' ) . addClass ( 'with-reference' ) ;
2013-12-30 12:42:36 -02:00
}
2016-09-06 17:49:29 +03:00
function favPopup ( event , post , textArea ) {
event . stopPropagation ( ) ;
if ( ! defaultScreenName ) {
alertPopup ( {
txtMessage : polyglot . t ( 'You have to log in to favorite messages.' )
} ) ;
return ;
}
if ( typeof post === 'undefined' )
2019-01-23 06:24:32 +05:00
post = JSON . parse ( $ ( event . target ) . closest ( '.post-data' ) . attr ( 'data-userpost' ) ) ;
2016-09-06 17:49:29 +03:00
var modal = openModal ( {
classBase : '.prompt-wrapper' ,
classAdd : 'fav-this' ,
title : polyglot . t ( 'fav_this' )
} ) ;
modal . content
. append ( postToElem ( post , '' ) )
. append ( $ ( '#fav-modal-template' ) . children ( ) . clone ( true ) )
;
/ *
//TODO: favs can be also commented
var replyArea = modal . content . find ( '.post-area .post-area-new' ) ;
if ( typeof textArea === 'undefined' ) {
textArea = replyArea . find ( 'textarea' ) ;
var textAreaPostInline = modal . content . find ( '.post .post-area-new textarea' ) ;
$ . each ( [ 'placeholder' , 'data-reply-to' ] , function ( i , attribute ) {
textArea . attr ( attribute , textAreaPostInline . attr ( attribute ) ) ;
} ) ;
} else {
replyArea . find ( 'textarea' ) . replaceWith ( textArea ) ;
if ( textArea . val ( ) ) {
2019-01-23 06:24:32 +05:00
textArea . trigger ( 'focus' ) ;
2016-09-06 17:49:29 +03:00
replyArea . addClass ( 'open' ) ;
}
}
replyArea . find ( '.post-submit' ) . addClass ( 'with-reference' ) ;
* /
}
2015-05-23 22:26:23 +05:00
// Expande Área do Novo post
2015-08-20 17:19:54 +05:00
function replyInitPopup ( e , post , textArea ) {
2015-05-23 22:26:23 +05:00
var modal = openModal ( {
classBase : '.prompt-wrapper' ,
classAdd : 'reply' ,
title : polyglot . t ( 'reply_to' , { fullname : '<span class="fullname">' + post . userpost . n + '</span>' } )
} ) ;
2013-12-30 12:42:36 -02:00
2015-05-23 22:26:23 +05:00
getFullname ( post . userpost . n , modal . self . find ( 'h3 .fullname' ) ) ;
2013-12-30 12:42:36 -02:00
2015-05-23 22:26:23 +05:00
modal . content
. append ( postToElem ( post , '' ) )
2015-08-12 22:10:42 +05:00
. append ( $ ( '#reply-modal-template' ) . children ( ) . clone ( true ) )
2015-05-23 22:26:23 +05:00
;
2013-12-30 12:42:36 -02:00
2015-08-20 17:19:54 +05:00
modal . content . find ( '.switch-mode' )
. text ( polyglot . t ( 'Switch to Retransmit' ) )
2016-01-31 11:23:07 +05:00
. on ( 'click' , { post : post } ,
function ( event ) {
var textArea = $ ( event . target ) . closest ( 'form' ) . find ( 'textarea' ) . detach ( ) ;
closePrompt ( event . target ) ;
reTwistPopup ( event , event . data . post , textArea ) ;
}
)
2015-08-20 17:19:54 +05:00
;
2013-12-30 12:42:36 -02:00
2015-08-20 17:19:54 +05:00
var replyArea = modal . content . find ( '.post-area .post-area-new' ) . addClass ( 'open' ) ;
if ( typeof textArea === 'undefined' ) {
textArea = replyArea . find ( 'textarea' ) ;
var textAreaPostInline = modal . content . find ( '.post .post-area-new textarea' ) ;
$ . each ( [ 'placeholder' , 'data-reply-to' ] , function ( i , attribute ) {
textArea . attr ( attribute , textAreaPostInline . attr ( attribute ) ) ;
} ) ;
} else {
replyArea . find ( 'textarea' ) . replaceWith ( textArea ) ;
}
2015-05-23 22:26:23 +05:00
composeNewPost ( e , replyArea ) ;
2013-12-30 12:42:36 -02:00
}
2015-05-23 22:26:23 +05:00
// abre o menu dropdown de configurações
2015-03-04 04:26:20 +05:00
function dropDownMenu ( ) {
2015-05-23 22:26:23 +05:00
$ ( '.config-menu' ) . slideToggle ( 'fast' ) ;
2013-12-30 12:42:36 -02:00
}
2015-05-23 22:26:23 +05:00
// fecha o config menu ao clicar em qualquer lugar da tela
2015-03-04 04:26:20 +05:00
function closeThis ( ) {
2015-05-23 22:26:23 +05:00
$ ( this ) . slideUp ( 'fast' ) ;
2015-03-04 04:26:20 +05:00
}
2016-02-03 01:29:41 +05:00
function muteEvent ( event , preventDefault ) {
event . stopPropagation ( ) ;
if ( preventDefault || ( event . data && event . data . preventDefault ) )
2016-02-22 16:03:09 +05:00
event . preventDefault ( ) ;
2016-02-03 01:29:41 +05:00
}
2016-01-31 11:23:07 +05:00
function toggleFollowButton ( req ) {
if ( ! req || ! req . peerAlias )
2015-03-04 04:26:20 +05:00
return ;
2016-01-31 11:23:07 +05:00
if ( req . toggleUnfollow ) {
if ( ! req . button || ! req . button . jquery )
req . button = getElem ( '[data-screen-name="' + req . peerAlias + '"]' ) . find ( '.follow' ) ;
req . button
. text ( polyglot . t ( 'Unfollow' ) )
2015-05-23 22:26:23 +05:00
. removeClass ( 'follow' )
. addClass ( 'unfollow' )
. off ( 'click' )
2016-01-31 11:23:07 +05:00
. on ( 'click' , { peerAlias : req . peerAlias } , clickUnfollow )
2015-05-23 22:26:23 +05:00
;
2015-03-07 07:16:45 +05:00
} else {
2016-01-31 11:23:07 +05:00
if ( ! req . button || ! req . button . jquery )
req . button = getElem ( '[data-screen-name="' + req . peerAlias + '"]' ) . find ( '.unfollow' ) ;
req . button
. text ( polyglot . t ( 'Follow' ) )
2015-05-23 22:26:23 +05:00
. removeClass ( 'unfollow' )
. addClass ( 'follow' )
. off ( 'click' )
2016-01-31 11:23:07 +05:00
. on ( 'click' , { peerAlias : req . peerAlias } , clickFollow )
;
2015-03-07 07:16:45 +05:00
}
2015-03-04 04:26:20 +05:00
}
2013-12-30 12:42:36 -02:00
2016-01-31 11:23:07 +05:00
function clickFollow ( event ) {
event . preventDefault ( ) ;
event . stopPropagation ( ) ;
if ( ! defaultScreenName ) {
alertPopup ( {
//txtTitle: polyglot.t(''), add some title (not 'error', please) or just KISS
txtMessage : polyglot . t ( 'You have to log in to follow users.' )
} ) ;
return ;
}
var peerAlias = ( event . data && event . data . peerAlias ) ? event . data . peerAlias
: $ ( event . target ) . closest ( '[data-screen-name]' ) . attr ( 'data-screen-name' ) ;
var content = $ ( '#following-config-modal-template' ) . children ( ) . clone ( true ) ;
content . closest ( '.following-config-modal-content' ) . attr ( 'data-screen-name' , peerAlias ) ;
2016-02-22 17:09:01 +05:00
fillElemWithTxt ( content . find ( '.following-config-method-message' ) ,
polyglot . t ( 'select_way_to_follow_@' , { alias : peerAlias } ) , { markout : 'apply' } ) ;
2016-01-31 11:23:07 +05:00
content . find ( '.following-screen-name b' ) . text ( peerAlias ) ;
openModal ( {
classBase : '.prompt-wrapper' , // FIXME it will be modal with advanced following set up in future
classAdd : 'following-config-modal' ,
content : content ,
title : polyglot . t ( 'Following config' )
} ) ;
}
function clickUnfollow ( event ) {
event . preventDefault ( ) ;
event . stopPropagation ( ) ;
var peerAlias = ( event . data && event . data . peerAlias ) ? event . data . peerAlias
: $ ( event . target ) . closest ( '[data-screen-name]' ) . attr ( 'data-screen-name' ) ;
confirmPopup ( {
txtMessage : polyglot . t ( 'confirm_unfollow_@' , { alias : peerAlias } ) ,
cbConfirm : function ( peerAlias ) {
unfollow ( peerAlias ,
function ( req ) {
$ ( '.mini-profile .following-count' ) . text ( followingUsers . length - 1 ) ;
$ ( '.wrapper .postboard .post' ) . each ( function ( ) {
var elem = $ ( this ) ;
if ( ( elem . find ( '[data-screen-name="' + req . peerAlias + '"]' ) . length
&& ! elem . find ( ".post-rt-by .open-profile-modal" ) . text ( ) )
|| elem . find ( ".post-rt-by .open-profile-modal" ) . text ( ) === '@' + req . peerAlias )
elem . remove ( ) ;
} ) ; // FIXME also need to check list of pending posts to remove from there
toggleFollowButton ( { peerAlias : req . peerAlias } ) ;
var followingList = getElem ( '.following-own-modal .following-list' ) ;
if ( followingList . length )
followingList . find ( 'li[data-peer-alias="' + req . peerAlias + '"]' ) . remove ( ) ;
} , { peerAlias : peerAlias }
) ;
} ,
cbConfirmReq : peerAlias
} ) ;
}
function setFollowingMethod ( event ) {
event . preventDefault ( ) ;
event . stopPropagation ( ) ;
var button = $ ( event . target ) ;
var peerAlias = button . closest ( '.following-config-modal-content' ) . attr ( 'data-screen-name' ) ;
follow ( peerAlias , button . hasClass ( 'private' ) ? false : true ,
function ( req ) {
$ ( '.mini-profile .following-count' ) . text ( followingUsers . length - 1 ) ;
setTimeout ( requestTimelineUpdate , 1000 , 'latest' , postsPerRefresh , [ req . peerAlias ] , promotedPostsOnly ) ;
toggleFollowButton ( { peerAlias : req . peerAlias , toggleUnfollow : req . toggleUnfollow } ) ;
var followingList = getElem ( '.following-own-modal .following-list' ) ;
if ( followingList . length )
2016-02-15 22:56:05 +05:00
addPeerToFollowingList ( followingList , req . peerAlias ) ;
2016-01-31 11:23:07 +05:00
} , { peerAlias : peerAlias , toggleUnfollow : true }
) ;
}
function followingListPublicCheckbox ( event ) {
event . preventDefault ( ) ;
event . stopPropagation ( ) ;
var tickSelection = function ( req ) {
if ( req . isPublic === req . wasPublic ) return ;
var elem = $ ( '.mini-profile-info[data-screen-name="' + req . peerAlias + '"] .public-following' ) ;
elem . toggleClass ( 'private' ) ;
if ( ! req . isPublic )
elem . text ( polyglot . t ( 'Private' ) ) ;
else
elem . text ( polyglot . t ( 'Public' ) ) ;
//console.log('set following method of @' + peerAlias + ' for ' + isPublic);
follow ( req . peerAlias , req . isPublic ) ;
} ;
var elem = $ ( event . target ) ;
var peerAlias = elem . closest ( '.mini-profile-info' ) . attr ( 'data-screen-name' ) ;
var wasPublic = ! elem . hasClass ( 'private' ) ;
confirmPopup ( {
txtMessage : polyglot . t ( 'select_way_to_follow_@' , { alias : peerAlias } ) ,
txtConfirm : polyglot . t ( 'Public' ) ,
cbConfirm : tickSelection ,
cbConfirmReq : { isPublic : true , wasPublic : wasPublic , peerAlias : peerAlias } ,
txtCancel : polyglot . t ( 'Private' ) ,
cbCancel : tickSelection ,
cbCancelReq : { isPublic : false , wasPublic : wasPublic , peerAlias : peerAlias }
} ) ;
}
2015-05-23 22:26:23 +05:00
function postExpandFunction ( e , postLi ) {
if ( ! postLi . hasClass ( 'original' ) )
2013-12-30 12:42:36 -02:00
return ;
2015-05-23 22:26:23 +05:00
var openClass = 'open' ;
var originalPost = postLi . find ( '.post-data.original' ) ;
var postInteractionText = originalPost . find ( '.post-expand' ) ;
var postExpandedContent = originalPost . find ( '.expanded-content' ) ;
var postsRelated = postLi . find ( '.related' ) ;
2014-10-16 10:47:08 +02:00
2015-05-23 22:26:23 +05:00
if ( ! postLi . hasClass ( openClass ) ) {
2013-12-30 12:42:36 -02:00
originalPost . detach ( ) ;
postLi . empty ( ) ;
2015-05-23 22:26:23 +05:00
postLi . addClass ( openClass ) ;
postInteractionText . text ( polyglot . t ( 'Collapse' ) ) ;
2013-12-30 12:42:36 -02:00
2015-05-23 22:26:23 +05:00
var itemOl = $ ( '<ol/>' , { class : 'expanded-post' } ) . appendTo ( postLi ) ;
var originalLi = $ ( '<li/>' , { class : 'module post original' } ) . appendTo ( itemOl )
. append ( originalPost ) ;
2013-12-30 12:42:36 -02:00
2016-09-10 18:14:08 +03:00
setPostImagePreview ( postExpandedContent , originalPost . find ( 'a[rel^="nofollow"]' ) ) ;
2015-08-10 03:02:15 +05:00
2015-05-23 22:26:23 +05:00
postExpandedContent . slideDown ( 'fast' ) ;
2014-10-16 10:47:08 +02:00
2015-05-23 22:26:23 +05:00
// insert 'reply_to' before
2013-12-30 12:42:36 -02:00
requestRepliedBefore ( originalLi ) ;
// insert replies to this post after
requestRepliesAfter ( originalLi ) ;
// RTs faces and counter
2018-11-02 02:50:12 +05:00
requestRTs ( originalPost . attr ( 'data-screen-name' ) , originalPost . attr ( 'data-id' ) ) ;
2015-05-23 22:26:23 +05:00
} else {
postLi . removeClass ( openClass ) ;
2014-05-07 13:28:16 +03:00
2015-05-23 22:26:23 +05:00
postInteractionText . text (
( typeof postLi . find ( '.post-data.original' ) . attr ( 'data-replied-to-id' ) === 'undefined' ) ?
polyglot . t ( 'Expand' ) : polyglot . t ( 'Show conversation' )
) ;
2014-05-07 13:28:16 +03:00
2015-05-23 22:26:23 +05:00
if ( postsRelated )
postsRelated . slideUp ( 'fast' ) ;
2013-12-30 12:42:36 -02:00
2015-05-23 22:26:23 +05:00
postExpandedContent . slideUp ( 'fast' , function ( ) {
originalPost . detach ( ) . appendTo ( postLi . empty ( ) ) ;
2013-12-30 12:42:36 -02:00
} ) ;
}
e . stopPropagation ( ) ;
}
2016-01-31 11:23:07 +05:00
function postReplyClick ( event ) {
2015-05-23 22:26:23 +05:00
if ( ! defaultScreenName ) {
2016-01-31 11:23:07 +05:00
event . stopPropagation ( ) ;
alertPopup ( {
//txtTitle: polyglot.t(''), add some title (not 'error', please) or just KISS
txtMessage : polyglot . t ( 'You have to log in to post replies.' )
} ) ;
2015-05-19 23:33:59 +05:00
return ;
2014-03-29 18:16:24 +01:00
}
2015-05-23 22:26:23 +05:00
var post = $ ( this ) . closest ( '.post' ) ;
if ( ! post . hasClass ( 'original' ) )
2019-01-23 06:24:32 +05:00
replyInitPopup ( event , JSON . parse ( post . find ( '.post-data' ) . attr ( 'data-userpost' ) ) ) ;
2015-05-23 22:26:23 +05:00
else {
2015-10-24 01:39:22 +05:00
if ( ! post . closest ( '.post.open' ) . length )
2016-01-31 11:23:07 +05:00
postExpandFunction ( event , post ) ;
composeNewPost ( event , post . find ( '.post-area-new' ) ) ;
2013-12-30 12:42:36 -02:00
}
2015-05-23 22:26:23 +05:00
2016-01-31 11:23:07 +05:00
event . stopPropagation ( ) ;
2013-12-30 12:42:36 -02:00
}
2022-09-01 14:26:00 +03:00
function postTranslateClick ( event ) {
if ( $ . Options . TranslationEnabled . val === 'enable' && $ . Options . TranslationAPI . val . length > 0 ) {
var post = $ ( this ) . closest ( '.post' ) . find ( '.post-text' ) ;
// prevent merge string on tabulation usage
post . html ( post . html ( ) . replaceAll ( '<br>' , ' ' ) ) ;
// some APIs not accept slash as the part of the get request routing
var request = post . text ( ) . replaceAll ( '/' , '|' ) ;
$ . ajax ( {
dataType : 'json' ,
url : $ . Options . TranslationAPI . val + encodeURIComponent ( request ) ,
success : function ( json ) {
if ( json . translation ) {
var translation = htmlFormatMsg ( json . translation . replaceAll ( '|' , '/' ) ) ;
post . html ( translation . html ) ;
}
if ( json . error ) {
alert ( json . error ) ;
}
} ,
error : function ( xhr , ajaxOptions , thrownError ) {
alert ( thrownError + "\r\n" + xhr . statusText + "\r\n" + xhr . responseText ) ;
}
} )
}
event . stopPropagation ( ) ;
}
2015-05-23 22:26:23 +05:00
// Expande Área do Novo post
function composeNewPost ( e , postAreaNew ) {
2013-12-30 12:42:36 -02:00
e . stopPropagation ( ) ;
2015-05-23 22:26:23 +05:00
if ( ! postAreaNew . hasClass ( 'open' ) ) {
postAreaNew . addClass ( 'open' ) ;
2013-12-30 12:42:36 -02:00
//se o usuário clicar fora é pra fechar
2016-03-23 21:46:02 +05:00
postAreaNew . clickoutside ( unfocusPostAreaNew ) ;
2013-12-30 12:42:36 -02:00
}
2015-05-23 22:26:23 +05:00
var textArea = postAreaNew . find ( 'textarea' ) ;
if ( textArea . attr ( 'data-reply-to' ) && ! textArea . val ( ) . length ) {
textArea . val ( textArea . attr ( 'data-reply-to' ) ) ;
2016-03-23 21:46:02 +05:00
poseTextareaPostTools ( textArea ) ;
2013-12-30 12:42:36 -02:00
}
2015-05-23 22:26:23 +05:00
if ( ! postAreaNew . find ( 'textarea:focus' ) . length )
2019-01-23 06:24:32 +05:00
postAreaNew . find ( 'textarea:last' ) . trigger ( 'focus' ) ;
2013-12-30 12:42:36 -02:00
}
2019-01-24 15:58:37 +05:00
function prepareTextareaToInput ( event ) {
var elem = $ ( event . target ) ;
if ( ! elem . closest ( '.directMessages' ) . length
&& ! elem . closest ( '.network' ) . length
&& ( $ . Options . splitPosts . val === 'enable'
|| ( $ . Options . splitPosts . val === 'only-new'
&& ! ( elem . closest ( '.post-data' ) . length
|| elem . closest ( '.modal-content' ) . find ( '.post-data' ) . length ) ) ) )
twister . var . isCurrentInputSplittable = true ;
else
twister . var . isCurrentInputSplittable = false ;
poseTextareaPostTools ( elem ) ;
}
2016-03-23 21:46:02 +05:00
function poseTextareaPostTools ( event ) {
if ( event . jquery )
var textArea = event ;
else
var textArea = $ ( event . target ) ;
posePostPreview ( textArea ) ;
poseTextareaEditBar ( textArea ) ;
}
function posePostPreview ( textArea ) {
2015-07-25 02:48:19 +05:00
if ( ! $ . Options . postPreview . val )
return ;
2015-07-24 00:51:24 +05:00
var postPreview = textArea . siblings ( '#post-preview' ) ;
2015-07-25 02:19:52 +05:00
if ( ! postPreview . length ) {
postPreview = $ ( '#post-preview-template' ) . children ( ) . clone ( )
. css ( 'margin-left' , textArea . css ( 'margin-left' ) )
. css ( 'margin-right' , textArea . css ( 'margin-right' ) )
;
2015-07-25 03:19:12 +05:00
postPreview . width ( textArea . width ( ) ) ;
postPreview . width ( postPreview . width ( ) // width is not accurate if we do it with textArea.width() directly, don't know why
2015-07-25 02:19:52 +05:00
- postPreview . css ( 'padding-left' ) - postPreview . css ( 'padding-right' ) ) ;
}
2015-07-25 04:13:53 +05:00
if ( textArea [ 0 ] . value . length )
2016-02-22 17:09:01 +05:00
fillElemWithTxt ( postPreview . show ( ) , textArea [ 0 ] . value ) ;
2015-07-25 04:13:53 +05:00
else
2015-08-21 10:10:46 +05:00
postPreview . hide ( ) ;
2015-07-24 00:51:24 +05:00
textArea . before ( postPreview ) ;
}
2016-03-23 21:46:02 +05:00
function poseTextareaEditBar ( textArea ) {
var editBar = textArea . siblings ( '.post-textarea-edit-bar' ) ;
if ( ! editBar . length ) {
editBar = twister . tmpl . postTextareaEditBar . clone ( true )
. css ( 'margin-left' , textArea . css ( 'margin-left' ) )
. css ( 'margin-right' , textArea . css ( 'margin-right' ) )
;
2019-01-24 16:12:19 +05:00
if ( textArea . closest ( '.network' ) . length )
editBar . find ( '.shorten-uri' ) . remove ( ) ;
else
editBar . find ( '.shorten-uri' ) . text ( polyglot . t ( 'shorten_URI' ) ) ;
2016-03-23 21:46:02 +05:00
}
editBar . insertAfter ( textArea ) . show ( ) ;
}
2015-05-23 22:26:23 +05:00
// Reduz Área do Novo post
2016-03-23 21:46:02 +05:00
function unfocusPostAreaNew ( ) {
var postAreaNew = $ ( this ) . removeClass ( 'open' ) ;
postAreaNew . find ( '#post-preview' ) . slideUp ( 'fast' ) ;
postAreaNew . find ( '.post-textarea-edit-bar' ) . hide ( ) ;
2013-12-30 12:42:36 -02:00
}
2014-05-11 22:18:33 +03:00
function checkPostForMentions ( post , mentions , max ) {
2015-05-23 22:26:23 +05:00
return new RegExp ( '^.{0,' + max . toString ( ) + '}(?:' + mentions . trim ( ) . replace ( / /g , '|' ) + ')' ) . test ( post ) ;
2014-05-11 22:18:33 +03:00
}
2015-05-23 22:26:23 +05:00
var splitedPostsCount = 1 ; // FIXME it could be property of future textAreaInput and composeNewPost united thing; currently stuff is hell
2014-04-17 01:21:17 +03:00
2015-05-23 22:26:23 +05:00
function replyTextInput ( event ) {
var textArea = $ ( event . target ) ;
2015-10-24 01:39:22 +05:00
var textAreaForm = textArea . closest ( 'form' ) ;
2015-05-23 22:26:23 +05:00
if ( textAreaForm . length ) {
2015-05-19 23:33:59 +05:00
if ( $ . Options . unicodeConversion . val !== 'disable' )
2015-05-23 22:26:23 +05:00
textArea . val ( convert2Unicodes ( textArea . val ( ) , textArea ) ) ;
2019-01-24 15:58:37 +05:00
if ( twister . var . isCurrentInputSplittable ) {
2015-05-23 22:26:23 +05:00
var caretPos = textArea . caret ( ) ;
var reply _to = textArea . attr ( 'data-reply-to' ) ;
var tas = textAreaForm . find ( 'textarea' ) ;
splitedPostsCount = tas . length ;
var icurrentta = tas . index ( event . target ) ; // current textarea tas index
var i = 0 ;
var pml = getPostSplitingPML ( ) ;
2015-04-10 06:59:57 +05:00
var cci = getPostSpittingCI ( icurrentta ) ;
2014-05-08 23:28:16 +03:00
2015-05-23 22:26:23 +05:00
for ( ; i < tas . length ; i ++ ) {
pml = getPostSplitingPML ( ) ;
if ( tas [ i ] . value . length > pml ) {
2015-04-10 06:59:57 +05:00
var ci = getPostSpittingCI ( i ) ;
2014-05-11 22:18:33 +03:00
if ( i < splitedPostsCount - 1 ) {
2015-05-23 22:26:23 +05:00
tas [ i + 1 ] . value = tas [ i ] . value . substr ( ci ) + tas [ i + 1 ] . value ;
tas [ i ] . value = tas [ i ] . value . substr ( 0 , ci ) ;
2015-04-10 06:59:57 +05:00
if ( caretPos > cci ) {
2015-04-09 22:09:05 +05:00
caretPos -= ci ;
icurrentta += 1 ;
2015-04-10 06:59:57 +05:00
cci = getPostSpittingCI ( icurrentta ) ;
2015-05-23 22:26:23 +05:00
var targetta = $ ( tas [ icurrentta ] ) ;
2015-04-09 22:09:05 +05:00
} else if ( i === icurrentta )
2015-05-23 22:26:23 +05:00
$ ( tas [ i ] ) . caret ( caretPos ) ;
2014-05-11 22:18:33 +03:00
} else {
2015-05-23 22:26:23 +05:00
var oldta = $ ( tas [ i ] ) ;
if ( $ . fn . textcomplete ) {
oldta . textcomplete ( 'destroy' ) ;
event . stopImmediatePropagation ( ) ; // something goes wrong in $.fn.textcomplete if we don't stop this immediately
2015-04-09 22:09:05 +05:00
}
2015-05-23 22:26:23 +05:00
var cp = oldta . val ( ) ;
var newta = $ ( oldta ) . clone ( true )
. val ( cp . substr ( ci ) )
. insertAfter ( oldta )
;
oldta . val ( cp . substr ( 0 , ci ) )
. addClass ( 'splited-post' )
. on ( 'focus' , function ( ) { this . style . height = '80px' ; } )
. on ( 'focusout' , function ( ) { this . style . height = '28px' ; } ) // FIXME move this to CSS
;
tas = textAreaForm . find ( 'textarea' ) ;
splitedPostsCount = tas . length ;
2015-04-13 01:20:41 +05:00
pml = getPostSplitingPML ( ) ;
2015-04-10 06:59:57 +05:00
if ( caretPos > cci ) {
2015-04-09 22:09:05 +05:00
caretPos -= ci ;
icurrentta += 1 ;
2015-04-10 06:59:57 +05:00
cci = getPostSpittingCI ( icurrentta ) ;
2015-05-23 22:26:23 +05:00
var targetta = newta ;
oldta [ 0 ] . style . height = '28px' ; // FIXME move this to CSS
2015-04-10 06:59:57 +05:00
} else if ( i === icurrentta ) {
2015-05-23 22:26:23 +05:00
$ ( tas [ i ] ) . caret ( caretPos ) ;
replyTextUpdateRemaining ( tas [ i ] ) ;
if ( $ . fn . textcomplete )
2015-09-05 04:14:16 +05:00
setTextcompleteOnElement ( tas [ i ] , getMentionsForAutoComplete ( ) ) ;
2015-04-10 06:59:57 +05:00
}
2014-05-11 22:18:33 +03:00
}
2015-05-23 22:26:23 +05:00
} else if ( tas . length > 1 && tas [ i ] . value . length === 0 ) {
if ( i === tas . length - 1 ) {
tas [ i ] . value = tas [ i - 1 ] . value ;
$ ( tas [ i - 1 ] ) . remove ( ) ;
} else
$ ( tas [ i ] ) . remove ( ) ;
tas = textAreaForm . find ( 'textarea' ) ;
i -- ;
splitedPostsCount = tas . length ;
pml = getPostSplitingPML ( ) ;
2015-04-09 22:09:05 +05:00
caretPos = - 1 ;
2015-04-10 06:59:57 +05:00
if ( icurrentta >= i && icurrentta > 0 ) {
2015-04-09 22:09:05 +05:00
icurrentta -= 1 ;
2015-04-10 06:59:57 +05:00
cci = getPostSpittingCI ( icurrentta ) ;
}
2015-05-23 22:26:23 +05:00
var targetta = $ ( tas [ icurrentta ] ) ;
2014-04-17 01:21:17 +03:00
}
2014-04-18 18:55:12 +03:00
}
2015-04-10 06:59:57 +05:00
2015-05-23 22:26:23 +05:00
if ( typeof targetta !== 'undefined' && targetta [ 0 ] !== document . activeElement ) {
textArea = targetta ;
2019-01-23 06:24:32 +05:00
textArea . trigger ( 'focus' ) ;
2015-05-23 22:26:23 +05:00
textArea . caret ( caretPos ) ;
2015-04-09 22:09:05 +05:00
}
2014-04-17 01:21:17 +03:00
}
2015-07-24 00:51:24 +05:00
2015-07-29 17:13:32 +05:00
if ( $ . Options . postPreview . val ) {
if ( textArea [ 0 ] . value . length )
2016-02-22 17:09:01 +05:00
fillElemWithTxt ( textAreaForm . find ( '#post-preview' ) . show ( ) , textArea [ 0 ] . value ) ;
2015-07-29 17:13:32 +05:00
else
2015-08-20 17:31:16 +05:00
textAreaForm . find ( '#post-preview' ) . html ( '' ) . slideUp ( 'fast' ) ;
2015-07-29 17:13:32 +05:00
}
2015-04-10 06:59:57 +05:00
}
2013-12-30 12:42:36 -02:00
2015-04-10 06:59:57 +05:00
function getPostSplitingPML ( ) {
2016-07-16 18:15:50 -03:00
var MaxPostSize = $ . Options . MaxPostEditorChars . val ;
2015-05-23 22:26:23 +05:00
if ( splitedPostsCount > 1 ) {
2016-07-16 18:15:50 -03:00
var pml = MaxPostSize - ( i + 1 ) . toString ( ) . length - splitedPostsCount . toString ( ) . length - 4 ;
2013-12-30 12:42:36 -02:00
2015-05-23 22:26:23 +05:00
// if mention exists, we shouldn't add it while posting.
if ( typeof reply _to !== 'undefined' &&
! checkPostForMentions ( tas [ i ] . value , reply _to , pml - reply _to . length ) ) {
pml -= reply _to . length ;
}
} else
2016-07-16 18:15:50 -03:00
var pml = MaxPostSize ;
2015-04-10 06:59:57 +05:00
return pml ;
}
function getPostSpittingCI ( ita ) {
var ci ;
2015-05-23 22:26:23 +05:00
var endings = tas [ ita ] . value . match ( /[\\\/\.,:;\?\!\*'"\]\)\}\^\|%\u201D\u2026\u2014\u4E00\u3002\uFF0C\uFF1A\uFF1F\uFF01\u3011>\s]/g ) // unicode escaped stuff is '”…—一。,:?!】
2015-04-10 06:59:57 +05:00
if ( endings ) {
2015-05-23 22:26:23 +05:00
ci = tas [ ita ] . value . lastIndexOf ( endings [ endings . length - 1 ] ) ;
for ( var j = endings . length - 2 ; j >= 0 && ci > pml ; j -- )
ci = tas [ ita ] . value . lastIndexOf ( endings [ j ] , ci - 1 ) ;
2013-12-30 12:42:36 -02:00
}
2015-04-10 06:59:57 +05:00
if ( ! ( ci > 0 ) )
ci = pml ;
return ( ci > pml ) ? pml : ci ;
2013-12-30 12:42:36 -02:00
}
}
2015-04-10 06:59:57 +05:00
function replyTextUpdateRemaining ( ta ) {
2015-05-23 22:26:23 +05:00
if ( ta . target )
ta = ta . target ;
if ( ta === document . activeElement ) {
var textArea = $ ( ta ) ;
2015-10-24 01:39:22 +05:00
var textAreaForm = textArea . closest ( 'form' ) ;
2015-05-23 22:26:23 +05:00
if ( textAreaForm . length ) {
var remainingCount = textAreaForm . find ( '.post-area-remaining' ) ;
2015-04-10 06:59:57 +05:00
var c = replyTextCountRemaining ( ta ) ;
2015-04-09 22:09:05 +05:00
2019-01-24 15:58:37 +05:00
if ( twister . var . isCurrentInputSplittable && splitedPostsCount > 1 )
2015-05-23 22:26:23 +05:00
remainingCount . text ( ( textAreaForm . find ( 'textarea' ) . index ( ta ) + 1 ) . toString ( )
+ '/' + splitedPostsCount . toString ( ) + ': ' + c . toString ( ) ) ;
2015-04-09 22:09:05 +05:00
else
remainingCount . text ( c . toString ( ) ) ;
2015-05-23 22:26:23 +05:00
var buttonSend = textAreaForm . find ( '.post-submit' ) ;
if ( ! buttonSend . length )
buttonSend = textAreaForm . find ( '.dm-submit' ) ;
2015-04-10 06:59:57 +05:00
var disable = false ;
2015-05-23 22:26:23 +05:00
textAreaForm . find ( 'textarea' ) . each ( function ( ) {
2015-04-10 06:59:57 +05:00
if ( replyTextCountRemaining ( this ) < 0 ) {
disable = true ; // alternatively we could call replyTextInput()
return false ;
}
} ) ;
2016-07-31 02:37:52 +05:00
if ( ! disable && c >= 0 && c < $ . Options . MaxPostEditorChars . val &&
2016-07-16 18:15:50 -03:00
textArea . val ( ) !== textArea . attr ( 'data-reply-to' ) ) {
2015-05-23 22:26:23 +05:00
remainingCount . removeClass ( 'warn' ) ;
$ . MAL . enableButton ( buttonSend ) ;
2015-04-10 06:59:57 +05:00
} else {
if ( disable )
2015-05-23 22:26:23 +05:00
remainingCount . addClass ( 'warn' ) ;
$ . MAL . disableButton ( buttonSend ) ;
2015-04-10 06:59:57 +05:00
}
2015-04-09 22:09:05 +05:00
}
}
}
2015-04-10 06:59:57 +05:00
function replyTextCountRemaining ( ta ) {
2015-05-23 22:26:23 +05:00
var textArea = $ ( ta ) ;
2015-04-09 22:09:05 +05:00
var c ;
2016-07-16 18:15:50 -03:00
var MaxPostSize = $ . Options . MaxPostEditorChars . val ;
2019-01-24 15:58:37 +05:00
if ( twister . var . isCurrentInputSplittable && splitedPostsCount > 1 ) {
2016-07-16 18:15:50 -03:00
c = MaxPostSize - ta . value . length - ( textArea . closest ( 'form' ) . find ( 'textarea' ) . index ( ta ) + 1 ) . toString ( ) . length - splitedPostsCount . toString ( ) . length - 4 ;
2015-05-23 22:26:23 +05:00
var reply _to = textArea . attr ( 'data-reply-to' ) ;
if ( typeof reply _to !== 'undefined' &&
2016-07-16 18:15:50 -03:00
! checkPostForMentions ( ta . value , reply _to , MaxPostSize - c - reply _to . length ) )
2015-05-23 22:26:23 +05:00
c -= reply _to . length ;
2015-04-09 22:09:05 +05:00
} else
2016-07-16 18:15:50 -03:00
c = MaxPostSize - ta . value . length ;
2015-04-09 22:09:05 +05:00
return c ;
}
2015-05-23 22:26:23 +05:00
function replyTextKeySend ( event ) {
if ( event . keyCode === 13 ) {
if ( ( ! event . metaKey && ! event . ctrlKey && $ . Options . keysSend . val === 'enter' &&
$ ( '.dropdown-menu' ) . css ( 'display' ) === 'none' )
|| ( ( event . metaKey || event . ctrlKey ) && $ . Options . keysSend . val === 'ctrlenter' ) ) {
var textArea = $ ( event . target ) ;
2015-10-24 01:39:22 +05:00
var textAreaForm = textArea . closest ( 'form' ) ;
2015-05-23 22:26:23 +05:00
var buttonSend = textAreaForm . find ( '.post-submit' ) ;
if ( ! buttonSend . length )
buttonSend = textAreaForm . find ( '.dm-submit' ) ;
if ( buttonSend . length ) {
textArea . val ( textArea . val ( ) . trim ( ) ) ;
if ( ! buttonSend . hasClass ( 'disabled' ) )
2019-01-23 06:24:32 +05:00
buttonSend . trigger ( 'click' ) ;
2015-04-10 06:59:57 +05:00
}
}
}
}
2014-04-07 19:11:41 +03:00
/ *
* unicode convertion list
* k : original string to be replaced
* u : unicode
* n : index of char to be stored and appended to result
* /
var unicodeConversionList = {
2015-05-23 22:26:23 +05:00
'punctuation' : [
2014-04-07 19:11:41 +03:00
{
2015-05-23 22:26:23 +05:00
'k' : /\.\.\./ ,
'u' : '\u2026' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /\.\../ ,
'u' : '\u2025' ,
'n' : 2
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /\?\?/ ,
'u' : '\u2047' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /\?!/ ,
'u' : '\u2048' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /!\?/ ,
'u' : '\u2049' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /!!/ ,
'u' : '\u203C' ,
'n' : - 1
2014-04-08 01:56:12 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /--/ ,
'u' : '\u2014' ,
'n' : - 1
2014-04-08 01:56:12 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /~~/ ,
'u' : '\u2053' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
}
] ,
2015-05-23 22:26:23 +05:00
'emotions' : [
2014-04-07 19:11:41 +03:00
{
2015-05-23 22:26:23 +05:00
'k' : /:.{0,1}D/ ,
'u' : '\uD83D\uDE03' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /(0|O):-{0,1}\)/i ,
'u' : '\uD83D\uDE07' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /:beer:/ ,
'u' : '\uD83C\uDF7A' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /3:-{0,1}\)/ ,
'u' : '\uD83D\uDE08' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /<3/ ,
'u' : '\u2764' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
2014-04-09 11:24:32 -03:00
// disabled due to urls :/
// {
2015-05-23 22:26:23 +05:00
// 'k': /o.O|:\/|:\\/,
// 'u': '\uD83D\uDE15',
// 'n': -1
2014-04-09 11:24:32 -03:00
// },
2014-04-07 19:11:41 +03:00
{
2015-05-23 22:26:23 +05:00
'k' : /:\'\(/ ,
'u' : '\uD83D\uDE22' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /(:|=)-{0,1}\(/ ,
'u' : '\uD83D\uDE1E' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /8(\)<|\|)/ ,
'u' : '\uD83D\uDE0E' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /(:|=)-{0,1}(\)|\])/ ,
'u' : '\uD83D\uDE0A' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /(\(|\[)-{0,1}(:|=)/ ,
'u' : '\uD83D\uDE0A' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /:\*/ ,
'u' : '\uD83D\uDE17' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /\^-{0,1}\^/ ,
'u' : '\uD83D\uDE06' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /:p/i ,
'u' : '\uD83D\uDE1B' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /;-{0,1}\)/ ,
'u' : '\uD83D\uDE09' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /\(-{0,1};/ ,
'u' : '\uD83D\uDE09' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /:(O|0)/ ,
'u' : '\uD83D\uDE2E' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /:@/ ,
'u' : '\uD83D\uDE31' ,
'n' : - 1
2014-04-08 01:56:12 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /:\|/ ,
'u' : '\uD83D\uDE10' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
}
] ,
2015-05-23 22:26:23 +05:00
'signs' : [
2014-04-07 19:11:41 +03:00
{
2015-05-23 22:26:23 +05:00
'k' : / tel(|:|=)/i ,
'u' : ' \u2121' ,
'n' : 4
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /^tel(|:|=)/i ,
'u' : '\u2121' ,
'n' : 3
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : / fax(|:|=)/i ,
'u' : ' \u213B' ,
'n' : 4
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /^fax(|:|=)/i ,
'u' : '\u213B' ,
'n' : 3
2014-04-07 19:11:41 +03:00
}
] ,
2015-05-23 22:26:23 +05:00
'fractions' : [
2014-04-07 19:11:41 +03:00
{
2015-05-23 22:26:23 +05:00
'k' : /1\/2/ ,
'u' : '\u00BD' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /1\/3/ ,
'u' : '\u2153' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /2\/3/ ,
'u' : '\u2154' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /1\/4/ ,
'u' : '\u00BC' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /3\/4/ ,
'u' : '\u00BE' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /1\/5/ ,
'u' : '\u2155' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /2\/5/ ,
'u' : '\u2156' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /3\/5/ ,
'u' : '\u2157' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /4\/5/ ,
'u' : '\u2158' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /1\/6/ ,
'u' : '\u2159' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /5\/6/ ,
'u' : '\u215A' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /1\/7/ ,
'u' : '\u2150' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /1\/8/ ,
'u' : '\u215B' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /3\/8/ ,
'u' : '\u215C' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /5\/8/ ,
'u' : '\u215D' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /7\/8/ ,
'u' : '\u215E' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /1\/9/ ,
'u' : '\u2151' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
} ,
{
2015-05-23 22:26:23 +05:00
'k' : /1\/10/ ,
'u' : '\u2152' ,
'n' : - 1
2014-04-07 19:11:41 +03:00
}
2015-05-23 22:26:23 +05:00
] } ;
2014-04-07 19:11:41 +03:00
2014-07-31 20:43:01 +02:00
// Marks ranges in a message where unicode replacements will be ignored (inside URLs).
2015-05-23 22:26:23 +05:00
function getRangesForUnicodeConversion ( msg ) {
if ( ! msg )
return ;
2014-10-16 10:47:08 +02:00
2014-07-31 20:43:01 +02:00
var tempMsg = msg ;
var results = [ ] ;
var regexHttpStart = /http[s]?:\/\// ;
var regexHttpEnd = /[ \n\t]/ ;
2015-05-23 22:26:23 +05:00
var start = 0 , end , position , rep = true ;
2014-10-16 10:47:08 +02:00
2014-07-31 20:43:01 +02:00
position = tempMsg . search ( regexHttpStart ) ;
2015-05-23 22:26:23 +05:00
while ( position !== - 1 ) {
2014-07-31 20:43:01 +02:00
end = start + position ;
2015-05-23 22:26:23 +05:00
if ( end > start )
2014-07-31 20:43:01 +02:00
results . push ( { start : start , end : end , replace : rep } ) ;
rep = ! rep ;
start = end ;
tempMsg = tempMsg . substring ( position , tempMsg . length ) ;
2014-10-16 10:47:08 +02:00
2015-05-23 22:26:23 +05:00
if ( rep === true )
2014-07-31 20:43:01 +02:00
position = tempMsg . search ( regexHttpStart ) ;
else
position = tempMsg . search ( regexHttpEnd ) ;
2014-04-07 19:11:41 +03:00
}
2015-05-23 22:26:23 +05:00
2014-07-31 20:43:01 +02:00
end = msg . length ;
2015-05-23 22:26:23 +05:00
if ( end > start )
2014-07-31 20:43:01 +02:00
results . push ( { start : start , end : end , replace : rep } ) ;
2014-10-16 10:47:08 +02:00
return results ;
2014-07-31 20:43:01 +02:00
}
2014-04-07 19:11:41 +03:00
2015-05-23 22:26:23 +05:00
function getUnicodeReplacement ( msg , list , ranges , ta ) {
if ( ! msg || ! list || ! ranges )
return ;
if ( ranges . length === 0 )
return '' ;
2014-10-16 10:47:08 +02:00
2014-07-31 20:43:01 +02:00
var position , substrings = [ ] ;
2015-05-23 22:26:23 +05:00
for ( var j = 0 ; j < ranges . length ; j ++ ) {
2014-07-31 20:43:01 +02:00
substrings [ j ] = msg . substring ( ranges [ j ] . start , ranges [ j ] . end ) ;
2015-05-23 22:26:23 +05:00
if ( ranges [ j ] . replace === true ) {
for ( var i = 0 ; i < list . length ; i ++ ) {
2014-07-31 20:43:01 +02:00
position = substrings [ j ] . search ( list [ i ] . k ) ;
2015-05-23 22:26:23 +05:00
if ( position !== - 1 && ta . data ( 'disabledUnicodeRules' ) . indexOf ( list [ i ] . u ) === - 1 ) {
2014-07-31 20:43:01 +02:00
var oldSubstring = substrings [ j ] ;
substrings [ j ] = substrings [ j ] . replace ( list [ i ] . k , list [ i ] . u ) ;
2014-10-16 10:47:08 +02:00
2014-07-31 20:43:01 +02:00
var len = oldSubstring . length - substrings [ j ] . length + list [ i ] . u . length ;
2015-05-23 22:26:23 +05:00
ta . data ( 'unicodeConversionStack' ) . unshift ( {
'k' : oldSubstring . substr ( position , len ) ,
'u' : list [ i ] . u ,
'p' : ranges [ j ] . start + position
2014-07-31 20:43:01 +02:00
} ) ;
}
}
}
}
var returnString = substrings [ 0 ] ;
2015-05-23 22:26:23 +05:00
for ( var j = 1 ; j < ranges . length ; j ++ ) {
2014-07-31 20:43:01 +02:00
returnString += substrings [ j ] ;
}
return returnString ;
}
2014-04-07 19:11:41 +03:00
2015-05-23 22:26:23 +05:00
function convert2Unicodes ( s , ta ) {
if ( ! ta . data ( 'unicodeConversionStack' ) ) // A stack of undo steps
ta . data ( 'unicodeConversionStack' , [ ] ) ;
if ( ! ta . data ( 'disabledUnicodeRules' ) ) // A list of conversion rules that are temporarily disabled
ta . data ( 'disabledUnicodeRules' , [ ] ) ;
2014-07-31 20:43:01 +02:00
var ranges = getRangesForUnicodeConversion ( s ) ;
2014-10-16 10:47:08 +02:00
2015-05-23 22:26:23 +05:00
if ( $ . Options . unicodeConversion . val === 'enable' || $ . Options . convertPunctuationsOpt . val )
s = getUnicodeReplacement ( s , unicodeConversionList . punctuation , ranges , ta ) ;
if ( $ . Options . unicodeConversion . val === 'enable' || $ . Options . convertEmotionsOpt . val )
s = getUnicodeReplacement ( s , unicodeConversionList . emotions , ranges , ta ) ;
if ( $ . Options . unicodeConversion . val === 'enable' || $ . Options . convertSignsOpt . val )
s = getUnicodeReplacement ( s , unicodeConversionList . signs , ranges , ta ) ;
if ( $ . Options . unicodeConversion . val === 'enable' || $ . Options . convertFractionsOpt . val )
s = getUnicodeReplacement ( s , unicodeConversionList . fractions , ranges , ta ) ;
if ( ta . data ( 'unicodeConversionStack' ) . length > 0 ) {
var ub = ta . closest ( '.post-area-new' ) . find ( '.undo-unicode' ) ;
ub . text ( polyglot . t ( 'undo' ) + ': ' + ta . data ( 'unicodeConversionStack' ) [ 0 ] . u ) ;
2014-04-07 19:11:41 +03:00
$ . MAL . enableButton ( ub ) ;
2015-05-19 23:33:59 +05:00
} else
2015-05-23 22:26:23 +05:00
$ . MAL . disableButton ( ta . closest ( '.post-area-new' ) . find ( '.undo-unicode' ) ) ;
2014-04-07 19:11:41 +03:00
2014-07-31 20:43:01 +02:00
return s ;
2014-04-07 19:11:41 +03:00
}
function undoLastUnicode ( e ) {
e . stopPropagation ( ) ;
e . preventDefault ( ) ;
2015-05-23 22:26:23 +05:00
var $ta = $ ( this ) . closest ( '.post-area-new' ) . find ( 'textarea' ) ;
if ( $ta . data ( 'unicodeConversionStack' ) . length === 0 )
2014-04-07 19:11:41 +03:00
return ;
2015-05-23 22:26:23 +05:00
var uc = $ta . data ( 'unicodeConversionStack' ) . shift ( ) ;
2014-04-07 19:11:41 +03:00
var pt = $ta . val ( ) ;
2014-10-16 10:47:08 +02:00
2014-07-31 20:43:01 +02:00
// If the text was shifted, and character is no longer at the saved position, this function
// searches for it to the right. If it is not there, it searches in the oposite direction.
// if it's not there either, it means it was deleted, so it is skipped.
var substrLeft = pt . substring ( 0 , uc . p ) ;
var substrRight = pt . substring ( uc . p , pt . length ) ;
2015-05-23 22:26:23 +05:00
if ( substrRight . search ( uc . u ) !== - 1 ) {
2014-07-31 20:43:01 +02:00
substrRight = substrRight . replace ( uc . u , uc . k ) ;
$ta . val ( substrLeft + substrRight ) ;
2015-05-23 22:26:23 +05:00
$ta . data ( 'disabledUnicodeRules' ) . push ( uc . u ) ;
} else if ( substrLeft . search ( uc . u ) !== - 1 ) {
2014-07-31 20:43:01 +02:00
var closestToTheLeft = substrLeft . lastIndexOf ( uc . u ) ;
var substrCenter = substrLeft . substring ( closestToTheLeft , substrLeft . length ) . replace ( uc . u , uc . k ) ;
substrLeft = substrLeft . substring ( 0 , closestToTheLeft ) ;
$ta . val ( substrLeft + substrCenter + substrRight ) ;
2015-05-23 22:26:23 +05:00
$ta . data ( 'disabledUnicodeRules' ) . push ( uc . u ) ;
2014-04-07 20:22:47 +03:00
}
2014-04-07 19:11:41 +03:00
2015-05-23 22:26:23 +05:00
if ( $ta . data ( 'unicodeConversionStack' ) . length > 0 )
$ ( this ) . text ( polyglot . t ( 'undo' ) + ': ' + $ta . data ( 'unicodeConversionStack' ) [ 0 ] . u ) ;
else {
$ ( this ) . text ( 'undo' ) ;
2014-04-07 19:11:41 +03:00
$ . MAL . disableButton ( $ ( this ) ) ;
2014-07-31 20:43:01 +02:00
}
2014-04-07 19:11:41 +03:00
}
2015-05-23 22:26:23 +05:00
function postSubmit ( e , oldLastPostId ) {
2015-08-12 22:10:42 +05:00
var btnPostSubmit ;
2014-05-19 22:00:31 +03:00
if ( e instanceof $ ) {
2015-08-12 22:10:42 +05:00
btnPostSubmit = e ;
2014-05-19 22:00:31 +03:00
//check if previous part was sent...
2015-05-23 22:26:23 +05:00
if ( oldLastPostId === lastPostId ) {
2015-08-12 22:10:42 +05:00
setTimeout ( postSubmit , 1000 , btnPostSubmit , oldLastPostId ) ;
2014-05-20 16:17:31 +03:00
return ;
}
2014-05-19 22:00:31 +03:00
} else {
2014-04-17 01:21:17 +03:00
e . stopPropagation ( ) ;
e . preventDefault ( ) ;
2015-08-12 22:10:42 +05:00
btnPostSubmit = $ ( this ) ;
2014-04-17 01:21:17 +03:00
}
2015-08-12 22:10:42 +05:00
$ . MAL . disableButton ( btnPostSubmit ) ;
2014-05-19 22:00:31 +03:00
2015-08-12 22:10:42 +05:00
var textArea = btnPostSubmit . closest ( '.post-area-new' ) . find ( 'textarea' ) ;
2013-12-30 12:42:36 -02:00
2015-08-20 17:31:16 +05:00
textArea . siblings ( '#post-preview' ) . slideUp ( 'fast' ) ;
2015-07-25 02:19:52 +05:00
2015-08-12 22:10:42 +05:00
var postData = btnPostSubmit . closest ( '.post-data' ) ;
if ( ! postData . length ) {
postData = btnPostSubmit . closest ( '.modal-content' ) . find ( '.post-data' ) ;
2014-01-10 16:23:39 -08:00
}
2014-04-17 01:21:17 +03:00
2015-08-12 22:10:42 +05:00
if ( btnPostSubmit . hasClass ( 'with-reference' ) ) {
2018-10-26 01:35:20 +05:00
var doSubmitPost = function ( postText , postDataElem ) {
2019-01-23 06:24:32 +05:00
newRtMsg ( JSON . parse ( postDataElem . attr ( 'data-content_to_rt' ) ) ,
2018-11-02 01:59:38 +05:00
postDataElem . attr ( 'data-content_to_sigrt' ) , postText , updateRTsWithOwnOne
) ;
2018-10-26 01:35:20 +05:00
} ;
2015-08-12 22:10:42 +05:00
} else {
if ( splitedPostsCount > 1 ) {
if ( textArea . length < splitedPostsCount ) {
//current part will be sent as reply to the previous part...
2018-10-21 06:04:01 +05:00
postData = $ ( '<div data-id="' + ( oldLastPostId + 1 ) . toString ( ) // (lastPostId - oldLastPostId) may be more than 1 because of an async nature of getting of the confirmation of sending
+ '" data-screen-name="' + defaultScreenName
2018-10-21 06:03:23 +05:00
+ '" data-reply-part-id="' + ( splitedPostsCount - textArea . length ) . toString ( )
+ '"></div>' ) ;
2015-08-12 22:10:42 +05:00
}
}
2018-10-21 06:03:23 +05:00
if ( postData . length )
var doSubmitPost = function ( postText , postDataElem ) {
var reply _n = postDataElem . attr ( 'data-screen-name' ) ;
var reply _k = parseInt ( postDataElem . attr ( 'data-id' ) ) ;
var part _id = postDataElem . attr ( 'data-reply-part-id' ) ;
var cbFunc = function ( req , ret ) {
2018-10-28 23:25:36 +05:00
var postDataElem = getElem ( '.expanded-post .post-data'
2018-10-21 06:03:23 +05:00
+ '[data-screen-name=\'' + req . reply _n + '\']'
+ '[data-id=\'' + req . reply _k + '\']' ) ;
if ( ! postDataElem . length ) {
if ( req . part _id && ++ req . tries < 5 )
setTimeout ( req . cbFunc , 2000 , req ) ;
return ;
}
2018-10-28 23:25:36 +05:00
for ( var k = 0 , twistElem = undefined ; k < postDataElem . length ; k ++ ) {
var formerPostElem = postDataElem . eq ( k ) . closest ( 'li.post' ) ;
if ( ! formerPostElem . next ( ) . hasClass ( 'post-replies' ) )
var containerElem = $ ( '<li class="post-replies"><ol class="sub-replies"></ol></li>' ) // FIXME replace with template as like as a reqRepAfterCB()'s similar thing
. insertAfter ( formerPostElem )
. children ( '.sub-replies' )
;
else {
var containerElem = formerPostElem . next ( ) . children ( '.sub-replies' ) ;
if ( containerElem . find ( '.post-data'
+ '[data-screen-name=\'' + ret . userpost . n + '\']'
+ '[data-id=\'' + ret . userpost . k + '\']' ) . length )
continue ;
}
2018-10-21 06:03:23 +05:00
2018-10-28 23:25:36 +05:00
if ( typeof twistElem !== 'undefined' )
twistElem . clone ( true )
. appendTo ( containerElem )
. slideDown ( 'fast' )
;
else
twistElem = postToElem ( ret , 'related' ) . hide ( )
. addClass ( 'new' )
. appendTo ( containerElem )
. slideDown ( 'fast' )
;
}
2018-10-21 06:03:23 +05:00
} ;
newPostMsg ( postText , reply _n , reply _k ,
cbFunc , { reply _n : reply _n , reply _k : reply _k , part _id : part _id ,
cbFunc : cbFunc , tries : 0 }
) ;
} ;
else
var doSubmitPost = function ( postText ) {
newPostMsg ( postText ) ;
} ;
2014-04-17 01:21:17 +03:00
}
2015-08-12 22:10:42 +05:00
if ( textArea . length <= 1 ) {
2014-05-08 23:28:16 +03:00
if ( splitedPostsCount > 1 ) {
2015-08-12 22:10:42 +05:00
var postText = '' ;
var reply _to = textArea . attr ( 'data-reply-to' ) ;
var val = textArea . val ( ) ;
2016-07-16 18:15:50 -03:00
if ( typeof reply _to === 'undefined' || checkPostForMentions ( val , reply _to , $ . Options . MaxPostEditorChars . val ) )
2015-08-12 22:10:42 +05:00
postText = val + ' (' + splitedPostsCount . toString ( ) + '/' + splitedPostsCount . toString ( ) + ')' ;
2014-05-08 23:28:16 +03:00
else
2015-08-12 22:10:42 +05:00
postText = reply _to + val + ' (' + splitedPostsCount . toString ( ) + '/' + splitedPostsCount . toString ( ) + ')' ;
2014-05-08 23:28:16 +03:00
2015-08-12 22:10:42 +05:00
doSubmitPost ( postText , postData ) ;
2014-05-08 23:28:16 +03:00
} else
2015-08-12 22:10:42 +05:00
doSubmitPost ( textArea . val ( ) , postData ) ;
2014-04-17 01:21:17 +03:00
splitedPostsCount = 1 ;
} else {
2015-08-12 22:10:42 +05:00
var postText = '' ;
var reply _to = textArea . attr ( 'data-reply-to' ) ;
var val = textArea [ 0 ] . value ;
2016-07-16 18:15:50 -03:00
if ( typeof reply _to === 'undefined' || checkPostForMentions ( val , reply _to , $ . Options . MaxPostEditorChars . val ) )
2015-08-12 22:10:42 +05:00
postText = val + ' (' + ( splitedPostsCount - textArea . length + 1 ) . toString ( ) + '/' + splitedPostsCount . toString ( ) + ')' ;
2014-05-11 22:18:33 +03:00
else
2015-08-12 22:10:42 +05:00
postText = reply _to + val + ' (' + ( splitedPostsCount - textArea . length + 1 ) . toString ( ) + '/' + splitedPostsCount . toString ( ) + ')' ;
2014-05-08 23:28:16 +03:00
2015-08-12 22:10:42 +05:00
$ ( textArea [ 0 ] ) . remove ( ) ;
2014-04-17 01:21:17 +03:00
2015-08-12 22:10:42 +05:00
doSubmitPost ( postText , postData ) ;
2016-01-08 23:58:56 +05:00
setTimeout ( postSubmit , 1000 , btnPostSubmit , lastPostId ) ;
2014-04-17 01:21:17 +03:00
return ;
}
2013-12-30 12:42:36 -02:00
2015-10-24 01:39:22 +05:00
if ( btnPostSubmit . closest ( '.prompt-wrapper' ) . length )
2016-01-31 11:23:07 +05:00
closePrompt ( btnPostSubmit ) ;
2015-08-12 22:10:42 +05:00
else {
textArea . val ( '' ) . attr ( 'placeholder' , polyglot . t ( 'Your message was sent!' ) ) ;
2016-07-16 18:15:50 -03:00
btnPostSubmit . closest ( 'form' ) . find ( '.post-area-remaining' ) . text ( '' ) ;
2015-08-12 22:10:42 +05:00
2018-10-20 04:43:24 +05:00
if ( btnPostSubmit . closest ( '.post-area,.post-reply-content' ) . length ) {
btnPostSubmit . closest ( '.post-area-new' ) . removeClass ( 'open' )
2019-01-23 06:24:32 +05:00
. find ( 'textarea' ) . trigger ( 'blur' ) ;
2018-10-20 04:43:24 +05:00
}
2015-08-12 22:10:42 +05:00
textArea . data ( 'unicodeConversionStack' , [ ] ) ;
textArea . data ( 'disabledUnicodeRules' , [ ] ) ;
}
2013-12-30 12:42:36 -02:00
}
2016-01-31 11:23:07 +05:00
function retweetSubmit ( event ) {
event . preventDefault ( ) ;
event . stopPropagation ( ) ;
2013-12-30 12:42:36 -02:00
2016-01-31 11:23:07 +05:00
var prompt = $ ( event . target ) . closest ( '.prompt-wrapper' ) ;
2018-10-26 01:35:20 +05:00
var postDataElem = prompt . find ( '.post-data' ) ;
2019-01-23 06:24:32 +05:00
newRtMsg ( JSON . parse ( postDataElem . attr ( 'data-content_to_rt' ) ) ,
2018-11-02 01:59:38 +05:00
postDataElem . attr ( 'data-content_to_sigrt' ) , '' , updateRTsWithOwnOne
) ;
2013-12-30 12:42:36 -02:00
2016-01-31 11:23:07 +05:00
closePrompt ( prompt ) ;
2013-12-30 12:42:36 -02:00
}
2016-09-06 17:49:29 +03:00
function favSubmit ( event ) {
event . preventDefault ( ) ;
event . stopPropagation ( ) ;
var prompt = $ ( event . target ) . closest ( '.prompt-wrapper' ) ;
var priv = ( event . target . className . indexOf ( 'private' ) > - 1 ) ;
newFavMsg ( prompt . find ( '.post-data' ) , priv ) ;
closePrompt ( prompt ) ;
}
2014-04-10 18:57:49 -03:00
function changeStyle ( ) {
var style , profile , menu ;
2015-05-19 23:33:59 +05:00
var theme = $ . Options . theme . val ;
2014-09-19 19:09:08 -03:00
2015-05-23 22:26:23 +05:00
if ( theme === 'nin' ) {
2014-09-19 19:09:08 -03:00
style = 'theme_nin/css/style.css' ;
profile = 'theme_nin/css/profile.css' ;
2019-02-02 19:44:11 +05:00
// we use .ajax because .getScript requires 'unsafe-inline' CSP rule for now, see https://github.com/jquery/jquery/issues/3969
2019-02-02 14:14:03 +05:00
$ . ajax ( { dataType : 'text' , url : 'theme_nin/js/theme_option.js' } )
2019-02-02 19:44:11 +05:00
. done ( function ( res ) { eval ( res ) ; } ) ;
2022-07-08 17:53:21 +03:00
} else if ( theme === 'nin_night' ) {
style = 'theme_nin_night/css/style.css' ;
profile = 'theme_nin_night/css/profile.css' ;
$ . ajax ( { dataType : 'text' , url : 'theme_nin_night/js/theme_option.js' } )
. done ( function ( res ) { eval ( res ) ; } ) ;
2021-05-10 01:32:15 +03:00
} else if ( theme === 'nin_original' ) {
theme = 'nin' ; // related to native theme in class definitions, so easiest way to integrate original version
style = 'theme_nin_original/css/style.css' ;
profile = 'theme_nin_original/css/profile.css' ;
$ . ajax ( { dataType : 'text' , url : 'theme_nin_original/js/theme_option.js' } )
. done ( function ( res ) { eval ( res ) ; } ) ;
2015-05-23 22:26:23 +05:00
} else if ( theme === 'calm' ) {
2014-04-10 18:57:49 -03:00
style = 'theme_calm/css/style.css' ;
profile = 'theme_calm/css/profile.css' ;
2015-05-23 22:26:23 +05:00
} else if ( theme === 'original' ) {
2014-09-19 19:09:08 -03:00
style = 'css/style.css' ;
profile = 'css/profile.css' ;
2019-02-02 19:44:11 +05:00
$ . ajax ( { dataType : 'text' , url : 'theme_original/js/theme_option.js' } )
. done ( function ( res ) { eval ( res ) ; } ) ;
2014-09-19 19:09:08 -03:00
}
2014-11-04 22:34:42 +06:00
2014-04-10 18:57:49 -03:00
$ ( '#stylecss' ) . attr ( 'href' , style ) ;
$ ( '#profilecss' ) . attr ( 'href' , profile ) ;
2015-05-23 22:26:23 +05:00
$ ( '<style type="text/css"> .selectable_theme:not(.theme_' + theme + ')' +
'{display:none!important;}\n</style>' ) . appendTo ( 'head' ) ;
2019-01-23 06:24:32 +05:00
setTimeout ( function ( ) { console . log ( $ ( menu ) ) ; $ ( menu ) . removeAttr ( 'style' ) ; } , 0 ) ;
2014-04-10 18:57:49 -03:00
}
2013-12-30 12:42:36 -02:00
2015-03-30 05:16:41 +05:00
function getMentionsForAutoComplete ( ) {
2015-05-23 22:26:23 +05:00
if ( defaultScreenName && typeof followingUsers !== 'undefined' ) {
2015-03-30 05:16:41 +05:00
var suggests = followingUsers . slice ( ) ;
if ( suggests . indexOf ( defaultScreenName ) > - 1 )
suggests . splice ( suggests . indexOf ( defaultScreenName ) , 1 ) ;
if ( suggests . length > 0 ) {
suggests . sort ( ) ;
return [ {
mentions : suggests ,
match : /\B@(\w*)$/ ,
search : function ( term , callback ) {
callback ( $ . map ( this . mentions , function ( mention ) {
return mention . indexOf ( term ) === 0 ? mention : null ;
} ) ) ;
} ,
index : 1 ,
replace : function ( mention ) {
return '@' + mention + ' ' ;
}
} ] ;
2014-04-10 18:57:49 -03:00
}
}
}
2013-12-30 12:42:36 -02:00
2014-05-20 19:38:34 +03:00
function replaceDashboards ( ) {
2015-05-23 22:26:23 +05:00
var width = $ ( window ) . width ( ) ;
var wrapper = $ ( '.wrapper' ) ;
if ( width >= 1200 && ! wrapper . hasClass ( 'w1200' ) ) {
wrapper . addClass ( 'w1200' ) ;
2014-05-20 19:38:34 +03:00
$ ( '.userMenu' ) . addClass ( 'w1200' ) ;
2015-04-21 02:00:05 +05:00
$ ( '.module.who-to-follow' ) . detach ( ) . appendTo ( $ ( '.dashboard.right' ) ) ;
$ ( '.module.twistday-reminder' ) . detach ( ) . appendTo ( $ ( '.dashboard.right' ) ) ;
2015-11-19 01:09:04 +05:00
$ ( '#modals-minimized' ) . addClass ( 'w1200' ) ;
2015-05-23 22:26:23 +05:00
} else if ( width < 1200 && wrapper . hasClass ( 'w1200' ) ) {
wrapper . removeClass ( 'w1200' ) ;
2014-05-20 19:38:34 +03:00
$ ( '.userMenu' ) . removeClass ( 'w1200' ) ;
2015-04-21 02:00:05 +05:00
$ ( '.module.who-to-follow' ) . detach ( ) . insertAfter ( $ ( '.module.mini-profile' ) ) ;
$ ( '.module.twistday-reminder' ) . detach ( ) . insertAfter ( $ ( '.module.toptrends' ) ) ;
2015-11-19 01:09:04 +05:00
$ ( '#modals-minimized' ) . removeClass ( 'w1200' ) ;
2014-05-20 19:38:34 +03:00
}
}
2013-12-30 12:42:36 -02:00
function initInterfaceCommon ( ) {
2016-09-11 17:15:58 +05:00
twister . tmpl . modalComponentWarn = extractTemplate ( '#template-inline-warn' ) ;
twister . tmpl . modalComponentWarn . find ( '.close' ) . on ( 'click' ,
function ( event ) {
var i = $ ( event . target ) . closest ( '.modal-wrapper' ) . attr ( 'data-modal-id' ) ;
if ( ! i || ! twister . modal [ i ] ) return ;
var modal = twister . modal [ i ] ;
modal . self . find ( '.inline-warn' ) . hide ( ) ;
modal . content . outerHeight ( modal . self . height ( ) - modal . self . find ( '.modal-header' ) . outerHeight ( ) ) ;
var windowHeight = $ ( window ) . height ( ) ;
if ( modal . self . outerHeight ( ) > windowHeight ) {
modal . content . outerHeight ( modal . content . outerHeight ( ) - modal . self . outerHeight ( ) + windowHeight ) ;
modal . self . outerHeight ( windowHeight ) ;
modal . self . css ( 'margin-top' , - windowHeight / 2 ) ;
}
}
) ;
2016-09-11 17:26:53 +05:00
twister . tmpl . modalComponentWarn . find ( '.options .never-again' ) . on ( 'change' ,
function ( event ) {
$ . Options . set ( 'skipWarn' + $ ( event . target ) . closest ( '.inline-warn' )
. attr ( 'data-warn-name' ) , event . target . checked ) ; // e.g. 'skipWarnFollowersNotAll'
}
) ;
2016-02-03 22:04:53 +05:00
twister . tmpl . commonDMsList = extractTemplate ( '#template-direct-messages-list' ) ;
2016-09-06 13:54:11 +05:00
twister . tmpl . uriShortenerMC = extractTemplate ( '#template-uri-shortener-modal-content' ) ;
twister . tmpl . uriShortenerMC
. find ( '.shorten-uri' ) . on ( 'click' ,
{ cbFunc :
function ( long , short ) {
if ( short ) {
var urisList = getElem ( '.uri-shortener-modal .uris-list' ) ;
if ( urisList . length ) {
var item = urisList . find ( '.short:contains("' + short + '")' ) . closest ( 'li' ) ;
if ( ! item . length ) {
item = twister . tmpl . uriShortenerUrisListItem . clone ( true ) ;
item . find ( '.short' ) . text ( short ) ;
item . find ( '.long' ) . text ( long ) . attr ( 'href' , long ) ;
item . appendTo ( urisList ) ;
}
urisList . children ( '.highlighted' ) . removeClass ( 'highlighted' ) ;
item . addClass ( 'highlighted' ) ;
var mc = urisList . closest ( '.modal-content' ) ;
mc . scrollTop ( item . offset ( ) . top - mc . offset ( ) . top + mc . scrollTop ( ) ) ;
}
showURIPair ( long , short ) ;
} else
showURIShortenerErrorRPC ( short ) ;
}
} ,
function ( event ) {
muteEvent ( event ) ;
openRequestShortURIForm ( event ) ;
}
)
. siblings ( '.clear-cache' ) . on ( 'click' ,
function ( ) {
confirmPopup ( {
txtMessage : polyglot . t ( 'confirm_uri_shortener_clear_cache' ) ,
cbConfirm : function ( ) {
twister . URIs = { } ;
$ . localStorage . set ( 'twistaURIs' , twister . URIs ) ;
getElem ( '.uri-shortener-modal .uris-list' ) . empty ( ) ;
}
} ) ;
}
)
;
twister . tmpl . uriShortenerUrisListItem = extractTemplate ( '#template-uri-shortener-uris-list-item' )
. on ( 'click' , function ( event ) {
var elem = $ ( event . target ) ;
elem . closest ( '.uris-list' ) . children ( '.highlighted' ) . removeClass ( 'highlighted' ) ;
elem . addClass ( 'highlighted' ) ;
} )
;
2016-02-03 22:04:53 +05:00
2017-02-02 14:17:10 +05:00
getElem ( '.modal-wrapper .modal-close, .modal-wrapper .modal-blackout' ) . on ( 'click' , closeModal ) ;
2015-01-21 12:10:04 +01:00
2015-11-19 01:09:04 +05:00
$ ( '.minimize-modal' ) . on ( 'click' , function ( event ) {
minimizeModal ( $ ( event . target ) . closest ( '.modal-wrapper' ) ) ;
} ) ;
2015-07-19 05:23:50 +05:00
$ ( '.modal-back' ) . on ( 'click' , function ( ) { history . back ( ) ; } ) ;
2015-09-04 03:21:33 +05:00
$ ( '.prompt-close' ) . on ( 'click' , closePrompt ) ;
2015-01-22 15:59:00 +01:00
2017-01-23 04:23:44 +05:00
getElem ( 'button.follow' , true ) . on ( 'click' , clickFollow ) ;
2016-01-31 11:23:07 +05:00
$ ( '.following-config-method-buttons .public-following' ) . on ( 'click' , function ( event ) {
setFollowingMethod ( event ) ;
closePrompt ( event ) ;
2014-04-10 18:57:49 -03:00
} ) ;
2015-01-21 12:10:04 +01:00
2016-07-31 02:37:52 +05:00
$ ( '.module.mini-profile .open-followers' ) . on ( 'mouseup' , { route : '#followers' } , routeOnClick ) ;
2016-02-15 22:27:37 +05:00
2016-02-03 01:29:41 +05:00
$ ( '.post-text' ) . on ( 'click' , 'a' , muteEvent ) ;
2015-05-23 22:26:23 +05:00
$ ( '.userMenu-config' ) . clickoutside ( closeThis . bind ( $ ( '.config-menu' ) ) ) ;
$ ( '.userMenu-config-dropdown' ) . on ( 'click' , dropDownMenu ) ;
$ ( '.post-area-new' )
2019-01-23 06:24:32 +05:00
. on ( 'click' , function ( e ) { composeNewPost ( e , $ ( this ) ) ; } )
2016-03-23 21:46:02 +05:00
. clickoutside ( unfocusPostAreaNew )
2015-05-23 22:26:23 +05:00
. children ( 'textarea' )
. on ( {
2019-01-24 15:58:37 +05:00
'focus' : prepareTextareaToInput ,
2015-05-23 22:26:23 +05:00
'input' : replyTextInput , // input event fires in modern browsers (IE9+) on any changes in textarea (and copypasting with mouse too)
'input focus' : replyTextUpdateRemaining ,
'keyup' : replyTextKeySend
} )
;
$ ( '.post-submit' ) . on ( 'click' , postSubmit ) ;
$ ( '.modal-propagate' ) . on ( 'click' , retweetSubmit ) ;
2016-09-06 17:49:29 +03:00
$ ( '.modal-fav-public' ) . on ( 'click' , favSubmit ) ;
$ ( '.modal-fav-private' ) . on ( 'click' , favSubmit ) ;
2015-05-19 23:33:59 +05:00
if ( $ . Options . unicodeConversion . val === 'disable' )
2015-05-23 22:26:23 +05:00
$ ( '.undo-unicode' ) . on ( 'click' , undoLastUnicode ) . css ( 'display' , 'none' ) ;
2014-04-07 19:11:41 +03:00
else
2015-05-23 22:26:23 +05:00
$ ( '.undo-unicode' ) . on ( 'click' , undoLastUnicode ) ;
2014-04-07 19:11:41 +03:00
2016-03-25 04:06:15 +05:00
getElem ( '.open-profile-modal' , true )
. on ( 'click' , muteEvent ) . on ( 'mouseup' , handleClickOpenProfileModal ) ;
2015-05-23 22:26:23 +05:00
//$('.open-hashtag-modal').on('click', openHashtagModal);
//$('.open-following-modal').on('click', openFollowingModal);
$ ( '.userMenu-connections a' ) . on ( 'click' , openMentionsModal ) ;
$ ( '.mentions-from-user' ) . on ( 'click' , openMentionsModal ) ;
2016-09-06 17:49:29 +03:00
$ ( '.userMenu-favs a' ) . on ( 'click' , openFavsModal ) ;
$ ( '.favs-from-user' ) . on ( 'click' , openFavsModal ) ;
2013-12-30 12:42:36 -02:00
2016-09-27 04:04:35 +05:00
getElem ( '.latest-activity' , true ) . on ( 'mouseup' ,
{ feeder : '.latest-activity' } , handleClickOpenConversation ) ;
2016-09-22 18:14:40 +03:00
2014-05-20 19:38:34 +03:00
replaceDashboards ( ) ;
2019-01-23 06:24:32 +05:00
$ ( window ) . on ( 'resize' , replaceDashboards ) ;
2015-04-14 22:39:20 +05:00
2015-10-24 01:20:41 +05:00
$ ( '.profile-card .profile-bio .group-description' )
. on ( 'focus' , function ( event ) {
$ ( event . target )
. siblings ( '.save' ) . show ( )
. siblings ( '.cancel' ) . show ( )
;
} )
. on ( 'input' ,
{ parentSelector : '.profile-bio' , enterSelector : '.save' } , inputEnterActivator )
. siblings ( '.save' ) . on ( 'click' , function ( event ) {
var elemEvent = $ ( event . target ) ;
var descElem = elemEvent . siblings ( '.group-description' ) ;
groupMsgSetGroupDescription ( elemEvent . closest ( '.profile-card' ) . attr ( 'data-screen-name' ) ,
descElem . val ( ) . trim ( ) ,
function ( req ) {
req . descElem . attr ( 'val-origin' , req . descElem . val ( ) . trim ( ) )
. siblings ( '.save' ) . hide ( )
. siblings ( '.cancel' ) . hide ( )
;
} , { descElem : descElem }
) ;
} )
. siblings ( '.cancel' ) . on ( 'click' , function ( event ) { // FIXME it would be nice to bind some 'clickoutside' event instead and remove cancel button, but current implementation of that doesn't unbind events when element dies
var descElem = $ ( event . target ) . hide ( )
. siblings ( '.save' ) . hide ( )
. siblings ( '.group-description' )
;
descElem . val ( descElem . attr ( 'val-origin' ) ) ;
} )
;
2015-05-23 22:26:23 +05:00
$ ( '.tox-ctc' ) . on ( 'click' , promptCopyAttrData ) ;
$ ( '.bitmessage-ctc' ) . on ( 'click' , promptCopyAttrData ) ;
2015-03-30 05:16:41 +05:00
2016-09-06 13:54:11 +05:00
$ ( '.uri-shortener' ) . on ( 'mouseup' , { route : '#/uri-shortener' } , routeOnClick ) ;
2016-03-22 01:54:56 +05:00
2017-01-23 03:32:34 +05:00
$ ( '.updates-check-client' ) . text ( polyglot . t ( 'updates_check_client' ) )
. on ( 'mouseup' , function ( event ) {
muteEvent ( event ) ;
checkUpdatesClient ( true ) ;
}
) ;
2016-03-23 04:02:20 +05:00
$ ( '.post-area-new textarea' )
. on ( 'focus' ,
function ( event ) {
twister . focus . textareaPostCur = $ ( event . target ) ;
if ( $ . fn . textcomplete ) { // because some pages don't have that. // network.html
event . data = { req : getMentionsForAutoComplete } ;
setTextcompleteOnEventTarget ( event ) ;
}
}
)
. on ( 'focusout' ,
function ( event ) {
twister . focus . textareaPostCur = undefined ;
twister . focus . textareaPostPrev = $ ( event . target ) ;
if ( $ . fn . textcomplete ) // because some pages don't have that. // network.html
unsetTextcompleteOnEventTarget ( event ) ;
}
)
;
2019-01-23 10:07:54 +05:00
twister . tmpl . post = extractTemplate ( '#template-post' )
. on ( 'click' , function ( event ) {
if ( event . button === 0 && window . getSelection ( ) == 0 )
postExpandFunction ( event , $ ( this ) ) ;
} )
;
2022-09-01 14:26:00 +03:00
if ( $ . Options . TranslationEnabled . val !== 'enable' ) {
twister . tmpl . post . find ( '.post-translate' ) . remove ( ) ;
}
twister . tmpl . post . find ( '.post-translate' ) . on ( 'click' , postTranslateClick ) ;
2019-01-23 10:07:54 +05:00
twister . tmpl . post . find ( '.post-reply' ) . on ( 'click' , postReplyClick ) ;
twister . tmpl . post . find ( '.post-propagate' ) . on ( 'click' , reTwistPopup ) ;
twister . tmpl . post . find ( '.post-favorite' ) . on ( 'click' , favPopup ) ;
twister . tmpl . post . find ( '.expanded-content .show-more' )
. on ( 'mouseup' ,
{ feeder : '.module.post.original.open .module.post.original .post-data' } ,
handleClickOpenConversation
)
. on ( 'click' , muteEvent ) // to prevent post collapsing
;
twister . tmpl . post . find ( '.new-replies-available button' ) . hide ( )
. on ( 'click' , function ( event ) {
event . stopPropagation ( ) ;
$ ( event . target ) . hide ( )
. closest ( 'li.post' ) . next ( '.post-replies' ) . find ( '.post.pending' )
. removeClass ( 'pending' ) . slideDown ( 'fast' )
;
} )
;
twister . tmpl . post . find ( '.post-stats' ) . hide ( ) ;
2019-02-02 14:14:03 +05:00
twister . tmpl . accountMC = extractTemplate ( '#template-account-modal' ) ;
twister . tmpl . accountMC . find ( '.avatar' ) . on ( 'click' ,
function ( event ) {
$ ( event . target ) . closest ( '.modal-content' ) . find ( '.input-avatar' ) . trigger ( 'click' ) ;
}
) ;
twister . tmpl . accountMC . find ( 'input[type="text"], textarea' ) . on ( 'input' , function ( event ) {
var saveElem = $ ( event . target ) . closest ( '.modal-content' ) . find ( '.submit-changes' )
. attr ( 'data-profile-changed' , 'true' ) ;
if ( saveElem . prop ( 'disabled' ) && saveElem . attr ( 'data-blocked' ) !== 'true' )
$ . MAL . enableButton ( saveElem ) ;
} ) ;
twister . tmpl . accountMC . filter ( '.input-avatar' ) . on ( 'change' , function ( event ) {
var containerElem = $ ( event . target ) . closest ( '.modal-content' ) ;
var saveElem = containerElem . find ( '.submit-changes' )
. attr ( 'data-avatar-changed' , 'true' ) ;
if ( saveElem . prop ( 'disabled' ) && saveElem . attr ( 'data-blocked' ) !== 'true' )
$ . MAL . enableButton ( saveElem ) ;
handleAvatarFileSelect ( event , containerElem . find ( '.avatar img' ) ) ;
} ) ;
twister . tmpl . accountMC . find ( '.submit-changes' ) . attr ( 'data-blocked' , 'true' )
. on ( 'click' , function ( event ) {
var saveElem = $ ( event . target ) ;
var containerElem = saveElem . closest ( '.modal-content' ) ;
$ . MAL . disableButton ( saveElem ) ;
var profileDataChanged = saveElem . attr ( 'data-profile-changed' ) === 'true' ;
var avatarDataChanged = saveElem . attr ( 'data-avatar-changed' ) === 'true' ;
if ( profileDataChanged ) {
saveElem . attr ( 'data-profile-changed' , 'false' ) ;
var profileData = {
fullname : containerElem . find ( '.input-name' ) . val ( ) ,
bio : containerElem . find ( '.input-bio' ) . val ( ) ,
location : containerElem . find ( '.input-location' ) . val ( ) ,
url : containerElem . find ( '.input-url' ) . val ( ) ,
tox : containerElem . find ( '.input-tox' ) . val ( ) ,
bitmessage : containerElem . find ( '.input-bitmessage' ) . val ( )
} ;
}
if ( avatarDataChanged ) {
saveElem . attr ( 'data-avatar-changed' , 'false' ) ;
var avatarData = containerElem . find ( '.avatar img' ) . attr ( 'src' ) ;
}
if ( profileDataChanged && avatarDataChanged ) {
redrawProfileAndAvatar ( defaultScreenName , profileData , avatarData ) ;
saveProfile ( defaultScreenName , profileData ,
function ( req ) {
saveAvatar ( req . peerAlias , req . avatarData ,
alertPopup ,
{
txtMessage : polyglot . t ( 'profile_saved' )
} ,
alertPopup ,
{
txtMessage : polyglot . t ( 'profile_not_saved' ) + '\n\nCan\'t save avatar data.' ,
cbConfirm : function ( req ) {
$ . MAL . enableButton ( req . attr ( 'data-avatar-changed' , 'true' ) ) ;
} ,
cbConfirmReq : req . saveElem ,
cbClose : 'cbConfirm'
}
) ;
} ,
{
peerAlias : defaultScreenName ,
avatarData : avatarData ,
saveElem : saveElem
} ,
alertPopup ,
{
txtMessage : polyglot . t ( 'profile_not_saved' ) + '\n\nCan\'t save profile data.' ,
cbConfirm : function ( req ) {
$ . MAL . enableButton ( req . attr ( 'data-profile-changed' , 'true' ) ) ;
} ,
cbConfirmReq : saveElem ,
cbClose : 'cbConfirm'
}
) ;
} else if ( profileDataChanged ) {
redrawProfile ( defaultScreenName , profileData ) ;
saveProfile ( defaultScreenName , profileData ,
alertPopup ,
{
txtMessage : polyglot . t ( 'profile_saved' )
} ,
alertPopup ,
{
txtMessage : polyglot . t ( 'profile_not_saved' ) + '\n\nCan\'t save profile data.' ,
cbConfirm : function ( req ) {
$ . MAL . enableButton ( req . attr ( 'data-profile-changed' , 'true' ) ) ;
} ,
cbConfirmReq : saveElem ,
cbClose : 'cbConfirm'
}
) ;
} else if ( avatarDataChanged ) {
redrawAvatar ( defaultScreenName , avatarData ) ;
saveAvatar ( defaultScreenName , avatarData ,
alertPopup ,
{
txtMessage : polyglot . t ( 'profile_saved' )
} ,
alertPopup ,
{
txtMessage : polyglot . t ( 'profile_not_saved' ) + '\n\nCan\'t save avatar data.' ,
cbConfirm : function ( req ) {
$ . MAL . enableButton ( req . attr ( 'data-avatar-changed' , 'true' ) ) ;
} ,
cbConfirmReq : saveElem ,
cbClose : 'cbConfirm'
}
) ;
} else
$ . MAL . enableButton ( saveElem ) ;
} )
;
$ . MAL . disableButton ( twister . tmpl . accountMC . find ( '.submit-changes' ) ) ;
twister . tmpl . accountMC . filter ( '.secret-key-container' ) . hide ( ) ;
twister . tmpl . accountMC . find ( '.toggle-secret-key' ) . on ( 'click' , function ( event ) {
var containerElem = $ ( event . target ) . closest ( '.modal-content' ) . find ( '.secret-key-container' ) ;
if ( containerElem . is ( ':visible' ) )
containerElem . slideUp ( 'fast' ) ;
else {
var secretKeyElem = containerElem . find ( '.secret-key' ) ;
if ( secretKeyElem . text ( ) )
containerElem . slideDown ( 'fast' ) ;
else
dumpPrivkey ( defaultScreenName ,
function ( req , res ) {
req . secretKeyElem . text ( res ) ;
req . containerElem . slideDown ( 'fast' ) ;
} ,
{
containerElem : containerElem ,
secretKeyElem : secretKeyElem
}
) ;
}
} ) ;
2015-03-30 05:16:41 +05:00
}
2016-01-31 11:23:07 +05:00
function extractTemplate ( selector ) {
2016-02-03 12:02:47 +05:00
return $ ( selector ) . appendTo ( twister . tmpl . root ) . children ( ) ;
2016-01-31 11:23:07 +05:00
}
2015-05-23 22:26:23 +05:00
function promptCopyAttrData ( event ) {
window . prompt ( polyglot . t ( 'copy_to_clipboard' ) , $ ( event . target ) . attr ( 'data' ) ) ;
}
2015-04-21 02:00:05 +05:00
function initInterfaceModule ( module ) {
return $ ( '.module.' + module ) . html ( $ ( '#' + module + '-template' ) . html ( ) ) . show ( ) ;
}
function killInterfaceModule ( module ) {
$ ( '.module.' + module ) . empty ( ) . hide ( ) ;
}
2015-03-30 05:16:41 +05:00
2015-10-22 02:05:18 +05:00
function elemFitNextIntoParentHeight ( elem ) {
var parent = elem . parent ( ) ;
var elemNext = elem . nextAll ( ) ;
var elemNextHeight = parent . actual ( 'height' ) - elem . actual ( 'outerHeight' , { includeMargin : true } ) ;
if ( elemNextHeight > 0 ) // FIXME actually it's here because of nin theme's two vertical columns layout of profile modal
elemNext . outerHeight ( elemNextHeight ) ;
else
elemNext . outerHeight ( parent . actual ( 'outerHeight' ) ) ;
}
2015-09-04 03:21:33 +05:00
function inputEnterActivator ( event ) {
var elemEvent = $ ( event . target ) ;
2015-10-24 01:39:22 +05:00
elemEvent . closest ( event . data . parentSelector ) . find ( event . data . enterSelector )
2019-01-23 06:24:32 +05:00
. prop ( 'disabled' , elemEvent . val ( ) . trim ( ) === '' ) ;
2015-09-04 03:21:33 +05:00
}
2016-03-23 04:02:20 +05:00
function pasteToTextarea ( ta , p ) {
if ( ! ta || typeof ta . val !== 'function' ) return ;
var s = ta . val ( ) ;
var c = ta . caret ( ) ;
if ( s . length ) {
if ( c < 1 )
ta . val ( p + ( s [ c ] . match ( /\s/ ) ? '' : ' ' ) + s ) ;
else if ( c < s . length )
ta . val ( s . slice ( 0 , c ) + ( s [ c - 1 ] . match ( /\s/ ) ? '' : ' ' ) + p + ( s [ c ] . match ( /\s/ ) ? '' : ' ' ) + s . slice ( c ) ) ;
else
ta . val ( s + ( s [ c - 1 ] . match ( /\s/ ) ? '' : ' ' ) + p + ' ' ) ;
} else
ta . val ( p + ' ' ) ;
2019-01-23 06:24:32 +05:00
ta . trigger ( 'focus' ) . caret ( c + p . length + ( ( ta . val ( ) . length - s . length - p . length ) > 1 ? 2 : 1 ) ) ;
2016-03-23 04:02:20 +05:00
}
2015-05-23 22:26:23 +05:00
function setTextcompleteOnEventTarget ( event ) {
// cursor has not set yet and we need to wait 100ms to skip global click event
2015-09-05 04:14:16 +05:00
setTimeout ( setTextcompleteOnElement , 100 , event . target ,
typeof event . data . req === 'function' ? event . data . req ( ) : event . data . req ) ;
2015-04-14 22:39:20 +05:00
}
2015-09-05 04:14:16 +05:00
function setTextcompleteOnElement ( elem , req ) {
2015-05-23 22:26:23 +05:00
elem = $ ( elem ) ;
2015-09-05 04:14:16 +05:00
elem . textcomplete ( req , {
2015-10-24 01:39:22 +05:00
appendTo : ( elem . closest ( '.dashboard' ) . length ) ? elem . parent ( ) : $ ( 'body' ) ,
2015-05-23 22:26:23 +05:00
listPosition : setTextcompleteDropdownListPos
} ) ;
2015-04-10 06:59:57 +05:00
}
2015-09-05 04:14:16 +05:00
function unsetTextcompleteOnEventTarget ( event ) {
$ ( event . target ) . textcomplete ( 'destroy' ) ;
}
2015-03-30 05:16:41 +05:00
// following workaround function is for calls from $.fn.textcomplete only
// we need this because currently implementation of caret position detection is way too imperfect
function setTextcompleteDropdownListPos ( position ) {
position = this . _applyPlacement ( position ) ;
2015-10-24 01:39:22 +05:00
if ( this . option . appendTo . closest ( '.dashboard' ) . length > 0 ) {
2015-05-23 22:26:23 +05:00
position . position = 'fixed' ;
position . top = ( parseFloat ( position . top ) - window . pageYOffset ) . toString ( ) + 'px' ;
} else
position . position = 'absolute' ;
2015-03-30 05:16:41 +05:00
this . $el . css ( position ) ;
return this ;
2014-01-18 13:09:32 -02:00
}
2015-07-17 17:17:28 -03:00
2019-01-23 06:24:32 +05:00
$ ( function ( ) {
2018-01-14 01:19:45 +05:00
setupTimeGmtToText ( $ . Options . locLang . val ) ;
2016-03-21 00:55:32 +05:00
if ( $ . localStorage . isSet ( 'twistaURIs' ) )
twister . URIs = $ . localStorage . get ( 'twistaURIs' ) ;
2016-01-31 11:23:07 +05:00
twister . html . blanka . appendTo ( 'body' ) . hide ( ) ;
2017-02-02 14:17:10 +05:00
twister . tmpl . loginMC = extractTemplate ( '#template-login-modal' ) ;
twister . tmpl . loginMC . find ( '.login' ) . on ( 'click' , handleClickAccountLoginLogin ) ;
var module = twister . tmpl . loginMC . closest ( '.create-account' ) ;
module . find ( '.alias' ) . on ( 'input' , handleInputAccountCreateSetReq ) ;
module . find ( '.check' ) . on ( 'click' , handleClickAccountCreateCheckReq ) ;
module . find ( '.create' ) . on ( 'click' , handleClickAccountCreateCreate ) ;
module = twister . tmpl . loginMC . closest ( '.import-account' ) ;
module . find ( '.secret-key' ) . on ( 'input' , handleInputAccountImportSetReq ) ;
module . find ( '.alias' ) . on ( 'input' , handleInputAccountImportSetReq ) ;
module . find ( '.import' ) . on ( 'click' , handleClickAccountImportImport ) ;
2016-02-15 22:27:37 +05:00
twister . tmpl . followersList = extractTemplate ( '#template-followers-list' ) ;
twister . tmpl . followersPeer = extractTemplate ( '#template-followers-peer' ) ;
2016-01-31 11:23:07 +05:00
twister . tmpl . followingList = extractTemplate ( '#template-following-list' ) ;
2016-02-15 22:56:05 +05:00
twister . tmpl . followingPeer = extractTemplate ( '#template-following-peer' ) ;
2016-11-25 02:46:41 +05:00
twister . tmpl . whoTofollowPeer = extractTemplate ( '#template-whotofollow-peer' ) ;
2016-02-03 22:04:53 +05:00
twister . tmpl . commonDMsListItem = extractTemplate ( '#template-direct-messages-list-item' )
. on ( 'mouseup' , function ( event ) {
event . data = { route :
$ . MAL . dmchatUrl ( $ ( event . target ) . closest ( '.module' ) . attr ( 'data-screen-name' ) ) } ;
routeOnClick ( event ) ;
} )
;
2016-03-23 04:02:20 +05:00
twister . tmpl . shortenUri = extractTemplate ( '#template-shorten-uri' )
. on ( 'click' ,
{ cbFunc :
function ( uriLong , uriShort , textArea ) {
if ( uriShort )
pasteToTextarea ( textArea , uriShort ) ;
else
showURIShortenerErrorRPC ( uriShort ) ;
}
} ,
function ( event ) {
muteEvent ( event ) ;
2016-03-23 21:46:02 +05:00
var postAreaNew = $ ( event . target ) . closest ( '.post-area-new' ) ;
var textArea = postAreaNew . find ( twister . focus . textareaPostCur ) ;
if ( ! textArea . length ) textArea = postAreaNew . find ( twister . focus . textareaPostPrev ) ;
if ( ! textArea . length ) textArea = postAreaNew . find ( 'textarea:last' ) ;
2016-03-23 04:02:20 +05:00
event . data . cbReq = textArea ;
2016-03-29 16:17:18 +05:00
if ( postAreaNew . closest ( '.directMessages' ) . length )
confirmPopup ( {
txtMessage : polyglot . t ( 'shorten_URI_its_public_is_it_ok' ) ,
txtConfirm : polyglot . t ( 'shorten_URI' ) ,
cbConfirm : openRequestShortURIForm ,
cbConfirmReq : event
} ) ;
2016-03-30 04:34:47 +05:00
else if ( $ . mobile && postAreaNew . closest ( '.dm-form' ) . length ) {
if ( confirm ( polyglot . t ( 'shorten_URI_its_public_is_it_ok' ) ) )
openRequestShortURIForm ( event ) ;
} else
2016-03-29 16:17:18 +05:00
openRequestShortURIForm ( event ) ;
2016-03-23 04:02:20 +05:00
}
)
;
2016-03-23 21:46:02 +05:00
twister . tmpl . postTextareaEditBar = extractTemplate ( '#template-post-textarea-edit-bar' )
. append ( twister . tmpl . shortenUri . clone ( true ) )
;
2016-01-31 11:23:07 +05:00
twister . tmpl . postRtReference = extractTemplate ( '#template-post-rt-reference' )
2016-03-23 23:56:36 +05:00
. on ( 'mouseup' , { feeder : '.post-rt-reference' } , handleClickOpenConversation )
2016-02-03 01:29:41 +05:00
. on ( 'click' , muteEvent ) // to prevent post expanding or collapsing
;
2018-11-01 00:48:32 +05:00
twister . tmpl . avatarTiny = extractTemplate ( '#template-avatar-tiny' ) ;
2016-01-31 11:23:07 +05:00
twister . tmpl . postRtBy = extractTemplate ( '#template-post-rt-by' ) ;
2016-02-16 03:48:39 +05:00
twister . tmpl . profileShowMoreFollowers = extractTemplate ( '#template-profile-show-more-followers' ) ;
2016-01-31 11:23:07 +05:00
2015-07-17 17:17:28 -03:00
var path = window . location . pathname ;
var page = path . split ( "/" ) . pop ( ) ;
2017-02-02 14:17:10 +05:00
if ( page . indexOf ( 'network.html' ) === 0 ) {
2015-07-17 17:17:28 -03:00
initInterfaceNetwork ( ) ;
2015-07-19 05:23:50 +05:00
} else if ( page . indexOf ( 'options.html' ) === 0 ) {
initInterfaceCommon ( ) ;
$ . Options . initControls ( ) ;
2015-07-17 17:17:28 -03:00
}
changeStyle ( ) ;
2016-03-09 01:23:39 +05:00
// need to play something once in result of click on Firefox to be able to play sound notifications
// FXIME it's just workaround of FF's bug, see https://bugzilla.mozilla.org/show_bug.cgi?id=1197631
$ ( 'body' ) . on ( 'click' , function ( ) {
2019-01-24 17:43:20 +05:00
if ( $ . Options . sndMention . val !== 'false' )
playSound ( 'player' , $ . Options . sndMention . val , 0.001 ) ;
2016-03-09 01:23:39 +05:00
setTimeout ( function ( ) { $ ( 'body' ) . off ( 'click' ) ; } , 10000 ) ; // sound must be played before unbinding
} ) ;
2015-07-17 17:17:28 -03:00
} ) ;