123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127 |
- /**
- * This file is part of Threema Web.
- *
- * Threema Web is free software: you can redistribute it and/or modify it
- * under the terms of the GNU Affero General Public License as published by
- * the Free Software Foundation, either version 3 of the License, or (at
- * your option) any later version.
- *
- * This program is distributed in the hope that it will be useful, but
- * WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
- * General Public License for more details.
- *
- * You should have received a copy of the GNU Affero General Public License
- * along with Threema Web. If not, see <http://www.gnu.org/licenses/>.
- */
- // tslint:disable:max-line-length
- import {hasValue} from '../helpers';
- import {WebClientService} from '../services/webclient';
- // Get text depending on type
- function getText(message: threema.Message): string {
- switch (message.type) {
- case 'text':
- return message.body;
- case 'location':
- return message.location.description;
- case 'file':
- // Prefer caption for file messages, if available
- if (message.caption && message.caption.length > 0) {
- return message.caption;
- }
- return message.file.name;
- }
- return message.caption;
- }
- export default [
- function() {
- return {
- restrict: 'EA',
- scope: {},
- bindToController: {
- message: '=',
- multiLine: '@?multiLine',
- linkify: '@?linkify',
- },
- link: function(scope, elem, attrs) {
- scope.$watch(
- () => scope.ctrl.message.id,
- (newId, oldId) => {
- // Register for message ID changes. When it changes, update the text.
- // This prevents processing the text more than once.
- if (hasValue(newId) && newId !== oldId) {
- scope.ctrl.updateText();
- }
- },
- );
- scope.$watch(
- () => scope.ctrl.message.caption,
- (newCaption, oldCaption) => {
- // Register for message caption changes. When it changes, update the text.
- //
- // Background: The caption may change because image messages may be sent from the
- // app before the image has been downloaded and parsed. That information
- // (thumbnail + caption) is sent later on as an update.
- if (hasValue(newCaption) && newCaption !== oldCaption) {
- scope.ctrl.updateText();
- }
- },
- );
- },
- controllerAs: 'ctrl',
- controller: ['WebClientService', '$filter', function(webClientService: WebClientService, $filter: ng.IFilterService) {
- // TODO: Extract filters into helper functions
- const escapeHtml = $filter('escapeHtml') as any;
- const markify = $filter('markify') as any;
- const emojify = $filter('emojify') as any;
- const enlargeSingleEmoji = $filter('enlargeSingleEmoji') as any;
- const mentionify = $filter('mentionify') as any;
- const linkify = $filter('linkify') as any;
- const nlToBr = $filter('nlToBr') as any;
- /**
- * Apply filters to text.
- */
- function processText(text: string, largeSingleEmoji: boolean, multiLine: boolean, linkifyText: boolean): string {
- const nonLinkified = mentionify(enlargeSingleEmoji(emojify(markify(escapeHtml(text))), largeSingleEmoji));
- const maybeLinkified = linkifyText ? linkify(nonLinkified) : nonLinkified;
- return nlToBr(maybeLinkified, multiLine);
- }
- /**
- * Text update function.
- */
- this.updateText = () => {
- // Because this.multiLine and this.linkify are bound using an `@` binding,
- // they are either undefined or a string. Convert to boolean.
- const multiLine = (this.multiLine === undefined || this.multiLine !== 'false');
- const linkifyText = (this.linkify === undefined || this.linkify !== 'false');
- // Process text once, apply all filter functions
- const text = getText(this.message);
- this.text = processText(
- text,
- this.largeSingleEmoji,
- multiLine,
- linkifyText,
- );
- };
- this.largeSingleEmoji = webClientService.appConfig.largeSingleEmoji;
- this.$onInit = function() {
- // Process initial text
- this.updateText();
- };
- }],
- template: `
- <span click-action ng-bind-html="ctrl.text"></span>
- `,
- };
- },
- ];
|