VisionPro开发实践3 - ImmersiveSpace和3D手势

教程的第三篇文章拖的有点久了,以致于过年期间做了一个更精简的Demo《迎财神》,那么我就基于这个Demo来讲解一下ImmersiveSpace和3D手势吧。 可以先看一下视频: www.bilibili.com/video/BV1FK...

一、ImmersiveSpace

1.初始化

ImmersiveSpace需要一开始在初始View中就进行定义,然后通过openImmersiveSpace和dismissImmersiveSpace两个方法来进行打开和关闭。

scss 复制代码
ImmersiveSpace(id: "ImmersiveSpace") {
    ImmersiveView()
}

然后在详情View中通过"空间展示"这个按钮来进行ImmersiveSpace的展示和关闭

代码如下:

typescript 复制代码
.onChange(of: showImmersiveSpace) { _, newValue in
    Task {
        if newValue {
            switch await openImmersiveSpace(id: "ImmersiveSpace") {
            case .opened:
                immersiveSpaceIsShown = true
            case .error, .userCancelled:
                fallthrough
            @unknown default:
                immersiveSpaceIsShown = false
                showImmersiveSpace = false
            }
        } else if immersiveSpaceIsShown {
            await dismissImmersiveSpace()
            immersiveSpaceIsShown = false
        }
    }
}

2.ImmersiveSpace中的RealityView

加载模型比较简单,模型加载之后,做了三件事:

  1. 设置 旋转组件 用来让模型自动旋转
  2. 设置 输入类型 和 碰撞体积,用来支持旋转和拖拽
  3. 复制模型并放置在指定位置

接下来我会逐一讲解每一步。

二、移动与旋转手势

1. 创建Modifer

PlacementGesturesModifier是用来实现模型移动的Modifier,此代码直接Copy于官方Demo项目,不需要做修改。

DragRotationModifier是用来实现模型拖拽旋转的Modifier,此代码直接Copy于官方Demo项目。

以上两个Modifier,我会在后续的文章中继续深入讲解。

2. 在View层调用以上的Modifier

css 复制代码
ImmersiveSpace(id: "ImmersiveSpace") {
    ImmersiveView()
    .dragRotation(pitchLimit: .degrees(90))
    .placementGestures(initialPosition: Point3D([600, -600, -1800.0]))
}

在上文提到的ImmersiveView下面增加两个Modifer的调用

3. 增加模型的输入和碰撞体积

php 复制代码
scene.components[InputTargetComponent.self] = InputTargetComponent(allowedInputTypes: .all)
scene.generateCollisionShapes(recursive: true)

这两行代码非常的关键,因为如果在Reality Composer Pro中没有设置模型的输入类型和碰撞体积的,模型是无法被鼠标点击的,在真机中也无法于视线和手势交互。

基于以上三个操作,模型就可以在3D空间内被移动和拖拽旋转了。

三、自动旋转

1. 添加组件与系统

此代码依然拷贝于官方的Demo

2. 在主文件中注册组件与系统

scss 复制代码
    init() {
        RotationComponent.registerComponent()
        RotationSystem.registerSystem()
        
    }

3. 模型设置旋转系统

c 复制代码
scene.components.set(RotationComponent())

通过此行代码,保证模型可以启用RotationComponent

四、模型复制与调整位置

因为想把空间填满一些,所以粗暴的办法就是把模型复制

ini 复制代码
let upEntity = scene.clone(recursive: true)
upEntity.position.z -= 0.2
upEntity.orientation *= simd_quatf(angle: .pi, axis: [1, 0, 0])
upEntity.position.y += 2.2

let leftEntity = scene.clone(recursive: true)
leftEntity.position.x -= 0.8

let rightEntity = scene.clone(recursive: true)
rightEntity.position.x += 0.8

let downEntity = scene.clone(recursive: true)
downEntity.position.y -= 1.1

可以看到代码中,只是单纯的复制模型和移动位置。上面的模型,还做了一个沿x轴360度的旋转。

后记

终于完成了这个初级的教程,期间去了苹果实验室,在B站上认识了一个客户,帮他们顺利完成1.0版本的开发并顺利上架应用市场,苹果发售了Vision Pro,过年期间做了一个迎财神的Demo并完成了教程。

期待2024年更多的Vision Pro的应用可以落地!

相关推荐
晓得迷路了19 分钟前
栗子前端技术周刊第 97 期 - Viteland:8 月回顾、Redux Toolkit 2.9、Nuxt 4.1...
前端·javascript·nuxt.js
前端双越老师22 分钟前
前端开发 AI Agent 智能体,需要掌握哪些知识?
前端·node.js·agent
EndingCoder23 分钟前
Electron 安全性最佳实践:防范常见漏洞
前端·javascript·electron·前端框架·node.js·桌面端
学前端搞口饭吃31 分钟前
React props的使用
前端·javascript·react.js
灵感__idea1 小时前
JavaScript高级程序设计(第5版):前端的能力边界
前端·javascript·程序员
华洛1 小时前
SEO还没死,GEO之战已经开始
前端·javascript·产品
IT_陈寒1 小时前
Python性能优化:5个被低估的魔法方法让你的代码提速50%
前端·人工智能·后端
As33100101 小时前
Chrome 插件开发入门指南:从基础到实践
前端·chrome
不想上班只想要钱1 小时前
vue3 ts:声明的一个数组不能将类型“boolean”分配给类型“never”。
前端·vue.js
OEC小胖胖1 小时前
Next.js 介绍:为什么选择它来构建你的下一个 Web 应用?
开发语言·前端·web·next.js