ScreenShotComponent.ts 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. import { _decorator, Camera, Component, Node, RenderTexture, Sprite, SpriteFrame, Vec3 } from 'cc';
  2. import { AliensGlobalInstance } from '../AliensGlobalInstance';
  3. import { EventDispatcher } from 'db://assets/core_tgx/easy_ui_framework/EventDispatcher';
  4. import { GameEvent } from '../Enum/GameEvent';
  5. const { ccclass, property } = _decorator;
  6. /** 截图组件脚本*/
  7. @ccclass('ScreenShotComponent')
  8. export class ScreenShotComponent extends Component {
  9. @property(Sprite)
  10. public sprite: Sprite = null!;
  11. _renderTex: RenderTexture | null = null;
  12. private _originalTargetTexture: RenderTexture | null = null;
  13. private _originalCameraPosition: Vec3 = new Vec3();
  14. private _originalCameraRotation: Vec3 = new Vec3();
  15. async start() {
  16. this.registerEvent();
  17. //获取相机组件
  18. const camera = await this.getSceneCamera();
  19. this._originalTargetTexture = camera.targetTexture;
  20. }
  21. private registerEvent(){
  22. EventDispatcher.instance.on(GameEvent.EVENT_CAMERA_SCREENSHOT,this.screenShot,this);
  23. }
  24. private unregisterEvent(){
  25. EventDispatcher.instance.off(GameEvent.EVENT_CAMERA_SCREENSHOT,this.screenShot,this);
  26. }
  27. //截图
  28. public async screenShot() {
  29. this.node.active = true;
  30. //获取相机组件
  31. const camera = await this.getSceneCamera();
  32. //获取目标节点
  33. const targetNode = await this.getTargetNode();
  34. const spriteFrame = this.sprite.spriteFrame!;
  35. const sp = new SpriteFrame();
  36. sp.reset({
  37. originalSize: spriteFrame.originalSize,
  38. rect: spriteFrame.rect,
  39. offset: spriteFrame.offset,
  40. isRotate: spriteFrame.rotated,
  41. borderTop: spriteFrame.insetTop,
  42. borderLeft: spriteFrame.insetLeft,
  43. borderBottom: spriteFrame.insetBottom,
  44. borderRight: spriteFrame.insetRight,
  45. });
  46. const renderTex = this._renderTex = new RenderTexture();
  47. renderTex.reset({
  48. width: 280, // 直接设置为最终大小
  49. height: 180,
  50. });
  51. if (targetNode) {
  52. // 设置渲染纹理
  53. sp.texture = renderTex;
  54. // 设置flipUV来翻转图像
  55. sp.flipUVY = true;
  56. this.sprite.spriteFrame = sp;
  57. // 移动相机对准目标节点
  58. const targetPos = targetNode.worldPosition.clone();
  59. camera.node.lookAt(targetPos, Vec3.UP);
  60. camera.targetTexture = renderTex;
  61. // 确保渲染完成
  62. this.scheduleOnce(() => {
  63. // 恢复相机状态
  64. camera.targetTexture = this._originalTargetTexture;
  65. camera.node.setWorldPosition(this._originalCameraPosition);
  66. camera.node.setRotationFromEuler(this._originalCameraRotation);
  67. // 强制更新材质
  68. this.sprite.markForUpdateRenderData();
  69. }, 0.1); // 稍微延长等待时间确保渲染完成
  70. }
  71. }
  72. //获取场景相机
  73. private async getSceneCamera() :Promise<Camera>{
  74. return new Promise<Camera>((resolve, reject) => {
  75. const levelNode = AliensGlobalInstance.instance.levels.children[0];
  76. if(!levelNode){return;}
  77. const camera = levelNode.getComponentInChildren(Camera)!;
  78. resolve(camera);
  79. });
  80. }
  81. //获取目标节点
  82. private async getTargetNode():Promise<Node> {
  83. return new Promise<Node>((resolve, reject) => {
  84. const levelNode = AliensGlobalInstance.instance.levels.children[0];
  85. const et = levelNode.getChildByName('et');
  86. resolve(et.children[0]);
  87. });
  88. }
  89. //更新相机最新的位置和旋转角度
  90. public saveCameraState(pos:Vec3,rotation:Vec3){
  91. this._originalCameraPosition = pos;
  92. this._originalCameraRotation = rotation;
  93. }
  94. protected onDestroy(): void {
  95. this.unregisterEvent();
  96. }
  97. }