摄像机
相机是为用户显示或捕获虚拟世界的工具,好比现实世界中观察事物的眼睛,所有炫酷的影像都需要通过相机渲染。在每个场景中必须至少有一个相机才能查看到场景内的对象。Orillusion
已经封装了常用的 相机类型 和 控制器,用户也可以通过 自定义组件 为相机扩展功能。
基本用法
import { Object3D, Scene3D, Camera3D } from '@orillusion/core'
// 实例化一个场景
let scene = new Scene3D();
// 实例化一个节点
let cameraObj = new Object3D();
// 加载一个相机组件
let camera = cameraObj.addComponent(Camera3D);
// 将相机添加至场景
scene.addChild(cameraObj);
// 创建3D视图
let view = new View3D();
// 填充场景至3D视图
view.scene = scene;
// 填充相机至3D视图
view.camera = camera;
// 开始渲染
Engine3D.startRenderView(view);
如果场景内有多个相机,可以通过设置 view.camera
来切换目标相机:
// 如果有多个相机
let cameraObj1 = new Object3D();
let camera1 = cameraObj.addComponent(Camera3D);
let cameraObj2 = new Object3D();
let camera2 = cameraObj.addComponent(Camera3D);
// 创建3D视图
let view = new View3D();
// 设置渲染场景
view.scene = scene;
// 设置 camera1
view.camera = camera1;
...
// 切换使用 camera2 进行渲染
view.camera = camera2;
相机位置
更改相机位置主要有三种方式:
- 通过
TransForm
变换:通过相机节点Object3D
的 transForm 属性可以手动设置相机的位置和方向角度:
// 创建一个节点
let cameraObj = new Object3D();
// 添加相机组件
let camera = cameraObj.addComponent(Camera3D);
// 设定 Object3D 的 Position 或 Rotation
cameraObj.x = 10;
cameraObj.rotateX = 90;
...
- 通过组件
lookAt
函数:相机组件提供的 lookAt 函数 可以同时设置相机Object3D
的位置和观察目标位置:
// 创建一个节点
let cameraObj = new Object3D();
// 添加相机组件
let camera = cameraObj.addComponent(Camera3D);
// 使用 Camera3D 组件 lookAt 改变 Object3D 的位置和方向角度
camera.lookAt(new Vector3(0,0,10), new Vector3(0,0,0), new Vector3(0,0,1));
参数 | 类型 | 描述 | 示例 |
---|---|---|---|
pos | Vector3 | 自身的位置 (全局) | Vector3(0, 0, 0) |
target | Vector3 | 目标的位置 (全局) | Vector3(0, 1, 0) |
up | Vector3 | 相机朝上方向的坐标轴 | Vector3(0, 1, 0) |
- 相机控制器:引擎内置了几种常用的 控制器组件,可以根据用户的输入交互自动调整相机的位置属性。
相机类型
目前主要支持正交相机、透视相机供开发者使用。
正交投影
在正交相机模式下无论物体距离相机远或近在渲染结果中物体的大小不变,我们通常在2D绘图中使用正交相机,并在我们的几何图形中将 z
坐标设为 0.0
。但是 z
轴可以延伸到任何我们想要的长度。使用正交相机对显示对象进行投影,得到的结果是按照同比例缩放,不会有畸变产生。
调用 camera.orthoOffCenter API 可以自定义一个正交相机空间:
参数 | 类型 | 描述 | 示例 |
---|---|---|---|
left | number | 视锥体x轴最小值 | -window.innerWidth / 2 |
right | number | 视锥体x轴最大值 | window.innerWidth / 2 |
bottom | number | 视锥体y轴最小值 | -window.innerHeight / 2 |
top | number | 视锥体y轴最大值 | window.innerHeight / 2 |
near | number | 视锥体近截面z值 | 1 |
far | number | 视锥体远截面z值 | 5000 |
一般情况下,我们可以通过 camera.ortho 快速设置一个以相机目标为中心,以 frustumSize
为高度,以 frustumSize
为深度的正交空间。它会保持屏幕比例自动计算 left
和 right
,且自动计算以相机目标为基点的视椎体 near
和 far
值
参数 | 类型 | 描述 | 示例 |
---|---|---|---|
frustumSize | number | 视锥体高度 | 100 |
frustumDepth | number | 视锥体深度 | 100 |
透视投影
透视投影会利用透视除法对距离观察者很远的对象进行缩短和收缩,逻辑尺寸相同的对象在可视区域靠前位置比靠后位置显得更大,可以实现逼近人眼的观察效果,是3D场景中最常用的投影模式。
调用 camera.perspective 可以按照需求设置相机为透视相机:
参数 | 类型 | 描述 | 示例 |
---|---|---|---|
fov | number | 透视度 | 60 |
aspect | number | 视口比例 | window.innerWidth / window.innerHeight |
near | number | 近截面 | 0.1 |
far | number | 远截面 | 1000 |
import { Engine3D, Scene3D, AtmosphericComponent, HoverCameraController, Object3D, MeshRenderer, BoxGeometry, LitMaterial, DirectLight, View3D, Camera3D, Frustum, OrbitController, Vector3, Color, AxisObject, GridObject } from "@orillusion/core";
import * as dat from "dat.gui"
// initializa engine
await Engine3D.init();
// create new scene as root node
let scene3D: Scene3D = new Scene3D();
// add an Atmospheric sky enviroment
let sky = scene3D.addComponent(AtmosphericComponent);
sky.sunY = 0.6;
// create camera
let cameraObj: Object3D = new Object3D();
let camera = cameraObj.addComponent(Camera3D);
// adjust camera view
camera.perspective(45, Engine3D.aspect, 0.1, 1000.0);
camera.lookAt(new Vector3(0, 10, 10), Vector3.ZERO, Vector3.UP)
// set camera controller
let controller = cameraObj.addComponent(OrbitController);
controller.maxDistance = 200;
// add camera node
scene3D.addChild(cameraObj);
// create light obj
let light: Object3D = new Object3D();
// adjust light rotation
light.rotationX = 45;
light.rotationY = 30;
// add direct light component
let dirLight: DirectLight = light.addComponent(DirectLight);
dirLight.intensity = 3;
// add light object to scene
scene3D.addChild(light);
// create a box
const box: Object3D = new Object3D();
// add MeshRenderer
let mr: MeshRenderer = box.addComponent(MeshRenderer);
// set geometry
mr.geometry = new BoxGeometry(1, 1, 1);
// set material
mr.material = new LitMaterial();
// set rotation
box.y = 0
scene3D.addChild(box);
// create a box
const box2: Object3D = new Object3D();
// add MeshRenderer
let mr2: MeshRenderer = box2.addComponent(MeshRenderer);
// set geometry
mr2.geometry = new BoxGeometry(1, 1, 1);
// set material
mr2.material = new LitMaterial();
mr2.material.baseColor = Color.COLOR_RED
// set rotation
box2.y = 1
box2.x = 1
scene3D.addChild(box2);
// create a box
const box3: Object3D = new Object3D();
// add MeshRenderer
let mr3: MeshRenderer = box3.addComponent(MeshRenderer);
// set geometry
mr3.geometry = new BoxGeometry(1, 1, 1);
// set material
mr3.material = new LitMaterial();
mr3.material.baseColor = Color.COLOR_BLUE
// set rotation
box3.y = -1
box3.x = -1
scene3D.addChild(box3);
scene3D.addChild(new AxisObject(10));
scene3D.addChild(new GridObject(1000, 100));
// create a view with target scene and camera
let view = new View3D();
view.scene = scene3D;
view.camera = camera;
// start render
Engine3D.startRenderView(view);
// add debug GUI
let gui = new dat.GUI();
let f = gui.addFolder('Camera')
let options = {
'ortho': () => {
camera.ortho(camera.frustumSize || 10, camera.frustumDepth || 50)
},
'perspective': () => {
camera.near = 0.1
camera.perspective(camera.fov, camera.aspect, camera.near, camera.far)
}
}
f.add(camera, 'near', 0.1, 100).listen().onChange(() => {
camera.type === 1 ? options.perspective() : options.ortho()
})
f.add(camera, 'far', 1, 1000).listen().onChange(() => {
camera.type === 1 ? options.perspective() : options.ortho()
})
f.add(options, 'perspective')
f.add(camera, 'fov', 1, 179).listen().onChange(() => options.perspective())
f.add(options, 'ortho')
f.add(camera, 'frustumSize', 1, 200).listen().onChange(() => options.ortho())
f.add(camera, 'frustumDepth', 1, 200).listen().onChange(() => options.ortho())
f.open()
相机组件
相机组件为相机提供灵活的扩展支持,可以直接使用预定义组件,也可以自定义组件实现更个性化的需求。组件通过自身的 update
函数,执行与 Engine3D
主循环同步的自我更新逻辑。
自由相机
该相机控制器 ,实现摄像机自由移动。它的交互功能有:
- 通过 W A S D 向着朝向方向 前进 后退 左右移动
- 通过按住鼠标左键控制相机的移动朝向
import { Engine3D, Scene3D, Object3D, Camera3D, Vector3, PlaneGeometry, DirectLight, Color, KelvinUtil, FlyCameraController, AtmosphericComponent, LitMaterial, BoxGeometry, MeshRenderer, View3D } from '@orillusion/core';
await Engine3D.init();
let scene: Scene3D = new Scene3D();
let cameraObj = new Object3D();
cameraObj.y = 0;
let camera = cameraObj.addComponent(Camera3D);
camera.perspective(60, Engine3D.aspect, 0.1, 5000.0);
// add Camera Controller
let flyController = cameraObj.addComponent(FlyCameraController);
flyController.setCamera(new Vector3(0, 15, 15), new Vector3(0, 10, 0));
flyController.moveSpeed = 10;
scene.addChild(cameraObj);
const boxObj: Object3D = new Object3D();
boxObj.localPosition = new Vector3(0, 10, 0);
let boxMr: MeshRenderer = boxObj.addComponent(MeshRenderer);
boxMr.geometry = new BoxGeometry(2, 2, 2);
boxMr.material = new LitMaterial();
boxMr.material.baseColor = new Color(1.0, 1.0, 1.0, 1.0);
scene.addChild(boxObj);
let groundObj = new Object3D();
groundObj.localPosition = new Vector3(0, 9, 0);
let planeMr = groundObj.addComponent(MeshRenderer);
planeMr.geometry = new PlaneGeometry(10, 10);
planeMr.material = new LitMaterial();
scene.addChild(groundObj);
{
let lightObj = new Object3D();
lightObj.x = 0;
lightObj.y = 0;
lightObj.z = 0;
lightObj.rotationX = 0;
lightObj.rotationY = 0;
lightObj.rotationZ = 0;
let lc = lightObj.addComponent(DirectLight);
lc.lightColor = KelvinUtil.color_temperature_to_rgb(5355);
lc.castShadow = true;
lc.intensity = 1.7;
scene.addChild(lightObj);
}
// add an Atmospheric sky enviroment
scene.addComponent(AtmosphericComponent).sunY = 0.6;
// create a view with target scene and camera
let view = new View3D();
view.scene = scene;
view.camera = camera;
// start render
Engine3D.startRenderView(view);
基本用法:
import { Scene3D, Camera3D, FlyCameraController } from '@orillusion/core'
// 实例化一个节点
let cameraObj = new Object3D();
// 加载一个相机组件
let camera = cameraObj.addComponent(Camera3D);
// 加载控制器组件
let flyController = cameraObj.addComponent(FlyCameraController);
// 通过组件 setCamera 设置相机位置
flyController.setCamera(new Vector3(0, 0, 15), new Vector3(0, 0, 0));
// 设置鼠标移动速度
flyController.moveSpeed = 10;
自由相机可以通过 setCamera 设定自身位置和朝向
参数 | 类型 | 描述 | 示例 |
---|---|---|---|
targetPos | Vector3 | 自身位置 | new Vector3(0,0,10) |
lookAtPos | Vector3 | 目标位置 | new Vector3(0,0,0) |
还可以修改 moveSpeed
来调整移动的速度快慢
参数 | 类型 | 描述 | 示例 |
---|---|---|---|
moveSpeed | number | 移动速度 | 10 |
盘旋相机
该相机控制器,实现摄像机在 xz
平面移动/围绕当前观察点旋转。它的交互功能有:
- 按下鼠标左键并移动鼠标,可以使相机围绕当前观察目标进行旋转。
- 按下鼠标右键并移动鼠标,可以使相机根据鼠标移动的方向与距离平滑移动当前场景可视区域
- 滑动鼠标滚轮可以控制摄像机的视距
import { Engine3D, Scene3D, Object3D, Camera3D, Vector3, HoverCameraController, AtmosphericComponent, LitMaterial, BoxGeometry, MeshRenderer, View3D, DirectLight } from '@orillusion/core';
await Engine3D.init();
let scene: Scene3D = new Scene3D();
let cameraObj = new Object3D();
let camera = cameraObj.addComponent(Camera3D);
camera.perspective(60, Engine3D.aspect, 0.1, 5000.0);
// add camera controller
let hoverController = cameraObj.addComponent(HoverCameraController);
hoverController.setCamera(15, -15, 15, new Vector3(0, 0, 0));
scene.addChild(cameraObj);
// add a base light
let lightObj = new Object3D();
lightObj.addComponent(DirectLight);
scene.addChild(lightObj);
const boxObj: Object3D = new Object3D();
boxObj.localPosition = new Vector3(0, 0, 0);
let mr: MeshRenderer = boxObj.addComponent(MeshRenderer);
mr.geometry = new BoxGeometry(5, 5, 5);
mr.material = new LitMaterial();
scene.addChild(boxObj);
// add an Atmospheric sky enviroment
scene.addComponent(AtmosphericComponent).sunY = 0.6;
// create a view with target scene and camera
let view = new View3D();
view.scene = scene;
view.camera = camera;
// start render
Engine3D.startRenderView(view);
基本用法:
import { Scene3D, Camera3D, HoverCameraController } from '@orillusion/core'
// 实例化一个节点
let cameraObj = new Object3D();
// 加载一个相机组件
let camera = cameraObj.addComponent(Camera3D);
// 加载控制器组件
let hoverCameraController = cameraObj.addComponent(HoverCameraController);
// 通过组件 setCamera 设置相机位置
hoverController.setCamera(15, -15, 15, new Vector3(0, 0, 0));
盘旋相机可以通过 setCamera 控制相机位置和朝向
参数 | 类型 | 描述 | 示例 |
---|---|---|---|
roll | number | 围绕y轴旋转 | 0 |
pitch | number | 围绕x轴旋转 | 0 |
distance | number | 相机与目标的距离 | 10 |
target | Vector3 | 朝向目标坐标 | new Vector3(0,0,0) |
轨道相机
该相机控制器,跟盘旋相机很相似,也是围绕一个坐标观察点旋转。但它可以直接设置相机的 Object3D
的位置和渲染来控制视角位置和朝向。主要特性如下:
- 按下鼠标左键并移动鼠标,可以使相机围绕当前观察目标进行全向旋转
- 按下鼠标右键并移动鼠标,可以根据鼠标移动方向在空间各个方向内移动相机中心,不仅可以在
xz
平面内自由移动,也支持y
方向的自由移动 - 滑动鼠标滚轮可以控制摄像机和中心的距离
- 可以设置相机自动旋转
- 可以设置旋转,缩放,平移的速度
- 可以设置仰角最大值和最小值
import { Engine3D, Scene3D, Object3D, Camera3D, Vector3, OrbitController, AtmosphericComponent, LitMaterial, BoxGeometry, MeshRenderer, View3D, DirectLight } from '@orillusion/core';
await Engine3D.init();
let scene: Scene3D = new Scene3D();
let cameraObj = new Object3D();
let camera = cameraObj.addComponent(Camera3D);
camera.perspective(60, Engine3D.aspect, 0.1, 5000.0);
cameraObj.localPosition.set(0, 10, 20);
// add camera controller
let orbit = cameraObj.addComponent(OrbitController);
// auto rotation
orbit.autoRotateSpeed = 0.5;
orbit.autoRotate = true;
scene.addChild(cameraObj);
// add a base light
let lightObj = new Object3D();
lightObj.addComponent(DirectLight);
scene.addChild(lightObj);
const boxObj: Object3D = new Object3D();
boxObj.localPosition = new Vector3(0, 0, 0);
let mr: MeshRenderer = boxObj.addComponent(MeshRenderer);
mr.geometry = new BoxGeometry(5, 5, 5);
mr.material = new LitMaterial();
scene.addChild(boxObj);
// add an Atmospheric sky enviroment
scene.addComponent(AtmosphericComponent).sunY = 0.6;
// create a view with target scene and camera
let view = new View3D();
view.scene = scene;
view.camera = camera;
// start render
Engine3D.startRenderView(view);
基本用法:
import { Scene3D, Camera3D, OrbitController } from '@orillusion/core'
// 实例化一个节点
let cameraObj = new Object3D();
// 加载一个相机组件
let camera = cameraObj.addComponent(Camera3D);
// 加载控制器组件
let orbit = cameraObj.addComponent(OrbitController);
// 设置相机 Object3D 位置
cameraObj.localPosition.set(0, 10, 30);
// 开启自动旋转
orbit.autoRotate = true
// 自动旋转速度
orbit.autoRotateSpeed = 0.1
// 缩放速度系数
orbit.zoomFactor = 0.1
// 视角平移速度系数
orbit.panFactor = 0.25
// 视角平滑系数
orbit.smooth = 5
// 缩放最小距离
orbit.minDistance = 1
// 缩放最大距离
orbit.maxDistance = 1000
// 仰角最小值
orbit.minPolarAngle = -90
// 仰角最大值
orbit.minPolarAngle = 90
自定义控制器
用户可以通过 自定义组件 来扩展额外的相机组件,可以参考 OrbitController 的实现方式。