Payments: bubble attachment & service message
This commit is contained in:
parent
916b77e567
commit
3568c5dbe0
@ -49,7 +49,6 @@ import { getMiddleware } from "../../helpers/middleware";
|
|||||||
import cancelEvent from "../../helpers/dom/cancelEvent";
|
import cancelEvent from "../../helpers/dom/cancelEvent";
|
||||||
import { attachClickEvent, simulateClickEvent } from "../../helpers/dom/clickEvent";
|
import { attachClickEvent, simulateClickEvent } from "../../helpers/dom/clickEvent";
|
||||||
import htmlToDocumentFragment from "../../helpers/dom/htmlToDocumentFragment";
|
import htmlToDocumentFragment from "../../helpers/dom/htmlToDocumentFragment";
|
||||||
import positionElementByIndex from "../../helpers/dom/positionElementByIndex";
|
|
||||||
import reflowScrollableElement from "../../helpers/dom/reflowScrollableElement";
|
import reflowScrollableElement from "../../helpers/dom/reflowScrollableElement";
|
||||||
import replaceContent from "../../helpers/dom/replaceContent";
|
import replaceContent from "../../helpers/dom/replaceContent";
|
||||||
import setInnerHTML from "../../helpers/dom/setInnerHTML";
|
import setInnerHTML from "../../helpers/dom/setInnerHTML";
|
||||||
@ -106,10 +105,10 @@ import { cancelContextMenuOpening } from "../../helpers/dom/attachContextMenuLis
|
|||||||
import contextMenuController from "../../helpers/contextMenuController";
|
import contextMenuController from "../../helpers/contextMenuController";
|
||||||
import { AckedResult } from "../../lib/mtproto/superMessagePort";
|
import { AckedResult } from "../../lib/mtproto/superMessagePort";
|
||||||
import middlewarePromise from "../../helpers/middlewarePromise";
|
import middlewarePromise from "../../helpers/middlewarePromise";
|
||||||
import findAndSplice from "../../helpers/array/findAndSplice";
|
|
||||||
import { EmoticonsDropdown } from "../emoticonsDropdown";
|
import { EmoticonsDropdown } from "../emoticonsDropdown";
|
||||||
import indexOfAndSplice from "../../helpers/array/indexOfAndSplice";
|
import indexOfAndSplice from "../../helpers/array/indexOfAndSplice";
|
||||||
import noop from "../../helpers/noop";
|
import noop from "../../helpers/noop";
|
||||||
|
import paymentsWrapCurrencyAmount from "../../helpers/paymentsWrapCurrencyAmount";
|
||||||
|
|
||||||
const USE_MEDIA_TAILS = false;
|
const USE_MEDIA_TAILS = false;
|
||||||
const IGNORE_ACTIONS: Set<Message.messageService['action']['_']> = new Set([
|
const IGNORE_ACTIONS: Set<Message.messageService['action']['_']> = new Set([
|
||||||
@ -3580,7 +3579,7 @@ export default class ChatBubbles {
|
|||||||
rowDiv.classList.add('reply-markup-row');
|
rowDiv.classList.add('reply-markup-row');
|
||||||
|
|
||||||
buttons.forEach((button) => {
|
buttons.forEach((button) => {
|
||||||
const text = wrapRichText(button.text, {noLinks: true, noLinebreaks: true});
|
let text: DocumentFragment | HTMLElement | string = wrapRichText(button.text, {noLinks: true, noLinebreaks: true});
|
||||||
|
|
||||||
let buttonEl: HTMLButtonElement | HTMLAnchorElement;
|
let buttonEl: HTMLButtonElement | HTMLAnchorElement;
|
||||||
|
|
||||||
@ -3596,14 +3595,14 @@ export default class ChatBubbles {
|
|||||||
});
|
});
|
||||||
|
|
||||||
buttonEl = htmlToDocumentFragment(r).firstElementChild as HTMLAnchorElement;
|
buttonEl = htmlToDocumentFragment(r).firstElementChild as HTMLAnchorElement;
|
||||||
buttonEl.classList.add('is-link', 'tgico');
|
buttonEl.classList.add('is-link');
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case 'keyboardButtonSwitchInline': {
|
case 'keyboardButtonSwitchInline': {
|
||||||
buttonEl = document.createElement('button');
|
buttonEl = document.createElement('button');
|
||||||
buttonEl.classList.add('is-switch-inline', 'tgico');
|
buttonEl.classList.add('is-switch-inline');
|
||||||
attachClickEvent(buttonEl, (e) => {
|
attachClickEvent(buttonEl, (e) => {
|
||||||
cancelEvent(e);
|
cancelEvent(e);
|
||||||
|
|
||||||
@ -3637,13 +3636,26 @@ export default class ChatBubbles {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case 'keyboardButtonBuy': {
|
||||||
|
buttonEl = document.createElement('button');
|
||||||
|
buttonEl.classList.add('is-buy');
|
||||||
|
|
||||||
|
if(messageMedia?._ === 'messageMediaInvoice') {
|
||||||
|
if(messageMedia.receipt_msg_id) {
|
||||||
|
text = i18n('Message.ReplyActionButtonShowReceipt');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
default: {
|
default: {
|
||||||
buttonEl = document.createElement('button');
|
buttonEl = document.createElement('button');
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
buttonEl.classList.add('reply-markup-button', 'rp');
|
buttonEl.classList.add('reply-markup-button', 'rp', 'tgico');
|
||||||
if(typeof(text) === 'string') {
|
if(typeof(text) === 'string') {
|
||||||
buttonEl.insertAdjacentHTML('beforeend', text);
|
buttonEl.insertAdjacentHTML('beforeend', text);
|
||||||
} else {
|
} else {
|
||||||
@ -4177,6 +4189,61 @@ export default class ChatBubbles {
|
|||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case 'messageMediaInvoice': {
|
||||||
|
const isTest = messageMedia.pFlags.test;
|
||||||
|
const photo = messageMedia.photo;
|
||||||
|
|
||||||
|
const priceEl = document.createElement(photo ? 'span' : 'div');
|
||||||
|
const f = document.createDocumentFragment();
|
||||||
|
const l = i18n(messageMedia.receipt_msg_id ? 'PaymentReceipt' : (isTest ? 'PaymentTestInvoice' : 'PaymentInvoice'));
|
||||||
|
l.classList.add('text-uppercase');
|
||||||
|
const joiner = ' ';
|
||||||
|
const p = document.createElement('span');
|
||||||
|
p.classList.add('text-bold');
|
||||||
|
p.textContent = paymentsWrapCurrencyAmount(messageMedia.total_amount, messageMedia.currency) + joiner;
|
||||||
|
f.append(p, l);
|
||||||
|
if(isTest && messageMedia.receipt_msg_id) {
|
||||||
|
const a = document.createElement('span');
|
||||||
|
a.classList.add('text-uppercase', 'pre-wrap');
|
||||||
|
a.append(joiner + '(Test)');
|
||||||
|
f.append(a);
|
||||||
|
}
|
||||||
|
setInnerHTML(priceEl, f);
|
||||||
|
|
||||||
|
if(photo) {
|
||||||
|
const mediaSize = mediaSizes.active.invoice;
|
||||||
|
wrapPhoto({
|
||||||
|
photo,
|
||||||
|
container: attachmentDiv,
|
||||||
|
withTail: false,
|
||||||
|
isOut,
|
||||||
|
lazyLoadQueue: this.lazyLoadQueue,
|
||||||
|
middleware: this.getMiddleware(),
|
||||||
|
loadPromises,
|
||||||
|
boxWidth: mediaSize.width,
|
||||||
|
boxHeight: mediaSize.height
|
||||||
|
});
|
||||||
|
|
||||||
|
bubble.classList.add('photo');
|
||||||
|
|
||||||
|
priceEl.classList.add('video-time');
|
||||||
|
attachmentDiv.append(priceEl);
|
||||||
|
} else {
|
||||||
|
attachmentDiv = undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
const titleDiv = document.createElement('div');
|
||||||
|
titleDiv.classList.add('bubble-primary-color');
|
||||||
|
setInnerHTML(titleDiv, wrapRichText(messageMedia.title));
|
||||||
|
|
||||||
|
const richText = wrapRichText(messageMedia.description);
|
||||||
|
messageDiv.prepend(...[titleDiv, !photo && priceEl, richText].filter(Boolean));
|
||||||
|
|
||||||
|
bubble.classList.remove('is-message-empty');
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
attachmentDiv = undefined;
|
attachmentDiv = undefined;
|
||||||
|
@ -9,7 +9,8 @@ import { formatTime } from "../../helpers/date";
|
|||||||
import htmlToSpan from "../../helpers/dom/htmlToSpan";
|
import htmlToSpan from "../../helpers/dom/htmlToSpan";
|
||||||
import setInnerHTML from "../../helpers/dom/setInnerHTML";
|
import setInnerHTML from "../../helpers/dom/setInnerHTML";
|
||||||
import formatCallDuration from "../../helpers/formatCallDuration";
|
import formatCallDuration from "../../helpers/formatCallDuration";
|
||||||
import { MessageAction } from "../../layer";
|
import paymentsWrapCurrencyAmount from "../../helpers/paymentsWrapCurrencyAmount";
|
||||||
|
import { Message, MessageAction } from "../../layer";
|
||||||
import { MyMessage } from "../../lib/appManagers/appMessagesManager";
|
import { MyMessage } from "../../lib/appManagers/appMessagesManager";
|
||||||
import I18n, { FormatterArgument, FormatterArguments, i18n, join, langPack, LangPackKey, _i18n } from "../../lib/langPack";
|
import I18n, { FormatterArgument, FormatterArguments, i18n, join, langPack, LangPackKey, _i18n } from "../../lib/langPack";
|
||||||
import wrapEmojiText from "../../lib/richTextProcessor/wrapEmojiText";
|
import wrapEmojiText from "../../lib/richTextProcessor/wrapEmojiText";
|
||||||
@ -21,6 +22,14 @@ import getPeerTitle from "./getPeerTitle";
|
|||||||
import wrapJoinVoiceChatAnchor from "./joinVoiceChatAnchor";
|
import wrapJoinVoiceChatAnchor from "./joinVoiceChatAnchor";
|
||||||
import wrapMessageForReply from "./messageForReply";
|
import wrapMessageForReply from "./messageForReply";
|
||||||
|
|
||||||
|
async function wrapLinkToMessage(message: Message.message | Message.messageService, plain?: boolean) {
|
||||||
|
const a = document.createElement('i');
|
||||||
|
a.dataset.savedFrom = message.peerId + '_' + message.mid;
|
||||||
|
a.dir = 'auto';
|
||||||
|
a.append(await wrapMessageForReply(message, undefined, undefined, plain as any));
|
||||||
|
return a;
|
||||||
|
}
|
||||||
|
|
||||||
export default async function wrapMessageActionTextNewUnsafe(message: MyMessage, plain?: boolean) {
|
export default async function wrapMessageActionTextNewUnsafe(message: MyMessage, plain?: boolean) {
|
||||||
const element: HTMLElement = plain ? undefined : document.createElement('span');
|
const element: HTMLElement = plain ? undefined : document.createElement('span');
|
||||||
const action = 'action' in message && message.action;
|
const action = 'action' in message && message.action;
|
||||||
@ -150,29 +159,10 @@ export default async function wrapMessageActionTextNewUnsafe(message: MyMessage,
|
|||||||
langPackKey = 'ActionPinnedNoText';
|
langPackKey = 'ActionPinnedNoText';
|
||||||
|
|
||||||
if(message.reply_to_mid) { // refresh original message
|
if(message.reply_to_mid) { // refresh original message
|
||||||
managers.appMessagesManager.fetchMessageReplyTo(message).then(async(originalMessage) => {
|
managers.appMessagesManager.fetchMessageReplyTo(message);
|
||||||
if(originalMessage && message) {
|
|
||||||
rootScope.dispatchEvent('message_edit', {
|
|
||||||
storageKey: `${peerId}_history`,
|
|
||||||
peerId: peerId,
|
|
||||||
mid: message.mid,
|
|
||||||
message
|
|
||||||
});
|
|
||||||
|
|
||||||
if(managers.appMessagesManager.isMessageIsTopMessage(message)) {
|
|
||||||
rootScope.dispatchEvent('dialogs_multiupdate', {
|
|
||||||
[peerId]: await managers.appMessagesManager.getDialogOnly(peerId)
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
const a = document.createElement('i');
|
args.push(wrapLinkToMessage(pinnedMessage, plain));
|
||||||
a.dataset.savedFrom = pinnedMessage.peerId + '_' + pinnedMessage.mid;
|
|
||||||
a.dir = 'auto';
|
|
||||||
a.append(await wrapMessageForReply(pinnedMessage, undefined, undefined, plain as any));
|
|
||||||
args.push(a);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
@ -258,6 +248,24 @@ export default async function wrapMessageActionTextNewUnsafe(message: MyMessage,
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case 'messageActionPaymentSent': {
|
||||||
|
langPackKey = 'PaymentSuccessfullyPaidNoItem';
|
||||||
|
const price = paymentsWrapCurrencyAmount(action.total_amount, action.currency);
|
||||||
|
args = [price, getNameDivHTML(message.peerId, plain)];
|
||||||
|
|
||||||
|
if(message.reply_to_mid) {
|
||||||
|
const invoiceMessage = await managers.appMessagesManager.getMessageByPeer(message.peerId, message.reply_to_mid);
|
||||||
|
if(!invoiceMessage) {
|
||||||
|
managers.appMessagesManager.fetchMessageReplyTo(message);
|
||||||
|
} else {
|
||||||
|
langPackKey = 'PaymentSuccessfullyPaid';
|
||||||
|
args.push(wrapLinkToMessage(invoiceMessage, plain));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
langPackKey = (langPack[_] || `[${action._}]`) as any;
|
langPackKey = (langPack[_] || `[${action._}]`) as any;
|
||||||
break;
|
break;
|
||||||
|
@ -158,6 +158,11 @@ export default async function wrapMessageForReply(message: MyMessage | MyDraftMe
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case 'messageMediaInvoice': {
|
||||||
|
addPart(undefined, plain ? media.title : wrapEmojiText(media.title));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
case 'messageMediaUnsupported': {
|
case 'messageMediaUnsupported': {
|
||||||
addPart(UNSUPPORTED_LANG_PACK_KEY);
|
addPart(UNSUPPORTED_LANG_PACK_KEY);
|
||||||
break;
|
break;
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
|
|
||||||
import renderImageWithFadeIn from "../../helpers/dom/renderImageWithFadeIn";
|
import renderImageWithFadeIn from "../../helpers/dom/renderImageWithFadeIn";
|
||||||
import mediaSizes from "../../helpers/mediaSizes";
|
import mediaSizes from "../../helpers/mediaSizes";
|
||||||
import { Message, PhotoSize } from "../../layer";
|
import { Message, PhotoSize, WebDocument } from "../../layer";
|
||||||
import { MyDocument } from "../../lib/appManagers/appDocsManager";
|
import { MyDocument } from "../../lib/appManagers/appDocsManager";
|
||||||
import { MyPhoto } from "../../lib/appManagers/appPhotosManager";
|
import { MyPhoto } from "../../lib/appManagers/appPhotosManager";
|
||||||
import rootScope from "../../lib/rootScope";
|
import rootScope from "../../lib/rootScope";
|
||||||
@ -19,9 +19,10 @@ import setAttachmentSize from "../../helpers/setAttachmentSize";
|
|||||||
import choosePhotoSize from "../../lib/appManagers/utils/photos/choosePhotoSize";
|
import choosePhotoSize from "../../lib/appManagers/utils/photos/choosePhotoSize";
|
||||||
import type { ThumbCache } from "../../lib/storages/thumbs";
|
import type { ThumbCache } from "../../lib/storages/thumbs";
|
||||||
import appDownloadManager from "../../lib/appManagers/appDownloadManager";
|
import appDownloadManager from "../../lib/appManagers/appDownloadManager";
|
||||||
|
import isWebDocument from "../../lib/appManagers/utils/webDocs/isWebDocument";
|
||||||
|
|
||||||
export default async function wrapPhoto({photo, message, container, boxWidth, boxHeight, withTail, isOut, lazyLoadQueue, middleware, size, withoutPreloader, loadPromises, autoDownloadSize, noBlur, noThumb, noFadeIn, blurAfter, managers = rootScope.managers}: {
|
export default async function wrapPhoto({photo, message, container, boxWidth, boxHeight, withTail, isOut, lazyLoadQueue, middleware, size, withoutPreloader, loadPromises, autoDownloadSize, noBlur, noThumb, noFadeIn, blurAfter, managers = rootScope.managers}: {
|
||||||
photo: MyPhoto | MyDocument,
|
photo: MyPhoto | MyDocument | WebDocument,
|
||||||
message?: Message.message | Message.messageService,
|
message?: Message.message | Message.messageService,
|
||||||
container: HTMLElement,
|
container: HTMLElement,
|
||||||
boxWidth?: number,
|
boxWidth?: number,
|
||||||
@ -40,7 +41,8 @@ export default async function wrapPhoto({photo, message, container, boxWidth, bo
|
|||||||
blurAfter?: boolean,
|
blurAfter?: boolean,
|
||||||
managers?: AppManagers,
|
managers?: AppManagers,
|
||||||
}) {
|
}) {
|
||||||
if(!((photo as MyPhoto).sizes || (photo as MyDocument).thumbs)) {
|
const isWebDoc = isWebDocument(photo);
|
||||||
|
if(!((photo as MyPhoto).sizes || (photo as MyDocument).thumbs) && !isWebDoc) {
|
||||||
if(boxWidth && boxHeight && !size && photo._ === 'document') {
|
if(boxWidth && boxHeight && !size && photo._ === 'document') {
|
||||||
setAttachmentSize(photo, container, boxWidth, boxHeight, undefined, message);
|
setAttachmentSize(photo, container, boxWidth, boxHeight, undefined, message);
|
||||||
}
|
}
|
||||||
@ -92,7 +94,7 @@ export default async function wrapPhoto({photo, message, container, boxWidth, bo
|
|||||||
isFit = set.isFit;
|
isFit = set.isFit;
|
||||||
cacheContext = await managers.thumbsStorage.getCacheContext(photo, size.type);
|
cacheContext = await managers.thumbsStorage.getCacheContext(photo, size.type);
|
||||||
|
|
||||||
if(!isFit) {
|
if(!isFit && !isWebDoc) {
|
||||||
aspecter = document.createElement('div');
|
aspecter = document.createElement('div');
|
||||||
aspecter.classList.add('media-container-aspecter');
|
aspecter.classList.add('media-container-aspecter');
|
||||||
aspecter.style.width = set.size.width + 'px';
|
aspecter.style.width = set.size.width + 'px';
|
||||||
@ -141,7 +143,7 @@ export default async function wrapPhoto({photo, message, container, boxWidth, bo
|
|||||||
cacheContext = await managers.thumbsStorage.getCacheContext(photo, size?.type);
|
cacheContext = await managers.thumbsStorage.getCacheContext(photo, size?.type);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!noThumb) {
|
if(!noThumb && !isWebDoc) {
|
||||||
const gotThumb = getStrippedThumbIfNeeded(photo, cacheContext, !noBlur);
|
const gotThumb = getStrippedThumbIfNeeded(photo, cacheContext, !noBlur);
|
||||||
if(gotThumb) {
|
if(gotThumb) {
|
||||||
loadThumbPromise = Promise.all([loadThumbPromise, gotThumb.loadPromise]);
|
loadThumbPromise = Promise.all([loadThumbPromise, gotThumb.loadPromise]);
|
||||||
|
18
src/config/currencies.ts
Normal file
18
src/config/currencies.ts
Normal file
File diff suppressed because one or more lines are too long
@ -18,7 +18,8 @@ type MediaTypeSizes = {
|
|||||||
emojiSticker: MediaSize,
|
emojiSticker: MediaSize,
|
||||||
poll: MediaSize,
|
poll: MediaSize,
|
||||||
round: MediaSize,
|
round: MediaSize,
|
||||||
documentName: MediaSize
|
documentName: MediaSize,
|
||||||
|
invoice: MediaSize
|
||||||
};
|
};
|
||||||
|
|
||||||
export type MediaSizeType = keyof MediaTypeSizes;
|
export type MediaSizeType = keyof MediaTypeSizes;
|
||||||
@ -54,7 +55,8 @@ class MediaSizes extends EventListenerBase<{
|
|||||||
emojiSticker: makeMediaSize(112, 112),
|
emojiSticker: makeMediaSize(112, 112),
|
||||||
poll: makeMediaSize(240, 0),
|
poll: makeMediaSize(240, 0),
|
||||||
round: makeMediaSize(200, 200),
|
round: makeMediaSize(200, 200),
|
||||||
documentName: makeMediaSize(200, 0)
|
documentName: makeMediaSize(200, 0),
|
||||||
|
invoice: makeMediaSize(240, 240)
|
||||||
},
|
},
|
||||||
desktop: {
|
desktop: {
|
||||||
regular: makeMediaSize(420, 340),
|
regular: makeMediaSize(420, 340),
|
||||||
@ -66,7 +68,8 @@ class MediaSizes extends EventListenerBase<{
|
|||||||
emojiSticker: makeMediaSize(112, 112),
|
emojiSticker: makeMediaSize(112, 112),
|
||||||
poll: makeMediaSize(330, 0),
|
poll: makeMediaSize(330, 0),
|
||||||
round: makeMediaSize(280, 280),
|
round: makeMediaSize(280, 280),
|
||||||
documentName: makeMediaSize(240, 0)
|
documentName: makeMediaSize(240, 0),
|
||||||
|
invoice: makeMediaSize(320, 260)
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
70
src/helpers/paymentsWrapCurrencyAmount.ts
Normal file
70
src/helpers/paymentsWrapCurrencyAmount.ts
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
import Currencies from "../config/currencies";
|
||||||
|
|
||||||
|
// https://stackoverflow.com/a/34141813
|
||||||
|
function number_format(number: any, decimals: any, dec_point: any, thousands_sep: any) {
|
||||||
|
// Strip all characters but numerical ones.
|
||||||
|
number = (number + '').replace(/[^0-9+\-Ee.]/g, '');
|
||||||
|
var n = !isFinite(+number) ? 0 : +number,
|
||||||
|
prec = !isFinite(+decimals) ? 0 : Math.abs(decimals),
|
||||||
|
sep = (typeof thousands_sep === 'undefined') ? ',' : thousands_sep,
|
||||||
|
dec = (typeof dec_point === 'undefined') ? '.' : dec_point,
|
||||||
|
s: any = '',
|
||||||
|
toFixedFix = function(n: number, prec: number) {
|
||||||
|
var k = Math.pow(10, prec);
|
||||||
|
return '' + Math.round(n * k) / k;
|
||||||
|
};
|
||||||
|
// Fix for IE parseFloat(0.55).toFixed(0) = 0;
|
||||||
|
s = (prec ? toFixedFix(n, prec) : '' + Math.round(n)).split('.');
|
||||||
|
if (s[0].length > 3) {
|
||||||
|
s[0] = s[0].replace(/\B(?=(?:\d{3})+(?!\d))/g, sep);
|
||||||
|
}
|
||||||
|
if ((s[1] || '').length < prec) {
|
||||||
|
s[1] = s[1] || '';
|
||||||
|
s[1] += new Array(prec - s[1].length + 1).join('0');
|
||||||
|
}
|
||||||
|
return s.join(dec);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function paymentsWrapCurrencyAmount($amount: number | string, $currency: string) {
|
||||||
|
$amount = +$amount;
|
||||||
|
|
||||||
|
const $currency_data = Currencies[$currency]; // вытащить из json
|
||||||
|
if(!$currency_data) {
|
||||||
|
throw new Error('CURRENCY_WRAP_INVALID');
|
||||||
|
}
|
||||||
|
|
||||||
|
const $amount_exp = $amount / Math.pow(10, $currency_data['exp']);
|
||||||
|
|
||||||
|
let $decimals = $currency_data['exp'];
|
||||||
|
if($currency == 'IRR' &&
|
||||||
|
Math.floor($amount_exp) == $amount_exp) {
|
||||||
|
$decimals = 0; // у иранцев копейки почти всегда = 0 и не показываются в UI
|
||||||
|
}
|
||||||
|
|
||||||
|
const $formatted = number_format($amount_exp, $decimals, $currency_data['decimal_sep'], $currency_data['thousands_sep']);
|
||||||
|
|
||||||
|
const $splitter = $currency_data['space_between'] ? "\xc2\xa0" : '';
|
||||||
|
let $formatted_intern: string;
|
||||||
|
if($currency_data['symbol_left']) {
|
||||||
|
$formatted_intern = $currency_data['symbol'] + $splitter + $formatted;
|
||||||
|
} else {
|
||||||
|
$formatted_intern = $formatted + $splitter + $currency_data['symbol'];
|
||||||
|
}
|
||||||
|
return $formatted_intern;
|
||||||
|
}
|
||||||
|
|
||||||
|
function paymentsGetCurrencyExp($currency: string) {
|
||||||
|
if($currency == 'CLF') {
|
||||||
|
return 4;
|
||||||
|
}
|
||||||
|
if(['BHD','IQD','JOD','KWD','LYD','OMR','TND'].includes($currency)) {
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
if(['BIF','BYR','CLP','CVE','DJF','GNF','ISK','JPY','KMF','KRW','MGA', 'PYG','RWF','UGX','UYI','VND','VUV','XAF','XOF','XPF'].includes($currency)) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if($currency == 'MRO') {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return 2;
|
||||||
|
}
|
@ -4,15 +4,16 @@
|
|||||||
* https://github.com/morethanwords/tweb/blob/master/LICENSE
|
* https://github.com/morethanwords/tweb/blob/master/LICENSE
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { PhotoSize } from "../layer";
|
import { PhotoSize, WebDocument } from "../layer";
|
||||||
import { REPLIES_HIDDEN_CHANNEL_ID } from "../lib/mtproto/mtproto_config";
|
import { REPLIES_HIDDEN_CHANNEL_ID } from "../lib/mtproto/mtproto_config";
|
||||||
import { MyDocument } from "../lib/appManagers/appDocsManager";
|
import { MyDocument } from "../lib/appManagers/appDocsManager";
|
||||||
import { MyPhoto } from "../lib/appManagers/appPhotosManager";
|
import { MyPhoto } from "../lib/appManagers/appPhotosManager";
|
||||||
import choosePhotoSize from "../lib/appManagers/utils/photos/choosePhotoSize";
|
import choosePhotoSize from "../lib/appManagers/utils/photos/choosePhotoSize";
|
||||||
import { MediaSize, makeMediaSize } from "./mediaSize";
|
import { MediaSize, makeMediaSize } from "./mediaSize";
|
||||||
|
import isWebDocument from "../lib/appManagers/utils/webDocs/isWebDocument";
|
||||||
|
|
||||||
export default function setAttachmentSize(
|
export default function setAttachmentSize(
|
||||||
photo: MyPhoto | MyDocument,
|
photo: MyPhoto | MyDocument | WebDocument,
|
||||||
element: HTMLElement | SVGForeignObjectElement,
|
element: HTMLElement | SVGForeignObjectElement,
|
||||||
boxWidth: number,
|
boxWidth: number,
|
||||||
boxHeight: number,
|
boxHeight: number,
|
||||||
@ -21,6 +22,11 @@ export default function setAttachmentSize(
|
|||||||
pushDocumentSize?: boolean,
|
pushDocumentSize?: boolean,
|
||||||
photoSize?: ReturnType<typeof choosePhotoSize>
|
photoSize?: ReturnType<typeof choosePhotoSize>
|
||||||
) {
|
) {
|
||||||
|
const _isWebDocument = isWebDocument(photo);
|
||||||
|
// if(_isWebDocument && pushDocumentSize === undefined) {
|
||||||
|
// pushDocumentSize = true;
|
||||||
|
// }
|
||||||
|
|
||||||
if(!photoSize) {
|
if(!photoSize) {
|
||||||
photoSize = choosePhotoSize(photo, boxWidth, boxHeight, undefined, pushDocumentSize);
|
photoSize = choosePhotoSize(photo, boxWidth, boxHeight, undefined, pushDocumentSize);
|
||||||
}
|
}
|
||||||
@ -28,8 +34,8 @@ export default function setAttachmentSize(
|
|||||||
|
|
||||||
let size: MediaSize;
|
let size: MediaSize;
|
||||||
const isDocument = photo._ === 'document';
|
const isDocument = photo._ === 'document';
|
||||||
if(isDocument) {
|
if(isDocument || _isWebDocument) {
|
||||||
size = makeMediaSize((photo as MyDocument).w || (photoSize as PhotoSize.photoSize).w || 512, (photo as MyDocument).h || (photoSize as PhotoSize.photoSize).h || 512);
|
size = makeMediaSize(photo.w || (photoSize as PhotoSize.photoSize).w || 512, photo.h || (photoSize as PhotoSize.photoSize).h || 512);
|
||||||
} else {
|
} else {
|
||||||
size = makeMediaSize((photoSize as PhotoSize.photoSize).w || 100, (photoSize as PhotoSize.photoSize).h || 100);
|
size = makeMediaSize((photoSize as PhotoSize.photoSize).w || 100, (photoSize as PhotoSize.photoSize).h || 100);
|
||||||
}
|
}
|
||||||
@ -40,7 +46,7 @@ export default function setAttachmentSize(
|
|||||||
|
|
||||||
let isFit = true;
|
let isFit = true;
|
||||||
|
|
||||||
if(!isDocument || ['video', 'gif'].includes((photo as MyDocument).type)) {
|
if(!isDocument || ['video', 'gif'].includes(photo.type) || _isWebDocument) {
|
||||||
if(boxSize.width < 200 && boxSize.height < 200) { // make at least one side this big
|
if(boxSize.width < 200 && boxSize.height < 200) { // make at least one side this big
|
||||||
boxSize = size = size.aspectCovered(makeMediaSize(200, 200));
|
boxSize = size = size.aspectCovered(makeMediaSize(200, 200));
|
||||||
}
|
}
|
||||||
|
@ -695,6 +695,13 @@ const lang = {
|
|||||||
"ScamMessage": "SCAM",
|
"ScamMessage": "SCAM",
|
||||||
"FakeMessage": "FAKE",
|
"FakeMessage": "FAKE",
|
||||||
"TextCopied": "Text copied to clipboard",
|
"TextCopied": "Text copied to clipboard",
|
||||||
|
"PaymentInvoice": "INVOICE",
|
||||||
|
"PaymentTestInvoice": "TEST INVOICE",
|
||||||
|
"PaymentReceipt": "Receipt",
|
||||||
|
"PaymentSuccessfullyPaid": "You successfully transferred %1$s to %2$s for %3$s",
|
||||||
|
"PaymentSuccessfullyPaidNoItem": "You successfully transferred %1$s to %2$s",
|
||||||
|
// "PaymentSuccessfullyPaidRecurrent": "You successfully transferred %1$s to %2$s for %3$s and allowed future recurring payments",
|
||||||
|
// "PaymentSuccessfullyPaidNoItemRecurrent": "You successfully transferred %1$s to %2$s and allowed future recurring payments",
|
||||||
|
|
||||||
// * macos
|
// * macos
|
||||||
"AccountSettings.Filters": "Chat Folders",
|
"AccountSettings.Filters": "Chat Folders",
|
||||||
@ -913,6 +920,7 @@ const lang = {
|
|||||||
"Message.Context.Pin": "Pin",
|
"Message.Context.Pin": "Pin",
|
||||||
"Message.Context.Unpin": "Unpin",
|
"Message.Context.Unpin": "Unpin",
|
||||||
"Message.Context.Goto": "Show Message",
|
"Message.Context.Goto": "Show Message",
|
||||||
|
"Message.ReplyActionButtonShowReceipt": "Show Receipt",
|
||||||
"MessageContext.CopyMessageLink1": "Copy Message Link",
|
"MessageContext.CopyMessageLink1": "Copy Message Link",
|
||||||
"Modal.Send": "Send",
|
"Modal.Send": "Send",
|
||||||
"NewPoll.Anonymous": "Anonymous Voting",
|
"NewPoll.Anonymous": "Anonymous Voting",
|
||||||
|
8
src/layer.d.ts
vendored
8
src/layer.d.ts
vendored
@ -5881,7 +5881,9 @@ export namespace WebDocument {
|
|||||||
access_hash: string | number,
|
access_hash: string | number,
|
||||||
size: number,
|
size: number,
|
||||||
mime_type: string,
|
mime_type: string,
|
||||||
attributes: Array<DocumentAttribute>
|
attributes: Array<DocumentAttribute>,
|
||||||
|
h?: number,
|
||||||
|
w?: number
|
||||||
};
|
};
|
||||||
|
|
||||||
export type webDocumentNoProxy = {
|
export type webDocumentNoProxy = {
|
||||||
@ -5889,7 +5891,9 @@ export namespace WebDocument {
|
|||||||
url: string,
|
url: string,
|
||||||
size: number,
|
size: number,
|
||||||
mime_type: string,
|
mime_type: string,
|
||||||
attributes: Array<DocumentAttribute>
|
attributes: Array<DocumentAttribute>,
|
||||||
|
h?: number,
|
||||||
|
w?: number
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
|
|
||||||
import type { ApiFileManager, DownloadMediaOptions, DownloadOptions } from "../mtproto/apiFileManager";
|
import type { ApiFileManager, DownloadMediaOptions, DownloadOptions } from "../mtproto/apiFileManager";
|
||||||
import deferredPromise, { CancellablePromise } from "../../helpers/cancellablePromise";
|
import deferredPromise, { CancellablePromise } from "../../helpers/cancellablePromise";
|
||||||
import { Document, InputFile, Photo, PhotoSize } from "../../layer";
|
import { Document, InputFile, Photo, PhotoSize, WebDocument } from "../../layer";
|
||||||
import { getFileNameByLocation } from "../../helpers/fileName";
|
import { getFileNameByLocation } from "../../helpers/fileName";
|
||||||
import getFileNameForUpload from "../../helpers/getFileNameForUpload";
|
import getFileNameForUpload from "../../helpers/getFileNameForUpload";
|
||||||
import { AppManagers } from "./managers";
|
import { AppManagers } from "./managers";
|
||||||
|
@ -2759,8 +2759,7 @@ export class AppMessagesManager extends AppManager {
|
|||||||
break; */
|
break; */
|
||||||
|
|
||||||
case 'messageMediaInvoice': {
|
case 'messageMediaInvoice': {
|
||||||
unsupported = true;
|
message.media.photo = this.appWebDocsManager.saveWebDocument(message.media.photo);
|
||||||
message.media = {_: 'messageMediaUnsupported'};
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -5630,6 +5629,22 @@ export class AppMessagesManager extends AppManager {
|
|||||||
delete message.reply_to_mid; // ! WARNING!
|
delete message.reply_to_mid; // ! WARNING!
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(message._ === 'messageService') {
|
||||||
|
const peerId = message.peerId;
|
||||||
|
this.rootScope.dispatchEvent('message_edit', {
|
||||||
|
storageKey: `${peerId}_history`,
|
||||||
|
peerId: peerId,
|
||||||
|
mid: message.mid,
|
||||||
|
message
|
||||||
|
});
|
||||||
|
|
||||||
|
if(this.isMessageIsTopMessage(message)) {
|
||||||
|
this.rootScope.dispatchEvent('dialogs_multiupdate', {
|
||||||
|
[peerId]: this.getDialogOnly(peerId)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return originalMessage;
|
return originalMessage;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
23
src/lib/appManagers/appWebDocsManager.ts
Normal file
23
src/lib/appManagers/appWebDocsManager.ts
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
/*
|
||||||
|
* https://github.com/morethanwords/tweb
|
||||||
|
* Copyright (C) 2019-2021 Eduard Kuzmenko
|
||||||
|
* https://github.com/morethanwords/tweb/blob/master/LICENSE
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { DocumentAttribute, WebDocument } from "../../layer";
|
||||||
|
|
||||||
|
export default class AppWebDocsManager {
|
||||||
|
public saveWebDocument(webDocument: WebDocument) {
|
||||||
|
if(!webDocument) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const attribute: DocumentAttribute.documentAttributeImageSize = webDocument.attributes.find((attribute) => attribute._ === 'documentAttributeImageSize') as any;
|
||||||
|
if(attribute) {
|
||||||
|
webDocument.w = attribute.w;
|
||||||
|
webDocument.h = attribute.h;
|
||||||
|
}
|
||||||
|
|
||||||
|
return webDocument;
|
||||||
|
}
|
||||||
|
}
|
@ -43,6 +43,7 @@ import { AppStoragesManager } from "./appStoragesManager";
|
|||||||
import cryptoMessagePort from "../crypto/cryptoMessagePort";
|
import cryptoMessagePort from "../crypto/cryptoMessagePort";
|
||||||
import appStateManager from "./appStateManager";
|
import appStateManager from "./appStateManager";
|
||||||
import filterUnique from "../../helpers/array/filterUnique";
|
import filterUnique from "../../helpers/array/filterUnique";
|
||||||
|
import AppWebDocsManager from "./appWebDocsManager";
|
||||||
|
|
||||||
export default function createManagers(appStoragesManager: AppStoragesManager, userId: UserId) {
|
export default function createManagers(appStoragesManager: AppStoragesManager, userId: UserId) {
|
||||||
const managers = {
|
const managers = {
|
||||||
@ -82,7 +83,8 @@ export default function createManagers(appStoragesManager: AppStoragesManager, u
|
|||||||
dcConfigurator: new DcConfigurator,
|
dcConfigurator: new DcConfigurator,
|
||||||
timeManager: new TimeManager,
|
timeManager: new TimeManager,
|
||||||
appStoragesManager: appStoragesManager,
|
appStoragesManager: appStoragesManager,
|
||||||
appStateManager: appStateManager
|
appStateManager: appStateManager,
|
||||||
|
appWebDocsManager: new AppWebDocsManager
|
||||||
};
|
};
|
||||||
|
|
||||||
type T = typeof managers;
|
type T = typeof managers;
|
||||||
|
@ -40,6 +40,7 @@ import type { AppStateManager } from "./appStateManager";
|
|||||||
import type { AppStickersManager } from "./appStickersManager";
|
import type { AppStickersManager } from "./appStickersManager";
|
||||||
import type { AppStoragesManager } from "./appStoragesManager";
|
import type { AppStoragesManager } from "./appStoragesManager";
|
||||||
import type { AppUsersManager } from "./appUsersManager";
|
import type { AppUsersManager } from "./appUsersManager";
|
||||||
|
import type AppWebDocsManager from "./appWebDocsManager";
|
||||||
import type { AppWebPagesManager } from "./appWebPagesManager";
|
import type { AppWebPagesManager } from "./appWebPagesManager";
|
||||||
import type { AppManagers } from "./managers";
|
import type { AppManagers } from "./managers";
|
||||||
|
|
||||||
@ -82,6 +83,7 @@ export class AppManager {
|
|||||||
protected timeManager: TimeManager;
|
protected timeManager: TimeManager;
|
||||||
protected appStoragesManager: AppStoragesManager;
|
protected appStoragesManager: AppStoragesManager;
|
||||||
protected appStateManager: AppStateManager;
|
protected appStateManager: AppStateManager;
|
||||||
|
protected appWebDocsManager: AppWebDocsManager;
|
||||||
|
|
||||||
public clear: (init?: boolean) => void;
|
public clear: (init?: boolean) => void;
|
||||||
|
|
||||||
|
@ -4,14 +4,21 @@
|
|||||||
* https://github.com/morethanwords/tweb/blob/master/LICENSE
|
* https://github.com/morethanwords/tweb/blob/master/LICENSE
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import type { DownloadMediaOptions } from "../../../mtproto/apiFileManager";
|
import type { DownloadMediaOptions, DownloadOptions } from "../../../mtproto/apiFileManager";
|
||||||
import getDocumentDownloadOptions from "../docs/getDocumentDownloadOptions";
|
import getDocumentDownloadOptions from "../docs/getDocumentDownloadOptions";
|
||||||
import getPhotoDownloadOptions from "../photos/getPhotoDownloadOptions";
|
import getPhotoDownloadOptions from "../photos/getPhotoDownloadOptions";
|
||||||
|
import getWebDocumentDownloadOptions from "../webDocs/getWebDocumentDownloadOptions";
|
||||||
|
import isWebDocument from "../webDocs/isWebDocument";
|
||||||
import getDownloadFileNameFromOptions from "./getDownloadFileNameFromOptions";
|
import getDownloadFileNameFromOptions from "./getDownloadFileNameFromOptions";
|
||||||
|
|
||||||
export default function getDownloadMediaDetails(options: DownloadMediaOptions) {
|
export default function getDownloadMediaDetails(options: DownloadMediaOptions) {
|
||||||
const {media, thumb, queueId, onlyCache} = options;
|
const {media, thumb, queueId, onlyCache} = options;
|
||||||
const downloadOptions = media._ === 'document' ? getDocumentDownloadOptions(media, thumb as any, queueId, onlyCache) : getPhotoDownloadOptions(media as any, thumb, queueId, onlyCache);
|
|
||||||
|
let downloadOptions: DownloadOptions;
|
||||||
|
if(media._ === 'document') downloadOptions = getDocumentDownloadOptions(media, thumb as any, queueId, onlyCache);
|
||||||
|
else if(media._ === 'photo') downloadOptions = getPhotoDownloadOptions(media, thumb, queueId, onlyCache);
|
||||||
|
else if(isWebDocument(media)) downloadOptions = getWebDocumentDownloadOptions(media);
|
||||||
|
|
||||||
const fileName = getDownloadFileNameFromOptions(downloadOptions);
|
const fileName = getDownloadFileNameFromOptions(downloadOptions);
|
||||||
return {fileName, downloadOptions};
|
return {fileName, downloadOptions};
|
||||||
}
|
}
|
||||||
|
@ -6,11 +6,11 @@
|
|||||||
|
|
||||||
import type { MyDocument } from "../../appDocsManager";
|
import type { MyDocument } from "../../appDocsManager";
|
||||||
import type { MyPhoto } from "../../appPhotosManager";
|
import type { MyPhoto } from "../../appPhotosManager";
|
||||||
|
import type { PhotoSize, WebDocument } from "../../../../layer";
|
||||||
import calcImageInBox from "../../../../helpers/calcImageInBox";
|
import calcImageInBox from "../../../../helpers/calcImageInBox";
|
||||||
import { PhotoSize } from "../../../../layer";
|
|
||||||
|
|
||||||
export default function choosePhotoSize(
|
export default function choosePhotoSize(
|
||||||
photo: MyPhoto | MyDocument,
|
photo: MyPhoto | MyDocument | WebDocument,
|
||||||
boxWidth = 0,
|
boxWidth = 0,
|
||||||
boxHeight = 0,
|
boxHeight = 0,
|
||||||
useBytes = false,
|
useBytes = false,
|
||||||
@ -34,12 +34,12 @@ export default function choosePhotoSize(
|
|||||||
|
|
||||||
let bestPhotoSize: PhotoSize = {_: 'photoSizeEmpty', type: ''};
|
let bestPhotoSize: PhotoSize = {_: 'photoSizeEmpty', type: ''};
|
||||||
let sizes = (photo as MyPhoto).sizes || (photo as MyDocument).thumbs as PhotoSize[];
|
let sizes = (photo as MyPhoto).sizes || (photo as MyDocument).thumbs as PhotoSize[];
|
||||||
if(pushDocumentSize && sizes && photo._ === 'document') {
|
if(pushDocumentSize && sizes && photo._ !== 'photo') {
|
||||||
sizes = sizes.concat({
|
sizes = sizes.concat({
|
||||||
_: 'photoSize',
|
_: 'photoSize',
|
||||||
w: (photo as MyDocument).w,
|
w: photo.w,
|
||||||
h: (photo as MyDocument).h,
|
h: photo.h,
|
||||||
size: (photo as MyDocument).size,
|
size: photo.size,
|
||||||
type: undefined
|
type: undefined
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,15 @@
|
|||||||
|
import { WebDocument } from "../../../../layer";
|
||||||
|
import { DownloadOptions } from "../../../mtproto/apiFileManager";
|
||||||
|
|
||||||
|
export default function getWebDocumentDownloadOptions(webDocument: WebDocument): DownloadOptions {
|
||||||
|
return {
|
||||||
|
dcId: 4,
|
||||||
|
location: {
|
||||||
|
_: 'inputWebFileLocation',
|
||||||
|
access_hash: (webDocument as WebDocument.webDocument).access_hash,
|
||||||
|
url: webDocument.url
|
||||||
|
},
|
||||||
|
size: webDocument.size,
|
||||||
|
mimeType: webDocument.mime_type
|
||||||
|
};
|
||||||
|
}
|
5
src/lib/appManagers/utils/webDocs/isWebDocument.ts
Normal file
5
src/lib/appManagers/utils/webDocs/isWebDocument.ts
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
import { WebDocument } from "../../../../layer";
|
||||||
|
|
||||||
|
export default function isWebDocument(webDocument: any): webDocument is WebDocument {
|
||||||
|
return !!(webDocument && (webDocument._ === 'webDocument' || webDocument._ === 'webDocumentNoProxy'));
|
||||||
|
}
|
@ -34,6 +34,14 @@ export class FileManager {
|
|||||||
throw false;
|
throw false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// sometimes file size can be bigger than the prov
|
||||||
|
const endOffset = offset + part.byteLength;
|
||||||
|
if(endOffset > bytes.byteLength) {
|
||||||
|
const newBytes = new Uint8Array(endOffset);
|
||||||
|
newBytes.set(bytes, 0);
|
||||||
|
bytes = newBytes;
|
||||||
|
}
|
||||||
|
|
||||||
bytes.set(part, offset);
|
bytes.set(part, offset);
|
||||||
},
|
},
|
||||||
truncate: () => {
|
truncate: () => {
|
||||||
|
@ -64,7 +64,7 @@ export const langPack: {[actionType: string]: LangPackKey} = {
|
|||||||
"messageActionGroupCall.ended_by": "Chat.Service.VoiceChatFinished",
|
"messageActionGroupCall.ended_by": "Chat.Service.VoiceChatFinished",
|
||||||
"messageActionGroupCall.ended_byYou": "Chat.Service.VoiceChatFinishedYou",
|
"messageActionGroupCall.ended_byYou": "Chat.Service.VoiceChatFinishedYou",
|
||||||
|
|
||||||
"messageActionBotAllowed": "Chat.Service.BotPermissionAllowed"
|
"messageActionBotAllowed": "Chat.Service.BotPermissionAllowed",
|
||||||
};
|
};
|
||||||
|
|
||||||
export type LangPackKey = /* string | */keyof typeof lang | keyof typeof langSign;
|
export type LangPackKey = /* string | */keyof typeof lang | keyof typeof langSign;
|
||||||
|
@ -14,7 +14,7 @@ import Modes from "../../config/modes";
|
|||||||
import deferredPromise, { CancellablePromise } from "../../helpers/cancellablePromise";
|
import deferredPromise, { CancellablePromise } from "../../helpers/cancellablePromise";
|
||||||
import { getFileNameByLocation } from "../../helpers/fileName";
|
import { getFileNameByLocation } from "../../helpers/fileName";
|
||||||
import { randomLong } from "../../helpers/random";
|
import { randomLong } from "../../helpers/random";
|
||||||
import { Document, InputFile, InputFileLocation, InputWebFileLocation, Photo, PhotoSize, UploadFile, UploadWebFile } from "../../layer";
|
import { Document, InputFile, InputFileLocation, InputWebFileLocation, Photo, PhotoSize, UploadFile, UploadWebFile, WebDocument } from "../../layer";
|
||||||
import { DcId } from "../../types";
|
import { DcId } from "../../types";
|
||||||
import CacheStorageController from "../cacheStorage";
|
import CacheStorageController from "../cacheStorage";
|
||||||
import fileManager from "../fileManager";
|
import fileManager from "../fileManager";
|
||||||
@ -54,7 +54,7 @@ export type DownloadOptions = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export type DownloadMediaOptions = {
|
export type DownloadMediaOptions = {
|
||||||
media: Photo | Document.document,
|
media: Photo.photo | Document.document | WebDocument,
|
||||||
thumb?: PhotoSize,
|
thumb?: PhotoSize,
|
||||||
queueId?: number,
|
queueId?: number,
|
||||||
onlyCache?: boolean
|
onlyCache?: boolean
|
||||||
@ -599,14 +599,15 @@ export class ApiFileManager extends AppManager {
|
|||||||
public downloadMedia(options: DownloadMediaOptions): DownloadPromise {
|
public downloadMedia(options: DownloadMediaOptions): DownloadPromise {
|
||||||
let {media, thumb} = options;
|
let {media, thumb} = options;
|
||||||
const isPhoto = media._ === 'photo';
|
const isPhoto = media._ === 'photo';
|
||||||
if(media._ === 'photoEmpty' || (isPhoto && !thumb)) {
|
if(isPhoto && !thumb) {
|
||||||
return Promise.reject('preloadPhoto photoEmpty!');
|
return Promise.reject('preloadPhoto photoEmpty!');
|
||||||
}
|
}
|
||||||
|
|
||||||
// get original instance with correct file_reference instead of using copies
|
// get original instance with correct file_reference instead of using copies
|
||||||
const isDocument = media._ === 'document';
|
const isDocument = media._ === 'document';
|
||||||
if(isDocument) media = this.appDocsManager.getDoc(media.id);
|
// const isWebDocument = media._ === 'webDocument';
|
||||||
else if(isPhoto) media = this.appPhotosManager.getPhoto(media.id);
|
if(isDocument) media = this.appDocsManager.getDoc((media as Photo.photo).id);
|
||||||
|
else if(isPhoto) media = this.appPhotosManager.getPhoto((media as Document.document).id);
|
||||||
|
|
||||||
const {fileName, downloadOptions} = getDownloadMediaDetails(options);
|
const {fileName, downloadOptions} = getDownloadMediaDetails(options);
|
||||||
|
|
||||||
@ -615,9 +616,9 @@ export class ApiFileManager extends AppManager {
|
|||||||
promise = this.download(downloadOptions);
|
promise = this.download(downloadOptions);
|
||||||
|
|
||||||
if(isDocument && !thumb) {
|
if(isDocument && !thumb) {
|
||||||
this.rootScope.dispatchEvent('document_downloading', media.id);
|
this.rootScope.dispatchEvent('document_downloading', (media as Document.document).id);
|
||||||
promise.catch(noop).finally(() => {
|
promise.catch(noop).finally(() => {
|
||||||
this.rootScope.dispatchEvent('document_downloaded', media.id);
|
this.rootScope.dispatchEvent('document_downloaded', (media as Document.document).id);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
* https://github.com/morethanwords/tweb/blob/master/LICENSE
|
* https://github.com/morethanwords/tweb/blob/master/LICENSE
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import type { WebDocument } from "../../layer";
|
||||||
import type { MyDocument } from "../appManagers/appDocsManager";
|
import type { MyDocument } from "../appManagers/appDocsManager";
|
||||||
import type { MyPhoto } from "../appManagers/appPhotosManager";
|
import type { MyPhoto } from "../appManagers/appPhotosManager";
|
||||||
|
|
||||||
@ -21,29 +22,33 @@ export type ThumbsCache = {
|
|||||||
|
|
||||||
const thumbFullSize = 'full';
|
const thumbFullSize = 'full';
|
||||||
|
|
||||||
|
export type ThumbStorageMedia = MyPhoto | MyDocument | WebDocument;
|
||||||
|
|
||||||
export default class ThumbsStorage {
|
export default class ThumbsStorage {
|
||||||
private thumbsCache: ThumbsCache = {};
|
private thumbsCache: ThumbsCache = {};
|
||||||
|
|
||||||
public getCacheContext(media: MyPhoto | MyDocument, thumbSize: string = thumbFullSize): ThumbCache {
|
private getKey(media: ThumbStorageMedia) {
|
||||||
|
return media._ + ((media as MyPhoto).id ?? (media as WebDocument).url);
|
||||||
|
}
|
||||||
|
|
||||||
|
public getCacheContext(media: ThumbStorageMedia, thumbSize: string = thumbFullSize): ThumbCache {
|
||||||
/* if(media._ === 'photo' && thumbSize !== 'i') {
|
/* if(media._ === 'photo' && thumbSize !== 'i') {
|
||||||
thumbSize = thumbFullSize;
|
thumbSize = thumbFullSize;
|
||||||
} */
|
} */
|
||||||
|
|
||||||
const key = media._ + media.id;
|
const cache = this.thumbsCache[this.getKey(media)] ??= {};
|
||||||
const cache = this.thumbsCache[key] ??= {};
|
|
||||||
return cache[thumbSize] ??= {downloaded: 0, url: '', type: thumbSize};
|
return cache[thumbSize] ??= {downloaded: 0, url: '', type: thumbSize};
|
||||||
}
|
}
|
||||||
|
|
||||||
public setCacheContextURL(media: MyPhoto | MyDocument, thumbSize: string = thumbFullSize, url: string, downloaded: number = 0) {
|
public setCacheContextURL(media: ThumbStorageMedia, thumbSize: string = thumbFullSize, url: string, downloaded: number = 0) {
|
||||||
const cacheContext = this.getCacheContext(media, thumbSize);
|
const cacheContext = this.getCacheContext(media, thumbSize);
|
||||||
cacheContext.url = url;
|
cacheContext.url = url;
|
||||||
cacheContext.downloaded = downloaded;
|
cacheContext.downloaded = downloaded;
|
||||||
return cacheContext;
|
return cacheContext;
|
||||||
}
|
}
|
||||||
|
|
||||||
public deleteCacheContext(media: MyPhoto | MyDocument, thumbSize: string = thumbFullSize) {
|
public deleteCacheContext(media: ThumbStorageMedia, thumbSize: string = thumbFullSize) {
|
||||||
const key = media._ + media.id;
|
const cache = this.thumbsCache[this.getKey(media)];
|
||||||
const cache = this.thumbsCache[key];
|
|
||||||
if(cache) {
|
if(cache) {
|
||||||
delete cache[thumbSize];
|
delete cache[thumbSize];
|
||||||
}
|
}
|
||||||
|
@ -338,4 +338,16 @@
|
|||||||
{"name": "file_size_max", "type": "number"},
|
{"name": "file_size_max", "type": "number"},
|
||||||
{"name": "video_size_max", "type": "number"}
|
{"name": "video_size_max", "type": "number"}
|
||||||
]
|
]
|
||||||
|
}, {
|
||||||
|
"predicate": "webDocument",
|
||||||
|
"params": [
|
||||||
|
{"name": "h", "type": "number"},
|
||||||
|
{"name": "w", "type": "number"}
|
||||||
|
]
|
||||||
|
}, {
|
||||||
|
"predicate": "webDocumentNoProxy",
|
||||||
|
"params": [
|
||||||
|
{"name": "h", "type": "number"},
|
||||||
|
{"name": "w", "type": "number"}
|
||||||
|
]
|
||||||
}]
|
}]
|
@ -2856,3 +2856,8 @@ $bubble-beside-button-width: 38px;
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.bubble-primary-color {
|
||||||
|
color: var(--message-primary-color);
|
||||||
|
font-weight: var(--font-weight-bold);
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user