【iOS ARKit】PhysicsMotionComponent

2024-03-15 23:52

本文主要是介绍【iOS ARKit】PhysicsMotionComponent,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

      使用 Physics BodyComponent 组件,通过设置物理参数、物理材质、施加作用力,能完全模拟物体在真实世界中的行为,这种方式的优点是遵循物理学规律、控制精确,但缺点是不直观。使用 PhysicsMotion Component组件则可以通过直接设置速度进行物理模拟,但需要明白的是,对物体施加力与设置物体速度是两种完全不同且不相容的操作,无法混合使用。

     下面我们使用 PhysicsMotionComponent组件进行演示。在代上节码中,我们手工构建了模拟环境,这是件枯燥且容易出错的工作,而且很难构建复杂的场景,利用 Reality Composer 工具则可以快速地构建场最模型,本示例我们先使用 Reality Composer 构建基本的场景,然后通过设置速度的方式进行物理模拟。

     利用 Reality Composer 工具设置好各实体的大小、物理材质、碰撞属性和位置关系,然后在 Xcode 中导入 Reality 场景,具体代码如下。

//
//  PhysicsMotionView.swift
//  ARKitDeamo
//
//  Created by zhaoquan du on 2024/3/14.
//import SwiftUI
import RealityKit
import ARKitstruct PhysicsMotionView: View {var body: some View {PhysicsMotionViewContainer().navigationTitle("物理模拟2").edgesIgnoringSafeArea(.all)}
}struct PhysicsMotionViewContainer:UIViewRepresentable {func makeCoordinator() -> Coordinator {Coordinator()}func makeUIView(context: Context) -> some ARView {let arView = ARView(frame: .zero)let config = ARWorldTrackingConfiguration()config.planeDetection = .horizontalcontext.coordinator.arView = arViewcontext.coordinator.loadModel()arView.session.delegate  = context.coordinatorarView.session.run(config)return arView}func updateUIView(_ uiView: UIViewType, context: Context) {}class Coordinator: NSObject, ARSessionDelegate{var sphereEntity : ModelEntity!var arView:ARView? = nillet gameController = GameController()@MainActor func session(_ session: ARSession, didAdd anchors: [ARAnchor]) {guard let anchor = anchors.first as? ARPlaneAnchor,let arView = arView else{return}let planeAnchor = AnchorEntity(anchor:anchor)planeAnchor.addChild(gameController.gameAnchor)arView.scene.anchors.append(planeAnchor)gameController.gameAnchor.backWall?.visit { entity inentity.components[ModelComponent.self] = nil}gameController.gameAnchor.frontWall?.visit { entity inentity.components[ModelComponent.self] = nil}gameController.Ball13?.physicsBody?.massProperties.centerOfMass = ([0.001,0,0.001],simd_quatf(angle: 0, axis:  [0,1,0]))gameController.Ball4?.physicsBody?.material = PhysicsMaterialResource.generate(friction: 0.3, restitution: 0.3)gameController.Ball6?.physicsBody?.mode = .kinematic//gameController.Ball6?.collision?.shapes.removeAll()arView.session.delegate = nilarView.session.run(ARWorldTrackingConfiguration())}@MainActor func loadModel(){gameController.gameAnchor = try! Ball.loadBallGame()if let ball = gameController.gameAnchor.motherBall as? Entity & HasCollision {let gestureRecognizers = arView?.installGestures(.translation, for: ball)if let gestureRecognizer = gestureRecognizers?.first as? EntityTranslationGestureRecognizer {gameController.gestureRecognizer = gestureRecognizergestureRecognizer.removeTarget(nil, action: nil)gestureRecognizer.addTarget(self, action: #selector(self.handleTranslation))}}}@objcfunc handleTranslation(_ recognizer: EntityTranslationGestureRecognizer) {guard let ball = gameController.motherBall else { return }let settings = gameController.settingsif recognizer.state == .ended || recognizer.state == .cancelled {gameController.gestureStartLocation = nilball.physicsBody?.mode = .dynamicreturn}guard let gestureCurrentLocation = recognizer.translation(in: nil) else { return }guard let gestureStartLocation = gameController.gestureStartLocation else {gameController.gestureStartLocation = gestureCurrentLocationreturn}let delta = gestureStartLocation - gestureCurrentLocationlet distance = ((delta.x * delta.x) + (delta.y * delta.y) + (delta.z * delta.z)).squareRoot()if distance > settings.ballPlayDistanceThreshold {gameController.gestureStartLocation = nilball.physicsBody?.mode = .dynamicreturn}ball.physicsBody?.mode = .kinematiclet realVelocity = recognizer.velocity(in: nil)let ballParentVelocity = ball.parent!.convert(direction: realVelocity, from: nil)var clampedX = ballParentVelocity.xvar clampedZ = ballParentVelocity.z// 夹断if clampedX > settings.ballVelocityMaxX {clampedX = settings.ballVelocityMaxX} else if clampedX < settings.ballVelocityMinX {clampedX = settings.ballVelocityMinX}// 夹断if clampedZ > settings.ballVelocityMaxZ {clampedZ = settings.ballVelocityMaxZ} else if clampedZ < settings.ballVelocityMinZ {clampedZ = settings.ballVelocityMinZ}let clampedVelocity: SIMD3<Float> = [clampedX, 0.0, clampedZ]ball.physicsMotion?.linearVelocity = clampedVelocity}}
}
extension Entity {func visit(using block: (Entity) -> Void) {block(self)for child in children {child.visit(using: block)}}
}
#Preview {PhysicsMotionView()
}

       在代码中,实现的功能如下:

    (1)加载模拟场景并进行相应的处理。

    (2) 通过设置物体速度,对物体运动进行物理模拟。

      在功能1中,我们首先使用 loadModel()方法加载 Reality 场景,然后通过 session(- session: ARSesion,didAdd anchors: [ARAnchor])方法对平面检测情况进行监视,当ARKit检测到符合要求的水平平面后,将加载的场景挂载到 ARAnchor 下显示,对不需要显示的四周围栏进行了隐藏处理,然后设置了各球体的物理参数、物理材质并重启了 ARSession(为更好组织代码,方便场景管理,我们使用了 GameController类,具体可以参看本节源码)。

     在功能2中为方便控制,我们使用了 RealityKit 中的平移手势EntityTranslationGesture Recognizer,通过计算使用者手指在屏幕上滑动的速度生成物体速度,并将其作为母球的速度(为防止速度过大,我们使用了 GameSettings 结构体并定义了几个边界值,具体可以参github源码),通过直接赋予母球速度值就可以观察母球与场景中其他球体在物理引擎作用下的运动效果。

      编译后测试,使用平移手势操作母球,当母球与场景中的其他球体发生碰撞时,会产生相应的物理效果。通过本例可以看到,在 Xcode中也可以修改 Reality Composer 工具中设定的各球体的物理属性,如代码清单中第15 行到第17所示,读者也可以修改不同属性看一看它们如何影响物体的行为,取消碰撞体,看一看还能不能发生撞。

具体代码地址:GitHub - duzhaoquan/ARkitDemo

这篇关于【iOS ARKit】PhysicsMotionComponent的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/813678

相关文章

安卓链接正常显示,ios#符被转义%23导致链接访问404

原因分析: url中含有特殊字符 中文未编码 都有可能导致URL转换失败,所以需要对url编码处理  如下: guard let allowUrl = webUrl.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) else {return} 后面发现当url中有#号时,会被误伤转义为%23,导致链接无法访问

