|
|
|
import { StagedAttachmentType } from '../components/session/conversation/SessionCompositionBox';
|
|
|
|
import { SignalService } from '../protobuf';
|
|
|
|
|
|
|
|
export async function autoScale<T extends { contentType: string; file: any }>(
|
|
|
|
attachment: T
|
|
|
|
): Promise<T> {
|
|
|
|
const { contentType, file } = attachment;
|
|
|
|
if (contentType.split('/')[0] !== 'image' || contentType === 'image/tiff') {
|
|
|
|
// nothing to do
|
|
|
|
return Promise.resolve(attachment);
|
|
|
|
}
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const url = URL.createObjectURL(file);
|
|
|
|
const img = document.createElement('img');
|
|
|
|
img.onerror = reject;
|
|
|
|
img.onload = () => {
|
|
|
|
URL.revokeObjectURL(url);
|
|
|
|
|
|
|
|
const maxSize = 6000 * 1024;
|
|
|
|
const maxHeight = 4096;
|
|
|
|
const maxWidth = 4096;
|
|
|
|
if (
|
|
|
|
img.naturalWidth <= maxWidth &&
|
|
|
|
img.naturalHeight <= maxHeight &&
|
|
|
|
file.size <= maxSize
|
|
|
|
) {
|
|
|
|
resolve(attachment);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const gifMaxSize = 25000 * 1024;
|
|
|
|
if (file.type === 'image/gif' && file.size <= gifMaxSize) {
|
|
|
|
resolve(attachment);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (file.type === 'image/gif') {
|
|
|
|
reject(new Error('GIF is too large'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const canvas = window.loadImage.scale(img, {
|
|
|
|
canvas: true,
|
|
|
|
maxWidth,
|
|
|
|
maxHeight,
|
|
|
|
});
|
|
|
|
|
|
|
|
let quality = 0.95;
|
|
|
|
let i = 4;
|
|
|
|
let blob;
|
|
|
|
do {
|
|
|
|
i -= 1;
|
|
|
|
blob = window.dataURLToBlobSync(
|
|
|
|
canvas.toDataURL('image/jpeg', quality)
|
|
|
|
);
|
|
|
|
quality = (quality * maxSize) / blob.size;
|
|
|
|
// NOTE: During testing with a large image, we observed the
|
|
|
|
// `quality` value being > 1. Should we clamp it to [0.5, 1.0]?
|
|
|
|
// See: https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob#Syntax
|
|
|
|
if (quality < 0.5) {
|
|
|
|
quality = 0.5;
|
|
|
|
}
|
|
|
|
} while (i > 0 && blob.size > maxSize);
|
|
|
|
|
|
|
|
resolve({
|
|
|
|
...attachment,
|
|
|
|
file: blob,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
img.src = url;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function getFile(attachment: StagedAttachmentType) {
|
|
|
|
if (!attachment) {
|
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
const attachmentFlags = attachment.isVoiceMessage
|
|
|
|
? SignalService.AttachmentPointer.Flags.VOICE_MESSAGE
|
|
|
|
: null;
|
|
|
|
|
|
|
|
const scaled = await autoScale(attachment);
|
|
|
|
const fileRead = await readFile(scaled);
|
|
|
|
return {
|
|
|
|
...fileRead,
|
|
|
|
url: undefined,
|
|
|
|
flags: attachmentFlags || null,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function readFile(attachment: any): Promise<object> {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const FR = new FileReader();
|
|
|
|
FR.onload = e => {
|
|
|
|
const data = e?.target?.result as ArrayBuffer;
|
|
|
|
resolve({
|
|
|
|
...attachment,
|
|
|
|
data,
|
|
|
|
size: data.byteLength,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
FR.onerror = reject;
|
|
|
|
FR.onabort = reject;
|
|
|
|
FR.readAsArrayBuffer(attachment.file);
|
|
|
|
});
|
|
|
|
}
|