Files
mol 263cb5ef03
Some checks failed
continuous-integration/drone/push Build is failing
test
2024-07-06 22:23:31 +08:00

147 lines
5.8 KiB
JavaScript

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.IIPHandler = void 0;
const ImageRenderer_1 = require("./ImageRenderer");
const ImageStorage_1 = require("./ImageStorage");
const Base64Decoder_wasm_1 = require("xterm-wasm-parts/lib/base64/Base64Decoder.wasm");
const IIPHeaderParser_1 = require("./IIPHeaderParser");
const IIPMetrics_1 = require("./IIPMetrics");
// limit hold memory in base64 decoder
const KEEP_DATA = 4194304;
// default IIP header values
const DEFAULT_HEADER = {
name: 'Unnamed file',
size: 0,
width: 'auto',
height: 'auto',
preserveAspectRatio: 1,
inline: 0
};
class IIPHandler {
constructor(_opts, _renderer, _storage, _coreTerminal) {
this._opts = _opts;
this._renderer = _renderer;
this._storage = _storage;
this._coreTerminal = _coreTerminal;
this._aborted = false;
this._hp = new IIPHeaderParser_1.HeaderParser();
this._header = DEFAULT_HEADER;
this._dec = new Base64Decoder_wasm_1.default(KEEP_DATA);
this._metrics = IIPMetrics_1.UNSUPPORTED_TYPE;
}
reset() { }
start() {
this._aborted = false;
this._header = DEFAULT_HEADER;
this._metrics = IIPMetrics_1.UNSUPPORTED_TYPE;
this._hp.reset();
}
put(data, start, end) {
if (this._aborted)
return;
if (this._hp.state === 4 /* HeaderState.END */) {
if (this._dec.put(data, start, end)) {
this._dec.release();
this._aborted = true;
}
}
else {
const dataPos = this._hp.parse(data, start, end);
if (dataPos === -1) {
this._aborted = true;
return;
}
if (dataPos > 0) {
this._header = Object.assign({}, DEFAULT_HEADER, this._hp.fields);
if (!this._header.inline || !this._header.size || this._header.size > this._opts.iipSizeLimit) {
this._aborted = true;
return;
}
this._dec.init(this._header.size);
if (this._dec.put(data, dataPos, end)) {
this._dec.release();
this._aborted = true;
}
}
}
}
end(success) {
if (this._aborted)
return true;
let w = 0;
let h = 0;
// early exit condition chain
let cond = true;
if (cond = success) {
if (cond = !this._dec.end()) {
this._metrics = (0, IIPMetrics_1.imageType)(this._dec.data8);
if (cond = this._metrics.mime !== 'unsupported') {
w = this._metrics.width;
h = this._metrics.height;
if (cond = w && h && w * h < this._opts.pixelLimit) {
[w, h] = this._resize(w, h).map(Math.floor);
cond = w && h && w * h < this._opts.pixelLimit;
}
}
}
}
if (!cond) {
this._dec.release();
return true;
}
const blob = new Blob([this._dec.data8], { type: this._metrics.mime });
this._dec.release();
if (!window.createImageBitmap) {
const url = URL.createObjectURL(blob);
const img = new Image();
return new Promise(r => {
img.addEventListener('load', () => {
var _a;
URL.revokeObjectURL(url);
const canvas = ImageRenderer_1.ImageRenderer.createCanvas(window.document, w, h);
(_a = canvas.getContext('2d')) === null || _a === void 0 ? void 0 : _a.drawImage(img, 0, 0, w, h);
this._storage.addImage(canvas);
r(true);
});
img.src = url;
// sanity measure to avoid terminal blocking from dangling promise
// happens from corrupt data (onload never gets fired)
setTimeout(() => r(true), 1000);
});
}
return createImageBitmap(blob, { resizeWidth: w, resizeHeight: h })
.then(bm => {
this._storage.addImage(bm);
return true;
});
}
_resize(w, h) {
var _a, _b, _c, _d;
const cw = ((_a = this._renderer.dimensions) === null || _a === void 0 ? void 0 : _a.css.cell.width) || ImageStorage_1.CELL_SIZE_DEFAULT.width;
const ch = ((_b = this._renderer.dimensions) === null || _b === void 0 ? void 0 : _b.css.cell.height) || ImageStorage_1.CELL_SIZE_DEFAULT.height;
const width = ((_c = this._renderer.dimensions) === null || _c === void 0 ? void 0 : _c.css.canvas.width) || cw * this._coreTerminal.cols;
const height = ((_d = this._renderer.dimensions) === null || _d === void 0 ? void 0 : _d.css.canvas.height) || ch * this._coreTerminal.rows;
const rw = this._dim(this._header.width, width, cw);
const rh = this._dim(this._header.height, height, ch);
if (!rw && !rh) {
const wf = width / w; // TODO: should this respect initial cursor offset?
const hf = (height - ch) / h; // TODO: fix offset issues from float cell height
const f = Math.min(wf, hf);
return f < 1 ? [w * f, h * f] : [w, h];
}
return !rw
? [w * rh / h, rh]
: this._header.preserveAspectRatio || !rw || !rh
? [rw, h * rw / w] : [rw, rh];
}
_dim(s, total, cdim) {
if (s === 'auto')
return 0;
if (s.endsWith('%'))
return parseInt(s.slice(0, -1)) * total / 100;
if (s.endsWith('px'))
return parseInt(s.slice(0, -2));
return parseInt(s) * cdim;
}
}
exports.IIPHandler = IIPHandler;