2014-01-05 20:07:11 +04:00
/ * !
2014-07-03 14:22:23 +04:00
* Webogram v0 . 2 - messaging web application for MTProto
2014-01-05 20:07:11 +04:00
* https : //github.com/zhukov/webogram
* Copyright ( C ) 2014 Igor Zhukov < igor . beatle @ gmail . com >
* https : //github.com/zhukov/webogram/blob/master/LICENSE
* /
'use strict' ;
/* Directives */
angular . module ( 'myApp.directives' , [ 'myApp.filters' ] )
2014-06-17 22:02:35 +04:00
. directive ( 'myHead' , function ( ) {
return {
restrict : 'AE' ,
scope : true ,
templateUrl : 'partials/head.html'
} ;
} )
2014-01-05 20:07:11 +04:00
. directive ( 'myDialog' , function ( ) {
return {
restrict : 'AE' ,
scope : true ,
translude : false ,
2014-03-09 18:12:41 -03:00
templateUrl : 'partials/dialog.html'
2014-01-05 20:07:11 +04:00
} ;
} )
. directive ( 'myMessage' , function ( ) {
return {
restrict : 'AE' ,
scope : true ,
translude : false ,
2014-03-09 18:12:41 -03:00
templateUrl : 'partials/message.html'
2014-01-05 20:07:11 +04:00
} ;
} )
2014-06-27 18:14:54 +04:00
. directive ( 'myDialogs' , function ( $modalStack , $transition , $window , $timeout ) {
2014-06-03 20:54:37 +04:00
return {
link : link
} ;
function link ( $scope , element , attrs ) {
var dialogsWrap = $ ( '.im_dialogs_wrap' , element ) [ 0 ] ,
scrollableWrap = $ ( '.im_dialogs_scrollable_wrap' , element ) [ 0 ] ,
searchField = $ ( '.im_dialogs_search_field' , element ) [ 0 ] ,
2014-06-27 18:14:54 +04:00
panelWrap = $ ( '.im_dialogs_panel' , element ) [ 0 ] ,
2014-06-13 12:40:14 +04:00
tabsWrap = $ ( '.im_dialogs_tabs_wrap' , element ) [ 0 ] ,
2014-06-03 20:54:37 +04:00
searchFocused = false ;
$ ( searchField ) . on ( 'focus blur' , function ( e ) {
searchFocused = e . type == 'focus' ;
if ( ! searchFocused ) {
2014-06-03 20:57:41 +04:00
$ ( scrollableWrap ) . find ( '.im_dialog_selected' ) . removeClass ( 'im_dialog_selected' ) ;
2014-06-27 18:14:54 +04:00
if ( ! searchField . value ) {
$scope . $emit ( 'ui_dialogs_search_clear' ) ;
}
2014-06-03 20:54:37 +04:00
}
} ) ;
2014-06-27 18:14:54 +04:00
$scope . $on ( 'dialogs_search_toggle' , function ( ) {
$ ( panelWrap ) . addClass ( 'im_dialogs_panel_search' ) ;
$scope . $broadcast ( 'ui_dialogs_search' ) ;
2014-06-27 21:52:24 +04:00
$ ( $window ) . scrollTop ( 0 ) ;
2014-06-27 18:14:54 +04:00
$timeout ( function ( ) {
searchField . focus ( ) ;
} )
} ) ;
$scope . $on ( 'search_clear' , function ( ) {
$ ( panelWrap ) . removeClass ( 'im_dialogs_panel_search' ) ;
$scope . $broadcast ( 'ui_dialogs_search' ) ;
} )
2014-06-13 12:40:14 +04:00
attrs . $observe ( 'hasTabs' , function ( newValue ) {
newValue = newValue == 'true' ;
2014-06-13 16:26:14 +04:00
$ ( tabsWrap ) . toggle ( newValue ) ;
$scope . $broadcast ( 'ui_dialogs_tabs' , newValue ) ;
2014-06-13 12:40:14 +04:00
} ) ;
$ ( document ) . on ( 'keydown' , onKeyDown ) ;
$scope . $on ( '$destroy' , function ( ) {
$ ( document ) . off ( 'keydown' , onKeyDown ) ;
} ) ;
2014-06-03 20:54:37 +04:00
function onKeyDown ( e ) {
if ( ! searchFocused && $modalStack . getTop ( ) ) {
return true ;
}
2014-06-04 20:59:46 +04:00
if ( e . keyCode == 36 && ! e . shiftKey && ! e . ctrlKey && e . altKey ) { // Alt + Home
var currentSelected = $ ( scrollableWrap ) . find ( '.im_dialog_wrap a' ) [ 0 ] ;
if ( currentSelected ) {
currentSelected . click ( ) ;
scrollableWrap . scrollTop = 0 ;
$ ( dialogsWrap ) . nanoScroller ( { flash : true } ) ;
}
return cancelEvent ( e ) ;
}
2014-06-05 15:40:52 +04:00
if ( e . keyCode == 27 || e . keyCode == 9 && e . shiftKey && ! e . ctrlKey && ! e . metaKey ) { // ESC or Shift + Tab
2014-06-03 20:54:37 +04:00
if ( ! searchFocused ) {
searchField . focus ( ) ;
if ( searchField . value ) {
searchField . select ( ) ;
}
}
return cancelEvent ( e ) ;
}
if ( searchFocused && e . keyCode == 13 ) { // Enter
var currentSelected = $ ( scrollableWrap ) . find ( '.im_dialog_selected' ) [ 0 ] || $ ( scrollableWrap ) . find ( '.im_dialog_wrap a' ) [ 0 ] ;
if ( currentSelected ) {
currentSelected . click ( ) ;
}
return cancelEvent ( e ) ;
}
if ( e . keyCode == 38 || e . keyCode == 40 ) { // UP, DOWN
var skip = ! e . shiftKey && e . altKey ;
if ( ! skip && ( ! searchFocused || e . metaKey ) ) {
return true ;
}
var next = e . keyCode == 40 ,
currentSelected = ! skip && $ ( scrollableWrap ) . find ( '.im_dialog_selected' ) [ 0 ] || $ ( scrollableWrap ) . find ( '.active a.im_dialog' ) [ 0 ] ,
currentSelectedWrap = currentSelected && currentSelected . parentNode ,
nextDialogWrap ;
if ( currentSelectedWrap ) {
var nextDialogWrap = currentSelected [ next ? 'nextSibling' : 'previousSibling' ] ;
if ( ! nextDialogWrap || ! nextDialogWrap . className || nextDialogWrap . className . indexOf ( 'im_dialog_wrap' ) == - 1 ) {
var dialogWraps = $ ( scrollableWrap ) . find ( '.im_dialog_wrap' ) ,
pos = dialogWraps . index ( currentSelected . parentNode ) ,
nextPos = pos + ( next ? 1 : - 1 ) ;
nextDialogWrap = dialogWraps [ nextPos ] ;
}
} else {
var dialogWraps = $ ( scrollableWrap ) . find ( '.im_dialog_wrap' ) ;
if ( next ) {
nextDialogWrap = dialogWraps [ 0 ] ;
} else {
nextDialogWrap = dialogWraps [ dialogWraps . length - 1 ] ;
}
}
if ( skip ) {
if ( nextDialogWrap ) {
$ ( nextDialogWrap ) . find ( 'a' ) [ 0 ] . click ( ) ;
}
} else {
if ( currentSelectedWrap && nextDialogWrap ) {
$ ( currentSelectedWrap ) . find ( 'a' ) . removeClass ( 'im_dialog_selected' ) ;
}
if ( nextDialogWrap ) {
$ ( nextDialogWrap ) . find ( 'a' ) . addClass ( 'im_dialog_selected' ) ;
}
}
if ( nextDialogWrap ) {
var elTop = nextDialogWrap . offsetTop ,
elHeight = nextDialogWrap . offsetHeight ,
scrollTop = scrollableWrap . scrollTop ,
viewportHeight = scrollableWrap . clientHeight ;
if ( scrollTop > elTop ) {
scrollableWrap . scrollTop = elTop ;
$ ( dialogsWrap ) . nanoScroller ( { flash : true } ) ;
}
else if ( scrollTop < elTop + elHeight - viewportHeight ) {
scrollableWrap . scrollTop = elTop + elHeight - viewportHeight ;
$ ( dialogsWrap ) . nanoScroller ( { flash : true } ) ;
}
}
return cancelEvent ( e ) ;
}
}
}
} )
2014-01-05 20:07:11 +04:00
. directive ( 'myDialogsList' , function ( $window , $timeout ) {
return {
2014-06-13 16:26:14 +04:00
link : link
2014-01-05 20:07:11 +04:00
} ;
2014-01-27 21:47:04 +04:00
2014-03-24 16:13:34 +04:00
function link ( $scope , element , attrs ) {
2014-01-27 21:47:04 +04:00
var dialogsWrap = $ ( '.im_dialogs_wrap' , element ) [ 0 ] ,
2014-06-27 18:14:54 +04:00
dialogsColWrap = $ ( '.im_dialogs_col_wrap' ) [ 0 ] ,
2014-01-27 21:47:04 +04:00
scrollableWrap = $ ( '.im_dialogs_scrollable_wrap' , element ) [ 0 ] ,
2014-02-12 19:10:05 +04:00
headWrap = $ ( '.tg_page_head' ) [ 0 ] ,
2014-06-27 18:14:54 +04:00
panelWrap = $ ( '.im_dialogs_panel' ) [ 0 ] ,
2014-02-10 21:25:21 +04:00
footer = $ ( '.im_page_footer' ) [ 0 ] ,
2014-06-13 12:40:14 +04:00
hasTabs = false ,
2014-01-05 20:07:11 +04:00
moreNotified = false ;
onContentLoaded ( function ( ) {
2014-02-10 21:30:44 +04:00
$ ( dialogsWrap ) . nanoScroller ( { preventPageScrolling : true , tabIndex : - 1 , iOSNativeScrolling : true } ) ;
2014-01-05 20:07:11 +04:00
} ) ;
var updateScroller = function ( ) {
onContentLoaded ( function ( ) {
$ ( dialogsWrap ) . nanoScroller ( ) ;
} ) ;
}
2014-03-24 16:13:34 +04:00
$scope . $on ( 'ui_dialogs_prepend' , updateScroller ) ;
2014-01-05 20:07:11 +04:00
2014-06-13 12:40:14 +04:00
$scope . $on ( 'ui_dialogs_tabs' , function ( e , newHasTabs ) {
hasTabs = newHasTabs ;
updateSizes ( ) ;
2014-06-27 18:14:54 +04:00
} ) ;
$scope . $on ( 'ui_dialogs_search' , updateSizes ) ;
$scope . $on ( 'ui_dialogs_update' , updateSizes ) ;
2014-06-13 12:40:14 +04:00
2014-01-05 20:07:11 +04:00
2014-03-24 16:13:34 +04:00
$scope . $on ( 'ui_dialogs_append' , function ( ) {
2014-01-05 20:07:11 +04:00
onContentLoaded ( function ( ) {
updateScroller ( ) ;
moreNotified = false ;
2014-03-21 12:13:04 +04:00
$timeout ( function ( ) {
$ ( scrollableWrap ) . trigger ( 'scroll' ) ;
} ) ;
2014-01-05 20:07:11 +04:00
} ) ;
} ) ;
2014-03-24 16:13:34 +04:00
$scope . $on ( 'ui_dialogs_change' , function ( ) {
2014-01-05 20:07:11 +04:00
onContentLoaded ( function ( ) {
updateScroller ( ) ;
moreNotified = false ;
2014-03-21 12:13:04 +04:00
$timeout ( function ( ) {
$ ( scrollableWrap ) . trigger ( 'scroll' ) ;
} ) ;
2014-01-05 20:07:11 +04:00
} ) ;
} ) ;
$ ( scrollableWrap ) . on ( 'scroll' , function ( e ) {
2014-06-30 17:30:42 +04:00
if ( ! element . is ( ':visible' ) ) return ;
2014-01-27 21:47:04 +04:00
// console.log('scroll', moreNotified);
2014-01-05 20:07:11 +04:00
if ( ! moreNotified && scrollableWrap . scrollTop >= scrollableWrap . scrollHeight - scrollableWrap . clientHeight - 300 ) {
2014-01-27 21:47:04 +04:00
// console.log('emit need more');
2014-03-24 16:13:34 +04:00
$scope . $emit ( 'dialogs_need_more' ) ;
2014-01-05 20:07:11 +04:00
moreNotified = true ;
}
} ) ;
function updateSizes ( ) {
2014-03-01 17:54:53 +01:00
if ( attrs . modal ) {
$ ( element ) . css ( {
2014-06-17 22:02:35 +04:00
height : $ ( $window ) . height ( ) -
( Config . Navigator . mobile ? 100 : 200 )
2014-03-01 17:54:53 +01:00
} ) ;
updateScroller ( ) ;
return ;
}
2014-03-07 16:34:25 +01:00
if ( ! headWrap || ! headWrap . offsetHeight ) {
2014-02-12 19:10:05 +04:00
headWrap = $ ( '.tg_page_head' ) [ 0 ] ;
}
2014-03-07 16:34:25 +01:00
if ( ! footer || ! footer . offsetHeight ) {
footer = $ ( '.im_page_footer' ) [ 0 ] ;
}
2014-06-27 18:14:54 +04:00
if ( ! panelWrap || ! panelWrap . offsetHeight ) {
panelWrap = $ ( '.im_dialogs_panel' ) [ 0 ] ;
}
if ( ! dialogsColWrap || ! dialogsColWrap . offsetHeight ) {
dialogsColWrap = $ ( '.im_dialogs_col_wrap' ) [ 0 ] ;
}
2014-03-04 23:52:26 +01:00
$ ( element ) . css ( {
2014-06-27 18:14:54 +04:00
height : $ ( $window ) . height ( ) -
footer . offsetHeight -
( headWrap ? headWrap . offsetHeight : 44 ) -
( panelWrap ? panelWrap . offsetHeight : 58 ) -
parseInt ( $ ( dialogsColWrap ) . css ( 'paddingBottom' ) || 0 )
2014-03-04 23:52:26 +01:00
} ) ;
2014-03-07 16:34:25 +01:00
2014-03-04 23:52:26 +01:00
updateScroller ( ) ;
2014-01-05 20:07:11 +04:00
}
$ ( $window ) . on ( 'resize' , updateSizes ) ;
updateSizes ( ) ;
2014-03-04 23:52:26 +01:00
setTimeout ( updateSizes , 1000 ) ;
2014-01-05 20:07:11 +04:00
} ;
} )
2014-03-01 17:54:53 +01:00
. directive ( 'myContactsList' , function ( $window , $timeout ) {
return {
link : link
} ;
2014-03-24 16:13:34 +04:00
function link ( $scope , element , attrs ) {
2014-03-01 17:54:53 +01:00
var searchWrap = $ ( '.contacts_modal_search' ) [ 0 ] ,
panelWrap = $ ( '.contacts_modal_panel' ) [ 0 ] ,
contactsWrap = $ ( '.contacts_wrap' , element ) [ 0 ] ;
onContentLoaded ( function ( ) {
$ ( contactsWrap ) . nanoScroller ( { preventPageScrolling : true , tabIndex : - 1 , iOSNativeScrolling : true } ) ;
updateSizes ( ) ;
} ) ;
function updateSizes ( ) {
$ ( element ) . css ( {
2014-06-17 22:02:35 +04:00
height : $ ( $window ) . height ( ) -
( panelWrap && panelWrap . offsetHeight || 0 ) -
( searchWrap && searchWrap . offsetHeight || 0 ) -
2014-06-21 01:44:23 +04:00
( Config . Navigator . mobile ? 100 : 200 )
2014-03-01 17:54:53 +01:00
} ) ;
$ ( contactsWrap ) . nanoScroller ( ) ;
}
$ ( $window ) . on ( 'resize' , updateSizes ) ;
2014-03-24 16:13:34 +04:00
$scope . $on ( 'contacts_change' , function ( ) {
2014-03-01 17:54:53 +01:00
onContentLoaded ( updateSizes )
} ) ;
} ;
} )
2014-04-27 16:25:04 +08:00
. directive ( 'myCountriesList' , function ( $window , $timeout ) {
return {
link : link
} ;
function link ( $scope , element , attrs ) {
var searchWrap = $ ( '.countries_modal_search' ) [ 0 ] ,
panelWrap = $ ( '.countries_modal_panel' ) [ 0 ] ,
countriesWrap = $ ( '.countries_wrap' , element ) [ 0 ] ;
onContentLoaded ( function ( ) {
$ ( countriesWrap ) . nanoScroller ( { preventPageScrolling : true , tabIndex : - 1 , iOSNativeScrolling : true } ) ;
updateSizes ( ) ;
} ) ;
function updateSizes ( ) {
$ ( element ) . css ( {
height : $ ( $window ) . height ( ) - ( panelWrap && panelWrap . offsetHeight || 0 ) - ( searchWrap && searchWrap . offsetHeight || 0 ) - 200
} ) ;
$ ( countriesWrap ) . nanoScroller ( ) ;
}
$ ( $window ) . on ( 'resize' , updateSizes ) ;
$scope . $on ( 'contacts_change' , function ( ) {
onContentLoaded ( updateSizes )
} ) ;
} ;
} )
2014-06-16 17:18:23 +04:00
. directive ( 'myHistory' , function ( $window , $timeout , $rootScope , $transition ) {
2014-01-05 20:07:11 +04:00
return {
link : link
} ;
2014-03-24 16:13:34 +04:00
function link ( $scope , element , attrs ) {
2014-01-27 21:47:04 +04:00
var historyWrap = $ ( '.im_history_wrap' , element ) [ 0 ] ,
2014-03-20 21:56:03 +04:00
historyMessagesEl = $ ( '.im_history_messages' , element ) [ 0 ] ,
2014-01-27 21:47:04 +04:00
historyEl = $ ( '.im_history' , element ) [ 0 ] ,
scrollableWrap = $ ( '.im_history_scrollable_wrap' , element ) [ 0 ] ,
scrollable = $ ( '.im_history_scrollable' , element ) [ 0 ] ,
2014-01-05 20:07:11 +04:00
panelWrap = $ ( '.im_history_panel_wrap' , element ) [ 0 ] ,
2014-03-05 18:34:56 +01:00
bottomPanelWrap = $ ( '.im_bottom_panel_wrap' , element ) [ 0 ] ,
2014-01-05 20:07:11 +04:00
sendFormWrap = $ ( '.im_send_form_wrap' , element ) [ 0 ] ,
2014-02-10 21:25:21 +04:00
headWrap = $ ( '.tg_page_head' ) [ 0 ] ,
footer = $ ( '.im_page_footer' ) [ 0 ] ,
sendForm = $ ( '.im_send_form' , element ) [ 0 ] ,
2014-06-13 01:46:46 +04:00
moreNotified = false ,
lessNotified = false ;
2014-01-05 20:07:11 +04:00
onContentLoaded ( function ( ) {
2014-01-17 13:27:49 +04:00
scrollableWrap . scrollTop = scrollableWrap . scrollHeight ;
2014-01-05 20:07:11 +04:00
} ) ;
2014-06-27 18:14:54 +04:00
$ ( historyWrap ) . nanoScroller ( { preventPageScrolling : true , tabIndex : - 1 , iOSNativeScrolling : true } ) ;
2014-01-05 20:07:11 +04:00
var updateScroller = function ( delay ) {
2014-06-13 01:46:46 +04:00
// console.trace('scroller update', delay);
2014-01-05 20:07:11 +04:00
$timeout ( function ( ) {
2014-01-17 13:27:49 +04:00
if ( ! $ ( scrollableWrap ) . hasClass ( 'im_history_to_bottom' ) ) {
$ ( historyWrap ) . nanoScroller ( ) ;
}
2014-01-05 20:07:11 +04:00
} , delay || 0 ) ;
}
2014-03-20 21:56:03 +04:00
var transform = false ,
trs = [ 'transform' , 'webkitTransform' , 'MozTransform' , 'msTransform' , 'OTransform' ] ,
2014-03-21 12:13:04 +04:00
i ;
2014-03-20 21:56:03 +04:00
for ( i = 0 ; i < trs . length ; i ++ ) {
if ( trs [ i ] in historyMessagesEl . style ) {
transform = trs [ i ] ;
break ;
}
}
2014-06-17 22:02:35 +04:00
var animated = transform ? true : false ,
2014-01-17 13:27:49 +04:00
curAnimation = false ;
2014-03-20 21:56:03 +04:00
2014-06-13 01:46:46 +04:00
$scope . $on ( 'ui_history_append_new' , function ( e , options ) {
2014-01-29 18:37:18 +04:00
if ( ! atBottom && ! options . my ) {
2014-01-17 13:27:49 +04:00
return ;
}
2014-06-30 17:30:42 +04:00
var curAnimated = animated &&
! $rootScope . idle . isIDLE &&
historyMessagesEl . clientHeight > 0 ,
2014-06-17 22:02:35 +04:00
wasH ;
2014-06-30 17:30:42 +04:00
if ( curAnimated ) {
wasH = scrollableWrap . scrollHeight ;
} else {
2014-01-17 13:27:49 +04:00
$ ( scrollable ) . css ( { bottom : 0 } ) ;
$ ( scrollableWrap ) . addClass ( 'im_history_to_bottom' ) ;
}
2014-01-09 16:13:12 +04:00
2014-01-17 13:27:49 +04:00
onContentLoaded ( function ( ) {
2014-06-17 22:02:35 +04:00
if ( curAnimated ) {
2014-01-17 13:27:49 +04:00
curAnimation = true ;
2014-03-20 21:56:03 +04:00
$ ( historyMessagesEl ) . removeClass ( 'im_history_appending' ) ;
scrollableWrap . scrollTop = scrollableWrap . scrollHeight ;
$ ( historyMessagesEl ) . css ( transform , 'translate(0px, ' + ( scrollableWrap . scrollHeight - wasH ) + 'px)' ) ;
2014-03-29 23:34:40 +04:00
$ ( historyWrap ) . nanoScroller ( ) ;
var styles = { } ;
styles [ transform ] = 'translate(0px, 0px)' ;
$ ( historyMessagesEl ) . addClass ( 'im_history_appending' ) ;
$transition ( $ ( historyMessagesEl ) , styles ) . then ( function ( ) {
curAnimation = false ;
$ ( historyMessagesEl ) . removeClass ( 'im_history_appending' ) ;
updateBottomizer ( ) ;
} ) ;
2014-01-17 13:27:49 +04:00
} else {
2014-01-05 20:07:11 +04:00
$ ( scrollableWrap ) . removeClass ( 'im_history_to_bottom' ) ;
2014-01-09 16:13:12 +04:00
$ ( scrollable ) . css ( { bottom : '' } ) ;
2014-01-17 13:27:49 +04:00
scrollableWrap . scrollTop = scrollableWrap . scrollHeight ;
2014-03-20 21:56:03 +04:00
updateBottomizer ( ) ;
2014-01-17 13:27:49 +04:00
}
} ) ;
2014-01-05 20:07:11 +04:00
} ) ;
2014-06-13 01:46:46 +04:00
function changeScroll ( ) {
var unreadSplit , focusMessage ;
if ( focusMessage = $ ( '.im_message_focus' , scrollableWrap ) [ 0 ] ) {
2014-06-13 12:40:14 +04:00
scrollableWrap . scrollTop = Math . max ( 0 , focusMessage . offsetTop - Math . floor ( scrollableWrap . clientHeight / 2 ) + 26 ) ;
2014-06-13 01:46:46 +04:00
atBottom = false ;
} else if ( unreadSplit = $ ( '.im_message_unread_split' , scrollableWrap ) [ 0 ] ) {
scrollableWrap . scrollTop = Math . max ( 0 , unreadSplit . offsetTop - 52 ) ;
atBottom = false ;
} else {
scrollableWrap . scrollTop = scrollableWrap . scrollHeight ;
2014-07-02 23:47:28 +04:00
atBottom = true ;
2014-06-13 01:46:46 +04:00
}
updateScroller ( ) ;
$timeout ( function ( ) {
$ ( scrollableWrap ) . trigger ( 'scroll' ) ;
} ) ;
} ;
2014-03-24 16:13:34 +04:00
$scope . $on ( 'ui_history_change' , function ( ) {
2014-01-05 20:07:11 +04:00
$ ( scrollableWrap ) . addClass ( 'im_history_to_bottom' ) ;
$ ( scrollable ) . css ( { bottom : 0 } ) ;
onContentLoaded ( function ( ) {
$ ( scrollableWrap ) . removeClass ( 'im_history_to_bottom' ) ;
$ ( scrollable ) . css ( { bottom : '' } ) ;
2014-03-09 22:43:45 +01:00
updateSizes ( true ) ;
2014-01-05 20:07:11 +04:00
moreNotified = false ;
2014-06-13 01:46:46 +04:00
lessNotified = false ;
changeScroll ( ) ;
2014-01-05 20:07:11 +04:00
} ) ;
} ) ;
2014-06-13 01:46:46 +04:00
$scope . $on ( 'ui_history_change_scroll' , function ( ) {
onContentLoaded ( changeScroll )
} ) ;
2014-03-24 16:13:34 +04:00
$scope . $on ( 'ui_history_focus' , function ( ) {
2014-01-29 18:37:18 +04:00
if ( ! atBottom ) {
scrollableWrap . scrollTop = scrollableWrap . scrollHeight ;
updateScroller ( ) ;
atBottom = true ;
}
} ) ;
2014-01-05 20:07:11 +04:00
2014-03-24 16:13:34 +04:00
$scope . $on ( 'ui_history_prepend' , function ( ) {
2014-01-05 20:07:11 +04:00
var sh = scrollableWrap . scrollHeight ,
st = scrollableWrap . scrollTop ,
2014-06-30 17:30:42 +04:00
pr = parseInt ( $ ( scrollableWrap ) . css ( 'paddingRight' ) ) ,
2014-01-05 20:07:11 +04:00
ch = scrollableWrap . clientHeight ;
$ ( scrollableWrap ) . addClass ( 'im_history_to_bottom' ) ;
2014-03-24 20:50:43 +04:00
scrollableWrap . scrollHeight ; // Some strange Chrome bug workaround
2014-06-30 17:30:42 +04:00
$ ( scrollable ) . css ( { bottom : - ( sh - st - ch ) , marginLeft : - Math . ceil ( pr / 2 ) } ) ;
2014-01-05 20:07:11 +04:00
2014-06-27 22:39:35 +04:00
var upd = function ( ) {
$ ( scrollableWrap ) . removeClass ( 'im_history_to_bottom' ) ;
2014-06-30 17:39:47 +04:00
$ ( scrollable ) . css ( { bottom : '' , marginLeft : '' } ) ;
2014-06-27 22:39:35 +04:00
scrollableWrap . scrollTop = st + scrollableWrap . scrollHeight - sh ;
2014-07-02 23:47:28 +04:00
2014-06-27 22:39:35 +04:00
updateBottomizer ( ) ;
moreNotified = false ;
2014-01-05 20:07:11 +04:00
2014-06-27 22:39:35 +04:00
$timeout ( function ( ) {
if ( scrollableWrap . scrollHeight != sh ) {
$ ( scrollableWrap ) . trigger ( 'scroll' ) ;
}
} ) ;
2014-03-20 21:56:03 +04:00
2014-06-27 22:39:35 +04:00
clearTimeout ( timer ) ;
unreg ( ) ;
} ,
timer = setTimeout ( upd , 0 ) ,
unreg = $scope . $on ( '$viewContentLoaded' , upd ) ;
2014-01-05 20:07:11 +04:00
} ) ;
2014-06-13 01:46:46 +04:00
$scope . $on ( 'ui_history_append' , function ( ) {
var sh = scrollableWrap . scrollHeight ;
onContentLoaded ( function ( ) {
2014-06-13 16:26:14 +04:00
atBottom = false ;
2014-06-13 01:46:46 +04:00
updateBottomizer ( ) ;
lessNotified = false ;
$timeout ( function ( ) {
if ( scrollableWrap . scrollHeight != sh ) {
$ ( scrollableWrap ) . trigger ( 'scroll' ) ;
}
} ) ;
} ) ;
} ) ;
2014-03-24 16:13:34 +04:00
$scope . $on ( 'ui_panel_update' , function ( ) {
2014-02-13 23:08:38 +04:00
onContentLoaded ( function ( ) {
2014-03-05 18:34:56 +01:00
updateSizes ( ) ;
2014-03-24 16:13:34 +04:00
$scope . $broadcast ( 'ui_message_send' ) ;
2014-03-20 22:16:11 +04:00
$timeout ( function ( ) {
$ ( scrollableWrap ) . trigger ( 'scroll' ) ;
} ) ;
2014-02-13 23:08:38 +04:00
} ) ;
} ) ;
2014-04-28 12:39:16 +08:00
$scope . $on ( 'ui_selection_clear' , function ( ) {
if ( window . getSelection ) {
if ( window . getSelection ( ) . empty ) { // Chrome
window . getSelection ( ) . empty ( ) ;
} else if ( window . getSelection ( ) . removeAllRanges ) { // Firefox
window . getSelection ( ) . removeAllRanges ( ) ;
}
} else if ( document . selection ) { // IE?
document . selection . empty ( ) ;
}
} ) ;
2014-03-24 16:13:34 +04:00
$scope . $on ( 'ui_editor_resize' , updateSizes ) ;
2014-04-08 16:38:24 +02:00
$scope . $on ( 'ui_height' , function ( ) {
2014-04-17 19:06:00 -04:00
onContentLoaded ( updateSizes ) ;
// updateSizes();
2014-04-08 16:38:24 +02:00
} ) ;
2014-01-31 23:05:24 +04:00
2014-01-05 20:07:11 +04:00
var atBottom = true ;
$ ( scrollableWrap ) . on ( 'scroll' , function ( e ) {
2014-06-30 17:30:42 +04:00
if ( ! element . is ( ':visible' ) ||
$ ( scrollableWrap ) . hasClass ( 'im_history_to_bottom' ) ||
curAnimation ) {
2014-01-17 13:27:49 +04:00
return ;
}
2014-01-05 20:07:11 +04:00
atBottom = scrollableWrap . scrollTop >= scrollableWrap . scrollHeight - scrollableWrap . clientHeight ;
if ( ! moreNotified && scrollableWrap . scrollTop <= 300 ) {
moreNotified = true ;
2014-03-24 16:13:34 +04:00
$scope . $emit ( 'history_need_more' ) ;
2014-01-05 20:07:11 +04:00
}
2014-06-13 01:46:46 +04:00
else if ( ! lessNotified && scrollableWrap . scrollTop >= scrollableWrap . scrollHeight - scrollableWrap . clientHeight - 300 ) {
lessNotified = true ;
$scope . $emit ( 'history_need_less' ) ;
}
2014-01-05 20:07:11 +04:00
} ) ;
2014-01-09 16:13:12 +04:00
function updateSizes ( heightOnly ) {
2014-07-02 23:47:28 +04:00
if ( ! element . is ( ':visible' ) && ! $ ( element [ 0 ] . parentNode . parentNode ) . is ( ':visible' ) ) {
2014-03-25 18:39:17 +04:00
return ;
}
2014-03-05 18:34:56 +01:00
if ( $ ( sendFormWrap ) . is ( ':visible' ) ) {
$ ( sendFormWrap ) . css ( {
height : $ ( sendForm ) . height ( )
} ) ;
}
2014-02-10 21:25:21 +04:00
2014-03-07 16:34:25 +01:00
if ( ! headWrap || ! headWrap . offsetHeight ) {
headWrap = $ ( '.tg_page_head' ) [ 0 ] ;
}
if ( ! footer || ! footer . offsetHeight ) {
footer = $ ( '.im_page_footer' ) [ 0 ] ;
}
2014-03-05 18:34:56 +01:00
var historyH = $ ( $window ) . height ( ) - panelWrap . offsetHeight - bottomPanelWrap . offsetHeight - ( headWrap ? headWrap . offsetHeight : 44 ) - footer . offsetHeight ;
2014-01-05 20:07:11 +04:00
$ ( historyWrap ) . css ( {
2014-02-10 21:25:21 +04:00
height : historyH
2014-01-05 20:07:11 +04:00
} ) ;
2014-01-09 16:13:12 +04:00
2014-03-20 21:56:03 +04:00
updateBottomizer ( ) ;
2014-03-24 20:50:43 +04:00
if ( heightOnly === true ) return ;
2014-01-05 20:07:11 +04:00
if ( atBottom ) {
onContentLoaded ( function ( ) {
2014-01-17 13:27:49 +04:00
scrollableWrap . scrollTop = scrollableWrap . scrollHeight ;
updateScroller ( ) ;
2014-01-05 20:07:11 +04:00
} ) ;
}
updateScroller ( 100 ) ;
}
2014-03-20 21:56:03 +04:00
function updateBottomizer ( ) {
$ ( historyMessagesEl ) . css ( { marginTop : 0 } ) ;
2014-06-30 17:30:42 +04:00
var marginTop = scrollableWrap . offsetHeight - historyMessagesEl . offsetHeight - 20 - 49 ;
if ( historyMessagesEl . offsetHeight > 0 && marginTop > 0 ) {
$ ( historyMessagesEl ) . css ( { marginTop : marginTop } ) ;
2014-03-20 21:56:03 +04:00
}
$ ( historyWrap ) . nanoScroller ( ) ;
}
2014-01-05 20:07:11 +04:00
$ ( $window ) . on ( 'resize' , updateSizes ) ;
2014-03-04 23:52:26 +01:00
updateSizes ( ) ;
2014-01-05 20:07:11 +04:00
onContentLoaded ( updateSizes ) ;
}
} )
2014-06-20 19:34:14 +04:00
. directive ( 'mySendForm' , function ( $timeout , $modalStack , Storage , ErrorService ) {
2014-01-05 20:07:11 +04:00
return {
link : link ,
scope : {
draftMessage : '='
}
} ;
2014-03-24 16:13:34 +04:00
function link ( $scope , element , attrs ) {
2014-01-05 20:07:11 +04:00
var messageField = $ ( 'textarea' , element ) [ 0 ] ,
2014-01-31 19:05:47 +04:00
fileSelects = $ ( 'input' , element ) ,
2014-01-05 20:07:11 +04:00
dropbox = $ ( '.im_send_dropbox_wrap' , element ) [ 0 ] ,
emojiButton = $ ( '.im_emoji_btn' , element ) [ 0 ] ,
editorElement = messageField ,
dragStarted , dragTimeout ,
2014-02-03 22:39:29 +04:00
emojiArea = $ ( messageField ) . emojiarea ( { button : emojiButton , norealTime : true } ) ,
2014-01-27 21:47:04 +04:00
emojiMenu = $ ( '.emoji-menu' , element ) [ 0 ] ,
2014-06-25 19:21:00 +04:00
submitBtn = $ ( '.im_submit' , element ) [ 0 ] ,
2014-01-05 20:07:11 +04:00
richTextarea = $ ( '.emoji-wysiwyg-editor' , element ) [ 0 ] ;
if ( richTextarea ) {
editorElement = richTextarea ;
$ ( richTextarea ) . addClass ( 'form-control' ) ;
$ ( richTextarea ) . attr ( 'placeholder' , $ ( messageField ) . attr ( 'placeholder' ) ) ;
2014-01-31 23:05:24 +04:00
2014-02-03 22:39:29 +04:00
var updatePromise ;
2014-06-25 19:21:00 +04:00
$ ( richTextarea )
. on ( 'DOMNodeInserted' , onPastedImageEvent )
. on ( 'keyup' , function ( e ) {
updateHeight ( ) ;
2014-01-05 20:07:11 +04:00
2014-06-25 19:21:00 +04:00
$scope . draftMessage . text = richTextarea . innerText ;
2014-01-05 20:07:11 +04:00
2014-06-25 19:21:00 +04:00
$timeout . cancel ( updatePromise ) ;
updatePromise = $timeout ( updateValue , 1000 ) ;
} ) ;
2014-02-03 22:39:29 +04:00
}
2014-01-05 20:07:11 +04:00
2014-06-19 20:03:00 +04:00
// Head is sometimes slower
$timeout ( function ( ) {
fileSelects
. add ( '.im_head_attach input' )
. on ( 'change' , function ( ) {
var self = this ;
$scope . $apply ( function ( ) {
$scope . draftMessage . files = Array . prototype . slice . call ( self . files ) ;
$scope . draftMessage . isMedia = $ ( self ) . hasClass ( 'im_media_attach_input' ) ;
setTimeout ( function ( ) {
try {
self . value = '' ;
} catch ( e ) { } ;
} , 1000 ) ;
} ) ;
} ) ;
} , 1000 ) ;
2014-01-05 20:07:11 +04:00
2014-02-04 23:55:20 +04:00
var sendOnEnter = true ,
updateSendSettings = function ( ) {
2014-06-20 19:34:14 +04:00
Storage . get ( 'send_ctrlenter' ) . then ( function ( sendOnCtrl ) {
2014-02-04 23:55:20 +04:00
sendOnEnter = ! sendOnCtrl ;
} ) ;
} ;
2014-03-24 16:13:34 +04:00
$scope . $on ( 'settings_changed' , updateSendSettings ) ;
2014-02-04 23:55:20 +04:00
updateSendSettings ( ) ;
2014-01-05 20:07:11 +04:00
$ ( editorElement ) . on ( 'keydown' , function ( e ) {
2014-02-10 21:25:21 +04:00
if ( richTextarea ) {
updateHeight ( ) ;
}
2014-02-03 22:39:29 +04:00
if ( e . keyCode == 13 ) {
var submit = false ;
if ( sendOnEnter && ! e . shiftKey ) {
submit = true ;
} else if ( ! sendOnEnter && ( e . ctrlKey || e . metaKey ) ) {
submit = true ;
}
if ( submit ) {
$ ( element ) . trigger ( 'submit' ) ;
2014-05-30 18:43:33 +04:00
$ ( element ) . trigger ( 'message_send' ) ;
2014-06-13 01:46:46 +04:00
resetAfterSubmit ( ) ;
2014-02-03 22:39:29 +04:00
return cancelEvent ( e ) ;
}
2014-01-05 20:07:11 +04:00
}
2014-02-03 22:39:29 +04:00
} ) ;
2014-01-05 20:07:11 +04:00
2014-06-27 18:14:54 +04:00
$ ( submitBtn ) . on ( 'mousedown touchstart' , function ( e ) {
2014-06-04 13:42:01 +04:00
$ ( element ) . trigger ( 'submit' ) ;
$ ( element ) . trigger ( 'message_send' ) ;
2014-06-13 01:46:46 +04:00
resetAfterSubmit ( ) ;
2014-06-04 13:42:01 +04:00
return cancelEvent ( e ) ;
} ) ;
2014-06-13 01:46:46 +04:00
var lastTyping = 0 ,
lastLength ;
2014-02-03 22:39:29 +04:00
$ ( editorElement ) . on ( 'keyup' , function ( e ) {
2014-06-13 01:46:46 +04:00
var now = tsNow ( ) ,
2014-06-30 17:30:42 +04:00
length = ( editorElement [ richTextarea ? 'textContent' : 'value' ] ) . length ;
2014-06-13 01:46:46 +04:00
if ( now - lastTyping > 5000 && length != lastLength ) {
lastTyping = now ;
lastLength = length ;
$scope . $emit ( 'ui_typing' ) ;
2014-01-05 20:07:11 +04:00
}
} ) ;
2014-06-13 01:46:46 +04:00
function resetAfterSubmit ( ) {
lastTyping = 0 ;
lastLength = 0 ;
} ;
2014-02-03 22:39:29 +04:00
function updateField ( ) {
if ( richTextarea ) {
$timeout . cancel ( updatePromise ) ;
2014-03-24 16:13:34 +04:00
var html = $ ( '<div>' ) . text ( $scope . draftMessage . text || '' ) . html ( ) ;
2014-02-03 22:39:29 +04:00
html = html . replace ( /\n/g , '<br/>' ) ;
$ ( richTextarea ) . html ( html ) ;
updateHeight ( ) ;
}
2014-01-05 20:07:11 +04:00
}
2014-02-03 22:39:29 +04:00
function updateValue ( ) {
if ( richTextarea ) {
$ ( richTextarea ) . trigger ( 'change' ) ;
updateHeight ( ) ;
}
2014-01-24 19:04:31 +04:00
}
2014-01-05 20:07:11 +04:00
2014-02-03 22:56:30 +04:00
var height = richTextarea . offsetHeight ;
2014-02-03 22:39:29 +04:00
function updateHeight ( ) {
2014-02-03 22:56:30 +04:00
var newHeight = richTextarea . offsetHeight ;
2014-02-03 22:39:29 +04:00
if ( height != newHeight ) {
height = newHeight ;
2014-03-24 16:13:34 +04:00
$scope . $emit ( 'ui_editor_resize' ) ;
2014-02-03 22:39:29 +04:00
}
} ;
2014-06-03 20:54:37 +04:00
function onKeyDown ( e ) {
2014-06-05 15:40:52 +04:00
if ( e . keyCode == 9 && ! e . shiftKey && ! e . ctrlKey && ! e . metaKey && ! $modalStack . getTop ( ) ) { // TAB
2014-06-03 20:54:37 +04:00
editorElement . focus ( ) ;
return cancelEvent ( e ) ;
}
}
$ ( document ) . on ( 'keydown' , onKeyDown ) ;
2014-01-05 20:07:11 +04:00
$ ( 'body' ) . on ( 'dragenter dragleave dragover drop' , onDragDropEvent ) ;
2014-03-09 15:11:41 +01:00
$ ( document ) . on ( 'paste' , onPasteEvent ) ;
2014-01-05 20:07:11 +04:00
2014-06-17 22:02:35 +04:00
if ( ! Config . Navigator . touch ) {
2014-06-02 19:31:40 +04:00
$scope . $on ( 'ui_peer_change' , focusField ) ;
$scope . $on ( 'ui_history_focus' , focusField ) ;
$scope . $on ( 'ui_history_change' , focusField ) ;
}
2014-03-24 16:13:34 +04:00
$scope . $on ( 'ui_message_send' , focusField ) ;
2014-06-02 19:31:40 +04:00
2014-03-24 16:13:34 +04:00
$scope . $on ( 'ui_peer_draft' , updateField ) ;
$scope . $on ( 'ui_message_before_send' , updateValue ) ;
2014-02-03 22:39:29 +04:00
2014-01-05 20:07:11 +04:00
function focusField ( ) {
onContentLoaded ( function ( ) {
2014-02-03 22:39:29 +04:00
editorElement . focus ( ) ;
2014-01-05 20:07:11 +04:00
} ) ;
}
2014-03-10 03:20:55 -03:00
function onPastedImageEvent ( e ) {
var element = e && e . target ;
var src ;
if ( element && ( src = element . src ) && src . indexOf ( 'data' ) === 0 ) {
element . parentNode . removeChild ( element ) ;
src = src . substr ( 5 ) . split ( ';' ) ;
var contentType = src [ 0 ] ;
var base64 = atob ( src [ 1 ] . split ( ',' ) [ 1 ] ) ;
var array = new Uint8Array ( base64 . length ) ;
for ( var i = 0 ; i < base64 . length ; i ++ ) {
array [ i ] = base64 . charCodeAt ( i ) ;
}
var blob = new Blob ( [ array ] , { type : contentType } ) ;
2014-04-08 13:07:38 +02:00
ErrorService . confirm ( { type : 'FILE_CLIPBOARD_PASTE' } ) . then ( function ( ) {
$scope . draftMessage . files = [ blob ] ;
$scope . draftMessage . isMedia = true ;
} ) ;
2014-03-10 03:20:55 -03:00
}
} ;
2014-03-09 15:11:41 +01:00
function onPasteEvent ( e ) {
var cData = ( e . originalEvent || e ) . clipboardData ,
items = cData && cData . items || [ ] ,
files = [ ] ,
i ;
for ( i = 0 ; i < items . length ; i ++ ) {
if ( items [ i ] . kind == 'file' ) {
files . push ( items [ i ] . getAsFile ( ) ) ;
}
}
2014-04-08 13:07:38 +02:00
if ( files . length > 0 ) {
ErrorService . confirm ( { type : 'FILES_CLIPBOARD_PASTE' , files : files } ) . then ( function ( ) {
2014-03-24 16:13:34 +04:00
$scope . draftMessage . files = files ;
$scope . draftMessage . isMedia = true ;
2014-03-09 15:11:41 +01:00
} ) ;
}
}
2014-01-05 20:07:11 +04:00
function onDragDropEvent ( e ) {
var dragStateChanged = false ;
if ( ! dragStarted || dragStarted == 1 ) {
dragStarted = checkDragEvent ( e ) ? 2 : 1 ;
dragStateChanged = true ;
}
if ( dragStarted == 2 ) {
if ( dragTimeout ) {
setTimeout ( function ( ) {
clearTimeout ( dragTimeout ) ;
dragTimeout = false ;
} , 0 ) ;
}
if ( e . type == 'dragenter' || e . type == 'dragover' ) {
if ( dragStateChanged ) {
$ ( dropbox )
. css ( { height : $ ( editorElement ) . height ( ) + 12 , width : $ ( editorElement ) . width ( ) + 12 } )
. show ( ) ;
}
} else {
if ( e . type == 'drop' ) {
2014-03-24 16:13:34 +04:00
$scope . $apply ( function ( ) {
$scope . draftMessage . files = Array . prototype . slice . call ( e . originalEvent . dataTransfer . files ) ;
$scope . draftMessage . isMedia = true ;
2014-01-05 20:07:11 +04:00
} ) ;
}
dragTimeout = setTimeout ( function ( ) {
$ ( dropbox ) . hide ( ) ;
dragStarted = false ;
dragTimeout = false ;
} , 300 ) ;
}
}
return cancelEvent ( e ) ;
} ;
2014-06-25 19:21:00 +04:00
$scope . $on ( '$destroy' , function cleanup ( ) {
$ ( 'body' ) . off ( 'dragenter dragleave dragover drop' , onDragDropEvent ) ;
$ ( document ) . off ( 'paste' , onPasteEvent ) ;
$ ( document ) . off ( 'keydown' , onKeyDown ) ;
$ ( submitBtn ) . off ( 'mousedown' )
fileSelects . off ( 'change' ) ;
if ( richTextarea ) {
$ ( richTextarea ) . off ( 'DOMNodeInserted keyup' , onPastedImageEvent ) ;
}
$ ( editorElement ) . off ( 'keydown' ) ;
} ) ;
if ( ! Config . Navigator . touch ) {
focusField ( ) ;
}
}
2014-01-05 20:07:11 +04:00
} )
. directive ( 'myLoadThumb' , function ( MtpApiFileManager ) {
return {
link : link ,
scope : {
thumb : '='
}
} ;
2014-03-24 16:13:34 +04:00
function link ( $scope , element , attrs ) {
2014-01-23 17:49:24 +04:00
var counter = 0 ;
2014-02-03 22:39:29 +04:00
2014-03-03 20:48:42 +01:00
var cachedSrc = MtpApiFileManager . getCachedFile (
2014-03-24 16:13:34 +04:00
$scope . thumb &&
$scope . thumb . location &&
! $scope . thumb . location . empty &&
$scope . thumb . location
2014-03-03 20:48:42 +01:00
) ;
2014-02-03 22:39:29 +04:00
if ( cachedSrc ) {
element . attr ( 'src' , cachedSrc ) ;
}
2014-03-30 00:56:10 +04:00
if ( $scope . thumb && $scope . thumb . width && $scope . thumb . height ) {
element . attr ( 'width' , $scope . thumb . width ) ;
element . attr ( 'height' , $scope . thumb . height ) ;
}
2014-02-03 22:39:29 +04:00
2014-03-30 00:56:10 +04:00
var stopWatching = $scope . $watchCollection ( 'thumb.location' , function ( newLocation ) {
if ( $scope . thumb && $scope . thumb . width && $scope . thumb . height ) {
element . attr ( 'width' , $scope . thumb . width ) ;
element . attr ( 'height' , $scope . thumb . height ) ;
2014-06-02 19:31:40 +04:00
$scope . $emit ( 'ui_height' ) ;
2014-03-30 00:56:10 +04:00
}
2014-03-04 13:08:15 +01:00
// console.log('new loc', newLocation, arguments);
2014-01-23 17:49:24 +04:00
var counterSaved = ++ counter ;
2014-03-03 20:48:42 +01:00
if ( ! newLocation || newLocation . empty ) {
2014-03-24 16:13:34 +04:00
element . attr ( 'src' , $scope . thumb && $scope . thumb . placeholder || 'img/blank.gif' ) ;
2014-03-30 00:56:10 +04:00
cleanup ( ) ;
2014-01-05 20:07:11 +04:00
return ;
}
2014-02-03 22:39:29 +04:00
var cachedSrc = MtpApiFileManager . getCachedFile ( newLocation ) ;
2014-01-23 17:49:24 +04:00
if ( cachedSrc ) {
element . attr ( 'src' , cachedSrc ) ;
2014-03-30 00:56:10 +04:00
cleanup ( ) ;
2014-01-23 17:49:24 +04:00
return ;
}
2014-03-02 14:37:24 +01:00
if ( ! element . attr ( 'src' ) ) {
2014-03-24 16:13:34 +04:00
element . attr ( 'src' , $scope . thumb . placeholder || 'img/blank.gif' ) ;
2014-03-02 14:37:24 +01:00
}
2014-01-23 17:49:24 +04:00
2014-03-31 19:06:05 +04:00
MtpApiFileManager . downloadSmallFile ( $scope . thumb . location ) . then ( function ( url ) {
2014-01-23 17:49:24 +04:00
if ( counterSaved == counter ) {
element . attr ( 'src' , url ) ;
2014-03-30 00:56:10 +04:00
cleanup ( ) ;
2014-01-23 17:49:24 +04:00
}
2014-01-05 20:07:11 +04:00
} , function ( e ) {
2014-03-31 19:06:05 +04:00
console . log ( 'Download image failed' , e , $scope . thumb . location , element [ 0 ] ) ;
2014-01-23 17:49:24 +04:00
if ( counterSaved == counter ) {
2014-03-24 16:13:34 +04:00
element . attr ( 'src' , $scope . thumb . placeholder || 'img/blank.gif' ) ;
2014-03-30 00:56:10 +04:00
cleanup ( ) ;
2014-01-23 17:49:24 +04:00
}
2014-01-05 20:07:11 +04:00
} ) ;
} )
2014-03-30 00:56:10 +04:00
var cleanup = angular . noop ;
// function () {
// setTimeout(function () {
// $scope.$destroy()
// stopWatching();
// }, 0);
// };
2014-01-05 20:07:11 +04:00
}
} )
. directive ( 'myLoadFullPhoto' , function ( MtpApiFileManager ) {
return {
link : link ,
transclude : true ,
2014-04-17 19:06:00 -04:00
templateUrl : 'partials/full_photo.html' ,
2014-01-05 20:07:11 +04:00
scope : {
fullPhoto : '=' ,
thumbLocation : '='
}
} ;
2014-03-24 16:13:34 +04:00
function link ( $scope , element , attrs ) {
2014-01-05 20:07:11 +04:00
2014-04-09 22:31:19 +02:00
var imgElement = $ ( 'img' , element ) [ 0 ] ,
resizeElements = $ ( '.img_fullsize_with_progress_wrap' , element )
. add ( '.img_fullsize_progress_wrap' , element )
. add ( $ ( imgElement ) ) ,
resize = function ( ) {
resizeElements . css ( { width : $scope . fullPhoto . width , height : $scope . fullPhoto . height } ) ;
$scope . $emit ( 'ui_height' ) ;
} ;
2014-01-05 20:07:11 +04:00
2014-04-09 22:31:19 +02:00
var jump = 0 ;
$scope . $watchCollection ( 'fullPhoto.location' , function ( ) {
var cachedSrc = MtpApiFileManager . getCachedFile ( $scope . thumbLocation ) ,
curJump = ++ jump ;
2014-01-21 21:56:30 +04:00
2014-04-09 22:31:19 +02:00
if ( cachedSrc ) {
imgElement . src = cachedSrc ;
resize ( ) ;
} else {
imgElement . src = '' ;
}
2014-01-05 20:07:11 +04:00
2014-04-09 22:31:19 +02:00
if ( ! $scope . fullPhoto . location ) {
return ;
}
2014-01-05 20:07:11 +04:00
2014-04-09 22:31:19 +02:00
var apiPromise ;
if ( $scope . fullPhoto . size ) {
var inputLocation = {
_ : 'inputFileLocation' ,
volume _id : $scope . fullPhoto . location . volume _id ,
local _id : $scope . fullPhoto . location . local _id ,
secret : $scope . fullPhoto . location . secret
} ;
apiPromise = MtpApiFileManager . downloadFile ( $scope . fullPhoto . location . dc _id , inputLocation , $scope . fullPhoto . size ) ;
} else {
apiPromise = MtpApiFileManager . downloadSmallFile ( $scope . fullPhoto . location ) ;
}
2014-01-05 20:07:11 +04:00
2014-04-09 22:31:19 +02:00
$scope . progress = { enabled : true , percent : 0 } ;
2014-01-05 20:07:11 +04:00
2014-04-09 22:31:19 +02:00
apiPromise . then ( function ( url ) {
if ( curJump == jump ) {
$scope . progress . enabled = false ;
imgElement . src = url ;
resize ( ) ;
}
} , function ( e ) {
console . log ( 'Download image failed' , e , $scope . fullPhoto . location ) ;
$scope . progress . enabled = false ;
2014-01-05 20:07:11 +04:00
2014-04-09 22:31:19 +02:00
if ( e && e . type == 'FS_BROWSER_UNSUPPORTED' ) {
$scope . error = { html : 'Your browser doesn\'t support <a href="https://developer.mozilla.org/en-US/docs/Web/API/LocalFileSystem" target="_blank">LocalFileSystem</a> feature which is needed to display this image.<br/>Please, install <a href="http://google.com/chrome" target="_blank">Google Chrome</a> or use <a href="https://telegram.org/" target="_blank">mobile app</a> instead.' } ;
} else {
$scope . error = { text : 'Download failed' , error : e } ;
}
} , function ( progress ) {
$scope . progress . percent = Math . max ( 1 , Math . floor ( 100 * progress . done / progress . total ) ) ;
} ) ;
} )
2014-01-05 20:07:11 +04:00
2014-04-09 22:31:19 +02:00
resize ( ) ;
2014-01-05 20:07:11 +04:00
}
} )
. directive ( 'myLoadVideo' , function ( $sce , MtpApiFileManager ) {
return {
link : link ,
transclude : true ,
2014-04-17 19:06:00 -04:00
templateUrl : 'partials/full_video.html' ,
2014-01-05 20:07:11 +04:00
scope : {
video : '='
}
} ;
2014-03-24 16:13:34 +04:00
function link ( $scope , element , attrs ) {
2014-01-05 20:07:11 +04:00
2014-03-24 16:13:34 +04:00
$scope . progress = { enabled : true , percent : 1 } ;
$scope . player = { } ;
2014-01-05 20:07:11 +04:00
var inputLocation = {
_ : 'inputVideoFileLocation' ,
2014-03-24 16:13:34 +04:00
id : $scope . video . id ,
access _hash : $scope . video . access _hash
2014-01-05 20:07:11 +04:00
} ;
2014-01-06 23:52:18 +04:00
var hasQt = false , i ;
2014-03-07 18:39:59 +01:00
if ( navigator . plugins ) {
for ( i = 0 ; i < navigator . plugins . length ; i ++ ) {
if ( navigator . plugins [ i ] . name . indexOf ( 'QuickTime' ) >= 0 ) {
hasQt = true ;
}
}
}
2014-01-06 23:52:18 +04:00
2014-05-08 17:06:25 +01:00
var downloadPromise = MtpApiFileManager . downloadFile ( $scope . video . dc _id , inputLocation , $scope . video . size , { mime : 'video/mp4' } ) ;
2014-04-02 14:16:09 +04:00
downloadPromise . then ( function ( url ) {
2014-03-24 16:13:34 +04:00
$scope . progress . enabled = false ;
// $scope.progress = {enabled: true, percent: 50};
$scope . player . hasQuicktime = hasQt ;
$scope . player . quicktime = false ;
$scope . player . src = $sce . trustAsResourceUrl ( url ) ;
2014-06-02 19:31:40 +04:00
$scope . $emit ( 'ui_height' ) ;
2014-01-05 20:07:11 +04:00
} , function ( e ) {
2014-03-24 16:13:34 +04:00
console . log ( 'Download video failed' , e , $scope . video ) ;
$scope . progress . enabled = false ;
$scope . player . src = '' ;
2014-01-21 21:56:30 +04:00
if ( e && e . type == 'FS_BROWSER_UNSUPPORTED' ) {
2014-03-24 16:13:34 +04:00
$scope . error = { html : 'Your browser doesn\'t support <a href="https://developer.mozilla.org/en-US/docs/Web/API/LocalFileSystem" target="_blank">LocalFileSystem</a> feature which is needed to play this video.<br/>Please, install <a href="http://google.com/chrome" target="_blank">Google Chrome</a> or use <a href="https://telegram.org/" target="_blank">mobile app</a> instead.' } ;
2014-01-21 21:56:30 +04:00
} else {
2014-03-24 16:13:34 +04:00
$scope . error = { text : 'Video download failed' , error : e } ;
2014-01-21 21:56:30 +04:00
}
2014-01-05 20:07:11 +04:00
} , function ( progress ) {
2014-03-24 16:13:34 +04:00
$scope . progress . percent = Math . max ( 1 , Math . floor ( 100 * progress . done / progress . total ) ) ;
2014-01-05 20:07:11 +04:00
} ) ;
2014-04-01 21:27:30 +04:00
2014-06-02 19:31:40 +04:00
$scope . $emit ( 'ui_height' ) ;
2014-04-01 21:27:30 +04:00
$scope . $on ( '$destroy' , function ( ) {
2014-04-02 14:16:09 +04:00
downloadPromise . cancel ( ) ;
2014-04-01 21:27:30 +04:00
} ) ;
2014-01-05 20:07:11 +04:00
}
} )
2014-04-17 19:06:00 -04:00
. directive ( 'myLoadGif' , function ( MtpApiFileManager ) {
return {
link : link ,
templateUrl : 'partials/full_gif.html' ,
scope : {
document : '='
}
} ;
function link ( $scope , element , attrs ) {
var downloadPromise = false ,
inputFileLocation = {
_ : 'inputDocumentFileLocation' ,
id : $scope . document . id ,
access _hash : $scope . document . access _hash
} ;
$scope . isActive = false ;
$scope . document . url = MtpApiFileManager . getCachedFile ( inputFileLocation ) ;
/*return $scope.document.progress = {enabled: true, percent: 30, total: $scope.document.size};*/
$scope . toggle = function ( ) {
if ( $scope . document . url ) {
$scope . isActive = ! $scope . isActive ;
$scope . $emit ( 'ui_height' ) ;
return ;
}
if ( downloadPromise ) {
downloadPromise . cancel ( ) ;
downloadPromise = false ;
return ;
}
$scope . document . progress = { enabled : true , percent : 1 , total : $scope . document . size } ;
downloadPromise = MtpApiFileManager . downloadFile (
$scope . document . dc _id ,
inputFileLocation ,
$scope . document . size ,
null ,
{ mime : $scope . document . mime _type }
) ;
downloadPromise . then ( function ( url ) {
$scope . document . url = url ;
$scope . isActive = true ;
delete $scope . document . progress ;
console . log ( 'file save done' ) ;
$scope . $emit ( 'ui_height' ) ;
} , function ( ) {
$scope . document . progress . enabled = false ;
} , function ( progress ) {
console . log ( 'dl progress' , progress ) ;
$scope . document . progress . done = progress . done ;
$scope . document . progress . percent = Math . max ( 1 , Math . floor ( 100 * progress . done / progress . total ) ) ;
} )
}
}
} )
2014-01-05 20:07:11 +04:00
. directive ( 'myMapPoint' , function ( ExternalResourcesManager ) {
return {
link : link ,
scope : {
point : '='
}
} ;
2014-03-24 16:13:34 +04:00
function link ( $scope , element , attrs ) {
2014-01-05 20:07:11 +04:00
var apiKey = 'AIzaSyC32ij28dCa0YzEV_HqbWfIwTZQql-RNS0' ;
2014-03-24 16:13:34 +04:00
var src = 'https://maps.googleapis.com/maps/api/staticmap?sensor=false¢er=' + $scope . point [ 'lat' ] + ',' + $scope . point [ 'long' ] + '&zoom=13&size=200x100&scale=2&key=' + apiKey ;
2014-01-05 20:07:11 +04:00
ExternalResourcesManager . downloadImage ( src ) . then ( function ( url ) {
element . append ( '<img src="' + url + '" width="200" height="100"/>' ) ;
} ) ;
2014-03-24 16:13:34 +04:00
element . attr ( 'href' , 'https://maps.google.com/?q=' + $scope . point [ 'lat' ] + ',' + $scope . point [ 'long' ] ) ;
2014-01-05 20:07:11 +04:00
element . attr ( 'target' , '_blank' ) ;
}
} )
2014-01-06 23:52:18 +04:00
2014-03-25 18:39:17 +04:00
. directive ( 'myLoadingDots' , function ( $interval ) {
2014-01-06 23:52:18 +04:00
return {
link : link ,
} ;
2014-03-24 16:13:34 +04:00
function link ( $scope , element , attrs ) {
2014-04-01 15:48:26 +04:00
element . html ( isAnimationSupported ( element [ 0 ] )
? '<div class="loading_dots"><span></span><span></span><span></span></div>'
: '...'
) ;
}
var animationSupported ;
function isAnimationSupported ( el ) {
if ( animationSupported === undefined ) {
animationSupported = el . style . animationName !== undefined ;
if ( animationSupported === false ) {
var domPrefixes = 'Webkit Moz O ms Khtml' . split ( ' ' ) , i ;
for ( i = 0 ; i < domPrefixes . length ; i ++ ) {
if ( el . style [ domPrefixes [ i ] + 'AnimationName' ] !== undefined ) {
animationSupported = true ;
break ;
}
}
}
}
return animationSupported ;
2014-01-06 23:52:18 +04:00
}
} )
2014-02-05 23:44:31 +04:00
. directive ( 'myAudioAutoplay' , function ( ) {
return {
link : link ,
scope : {
audio : '='
}
} ;
2014-03-24 16:13:34 +04:00
function link ( $scope , element , attrs ) {
$scope . $watch ( 'audio.autoplay' , function ( autoplay ) {
2014-02-05 23:44:31 +04:00
if ( autoplay ) {
element . autoplay = true ;
element [ 0 ] . play ( ) ;
} else {
element . autoplay = false ;
}
} ) ;
}
} )
2014-02-12 19:10:05 +04:00
. directive ( 'myFocused' , function ( ) {
return {
2014-03-24 16:13:34 +04:00
link : function ( $scope , element , attrs ) {
2014-06-17 22:02:35 +04:00
if ( Config . Navigator . touch ) {
2014-06-02 19:31:40 +04:00
return false ;
}
2014-02-12 19:10:05 +04:00
setTimeout ( function ( ) {
element [ 0 ] . focus ( ) ;
} , 100 ) ;
}
} ;
2014-02-23 19:53:14 -03:00
} )
2014-04-27 16:25:04 +08:00
. directive ( 'myFocusOn' , function ( ) {
return {
link : function ( $scope , element , attrs ) {
$scope . $on ( attrs . myFocusOn , function ( ) {
2014-06-17 22:02:35 +04:00
if ( Config . Navigator . touch ) {
2014-06-02 19:31:40 +04:00
return false ;
}
2014-04-27 16:25:04 +08:00
onContentLoaded ( function ( ) {
element [ 0 ] . focus ( ) ;
} ) ;
} ) ;
}
} ;
} )
2014-03-02 14:37:24 +01:00
. directive ( 'myFileUpload' , function ( ) {
2014-02-23 19:53:14 -03:00
return {
link : link
} ;
2014-03-24 16:13:34 +04:00
function link ( $scope , element , attrs ) {
2014-03-02 14:37:24 +01:00
element . on ( 'change' , function ( ) {
2014-02-23 19:53:14 -03:00
var self = this ;
2014-03-24 16:13:34 +04:00
$scope . $apply ( function ( ) {
$scope . photo . file = self . files [ 0 ] ;
2014-02-23 19:53:14 -03:00
setTimeout ( function ( ) {
try {
self . value = '' ;
} catch ( e ) { } ;
} , 1000 ) ;
} ) ;
} ) ;
} ;
2014-03-21 13:36:51 +04:00
} )
. directive ( 'myModalWidth' , function ( ) {
return {
link : link
} ;
2014-03-24 16:13:34 +04:00
function link ( $scope , element , attrs ) {
2014-03-21 13:36:51 +04:00
attrs . $observe ( 'myModalWidth' , function ( newW ) {
2014-06-17 22:02:35 +04:00
$ ( element [ 0 ] . parentNode . parentNode ) . css ( { width : parseInt ( newW ) + ( Config . Navigator . mobile ? 0 : 36 ) } ) ;
2014-03-21 13:36:51 +04:00
} ) ;
} ;
2014-03-24 16:13:34 +04:00
} )
2014-04-09 22:31:19 +02:00
. directive ( 'myModalNav' , function ( ) {
return {
link : link
} ;
function link ( $scope , element , attrs ) {
var onKeyDown = function ( event ) {
var target = event . target ;
if ( target && ( target . tagName == 'INPUT' || target . tagName == 'TEXTAREA' ) ) {
2014-06-27 21:52:24 +04:00
return ;
2014-04-09 22:31:19 +02:00
}
switch ( event . keyCode ) {
case 39 : // right
case 32 : // space
case 34 : // pg down
case 40 : // down
$scope . $eval ( attrs . next ) ;
break ;
case 37 : // left
case 33 : // pg up
case 38 : // up
$scope . $eval ( attrs . prev ) ;
break ;
}
} ;
$ ( document ) . on ( 'keydown' , onKeyDown ) ;
$scope . $on ( '$destroy' , function ( ) {
$ ( document ) . off ( 'keydown' , onKeyDown ) ;
} ) ;
} ;
} )
2014-04-29 21:51:49 +08:00
. directive ( 'myCustomBackground' , function ( ) {
return {
link : link
} ;
function link ( $scope , element , attrs ) {
console . log ( dT ( ) , 'bg' , attrs . myCustomBackground ) ;
$ ( 'html' ) . css ( { background : attrs . myCustomBackground } ) ;
$scope . $on ( '$destroy' , function ( ) {
$ ( 'html' ) . css ( { background : '' } ) ;
} ) ;
} ;
} )
2014-04-30 11:09:40 +08:00
. directive ( 'myInfiniteScroller' , function ( ) {
return {
link : link ,
scope : true
} ;
function link ( $scope , element , attrs ) {
2014-07-02 20:36:17 +04:00
var scrollableWrap = $ ( '.nano-content' , element ) [ 0 ] ,
2014-04-30 11:09:40 +08:00
moreNotified = false ;
$ ( scrollableWrap ) . on ( 'scroll' , function ( e ) {
2014-06-30 17:30:42 +04:00
if ( ! element . is ( ':visible' ) ) return ;
2014-04-30 11:09:40 +08:00
if ( ! moreNotified &&
scrollableWrap . scrollTop >= scrollableWrap . scrollHeight - scrollableWrap . clientHeight - 300 ) {
moreNotified = true ;
$scope . $apply ( function ( ) {
$scope . slice . limit += ( $scope . slice . limitDelta || 20 ) ;
} ) ;
onContentLoaded ( function ( ) {
moreNotified = false ;
$ ( element ) . nanoScroller ( ) ;
} ) ;
}
} ) ;
} ;
} )
2014-03-24 16:13:34 +04:00
. directive ( 'myModalPosition' , function ( $window , $timeout ) {
return {
link : link
} ;
function link ( $scope , element , attrs ) {
var updateMargin = function ( ) {
2014-06-17 22:02:35 +04:00
if ( Config . Navigator . mobile &&
$ ( element [ 0 ] . parentNode . parentNode . parentNode ) . hasClass ( 'page_modal' ) ) {
return ;
}
2014-03-24 16:13:34 +04:00
var height = element [ 0 ] . parentNode . offsetHeight ,
contHeight = element [ 0 ] . parentNode . parentNode . parentNode . offsetHeight ;
if ( height < contHeight ) {
$ ( element [ 0 ] . parentNode ) . css ( 'marginTop' , ( contHeight - height ) / 2 ) ;
} else {
$ ( element [ 0 ] . parentNode ) . css ( 'marginTop' , '' ) ;
}
2014-04-09 22:31:19 +02:00
if ( attrs . animation != 'no' ) {
$timeout ( function ( ) {
$ ( element [ 0 ] . parentNode ) . addClass ( 'modal-content-animated' ) ;
} , 300 ) ;
}
2014-03-24 16:13:34 +04:00
} ;
onContentLoaded ( updateMargin ) ;
$ ( $window ) . on ( 'resize' , updateMargin ) ;
$scope . $on ( 'ui_height' , function ( ) {
onContentLoaded ( updateMargin ) ;
} ) ;
} ;
2014-03-25 15:36:17 +04:00
} )
. directive ( 'myVerticalPosition' , function ( $window , $timeout ) {
return {
link : link
} ;
function link ( $scope , element , attrs ) {
2014-06-17 22:02:35 +04:00
var usePadding = attrs . padding === 'true' ,
prevMargin = 0 ;
2014-04-08 16:38:24 +02:00
2014-03-25 15:36:17 +04:00
var updateMargin = function ( ) {
var height = element [ 0 ] . offsetHeight ,
2014-06-17 22:02:35 +04:00
fullHeight = height - ( height && usePadding ? 2 * prevMargin : 0 ) ,
2014-03-25 15:36:17 +04:00
contHeight = $ ( $window ) . height ( ) ,
2014-03-25 20:33:23 +04:00
ratio = attrs . myVerticalPosition && parseFloat ( attrs . myVerticalPosition ) || 0.5 ,
2014-06-17 22:02:35 +04:00
margin = fullHeight < contHeight ? parseInt ( ( contHeight - fullHeight ) * ratio ) : '' ,
styles = usePadding
? { paddingTop : margin , paddingBottom : margin }
: { marginTop : margin , marginBottom : margin } ;
2014-03-25 15:36:17 +04:00
2014-06-17 22:02:35 +04:00
element . css ( styles ) ;
element . addClass ( 'vertical-aligned' ) ;
2014-04-08 16:38:24 +02:00
if ( prevMargin !== margin ) {
$scope . $emit ( 'ui_height' ) ;
}
prevMargin = margin ;
2014-03-25 15:36:17 +04:00
} ;
onContentLoaded ( updateMargin ) ;
$ ( $window ) . on ( 'resize' , updateMargin ) ;
$scope . $on ( 'ui_height' , function ( ) {
onContentLoaded ( updateMargin ) ;
} ) ;
2014-04-08 16:38:24 +02:00
2014-03-25 15:36:17 +04:00
} ;
} )