/**
* 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 .
*/
// 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: `
`,
};
},
];