ios - Swift 3 设备运动重力

标签 ios iphone swift gravity

我有一个 iPhone 应用程序,其中的关卡基于对 motionmanager 的设备运动调用的 gravityY 参数。我已将水平仪固定在手机的俯仰角上,因为我希望通过其 x 轴向用户展示手机相对于平面(平坦到地面)是升高还是下降......左右或旋转不相关。为此,我对应用程序进行了编程,使其沿着水平(条)滑动指示器(超出水平时为红色)……它的最大行程是水平的每一端。

级别非常好...并显示正确的值,直到用户锁定手机并将其放入他或她的后袋中。在此阶段,水平指示器移动到水平的一端(手机在口袋中升高的一端),当手机被拉出并解锁时,应用程序不会立即恢复水平 - 它仍然存在超出水平,即使我手动调用函数来恢复水平。大约 5 分钟后,电平似乎自行恢复。

代码如下:

func getElevation () {

    //now get the device orientation - want the gravity value
    if self.motionManager.isDeviceMotionAvailable {

        self.motionManager.deviceMotionUpdateInterval = 0.05

        self.motionManager.startDeviceMotionUpdates(
            to: OperationQueue.current!, withHandler: {
                deviceMotion, error -> Void in

                var gravityValueY:Double = 0

                if(error == nil) {
                    let gravityData = self.motionManager.deviceMotion
                    let gravityValueYRad = (gravityData!.gravity.y)
                    gravityValueY = round(180/(.pi) * (gravityValueYRad))

                    self.Angle.text = "\(String(round(gravityValueY)))"
                }
                else {
                    //handle the error
                    self.Angle.text = "0"
                    gravityValueY = 0
                }
                var elevationY = gravityValueY
                //limit movement of bubble
                if elevationY > 45 {
                    elevationY = 45
                }
                else if elevationY < -45 {
                    elevationY = -45
                }

                let outofLevel: UIImage? = #imageLiteral(resourceName: "levelBubble-1")
                let alignLevel: UIImage? = #imageLiteral(resourceName: "levelBubbleGR-1")

                let highElevation:Double = 1.75
                let lowElevation:Double = -1.75

                if highElevation < elevationY {
                    self.bubble.image = outofLevel
                }
                else if elevationY < lowElevation {
                    self.bubble.image = outofLevel
                }
                else {
                    self.bubble.image = alignLevel
                }
                // Move the bubble on the level
                if let bubble = self.bubble {
                    UIView.animate(withDuration: 1.5, animations: { () -> Void in
                        bubble.transform = CGAffineTransform(translationX: 0, y: CGFloat(elevationY))
                    })
                }
        })
    }
}

我希望关卡几乎立即恢复(2-3 秒内)。我无法强制校准或更新。这是我的第一篇文章....帮助表示赞赏。

编辑 - 我尝试使用以下代码设置一个没有任何动画的单独应用程序:

//

import UIKit
import CoreMotion

class ViewController: UIViewController {

let motionManager = CMMotionManager()

@IBOutlet weak var angle: UITextField!

override func viewDidLoad() {
    super.viewDidLoad()
    // Do any additional setup after loading the view, typically from a nib.
}

override func didReceiveMemoryWarning() {
    super.didReceiveMemoryWarning()
    // Dispose of any resources that can be recreated.
}

@IBAction func startLevel(_ sender: Any) {
    startLevel()
}

func startLevel() {
//now get the device orientation - want the gravity value
if self.motionManager.isDeviceMotionAvailable {

self.motionManager.deviceMotionUpdateInterval = 0.1
self.motionManager.startDeviceMotionUpdates(
to: OperationQueue.current!, withHandler: {
deviceMotion, error -> Void in

var gravityValueY:Double = 0

if(error == nil) {
    let gravityData = self.motionManager.deviceMotion
    let gravityValueYRad = (gravityData!.gravity.y)
    gravityValueY = round(180/(.pi) * (gravityValueYRad))
    }
else {
    //handle the error
     gravityValueY = 0
    }
self.angle.text = "(\(gravityValueY))"
})}
}
}

仍然以完全相同的方式表现......

最佳答案

好的....所以我通过反复试验弄明白了这一点。首先,我构建了一个 stopGravity 函数,如下所示:

    func stopGravity () {
if self.motionManager.isDeviceMotionAvailable {
    self.motionManager.stopDeviceMotionUpdates()
    }
}

我发现如果我调用该函数,级别总是设置正确,例如通过移动到新 View ,然后在返回原始 View 时重新启动更新。当锁定设备或单击主页按钮时,我需要调用相同的函数,然后在重新加载或返回 View 时重新启动重力功能。

为此,我在 viewDidLoad() 中插入了以下内容...

        NotificationCenter.default.addObserver(self, selector: #selector(stopGravity), name: NSNotification.Name.UIApplicationWillResignActive, object: nil)
    NotificationCenter.default.addObserver(self, selector: #selector(stopGravity), name: NSNotification.Name.UIApplicationWillTerminate, object: nil)

这会通知 AppDelegate 并运行函数。这立即解决了问题。

关于ios - Swift 3 设备运动重力,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47461903/

相关文章:

ios - 如何在 swift 3 中删除名为 .ALAssetsLibraryChanged 的​​观察者

ios - 如何在 Xcode 6 中使用 AutoLayout 创建动态大小的分页水平 UIScrollView

ios - 从 UITabviewController 导航回 Storyboard中的 ViewController

iphone - 我想知道我的应用程序如何访问 ios 应用程序中的 SQLite 数据库

iphone - EKEventEditViewController 不支持推送到 NavController 吗?请参阅附加的代码和错误

swift - NSBatchDeleteRequest 与 NSPredicate swift

iphone - 更新 IOS 应用程序。是否需要审查更新?

iphone - 从设备获取 NSUserDefaults 数据

ios - Xcode 7 上的 Swift 发生 fatal error

ios - 编辑导航栏的阴影线 UIImageView 无效