cocos-awesome/assets/Scene/Home/Transition.ts
2024-05-15 22:37:35 +08:00

181 lines
5.3 KiB
TypeScript

const { ccclass, property } = cc._decorator;
const TRANSITION_TIME = 2;
@ccclass
export default class Transition extends cc.Component {
@property() time: number = 0;
@property(cc.Material) materials = [];
private _inited = false;
private _texture1: cc.Texture2D;
private _texture2: cc.Texture2D;
private _sprite: cc.Sprite;
private _spriteNode: cc.Node;
private _camera: cc.Camera;
private _cameraNode: cc.Node;
private _spriteMaterial: cc.Material;
private _onLoadFinished = () => {};
loading = false;
private materialIndex = 0;
onLoad() {
cc.game.addPersistRootNode(this.node);
}
start() {
this.init();
cc.director.on(cc.Director.EVENT_AFTER_SCENE_LAUNCH, this._onLoadFinished, this);
cc.director.on('switchSceneByTransition', this.switchSceneByTransition.bind(this));
}
switchSceneByTransition() {
this.materialIndex = this.materialIndex + 1 > this.materials.length ? 0 : this.materialIndex + 1;
this.loadScene('Scene/Home/Home', 'Canvas/Main Camera', 'Canvas/Main Camera');
}
init() {
if (this._inited) return;
this._inited = true;
this.time = 0;
this._texture1 = this.createTexture();
this._texture2 = this.createTexture();
let spriteNode = cc.find('TRANSITION_SPRITE', this.node);
if (!spriteNode) {
spriteNode = new cc.Node('TRANSITION_SPRITE');
this._sprite = spriteNode.addComponent(cc.Sprite);
spriteNode.parent = this.node;
} else {
this._sprite = spriteNode.getComponent(cc.Sprite);
}
let spriteFrame = new cc.SpriteFrame();
spriteFrame.setTexture(this._texture1);
this._sprite.spriteFrame = spriteFrame;
spriteNode.active = false;
spriteNode.scaleY = -1;
this._spriteNode = spriteNode;
let cameraNode = cc.find('TRANSITION_CAMERA', this.node);
if (!cameraNode) {
cameraNode = new cc.Node('TRANSITION_CAMERA');
this._camera = cameraNode.addComponent(cc.Camera);
cameraNode.parent = this.node;
} else {
this._camera = cameraNode.getComponent(cc.Camera);
}
cameraNode.active = false;
this._cameraNode = cameraNode;
// @ts-ignore
this.node.groupIndex = cc.Node.BuiltinGroupIndex.DEBUG - 1;
this._camera.cullingMask = 1 << this.node.groupIndex;
this.updateSpriteMaterial();
}
updateSpriteMaterial() {
if (!this._sprite) return;
let newMaterial = cc.MaterialVariant.create(this.materials[this.materialIndex], this._sprite);
newMaterial.setProperty('texture', this._texture1);
newMaterial.setProperty('texture2', this._texture2);
newMaterial.setProperty('screenSize', new Float32Array([this._texture2.width, this._texture2.height]));
this._sprite.setMaterial(0, newMaterial);
this._spriteMaterial = newMaterial;
}
_onSceneLaunched() {
window['camera'] = cc.Camera.cameras;
}
loadScene(sceneUrl, fromCameraPath, toCameraPath) {
this.scheduleOnce(() => {
cc.director.preloadScene(sceneUrl, null, () => {
this._loadScene(
sceneUrl,
fromCameraPath,
toCameraPath,
() => {},
() => {}
);
});
});
return true;
}
createTexture() {
let texture = new cc.RenderTexture();
// @ts-ignore
texture.initWithSize(cc.visibleRect.width, cc.visibleRect.height, cc.gfx.RB_FMT_D24S8);
return texture;
}
_loadScene(sceneUrl, fromCameraPath, toCameraPath, onSceneLoaded, onTransitionFinished) {
this.init();
this._spriteNode.active = true;
this._cameraNode.active = true;
let fromCameraNode = cc.find(fromCameraPath);
let fromCamera = fromCameraNode && (fromCameraNode.getComponent(cc.Camera) as any);
if (!fromCamera) {
cc.warn(`Can not find fromCamera with path ${fromCameraPath}`);
return;
}
let originTargetTexture1 = fromCamera.targetTexture;
fromCamera.cullingMask &= ~this._camera.cullingMask;
fromCamera.targetTexture = this._texture1;
fromCamera.render(cc.director.getScene());
fromCamera.targetTexture = originTargetTexture1;
return cc.director.loadScene(sceneUrl, (arg) => {
onSceneLoaded && onSceneLoaded(...arg);
let toCameraNode = cc.find(toCameraPath);
let toCamera = toCameraNode && (toCameraNode.getComponent(cc.Camera) as any);
if (!toCamera) {
cc.warn(`Can not find toCamera with path ${toCameraPath}`);
return;
}
toCamera.cullingMask &= ~this._camera.cullingMask;
let originTargetTexture2 = toCamera.targetTexture;
toCamera.targetTexture = this._texture2;
toCamera.render(cc.director.getScene());
this._camera.depth = toCamera.depth;
this._camera.clearFlags = toCamera.clearFlags;
this._onLoadFinished = () => {
toCamera.targetTexture = originTargetTexture2;
this._spriteNode.active = false;
this._cameraNode.active = false;
onTransitionFinished && onTransitionFinished();
};
this.time = 0;
this.loading = true;
});
}
update(dt) {
if (this.loading) {
this.time += dt;
if (this.time >= TRANSITION_TIME) {
this.time = TRANSITION_TIME;
this.loading = false;
this._onLoadFinished && this._onLoadFinished();
this._onLoadFinished = null;
}
this._spriteMaterial.setProperty('time', this.time / TRANSITION_TIME);
}
}
}