56 changed files with 2567 additions and 1410 deletions
File diff suppressed because one or more lines are too long
@ -1,8 +1,8 @@
|
||||
<div> |
||||
<a ng-if="historyMessage.media.user_id > 0" class="im_message_contact_photo pull-left" my-peer-photolink="historyMessage.media.user_id" img-class="im_message_contact_photo" user-override="historyMessage.media"></a> |
||||
<div class="im_message_contact_name" ng-switch="historyMessage.media.user_id > 0"> |
||||
<a ng-switch-when="true" my-peer-link="historyMessage.media.user_id" user-override="historyMessage.media"></a> |
||||
<span ng-switch-default ng-bind-html="::historyMessage.media.rFullName"></span> |
||||
<div class="im_message_contact"> |
||||
<a ng-if="::media.user_id > 0" class="im_message_contact_photo pull-left" my-peer-photolink="media.user_id" img-class="im_message_contact_photo" user-override="media"></a> |
||||
<div class="im_message_contact_name" ng-switch="media.user_id > 0"> |
||||
<a ng-switch-when="true" my-peer-link="media.user_id" user-override="media"></a> |
||||
<span ng-switch-default ng-bind-html="::media.rFullName"></span> |
||||
</div> |
||||
<div class="im_message_contact_phone" ng-bind="::historyMessage.media.phone_number | phoneNumber"></div> |
||||
<div class="im_message_contact_phone" ng-bind="::media.phone_number | phoneNumber"></div> |
||||
</div> |
@ -1,21 +1,21 @@
|
||||
<div class="im_message_venue clearfix"> |
||||
|
||||
<a ng-href="{{::venue.mapUrl}}" target="_blank" class="im_message_venue_geopoint_wrap"> |
||||
<a ng-href="{{::media.mapUrl}}" target="_blank" class="im_message_venue_geopoint_wrap"> |
||||
<i class="icon icon-geo-point"></i> |
||||
<img |
||||
class="im_message_venue_geopoint_image" |
||||
my-geo-point-map="venue.geo" |
||||
my-geo-point-map="media.geo" |
||||
width="100" |
||||
height="100" |
||||
alt="[{{::'conversation_media_location' | i18n}} {{::venue.mapUrl}}]" |
||||
alt="[{{::'conversation_media_location' | i18n}} {{::media.mapUrl}}]" |
||||
/> |
||||
</a> |
||||
|
||||
<div class="im_message_venue_info"> |
||||
<div class="im_message_venue_title_wrap"> |
||||
<a ng-href="{{::venue.mapUrl}}" target="_blank" class="im_message_document_name" ng-bind="::venue.title"></a> |
||||
<a ng-href="{{::media.mapUrl}}" target="_blank" class="im_message_document_name" ng-bind="::media.title"></a> |
||||
</div> |
||||
<div class="im_message_venue_address" ng-bind="::venue.address"></div> |
||||
<div class="im_message_venue_address" ng-bind="::media.address"></div> |
||||
</div> |
||||
|
||||
</div> |
||||
|
@ -1,35 +0,0 @@
|
||||
<div class="im_message_video im_message_document_thumbed"> |
||||
<a class="im_message_video_thumb" ng-click="videoOpen()" ng-style="::{width: media.video.thumb.width + 'px'}"> |
||||
<span class="im_message_video_duration nocopy" data-content="{{::media.video.duration | duration}}"></span> |
||||
<i class="icon icon-videoplay"></i> |
||||
<img |
||||
class="im_message_video_thumb im_message_video_thumb_blurred" |
||||
my-load-thumb |
||||
thumb="media.video.thumb" |
||||
/> |
||||
</a> |
||||
|
||||
<div class="im_message_document_info"> |
||||
<div class="im_message_document_name_wrap"> |
||||
<span class="copyonly">[</span><span class="im_message_document_name" my-i18n="message_attach_video_video"></span><span class="copyonly"> <span ng-bind="::media.video.duration | duration"></span>]</span> |
||||
<span class="im_message_document_size" ng-if="!media.video.progress.enabled" ng-bind="::media.video.size | formatSize"></span> |
||||
<span class="im_message_document_size" ng-if="media.video.progress.enabled" ng-bind="media.video.progress | formatSizeProgress"></span> |
||||
</div> |
||||
<div class="im_message_document_actions noselect" ng-if="!media.video.progress.enabled"> |
||||
<a href="" ng-click="videoSave()" ng-switch="media.video.downloaded"> |
||||
<span class="nocopy" ng-switch-when="true" my-i18n="message_attach_video_save"></span> |
||||
<span class="nocopy" ng-switch-default my-i18n="message_attach_video_download"></span> |
||||
</a> |
||||
<a class="nocopy" href="" ng-click="videoOpen()" my-i18n="message_attach_video_play"></a> |
||||
</div> |
||||
<div class="clearfix im_message_cancelable_progress_wrap" ng-if="media.video.progress.enabled"> |
||||
<a class="im_message_media_progress_cancel pull-right nocopy" ng-click="media.video.progress.cancel()" my-i18n="modal_cancel"></a> |
||||
<div class="im_message_download_progress_wrap"> |
||||
<div class="progress tg_down_progress"> |
||||
<div class="progress-bar progress-bar-success" ng-style="{width: media.video.progress.percent + '%'}"></div> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
<div ng-if="::media.rCaption" class="im_message_video_caption" ng-bind-html="::media.rCaption"></div> |
@ -1,59 +1,59 @@
|
||||
<div ng-show="webpage._ == 'webPage'" class="im_message_webpage_wrap clearfix" ng-switch="webpage.type"> |
||||
<div ng-show="media.webpage._ == 'webPage'" class="im_message_webpage_wrap clearfix" ng-switch="media.webpage.type"> |
||||
<div ng-switch-when="photo" class="im_message_webpage_photo"> |
||||
<div class="im_message_webpage_title"> |
||||
<a href="{{webpage.url}}" target="_blank" ng-bind-html="webpage.rTitle"></a> |
||||
<a href="{{media.webpage.url}}" target="_blank" ng-bind-html="media.webpage.rTitle"></a> |
||||
</div> |
||||
<div ng-if="webpage.description.length" class="im_message_webpage_description" ng-bind-html="webpage.rDescription"></div> |
||||
<a class="im_message_photo_thumb" ng-click="openPhoto(webpage.photo.id, {w: webpage.id, m: messageId})" ng-style="::{width: webpage.photo.thumb.width + 'px'}" ng-mouseover="preloadPhoto(webpage.photo.id)"> |
||||
<div ng-if="media.webpage.description.length" class="im_message_webpage_description" ng-bind-html="media.webpage.rDescription"></div> |
||||
<a class="im_message_photo_thumb" ng-click="openPhoto(media.webpage.photo.id, {w: media.webpage.id, m: messageId})" ng-style="::{width: media.webpage.photo.thumb.width + 'px'}" ng-mouseover="preloadPhoto(media.webpage.photo.id)"> |
||||
<img |
||||
class="im_message_photo_thumb" |
||||
my-load-thumb |
||||
thumb="webpage.photo.thumb" |
||||
thumb="media.webpage.photo.thumb" |
||||
alt="[{{::'conversation_media_photo' | i18n}}]" |
||||
/> |
||||
</a> |
||||
</div> |
||||
<div ng-switch-when="video" class="im_message_webpage_video"> |
||||
<div class="im_message_webpage_site" ng-bind="webpage.site_name || webpage.display_url"></div> |
||||
<div class="im_message_webpage_site" ng-bind="media.webpage.site_name || media.webpage.display_url"></div> |
||||
<div class="im_message_webpage_title"> |
||||
<a ng-click="openEmbed($event)" href="{{webpage.url}}" target="_blank" ng-bind-html="webpage.rTitle"></a> |
||||
<a ng-click="openEmbed($event)" href="{{media.webpage.url}}" target="_blank" ng-bind-html="media.webpage.rTitle"></a> |
||||
</div> |
||||
<div ng-if="webpage.description.length" class="im_message_webpage_description" ng-bind-html="webpage.rDescription"></div> |
||||
<a class="im_message_video_thumb" ng-click="openEmbed($event)" ng-href="{{webpage.url}}" target="_blank" ng-style="::{width: video.thumb.width + 'px'}"> |
||||
<span ng-if="webpage.duration > 0" class="im_message_video_duration nocopy" data-content="{{::webpage.duration | duration}}"></span> |
||||
<div ng-if="media.webpage.description.length" class="im_message_webpage_description" ng-bind-html="media.webpage.rDescription"></div> |
||||
<a class="im_message_video_thumb" ng-click="openEmbed($event)" ng-href="{{media.webpage.url}}" target="_blank" ng-style="::{width: video.thumb.width + 'px'}"> |
||||
<span ng-if="media.webpage.duration > 0" class="im_message_video_duration nocopy" data-content="{{::media.webpage.duration | duration}}"></span> |
||||
<i class="icon icon-videoplay"></i> |
||||
<img |
||||
class="im_message_video_thumb" |
||||
my-load-thumb |
||||
thumb="webpage.photo.thumb" |
||||
thumb="media.webpage.photo.thumb" |
||||
alt="[{{::'conversation_media_video' | i18n}}]" |
||||
/> |
||||
</a> |
||||
</div> |
||||
|
||||
<div ng-switch-when="document" class="im_message_webpage_document"> |
||||
<div my-message-document="webpage" message-id="messageId"></div> |
||||
<div my-message-document="media.webpage" message-id="messageId"></div> |
||||
</div> |
||||
|
||||
<div ng-switch-when="gif" class="im_message_webpage_gif"> |
||||
<div class="im_message_webpage_title"> |
||||
<a href="{{webpage.url}}" target="_blank" ng-bind-html="webpage.rTitle"></a> |
||||
<a href="{{media.webpage.url}}" target="_blank" ng-bind-html="media.webpage.rTitle"></a> |
||||
</div> |
||||
<div my-message-document="webpage" message-id="messageId"></div> |
||||
<div my-message-document="media.webpage" message-id="messageId"></div> |
||||
</div> |
||||
|
||||
<div ng-switch-default class="im_message_webpage_article"> |
||||
<a ng-if="webpage.photo" href="{{webpage.url}}" target="_blank" class="im_message_webpage_article_photo pull-right"> |
||||
<a ng-if="media.webpage.photo" href="{{media.webpage.url}}" target="_blank" class="im_message_webpage_article_photo pull-right"> |
||||
<img |
||||
class="im_message_article_thumb" |
||||
my-load-thumb |
||||
thumb="webpage.photo.thumb" |
||||
thumb="media.webpage.photo.thumb" |
||||
/> |
||||
</a> |
||||
<div class="im_message_webpage_site" ng-bind="webpage.site_name"></div> |
||||
<div ng-if="media.webpage.site_name" class="im_message_webpage_site" ng-bind="media.webpage.site_name"></div> |
||||
<div class="im_message_webpage_title"> |
||||
<a ng-click="openEmbed($event)" href="{{webpage.url}}" target="_blank" ng-bind-html="webpage.rTitle"></a> |
||||
<a ng-click="openEmbed($event)" href="{{media.webpage.url}}" target="_blank" ng-bind-html="media.webpage.rTitle"></a> |
||||
</div> |
||||
<div ng-if="webpage.description.length" class="im_message_webpage_description" ng-bind-html="webpage.rDescription"></div> |
||||
<div ng-if="media.webpage.description.length" class="im_message_webpage_description" ng-bind-html="media.webpage.rDescription"></div> |
||||
</div> |
||||
</div> |
||||
|
@ -0,0 +1,14 @@
|
||||
<div ng-switch="::media._"> |
||||
<div ng-switch-when="messageMediaPhoto" my-message-photo="media" message-id="messageId"></div> |
||||
<div ng-switch-when="messageMediaDocument" my-message-document="media" message-id="messageId"></div> |
||||
<div ng-switch-when="messageMediaGeo" my-message-geo="media"></div> |
||||
<div ng-switch-when="messageMediaVenue" my-message-venue="media"></div> |
||||
<div ng-switch-when="messageMediaContact" my-message-contact="media"></div> |
||||
<div ng-switch-when="messageMediaWebPage" my-message-webpage="media" message-id="messageId"></div> |
||||
<div ng-switch-when="messageMediaPending" my-message-pending="media"></div> |
||||
<div ng-switch-when="messageMediaUnsupported"> |
||||
<div class="im_message_text" my-i18n="message_attach_unsupported"> |
||||
<my-i18n-param name="link"><a href="https://web.telegram.org" target="_blank">web.telegram.org</a></my-i18n-param> |
||||
</div> |
||||
</div> |
||||
</div> |
@ -0,0 +1 @@
|
||||
<span ng-switch="pinnedMessage.loading"><span ng-switch-when="true" my-i18n="im_reply_loading"><my-i18n-param name="dots"><span my-loading-dots></span></my-i18n-param></span><span ng-switch-default my-short-message="pinnedMessage"></span></span> |
@ -1,8 +1,9 @@
|
||||
<div class="reply_markup_wrap"> |
||||
<div class="reply_markup" ng-class="replyMarkup.splitCount ? 'reply_markup_h' + replyMarkup.splitCount : ''"> |
||||
<div class="reply_markup_row" ng-repeat="row in replyMarkup.rows"> |
||||
<div class="reply_markup_button_wrap" ng-class="'reply_markup_button_w' + row.buttons.length" ng-repeat="button in row.buttons"> |
||||
<button class="btn reply_markup_button" ng-bind-html="::button.rText" ng-click="buttonSend(button)"></button> |
||||
<div class="reply_markup_button_wrap" ng-class="'reply_markup_button_w' + row.buttons.length" ng-repeat="button in row.buttons" ng-switch="button._"> |
||||
<a ng-switch-when="keyboardButtonUrl" class="btn reply_markup_button" href="{{button.pUrl}}" ng-bind-html="::button.rText"></a> |
||||
<button ng-switch-default class="btn reply_markup_button" ng-bind-html="::button.rText" ng-click="buttonClick(button)"></button> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
|
@ -1,8 +1,8 @@
|
||||
<div> |
||||
<a ng-if="historyMessage.media.user_id > 0" class="im_message_contact_photo pull-left" my-peer-photolink="historyMessage.media.user_id" img-class="im_message_contact_photo" user-override="historyMessage.media"></a> |
||||
<div class="im_message_contact_name" ng-switch="historyMessage.media.user_id > 0"> |
||||
<a ng-switch-when="true" my-peer-link="historyMessage.media.user_id" user-override="historyMessage.media"></a> |
||||
<span ng-switch-default ng-bind-html="::historyMessage.media.rFullName"></span> |
||||
<div class="im_message_contact"> |
||||
<a ng-if="::media.user_id > 0" class="im_message_contact_photo pull-left" my-peer-photolink="media.user_id" img-class="im_message_contact_photo" user-override="media"></a> |
||||
<div class="im_message_contact_name" ng-switch="media.user_id > 0"> |
||||
<a ng-switch-when="true" my-peer-link="media.user_id" user-override="media"></a> |
||||
<span ng-switch-default ng-bind-html="::media.rFullName"></span> |
||||
</div> |
||||
<div class="im_message_contact_phone" ng-bind="::historyMessage.media.phone_number | phoneNumber"></div> |
||||
<div class="im_message_contact_phone" ng-bind="::media.phone_number | phoneNumber"></div> |
||||
</div> |
@ -1,20 +1,21 @@
|
||||
<div class="im_message_venue clearfix"> |
||||
|
||||
<a ng-href="{{::venue.mapUrl}}" target="_blank" class="im_message_venue_geopoint_wrap"> |
||||
<a ng-href="{{::media.mapUrl}}" target="_blank" class="im_message_venue_geopoint_wrap"> |
||||
<i class="icon icon-geo-point"></i> |
||||
<img |
||||
class="im_message_venue_geopoint_image" |
||||
my-geo-point-map="venue.geo" |
||||
my-geo-point-map="media.geo" |
||||
width="100" |
||||
height="100" |
||||
alt="[{{::'conversation_media_location' | i18n}} {{::media.mapUrl}}]" |
||||
/> |
||||
</a> |
||||
|
||||
<div class="im_message_venue_info"> |
||||
<div class="im_message_venue_title_wrap"> |
||||
<a ng-href="{{::venue.mapUrl}}" target="_blank" class="im_message_document_name" ng-bind="::venue.title"></a> |
||||
<a ng-href="{{::media.mapUrl}}" target="_blank" class="im_message_document_name" ng-bind="::media.title"></a> |
||||
</div> |
||||
<div class="im_message_venue_address" ng-bind="::venue.address"></div> |
||||
<div class="im_message_venue_address" ng-bind="::media.address"></div> |
||||
</div> |
||||
|
||||
</div> |
||||
|
@ -1,12 +0,0 @@
|
||||
<div class="im_message_video im_message_document_thumbed"> |
||||
<a class="im_message_video_thumb" href="" ng-click="videoOpen()" ng-style="::{width: media.video.thumb.width + 'px'}"> |
||||
<span class="im_message_video_duration" ng-bind="::media.video.duration | duration"></span> |
||||
<i class="icon icon-videoplay"></i> |
||||
<img |
||||
class="im_message_video_thumb im_message_video_thumb_blurred" |
||||
my-load-thumb |
||||
thumb="media.video.thumb" |
||||
/> |
||||
</a> |
||||
</div> |
||||
<div ng-if="::media.rCaption" class="im_message_video_caption" ng-bind-html="::media.rCaption"></div> |
@ -0,0 +1,254 @@
|
||||
/* |
||||
* Toastr |
||||
* Version 2.0.1 |
||||
* Copyright 2012 John Papa and Hans Fjallemark. |
||||
* All Rights Reserved. |
||||
* Use, reproduction, distribution, and modification of this code is subject to the terms and |
||||
* conditions of the MIT license, available at http://www.opensource.org/licenses/mit-license.php |
||||
* |
||||
* Author: John Papa and Hans Fjallemark |
||||
* Project: https://github.com/CodeSeven/toastr |
||||
*/ |
||||
.toast-title { |
||||
font-weight: bold; |
||||
} |
||||
.toast-message { |
||||
-ms-word-wrap: break-word; |
||||
word-wrap: break-word; |
||||
} |
||||
.toast-message a, |
||||
.toast-message label { |
||||
color: #ffffff; |
||||
} |
||||
.toast-message a:hover { |
||||
color: #cccccc; |
||||
text-decoration: none; |
||||
} |
||||
|
||||
.toast-close-button { |
||||
position: relative; |
||||
right: -0.3em; |
||||
top: -0.3em; |
||||
float: right; |
||||
font-size: 20px; |
||||
font-weight: bold; |
||||
color: #ffffff; |
||||
-webkit-text-shadow: 0 1px 0 #ffffff; |
||||
text-shadow: 0 1px 0 #ffffff; |
||||
opacity: 0.8; |
||||
-ms-filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=80); |
||||
filter: alpha(opacity=80); |
||||
} |
||||
.toast-close-button:hover, |
||||
.toast-close-button:focus { |
||||
color: #000000; |
||||
text-decoration: none; |
||||
cursor: pointer; |
||||
opacity: 0.4; |
||||
-ms-filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=40); |
||||
filter: alpha(opacity=40); |
||||
} |
||||
|
||||
/*Additional properties for button version |
||||
iOS requires the button element instead of an anchor tag. |
||||
If you want the anchor version, it requires `href="#"`.*/ |
||||
button.toast-close-button { |
||||
padding: 0; |
||||
cursor: pointer; |
||||
background: transparent; |
||||
border: 0; |
||||
-webkit-appearance: none; |
||||
} |
||||
.toast-top-full-width { |
||||
top: 0; |
||||
right: 0; |
||||
width: 100%; |
||||
} |
||||
.toast-bottom-full-width { |
||||
bottom: 0; |
||||
right: 0; |
||||
width: 100%; |
||||
} |
||||
.toast-top-left { |
||||
top: 12px; |
||||
left: 12px; |
||||
} |
||||
.toast-top-center { |
||||
top: 12px; |
||||
} |
||||
.toast-top-right { |
||||
top: 12px; |
||||
right: 12px; |
||||
} |
||||
.toast-bottom-right { |
||||
right: 12px; |
||||
bottom: 12px; |
||||
} |
||||
.toast-bottom-center { |
||||
bottom: 12px; |
||||
} |
||||
.toast-bottom-left { |
||||
bottom: 12px; |
||||
left: 12px; |
||||
} |
||||
.toast-center { |
||||
top: 45%; |
||||
} |
||||
#toast-container { |
||||
position: fixed; |
||||
z-index: 999999; |
||||
pointer-events: auto; |
||||
/*overrides*/ |
||||
|
||||
} |
||||
#toast-container.toast-center, |
||||
#toast-container.toast-top-center, |
||||
#toast-container.toast-bottom-center{ |
||||
width: 100%; |
||||
pointer-events: none; |
||||
} |
||||
#toast-container.toast-center > div, |
||||
#toast-container.toast-top-center > div, |
||||
#toast-container.toast-bottom-center > div{ |
||||
margin: auto; |
||||
pointer-events: auto; |
||||
} |
||||
#toast-container.toast-center > button, |
||||
#toast-container.toast-top-center > button, |
||||
#toast-container.toast-bottom-center > button{ |
||||
pointer-events: auto; |
||||
} |
||||
#toast-container * { |
||||
-moz-box-sizing: border-box; |
||||
-webkit-box-sizing: border-box; |
||||
box-sizing: border-box; |
||||
} |
||||
#toast-container > div { |
||||
margin: 0 0 6px; |
||||
padding: 15px 15px 15px 50px; |
||||
width: 300px; |
||||
-moz-border-radius: 3px 3px 3px 3px; |
||||
-webkit-border-radius: 3px 3px 3px 3px; |
||||
border-radius: 3px 3px 3px 3px; |
||||
background-position: 15px center; |
||||
background-repeat: no-repeat; |
||||
-moz-box-shadow: 0 0 12px #999999; |
||||
-webkit-box-shadow: 0 0 12px #999999; |
||||
box-shadow: 0 0 12px #999999; |
||||
color: #ffffff; |
||||
opacity: 0.8; |
||||
-ms-filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=80); |
||||
filter: alpha(opacity=80); |
||||
} |
||||
#toast-container > :hover { |
||||
-moz-box-shadow: 0 0 12px #000000; |
||||
-webkit-box-shadow: 0 0 12px #000000; |
||||
box-shadow: 0 0 12px #000000; |
||||
opacity: 1; |
||||
-ms-filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=100); |
||||
filter: alpha(opacity=100); |
||||
cursor: pointer; |
||||
} |
||||
/*#toast-container > .toast-info { |
||||
background-image: url("") !important; |
||||
}*/ |
||||
#toast-container > .toast-wait { |
||||
background-image: url("") !important; |
||||
} |
||||
#toast-container > .toast-error { |
||||
background-image: url("") !important; |
||||
} |
||||
#toast-container > .toast-success { |
||||
background-image: url("") !important; |
||||
} |
||||
#toast-container > .toast-warning { |
||||
background-image: url("") !important; |
||||
} |
||||
|
||||
#toast-container.toast-top-full-width > div, |
||||
#toast-container.toast-bottom-full-width > div { |
||||
width: 96%; |
||||
margin: auto; |
||||
} |
||||
.toast { |
||||
background-color: #030303; |
||||
} |
||||
.toast-success { |
||||
background-color: #51a351; |
||||
} |
||||
.toast-error { |
||||
background-color: #bd362f; |
||||
} |
||||
.toast-info { |
||||
background-color: #2f96b4; |
||||
} |
||||
.toast-wait { |
||||
background-color: #2f96b4; |
||||
} |
||||
.toast-warning { |
||||
background-color: #f89406; |
||||
} |
||||
|
||||
#toast-container > .toast-info { |
||||
background-color: #404040; |
||||
border-radius: 3px; |
||||
box-shadow: 0 0 2px rgba(0,0,0,.12),0 2px 4px rgba(0,0,0,.24); |
||||
color: #fff; |
||||
line-height: 20px; |
||||
padding: 16px; |
||||
font-size: 13px; |
||||
text-align: center; |
||||
} |
||||
|
||||
|
||||
/*Responsive Design*/ |
||||
@media all and (max-width: 240px) { |
||||
#toast-container > div { |
||||
padding: 8px 8px 8px 50px; |
||||
width: 11em; |
||||
} |
||||
#toast-container .toast-close-button { |
||||
right: -0.2em; |
||||
top: -0.2em; |
||||
} |
||||
} |
||||
@media all and (min-width: 241px) and (max-width: 480px) { |
||||
#toast-container > div { |
||||
padding: 8px 8px 8px 50px; |
||||
width: 18em; |
||||
} |
||||
#toast-container .toast-close-button { |
||||
right: -0.2em; |
||||
top: -0.2em; |
||||
} |
||||
} |
||||
@media all and (min-width: 481px) and (max-width: 768px) { |
||||
#toast-container > div { |
||||
padding: 15px 15px 15px 50px; |
||||
width: 25em; |
||||
} |
||||
} |
||||
|
||||
/* |
||||
* AngularJS-Toaster |
||||
* Version 0.3 |
||||
*/ |
||||
:not(.no-enter)#toast-container > div.ng-enter, |
||||
:not(.no-leave)#toast-container > div.ng-leave |
||||
{ |
||||
-webkit-transition: 1000ms cubic-bezier(0.250, 0.250, 0.750, 0.750) all; |
||||
-moz-transition: 1000ms cubic-bezier(0.250, 0.250, 0.750, 0.750) all; |
||||
-ms-transition: 1000ms cubic-bezier(0.250, 0.250, 0.750, 0.750) all; |
||||
-o-transition: 1000ms cubic-bezier(0.250, 0.250, 0.750, 0.750) all; |
||||
transition: 1000ms cubic-bezier(0.250, 0.250, 0.750, 0.750) all; |
||||
} |
||||
|
||||
:not(.no-enter)#toast-container > div.ng-enter.ng-enter-active, |
||||
:not(.no-leave)#toast-container > div.ng-leave { |
||||
opacity: 0.8; |
||||
} |
||||
|
||||
:not(.no-leave)#toast-container > div.ng-leave.ng-leave-active, |
||||
:not(.no-enter)#toast-container > div.ng-enter { |
||||
opacity: 0; |
||||
} |
@ -0,0 +1,507 @@
|
||||
/* global angular */ |
||||
(function(window, document) { |
||||
'use strict'; |
||||
|
||||
/* |
||||
* AngularJS Toaster |
||||
* Version: 2.0.0 |
||||
* |
||||
* Copyright 2013-2016 Jiri Kavulak. |
||||
* All Rights Reserved. |
||||
* Use, reproduction, distribution, and modification of this code is subject to the terms and |
||||
* conditions of the MIT license, available at http://www.opensource.org/licenses/mit-license.php
|
||||
* |
||||
* Author: Jiri Kavulak |
||||
* Related to project of John Papa, Hans Fjällemark and Nguyễn Thiện Hùng (thienhung1989) |
||||
*/ |
||||
|
||||
angular.module('toaster', []).constant( |
||||
'toasterConfig', { |
||||
'limit': 0, // limits max number of toasts
|
||||
'tap-to-dismiss': true, |
||||
'close-button': false, |
||||
'close-html': '<button class="toast-close-button" type="button">×</button>', |
||||
'newest-on-top': true, |
||||
'time-out': 5000, |
||||
'icon-classes': { |
||||
error: 'toast-error', |
||||
info: 'toast-info', |
||||
wait: 'toast-wait', |
||||
success: 'toast-success', |
||||
warning: 'toast-warning' |
||||
}, |
||||
'body-output-type': '', // Options: '', 'trustedHtml', 'template', 'templateWithData', 'directive'
|
||||
'body-template': 'toasterBodyTmpl.html', |
||||
'icon-class': 'toast-info', |
||||
'position-class': 'toast-top-right', // Options (see CSS):
|
||||
// 'toast-top-full-width', 'toast-bottom-full-width', 'toast-center',
|
||||
// 'toast-top-left', 'toast-top-center', 'toast-top-right',
|
||||
// 'toast-bottom-left', 'toast-bottom-center', 'toast-bottom-right',
|
||||
'title-class': 'toast-title', |
||||
'message-class': 'toast-message', |
||||
'prevent-duplicates': false, |
||||
'mouseover-timer-stop': true // stop timeout on mouseover and restart timer on mouseout
|
||||
} |
||||
).service( |
||||
'toaster', [ |
||||
'$rootScope', 'toasterConfig', function($rootScope, toasterConfig) { |
||||
// http://stackoverflow.com/questions/26501688/a-typescript-guid-class
|
||||
var Guid = (function() { |
||||
var Guid = {}; |
||||
Guid.newGuid = function() { |
||||
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c) { |
||||
var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8); |
||||
return v.toString(16); |
||||
}); |
||||
}; |
||||
return Guid; |
||||
}()); |
||||
|
||||
this.pop = function(type, title, body, timeout, bodyOutputType, clickHandler, toasterId, showCloseButton, toastId, onHideCallback) { |
||||
if (angular.isObject(type)) { |
||||
var params = type; // Enable named parameters as pop argument
|
||||
this.toast = { |
||||
type: params.type, |
||||
title: params.title, |
||||
body: params.body, |
||||
timeout: params.timeout, |
||||
bodyOutputType: params.bodyOutputType, |
||||
clickHandler: params.clickHandler, |
||||
showCloseButton: params.showCloseButton, |
||||
closeHtml: params.closeHtml, |
||||
toastId: params.toastId, |
||||
onShowCallback: params.onShowCallback, |
||||
onHideCallback: params.onHideCallback, |
||||
directiveData: params.directiveData |
||||
}; |
||||
toasterId = params.toasterId; |
||||
} else { |
||||
this.toast = { |
||||
type: type, |
||||
title: title, |
||||
body: body, |
||||
timeout: timeout, |
||||
bodyOutputType: bodyOutputType, |
||||
clickHandler: clickHandler, |
||||
showCloseButton: showCloseButton, |
||||
toastId: toastId, |
||||
onHideCallback: onHideCallback |
||||
}; |
||||
} |
||||
|
||||
if (!this.toast.toastId || !this.toast.toastId.length) { |
||||
this.toast.toastId = Guid.newGuid(); |
||||
} |
||||
|
||||
$rootScope.$emit('toaster-newToast', toasterId, this.toast.toastId); |
||||
|
||||
return { |
||||
toasterId: toasterId, |
||||
toastId: this.toast.toastId |
||||
}; |
||||
}; |
||||
|
||||
this.clear = function(toasterId, toastId) { |
||||
if (angular.isObject(toasterId)) { |
||||
$rootScope.$emit('toaster-clearToasts', toasterId.toasterId, toasterId.toastId); |
||||
} else { |
||||
$rootScope.$emit('toaster-clearToasts', toasterId, toastId); |
||||
} |
||||
}; |
||||
|
||||
// Create one method per icon class, to allow to call toaster.info() and similar
|
||||
for (var type in toasterConfig['icon-classes']) { |
||||
this[type] = createTypeMethod(type); |
||||
} |
||||
|
||||
function createTypeMethod(toasterType) { |
||||
return function(title, body, timeout, bodyOutputType, clickHandler, toasterId, showCloseButton, toastId, onHideCallback) { |
||||
if (angular.isString(title)) { |
||||
return this.pop( |
||||
toasterType, |
||||
title, |
||||
body, |
||||
timeout, |
||||
bodyOutputType, |
||||
clickHandler, |
||||
toasterId, |
||||
showCloseButton, |
||||
toastId, |
||||
onHideCallback); |
||||
} else { // 'title' is actually an object with options
|
||||
return this.pop(angular.extend(title, { type: toasterType })); |
||||
} |
||||
}; |
||||
} |
||||
}] |
||||
).factory( |
||||
'toasterEventRegistry', [ |
||||
'$rootScope', function($rootScope) { |
||||
var deregisterNewToast = null, deregisterClearToasts = null, newToastEventSubscribers = [], clearToastsEventSubscribers = [], toasterFactory; |
||||
|
||||
toasterFactory = { |
||||
setup: function() { |
||||
if (!deregisterNewToast) { |
||||
deregisterNewToast = $rootScope.$on( |
||||
'toaster-newToast', function(event, toasterId, toastId) { |
||||
for (var i = 0, len = newToastEventSubscribers.length; i < len; i++) { |
||||
newToastEventSubscribers[i](event, toasterId, toastId); |
||||
} |
||||
}); |
||||
} |
||||
|
||||
if (!deregisterClearToasts) { |
||||
deregisterClearToasts = $rootScope.$on( |
||||
'toaster-clearToasts', function(event, toasterId, toastId) { |
||||
for (var i = 0, len = clearToastsEventSubscribers.length; i < len; i++) { |
||||
clearToastsEventSubscribers[i](event, toasterId, toastId); |
||||
} |
||||
}); |
||||
} |
||||
}, |
||||
|
||||
subscribeToNewToastEvent: function(onNewToast) { |
||||
newToastEventSubscribers.push(onNewToast); |
||||
}, |
||||
subscribeToClearToastsEvent: function(onClearToasts) { |
||||
clearToastsEventSubscribers.push(onClearToasts); |
||||
}, |
||||
unsubscribeToNewToastEvent: function(onNewToast) { |
||||
var index = newToastEventSubscribers.indexOf(onNewToast); |
||||
if (index >= 0) { |
||||
newToastEventSubscribers.splice(index, 1); |
||||
} |
||||
|
||||
if (newToastEventSubscribers.length === 0) { |
||||
deregisterNewToast(); |
||||
deregisterNewToast = null; |
||||
} |
||||
}, |
||||
unsubscribeToClearToastsEvent: function(onClearToasts) { |
||||
var index = clearToastsEventSubscribers.indexOf(onClearToasts); |
||||
if (index >= 0) { |
||||
clearToastsEventSubscribers.splice(index, 1); |
||||
} |
||||
|
||||
if (clearToastsEventSubscribers.length === 0) { |
||||
deregisterClearToasts(); |
||||
deregisterClearToasts = null; |
||||
} |
||||
} |
||||
}; |
||||
return { |
||||
setup: toasterFactory.setup, |
||||
subscribeToNewToastEvent: toasterFactory.subscribeToNewToastEvent, |
||||
subscribeToClearToastsEvent: toasterFactory.subscribeToClearToastsEvent, |
||||
unsubscribeToNewToastEvent: toasterFactory.unsubscribeToNewToastEvent, |
||||
unsubscribeToClearToastsEvent: toasterFactory.unsubscribeToClearToastsEvent |
||||
}; |
||||
}] |
||||
) |
||||
.directive('directiveTemplate', ['$compile', '$injector', function($compile, $injector) { |
||||
return { |
||||
restrict: 'A', |
||||
scope: { |
||||
directiveName: '@directiveName', |
||||
directiveData: '@directiveData' |
||||
}, |
||||
replace: true, |
||||
link: function(scope, elm, attrs) { |
||||
scope.$watch('directiveName', function(directiveName) { |
||||
if (angular.isUndefined(directiveName) || directiveName.length <= 0) |
||||
throw new Error('A valid directive name must be provided via the toast body argument when using bodyOutputType: directive'); |
||||
|
||||
var directive; |
||||
|
||||
try { |
||||
directive = $injector.get(attrs.$normalize(directiveName) + 'Directive'); |
||||
} catch (e) { |
||||
throw new Error(directiveName + ' could not be found. ' + |
||||
'The name should appear as it exists in the markup, not camelCased as it would appear in the directive declaration,' + |
||||
' e.g. directive-name not directiveName.'); |
||||
} |
||||
|
||||
|
||||
var directiveDetails = directive[0]; |
||||
|
||||
if (directiveDetails.scope !== true && directiveDetails.scope) { |
||||
throw new Error('Cannot use a directive with an isolated scope. ' + |
||||
'The scope must be either true or falsy (e.g. false/null/undefined). ' + |
||||
'Occurred for directive ' + directiveName + '.'); |
||||
} |
||||
|
||||
if (directiveDetails.restrict.indexOf('A') < 0) { |
||||
throw new Error('Directives must be usable as attributes. ' + |
||||
'Add "A" to the restrict option (or remove the option entirely). Occurred for directive ' + |
||||
directiveName + '.'); |
||||
} |
||||
|
||||
if (scope.directiveData) |
||||
scope.directiveData = angular.fromJson(scope.directiveData); |
||||
|
||||
var template = $compile('<div ' + directiveName + '></div>')(scope); |
||||
|
||||
elm.append(template); |
||||
}); |
||||
} |
||||
}; |
||||
}]) |
||||
.directive( |
||||
'toasterContainer', [ |
||||
'$parse', '$rootScope', '$interval', '$sce', 'toasterConfig', 'toaster', 'toasterEventRegistry', |
||||
function($parse, $rootScope, $interval, $sce, toasterConfig, toaster, toasterEventRegistry) { |
||||
return { |
||||
replace: true, |
||||
restrict: 'EA', |
||||
scope: true, // creates an internal scope for this directive (one per directive instance)
|
||||
link: function(scope, elm, attrs) { |
||||
var mergedConfig; |
||||
|
||||
// Merges configuration set in directive with default one
|
||||
mergedConfig = angular.extend({}, toasterConfig, scope.$eval(attrs.toasterOptions)); |
||||
|
||||
scope.config = { |
||||
toasterId: mergedConfig['toaster-id'], |
||||
position: mergedConfig['position-class'], |
||||
title: mergedConfig['title-class'], |
||||
message: mergedConfig['message-class'], |
||||
tap: mergedConfig['tap-to-dismiss'], |
||||
closeButton: mergedConfig['close-button'], |
||||
closeHtml: mergedConfig['close-html'], |
||||
animation: mergedConfig['animation-class'], |
||||
mouseoverTimer: mergedConfig['mouseover-timer-stop'] |
||||
}; |
||||
|
||||
scope.$on( |
||||
"$destroy", function() { |
||||
toasterEventRegistry.unsubscribeToNewToastEvent(scope._onNewToast); |
||||
toasterEventRegistry.unsubscribeToClearToastsEvent(scope._onClearToasts); |
||||
} |
||||
); |
||||
|
||||
function setTimeout(toast, time) { |
||||
toast.timeoutPromise = $interval( |
||||
function() { |
||||
scope.removeToast(toast.toastId); |
||||
}, time, 1 |
||||
); |
||||
} |
||||
|
||||
scope.configureTimer = function(toast) { |
||||
var timeout = angular.isNumber(toast.timeout) ? toast.timeout : mergedConfig['time-out']; |
||||
if (typeof timeout === "object") timeout = timeout[toast.type]; |
||||
if (timeout > 0) { |
||||
setTimeout(toast, timeout); |
||||
} |
||||
}; |
||||
|
||||
function addToast(toast, toastId) { |
||||
toast.type = mergedConfig['icon-classes'][toast.type]; |
||||
if (!toast.type) { |
||||
toast.type = mergedConfig['icon-class']; |
||||
} |
||||
|
||||
if (mergedConfig['prevent-duplicates'] === true && scope.toasters.length) { |
||||
if (scope.toasters[scope.toasters.length - 1].body === toast.body) { |
||||
return; |
||||
} else { |
||||
var i, len, dupFound = false; |
||||
for (i = 0, len = scope.toasters.length; i < len; i++) { |
||||
if (scope.toasters[i].toastId === toastId) { |
||||
dupFound = true; |
||||
break; |
||||
} |
||||
} |
||||
|
||||
if (dupFound) return; |
||||
} |
||||
} |
||||
|
||||
|
||||
// set the showCloseButton property on the toast so that
|
||||
// each template can bind directly to the property to show/hide
|
||||
// the close button
|
||||
var closeButton = mergedConfig['close-button']; |
||||
|
||||
// if toast.showCloseButton is a boolean value,
|
||||
// it was specifically overriden in the pop arguments
|
||||
if (typeof toast.showCloseButton === "boolean") { |
||||
|
||||
} else if (typeof closeButton === "boolean") { |
||||
toast.showCloseButton = closeButton; |
||||
} else if (typeof closeButton === "object") { |
||||
var closeButtonForType = closeButton[toast.type]; |
||||
|
||||
if (typeof closeButtonForType !== "undefined" && closeButtonForType !== null) { |
||||
toast.showCloseButton = closeButtonForType; |
||||
} |
||||
} else { |
||||
// if an option was not set, default to false.
|
||||
toast.showCloseButton = false; |
||||
} |
||||
|
||||
if (toast.showCloseButton) { |
||||
toast.closeHtml = $sce.trustAsHtml(toast.closeHtml || scope.config.closeHtml); |
||||
} |
||||
|
||||
// Set the toast.bodyOutputType to the default if it isn't set
|
||||
toast.bodyOutputType = toast.bodyOutputType || mergedConfig['body-output-type']; |
||||
switch (toast.bodyOutputType) { |
||||
case 'trustedHtml': |
||||
toast.html = $sce.trustAsHtml(toast.body); |
||||
break; |
||||
case 'template': |
||||
toast.bodyTemplate = toast.body || mergedConfig['body-template']; |
||||
break; |
||||
case 'templateWithData': |
||||
var fcGet = $parse(toast.body || mergedConfig['body-template']); |
||||
var templateWithData = fcGet(scope); |
||||
toast.bodyTemplate = templateWithData.template; |
||||
toast.data = templateWithData.data; |
||||
break; |
||||
case 'directive': |
||||
toast.html = toast.body; |
||||
break; |
||||
} |
||||
|
||||
scope.configureTimer(toast); |
||||
|
||||
if (mergedConfig['newest-on-top'] === true) { |
||||
scope.toasters.unshift(toast); |
||||
if (mergedConfig['limit'] > 0 && scope.toasters.length > mergedConfig['limit']) { |
||||
scope.toasters.pop(); |
||||
} |
||||
} else { |
||||
scope.toasters.push(toast); |
||||
if (mergedConfig['limit'] > 0 && scope.toasters.length > mergedConfig['limit']) { |
||||
scope.toasters.shift(); |
||||
} |
||||
} |
||||
|
||||
if (angular.isFunction(toast.onShowCallback)) { |
||||
toast.onShowCallback(); |
||||
} |
||||
} |
||||
|
||||
scope.removeToast = function(toastId) { |
||||
var i, len; |
||||
for (i = 0, len = scope.toasters.length; i < len; i++) { |
||||
if (scope.toasters[i].toastId === toastId) { |
||||
removeToast(i); |
||||
break; |
||||
} |
||||
} |
||||
}; |
||||
|
||||
function removeToast(toastIndex) { |
||||
var toast = scope.toasters[toastIndex]; |
||||
|
||||
// toast is always defined since the index always has a match
|
||||
if (toast.timeoutPromise) { |
||||
$interval.cancel(toast.timeoutPromise); |
||||
} |
||||
scope.toasters.splice(toastIndex, 1); |
||||
|
||||
if (angular.isFunction(toast.onHideCallback)) { |
||||
toast.onHideCallback(); |
||||
} |
||||
} |
||||
|
||||
function removeAllToasts(toastId) { |
||||
for (var i = scope.toasters.length - 1; i >= 0; i--) { |
||||
if (isUndefinedOrNull(toastId)) { |
||||
removeToast(i); |
||||
} else { |
||||
if (scope.toasters[i].toastId == toastId) { |
||||
removeToast(i); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
|
||||
scope.toasters = []; |
||||
|
||||
function isUndefinedOrNull(val) { |
||||
return angular.isUndefined(val) || val === null; |
||||
} |
||||
|
||||
scope._onNewToast = function(event, toasterId, toastId) { |
||||
// Compatibility: if toaster has no toasterId defined, and if call to display
|
||||
// hasn't either, then the request is for us
|
||||
|
||||
if ((isUndefinedOrNull(scope.config.toasterId) && isUndefinedOrNull(toasterId)) || (!isUndefinedOrNull(scope.config.toasterId) && !isUndefinedOrNull(toasterId) && scope.config.toasterId == toasterId)) { |
||||
addToast(toaster.toast, toastId); |
||||
} |
||||
}; |
||||
scope._onClearToasts = function(event, toasterId, toastId) { |
||||
// Compatibility: if toaster has no toasterId defined, and if call to display
|
||||
// hasn't either, then the request is for us
|
||||
if (toasterId == '*' || (isUndefinedOrNull(scope.config.toasterId) && isUndefinedOrNull(toasterId)) || (!isUndefinedOrNull(scope.config.toasterId) && !isUndefinedOrNull(toasterId) && scope.config.toasterId == toasterId)) { |
||||
removeAllToasts(toastId); |
||||
} |
||||
}; |
||||
|
||||
toasterEventRegistry.setup(); |
||||
|
||||
toasterEventRegistry.subscribeToNewToastEvent(scope._onNewToast); |
||||
toasterEventRegistry.subscribeToClearToastsEvent(scope._onClearToasts); |
||||
}, |
||||
controller: [ |
||||
'$scope', '$element', '$attrs', function($scope, $element, $attrs) { |
||||
// Called on mouseover
|
||||
$scope.stopTimer = function(toast) { |
||||
if ($scope.config.mouseoverTimer === true) { |
||||
if (toast.timeoutPromise) { |
||||
$interval.cancel(toast.timeoutPromise); |
||||
toast.timeoutPromise = null; |
||||
} |
||||
} |
||||
}; |
||||
|
||||
// Called on mouseout
|
||||
$scope.restartTimer = function(toast) { |
||||
if ($scope.config.mouseoverTimer === true) { |
||||
if (!toast.timeoutPromise) { |
||||
$scope.configureTimer(toast); |
||||
} |
||||
} else if (toast.timeoutPromise === null) { |
||||
$scope.removeToast(toast.toastId); |
||||
} |
||||
}; |
||||
|
||||
$scope.click = function(toast, isCloseButton) { |
||||
if ($scope.config.tap === true || (toast.showCloseButton === true && isCloseButton === true)) { |
||||
var removeToast = true; |
||||
if (toast.clickHandler) { |
||||
if (angular.isFunction(toast.clickHandler)) { |
||||
removeToast = toast.clickHandler(toast, isCloseButton); |
||||
} else if (angular.isFunction($scope.$parent.$eval(toast.clickHandler))) { |
||||
removeToast = $scope.$parent.$eval(toast.clickHandler)(toast, isCloseButton); |
||||
} else { |
||||
console.log("TOAST-NOTE: Your click handler is not inside a parent scope of toaster-container."); |
||||
} |
||||
} |
||||
if (removeToast) { |
||||
$scope.removeToast(toast.toastId); |
||||
} |
||||
} |
||||
}; |
||||
}], |
||||
template: |
||||
'<div id="toast-container" ng-class="[config.position, config.animation]">' + |
||||
'<div ng-repeat="toaster in toasters" class="toast" ng-class="toaster.type" ng-click="click(toaster)" ng-mouseover="stopTimer(toaster)" ng-mouseout="restartTimer(toaster)">' + |
||||
'<div ng-if="toaster.showCloseButton" ng-click="click(toaster, true)" ng-bind-html="toaster.closeHtml"></div>' + |
||||
'<div ng-class="config.title">{{toaster.title}}</div>' + |
||||
'<div ng-class="config.message" ng-switch on="toaster.bodyOutputType">' + |
||||
'<div ng-switch-when="trustedHtml" ng-bind-html="toaster.html"></div>' + |
||||
'<div ng-switch-when="template"><div ng-include="toaster.bodyTemplate"></div></div>' + |
||||
'<div ng-switch-when="templateWithData"><div ng-include="toaster.bodyTemplate"></div></div>' + |
||||
'<div ng-switch-when="directive"><div directive-template directive-name="{{toaster.html}}" directive-data="{{toaster.directiveData}}"></div></div>' + |
||||
'<div ng-switch-default >{{toaster.body}}</div>' + |
||||
'</div>' + |
||||
'</div>' + |
||||
'</div>' |
||||
}; |
||||
}] |
||||
); |
||||
})(window, document); |
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
Loading…
Reference in new issue