【iOS】MVC模式

MVC模式 MVC模式MVC模式demo MVC模式 MVC模式全称为model(模型)view(视图)controller(控制器),他分为三个不同的层分别负责不同的职责。 View:该层用于存放视图,该层中我们可以对页面及控件进行布局。Model:模型一般都拥有很好的可复用性,在该层中,我们可以统一管理一些数据。Controlller:该层充当一个CPU的功能,即该应用程序

iOS剪贴板同步到Windows剪贴板(无需安装软件的方案)

摘要 剪贴板同步能够提高很多的效率,免去复制、发送、复制、粘贴的步骤,只需要在手机上复制,就可以直接在电脑上 ctrl+v 粘贴,这方面在 Apple 设备中是做的非常好的,Apple 设备之间的剪贴板同步功能(Universal Clipboard)确实非常方便,它可以在 iPhone、iPad 和 Mac 之间无缝传输剪贴板内容,从而大大提高工作效率。 但是,iPhone 如何和 Wind

iOS项目发布提交出现invalid code signing entitlements错误。

1、进入开发者账号,选择App IDs,找到自己项目对应的AppId,点击进去编辑, 2、看下错误提示出现  --Specifically, value "CVYZ6723728.*" for key "com.apple.developer.ubiquity-container-identifiers" in XX is not supported.-- 这样的错误提示 将ubiquity

