mirror of
https://github.com/jiawanlong/Cesium-Examples.git
synced 2025-07-05 07:37:31 +00:00
297 lines
40 KiB
JavaScript
297 lines
40 KiB
JavaScript
![]() |
var videos = [];
|
||
|
var video_dom;
|
||
|
var getCurrentMousePosition = function (scene, position, noPickEntity) {
|
||
|
var cartesian;
|
||
|
var pickedObject = scene.pick(position);
|
||
|
if (scene.pickPositionSupported && Cesium.defined(pickedObject)) {
|
||
|
var entity = pickedObject.id;
|
||
|
if (noPickEntity == null || (noPickEntity && entity !== noPickEntity)) {
|
||
|
var cartesian = scene.pickPosition(position);
|
||
|
if (Cesium.defined(cartesian)) {
|
||
|
var cartographic = Cesium.Cartographic.fromCartesian(cartesian);
|
||
|
var height = cartographic.height;
|
||
|
if (height >= 0) return cartesian;
|
||
|
|
||
|
if (!Cesium.defined(pickedObject.id) && height >= -500)
|
||
|
return cartesian;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (scene.mode === Cesium.SceneMode.SCENE3D) {
|
||
|
var pickRay = scene.camera.getPickRay(position);
|
||
|
cartesian = scene.globe.pick(pickRay, scene);
|
||
|
} else {
|
||
|
cartesian = scene.camera.pickEllipsoid(position, scene.globe.ellipsoid);
|
||
|
}
|
||
|
return cartesian;
|
||
|
}
|
||
|
|
||
|
class video {
|
||
|
constructor(viewer, config) {
|
||
|
this.viewer = viewer;
|
||
|
this.config = config;
|
||
|
|
||
|
}
|
||
|
|
||
|
creat() {
|
||
|
var viewer = this.viewer;
|
||
|
var config = this.config;
|
||
|
var videoElement = config.videoElement;
|
||
|
var positions = config.positions;
|
||
|
var clampToGround = config.clampToGround;
|
||
|
if (clampToGround) {
|
||
|
viewer.entities.add({
|
||
|
nam: "video",
|
||
|
polygon: {
|
||
|
hierarchy: Cesium.Cartesian3.fromDegreesArray(positions),
|
||
|
material: videoElement
|
||
|
}
|
||
|
});
|
||
|
} else {
|
||
|
viewer.entities.add({
|
||
|
nam: "video",
|
||
|
polygon: {
|
||
|
hierarchy: {
|
||
|
positions: Cesium.Cartesian3.fromDegreesArrayHeights(positions)
|
||
|
},
|
||
|
material: videoElement,
|
||
|
perPositionHeight: true,
|
||
|
outline: true
|
||
|
}
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
|
||
|
clearAll() {
|
||
|
var dd = viewer.entities._entities._array;
|
||
|
for (let index = 0; index < dd.length; index++) {
|
||
|
if (dd[index]._nam = "video") {
|
||
|
viewer.entities.remove(dd[index])
|
||
|
index--;
|
||
|
}
|
||
|
}
|
||
|
videos.forEach((v) => {
|
||
|
v.destroy();
|
||
|
})
|
||
|
}
|
||
|
|
||
|
change(object) {
|
||
|
var _this = this;
|
||
|
for (const key in object) {
|
||
|
const element = object[key];
|
||
|
_this.lightCamera.frustum[key] = element;
|
||
|
_this.clear()
|
||
|
_this.drawFrustumOutline();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
drawVideo() {
|
||
|
let _self = this;
|
||
|
var options = this.config;
|
||
|
this.handler = new Cesium.ScreenSpaceEventHandler(this.viewer.scene.canvas);
|
||
|
|
||
|
this.horizontalViewAngle = options.horizontalViewAngle || 60.0;
|
||
|
this.verticalViewAngle = options.verticalViewAngle || 40.0;
|
||
|
video_dom = document.getElementById(options.video);
|
||
|
|
||
|
this.options = options;
|
||
|
this.posArray = [];
|
||
|
this.state = "PREPARE";
|
||
|
if (options.viewPosition && options.viewPositionEnd) {
|
||
|
_self.viewPosition = options.viewPosition;
|
||
|
_self.viewPositionEnd = options.viewPositionEnd;
|
||
|
_self.viewDistance = Cesium.Cartesian3.distance(_self.viewPosition, _self.viewPositionEnd);
|
||
|
_self.viewHeading = getHeading(_self.viewPosition, _self.viewPositionEnd);
|
||
|
_self.viewPitch = getPitch(_self.viewPosition, _self.viewPositionEnd);
|
||
|
_self.createLightCamera();
|
||
|
} else {
|
||
|
this.action();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
action() {
|
||
|
let _self = this;
|
||
|
_self.handler.setInputAction(function (movement) {
|
||
|
var cartesian = getCurrentMousePosition(_self.viewer.scene, movement.position);
|
||
|
if (!cartesian) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (_self.posArray.length == 0) {
|
||
|
_self.posArray.push(cartesian);
|
||
|
_self.state = "OPERATING";
|
||
|
} else if (_self.posArray.length == 1) {
|
||
|
_self.viewPosition = _self.posArray[0];
|
||
|
_self.viewPositionEnd = cartesian;
|
||
|
_self.viewDistance = Cesium.Cartesian3.distance(_self.viewPosition, _self.viewPositionEnd);
|
||
|
_self.viewHeading = getHeading(_self.viewPosition, _self.viewPositionEnd);
|
||
|
_self.viewPitch = getPitch(_self.viewPosition, _self.viewPositionEnd);
|
||
|
|
||
|
_self.state = "END";
|
||
|
_self.handler.destroy();
|
||
|
_self.handler = null;
|
||
|
_self.createLightCamera();
|
||
|
}
|
||
|
}, Cesium.ScreenSpaceEventType.LEFT_CLICK);
|
||
|
}
|
||
|
|
||
|
//创建相机
|
||
|
createLightCamera() {
|
||
|
this.lightCamera = new Cesium.Camera(this.viewer.scene);
|
||
|
this.lightCamera.position = this.viewPosition;
|
||
|
|
||
|
this.lightCamera.frustum.near = this.viewDistance * 0.0001;
|
||
|
this.lightCamera.frustum.far = this.viewDistance;
|
||
|
const hr = Cesium.Math.toRadians(this.horizontalViewAngle);
|
||
|
const vr = Cesium.Math.toRadians(this.verticalViewAngle);
|
||
|
const aspectRatio =
|
||
|
(this.viewDistance * Math.tan(hr / 2) * 2) /
|
||
|
(this.viewDistance * Math.tan(vr / 2) * 2);
|
||
|
this.lightCamera.frustum.aspectRatio = aspectRatio;
|
||
|
if (hr > vr) {
|
||
|
this.lightCamera.frustum.fov = hr;
|
||
|
} else {
|
||
|
this.lightCamera.frustum.fov = vr;
|
||
|
}
|
||
|
this.lightCamera.setView({
|
||
|
destination: this.viewPosition,
|
||
|
orientation: {
|
||
|
heading: Cesium.Math.toRadians(this.viewHeading || 0),
|
||
|
pitch: Cesium.Math.toRadians(this.viewPitch || 0),
|
||
|
roll: 0
|
||
|
}
|
||
|
});
|
||
|
this.drawFrustumOutline();
|
||
|
}
|
||
|
|
||
|
//创建视锥线
|
||
|
drawFrustumOutline() {
|
||
|
const scratchRight = new Cesium.Cartesian3();
|
||
|
const scratchRotation = new Cesium.Matrix3();
|
||
|
const scratchOrientation = new Cesium.Quaternion();
|
||
|
const position = this.lightCamera.positionWC;
|
||
|
const direction = this.lightCamera.directionWC;
|
||
|
const up = this.lightCamera.upWC;
|
||
|
let right = this.lightCamera.rightWC;
|
||
|
right = Cesium.Cartesian3.negate(right, scratchRight);
|
||
|
let rotation = scratchRotation;
|
||
|
Cesium.Matrix3.setColumn(rotation, 0, right, rotation);
|
||
|
Cesium.Matrix3.setColumn(rotation, 1, up, rotation);
|
||
|
Cesium.Matrix3.setColumn(rotation, 2, direction, rotation);
|
||
|
let orientation = Cesium.Quaternion.fromRotationMatrix(rotation, scratchOrientation);
|
||
|
|
||
|
var newObj = _.cloneDeep(this.lightCamera.frustum);
|
||
|
newObj.near = newObj.far - 0.01;
|
||
|
|
||
|
var videoGeometryInstance1 = new Cesium.GeometryInstance({
|
||
|
geometry: new Cesium.FrustumGeometry({
|
||
|
frustum: newObj,
|
||
|
origin: this.viewPosition,
|
||
|
orientation: orientation
|
||
|
})
|
||
|
});
|
||
|
|
||
|
var p1s = new Cesium.Primitive({
|
||
|
geometryInstances: [videoGeometryInstance1],
|
||
|
appearance: createAppearance()
|
||
|
})
|
||
|
this.viewer.scene.primitives.add(
|
||
|
p1s
|
||
|
);
|
||
|
|
||
|
var videoGeometryInstance2 = new Cesium.GeometryInstance({
|
||
|
geometry: new Cesium.FrustumOutlineGeometry({
|
||
|
frustum: this.lightCamera.frustum,
|
||
|
origin: this.viewPosition,
|
||
|
orientation: orientation
|
||
|
}),
|
||
|
attributes: {
|
||
|
color: Cesium.ColorGeometryInstanceAttribute.fromColor(Cesium.Color.BLUE)
|
||
|
}
|
||
|
});
|
||
|
|
||
|
var p2s = new Cesium.Primitive({
|
||
|
geometryInstances: [videoGeometryInstance2],
|
||
|
appearance: new Cesium.PerInstanceColorAppearance(
|
||
|
{
|
||
|
flat: true,
|
||
|
// translucent : false
|
||
|
}
|
||
|
)
|
||
|
})
|
||
|
this.viewer.scene.primitives.add(
|
||
|
p2s
|
||
|
);
|
||
|
this.FrustumGeometry = p1s;
|
||
|
this.FrustumOutlineGeometry = p2s;
|
||
|
videos.push(p1s)
|
||
|
videos.push(p2s)
|
||
|
}
|
||
|
|
||
|
clear() {
|
||
|
this.FrustumGeometry.destroy()
|
||
|
this.FrustumOutlineGeometry.destroy()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
function createAppearance() {
|
||
|
let source = `czm_material czm_getMaterial(czm_materialInput materialInput)
|
||
|
{
|
||
|
czm_material material = czm_getDefaultMaterial(materialInput);
|
||
|
vec2 st = materialInput.st;
|
||
|
vec4 colorImage = texture2D(image, vec2(st.s, st.t));
|
||
|
vec4 maskImage = texture2D(tmask, vec2(st.s, st.t));
|
||
|
material.alpha = colorImage.a * color.a*maskImage.r;
|
||
|
material.diffuse = colorImage.rgb*color.rgb;
|
||
|
return material;
|
||
|
}`;
|
||
|
let material = new Cesium.Material({
|
||
|
fabric: {
|
||
|
type: "custome_1",
|
||
|
uniforms: {
|
||
|
color: new Cesium.Color(1.0, 1.0, 1.0, 1.0),
|
||
|
image: "
|
||
|
tmask: "
|
||
|
},
|
||
|
source: source
|
||
|
}
|
||
|
});
|
||
|
material.uniforms.image = video_dom;
|
||
|
let appearance = new Cesium.EllipsoidSurfaceAppearance({
|
||
|
material: material,
|
||
|
flat: true,
|
||
|
renderState: {
|
||
|
cull: {
|
||
|
enabled: false,
|
||
|
},
|
||
|
depthTest: {
|
||
|
enabled: false
|
||
|
}
|
||
|
}
|
||
|
});
|
||
|
return appearance;
|
||
|
}
|
||
|
|
||
|
//获取偏航角
|
||
|
function getHeading(fromPosition, toPosition) {
|
||
|
let finalPosition = new Cesium.Cartesian3();
|
||
|
let matrix4 = Cesium.Transforms.eastNorthUpToFixedFrame(fromPosition);
|
||
|
Cesium.Matrix4.inverse(matrix4, matrix4);
|
||
|
Cesium.Matrix4.multiplyByPoint(matrix4, toPosition, finalPosition);
|
||
|
Cesium.Cartesian3.normalize(finalPosition, finalPosition);
|
||
|
return Cesium.Math.toDegrees(Math.atan2(finalPosition.x, finalPosition.y));
|
||
|
}
|
||
|
|
||
|
//获取俯仰角
|
||
|
function getPitch(fromPosition, toPosition) {
|
||
|
let finalPosition = new Cesium.Cartesian3();
|
||
|
let matrix4 = Cesium.Transforms.eastNorthUpToFixedFrame(fromPosition);
|
||
|
Cesium.Matrix4.inverse(matrix4, matrix4);
|
||
|
Cesium.Matrix4.multiplyByPoint(matrix4, toPosition, finalPosition);
|
||
|
Cesium.Cartesian3.normalize(finalPosition, finalPosition);
|
||
|
return Cesium.Math.toDegrees(Math.asin(finalPosition.z));
|
||
|
}
|
||
|
|