2013-08-19 22:30:57 +09:00
/ * · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · ·
· ·
· ·
· Q V I T T E R ·
· ·
· http : //github.com/hannesmannerheim/qvitter ·
· ·
· ·
· < o ) ·
· / _ //// ·
· ( _ _ _ _ / ·
· ( o < ·
· o > \ \ \ \ _ \ ·
· \ \ ) \ _ _ _ _ ) ·
· ·
· ·
· ·
· Qvitter is free software : you can redistribute it and / or modify it ·
· under the terms of the GNU Affero General Public License as published by ·
· the Free Software Foundation , either version three of the License or ( at ·
· your option ) any later version . ·
· ·
· Qvitter is distributed in hope that it will be useful but WITHOUT ANY ·
· WARRANTY ; without even the implied warranty of MERCHANTABILTY or FITNESS ·
· FOR A PARTICULAR PURPOSE . See the GNU Affero General Public License for ·
· more details . ·
· ·
· You should have received a copy of the GNU Affero General Public License ·
· along with Qvitter . If not , see < http : //www.gnu.org/licenses/>. ·
· ·
· Contact h @ nnesmannerhe . im if you have any questions . ·
· ·
· · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · * /
2013-08-26 03:11:53 +09:00
2014-01-29 03:42:47 +09:00
// object to keep old states of streams in, to speed up stream change
window . oldStreams = new Object ( ) ;
2013-08-19 22:30:57 +09:00
/ * ·
·
2014-05-14 16:46:07 +09:00
· Update stream on back button
2013-08-19 22:30:57 +09:00
·
· · · · · · · · · · · · · * /
2014-05-14 16:46:07 +09:00
window . onpopstate = function ( event ) {
if ( event && event . state ) {
display _spinner ( ) ;
setNewCurrentStream ( event . state . strm , function ( ) {
remove _spinner ( ) ;
} , false ) ;
2013-08-19 22:30:57 +09:00
}
}
2013-09-03 01:13:15 +09:00
/ * ·
·
· fix login and register box to top when they reach top
·
· · · · · · · · · · · · · * /
window . loginContentStartPos = $ ( '.front-welcome-text' ) . height ( ) + 45 ;
$ ( window ) . scroll ( function ( e ) {
if ( $ ( this ) . scrollTop ( ) > window . loginContentStartPos && $ ( '#login-content' ) . css ( 'position' ) != 'fixed' ) {
$ ( '#login-content, .front-signup' ) . not ( '#popup-signup' ) . css ( { 'position' : 'fixed' , 'top' : '50px' } ) ;
}
else if ( $ ( this ) . scrollTop ( ) < window . loginContentStartPos && $ ( '#login-content' ) . css ( 'position' ) != 'absolute' ) {
$ ( '#login-content, .front-signup' ) . not ( '#popup-signup' ) . css ( { 'position' : 'absolute' , 'top' : 'auto' } ) ;
}
} ) ;
/ * ·
·
· Tooltip to show what federated means
·
· · · · · · · · · · · · · * /
$ ( '#federated-tooltip' ) . on ( 'mouseenter' , function ( ) {
$ ( '#what-is-federation' ) . fadeIn ( 100 ) ;
} ) ;
$ ( '#what-is-federation' ) . on ( 'mouseleave' , function ( ) {
$ ( '#what-is-federation' ) . fadeOut ( 100 ) ;
} ) ;
/ * ·
·
· Register
·
· · · · · · · · · · · · · * /
$ ( '.front-signup input, .front-signup button' ) . removeAttr ( 'disabled' ) ; // clear this onload
$ ( '#signup-btn-step1' ) . click ( function ( ) {
2014-01-29 03:42:47 +09:00
2013-09-03 01:13:15 +09:00
display _spinner ( ) ;
$ ( '.front-signup input, .front-signup button' ) . addClass ( 'disabled' ) ;
$ ( '.front-signup input, .front-signup button' ) . attr ( 'disabled' , 'disabled' ) ;
// 7 s timeout to annoy human spammers
setTimeout ( function ( ) {
remove _spinner ( ) ;
popUpAction ( 'popup-register' , window . sL . signUp , '<div id="popup-signup" class="front-signup">' +
2013-11-23 08:31:04 +09:00
'<div class="signup-input-container"><div id="atsign">@</div><input placeholder="' + window . sL . registerNickname + '" type="text" autocomplete="off" class="text-input" id="signup-user-nickname-step2"><div class="fieldhelp">a-z0-9</div></div>' +
2013-09-03 01:13:15 +09:00
'<div class="signup-input-container"><input placeholder="' + window . sL . signUpFullName + '" type="text" autocomplete="off" class="text-input" id="signup-user-name-step2" value="' + $ ( '#signup-user-name' ) . val ( ) + '"></div>' +
'<div class="signup-input-container"><input placeholder="' + window . sL . signUpEmail + '" type="text" autocomplete="off" id="signup-user-email-step2" value="' + $ ( '#signup-user-email' ) . val ( ) + '"></div>' +
2013-11-23 08:31:04 +09:00
'<div class="signup-input-container"><input placeholder="' + window . sL . registerHomepage + '" type="text" autocomplete="off" class="text-input" id="signup-user-homepage-step2"></div>' +
'<div class="signup-input-container"><input placeholder="' + window . sL . registerBio + '" type="text" autocomplete="off" class="text-input" id="signup-user-bio-step2"></div>' +
'<div class="signup-input-container"><input placeholder="' + window . sL . registerLocation + '" type="text" autocomplete="off" class="text-input" id="signup-user-location-step2"></div>' +
2013-09-03 01:13:15 +09:00
'<div class="signup-input-container"><input placeholder="' + window . sL . loginPassword + '" type="password" class="text-input" id="signup-user-password1-step2" value="' + $ ( '#signup-user-password' ) . val ( ) + '"><div class="fieldhelp">>5</div></div>' +
2013-11-23 08:31:04 +09:00
'<div class="signup-input-container"><input placeholder="' + window . sL . registerRepeatPassword + '" type="password" class="text-input" id="signup-user-password2-step2"></div>' +
2013-09-03 01:13:15 +09:00
'<button id="signup-btn-step2" class="signup-btn disabled" type="submit">' + window . sL . signUpButtonText + '</button>' +
'</div>' , false ) ;
// ask api if nickname is ok, if no typing for 1 s
$ ( '#signup-user-nickname-step2' ) . on ( 'keyup' , function ( ) {
clearTimeout ( window . checkNicknameTimeout ) ;
if ( $ ( '#signup-user-nickname-step2' ) . val ( ) . length > 1 && /^[a-zA-Z0-9]+$/ . test ( $ ( '#signup-user-nickname-step2' ) . val ( ) ) ) {
$ ( '#signup-user-nickname-step2' ) . addClass ( 'nickname-taken' ) ;
if ( $ ( '.spinner-wrap' ) . length == 0 ) {
$ ( '#signup-user-nickname-step2' ) . after ( '<div class="spinner-wrap"><div class="spinner"><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i></div></div>' ) ;
}
window . checkNicknameTimeout = setTimeout ( function ( ) {
getFromAPI ( 'check_nickname.json?nickname=' + encodeURIComponent ( $ ( '#signup-user-nickname-step2' ) . val ( ) ) , function ( data ) {
$ ( '.spinner-wrap' ) . remove ( ) ;
console . log ( $ ( '.spinner-wrap' ) . length ) ;
if ( data == 0 ) {
$ ( '#signup-user-password2-step2' ) . trigger ( 'keyup' ) ; // revalidates
}
else {
$ ( '#signup-user-nickname-step2' ) . removeClass ( 'nickname-taken' ) ;
$ ( '#signup-user-password2-step2' ) . trigger ( 'keyup' ) ;
}
} ) ;
} , 1000 ) ;
}
else {
$ ( '.spinner-wrap' ) . remove ( ) ;
}
} ) ;
// validate on keyup
$ ( '#popup-register input' ) . on ( 'keyup' , function ( ) {
if ( validateRegisterForm ( $ ( '#popup-register' ) ) ) {
if ( ! $ ( '#signup-user-nickname-step2' ) . hasClass ( 'nickname-taken' ) ) {
$ ( '#signup-btn-step2' ) . removeClass ( 'disabled' ) ;
}
else {
$ ( '#signup-btn-step2' ) . addClass ( 'disabled' ) ;
}
}
else {
$ ( '#signup-btn-step2' ) . addClass ( 'disabled' ) ;
}
} ) ;
$ ( '#popup-register input' ) . trigger ( 'keyup' ) ;
// submit on enter
$ ( 'input#signup-user-name-step2,input#signup-user-email-step2,input#signup-user-password1-step2, input#signup-user-password2-step2' ) . keyup ( function ( e ) {
if ( e . keyCode == 13 ) {
$ ( '#signup-btn-step2' ) . trigger ( 'click' ) ;
}
} ) ;
$ ( '#signup-btn-step2' ) . click ( function ( ) {
$ ( '#popup-register input,#popup-register button' ) . addClass ( 'disabled' ) ;
display _spinner ( ) ;
2014-05-14 16:46:07 +09:00
$ . ajax ( { url : window . qvitterApiRoot ,
2013-09-03 01:13:15 +09:00
type : "POST" ,
data : {
postRequest : 'account/register.json' ,
nickname : $ ( '#signup-user-nickname-step2' ) . val ( ) ,
email : $ ( '#signup-user-email-step2' ) . val ( ) ,
fullname : $ ( '#signup-user-name-step2' ) . val ( ) ,
homepage : $ ( '#signup-user-homepage-step2' ) . val ( ) ,
bio : $ ( '#signup-user-bio-step2' ) . val ( ) ,
location : $ ( '#signup-user-location-step2' ) . val ( ) ,
password : $ ( '#signup-user-password1-step2' ) . val ( ) ,
2013-11-23 08:31:04 +09:00
confirm : $ ( '#signup-user-password2-step2' ) . val ( ) ,
cBS : window . cBS ,
cBSm : window . cBSm ,
2013-09-03 01:13:15 +09:00
username : 'none' ,
} ,
dataType : "json" ,
error : function ( data ) { console . log ( 'error' ) ; console . log ( data ) ; } ,
success : function ( data ) {
remove _spinner ( ) ;
if ( typeof data . error == 'undefined' ) {
2014-05-14 16:46:07 +09:00
$ ( 'input#nickname' ) . val ( $ ( '#signup-user-nickname-step2' ) . val ( ) ) ;
2013-09-03 01:13:15 +09:00
$ ( 'input#password' ) . val ( $ ( '#signup-user-password1-step2' ) . val ( ) ) ;
$ ( 'input#rememberme' ) . prop ( 'checked' , true ) ;
$ ( '#submit-login' ) . trigger ( 'click' ) ;
$ ( '#popup-register' ) . remove ( ) ;
}
else {
alert ( 'Try again! ' + data . error ) ;
$ ( '#popup-register input,#popup-register button' ) . removeClass ( 'disabled' ) ;
}
}
} ) ;
} ) ;
// reactivate register form on popup close
$ ( '#popup-register' ) . on ( 'remove' , function ( ) {
$ ( '.front-signup input, .front-signup button' ) . removeAttr ( 'disabled' ) ;
$ ( '.front-signup input, .front-signup button' ) . removeClass ( 'disabled' ) ;
} ) ;
} , 7000 ) ;
} ) ;
// submit on enter
$ ( 'input#signup-user-name,input#signup-user-email,input#signup-user-password' ) . keyup ( function ( e ) {
if ( e . keyCode == 13 ) {
$ ( '#signup-btn-step1' ) . trigger ( 'click' ) ;
}
} ) ;
2013-08-19 22:30:57 +09:00
/ * ·
·
2013-09-03 01:13:15 +09:00
· autologin or show welcome screen
2013-08-19 22:30:57 +09:00
·
· · · · · · · · · · · · · * /
$ ( '#submit-login' ) . removeAttr ( 'disabled' ) ; // might be remebered by browser...
2013-08-26 05:34:09 +09:00
$ ( window ) . load ( function ( ) {
2013-08-19 22:30:57 +09:00
$ ( '#user-container' ) . css ( 'display' , 'block' ) ;
$ ( '#feed' ) . css ( 'display' , 'block' ) ;
2013-08-20 04:15:15 +09:00
// check for localstorage, if none, we remove possibility to remember login
var userInLocalStorage = false ;
if ( localStorageIsEnabled ( ) ) {
2014-05-14 16:46:07 +09:00
if ( typeof localStorage . autologinPassword != 'undefined' ) {
2013-08-20 04:15:15 +09:00
userInLocalStorage = true ;
}
}
2014-05-14 16:46:07 +09:00
// if we have a user logged in to localStorage, but not to gnusocial, delete
// and send them to front page and tell it to shake loginbox
if ( ! window . isLoggedIn && userInLocalStorage ) {
localStorage . doShake = true ;
delete localStorage . autologinUsername ;
delete localStorage . autologinPassword ;
window . location . href = window . siteInstanceURL ;
}
2013-11-23 08:31:04 +09:00
2014-05-14 16:46:07 +09:00
// if login credentials in localstorage got lost somewhere, logout
if ( window . isLoggedIn && ! userInLocalStorage ) {
window . location . href = window . siteInstanceURL + 'main/logout' ;
}
// autologin
else if ( window . isLoggedIn && userInLocalStorage ) {
$ ( 'input#nickname' ) . val ( localStorage . autologinUsername ) ;
$ ( 'input#password' ) . val ( localStorage . autologinPassword ) ;
2014-01-29 03:42:47 +09:00
doLogin ( "get stream from url" ) ;
2013-08-19 22:30:57 +09:00
}
else {
display _spinner ( ) ;
2013-08-26 03:11:53 +09:00
window . currentStream = '' ; // force reload stream
2013-08-19 22:30:57 +09:00
setNewCurrentStream ( getStreamFromUrl ( ) , function ( ) {
2013-08-26 03:11:53 +09:00
logoutWithoutReload ( false ) ;
2014-05-14 16:46:07 +09:00
remove _spinner ( ) ;
2013-08-19 22:30:57 +09:00
} , true ) ;
}
} ) ;
2013-08-26 03:11:53 +09:00
2013-08-19 22:30:57 +09:00
/ * ·
·
· Login action
·
· · · · · · · · · · · · · * /
2014-05-14 16:46:07 +09:00
$ ( '#form_login' ) . submit ( function ( e ) {
2013-11-23 08:31:04 +09:00
2014-05-14 16:46:07 +09:00
// store username and password in localstorage before submitting
if ( typeof localStorage . autologinPassword == 'undefined' ) {
e . preventDefault ( ) ;
localStorage . autologinPassword = $ ( 'input#password' ) . val ( ) ;
localStorage . autologinUsername = $ ( 'input#nickname' ) . val ( ) ;
$ ( this ) . submit ( ) ;
}
2013-11-23 08:31:04 +09:00
} ) ;
function doLogin ( streamToSet ) {
2013-08-19 22:30:57 +09:00
$ ( '#submit-login' ) . attr ( 'disabled' , 'disabled' ) ;
$ ( '#submit-login' ) . focus ( ) ; // prevents submit on enter to close alert-popup on wrong credentials
display _spinner ( ) ;
// login with ajax
2014-05-14 16:46:07 +09:00
checkLogin ( $ ( 'input#nickname' ) . val ( ) , $ ( 'input#password' ) . val ( ) , function ( user ) {
2013-08-19 22:30:57 +09:00
2013-08-26 03:11:53 +09:00
console . log ( user ) ;
2013-08-19 22:30:57 +09:00
// store credentials in global var
window . loginUsername = user . screen _name ;
window . loginPassword = $ ( 'input#password' ) . val ( ) ;
2013-08-26 05:34:09 +09:00
2014-01-29 03:42:47 +09:00
// maybe get stream from url (UGLY SORRRRRY)
if ( streamToSet == "get stream from url" ) {
streamToSet = getStreamFromUrl ( ) ; // called now becuase we want window.loginUsername to be set first...
}
2013-08-26 05:34:09 +09:00
// set colors if the api supports it
if ( typeof user . linkcolor != 'undefined' &&
typeof user . backgroundcolor != 'undefined' ) {
user . linkcolor = user . linkcolor || '' ; // no null value
user . backgroundcolor = user . backgroundcolor || '' ; // no null value
window . userLinkColor = user . linkcolor ;
window . userBackgroundColor = user . backgroundcolor ;
if ( window . userLinkColor . length != 6 ) {
window . userLinkColor = window . defaultLinkColor ;
}
if ( window . userBackgroundColor . length != 6 ) {
window . userBackgroundColor = window . defaultBackgroundColor ;
}
}
2013-08-19 22:30:57 +09:00
// add user data to DOM, show search form, remeber user id, show the feed
2013-11-23 08:31:04 +09:00
$ ( '#user-container' ) . css ( 'z-index' , '1000' ) ;
$ ( '#top-compose' ) . removeClass ( 'hidden' ) ;
2013-11-29 20:49:54 +09:00
$ ( '#user-avatar' ) . attr ( 'src' , user . profile _image _url _profile _size ) ;
2013-08-19 22:30:57 +09:00
$ ( '#user-name' ) . append ( user . name ) ;
$ ( '#user-screen-name' ) . append ( user . screen _name ) ;
$ ( '#user-profile-link' ) . append ( '<a href="' + user . statusnet _profile _url + '">' + window . sL . viewMyProfilePage + '</a>' ) ;
$ ( '#user-queets strong' ) . html ( user . statuses _count ) ;
$ ( '#user-following strong' ) . html ( user . friends _count ) ;
$ ( '#user-followers strong' ) . html ( user . followers _count ) ;
2013-08-26 03:11:53 +09:00
$ ( '#user-groups strong' ) . html ( user . groups _count ) ;
2013-11-23 08:31:04 +09:00
$ ( '.stream-selection.friends-timeline' ) . attr ( 'href' , user . statusnet _profile _url + '/all' ) ;
$ ( '.stream-selection.mentions' ) . attr ( 'href' , user . statusnet _profile _url + '/replies' ) ;
$ ( '.stream-selection.my-timeline' ) . attr ( 'href' , user . statusnet _profile _url ) ;
$ ( '.stream-selection.favorites' ) . attr ( 'href' , user . statusnet _profile _url + '/favorites' ) ;
2013-08-19 22:30:57 +09:00
window . myUserID = user . id ;
2013-08-26 03:11:53 +09:00
2013-08-19 22:30:57 +09:00
// if remeber me is checked, save credentials in local storage
if ( $ ( '#rememberme' ) . is ( ':checked' ) ) {
2013-08-20 04:15:15 +09:00
if ( localStorageIsEnabled ( ) ) {
localStorage . autologinPassword = $ ( 'input#password' ) . val ( ) ;
2014-05-14 16:46:07 +09:00
localStorage . autologinUsername = $ ( 'input#nickname' ) . val ( ) ;
2013-08-20 04:15:15 +09:00
}
2013-08-19 22:30:57 +09:00
}
// load history
loadHistoryFromLocalStorage ( ) ;
// set stream
window . currentStream = '' ; // always reload stream on login
setNewCurrentStream ( streamToSet , function ( ) {
2013-09-03 01:13:15 +09:00
$ ( '.language-dropdown' ) . css ( 'display' , 'none' ) ;
$ ( '.dropdown-menu.quitter-settings li.language' ) . css ( 'display' , 'block' ) ;
2013-08-19 22:30:57 +09:00
$ ( '#user-header' ) . animate ( { opacity : '1' } , 800 ) ;
$ ( '#user-body' ) . animate ( { opacity : '1' } , 800 ) ;
$ ( '#user-footer' ) . animate ( { opacity : '1' } , 800 ) ;
$ ( '.menu-container' ) . animate ( { opacity : '1' } , 800 ) ;
2013-08-26 03:11:53 +09:00
$ ( '#page-container' ) . animate ( { opacity : '1' } , 200 ) ;
2013-09-03 01:13:15 +09:00
$ ( '.front-welcome-text' ) . slideUp ( 1000 ) ;
2013-08-26 03:11:53 +09:00
$ ( '#settingslink' ) . fadeIn ( 'slow' ) ;
$ ( '#search' ) . fadeIn ( 'slow' ) ;
2013-08-19 22:30:57 +09:00
$ ( '#login-content' ) . css ( 'display' , 'none' ) ;
2013-09-03 01:13:15 +09:00
$ ( '.front-signup' ) . css ( 'display' , 'none' ) ;
2013-08-19 22:30:57 +09:00
remove _spinner ( ) ;
} , true ) ;
2013-11-23 08:31:04 +09:00
} ) ;
}
2013-08-19 22:30:57 +09:00
/ * ·
·
· In the login form , we want to check the remember - me - checkbox when its label is clicked
·
· · · · · · · · · · · · · * /
$ ( '#rememberme_label' ) . click ( function ( ) {
if ( $ ( '#rememberme' ) . prop ( 'checked' ) ) {
$ ( '#rememberme' ) . prop ( 'checked' , false ) ;
}
else {
$ ( '#rememberme' ) . prop ( 'checked' , true ) ;
}
} ) ;
2013-08-26 03:11:53 +09:00
$ ( '#rememberme_label' ) . disableSelection ( ) ;
2013-08-19 22:30:57 +09:00
/ * ·
·
· Logout by deleting local storage credentials ( if there are any ) and reload
·
· · · · · · · · · · · · · * /
$ ( '#logout' ) . click ( function ( ) {
2013-08-20 04:15:15 +09:00
if ( localStorageIsEnabled ( ) ) {
delete localStorage . autologinUsername ;
delete localStorage . autologinPassword ;
}
2014-05-14 16:46:07 +09:00
window . location . href = window . siteInstanceURL + 'main/logout' ;
2013-08-19 22:30:57 +09:00
} ) ;
2013-08-26 03:11:53 +09:00
/ * ·
·
· Settings
·
· · · · · · · · · · · · · * /
$ ( '#settings' ) . click ( function ( ) {
2013-08-26 05:34:09 +09:00
// buttons to add later: '<div class="right"><button class="close">' + window.sL.cancelVerb + '</button><button class="primary disabled">' + window.sL.saveChanges + '</button></div>'
2014-05-14 16:46:07 +09:00
popUpAction ( 'popup-settings' , window . sL . settings , '<div id="settings-container"><div><label for="link-color-selection">' + window . sL . linkColor + '</label><input id="link-color-selection" type="text" value="#' + window . userLinkColor + '" /></div><div><label for="link-color-selection">' + window . sL . backgroundColor + '</label><input id="background-color-selection" type="text" value="#' + window . userBackgroundColor + '" /></div><a href="' + window . siteInstanceURL + 'settings/profile" id="moresettings">' + window . sL . moreSettings + '</a></div>' , false ) ;
2013-08-26 03:11:53 +09:00
$ ( '#link-color-selection' ) . minicolors ( {
change : function ( hex ) {
changeLinkColor ( hex ) ;
postNewLinkColor ( hex . substring ( 1 ) ) ;
}
} ) ;
2013-08-26 05:34:09 +09:00
$ ( '#background-color-selection' ) . minicolors ( {
change : function ( hex ) {
$ ( 'body' ) . css ( 'background-color' , hex ) ;
postNewBackgroundColor ( hex . substring ( 1 ) ) ;
}
} ) ;
2013-11-23 08:31:04 +09:00
// also on keyup in input (minicolors 'change' event does not do this, apparently)
$ ( '#link-color-selection' ) . on ( 'keyup' , function ( ) {
keyupSetLinkColor ( $ ( this ) . val ( ) ) ;
} ) ;
$ ( '#background-color-selection' ) . on ( 'keyup' , function ( ) {
keyupSetBGColor ( $ ( this ) . val ( ) ) ;
} ) ;
} ) ;
// idle function for linkcolor selection by keyboard input
var keyupLinkColorTimer ;
function keyupSetLinkColor ( hex ) {
clearTimeout ( keyupLinkColorTimer ) ;
keyupLinkColorTimer = setTimeout ( function ( ) {
$ ( '#link-color-selection' ) . minicolors ( 'value' , hex ) ;
changeLinkColor ( $ ( '#link-color-selection' ) . val ( ) ) ;
postNewLinkColor ( $ ( '#link-color-selection' ) . val ( ) . substring ( 1 ) ) ;
} , 500 ) ;
}
// idle function for bgcolor selection by keyboard input
var keyupBGColorTimer ;
function keyupSetBGColor ( hex ) {
clearTimeout ( keyupBGColorTimer ) ;
keyupBGColorTimer = setTimeout ( function ( ) {
$ ( '#background-color-selection' ) . minicolors ( 'value' , hex ) ;
$ ( 'body' ) . css ( 'background-color' , $ ( '#background-color-selection' ) . val ( ) ) ;
postNewBackgroundColor ( $ ( '#background-color-selection' ) . val ( ) . substring ( 1 ) ) ;
} , 500 ) ;
}
// go to standard settingspage
$ ( 'body' ) . on ( 'click' , '#moresettings' , function ( ) {
$ ( document . body ) . append ( '<iframe id="logout-iframe" src="https://quitter.se/main/logout" style="display:none;">' ) ; // we need to logout before login, otherwise redirection to settingspage doesn't work
$ ( 'iframe#logout-iframe' ) . load ( function ( ) {
$ ( '#moresettings' ) . children ( 'form' ) . submit ( ) ; // submit hidden form and open settingspage in new tab
} ) ;
2013-08-26 03:11:53 +09:00
} ) ;
2013-11-23 08:31:04 +09:00
2013-08-26 03:11:53 +09:00
/ * ·
·
· Do a logout without reloading , i . e . on login errors
·
· · · · · · · · · · · · · * /
function logoutWithoutReload ( doShake ) {
2013-08-26 06:06:06 +09:00
if ( window . currentStream == 'statuses/public_timeline.json' ) {
2014-05-14 16:46:07 +09:00
$ ( 'body' ) . css ( 'background-image' , 'url(' + window . fullUrlToThisQvitterApp + 'img/mela.jpg)' ) ;
2013-08-26 06:06:06 +09:00
}
$ ( '#submit-login' ) . removeAttr ( 'disabled' ) ;
// delete any locally stored credentials
if ( localStorageIsEnabled ( ) ) {
delete localStorage . autologinUsername ;
delete localStorage . autologinPassword ;
}
$ ( '#user-header' ) . animate ( { opacity : '0' } , 200 ) ;
$ ( '#user-body' ) . animate ( { opacity : '0' } , 200 ) ;
$ ( '#user-footer' ) . animate ( { opacity : '0' } , 200 ) ;
$ ( '.menu-container' ) . animate ( { opacity : '0' } , 200 ) ;
2013-09-03 01:13:15 +09:00
$ ( '.language-dropdown' ) . css ( 'display' , 'block' ) ;
$ ( '.dropdown-menu.quitter-settings li.language' ) . css ( 'display' , 'none' ) ;
2013-08-26 06:06:06 +09:00
$ ( '#settingslink' ) . fadeOut ( 'slow' ) ;
$ ( '#search' ) . fadeOut ( 'slow' ) ;
2014-05-14 16:46:07 +09:00
$ ( 'input#nickname' ) . focus ( ) ;
2013-09-03 01:13:15 +09:00
$ ( '.front-signup' ) . animate ( { opacity : '1' } , 200 ) ;
2014-05-14 16:46:07 +09:00
if ( doShake || localStorage . doShake ) {
$ ( 'input#nickname' ) . css ( 'background-color' , 'pink' ) ;
2013-08-26 06:06:06 +09:00
$ ( 'input#password' ) . css ( 'background-color' , 'pink' ) ;
}
$ ( '#login-content' ) . animate ( { opacity : '1' } , 200 , function ( ) {
2014-05-14 16:46:07 +09:00
if ( doShake || localStorage . doShake ) {
2013-08-26 06:06:06 +09:00
$ ( '#login-content' ) . effect ( 'shake' , { distance : 5 , times : 2 } , function ( ) {
2014-05-14 16:46:07 +09:00
$ ( 'input#nickname' ) . animate ( { backgroundColor : '#fff' } , 1000 ) ;
2013-08-26 06:06:06 +09:00
$ ( 'input#password' ) . animate ( { backgroundColor : '#fff' } , 1000 ) ;
2014-05-14 16:46:07 +09:00
delete localStorage . doShake ;
2013-08-26 06:06:06 +09:00
} ) ;
2013-08-26 03:11:53 +09:00
}
2013-09-03 01:13:15 +09:00
$ ( '.front-welcome-text' ) . fadeIn ( 3000 ) ;
2013-08-26 06:06:06 +09:00
} ) ;
$ ( '#page-container' ) . animate ( { opacity : '1' } , 200 ) ;
2013-08-26 03:11:53 +09:00
}
2013-08-19 22:30:57 +09:00
/ * ·
·
· Handling the language dropdown selection
·
· · · · · · · · · · · · · * /
$ ( '.dropdown' ) . click ( function ( ) { $ ( this ) . toggleClass ( 'dropped' ) } ) ;
$ ( '.dropdown' ) . disableSelection ( ) ;
$ ( document ) . bind ( 'click' , function ( e ) {
2013-08-26 03:11:53 +09:00
if ( ! $ ( e . target ) . is ( '#logo' ) && ! $ ( e . target ) . is ( '#settingslink' ) && ! $ ( e . target ) . is ( '.nav-session' ) && ! $ ( e . target ) . is ( '.dropdown-toggle' ) && ! $ ( e . target ) . is ( '.dropdown-toggle small' ) && ! $ ( e . target ) . is ( '.dropdown-toggle span' ) && ! $ ( e . target ) . is ( '.dropdown-toggle b' ) ) {
2013-08-19 22:30:57 +09:00
$ ( '.dropdown' ) . removeClass ( 'dropped' ) ;
$ ( '.quitter-settings.dropdown-menu' ) . removeClass ( 'dropped' ) ;
}
} ) ;
$ ( '.language-link' ) . click ( function ( ) {
2013-08-20 04:15:15 +09:00
if ( localStorageIsEnabled ( ) ) {
localStorage . selectedLanguage = $ ( this ) . attr ( 'data-lang-code' ) ; // save langage selection
}
2013-08-19 22:30:57 +09:00
location . reload ( ) ; // reload
} ) ;
/ * ·
·
· Show the logo menu dropdown on click
·
· · · · · · · · · · · · · * /
2013-08-26 03:11:53 +09:00
$ ( '#settingslink' ) . click ( function ( ) {
2013-08-19 22:30:57 +09:00
if ( ! $ ( '.quitter-settings' ) . hasClass ( 'dropped' ) ) { $ ( '.quitter-settings' ) . addClass ( 'dropped' ) ; }
else { $ ( '.quitter-settings' ) . removeClass ( 'dropped' ) ; }
} ) ;
/ * ·
·
2014-01-29 03:42:47 +09:00
· When clicking a external follow button
2013-08-19 22:30:57 +09:00
·
· · · · · · · · · · · · · * /
2013-11-23 08:31:04 +09:00
$ ( 'body' ) . on ( 'click' , '.external-follow-button' , function ( event ) {
2013-11-29 20:49:54 +09:00
popUpAction ( 'popup-external-follow' , window . sL . userExternalFollow + ' ' + $ ( '.profile-card-inner .screen-name' ) . html ( ) , '<form method="post" action="' + window . siteInstanceURL . replace ( 'https://' , 'http://' ) + 'main/ostatus"><input type="hidden" id="nickname" name="nickname" value="' + $ ( '.profile-card-inner .screen-name' ) . html ( ) . substring ( 1 ) + '"><input type="text" id="profile" name="profile" placeholder="' + window . sL . userExternalFollowHelp + '" /></form>' , '<div class="right"><button class="close">' + window . sL . cancelVerb + '</button><button class="primary">' + window . sL . userExternalFollow + '</button></div>' ) ;
2013-11-23 08:31:04 +09:00
$ ( '#popup-external-follow form input#profile' ) . focus ( ) ;
$ ( '#popup-external-follow button.primary' ) . click ( function ( ) {
$ ( '#popup-external-follow form' ) . submit ( ) ;
} ) ;
} ) ;
2014-01-29 03:42:47 +09:00
/ * ·
·
· When clicking a external join button
·
· · · · · · · · · · · · · * /
$ ( 'body' ) . on ( 'click' , '.external-member-button' , function ( event ) {
popUpAction ( 'popup-external-join' , window . sL . joinExternalGroup + ' ' + $ ( '.profile-card-inner .screen-name' ) . html ( ) , '<form method="post" action="' + window . siteInstanceURL . replace ( 'https://' , 'http://' ) + 'main/ostatus"><input type="hidden" id="group" name="group" value="' + $ ( '.profile-card-inner .screen-name' ) . html ( ) . substring ( 1 ) + '"><input type="text" id="profile" name="profile" placeholder="' + window . sL . userExternalFollowHelp + '" /></form>' , '<div class="right"><button class="close">' + window . sL . cancelVerb + '</button><button class="primary">' + window . sL . userExternalFollow + '</button></div>' ) ;
$ ( '#popup-external-join form input#profile' ) . focus ( ) ;
$ ( '#popup-external-join button.primary' ) . click ( function ( ) {
$ ( '#popup-external-join form' ) . submit ( ) ;
} ) ;
} ) ;
2013-11-23 08:31:04 +09:00
2013-08-19 22:30:57 +09:00
$ ( 'body' ) . on ( 'click' , '.follow-button' , function ( event ) {
if ( ! $ ( this ) . hasClass ( 'disabled' ) ) {
$ ( this ) . addClass ( 'disabled' ) ;
// get user id
var user _id = $ ( this ) . attr ( 'data-follow-user-id' ) ;
// follow or unfollow?
if ( $ ( this ) . hasClass ( 'following' ) ) {
var followOrUnfollow = 'unfollow' ;
}
else {
var followOrUnfollow = 'follow' ;
}
// post to api
display _spinner ( ) ;
APIFollowOrUnfollowUser ( followOrUnfollow , user _id , this , function ( data , this _element ) {
remove _spinner ( ) ;
$ ( this _element ) . removeClass ( 'disabled' ) ;
if ( data ) {
if ( data . following ) {
$ ( this _element ) . addClass ( 'following' ) ;
$ ( '#user-following strong' ) . html ( parseInt ( $ ( '#user-following strong' ) . html ( ) , 10 ) + 1 ) ;
}
else {
$ ( this _element ) . removeClass ( 'following' ) ;
$ ( '#user-following strong' ) . html ( parseInt ( $ ( '#user-following strong' ) . html ( ) , 10 ) - 1 ) ;
}
}
} ) ;
}
} ) ;
/ * ·
·
· When clicking a join group button
·
· · · · · · · · · · · · · * /
$ ( 'body' ) . on ( 'click' , '.member-button' , function ( event ) {
if ( ! $ ( this ) . hasClass ( 'disabled' ) ) {
$ ( this ) . addClass ( 'disabled' ) ;
// get group id
var group _id = $ ( this ) . attr ( 'data-group-id' ) ;
// join or leave?
if ( $ ( this ) . hasClass ( 'member' ) ) {
var joinOrLeave = 'leave' ;
}
else {
var joinOrLeave = 'join' ;
}
// post to api
display _spinner ( ) ;
APIJoinOrLeaveGroup ( joinOrLeave , group _id , this , function ( data , this _element ) {
remove _spinner ( ) ;
$ ( this _element ) . removeClass ( 'disabled' ) ;
if ( data ) {
if ( data . member ) {
$ ( this _element ) . addClass ( 'member' ) ;
$ ( '.profile-card .member-stats strong' ) . html ( parseInt ( $ ( '.profile-card .member-stats strong' ) . html ( ) , 10 ) + 1 ) ;
$ ( '#user-groups strong' ) . html ( parseInt ( $ ( '#user-groups strong' ) . html ( ) , 10 ) + 1 ) ;
}
else if ( data . member === false ) {
$ ( this _element ) . removeClass ( 'member' ) ;
$ ( '.profile-card .member-stats strong' ) . html ( parseInt ( $ ( '.profile-card .member-stats strong' ) . html ( ) , 10 ) - 1 ) ;
$ ( '#user-groups strong' ) . html ( parseInt ( $ ( '#user-groups strong' ) . html ( ) , 10 ) - 1 ) ;
}
}
} ) ;
}
} ) ;
/ * ·
·
· Select a stream when the logged in user clicks their own queets , followers etc
·
· · · · · · · · · · · · · * /
$ ( '#user-header, #user-queets, #user-following, #user-followers, #user-groups' ) . on ( 'click' , function ( ) {
if ( $ ( this ) . attr ( 'id' ) == 'user-header' || $ ( this ) . attr ( 'id' ) == 'user-queets' ) {
setNewCurrentStream ( 'statuses/user_timeline.json?screen_name=' + window . loginUsername , function ( ) { } , true ) ;
}
else if ( $ ( this ) . attr ( 'id' ) == 'user-following' ) {
2013-08-20 22:26:25 +09:00
setNewCurrentStream ( 'statuses/friends.json?count=20' , function ( ) { } , true ) ;
2013-08-19 22:30:57 +09:00
}
else if ( $ ( this ) . attr ( 'id' ) == 'user-followers' ) {
2013-08-20 22:26:25 +09:00
setNewCurrentStream ( 'statuses/followers.json?count=20' , function ( ) { } , true ) ;
2013-08-19 22:30:57 +09:00
}
else if ( $ ( this ) . attr ( 'id' ) == 'user-groups' ) {
2013-08-20 22:26:25 +09:00
setNewCurrentStream ( 'statusnet/groups/list.json?count=10' , function ( ) { } , true ) ;
2013-08-19 22:30:57 +09:00
}
} ) ;
/ * ·
·
2013-08-20 22:26:25 +09:00
· Select a stream when clicking on queets , followers etc in a profile card or feed header
2013-08-19 22:30:57 +09:00
·
· · · · · · · · · · · · · * /
2013-08-20 22:26:25 +09:00
$ ( 'body' ) . on ( 'click' , '.profile-banner-footer .stats li a, .queet-stream' , function ( ) {
2013-08-19 22:30:57 +09:00
var screenName = $ ( '.profile-card-inner .screen-name' ) . html ( ) . substring ( 1 ) ;
if ( $ ( this ) . hasClass ( 'tweet-stats' ) ) {
setNewCurrentStream ( 'statuses/user_timeline.json?screen_name=' + screenName , function ( ) { } , true ) ;
}
else if ( $ ( this ) . hasClass ( 'following-stats' ) ) {
2013-08-20 22:26:25 +09:00
setNewCurrentStream ( 'statuses/friends.json?count=20&screen_name=' + screenName , function ( ) { } , true ) ;
2013-08-19 22:30:57 +09:00
}
else if ( $ ( this ) . hasClass ( 'follower-stats' ) ) {
2013-08-20 22:26:25 +09:00
setNewCurrentStream ( 'statuses/followers.json?count=20&screen_name=' + screenName , function ( ) { } , true ) ;
2013-08-19 22:30:57 +09:00
}
else if ( $ ( this ) . hasClass ( 'groups-stats' ) ) {
2013-08-20 22:26:25 +09:00
setNewCurrentStream ( 'statusnet/groups/list.json?count=10&screen_name=' + screenName , function ( ) { } , true ) ;
2013-08-19 22:30:57 +09:00
}
else if ( $ ( this ) . hasClass ( 'queets' ) ) {
setNewCurrentStream ( 'statuses/user_timeline.json?screen_name=' + screenName , function ( ) { } , true ) ;
}
else if ( $ ( this ) . hasClass ( 'mentions' ) ) {
setNewCurrentStream ( 'statuses/mentions.json?screen_name=' + screenName , function ( ) { } , true ) ;
}
else if ( $ ( this ) . hasClass ( 'favorites' ) ) {
setNewCurrentStream ( 'favorites.json?screen_name=' + screenName , function ( ) { } , true ) ;
2013-08-20 22:26:25 +09:00
}
else if ( $ ( this ) . hasClass ( 'member-stats' ) ) {
setNewCurrentStream ( 'statusnet/groups/membership/' + screenName + '.json?count=20' , function ( ) { } , true ) ;
}
else if ( $ ( this ) . hasClass ( 'admin-stats' ) ) {
setNewCurrentStream ( 'statusnet/groups/admins/' + screenName + '.json?count=20' , function ( ) { } , true ) ;
}
2013-08-19 22:30:57 +09:00
} ) ;
/ * ·
·
· Searching
·
· · · · · · · · · · · · · * /
$ ( '#search-query' ) . on ( 'keyup' , function ( e ) { if ( e . keyCode == 13 ) { showSearchStream ( ) ; } } ) ; // on enter in input field
$ ( 'button.icon.nav-search' ) . on ( 'click' , function ( e ) { showSearchStream ( ) ; } ) ; // on click on magnifying glass
function showSearchStream ( ) {
streamName = 'search.json?q=' + encodeURIComponent ( $ ( '#search-query' ) . val ( ) ) ;
setNewCurrentStream ( streamName , function ( ) { } , true ) ;
}
/ * ·
2013-11-23 08:31:04 +09:00
· < o
· Hijack all links and look for local users , tags , searches and groups . ( //
2013-08-19 22:30:57 +09:00
·
· If found , select that stream and prevent links default behaviour
·
· · · · · · · · · · · · · * /
2013-11-23 08:31:04 +09:00
$ ( document ) . on ( 'click' , 'a' , function ( e ) {
// not if metakeys are pressed!
if ( e . ctrlKey || e . altKey || e . shiftKey || e . metaKey ) {
return ;
}
// ugly fix: if this is the x remove users from history, prevent link but don't set a new currentstream
2014-01-29 03:42:47 +09:00
if ( $ ( e . target ) . is ( 'i.chev-right' ) ) {
2013-11-23 08:31:04 +09:00
e . preventDefault ( ) ;
return ;
}
// all non-hijacked links opens in new tab
$ ( this ) . attr ( 'target' , '_blank' ) ;
2013-08-19 22:30:57 +09:00
if ( typeof $ ( this ) . attr ( 'href' ) != 'undefined' ) {
2013-11-23 08:31:04 +09:00
// site root
if ( $ ( this ) . attr ( 'href' ) . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) . replace ( window . siteRootDomain , '' ) == '/' ) {
e . preventDefault ( ) ;
setNewCurrentStream ( 'statuses/public_timeline.json' , function ( ) { } , true ) ;
}
2014-01-29 03:42:47 +09:00
// whole network feed
else if ( $ ( this ) . attr ( 'href' ) . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) . replace ( window . siteRootDomain , '' ) == '/main/all' ) {
e . preventDefault ( ) ;
setNewCurrentStream ( 'statuses/public_and_external_timeline.json?since_id=1' , function ( ) { } , true ) ;
}
2013-11-23 08:31:04 +09:00
// logged in users streams
else if ( $ ( this ) . attr ( 'href' ) . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) . replace ( window . siteRootDomain + '/' + window . loginUsername , '' ) == '/all' ) {
e . preventDefault ( ) ;
setNewCurrentStream ( 'statuses/friends_timeline.json' , function ( ) { } , true ) ;
}
else if ( $ ( this ) . attr ( 'href' ) . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) . replace ( window . siteRootDomain + '/' + window . loginUsername , '' ) == '/replies' ) {
e . preventDefault ( ) ;
setNewCurrentStream ( 'statuses/mentions.json' , function ( ) { } , true ) ;
}
// else if ($(this).attr('href').replace('http://','').replace('https://','').replace(window.siteRootDomain + '/','') == window.loginUsername) {
// e.preventDefault();
// setNewCurrentStream('statuses/user_timeline.json',function(){},true);
// }
else if ( $ ( this ) . attr ( 'href' ) . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) . replace ( window . siteRootDomain + '/' + window . loginUsername , '' ) == '/favorites' ) {
e . preventDefault ( ) ;
setNewCurrentStream ( 'favorites.json' , function ( ) { } , true ) ;
}
2013-08-19 22:30:57 +09:00
// profiles
2013-11-23 08:31:04 +09:00
else if ( ( /^[a-zA-Z0-9]+$/ . test ( $ ( this ) . attr ( 'href' ) . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) . replace ( window . siteRootDomain + '/' , '' ) ) ) ) {
2013-08-19 22:30:57 +09:00
e . preventDefault ( ) ;
if ( $ ( this ) . parent ( ) . attr ( 'id' ) == 'user-profile-link' ) { // logged in user
setNewCurrentStream ( 'statuses/user_timeline.json?screen_name=' + window . loginUsername , function ( ) { } , true ) ;
}
else if ( $ ( this ) . hasClass ( 'account-group' ) ) { // any user
setNewCurrentStream ( 'statuses/user_timeline.json?screen_name=' + $ ( this ) . find ( '.screen-name' ) . text ( ) . substring ( 1 ) . toLowerCase ( ) , function ( ) { } , true ) ;
}
else { // any user
setNewCurrentStream ( 'statuses/user_timeline.json?screen_name=' + $ ( this ) . attr ( 'href' ) . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) . replace ( window . siteRootDomain + '/' , '' ) , function ( ) { } , true ) ;
}
}
else if ( ( /^[0-9]+$/ . test ( $ ( this ) . attr ( 'href' ) . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) . replace ( window . siteRootDomain + '/user/' , '' ) ) ) ) {
e . preventDefault ( ) ;
setNewCurrentStream ( 'statuses/user_timeline.json?screen_name=' + $ ( this ) . text ( ) . toLowerCase ( ) , function ( ) { } , true ) ;
}
// tags
else if ( $ ( this ) . attr ( 'href' ) . indexOf ( window . siteRootDomain + '/tag/' ) > - 1 ) {
e . preventDefault ( ) ;
2014-01-29 03:42:47 +09:00
setNewCurrentStream ( 'statusnet/tags/timeline/' + $ ( this ) . text ( ) . toLowerCase ( ) . replace ( '#' , '' ) + '.json' , function ( ) { } , true ) ;
2013-08-19 22:30:57 +09:00
}
// groups
else if ( /^[0-9]+$/ . test ( $ ( this ) . attr ( 'href' ) . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) . replace ( window . siteRootDomain + '/group/' , '' ) . replace ( '/id' , '' ) ) ) {
e . preventDefault ( ) ;
if ( $ ( this ) . hasClass ( 'account-group' ) ) {
var groupName = $ ( this ) . find ( '.screen-name' ) . html ( ) . substring ( 1 ) ;
}
else {
var groupName = $ ( this ) . text ( ) . toLowerCase ( ) ;
}
setNewCurrentStream ( 'statusnet/groups/timeline/' + groupName + '.json' , function ( ) { } , true ) ;
}
else if ( $ ( this ) . attr ( 'href' ) . indexOf ( window . siteRootDomain + '/group/' ) > - 1 ) {
e . preventDefault ( ) ;
setNewCurrentStream ( 'statusnet/groups/timeline/' + $ ( this ) . attr ( 'href' ) . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) . replace ( window . siteRootDomain + '/group/' , '' ) + '.json' , function ( ) { } , true ) ;
2013-11-23 08:31:04 +09:00
}
// search
else if ( $ ( this ) . attr ( 'href' ) . indexOf ( '/search/notice?q=' ) > - 1 ) {
e . preventDefault ( ) ;
var searchToStream = $ ( this ) . attr ( 'href' ) . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) . replace ( window . siteRootDomain , '' ) . replace ( '/search/notice?q=' , '' ) ;
setNewCurrentStream ( 'search.json?q=' + searchToStream , function ( ) { } , true ) ;
}
2013-08-19 22:30:57 +09:00
// profile picture
else if ( $ ( this ) . hasClass ( 'profile-picture' ) ) {
e . preventDefault ( ) ;
popUpAction ( 'popup-profile-picture' , $ ( '.profile-card-inner .screen-name' ) . html ( ) , '<img style="width:100%" src="' + $ ( this ) . attr ( 'href' ) + '" />' , false ) ;
}
// external profiles
else if ( ( $ ( this ) . children ( 'span.mention' ) . length > 0 // if it's a mention
|| ( $ ( this ) . hasClass ( 'account-group' ) && $ ( this ) . attr ( 'href' ) . indexOf ( '/group/' ) == - 1 ) // or if this is queet stream item header but not a group
|| ( $ ( this ) . closest ( '.stream-item' ) . hasClass ( 'activity' ) && $ ( this ) . attr ( 'href' ) . indexOf ( '/group/' ) == - 1 ) ) // or if it's a activity notice but not a group link
&& typeof window . loginUsername != 'undefined' ) { // if logged in
e . preventDefault ( ) ;
display _spinner ( ) ;
getFromAPI ( 'externalprofile/show.json?profileurl=' + encodeURIComponent ( $ ( this ) . attr ( 'href' ) ) , function ( data ) {
// external user found locally
if ( data ) {
2013-11-23 08:31:04 +09:00
console . log ( data ) ;
2013-08-19 22:30:57 +09:00
// empty strings and zeros instead of null
data . name = data . name || '' ;
data . profile _image _url = data . profile _image _url || '' ;
data . profile _image _url _profile _size = data . profile _image _url _profile _size || '' ;
data . profile _image _url _original = data . profile _image _url _original || '' ;
data . screen _name = data . screen _name || '' ;
data . description = data . description || '' ;
data . location = data . location || '' ;
data . url = data . url || '' ;
data . statusnet _profile _url = data . statusnet _profile _url || '' ;
data . statuses _count = data . statuses _count || 0 ;
data . followers _count = data . followers _count || 0 ;
data . friends _count = data . friends _count || 0 ;
// profile card
var followingClass = '' ;
if ( data . following ) {
followingClass = 'following' ;
}
2013-11-23 08:31:04 +09:00
var serverUrl = data . statusnet _profile _url . replace ( '/' + data . screen _name , '' ) ;
var userApiUrl = serverUrl + '/api/statuses/user_timeline.json?screen_name=' + data . screen _name ;
var screenNameWithServer = '@' + data . screen _name + '@' + serverUrl . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) ;
2013-08-19 22:30:57 +09:00
var followButton = '<div class="user-actions"><button data-follow-user-id="' + data . id + '" data-follow-user="' + data . statusnet _profile _url + '" type="button" class="follow-button ' + followingClass + '"><span class="button-text follow-text"><i class="follow"></i>' + window . sL . userFollow + '</span><span class="button-text following-text">' + window . sL . userFollowing + '</span><span class="button-text unfollow-text">' + window . sL . userUnfollow + '</span></button></div>' ;
var profileCard = '<div class="profile-card"><div class="profile-header-inner" style="background-image:url(' + data . profile _image _url _original + ')"><div class="profile-header-inner-overlay"></div><a class="profile-picture"><img src="' + data . profile _image _url _profile _size + '" /></a><div class="profile-card-inner"><h1 class="fullname">' + data . name + '<span></span></h1><h2 class="username"><span class="screen-name"><a target="_blank" href="' + data . statusnet _profile _url + '">' + screenNameWithServer + '</a></span><span class="follow-status"></span></h2><div class="bio-container"><p>' + data . description + '</p></div><p class="location-and-url"><span class="location">' + data . location + '</span><span class="divider"> · </span><span class="url"><a target="_blank" href="' + data . url + '">' + data . url . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) + '</a></span></p></div></div><div class="profile-banner-footer"><ul class="stats"><li><a target="_blank" href="' + data . statusnet _profile _url + '"><strong>' + data . statuses _count + '</strong>' + window . sL . notices + '</a></li><li><a target="_blank" href="' + data . statusnet _profile _url + '/subscriptions"><strong>' + data . friends _count + '</strong>' + window . sL . following + '</a></li><li><a target="_blank" href="' + data . statusnet _profile _url + '/subscribers"><strong>' + data . followers _count + '</strong>' + window . sL . followers + '</a></li></ul>' + followButton + '<div class="clearfix"></div></div></div><div class="clearfix"></div>' ;
popUpAction ( 'popup-external-profile' , screenNameWithServer , profileCard , false ) ;
2013-11-23 08:31:04 +09:00
// if remote server is https, do jsonp request directly, otherwise proxy
if ( serverUrl . substring ( 0 , 8 ) == 'https://' ) {
console . log ( userApiUrl ) ;
$ . ajax ( { url : userApiUrl , type : "GET" , dataType : "jsonp" , success : function ( data ) {
console . log ( data ) ;
}
} ) ;
}
else {
getFromAPI ( 'externalproxy.json?url=' + encodeURIComponent ( userApiUrl ) , function ( data ) {
if ( data ) {
}
} ) ;
}
2013-08-19 22:30:57 +09:00
remove _spinner ( ) ;
}
// external user not found locally, try externally
else {
// TODO!
}
} ) ;
}
// external groups
else if ( ( $ ( this ) . children ( 'span.group' ) . length > 0 // if it's a group mention
|| ( $ ( this ) . hasClass ( 'account-group' ) && $ ( this ) . attr ( 'href' ) . indexOf ( '/group/' ) > - 1 ) // or if this is group stream item header
|| ( $ ( this ) . closest ( '.stream-item' ) . hasClass ( 'activity' ) && $ ( this ) . attr ( 'href' ) . indexOf ( '/group/' ) > - 1 ) ) // or if it's a activity notice
&& typeof window . loginUsername != 'undefined' ) { // if logged in
e . preventDefault ( ) ;
display _spinner ( ) ;
getFromAPI ( 'statusnet/groups/show.json?id=foo&uri=' + encodeURIComponent ( $ ( this ) . attr ( 'href' ) ) , function ( data ) { if ( data ) {
data . nickname = data . nickname || '' ;
data . fullname = data . fullname || '' ;
data . stream _logo = data . stream _logo || 'http://quitter.se/theme/quitter-theme2/default-avatar-stream.png' ;
data . homepage _logo = data . homepage _logo || 'http://quitter.se/theme/quitter-theme2/default-avatar-profile.png' ;
data . original _logo = data . original _logo || 'http://quitter.se/theme/quitter-theme2/default-avatar-profile.png' ;
data . description = data . description || '' ;
data . homepage = data . homepage || '' ;
data . url = data . url || '' ;
data . member _count = data . member _count || 0 ;
data . admin _count = data . admin _count || 0 ;
// show user actions if logged in
var memberClass = '' ;
if ( data . member ) {
memberClass = 'member' ;
}
var groupRoot = data . url . substring ( 0 , data . url . indexOf ( '/group/' ) ) ;
var groupServer = groupRoot . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) ;
var memberButton = '<div class="user-actions"><button data-group-id="' + data . id + '" type="button" class="member-button ' + memberClass + '"><span class="button-text join-text"><i class="join"></i>' + window . sL . joinGroup + '</span><span class="button-text ismember-text">' + window . sL . isMemberOfGroup + '</span><span class="button-text leave-text">' + window . sL . leaveGroup + '</span></button></div>' ;
// get local member avatars
getFromAPI ( 'statusnet/groups/membership.json?id=' + data . id , function ( user _data ) { if ( user _data ) {
var avatars = '' ;
var i = 0 ;
$ . each ( user _data , function ( k , v ) {
if ( i < 7 ) {
avatars = avatars + '<img class="avatar size30" src="' + v . profile _image _url + '" />' ;
}
i ++ ;
} ) ;
2013-09-03 01:13:15 +09:00
var profileCard = '<div class="profile-card"><div class="profile-header-inner" style="background-image:url(' + data . original _logo + ')"><div class="profile-header-inner-overlay"></div><a class="profile-picture"><img src="' + data . homepage _logo + '" /></a><div class="profile-card-inner"><h1 class="fullname">' + data . fullname + '<span></span></h1><h2 class="username"><span class="screen-name"><a target="_blank" href="' + groupRoot + '/group/' + data . nickname + '">!' + data . nickname + '@' + groupServer + '</a></span></span></h2><div class="bio-container"><p>' + data . description + '</p></div><p class="location-and-url"></span><span class="url"><a href="' + data . homepage + '">' + data . homepage . replace ( 'http://' , '' ) . replace ( 'https://' , '' ) + '</a></span></p></div></div><div class="profile-banner-footer"><ul class="stats"><li><a target="_blank" href="' + groupRoot + '/group/' + data . nickname + '/members" class="member-stats">' + avatars + '</a></li></ul>' + memberButton + '<div class="clearfix"></div></div></div>' ;
2013-08-19 22:30:57 +09:00
popUpAction ( 'popup-external-group-profile' , '!' + data . nickname + '@' + groupServer , profileCard , false ) ;
remove _spinner ( ) ;
} } ) ;
} } ) ;
}
}
} ) ;
/ * ·
·
· When user clicks the x to remove a menu history item
·
· · · · · · · · · · · · · * /
2014-01-29 03:42:47 +09:00
$ ( 'body' ) . on ( 'click' , '#history-container .chev-right' , function ( event ) {
2013-08-19 22:30:57 +09:00
$ ( this ) . parent ( '.stream-selection' ) . remove ( ) ;
updateHistoryLocalStorage ( ) ;
} ) ;
/ * ·
·
· When sorting the history menu
·
· · · · · · · · · · · · · * /
2014-01-29 03:42:47 +09:00
2013-08-19 22:30:57 +09:00
$ ( '#history-container' ) . on ( "sortupdate" , function ( ) {
updateHistoryLocalStorage ( ) ;
} ) ;
/ * ·
·
· Load more from the current stream when scroll is 1000 px from bottom
·
· The search API is crap and don ' t do max _id and last _id , so we have to do pages there ...
·
· · · · · · · · · · · · · * /
$ ( window ) . scroll ( function ( ) {
if ( $ ( window ) . scrollTop ( ) + $ ( window ) . height ( ) > $ ( document ) . height ( ) - 1000 ) {
2013-11-29 20:49:54 +09:00
// not if we're already loading or if no stream is set yet
if ( ! $ ( 'body' ) . hasClass ( 'loading-older' ) && typeof window . currentStream != "undefined" ) {
2013-08-20 22:26:25 +09:00
$ ( 'body' ) . addClass ( 'loading-older' ) ;
// remove loading class in 10 seconds, i.e. try again if failed to load within 10 s
if ( window . currentStream . substring ( 0 , 6 ) != 'search' ) {
setTimeout ( function ( ) { $ ( 'body' ) . removeClass ( 'loading-older' ) ; } , 10000 ) ;
}
var lastStreamItemId = $ ( '#feed-body' ) . children ( '.stream-item' ) . last ( ) . attr ( 'id' ) ;
// if this is search or users lists, we need page and rpp vars, we store page number in an attribute
if ( window . currentStream . substring ( 0 , 6 ) == 'search'
|| window . currentStream . substring ( 0 , 23 ) == 'statuses/followers.json'
|| window . currentStream . substring ( 0 , 21 ) == 'statuses/friends.json'
|| window . currentStream . substring ( 0 , 26 ) == 'statusnet/groups/list.json'
|| window . currentStream . substring ( 0 , 28 ) == 'statusnet/groups/membership/'
|| window . currentStream . substring ( 0 , 24 ) == 'statusnet/groups/admins/' ) {
if ( typeof $ ( '#feed-body' ) . attr ( 'data-search-page-number' ) != 'undefined' ) {
var searchPage = parseInt ( $ ( '#feed-body' ) . attr ( 'data-search-page-number' ) , 10 ) ;
2013-08-19 22:30:57 +09:00
}
else {
2013-08-20 22:26:25 +09:00
var searchPage = 2 ;
2013-08-19 22:30:57 +09:00
}
2013-08-20 22:26:25 +09:00
var nextPage = searchPage + 1 ;
var getVars = qOrAmp ( window . currentStream ) + 'rpp=20&page=' + searchPage ; // search uses 'rrp' var and others 'count' for paging, though we can add rrp to others aswell without any problem
}
// normal streams
else {
var getVars = qOrAmp ( window . currentStream ) + 'max_id=' + $ ( '#feed-body' ) . children ( '.stream-item' ) . last ( ) . attr ( 'data-quitter-id-in-stream' ) ;
}
display _spinner ( ) ;
getFromAPI ( window . currentStream + getVars , function ( data ) {
if ( data ) {
addToFeed ( data , lastStreamItemId , 'visible' ) ;
$ ( 'body' ) . removeClass ( 'loading-older' ) ;
2013-08-19 22:30:57 +09:00
2013-08-20 22:26:25 +09:00
// if this is search our group users lists, we remember page number
if ( window . currentStream . substring ( 0 , 6 ) == 'search'
|| window . currentStream . substring ( 0 , 23 ) == 'statuses/followers.json'
|| window . currentStream . substring ( 0 , 21 ) == 'statuses/friends.json'
|| window . currentStream . substring ( 0 , 26 ) == 'statusnet/groups/list.json'
|| window . currentStream . substring ( 0 , 28 ) == 'statusnet/groups/membership/'
|| window . currentStream . substring ( 0 , 24 ) == 'statusnet/groups/admins/' ) {
$ ( '#feed-body' ) . attr ( 'data-search-page-number' , nextPage ) ;
2013-08-19 22:30:57 +09:00
}
2013-08-20 22:26:25 +09:00
remove _spinner ( ) ;
}
} ) ;
}
}
2013-08-19 22:30:57 +09:00
} ) ;
/ * ·
·
· Updates all queets ' times / dates
·
· · · · · · · · · · · · · * /
var updateTimesInterval = self . setInterval ( function ( ) {
$ ( '.created-at' ) . each ( function ( ) {
$ ( this ) . children ( 'a' ) . html ( parseTwitterDate ( $ ( this ) . attr ( 'data-created-at' ) ) ) ;
} ) ;
} , 10000 ) ;
/ * ·
·
· Check for new queets
·
· · · · · · · · · · · · · * /
var checkForNewQueetsInterval = window . setInterval ( function ( ) { checkForNewQueets ( ) } , window . timeBetweenPolling ) ;
function checkForNewQueets ( ) {
// no new requests if requests are very slow, e.g. searches
if ( ! $ ( 'body' ) . hasClass ( 'loading-newer' ) ) {
$ ( 'body' ) . addClass ( 'loading-newer' ) ;
// only of logged in and not user stream
if ( $ ( '#user-container' ) . css ( 'display' ) == 'block' && $ ( '.stream-item.user' ) . length == 0 ) {
var lastId = $ ( '#feed-body' ) . children ( '.stream-item' ) . not ( '.temp-post' ) . attr ( 'data-quitter-id-in-stream' ) ;
var addThisStream = window . currentStream ;
getFromAPI ( addThisStream + qOrAmp ( window . currentStream ) + 'since_id=' + lastId , function ( data ) {
if ( data ) {
$ ( 'body' ) . removeClass ( 'loading-newer' ) ;
if ( addThisStream == window . currentStream ) {
addToFeed ( data , false , 'hidden' ) ;
}
}
} ) ;
}
// if we have hidden items, show new-queets-bar
if ( $ ( '#feed-body' ) . find ( '.stream-item.hidden' ) . length > 0 ) {
var new _queets _num = $ ( '#feed-body' ) . find ( '.stream-item.hidden' ) . length ;
document . title = window . siteTitle + ' (' + new _queets _num + ')' ;
$ ( '#new-queets-bar' ) . parent ( ) . removeClass ( 'hidden' ) ;
// text plural
if ( new _queets _num == 1 ) {
var q _txt = ' ' + window . sL . newQueet ;
}
else {
var q _txt = ' ' + window . sL . newQueets ;
}
$ ( '#new-queets-bar' ) . html ( new _queets _num + q _txt ) ;
2013-08-20 07:16:01 +09:00
}
2013-08-19 22:30:57 +09:00
}
}
2013-08-20 07:16:01 +09:00
2013-08-19 22:30:57 +09:00
/ * ·
·
· Show hidden queets when user clicks on new - queets - bar
·
· · · · · · · · · · · · · * /
2014-01-29 03:42:47 +09:00
$ ( 'body' ) . on ( 'click' , '#new-queets-bar' , function ( ) {
2013-08-19 22:30:57 +09:00
document . title = window . siteTitle ;
$ ( '.stream-item.hidden' ) . css ( 'opacity' , '0' )
$ ( '.stream-item.hidden' ) . animate ( { opacity : '1' } , 200 ) ;
$ ( '.stream-item.hidden' ) . removeClass ( 'hidden' ) ;
$ ( '#new-queets-bar' ) . parent ( ) . addClass ( 'hidden' ) ;
} ) ;
/ * ·
·
· Expand and de - expand queets when clicking anywhere but on a few element types
·
· · · · · · · · · · · · · * /
2014-01-29 03:42:47 +09:00
$ ( 'body' ) . on ( 'click' , '.queet' , function ( event ) {
2013-08-19 22:30:57 +09:00
if ( ! $ ( event . target ) . is ( 'a' )
2013-11-23 08:31:04 +09:00
&& ! $ ( event . target ) . is ( '.CodeMirror-scroll' )
&& ! $ ( event . target ) . is ( '.cm-mention' )
&& ! $ ( event . target ) . is ( '.cm-tag' )
&& ! $ ( event . target ) . is ( '.cm-group' )
&& ! $ ( event . target ) . is ( '.cm-url' )
&& ! $ ( event . target ) . is ( 'pre' )
2013-08-19 22:30:57 +09:00
&& ! $ ( event . target ) . is ( '.name' )
&& ! $ ( event . target ) . is ( '.queet-box-template' )
2013-11-23 08:31:04 +09:00
&& ! $ ( event . target ) . is ( 'img' )
2013-08-19 22:30:57 +09:00
&& ! $ ( event . target ) . is ( 'button' )
&& ! $ ( event . target ) . is ( '.show-full-conversation' )
&& ! $ ( event . target ) . is ( 'span.mention' )
&& ! $ ( event . target ) . is ( '.action-reply-container a span' )
&& ! $ ( event . target ) . is ( '.action-reply-container a b' )
&& ! $ ( event . target ) . is ( '.action-rt-container a span' )
&& ! $ ( event . target ) . is ( '.action-rt-container a b' )
&& ! $ ( event . target ) . is ( '.action-del-container a span' )
&& ! $ ( event . target ) . is ( '.action-del-container a b' )
&& ! $ ( event . target ) . is ( '.action-fav-container a span' )
&& ! $ ( event . target ) . is ( '.action-fav-container a b' )
&& ! $ ( event . target ) . is ( 'span.group' )
&& ! $ ( event . target ) . is ( '.longdate' )
&& ! $ ( event . target ) . is ( '.screen-name' )
2013-09-03 01:13:15 +09:00
&& ! $ ( this ) . parent ( '.stream-item' ) . hasClass ( 'user' ) ) { // not if user stream
2013-08-19 22:30:57 +09:00
expand _queet ( $ ( this ) . parent ( ) ) ;
}
} ) ;
2013-11-23 08:31:04 +09:00
/ * ·
·
· Collapse all open conversations on esc or when clicking the margin
·
· · · · · · · · · · · · · * /
2013-08-19 22:30:57 +09:00
2013-11-23 08:31:04 +09:00
$ ( 'body' ) . click ( function ( event ) {
if ( $ ( event . target ) . is ( 'body' ) ) {
$ . each ( $ ( '.stream-item.expanded' ) , function ( ) {
expand _queet ( $ ( this ) , false ) ;
} ) ;
}
} ) ;
$ ( document ) . keyup ( function ( e ) {
if ( e . keyCode == 27 ) { // esc
$ . each ( $ ( '.stream-item.expanded' ) , function ( ) {
expand _queet ( $ ( this ) , false ) ;
} ) ;
}
} ) ;
2013-08-19 22:30:57 +09:00
/ * ·
·
· When clicking the delete - button
·
· · · · · · · · · · · · · * /
2014-01-29 03:42:47 +09:00
$ ( 'body' ) . on ( 'click' , '.action-del-container' , function ( ) {
2013-08-19 22:30:57 +09:00
var this _stream _item = $ ( this ) . parent ( ) . parent ( ) . parent ( ) . parent ( ) . parent ( ) ;
var this _qid = this _stream _item . attr ( 'data-quitter-id' ) ;
var $queetHtml = $ ( '<div>' ) . append ( this _stream _item . html ( ) ) ;
var $stuffToRemove = $queetHtml . find ( '.stream-item-footer, .expanded-content, .inline-reply-queetbox, .stream-item.conversation, .view-more-container-top, .view-more-container-bottom' ) ;
$stuffToRemove . remove ( ) ;
var queetHtmlWithoutFooterAndConversation = $queetHtml . html ( ) ;
popUpAction ( 'popup-delete-' + this _qid , window . sL . deleteConfirmation , queetHtmlWithoutFooterAndConversation , '<div class="right"><button class="close">' + window . sL . cancelVerb + '</button><button class="primary">' + window . sL . deleteVerb + '</button></div>' ) ;
$ ( '#popup-delete-' + this _qid + ' button.primary' ) . on ( 'click' , function ( ) {
display _spinner ( ) ;
$ ( '.modal-container' ) . remove ( ) ;
// delete
postActionToAPI ( 'statuses/destroy/' + this _qid + '.json' , function ( data ) {
if ( data ) {
remove _spinner ( ) ;
// remove the stream-item clicked and all other displays of this object from dom (e.g. in conversation)
$ ( '.stream-item[data-quitter-id="' + this _qid + '"]' ) . find ( '.queet' ) . animate ( { opacity : '0' } , 700 , function ( ) {
$ ( '.stream-item[data-quitter-id="' + this _qid + '"]' ) . remove ( ) ;
} ) ;
}
else {
remove _spinner ( ) ;
}
} ) ;
} ) ;
} ) ;
/ * ·
·
· When clicking the requeet - button
·
· · · · · · · · · · · · · * /
2014-01-29 03:42:47 +09:00
$ ( 'body' ) . on ( 'click' , '.action-rt-container' , function ( ) {
2013-08-19 22:30:57 +09:00
var this _stream _item = $ ( this ) . parent ( ) . parent ( ) . parent ( ) . parent ( ) . parent ( ) ;
var this _action = $ ( this ) ;
// requeet
if ( ! this _action . children ( '.with-icn' ) . hasClass ( 'done' ) ) {
this _action . children ( '.with-icn' ) . addClass ( 'done' ) ;
this _action . find ( '.with-icn b' ) . html ( window . sL . requeetedVerb ) ;
this _stream _item . addClass ( 'requeeted' ) ;
// post requeet
postActionToAPI ( 'statuses/retweet/' + this _stream _item . attr ( 'data-quitter-id' ) + '.json' , function ( data ) {
if ( data ) {
// success
this _stream _item . attr ( 'data-requeeted-by-me-id' , data . id ) ;
}
else {
// error
this _action . children ( '.with-icn' ) . removeClass ( 'done' ) ;
this _action . find ( '.with-icn b' ) . html ( window . sL . requeetVerb ) ;
this _stream _item . removeClass ( 'requeeted' ) ;
}
} ) ;
}
// un-requeet
else if ( this _action . children ( '.with-icn' ) . hasClass ( 'done' ) ) {
display _spinner ( ) ;
// if we don't have the id od the repeat stored in DOM, we need to look it up
// (might be a problem if there's more than 100 repeats)
if ( typeof this _stream _item . attr ( 'data-requeeted-by-me-id' ) == 'undefined' ) {
getFavsOrRequeetsForQueet ( 'requeets' , this _stream _item . attr ( 'data-quitter-id' ) , function ( data ) {
$ . each ( data , function ( key , obj ) {
if ( window . myUserID == obj . user . id ) {
var my _rq _id = obj . id ;
unRequeet ( this _stream _item , this _action , my _rq _id ) ;
}
} ) ;
} ) ;
}
// if we have the id stored in DOM
else {
var my _rq _id = this _stream _item . attr ( 'data-requeeted-by-me-id' ) ;
unRequeet ( this _stream _item , this _action , my _rq _id ) ;
}
}
} ) ;
/ * ·
·
· When clicking the fav - button
·
· · · · · · · · · · · · · * /
2014-01-29 03:42:47 +09:00
$ ( 'body' ) . on ( 'click' , '.action-fav-container' , function ( ) {
2013-08-19 22:30:57 +09:00
var this _stream _item = $ ( this ) . parent ( ) . parent ( ) . parent ( ) . parent ( ) . parent ( ) ;
var this _action = $ ( this ) ;
// fav
if ( ! this _action . children ( '.with-icn' ) . hasClass ( 'done' ) ) {
this _action . children ( '.with-icn' ) . addClass ( 'done' ) ;
this _action . find ( '.with-icn b' ) . html ( window . sL . favoritedVerb ) ;
this _stream _item . addClass ( 'favorited' ) ;
// post fav
postActionToAPI ( 'favorites/create/' + this _stream _item . attr ( 'data-quitter-id' ) + '.json' , function ( data ) {
if ( data ) {
// success
}
else {
// error
this _action . children ( '.with-icn' ) . removeClass ( 'done' ) ;
this _action . find ( '.with-icn b' ) . html ( window . sL . favoriteVerb ) ;
this _stream _item . removeClass ( 'favorited' ) ;
}
} ) ;
}
// unfav
else {
display _spinner ( ) ;
this _action . children ( '.with-icn' ) . removeClass ( 'done' ) ;
this _action . find ( '.with-icn b' ) . html ( window . sL . favoriteVerb ) ;
this _stream _item . removeClass ( 'favorited' ) ;
// post unfav
postActionToAPI ( 'favorites/destroy/' + this _stream _item . attr ( 'data-quitter-id' ) + '.json' , function ( data ) {
if ( data ) {
// success
remove _spinner ( ) ;
}
else {
// error
remove _spinner ( ) ;
this _action . children ( '.with-icn' ) . addClass ( 'done' ) ;
this _action . find ( '.with-icn b' ) . html ( window . sL . favoritedVerb ) ;
this _stream _item . addClass ( 'favorited' ) ;
}
} ) ;
}
} ) ;
/ * ·
·
· When clicking the reply - button
·
· · · · · · · · · · · · · * /
2014-01-29 03:42:47 +09:00
$ ( 'body' ) . on ( 'click' , '.action-reply-container' , function ( ) {
2013-11-23 08:31:04 +09:00
var this _stream _item = $ ( this ) . closest ( '.stream-item' ) ;
2013-08-19 22:30:57 +09:00
var this _stream _item _id = this _stream _item . attr ( 'data-quitter-id' ) ;
2013-11-23 08:31:04 +09:00
// grabbing the queet and view it in the popup, stripped of footer, reply box and other sruff
var $queetHtml = $ ( '<div>' ) . append ( this _stream _item . children ( '.queet' ) . outerHTML ( ) ) ;
var $queetHtmlFooter = $queetHtml . find ( '.stream-item-footer' ) ;
$queetHtmlFooter . remove ( ) ;
var $queetHtmlQueetBox = $queetHtml . find ( '.inline-reply-queetbox' ) ;
$queetHtmlQueetBox . remove ( ) ;
var $queetHtmlExpandedContent = $queetHtml . find ( '.expanded-content' ) ;
$queetHtmlExpandedContent . remove ( ) ;
var queetHtmlWithoutFooter = $queetHtml . html ( ) ;
popUpAction ( 'popup-reply-' + this _stream _item _id , window . sL . replyTo + ' ' + this _stream _item . find ( '.screen-name' ) . html ( ) , replyFormHtml ( this _stream _item , this _stream _item _id ) , queetHtmlWithoutFooter ) ;
expandInlineQueetBox ( $ ( '#popup-reply-' + this _stream _item _id ) . find ( '.modal-body' ) . find ( '.queet-box-template' ) ) ;
} ) ;
2013-08-19 22:30:57 +09:00
2013-11-23 08:31:04 +09:00
/ * ·
·
· When clicking the compose button on mobile view
·
· · · · · · · · · · · · · * /
$ ( 'body' ) . on ( 'click' , '#top-compose' , function ( ) {
popUpAction ( 'popup-compose' , window . sL . compose , '<div class="inline-reply-queetbox"><div class="queet-box-template"></div></div>' , false ) ;
expandInlineQueetBox ( $ ( '#popup-compose' ) . find ( '.queet-box-template' ) ) ;
2013-08-19 22:30:57 +09:00
} ) ;
/ * ·
·
· Close popups
·
· · · · · · · · · · · · · * /
$ ( 'body' ) . on ( 'click' , '.modal-container button.close' , function ( ) {
$ ( '.modal-container' ) . remove ( ) ;
} ) ;
$ ( 'body' ) . on ( 'click' , '.modal-close' , function ( ) {
$ ( '.modal-container' ) . remove ( ) ;
} ) ;
$ ( document ) . keyup ( function ( e ) {
if ( e . keyCode == 27 ) {
$ ( '.modal-container' ) . remove ( ) ;
}
} ) ;
/ * ·
·
· Expand inline reply form when clicked
·
· · · · · · · · · · · · · * /
2014-01-29 03:42:47 +09:00
$ ( 'body' ) . on ( 'click' , '.queet-box-template' , function ( ) {
2013-08-19 22:30:57 +09:00
// expand inline queet box
expandInlineQueetBox ( $ ( this ) ) ;
} ) ;
/ * ·
·
· Post inline and popup replies
·
· · · · · · · · · · · · · * /
$ ( 'body' ) . on ( 'click' , '.queet-toolbar button' , function ( ) {
if ( $ ( this ) . hasClass ( 'enabled' ) ) {
// set temp post id
if ( $ ( '.temp-post' ) . length == 0 ) {
var tempPostId = 'stream-item-temp-post-i' ;
}
else {
var tempPostId = $ ( '.temp-post' ) . attr ( 'id' ) + 'i' ;
}
2013-11-23 08:31:04 +09:00
var queetBoxID = $ ( this ) . parent ( ) . parent ( ) . parent ( ) . find ( '.queet-box-template' ) . attr ( 'id' ) ;
var queetText = window [ 'codemirror-' + queetBoxID ] . getValue ( ) ;
2014-01-29 03:42:47 +09:00
var queetHtml = '<div id="' + tempPostId + '" class="stream-item conversation temp-post" style="opacity:1"><div class="queet"><span class="dogear"></span><div class="queet-content"><div class="stream-item-header"><a class="account-group"><img class="avatar" src="' + $ ( '#user-avatar' ) . attr ( 'src' ) + '" /><strong class="name">' + $ ( '#user-name' ) . html ( ) + '</strong> <span class="screen-name">@' + $ ( '#user-screen-name' ) . html ( ) + '</span></a><small class="created-at">posting</small></div><div class="queet-text">' + queetText + '</div><div class="stream-item-footer"><span class="stream-item-expand"> </span></div></div></div></div>' ;
queetHtml = detectRTL ( queetHtml ) ;
2013-11-23 08:31:04 +09:00
2013-08-19 22:30:57 +09:00
// get reply to id and add temp queet
if ( $ ( '.modal-container' ) . find ( '.queet-toolbar button' ) . length > 0 ) { // from popup
var in _reply _to _status _id = $ ( '.modal-container' ) . attr ( 'id' ) . substring ( 12 ) ; // removes "popup-reply-" from popups id
$ ( '.modal-container' ) . remove ( ) ;
2014-01-29 03:42:47 +09:00
queetHtml = detectRTL ( queetHtml ) ;
// try to find an expanded queet to add the temp queet to
if ( $ ( '.stream-item.expanded[data-quitter-id="' + in _reply _to _status _id + '"]' ) . length > 0 ) {
$ ( '.stream-item.expanded[data-quitter-id="' + in _reply _to _status _id + '"]' ) . append ( queetHtml ) ;
}
else if ( $ ( '.stream-item.conversation[data-quitter-id="' + in _reply _to _status _id + '"]' ) . not ( '.hidden-conversation' ) . length > 0 ) {
$ ( '.stream-item.conversation[data-quitter-id="' + in _reply _to _status _id + '"]' ) . not ( '.hidden-conversation' ) . parent ( ) . append ( queetHtml ) ;
}
// if we cant find a proper place, just add it to top and remove conversation class
else {
$ ( '#feed-body' ) . prepend ( queetHtml . replace ( 'class="stream-item conversation' , 'class="stream-item' ) ) ;
}
2013-08-19 22:30:57 +09:00
}
else { // from inline reply
var in _reply _to _status _id = $ ( this ) . parent ( ) . parent ( ) . parent ( ) . parent ( ) . parent ( ) . attr ( 'data-quitter-id' ) ;
$ ( this ) . parent ( ) . parent ( ) . parent ( ) . parent ( ) . parent ( ) . append ( queetHtml ) ;
}
// null reply box
2013-11-23 08:31:04 +09:00
$ ( this ) . parent ( ) . parent ( ) . parent ( ) . find ( '.queet-box-template' ) . css ( 'display' , 'block' ) ;
$ ( this ) . parent ( ) . parent ( ) . parent ( ) . find ( '.CodeMirror' ) . remove ( ) ;
$ ( this ) . parent ( ) . parent ( ) . parent ( ) . find ( 'textarea#codemirror-' + queetBoxID ) . remove ( ) ;
2013-08-19 22:30:57 +09:00
$ ( this ) . parent ( ) . parent ( ) . parent ( ) . find ( '.queet-toolbar' ) . remove ( ) ;
2013-11-23 08:31:04 +09:00
delete window [ 'codemirror-' + queetBoxID ] ;
2013-08-19 22:30:57 +09:00
// check for new queets (one second from) NOW
setTimeout ( 'checkForNewQueets()' , 1000 ) ;
// post queet
2013-11-23 08:31:04 +09:00
postReplyToAPI ( queetText , in _reply _to _status _id , function ( data ) { if ( data ) {
2013-08-19 22:30:57 +09:00
// show real queet
var new _queet = Array ( ) ;
new _queet [ 0 ] = data ;
addToFeed ( new _queet , tempPostId , 'visible' ) ;
// remove temp queet
$ ( '#' + tempPostId ) . remove ( ) ;
2013-11-23 08:31:04 +09:00
// queet count
$ ( '#user-queets strong' ) . html ( parseInt ( $ ( '#user-queets strong' ) . html ( ) , 10 ) + 1 ) ;
2013-08-19 22:30:57 +09:00
} } ) ;
}
} ) ;
2013-11-23 08:31:04 +09:00
2013-08-19 22:30:57 +09:00
/ * ·
·
· Post queet
·
· · · · · · · · · · · · · * /
$ ( '#queet-toolbar button' ) . click ( function ( ) {
if ( $ ( this ) . hasClass ( 'enabled' ) ) {
// set temp post id
if ( $ ( '.temp-post' ) . length == 0 ) {
var tempPostId = 'stream-item-temp-post-i' ;
}
else {
var tempPostId = $ ( '.temp-post' ) . attr ( 'id' ) + 'i' ;
}
2013-11-23 08:31:04 +09:00
var queetText = codemirrorQueetBox . getValue ( ) ;
2013-08-19 22:30:57 +09:00
// remove trailing <br> and convert other <br> to newline
2013-11-23 08:31:04 +09:00
queetText = $ . trim ( queetText ) ;
2013-08-19 22:30:57 +09:00
// show temporary queet
var queetHtml = '<div id="' + tempPostId + '" class="stream-item temp-post" style="opacity:1"><div class="queet"><span class="dogear"></span><div class="queet-content"><div class="stream-item-header"><a class="account-group"><img class="avatar" src="' + $ ( '#user-avatar' ) . attr ( 'src' ) + '" /><strong class="name">' + $ ( '#user-name' ) . html ( ) + '</strong> <span class="screen-name">@' + $ ( '#user-screen-name' ) . html ( ) + '</span></a><small class="created-at">posting</small></div><div class="queet-text">' + queetText + '</div><div class="stream-item-footer"><span class="stream-item-expand"> </span></div></div></div></div>' ;
// detect rtl
queetHtml = detectRTL ( queetHtml ) ;
$ ( '#feed-body' ) . prepend ( queetHtml ) ;
// check for new queets (one second from) NOW
setTimeout ( 'checkForNewQueets()' , 1000 ) ;
// null post form
2013-11-23 08:31:04 +09:00
codemirrorQueetBox . setValue ( '' ) ;
$ ( '#queet-toolbar' ) . css ( 'display' , 'none' ) ;
$ ( '#queet-box' ) . css ( 'display' , 'block' ) ;
$ ( '#user-footer .CodeMirror-wrap' ) . css ( 'display' , 'none' ) ;
2013-08-19 22:30:57 +09:00
// post queet
postQueetToAPI ( queetText , function ( data ) { if ( data ) {
// show real queet
var new _queet = Array ( ) ;
new _queet [ 0 ] = data ;
addToFeed ( new _queet , tempPostId , 'visible' ) ;
// remove temp queet
$ ( '#' + tempPostId ) . remove ( ) ;
2013-11-23 08:31:04 +09:00
// queet count
$ ( '#user-queets strong' ) . html ( parseInt ( $ ( '#user-queets strong' ) . html ( ) , 10 ) + 1 ) ;
2013-08-19 22:30:57 +09:00
} } ) ;
}
} ) ;
2013-11-23 08:31:04 +09:00
/ * ·
·
· Codemirror configuration for queet box
·
· · · · · · · · · · · · · * /
CodeMirror . defaults . lineWrapping = true ;
2013-11-29 20:49:54 +09:00
CodeMirror . defineMode ( "gnusocial" , function ( config , parserConfig ) {
2013-11-23 08:31:04 +09:00
function tokenBase ( stream , state ) {
stream . string = stream . string + ' ' ; // makes regexping easier..
var ch = stream . next ( ) ;
// regexps
var externalMentionInBeginningRE = /[a-zA-Z0-9]+(@)[\wåäö\-\.]+(\.)((ac|ad|aero|af|ag|ai|al|am|an|ao|aq|arpa|asia|at|au|aw|ax|az|ba|bb|bd|be|bf|bg|bh|biz|bj|bm|bn|bo|br|bs|bt|bv|bw|by|bz|ca|cat|cc|cd|cf|cg|ch|ci|ck|cl|cm|cn|com|coop|cr|cu|cv|cw|cx|cy|cz|de|dj|dk|dm|do|dz|ec|edu|ee|eg|er|es|et|eu|fi|fj|fk|fm|fo|fr|ga|gb|gd|ge|gf|gg|gh|gi|gl|gm|gn|gov|gp|gq|gr|gs|gt|gu|gw|gy|hk|hm|hn|hr|ht|hu|id|ie|il|im|info|int|io|iq|ir|is|it|je|jm|jobs|jp|ke|kg|kh|ki|km|kn|kp|kr|kw|ky|kz|la|lb|lc|li|lk|lr|ls|lt|lu|lv|ly|ma|mc|md|me|mg|mh|mil|mk|ml|mm|mn|mobi|mp|mq|mr|ms|mt|museum|mv|mw|mx|my|mz|name|nc|net|nf|ng|ni|nl|no|np|nr|nu|nz|om|org|pa|pe|pf|pg|ph|pk|pl|pm|pn|post|pro|ps|pt|pw|py|qa|re|ro|rs|ru|rw|sa|sb|sc|sd|se|sg|sh|si|sj|sk|sl|sm|sn|so|sr|st|su|sv|sx|sy|sz|tc|td|tel|tf|tg|th|tj|tk|tl|tm|tn|to|tp|travel|tt|tv|tw|tz|ua|ug|uk|us|uy|uz|va|vc|ve|vg|vi|vn|vu|wf|ws|xxx|ye|yt|za|zm|zw)|(ae|ar|as|bi|co|in|jo|mo|mu|na|ne|pr|tr))/ ;
var mentionInBeginningRE = /[a-zA-Z0-9]+/ ;
var tagInBeginningRE = /[\wåäö\-]+/ ;
var groupInBeginningRE = /[a-zA-Z0-9]+/ ;
var externalMentionRE = /([ ]+)?@[a-zA-Z0-9]+(@)[\wåäö\-\.]+(\.)((ac|ad|aero|af|ag|ai|al|am|an|ao|aq|arpa|asia|at|au|aw|ax|az|ba|bb|bd|be|bf|bg|bh|biz|bj|bm|bn|bo|br|bs|bt|bv|bw|by|bz|ca|cat|cc|cd|cf|cg|ch|ci|ck|cl|cm|cn|com|coop|cr|cu|cv|cw|cx|cy|cz|de|dj|dk|dm|do|dz|ec|edu|ee|eg|er|es|et|eu|fi|fj|fk|fm|fo|fr|ga|gb|gd|ge|gf|gg|gh|gi|gl|gm|gn|gov|gp|gq|gr|gs|gt|gu|gw|gy|hk|hm|hn|hr|ht|hu|id|ie|il|im|info|int|io|iq|ir|is|it|je|jm|jobs|jp|ke|kg|kh|ki|km|kn|kp|kr|kw|ky|kz|la|lb|lc|li|lk|lr|ls|lt|lu|lv|ly|ma|mc|md|me|mg|mh|mil|mk|ml|mm|mn|mobi|mp|mq|mr|ms|mt|museum|mv|mw|mx|my|mz|name|nc|net|nf|ng|ni|nl|no|np|nr|nu|nz|om|org|pa|pe|pf|pg|ph|pk|pl|pm|pn|post|pro|ps|pt|pw|py|qa|re|ro|rs|ru|rw|sa|sb|sc|sd|se|sg|sh|si|sj|sk|sl|sm|sn|so|sr|st|su|sv|sx|sy|sz|tc|td|tel|tf|tg|th|tj|tk|tl|tm|tn|to|tp|travel|tt|tv|tw|tz|ua|ug|uk|us|uy|uz|va|vc|ve|vg|vi|vn|vu|wf|ws|xxx|ye|yt|za|zm|zw)|(ae|ar|as|bi|co|in|jo|mo|mu|na|ne|pr|tr))/ ;
var mentionRE = /([ ]+)?@[a-zA-Z0-9]+/ ;
var tagRE = /([ ]+)?#[\wåäö\-]+/ ;
var groupRE = /([ ]+)?![a-zA-Z0-9]+/ ;
var urlWithoutHttpInBeginningRE = /([\wåäö\-\.]+)?(\.)((ac|ad|aero|af|ag|ai|al|am|an|ao|aq|arpa|asia|at|au|aw|ax|az|ba|bb|bd|be|bf|bg|bh|biz|bj|bm|bn|bo|br|bs|bt|bv|bw|by|bz|ca|cat|cc|cd|cf|cg|ch|ci|ck|cl|cm|cn|com|coop|cr|cu|cv|cw|cx|cy|cz|de|dj|dk|dm|do|dz|ec|edu|ee|eg|er|es|et|eu|fi|fj|fk|fm|fo|fr|ga|gb|gd|ge|gf|gg|gh|gi|gl|gm|gn|gov|gp|gq|gr|gs|gt|gu|gw|gy|hk|hm|hn|hr|ht|hu|id|ie|il|im|info|int|io|iq|ir|is|it|je|jm|jobs|jp|ke|kg|kh|ki|km|kn|kp|kr|kw|ky|kz|la|lb|lc|li|lk|lr|ls|lt|lu|lv|ly|ma|mc|md|me|mg|mh|mil|mk|ml|mm|mn|mobi|mp|mq|mr|ms|mt|museum|mv|mw|mx|my|mz|name|nc|net|nf|ng|ni|nl|no|np|nr|nu|nz|om|org|pa|pe|pf|pg|ph|pk|pl|pm|pn|post|pro|ps|pt|pw|py|qa|re|ro|rs|ru|rw|sa|sb|sc|sd|se|sg|sh|si|sj|sk|sl|sm|sn|so|sr|st|su|sv|sx|sy|sz|tc|td|tel|tf|tg|th|tj|tk|tl|tm|tn|to|tp|travel|tt|tv|tw|tz|ua|ug|uk|us|uy|uz|va|vc|ve|vg|vi|vn|vu|wf|ws|xxx|ye|yt|za|zm|zw)|(ae|ar|as|bi|co|in|jo|mo|mu|na|ne|pr|tr))(\/[\wåäö\%\!\*\'\(\)\;\:\@\&\=\+\$\,\/\?\#\[\]\-\_\.\~]+)?(\/)?( )/ ;
var urlWithoutHttpRE = /([ ]+)?[\wåäö\-\.]+(\.)((ac|ad|aero|af|ag|ai|al|am|an|ao|aq|arpa|asia|at|au|aw|ax|az|ba|bb|bd|be|bf|bg|bh|biz|bj|bm|bn|bo|br|bs|bt|bv|bw|by|bz|ca|cat|cc|cd|cf|cg|ch|ci|ck|cl|cm|cn|com|coop|cr|cu|cv|cw|cx|cy|cz|de|dj|dk|dm|do|dz|ec|edu|ee|eg|er|es|et|eu|fi|fj|fk|fm|fo|fr|ga|gb|gd|ge|gf|gg|gh|gi|gl|gm|gn|gov|gp|gq|gr|gs|gt|gu|gw|gy|hk|hm|hn|hr|ht|hu|id|ie|il|im|info|int|io|iq|ir|is|it|je|jm|jobs|jp|ke|kg|kh|ki|km|kn|kp|kr|kw|ky|kz|la|lb|lc|li|lk|lr|ls|lt|lu|lv|ly|ma|mc|md|me|mg|mh|mil|mk|ml|mm|mn|mobi|mp|mq|mr|ms|mt|museum|mv|mw|mx|my|mz|name|nc|net|nf|ng|ni|nl|no|np|nr|nu|nz|om|org|pa|pe|pf|pg|ph|pk|pl|pm|pn|post|pro|ps|pt|pw|py|qa|re|ro|rs|ru|rw|sa|sb|sc|sd|se|sg|sh|si|sj|sk|sl|sm|sn|so|sr|st|su|sv|sx|sy|sz|tc|td|tel|tf|tg|th|tj|tk|tl|tm|tn|to|tp|travel|tt|tv|tw|tz|ua|ug|uk|us|uy|uz|va|vc|ve|vg|vi|vn|vu|wf|ws|xxx|ye|yt|za|zm|zw)|(ae|ar|as|bi|co|in|jo|mo|mu|na|ne|pr|tr))(\/[\wåäö\%\!\*\'\(\)\;\:\@\&\=\+\$\,\/\?\#\[\]\-\_\.\~]+)?(\/)?( )/ ;
var urlInBeginningRE = /(ttp\:\/\/|ttps\:\/\/)([\wåäö\-\.]+)?(\.)((ac|ad|aero|af|ag|ai|al|am|an|ao|aq|arpa|asia|at|au|aw|ax|az|ba|bb|bd|be|bf|bg|bh|biz|bj|bm|bn|bo|br|bs|bt|bv|bw|by|bz|ca|cat|cc|cd|cf|cg|ch|ci|ck|cl|cm|cn|com|coop|cr|cu|cv|cw|cx|cy|cz|de|dj|dk|dm|do|dz|ec|edu|ee|eg|er|es|et|eu|fi|fj|fk|fm|fo|fr|ga|gb|gd|ge|gf|gg|gh|gi|gl|gm|gn|gov|gp|gq|gr|gs|gt|gu|gw|gy|hk|hm|hn|hr|ht|hu|id|ie|il|im|info|int|io|iq|ir|is|it|je|jm|jobs|jp|ke|kg|kh|ki|km|kn|kp|kr|kw|ky|kz|la|lb|lc|li|lk|lr|ls|lt|lu|lv|ly|ma|mc|md|me|mg|mh|mil|mk|ml|mm|mn|mobi|mp|mq|mr|ms|mt|museum|mv|mw|mx|my|mz|name|nc|net|nf|ng|ni|nl|no|np|nr|nu|nz|om|org|pa|pe|pf|pg|ph|pk|pl|pm|pn|post|pro|ps|pt|pw|py|qa|re|ro|rs|ru|rw|sa|sb|sc|sd|se|sg|sh|si|sj|sk|sl|sm|sn|so|sr|st|su|sv|sx|sy|sz|tc|td|tel|tf|tg|th|tj|tk|tl|tm|tn|to|tp|travel|tt|tv|tw|tz|ua|ug|uk|us|uy|uz|va|vc|ve|vg|vi|vn|vu|wf|ws|xxx|ye|yt|za|zm|zw)|(ae|ar|as|bi|co|in|jo|mo|mu|na|ne|pr|tr))(\/[\wåäö\%\!\*\'\(\)\;\:\@\&\=\+\$\,\/\?\#\[\]\-\_\.\~]+)?(\/)?( )/ ;
var urlRE = /([ ]+)?(http\:\/\/|https\:\/\/)([\wåäö\-\.]+)?(\.)((ac|ad|aero|af|ag|ai|al|am|an|ao|aq|arpa|asia|at|au|aw|ax|az|ba|bb|bd|be|bf|bg|bh|biz|bj|bm|bn|bo|br|bs|bt|bv|bw|by|bz|ca|cat|cc|cd|cf|cg|ch|ci|ck|cl|cm|cn|com|coop|cr|cu|cv|cw|cx|cy|cz|de|dj|dk|dm|do|dz|ec|edu|ee|eg|er|es|et|eu|fi|fj|fk|fm|fo|fr|ga|gb|gd|ge|gf|gg|gh|gi|gl|gm|gn|gov|gp|gq|gr|gs|gt|gu|gw|gy|hk|hm|hn|hr|ht|hu|id|ie|il|im|info|int|io|iq|ir|is|it|je|jm|jobs|jp|ke|kg|kh|ki|km|kn|kp|kr|kw|ky|kz|la|lb|lc|li|lk|lr|ls|lt|lu|lv|ly|ma|mc|md|me|mg|mh|mil|mk|ml|mm|mn|mobi|mp|mq|mr|ms|mt|museum|mv|mw|mx|my|mz|name|nc|net|nf|ng|ni|nl|no|np|nr|nu|nz|om|org|pa|pe|pf|pg|ph|pk|pl|pm|pn|post|pro|ps|pt|pw|py|qa|re|ro|rs|ru|rw|sa|sb|sc|sd|se|sg|sh|si|sj|sk|sl|sm|sn|so|sr|st|su|sv|sx|sy|sz|tc|td|tel|tf|tg|th|tj|tk|tl|tm|tn|to|tp|travel|tt|tv|tw|tz|ua|ug|uk|us|uy|uz|va|vc|ve|vg|vi|vn|vu|wf|ws|xxx|ye|yt|za|zm|zw)|(ae|ar|as|bi|co|in|jo|mo|mu|na|ne|pr|tr))(\/[\wåäö\%\!\*\'\(\)\;\:\@\&\=\+\$\,\/\?\#\[\]\-\_\.\~]+)?(\/)?( )/ ;
var emailRE = /([ ]+)?([a-zA-Z0-9\!\#\$\%\&\'\*\+\-\/\=\?\^\_\`\{\|\}\~\.]+)?(@)[\wåäö\-\.]+(\.)((ac|ad|aero|af|ag|ai|al|am|an|ao|aq|arpa|asia|at|au|aw|ax|az|ba|bb|bd|be|bf|bg|bh|biz|bj|bm|bn|bo|br|bs|bt|bv|bw|by|bz|ca|cat|cc|cd|cf|cg|ch|ci|ck|cl|cm|cn|com|coop|cr|cu|cv|cw|cx|cy|cz|de|dj|dk|dm|do|dz|ec|edu|ee|eg|er|es|et|eu|fi|fj|fk|fm|fo|fr|ga|gb|gd|ge|gf|gg|gh|gi|gl|gm|gn|gov|gp|gq|gr|gs|gt|gu|gw|gy|hk|hm|hn|hr|ht|hu|id|ie|il|im|info|int|io|iq|ir|is|it|je|jm|jobs|jp|ke|kg|kh|ki|km|kn|kp|kr|kw|ky|kz|la|lb|lc|li|lk|lr|ls|lt|lu|lv|ly|ma|mc|md|me|mg|mh|mil|mk|ml|mm|mn|mobi|mp|mq|mr|ms|mt|museum|mv|mw|mx|my|mz|name|nc|net|nf|ng|ni|nl|no|np|nr|nu|nz|om|org|pa|pe|pf|pg|ph|pk|pl|pm|pn|post|pro|ps|pt|pw|py|qa|re|ro|rs|ru|rw|sa|sb|sc|sd|se|sg|sh|si|sj|sk|sl|sm|sn|so|sr|st|su|sv|sx|sy|sz|tc|td|tel|tf|tg|th|tj|tk|tl|tm|tn|to|tp|travel|tt|tv|tw|tz|ua|ug|uk|us|uy|uz|va|vc|ve|vg|vi|vn|vu|wf|ws|xxx|ye|yt|za|zm|zw)|(ae|ar|as|bi|co|in|jo|mo|mu|na|ne|pr|tr))( )/ ;
if ( stream . start == 0 && ch == "@" && stream . match ( externalMentionInBeginningRE ) ) { return "mention" }
else if ( stream . start == 0 && ch == "@" && stream . match ( mentionInBeginningRE ) ) { return "mention" }
else if ( stream . start == 0 && ch == "#" && stream . match ( tagInBeginningRE ) ) { return "mention" }
else if ( stream . start == 0 && ch == "!" && stream . match ( groupInBeginningRE ) ) { return "mention" }
else if ( stream . start == 0 && ch . match ( /[a-z0-9]/ ) && stream . match ( urlWithoutHttpInBeginningRE ) ) { stream . backUp ( 1 ) ; return "url" ; }
2013-11-29 20:49:54 +09:00
else if ( stream . start == 0 && ch == "h" && stream . match ( urlInBeginningRE ) ) { stream . backUp ( 1 ) ; return "url" ; }
2013-11-23 08:31:04 +09:00
else if ( ch == " " && stream . match ( externalMentionRE ) ) { return "mention" }
else if ( ch == " " && stream . match ( mentionRE ) ) { return "mention" }
else if ( ch == " " && stream . match ( tagRE ) ) { return "tag" ; }
else if ( ch == " " && stream . match ( groupRE ) ) { return "group" ; }
else if ( ch == " " && stream . match ( urlWithoutHttpRE ) ) { stream . backUp ( 1 ) ; return "url" ; }
2013-11-29 20:49:54 +09:00
else if ( ch == " " && stream . match ( urlRE ) ) { stream . backUp ( 1 ) ; return "url" ; }
2013-11-23 08:31:04 +09:00
else if ( ! ( ch == ' ' && stream . next ( ) == '.' ) && ! ( stream . start == 0 && ch == '.' ) && ( stream . start == 0 || ch == ' ' ) && stream . match ( emailRE ) ) {
stream . backUp ( 1 ) ;
return "email" ;
}
}
return {
startState : function ( base ) {
return { tokenize : tokenBase } ;
} ,
token : function ( stream , state ) {
state . tokenize = state . tokenize || tokenBase ;
var style = state . tokenize ( stream , state ) ;
return style ;
}
} ;
} ) ;
// activate queet box
var codemirrorQueetBox = CodeMirror . fromTextArea ( document . getElementById ( "codemirror-queet-box" ) , {
// submit on enter
onKeyEvent : function ( editor , event ) {
event = $ . event . fix ( event ) ;
var enterKeyHasBeenPressed = event . type == "keyup" && event . keyCode == 13 && ( event . ctrlKey || event . altKey ) ;
if ( enterKeyHasBeenPressed ) {
$ ( '#queet-toolbar button' ) . trigger ( 'click' ) ;
}
2013-11-29 20:49:54 +09:00
2013-11-23 08:31:04 +09:00
}
} ) ;
2013-11-29 20:49:54 +09:00
2013-11-23 08:31:04 +09:00
2013-08-19 22:30:57 +09:00
/ * ·
·
2013-08-20 18:33:59 +09:00
· Count chars in queet box on keyup
2013-08-19 22:30:57 +09:00
·
· · · · · · · · · · · · · * /
2013-11-23 08:31:04 +09:00
codemirrorQueetBox . on ( 'change' , function ( ) {
countCharsInQueetBox ( codemirrorQueetBox . getValue ( ) , $ ( '#queet-counter' ) , $ ( '#queet-toolbar button' ) ) ;
2013-08-19 22:30:57 +09:00
} ) ;
2013-11-23 08:31:04 +09:00
2013-08-19 22:30:57 +09:00
/ * ·
·
· Expand / collapse queet box on click and blur
·
· · · · · · · · · · · · · * /
$ ( '#queet-box' ) . click ( function ( ) {
2013-11-23 08:31:04 +09:00
$ ( '#queet-box' ) . css ( 'display' , 'none' ) ;
$ ( '#user-footer .CodeMirror-wrap' ) . css ( 'display' , 'block' ) ;
$ ( '#queet-toolbar' ) . css ( 'display' , 'block' ) ;
$ ( '#queet-toolbar button' ) . addClass ( 'disabled' ) ;
codemirrorQueetBox . setValue ( '' ) ;
codemirrorQueetBox . focus ( ) ;
countCharsInQueetBox ( codemirrorQueetBox . getValue ( ) , $ ( '#queet-counter' ) , $ ( '#queet-toolbar button' ) ) ;
2013-08-19 22:30:57 +09:00
} ) ;
2013-11-23 08:31:04 +09:00
codemirrorQueetBox . on ( "blur" , function ( ) {
if ( codemirrorQueetBox . getValue ( ) . length == 0 ) {
2013-08-19 22:30:57 +09:00
$ ( '#queet-toolbar' ) . css ( 'display' , 'none' ) ;
2013-11-23 08:31:04 +09:00
$ ( '#queet-box' ) . css ( 'display' , 'block' ) ;
$ ( '#user-footer .CodeMirror-wrap' ) . css ( 'display' , 'none' ) ;
2013-08-19 22:30:57 +09:00
}
2013-11-23 08:31:04 +09:00
} ) ;
2013-08-19 22:30:57 +09:00
/ * ·
·
· Shorten URL : s in queet boxes on space
·
· · · · · · · · · · · · · * /
2013-11-23 08:31:04 +09:00
// $('body').on('keyup','#queet-box',function(e){
// if(e.keyCode == 32) {
// shortenUrlsInBox($('#queet-box'),$('#queet-counter'),$('#queet-toolbar button'));
// }
// });
2014-01-29 03:42:47 +09:00
// $('body').on('keyup','.queet-box-template',function(e){
2013-11-23 08:31:04 +09:00
// if(e.keyCode == 32) {
// shortenUrlsInBox($(this),$(this).find('.queet-counter'),$(this).find('.queet-toolbar button'));
// }
// });
2013-08-19 22:30:57 +09:00
/ * ·
·
· When clicking show more links , walk upwards or downwards
·
· · · · · · · · · · · · · * /
2014-01-29 03:42:47 +09:00
$ ( 'body' ) . on ( 'click' , '.view-more-container-bottom' , function ( ) {
2013-08-19 22:30:57 +09:00
findReplyToStatusAndShow ( $ ( this ) . parent ( '.stream-item' ) . attr ( 'data-quitter-id' ) , $ ( this ) . attr ( 'data-replies-after' ) ) ;
$ ( this ) . remove ( ) ;
} ) ;
2014-01-29 03:42:47 +09:00
$ ( 'body' ) . on ( 'click' , '.view-more-container-top' , function ( ) {
2013-08-19 22:30:57 +09:00
var this _qid = $ ( this ) . closest ( '.stream-item:not(.conversation)' ) . attr ( 'data-quitter-id' ) ;
var queet = $ ( this ) . siblings ( '.queet' ) ;
rememberMyScrollPos ( queet , 'moretop' + this _qid ) ;
findInReplyToStatusAndShow ( $ ( this ) . parent ( '.stream-item' ) . attr ( 'data-quitter-id' ) , $ ( this ) . attr ( 'data-trace-from' ) , false , true ) ;
$ ( this ) . remove ( ) ;
backToMyScrollPos ( queet , 'moretop' + this _qid , false ) ;
// remove the "show full conversation" link if nothing more to show
if ( $ ( '#stream-item-' + $ ( this ) . parent ( '.stream-item' ) . attr ( 'data-quitter-id' ) ) . find ( '.hidden-conversation' ) . length == 0 ) {
$ ( '#stream-item-' + $ ( this ) . parent ( '.stream-item' ) . attr ( 'data-quitter-id' ) ) . children ( '.queet' ) . find ( '.show-full-conversation' ) . remove ( ) ;
}
} ) ;
/ * ·
·
· When clicking "show full conversation" , show all hidden queets in conversation
·
· · · · · · · · · · · · · * /
2014-01-29 03:42:47 +09:00
$ ( 'body' ) . on ( 'click' , '.show-full-conversation' , function ( ) {
2013-08-19 22:30:57 +09:00
var this _q = $ ( this ) . closest ( '.queet' ) ;
var this _qid = $ ( this ) . closest ( '.stream-item:not(.conversation)' ) . attr ( 'data-quitter-id' ) ;
rememberMyScrollPos ( this _q , this _qid ) ;
$ ( '#stream-item-' + $ ( this ) . attr ( 'data-stream-item-id' ) ) . find ( '.view-more-container-top' ) . remove ( ) ;
$ ( '#stream-item-' + $ ( this ) . attr ( 'data-stream-item-id' ) ) . find ( '.view-more-container-bottom' ) . remove ( ) ;
$ . each ( $ ( '#stream-item-' + $ ( this ) . attr ( 'data-stream-item-id' ) ) . find ( '.hidden-conversation' ) , function ( key , obj ) {
$ ( obj ) . removeClass ( 'hidden-conversation' ) ;
$ ( obj ) . animate ( { opacity : '1' } , 400 , function ( ) {
$ ( obj ) . css ( 'background-color' , 'pink' ) . animate ( { backgroundColor : '#F6F6F6' } , 1000 ) ;
} ) ;
} ) ;
$ ( this ) . remove ( ) ;
backToMyScrollPos ( this _q , this _qid , false ) ;
2014-01-29 03:42:47 +09:00
} ) ;