-
-
Save Matt54/835a275bca08d782ea1b9be3fe6ab72b to your computer and use it in GitHub Desktop.
#ifndef cubeShaderTypes_h | |
#define cubeShaderTypes_h | |
#include <simd/simd.h> | |
struct CubeVertex { | |
vector_float3 position; | |
vector_float3 normal; | |
}; | |
struct CubeParams { | |
vector_float3 size; | |
uint32_t dimensions_x; | |
uint32_t dimensions_y; | |
float cubeSphereInterpolationRatio; | |
}; | |
#endif /* cubeShaderTypes_h */ |
import SwiftUI | |
import RealityKit | |
import Metal | |
struct CubeSphereState { | |
var size: SIMD3<Float> = [0.3, 0.3, 0.3] | |
var planeResolution: SIMD2<UInt32> = [16, 16] | |
var cubeSphereInterpolationRatio: Float = 0.0 | |
} | |
struct MetalCubeExample: View { | |
@State private var rootEntity: Entity? | |
@State private var mesh: LowLevelMesh? | |
@State var state: CubeSphereState = .init() | |
let device: MTLDevice | |
let commandQueue: MTLCommandQueue | |
let computePipeline: MTLComputePipelineState | |
@State var isForward: Bool = true | |
@State var time: Double = 0.0 | |
@State var timer: Timer? | |
@State var deadBandValue: Double = 0.5 | |
@State private var rotationAngles: SIMD3<Float> = [0, 0, 0] | |
@State private var lastRotationUpdateTime = CACurrentMediaTime() | |
let deadbandStep = 0.005 | |
let modulationStep: Float = 0.02 | |
let maxResolution: UInt32 = 128 | |
init() { | |
self.device = MTLCreateSystemDefaultDevice()! | |
self.commandQueue = device.makeCommandQueue()! | |
let library = device.makeDefaultLibrary()! | |
let updateFunction = library.makeFunction(name: "updateCubeMesh")! | |
self.computePipeline = try! device.makeComputePipelineState(function: updateFunction) | |
} | |
var body: some View { | |
RealityView { content in | |
let mesh = try! createMesh() | |
let resource = try! MeshResource(from: mesh) | |
let modelComponent = ModelComponent(mesh: resource, materials: [UnlitMaterial()]) | |
let entity = Entity() | |
entity.components.set(modelComponent) | |
content.add(entity) | |
self.mesh = mesh | |
updateMesh(with: state) | |
self.rootEntity = entity | |
} update: { content in | |
updateMesh(with: state) | |
} | |
.onAppear { startTimer() } | |
.onDisappear { stopTimer() } | |
} | |
private func startTimer() { | |
timer = Timer.scheduledTimer(withTimeInterval: 1/120.0, repeats: true) { _ in | |
updateRotation() | |
if deadBandValue > 0 { | |
deadBandValue -= deadbandStep | |
return | |
} | |
var ratio = state.cubeSphereInterpolationRatio | |
if isForward { | |
ratio += modulationStep | |
if ratio >= 1.0 { | |
deadBandValue = 1.0 | |
ratio = 1 | |
isForward = false | |
} | |
} else { | |
ratio -= modulationStep | |
if ratio <= 0.0 { | |
deadBandValue = 1.0 | |
ratio = 0.0 | |
isForward = true | |
} | |
} | |
state.cubeSphereInterpolationRatio = ratio | |
} | |
} | |
func updateRotation() { | |
let currentTime = CACurrentMediaTime() | |
let frameDuration = currentTime - lastRotationUpdateTime | |
self.time += frameDuration | |
// Rotate along all axis at different rates for a wonky roll effect | |
rotationAngles.x += Float(frameDuration * 0.25) | |
rotationAngles.y += Float(frameDuration * 0.125) | |
rotationAngles.z += Float(frameDuration * 0.0675) | |
let rotationX = simd_quatf(angle: rotationAngles.x, axis: [1, 0, 0]) | |
let rotationY = simd_quatf(angle: rotationAngles.y, axis: [0, 1, 0]) | |
let rotationZ = simd_quatf(angle: rotationAngles.z, axis: [0, 0, 1]) | |
rootEntity?.transform.rotation = rotationX * rotationY * rotationZ | |
lastRotationUpdateTime = currentTime | |
} | |
private func stopTimer() { | |
timer?.invalidate() | |
timer = nil | |
} | |
private var vertexCount: Int { | |
Int(state.planeResolution.x * state.planeResolution.y * 6) // 6 faces | |
} | |
private var vertexCapacity: Int { | |
Int(maxResolution * maxResolution * 6) | |
} | |
private var indexCount: Int { | |
Int(6 * (state.planeResolution.x - 1) * (state.planeResolution.y - 1) * 6) // 6 indices per quad, 6 faces | |
} | |
private var indexCapacity: Int { | |
Int(6 * (maxResolution - 1) * (maxResolution - 1) * 6) // 6 indices per quad, 6 faces | |
} | |
private func createMesh() throws -> LowLevelMesh { | |
let vertexAttributes = [ | |
LowLevelMesh.Attribute(semantic: .position, format: .float3, offset: 0), | |
LowLevelMesh.Attribute(semantic: .normal, format: .float3, offset: MemoryLayout<SIMD3<Float>>.stride) | |
] | |
let vertexLayouts = [ | |
LowLevelMesh.Layout(bufferIndex: 0, bufferStride: MemoryLayout<CubeVertex>.stride) | |
] | |
return try LowLevelMesh(descriptor: .init( | |
vertexCapacity: vertexCapacity, | |
vertexAttributes: vertexAttributes, | |
vertexLayouts: vertexLayouts, | |
indexCapacity: indexCapacity | |
)) | |
} | |
private func updateMesh(with state: CubeSphereState) { | |
guard let mesh = mesh, | |
let commandBuffer = commandQueue.makeCommandBuffer(), | |
let computeEncoder = commandBuffer.makeComputeCommandEncoder() else { return } | |
var params = CubeParams( | |
size: state.size, | |
dimensions_x: UInt32(state.planeResolution.x), | |
dimensions_y: UInt32(state.planeResolution.y), | |
cubeSphereInterpolationRatio: state.cubeSphereInterpolationRatio | |
) | |
let vertexBuffer = mesh.replace(bufferIndex: 0, using: commandBuffer) | |
let indexBuffer = mesh.replaceIndices(using: commandBuffer) | |
computeEncoder.setComputePipelineState(computePipeline) | |
computeEncoder.setBuffer(vertexBuffer, offset: 0, index: 0) | |
computeEncoder.setBuffer(indexBuffer, offset: 0, index: 1) | |
computeEncoder.setBytes(¶ms, length: MemoryLayout<CubeParams>.stride, index: 2) | |
let threadgroupSize = MTLSize(width: 64, height: 1, depth: 1) | |
let threadgroups = MTLSize( | |
width: (vertexCount + threadgroupSize.width - 1) / threadgroupSize.width, | |
height: 1, | |
depth: 1 | |
) | |
computeEncoder.dispatchThreadgroups(threadgroups, threadsPerThreadgroup: threadgroupSize) | |
computeEncoder.endEncoding() | |
commandBuffer.commit() | |
let halfSize = state.size * 0.5 | |
let bounds = BoundingBox( | |
min: -halfSize, | |
max: halfSize | |
) | |
mesh.parts.replaceAll([ | |
LowLevelMesh.Part( | |
indexCount: mesh.descriptor.indexCapacity, | |
topology: .line, | |
bounds: bounds | |
) | |
]) | |
} | |
} | |
#Preview { | |
MetalCubeExample() | |
} |
Sorry for the late reply here @rth238! Do you have a bridging header that includes the path to your header file that defines these types?
The file should include something like:
#include ".../cubeShaderTypes.h"
This bridging header file makes sure that Swift can see the C types - it's the bridge between C and Swift. You'll want to make sure that this is setup correctly in your project. Check "Objective-C Bridging Header" in Build Settings
OK, thanks, I've not had to use the bridge header or modify the build setting before, so this got me pointed in the right direction. (But instead of #include, it was #import) So thanks for this suggestion, and than thanks again for opening up all of your remarkable work by posting these examples!
Glad you got it working and happy to hear that you're enjoying the gists. I plan to create more organized resources for these soon. At the very least, including a project file would save readers from bridging header and path issues 😆
Thanks for providing these code examples; they have been very insightful! With this one, for some reason, although I have the .h file in the project, I get 3 error messages: Cannot find type 'CubeVertex' in scope and 2 of the same: Cannot find type 'CubeParams' in scope. I've removed the .h file, reinserted, etc., but it seems a setting may be preventing the use of the header file? Any suggestions would be welcomed!