harbour-fernschreiber/qml/js/functions.js
John Gibbon d0f33969eb support basic bot messages (reply markup)
only inlineKeyboardButtonTypeCallback and inlineKeyboardButtonTypeUrl are implemented.
2020-12-27 00:01:59 +01:00

459 lines
21 KiB
JavaScript

/*
Copyright (C) 2020 Sebastian J. Wolf and other contributors
This file is part of Fernschreiber.
Fernschreiber is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Fernschreiber 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with Fernschreiber. If not, see <http://www.gnu.org/licenses/>.
*/
.pragma library
.import "debug.js" as Debug
.import Sailfish.Silica 1.0 as Silica
var tdLibWrapper;
var appNotification;
function setGlobals(globals) {
tdLibWrapper = globals.tdLibWrapper;
appNotification = globals.appNotification;
}
function getUserName(userInformation) {
return ((userInformation.first_name || "") + " " + (userInformation.last_name || "")).trim();
}
function getMessageText(message, simple, myself, ignoreEntities) {
switch(message.content['@type']) {
case 'messageText':
if (simple) {
return message.content.text.text;
} else {
return enhanceMessageText(message.content.text, ignoreEntities);
}
case 'messageSticker':
return simple ? qsTr("Sticker: %1").arg(message.content.sticker.emoji) : "";
case 'messagePhoto':
if (message.content.caption.text !== "") {
return simple ? qsTr("Picture: %1").arg(message.content.caption.text) : enhanceMessageText(message.content.caption, ignoreEntities)
} else {
return simple ? (myself ? qsTr("sent a picture", "myself") : qsTr("sent a picture")) : "";
}
case 'messageVideo':
if (message.content.caption.text !== "") {
return simple ? qsTr("Video: %1").arg(message.content.caption.text) : enhanceMessageText(message.content.caption, ignoreEntities)
} else {
return simple ? (myself ? qsTr("sent a video", "myself") : qsTr("sent a video")) : "";
}
case 'messageVideoNote':
return simple ? (myself ? qsTr("sent a video note", "myself") : qsTr("sent a video note")) : "";
case 'messageAnimation':
if (message.content.caption.text !== "") {
return simple ? qsTr("Animation: %1").arg(message.content.caption.text) : enhanceMessageText(message.content.caption, ignoreEntities)
} else {
return simple ? (myself ? qsTr("sent an animation", "myself") : qsTr("sent an animation")) : "";
}
case 'messageAudio':
if (message.content.caption.text !== "") {
return simple ? qsTr("Audio: %1").arg(message.content.caption.text) : enhanceMessageText(message.content.caption, ignoreEntities)
} else {
return simple ? (myself ? qsTr("sent an audio", "myself") : qsTr("sent an audio")) : "";
}
case 'messageVoiceNote':
if (message.content.caption.text !== "") {
return simple ? qsTr("Voice Note: %1").arg(message.content.caption.text) : enhanceMessageText(message.content.caption, ignoreEntities)
} else {
return simple ? (myself ? qsTr("sent a voice note", "myself") : qsTr("sent a voice note")) : "";
}
case 'messageDocument':
if (message.content.document.file_name !== "") {
return simple ? qsTr("Document: %1").arg(message.content.document.file_name) : (message.content.document.file_name + ( message.content.caption.text !== "" ? ("<br />" + enhanceMessageText(message.content.caption, ignoreEntities) ) : "")).trim();
} else {
return simple ? (myself ? qsTr("sent a document", "myself") : qsTr("sent a document")) : "";
}
case 'messageLocation':
return simple ? (myself ? qsTr("sent a location", "myself") : qsTr("sent a location")) : "";
case 'messageVenue':
return simple ? (myself ? qsTr("sent a venue", "myself") : qsTr("sent a venue")) : ( "<b>" + message.content.venue.title + "</b>, " + message.content.venue.address );
case 'messageContactRegistered':
return myself ? qsTr("have registered with Telegram") : qsTr("has registered with Telegram");
case 'messageChatJoinByLink':
return myself ? qsTr("joined this chat", "myself") : qsTr("joined this chat");
case 'messageChatAddMembers':
return myself ? qsTr("were added to this chat", "myself") : qsTr("was added to this chat");
case 'messageChatDeleteMember':
return myself ? qsTr("left this chat", "myself") : qsTr("left this chat");
case 'messageChatChangeTitle':
return myself ? qsTr("changed the chat title to %1", "myself").arg(message.content.title) : qsTr("changed the chat title to %1").arg(message.content.title);
case 'messagePoll':
if (message.content.poll.type['@type'] === "pollTypeQuiz") {
if (message.content.poll.is_anonymous) {
return simple ? (myself ? qsTr("sent an anonymous quiz", "myself") : qsTr("sent an anonymous quiz")) : ("<b>" + qsTr("Anonymous Quiz") + "</b>");
}
return simple ? (myself ? qsTr("sent a quiz", "myself") : qsTr("sent a quiz")) : ("<b>" + qsTr("Quiz") + "</b>");
}
if (message.content.poll.is_anonymous) {
return simple ? (myself ? qsTr("sent an anonymous poll", "myself") : qsTr("sent an anonymous poll")) : ("<b>" + qsTr("Anonymous Poll") + "</b>");
}
return simple ? (myself ? qsTr("sent a poll", "myself") : qsTr("sent a poll")) : ("<b>" + qsTr("Poll") + "</b>");
case 'messageBasicGroupChatCreate':
case 'messageSupergroupChatCreate':
return myself ? qsTr("created this group", "myself") : qsTr("created this group");
case 'messageChatChangePhoto':
return myself ? qsTr("changed the chat photo", "myself") : qsTr("changed the chat photo");
case 'messageChatDeletePhoto':
return myself ? qsTr("deleted the chat photo", "myself") : qsTr("deleted the chat photo");
case 'messageChatSetTtl':
return myself ? qsTr("changed the secret chat TTL setting", "myself; TTL = Time To Live") : qsTr("changed the secret chat TTL setting", "TTL = Time To Live");
case 'messageChatUpgradeFrom':
case 'messageChatUpgradeTo':
return myself ? qsTr("upgraded this group to a supergroup", "myself") : qsTr("upgraded this group to a supergroup");
case 'messageCustomServiceAction':
return message.content.text;
case 'messagePinMessage':
return myself ? qsTr("changed the pinned message", "myself") : qsTr("changed the pinned message");
case 'messageExpiredPhoto':
return myself ? qsTr("sent a self-destructing photo that is expired", "myself") : qsTr("sent a self-destructing photo that is expired");
case 'messageExpiredVideo':
return myself ? qsTr("sent a self-destructing video that is expired", "myself") : qsTr("sent a self-destructing video that is expired");
case 'messageScreenshotTaken':
return myself ? qsTr("created a screenshot in this chat", "myself") : qsTr("created a screenshot in this chat");
case 'messageUnsupported':
return myself ? qsTr("sent an unsupported message", "myself") : qsTr("sent an unsupported message");
default:
return myself ? qsTr("sent an unsupported message: %1", "myself; %1 is message type").arg(message.content['@type'].substring(7)) : qsTr("sent an unsupported message: %1", "%1 is message type").arg(message.content['@type'].substring(7));
}
}
function getChatPartnerStatusText(statusType, was_online) {
switch(statusType) {
case "userStatusEmpty":
return qsTr("was never online");
case "userStatusLastMonth":
return qsTr("last online: last month");
case "userStatusLastWeek":
return qsTr("last online: last week");
case "userStatusOffline":
return qsTr("last online: %1").arg(getDateTimeElapsed(was_online));
case "userStatusOnline":
return qsTr("online");
case "userStatusRecently":
return qsTr("was recently online");
}
}
function getSecretChatStatus(secretChatDetails) {
switch (secretChatDetails.state["@type"]) {
case "secretChatStateClosed":
return "<b>" + qsTr("Closed!") + "</b>";
case "secretChatStatePending":
return qsTr("Pending acknowledgement");
case "secretChatStateReady":
return "";
}
}
function getChatMemberStatusText(statusType) {
switch(statusType) {
case "chatMemberStatusAdministrator":
return qsTr("Admin", "channel user role");
case "chatMemberStatusBanned":
return qsTr("Banned", "channel user role");
case "chatMemberStatusCreator":
return qsTr("Creator", "channel user role");
case "chatMemberStatusRestricted":
return qsTr("Restricted", "channel user role");
case "chatMemberStatusLeft":
case "chatMemberStatusMember":
return ""
}
return statusType || "";
}
function getShortenedCount(count) {
if (count >= 1000000) {
return qsTr("%1M").arg((count / 1000000).toLocaleString(Qt.locale(), 'f', 0));
} else if (count >= 1000 ) {
return qsTr("%1K").arg((count / 1000).toLocaleString(Qt.locale(), 'f', 0));
} else {
return count;
}
}
function getDateTimeElapsed(timestamp) {
return Silica.Format.formatDate(new Date(timestamp * 1000), Silica.Formatter.DurationElapsed);
}
function getDateTimeTranslated(timestamp) {
return new Date(timestamp * 1000).toLocaleString();
}
function handleHtmlEntity(messageText, messageInsertions, originalString, replacementString) {
var nextIndex = -1;
while ((nextIndex = messageText.indexOf(originalString, nextIndex + 1)) > -1) {
messageInsertions.push({ offset: nextIndex, insertionString: replacementString, removeLength: originalString.length });
}
}
var rawNewLineRegExp = /\r?\n/g;
var ampRegExp = /&/g;
var ltRegExp = /</g;
var gtRegExp = />/g;
function enhanceHtmlEntities(simpleText) {
return simpleText.replace(ampRegExp, "&amp;").replace(ltRegExp, "&lt;").replace(gtRegExp, "&gt;");//.replace(rawNewLineRegExp, "<br>");
}
function messageInsertionSorter(a, b) { return (b.offset+b.removeLength) - (a.offset+a.removeLength) }
function enhanceMessageText(formattedText, ignoreEntities) {
var messageInsertions = [];
var messageText = formattedText.text;
var entity;
if (ignoreEntities) {
return messageText;
}
if(formattedText.entities.length === 0) {
return messageText.replace(ampRegExp, "&amp;").replace(ltRegExp, "&lt;").replace(gtRegExp, "&gt;").replace(rawNewLineRegExp, "<br>");
}
for (var i = 0; i < formattedText.entities.length; i++) {
entity = formattedText.entities[i];
if (entity['@type'] !== "textEntity") {
continue;
}
switch(entity.type['@type']) {
case "textEntityTypeBold":
messageInsertions.push(
{ offset: entity.offset, insertionString: "<b>", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</b>", removeLength: 0 }
);
break;
case "textEntityTypeUrl":
messageInsertions.push(
{ offset: entity.offset, insertionString: "<a href=\"" + messageText.substring(entity.offset, ( entity.offset + entity.length )) + "\">", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</a>", removeLength: 0 }
);
break;
case "textEntityTypeCode":
messageInsertions.push(
{ offset: entity.offset, insertionString: "<pre>", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</pre>", removeLength: 0 }
);
break;
case "textEntityTypeEmailAddress":
messageInsertions.push(
{ offset: entity.offset, insertionString: "<a href=\"mailto:" + messageText.substring(entity.offset, ( entity.offset + entity.length )) + "\">", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</a>", removeLength: 0 }
);
break;
case "textEntityTypeItalic":
messageInsertions.push(
{ offset: entity.offset, insertionString: "<i>", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</i>", removeLength: 0 }
);
break;
case "textEntityTypeMention":
messageInsertions.push(
{ offset: entity.offset, insertionString: "<a href=\"user://" + messageText.substring(entity.offset, ( entity.offset + entity.length )) + "\">", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</a>", removeLength: 0 }
);
break;
case "textEntityTypeMentionName":
messageInsertions.push(
{ offset: entity.offset, insertionString: "<a href=\"userId://" + entity.type.user_id + "\">", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</a>", removeLength: 0 }
);
break;
case "textEntityTypePhoneNumber":
messageInsertions.push(
{ offset: entity.offset, insertionString: "<a href=\"tel:" + messageText.substring(entity.offset, ( entity.offset + entity.length )) + "\">", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</a>", removeLength: 0 }
);
break;
case "textEntityTypePre":
messageInsertions.push(
{ offset: entity.offset, insertionString: "<pre>", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</pre>", removeLength: 0 }
);
break;
case "textEntityTypePreCode":
messageInsertions.push(
{ offset: entity.offset, insertionString: "<pre>", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</pre>", removeLength: 0 }
);
break;
case "textEntityTypeTextUrl":
messageInsertions.push(
{ offset: entity.offset, insertionString: "<a href=\"" + entity.type.url + "\">", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</a>", removeLength: 0 }
);
break;
case "textEntityTypeUnderline":
messageInsertions.push(
{ offset: entity.offset, insertionString: "<u>", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</u>", removeLength: 0 }
);
break;
case "textEntityTypeBotCommand":
var command = messageText.substring(entity.offset, entity.offset + entity.length);
messageInsertions.push(
{ offset: entity.offset, insertionString: "<a href=\"botCommand://" + command + "\">", removeLength: 0 },
{ offset: (entity.offset + entity.length), insertionString: "</a>", removeLength: 0 }
);
break;
}
}
if(messageInsertions.length === 0) {
return messageText.replace(ampRegExp, "&amp;").replace(ltRegExp, "&lt;").replace(gtRegExp, "&gt;").replace(rawNewLineRegExp, "<br>");
}
handleHtmlEntity(messageText, messageInsertions, "&", "&amp;");
handleHtmlEntity(messageText, messageInsertions, "<", "&lt;");
handleHtmlEntity(messageText, messageInsertions, ">", "&gt;");
messageInsertions.sort(messageInsertionSorter);
for (var z = 0; z < messageInsertions.length; z++) {
messageText = messageText.substring(0, messageInsertions[z].offset)
+ messageInsertions[z].insertionString
+ messageText.substring(messageInsertions[z].offset + messageInsertions[z].removeLength);
}
messageText = messageText.replace(rawNewLineRegExp, "<br>");
return messageText;
}
function handleLink(link) {
var tMePrefix = tdLibWrapper.getOptionString("t_me_url");
if (link.indexOf("user://") === 0) {
var userName = link.substring(8);
var userInformation = tdLibWrapper.getUserInformationByName(userName);
if (typeof userInformation.id === "undefined") {
appNotification.show(qsTr("Unable to find user %1").arg(userName));
} else {
tdLibWrapper.createPrivateChat(userInformation.id);
}
} else if (link.indexOf("userId://") === 0) {
tdLibWrapper.createPrivateChat(link.substring(9));
} else if (link.indexOf("tg://") === 0) {
Debug.log("Special TG link: ", link);
if (link.indexOf("tg://join?invite=") === 0) {
tdLibWrapper.joinChatByInviteLink(tMePrefix + "joinchat/" + link.substring(17));
} else if (link.indexOf("tg://resolve?domain=") === 0) {
tdLibWrapper.searchPublicChat(link.substring(20));
}
} else if (link.indexOf("botCommand://") === 0) { // this gets returned to send on ChatPage
return link.substring(13);
} else {
if (link.indexOf(tMePrefix) === 0) {
if (link.indexOf("joinchat") !== -1) {
Debug.log("Joining Chat: ", link);
tdLibWrapper.joinChatByInviteLink(link);
// Do the necessary stuff to open the chat if successful
// Fail with nice error message if it doesn't work
} else {
Debug.log("Search public chat: ", link.substring(tMePrefix.length));
tdLibWrapper.searchPublicChat(link.substring(tMePrefix.length));
// Check responses for updateBasicGroup or updateSupergroup
// Fire createBasicGroupChat or createSupergroupChat
// Do the necessary stuff to open the chat
// Fail with nice error message if chat can't be found
}
} else {
Qt.openUrlExternally(link);
}
}
}
function getVideoHeight(videoWidth, videoData) {
if (typeof videoData !== "undefined") {
if (videoData.height === 0) {
return videoWidth;
} else {
var aspectRatio = videoData.height / videoData.width;
return Math.round(videoWidth * aspectRatio);
}
} else {
return 1;
}
}
function replaceUrlsWithLinks(string) {
return string.replace(/((\w+):\/\/[\w?=&.\/-;#~%-]+(?![\w\s?&.\/;#~%"=-]*>))/g, "<a href=\"$1\">$1</a>");
}
function sortMessagesArrayByDate(messages) {
messages.sort(function(a, b) {
return a.date - b.date;
});
}
function getMessagesArrayIds(messages) {
sortMessagesArrayByDate(messages);
return messages.map(function(message){return message.id.toString()});
}
function getMessagesArrayText(messages) {
sortMessagesArrayByDate(messages);
var lastSenderName = "";
var lines = [];
for(var i = 0; i < messages.length; i += 1) {
var senderName = getUserName(tdLibWrapper.getUserInformation(messages[i].sender.user_id));
if(senderName !== lastSenderName) {
lines.push(senderName);
}
lastSenderName = senderName;
lines.push(getMessageText(messages[i], true, false));
lines.push("");
}
return lines.join("\n");
}
function handleErrorMessage(code, message) {
if (code === 404) {
// Silently ignore 404 Not Found messages (occur sometimes, without clear context...)
return;
}
if (message === "USER_ALREADY_PARTICIPANT") {
appNotification.show(qsTr("You are already a member of this chat."));
} else {
appNotification.show(message);
}
}
function getMessagesNeededForwardPermissions(messages) {
var neededPermissions = ["can_send_messages"]
var mediaMessageTypes = ["messageAudio", "messageDocument", "messagePhoto", "messageVideo", "messageVideoNote", "messageVoiceNote"]
var otherMessageTypes = ["messageAnimation", "messageGame", "messageSticker"]
for(var i = 0; i < messages.length && neededPermissions.length < 3; i += 1) {
var type = messages[i]["content"]["@type"]
var permission = ""
if(type === "messageText") {
continue
} else if(type === "messagePoll") {
permission = "can_send_polls"
} else if(mediaMessageTypes.indexOf(type) > -1) {
permission = "can_send_media_messages"
} else if(otherMessageTypes.indexOf(type) > -1) {
permission = "can_send_other_messages"
}
if(permission !== "" && neededPermissions.indexOf(permission) === -1) {
neededPermissions.push(permission)
}
}
return neededPermissions
}