UNPKG

molstar

Version:

A comprehensive macromolecular library.

81 lines (80 loc) 3.79 kB
/** * Copyright (c) 2020-2024 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author David Sehnal <david.sehnal@gmail.com> */ import * as HME from 'h264-mp4-encoder'; import { PluginStateAnimation } from '../../mol-plugin-state/animation/model'; export async function encodeMp4Animation(plugin, ctx, params) { var _a, _b, _c; await ctx.update({ message: 'Initializing...', isIndeterminate: true }); validateViewport(params); const durationMs = PluginStateAnimation.getDuration(plugin, params.animation); if (durationMs === void 0) { throw new Error('The animation does not have the duration specified.'); } const encoder = await HME.createH264MP4Encoder(); const { width, height } = params; let vw = params.viewport.width, vh = params.viewport.height; // dimensions must be a multiple of 2 if (vw % 2 !== 0) vw -= 1; if (vh % 2 !== 0) vh -= 1; const normalizedViewport = { ...params.viewport, width: vw, height: vh }; encoder.width = vw; encoder.height = vh; if (params.quantizationParameter) encoder.quantizationParameter = params.quantizationParameter; if (params.fps) encoder.frameRate = params.fps; encoder.initialize(); const loop = plugin.animationLoop; const canvasProps = (_a = plugin.canvas3d) === null || _a === void 0 ? void 0 : _a.props; const wasAnimating = loop.isAnimating; let stoppedAnimation = true, finalized = false; try { loop.stop(); loop.resetTime(0); if (params.customBackground !== void 0) { (_b = plugin.canvas3d) === null || _b === void 0 ? void 0 : _b.setProps({ renderer: { backgroundColor: params.customBackground }, transparentBackground: false }, true); } const fps = encoder.frameRate; const N = Math.ceil(durationMs / 1000 * fps); const dt = durationMs / N; await ctx.update({ message: 'Rendering...', isIndeterminate: false, current: 0, max: N + 1 }); await params.pass.updateBackground(); await plugin.managers.animation.play(params.animation.definition, params.animation.params); stoppedAnimation = false; for (let i = 0; i <= N; i++) { await loop.tick(i * dt, { isSynchronous: true, animation: { currentFrame: i, frameCount: N }, manualDraw: true, updateControls: true }); const image = await params.pass.getImageData(ctx, width, height, normalizedViewport); encoder.addFrameRgba(image.data); if (ctx.shouldUpdate) { await ctx.update({ current: i + 1 }); } } await ctx.update({ message: 'Applying finishing touches...', isIndeterminate: true }); await plugin.managers.animation.stop(); stoppedAnimation = true; encoder.finalize(); finalized = true; return encoder.FS.readFile(encoder.outputFilename); } finally { if (finalized) encoder.delete(); if (params.customBackground !== void 0) { (_c = plugin.canvas3d) === null || _c === void 0 ? void 0 : _c.setProps({ renderer: { backgroundColor: canvasProps === null || canvasProps === void 0 ? void 0 : canvasProps.renderer.backgroundColor }, transparentBackground: canvasProps === null || canvasProps === void 0 ? void 0 : canvasProps.transparentBackground }); } if (!stoppedAnimation) await plugin.managers.animation.stop(); if (wasAnimating) loop.start(); } } function validateViewport(params) { if (params.viewport.x + params.viewport.width > params.width || params.viewport.y + params.viewport.height > params.height) { throw new Error('Viewport exceeds the canvas dimensions.'); } }