我尝试使用html2canvas.js渲染THREE.js场景+一些重叠的HTML元素。它大多数时间 ,但不是所有时间。
在失败的情况下,HTML元素被渲染(背景,叠加等),但没有别的。 THREE.js场景就像它完全是空的一样,即使它有可见的数据。我可以说它通常不适用于较大的模型,但只能在渲染的早期。在所有情况下,最终都能正常工作,但较大的模型大约需要30秒。好像我必须给缓冲区一些时间来稳定。
html2canvas正如您所期望的那样处理THREE.js画布 - 它只是使用drawImage
将THREE.js画布绘制到最终由库返回的新画布上。
否则,我会尝试确保没有其他任何东西忙于画布,就像在这个小提琴中一样:http://jsfiddle.net/TheJim01/k6dto5sk/63/(下面的js代码)
正如您所看到的,我尝试阻止渲染循环时会做很多事情,并且当我想捕获场景时再执行一次渲染。但即使所有这些预防措施似乎都没有帮助。
有没有更好的方法从THREE.js画布中获取图像?我可以手动执行该部分,然后将THREE.js画布替换为获取图像的时间足够让html2canvas完成它的工作,然后将THREE.js画布交换回来。我不想这样做这样,如果用户制作了大量快照(图像资源,图像资源无处不在......),我就不会弄乱DOM。
无论如何,这里是代码。欢迎任何想法或建议。谢谢!
var hostDiv, scene, renderer, camera, root, controls, light, shape, theta, aniLoopId, animating;
function snap() {
animating = false;
cancelAnimationFrame(aniLoopId);
renderer.render(scene, camera);
// html2canvas version:
/*
var element = document.getElementById('scenePlusOverlays');
// the input buttons represent my overlays
html2canvas( element, function(canvas) {
// I'd convert the returned canvas to a PNG
animating = true;
animate();
});
*/
// This is basically what html2canvas does with the THREE.js canvas.
var c = document.getElementById('rendererCanvas');
var toC = document.createElement('canvas');
toC.width = c.width;
toC.height = c.height;
var toCtx = toC.getContext('2d');
toCtx.drawImage(c, 0, 0);
console.log(toC.toDataURL('image/png'));
animating = true;
animate();
}
function addGeometry() {
//var geo = new THREE.BoxGeometry(1, 1, 1);
var geo = new THREE.SphereGeometry(5, 32, 32);
var beo = new THREE.BufferGeometry().fromGeometry(geo);
geo.dispose();
geo = null;
var mat = new THREE.MeshPhongMaterial({color:'red'});
var msh;
var count = 10;
count /= 2;
var i = 20;
var topLayer = new THREE.Object3D();
var zLayer, xLayer, yLayer;
for(var Z = -count; Z < count; Z++){
zLayer = new THREE.Object3D();
for(var X = -count; X < count; X++){
xLayer = new THREE.Object3D();
for(var Y = -count; Y < count; Y++){
yLayer = new THREE.Object3D();
msh = new THREE.Mesh(beo, mat);
yLayer.add(msh);
msh.position.set((X*i)+(i/2), (Y*i)+(i/2), (Z*i)+(i/2));
xLayer.add(yLayer);
}
zLayer.add(xLayer);
}
topLayer.add(zLayer);
}
scene.add(topLayer);
}
var WIDTH = '500';//window.innerWidth,
HEIGHT = '500';//window.innerHeight,
FOV = 35,
NEAR = 0.1,
FAR = 10000;
function init() {
hostDiv = document.getElementById('hostDiv');
document.body.insertBefore(hostDiv, document.body.firstElementChild);
renderer = new THREE.WebGLRenderer({ antialias: true, preserverDrawingBuffer: true });
renderer.setSize(WIDTH, HEIGHT);
renderer.domElement.setAttribute('id', 'rendererCanvas');
hostDiv.appendChild(renderer.domElement);
camera = new THREE.PerspectiveCamera(FOV, WIDTH / HEIGHT, NEAR, FAR);
camera.position.z = 500;
controls = new THREE.TrackballControls(camera, renderer.domElement);
light = new THREE.PointLight(0xffffff, 1, Infinity);
light.position.copy(camera.position);
scene = new THREE.Scene();
scene.add(camera);
scene.add(light);
animating = true;
animate();
}
function animate() {
if(animating){
light.position.copy(camera.position);
aniLoopId = requestAnimationFrame(animate);
}
renderer.render(scene, camera);
controls.update();
}
修改 以所述方式调用readPixels或toDataURL是可能的 - 我曾考虑过类似的方法来获取缓冲区,但却被所需的异步代码量所拖延。像这样:
var global_callback = null;
function snapshot_method() {
global_callback = function(returned_image) {
// do something with the image
}
}
// ...
function render() {
renderer.render();
if(global_callback !== null) {
global_callback(renderer.domElement.toDataURL());
global_callback = null;
}
}
答案 0 :(得分:6)
(为了未来Google员工的利益):当您实例化preserveDrawingBuffer
时,您需要将true
设置为WebGLRenderer
:
renderer = new THREE.WebGLRenderer({ antialias: true, preserveDrawingBuffer: true });
(原始代码有这个,但有一个错字)。这是必需的,因为默认情况下,WebGL不需要浏览器在每个绘图框之后保存深度/颜色缓冲区,因此如果要使用readPixels
或toDataURL
,则需要明确告知实现使用该标志保存绘图缓冲区。
然而,这样做有性能损失; the spec provides some guidance if this is a problem for you:
虽然有时需要保留绘图缓冲区,但它可以 在某些平台上造成重大性能损失。每当 可能此标志应保持为false并使用其他技术。 同步绘图缓冲区访问等技术(例如,调用 readPixels或toDataURL在渲染到的相同函数中 绘图缓冲区)可用于获取绘图缓冲区的内容。 如果作者需要在一系列渲染到相同的绘图缓冲区 对于调用,可以使用Framebuffer对象。
我不确定这些建议在当前版本的three.js(r68)中是否可行,但是另一种替代解决方案(在画布副本上调用readPixels
或toDataURL
)讨论了其他几个问题(Why does my canvas go blank after converting to image?,How do you save an image from a Three.js canvas?)。