如何动态更改分段控件中标签中的值? Swift 4,iOS

问题描述 投票:0回答:1

我对分段控件的实现存在一些问题。我来找你(经验较丰富的人),为了解决这个问题,我应该做些什么。我来啦:

对于“当前”选项卡,“面板的最佳方位角”和“面板的最佳倾斜角度”应该是“跟踪太阳”(我稍后将添加的代码),换句话说:这两个字段必须实时更新(再次,那段代码我会在以后再做)。 “面板的当前方位角”和“面板当前倾斜角度”基本上根据手机的位置而变化(我使用coreMotion和coreLocation来获取它们)。

问题是这个(解释后添加的图像):

对于“当前”选项卡,“面板的最佳方位角”和“面板的最佳倾斜角”字段必须随时间变化。 “面板的当前倾斜角度”必须从一开始就显示面板(手机的)倾斜,而不是这样做(仅当您选择另一个标签时才这样做)。

“面板的最佳倾斜角度”必须具有固定(但不同)的值,适用于3个月,6个月和“年”选项卡。 “面板的最佳方位角”对于3个月,6个月和“年”选项卡是相同的(该代码我将在稍后添加它)。然而,当我选择“当前”选项卡时,我遇到了问题,此处“面板的最佳方位角”获得了“跟踪太阳”的值,当我选择另一个选项卡时,该值应保持为“180º”。这种情况只发生在手机仍在水面上时,一旦我移动手机,该字段的值就会变为180º。当我从其他标签转到“当前”标签时(而不是显示“跟踪太阳”它显示“180º”,相反,当手机移动时会出现此值,但如果手机仍在,则“跟踪”太阳“显示出来了”。

基本上唯一正常工作的字段是分段控件中所有4个选项卡中的“Panel的当前方位角”。

请参考图片以获得更好的主意。

What I get in the initial view

What I want in the initial view

What I get if the phone is moving (this is correct for the 3, 6 months and year tabs)

What I get if the phone is still, if I chose the "current" tab while the phone was still and then I selected another tab

Same as previos image

所以,我的问题是:为了防止这种情况,我该怎么办?我试图在对应于分段控件的@IBAction中使用switch-case语句,但很明显它不起作用。在这里,我与您分享我用于项目的代码:

import UIKit
import CoreMotion
import CoreLocation

class PVOrientationViewController: UIViewController, CLLocationManagerDelegate {

//MARK: - Constants and Variables
let motionManager = CMMotionManager()
let locationManager = CLLocationManager()
var segIndex = 0

//MARK: - Outlets, views, actions
@IBOutlet weak var panelOptimalAz: UILabel!
@IBOutlet weak var panelOptimalTilt: UILabel!
@IBOutlet weak var panelCurrentAz: UILabel!
@IBOutlet weak var panelCurrentTilt: UILabel!

@IBOutlet weak var segmentedControl: UISegmentedControl!

//MARK: - Action of the Segmented Button.
@IBAction func indexChanged(_ sender: Any) {

    segIndex = segmentedControl.selectedSegmentIndex

    switch segIndex {

    case 0:
        print("Current Tab")
        panelOptimalAz.text = "Tracking the Sun"
        panelCurrentTilt.text = myDeviceMotion()
    case 1:
        print("3 months Tab")
        panelCurrentTilt.text = String(myDeviceMotion())

    case 2:
        print("6 months Tab")
        panelCurrentTilt.text = String(myDeviceMotion())

    case 3:
        print("Year tab")
        panelCurrentTilt.text = String(myDeviceMotion())

    default:
        panelOptimalAz.text = "No Optimal Azimuth given"
        panelCurrentAz.text = "No Current Azimuth given"
        panelCurrentTilt.text = "No Current Tilt given"
        panelOptimalTilt.text = "No Optimal Tilt given"

    }

}

//MARK: - viewDidLoad()
override func viewDidLoad() {
    super.viewDidLoad()

    locationManager.delegate = self

    // Azimuth
    if (CLLocationManager.headingAvailable()) {

        locationManager.headingFilter = 1
        locationManager.startUpdatingHeading()

    }

}

//MARK: - Motion methods
func myDeviceMotion() -> String{

    var currentTilt:String = "0.0º"
    if motionManager.isDeviceMotionAvailable {

        motionManager.deviceMotionUpdateInterval = 0.1

        motionManager.startDeviceMotionUpdates(to: OperationQueue()) {(motion, error) -> Void in

            if let attitude = motion?.attitude {

                DispatchQueue.main.async{

                    self.panelCurrentTilt.text = "\(String(format:"%.0f", attitude.pitch * 180 / Double.pi))º"

                    currentTilt = "\(String(format:"%.0f", attitude.pitch * 180 / Double.pi))º"

                }

            }

        }

        print("Device motion started")

    }else {

        print("Device motion unavailable")

    }

    return currentTilt

}

//MARK: - True heading and panel's heading
func locationManager(_ manager: CLLocationManager, didUpdateHeading heading: CLHeading) {

    let trueHeading = heading.trueHeading

    self.panelCurrentAz.text = "\(String(format: "%.0f", trueHeading))º"

    //If latitude >= 0, then panel's azimuth = 180º, else, it is 0º
    var panelHeading = 0.0
    if GlobalData.latit >= 0.0{

        self.panelOptimalAz.text = "180º"
        panelHeading = trueHeading - 180.0

        if panelHeading < 0{

            panelHeading += 360.0
            self.panelCurrentAz.text = "\(String(format: "%.0f", panelHeading))º"

        } else {

            panelHeading += 0.0
            self.panelCurrentAz.text = "\(String(format: "%.0f", panelHeading))º"

        }

    }else{

        self.panelOptimalAz.text = "0º"
        panelHeading = trueHeading + 180.0

        if panelHeading > 359.0{

            panelHeading -= 360.0
            self.panelCurrentAz.text = "\(String(format: "%.0f", panelHeading))º"

        } else {

            panelHeading += 0.0
            self.panelCurrentAz.text = "\(String(format: "%.0f", panelHeading))º"

        }

    }

}

}

提前感谢您的时间和建议。

问候。

ios swift uisegmentedcontrol segmentedcontrol
1个回答
0
投票

重述...

如果当前选定的段是“当前”,OptimalAz标签应该说“跟踪太阳”,因此在if ()中添加didUpdateHeading以仅在选择其他段时更新该标签。

启动viewDidLoad()中的运动管理器(以及位置管理器)...它将继续运行 - 并继续更新标签 - 因此不需要myDeviceMotion()功能。

viewDidLoad()结束时,触发indexChanged()初始化标签(以及任何其他所需的任务/变量),就像用户点击“当前”段一样。

尝试一下(我想我在代码中添加了足够的注释来解释任何更改):

import UIKit
import CoreMotion
import CoreLocation

class PVOrientationViewController: UIViewController, CLLocationManagerDelegate {

    //MARK: - Constants and Variables
    let motionManager = CMMotionManager()
    let locationManager = CLLocationManager()
    var segIndex = 0

    //MARK: - Outlets, views, actions
    @IBOutlet weak var panelOptimalAz: UILabel!
    @IBOutlet weak var panelOptimalTilt: UILabel!
    @IBOutlet weak var panelCurrentAz: UILabel!
    @IBOutlet weak var panelCurrentTilt: UILabel!

    @IBOutlet weak var segmentedControl: UISegmentedControl!

    //MARK: - Action of the Segmented Button.
    @IBAction func indexChanged(_ sender: Any) {

        segIndex = segmentedControl.selectedSegmentIndex

        switch segIndex {

        case 0:
            print("Current Tab")
            panelOptimalAz.text = "Tracking the Sun"

        case 1:
            print("3 months Tab")
            // do something related to 3 months

        case 2:
            print("6 months Tab")
            // do something related to 6 months

        case 3:
            print("Year tab")
            // do something related to Year

        default:
            panelOptimalAz.text = "No Optimal Azimuth given"
            panelCurrentAz.text = "No Current Azimuth given"
            panelCurrentTilt.text = "No Current Tilt given"
            panelOptimalTilt.text = "No Optimal Tilt given"

        }

    }

    //MARK: - viewDidLoad()
    override func viewDidLoad() {
        super.viewDidLoad()

        locationManager.delegate = self

        // Azimuth
        if (CLLocationManager.headingAvailable()) {

            locationManager.headingFilter = 1
            locationManager.startUpdatingHeading()

        }

        // start the motion manager here, since we want it running all the time
        if motionManager.isDeviceMotionAvailable {

            motionManager.deviceMotionUpdateInterval = 0.1

            motionManager.startDeviceMotionUpdates(to: OperationQueue()) {(motion, error) -> Void in

                if let attitude = motion?.attitude {

                    DispatchQueue.main.async{

                        self.panelCurrentTilt.text = "\(String(format:"%.0f", attitude.pitch * 180 / Double.pi))º"

                    }

                }

            }

            print("Device motion started")

        }else {

            print("Device motion unavailable")

        }

        // call segment changed to update on start (as if user tapped seg 0)
        self.indexChanged(segmentedControl)

    }

    //MARK: - Motion methods

    // no need for this
//  func myDeviceMotion() -> String{
//
//      var currentTilt:String = "0.0º"
//
//      return currentTilt
//
//  }

    //MARK: - True heading and panel's heading
    func locationManager(_ manager: CLLocationManager, didUpdateHeading heading: CLHeading) {

        let trueHeading = heading.trueHeading

        //If latitude >= 0, then panel's azimuth = 180º, else, it is 0º

        var panelHeading = 0.0

        // I don't have your GlobalData object, so
        //      just hard-coding 1.0 here...

//      if GlobalData.latit >= 0.0{
        if 1.0 >= 0.0 {

            // only update Optimal Az label if "Current" seg is NOT selected
            if self.segmentedControl.selectedSegmentIndex != 0 {
                self.panelOptimalAz.text = "180º"
            }

            panelHeading = trueHeading - 180.0

            if panelHeading < 0{

                panelHeading += 360.0

            }

            self.panelCurrentAz.text = "\(String(format: "%.0f", panelHeading))º"

        }else{

            // only update Optimal Az label if "Current" seg is NOT selected
            if self.segmentedControl.selectedSegmentIndex != 0 {
                self.panelOptimalAz.text = "0º"
            }

            panelHeading = trueHeading + 180.0

            if panelHeading > 359.0{

                panelHeading -= 360.0

            }

            self.panelCurrentAz.text = "\(String(format: "%.0f", panelHeading))º"

        }

    }

}
© www.soinside.com 2019 - 2024. All rights reserved.