背景
在体验HelloWorld时,很好奇每个功能是怎么实现的,但是这个demo复用了很多功能、数据模型,刚开始理解起来就比较困难。所以我就先从功能点来看,将复用的功能、数据模型都剔除掉,保证单一功能能解藕单独运行。
环境
Xcode:15.1 beta
VisionOS:1.0
梳理功能
设置光照
我是把官方demo中的功能单独拆分出来,方便学习。
data:image/s3,"s3://crabby-images/1aa97/1aa97cbb181277c430381e2b52ecef39d0971665" alt=""
swift
import SwiftUI
import RealityKit
import RealityKitContent
struct EarthSetSunlight: View {
@State var intensity: Float = 14
@State var showSunlight = false
@State var curEarth:Entity?
var body: some View {
ZStack{
RealityView { content in
guard let earth = await RealityKitContent.entity(named: "Earth") else {
return
}
content.add(earth)
earth.scale = SIMD3(repeating: 0.3)
curEarth = earth
} update: { content in
curEarth?.setSunlight(intensity: showSunlight ? intensity : 8)
}
Toggle("Sunlight", isOn: $showSunlight)
.toggleStyle(.button)
.padding(.top, 240)
}
}
}
#Preview {
EarthSetSunlight()
}
这段代码很简单,就是一个3D模型、一个切换开关,开关控制是否要设置光照。
这里有两个知识点:
1. 加载3D资源
1.1 资源文件
1.1.1 直接拖HelloWorld里面的资源到自己的项目
data:image/s3,"s3://crabby-images/64707/64707decc7a1050e54d3c6663137ce82306f65cd" alt=""
data:image/s3,"s3://crabby-images/46d60/46d608ff636478894bf040fa9e7a3c46a9ef84bc" alt=""
将所有的资源添加到自己的工程里面默认的RealityKitContent.rkassets
文件夹下面
1.1.2 自己创建
创建一个新的visionos
工程会自带一个3D资源。
data:image/s3,"s3://crabby-images/3a361/3a361f77bcaebbe6edaca83fdc4272924a84951d" alt=""
如果要编辑资源,可以打开Package
,点击右上角Open in Reality Composer Pro
data:image/s3,"s3://crabby-images/7b66a/7b66a5c8664779ae145bef6ad5e56c861915d02f" alt=""
打开后的样子
data:image/s3,"s3://crabby-images/bde7b/bde7b4a7ec22d89cd4cc5c122b4dccdc474b5d4a" alt=""
在Reality Composer Pro
里面添加一个地球3D模型资源。
data:image/s3,"s3://crabby-images/3aa57/3aa578fdbdde2219bddc31c16e495f1dbb485e5d" alt=""
data:image/s3,"s3://crabby-images/f50c9/f50c90a09aac0c42aaa3ff38655f418f7a0bba42" alt=""
双击资源,默认会把资源添加到Scene
,资源叫Earth.usdz
也可以拖动资源到下面的资源栏,这样就不属于任何scene。
data:image/s3,"s3://crabby-images/5678e/5678e78e19206de62f9cbdf46f7635ca83d5f5fc" alt=""
如果不想放在原有的Scene
,就可以新建一个scene叫Earth
data:image/s3,"s3://crabby-images/6a937/6a93733a64adb2cacc637df8a5dd2f14f95b45b4" alt=""
将Earth.usdz
资源拖动到Earth
scene里面,就生成了一个Earth.usda
文件,这个就是后续RealityView
加载的资源。
data:image/s3,"s3://crabby-images/04e57/04e57945fade004d63417b26fae8bb547373be6b" alt=""
1.2 加载资源
目前加载3D资源,我这边统一用的是RealityKit
,平常我接触到的有两个Model3D
、RealityView
。
Model3D
更简单一点,拿来即用,就像加载Image一样。
RealityView
自定义的程度更高一些,主要用来展示Entity
的。可以编辑资源Entity
,还可以在Entity
添加各种功能的组件Component
。还可以根据属性变化,来更新Entity
。
Entity
就是3D资源对象,可以改变它的大小、位置、旋转等,还可以添加各种功能的Component
,包括手势、碰撞、粒子效果、光照效果等。
2 添加光照
经过上面的介绍,光照其实就是一个功能Component
,添加到了3D资源对象上了。
2.1 光照资源
Swift
import SwiftUI
import RealityKit
extension Entity {
/// Adds an image-based light that emulates sunlight.
///
/// This method assumes that the project contains a folder called
/// `Sunlight.skybox` that contains an image of a white dot on a black
/// background. The position of the dot in the image dictates the direction
/// from which the sunlight appears to originate. Use a small dot
/// to maximize the point-like nature of the light source.
///
/// Tune the intensity parameter to get the brightness that you need.
/// Set the intensity to `nil` to remove the image-based light (IBL)
/// from the entity.
///
/// - Parameter intensity: The strength of the sunlight. Tune
/// this value to get the brightness you want. Set a value of `nil` to
/// remove the image based light from the entity.
func setSunlight(intensity: Float?) {
if let intensity {
Task {
guard let resource = try? await EnvironmentResource(named: "Sunlight") else { return }
var iblComponent = ImageBasedLightComponent(
source: .single(resource),
intensityExponent: intensity)
// Ensure that the light rotates with its entity. Omit this line
// for a light that remains fixed relative to the surroundings.
iblComponent.inheritsRotation = true
components.set(iblComponent)
components.set(ImageBasedLightReceiverComponent(imageBasedLight: self))
}
} else {
components.remove(ImageBasedLightComponent.self)
components.remove(ImageBasedLightReceiverComponent.self)
}
}
}
注释上说的很清楚了,就是Sunlight.skybox
文件夹下面有一个图片,图片黑色背景,有一个光点,光点就代表了光源的位置,物体的位置可以理解成是正中心。
data:image/s3,"s3://crabby-images/8f2a6/8f2a678695c391e2bcfbba32609d5d667eb6c1d0" alt=""
data:image/s3,"s3://crabby-images/4e5ab/4e5ab42939cb7345f7d0c3408aff94f011471d67" alt=""
红色就表示我们要设置光照的位置,光点就是光源的位置,这里就是光源在左边。
2.2 添加Component
Entity
对象可以添加多个Component,来实现不同的效果,这里就添加的是ImageBasedLightComponent
、ImageBasedLightReceiverComponent
来实现光照。