我的第一次份实习工作-iOS实习生-第三个月

第三个月 这个月有一个考核项目,是一个电子书阅读器,组长说很重要,是我的实习考核项目。 我的项目XTReader,这是我参考网上的一些代码,和模仿咪咕阅读做的,功能还不完善,数据的部分是用聚合数据做的。要收费的。   还有阅读页面,基本功能实现了一下。使用了autolayout,自适应布局,也是第一次用网络,第一次用数据库,第一次用自动布局。还有很多不足。 做了一周多,有个问题一直没

我的第一次份实习工作-iOS实习生-公司使用过的软件

bittorrentsync 素材,文件同步软件 cornerstone svn 软件开发合作 mark man 测量坐标的软件 SQLLite Manager 数据库操作软件

我的第一次份实习工作-iOS实习生-第二个月

第二个月 来公司过了一个月了。每天早上9点上班,到晚上6.30下班,上下班要指纹打卡,第一个月忘了打卡好多次(),然后还要去补打卡单。公司这边还安排了,工资卡办理,招商银行卡。开了一次新员工大会,认识了公司的一些过往,公司的要求等,还加了一下公司的企业QQ,还有其他的羽毛球群,篮球群。我加了下羽毛球群,也去打了一两次。第二个月的感受,感觉跟组里面的交流跟沟通都好少,基本上还有好多人不认识。想想也

我的第一次份实习工作-iOS实习生-第一个月

实习时间:2015-08-20 到 2015-12-25  实习公司;福建天棣互联有限公司 实习岗位:iOS开发实习生 第一个月: 第一天来公司,前台报道后,人资带我去我工作的地方。到了那,就由一个组长带我,当时还没有我的办公桌,组长在第三排给我找了一个位置,擦了下桌子,把旁边的准备的电脑帮我装了下,因为学的是iOS,实习生就只能用黑苹果了,这是我实习用的电脑。 帮我装了一下电脑后,开机

iOS如何隐藏系统状态栏

这里主要说明一下iOS7系统给状态栏的适配及隐藏带来的改变。 变化一: 不隐藏状态栏的情况下,StatusBar会直接显示在当前页面上,当前页面的会延伸到 StatusBar下方,顶到最上头。 这种显示方式在iOS7上是无法改变的,也无法通过设置或者配置类达到iOS6的状态栏效果。       所以在iOS7上进行页面布局的时候要考虑

ios %.2f是四舍五入吗?

实事上这个“四舍五入”并不是数学上的“四舍五入”,而是“四舍六入五成双”,英文中被称为”round half to even”或”Banker’s rounding”。 “四舍六入五成双”是指,当保留精度的下一位不是5时,按正常的四舍五入;当保留精度的下一位是5时,如果5的后面为0则舍弃;而如果5的后面还有大于0的部分时,则无论5的前一位是奇数还是偶数,都进行进位。 1.当保留精度的下一位不是