【问题标题】:How to pass an instance as a property using other properties in swift如何在swift中使用其他属性将实例作为属性传递
【发布时间】:2020-06-13 00:56:31
【问题描述】:

我是 swift 的新手,我正在尝试传递一个类 (CP) 的实例作为 swift UI 的 ContentView 结构的属性。此实例接受同一 ContentView 结构的另一个属性的参数。我希望能够从警报菜单中选择一个选项,将属性设置为一个变量,该变量将确定来自计时器的特定时间序列,并且由于此实例的参数是一个 inout 字符串,因此这些值将在实例中更新。我在这个网站上发现有人遇到类似情况,有人建议重写 viewDidLoad 函数。但是,这不适用于 SwiftUI 框架。

这是我认为足以理解问题的代码示例。 débatCP,将在其他元素中使用,因此在第一个按钮操作中声明它将不起作用。

struct ContentView: View {
@State var a = "Chronometre"
@State var partie = "Partie du débat"
@State var tempsString = "Temps ici"
@State var enCours = "CanadienParlementaire - Commencer"
@State var pausePlay = "pause"
@State var tempsMillieu = 420;
@State var tempsFermeture = 180
@State var round = 7;
@State var répartitionPM = "7/3"
var debatCP = CP(modePM: &répartitionPM)
@State private var showingAlert = false
//il va falloir un bouton pause
var body: some View {
    VStack {
        Text(String(self.round))
        Text(a)
            .font(.largeTitle)
            .fontWeight(.semibold)
            .lineLimit(nil)

        Text(partie)
        Button(action: {
            self.showingAlert = true
            if self.enCours != "Recommencer"{
            let chrono = Timer.scheduledTimer(withTimeInterval: 1.0, repeats: true) { (chrono) in
                self.debatCP.verifierEtatDebut(round: &self.round, tempsActuel: &self.tempsMillieu, pause: &self.pausePlay, partie: &self.partie, tempsStr: &self.tempsString)
                if self.round <= 2{
                     self.debatCP.verifierEtatFin(round: &self.round, tempsActuel: &self.tempsFermeture, pause: &self.pausePlay, partie: &self.partie, tempsStr: &self.tempsString)

                }
                if self.round <= 0 {
                    chrono.invalidate()
                    self.enCours = "Canadien Parlementaire - Commencer"
                }
            }
            }
            else{
                self.tempsMillieu = 420
                self.tempsFermeture = 180
                self.round = 7
            }
            self.enCours = "Recommencer"

            self.a = "CP"
        }, label: {
            Text(enCours)
                .alert(isPresented: $showingAlert) {
                    Alert(title: Text("6/4 ou 7/3"), message: Text("6/4 pour avoir plus de temps à la fin et 7/3 pour en avoir plus au début"), primaryButton: .default(Text("6/4"), action: {
                        self.répartitionPM = "6/4";
                    }), secondaryButton: .default(Text("7/3"), action: {
                        self.répartitionPM = "7/3"
                    }))
            }
        })

我收到此错误消息:无法在属性初始化程序中使用实例成员“répartitionPM”;属性初始化器在“self”可用之前运行

编辑

这是 CP 类

class CP:Debat{
init(modePM: inout String){
}
  override  var rondeFermeture:Int{
        return 2;
    }
    override var tempsTotalMillieu : Int{
        return 420;
    };
    override var tempsProtege:Int{//60
        return 60;
    }

    override var tempsLibre:Int{//360
        return 360;
    }
    override var tempsFermeture: Int{
        return 180;
}

连同一个Defat类的样本

class Debat{
var rondeFermeture:Int{
    return 0;
}
var tempsTotalMillieu:Int{//420
    return 0;
};
var tempsProtege:Int{//60
    return 0;
}
var tempsLibre:Int{//360
    return 0;
}
var tempsFermeture:Int{
    return 0
}
func formatTime(time:Int) -> String {
    let minutes:Int = time/60
    let seconds:Int = time%60
    return String(minutes) + ":" + String(seconds)
}
func prochainTour(time: inout Int, round : inout Int)->Void{
    if round > rondeFermeture {
    let tempsAvantLaFin = time % self.tempsTotalMillieu
        time -= tempsAvantLaFin
    }
    if round <= rondeFermeture{
        let tempsAvantLaFin = time % self.tempsFermeture
        time -= tempsAvantLaFin
    }
}

【问题讨论】:

  • 这是一个 inout 字符串 var debatCP = CP(modePM: &amp;"7/3")would 不能将不可变值作为 inout 参数传递:文字不可变

标签: swift swiftui


【解决方案1】:
@State var répartitionPM = "7/3"
var debatCP = CP(modePM: &répartitionPM)

SwiftUI 有不同的模式来处理@State 变量。目前尚不清楚CP 是什么,但无论如何以下是要走的路

struct AContentView: View {

    @State var repartitionPM = "7/3" // should be defined
    var debatCP:CP!=nil                  // declare-only

    init() {
        debatCP = CP(modePM: $repartitionPM) // pass as binding
    }
    ...

CP 应该是这样的

class CP:Debat{
@Binding var modePM:String

init(modePM:Binding <String>){
    self._modePM = modePM
    super.init()

}

【讨论】:

  • CP 是一个类,它继承自另一个类,所以我不认为结构在那里有用。 init(){ debatCP = CP(modePM:&amp;répartitionPM) } 在初始化所有存储属性之前使用此错误“自我”
  • @LouisCouture,class也可以包含@Binding,注意,我把绑定放在init,不是引用,其他的cmets很重要。
  • 有效!然而,当前的语法需要在我们使用它之前初始化所有属性(参见stackoverflow.com/questions/34474545/…)。这可以通过将 debatCP 设置为 nil 并使用 !将绑定变量分配给类的语法也不同。我会修改你的答案以反映变化
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2020-03-28
  • 1970-01-01
  • 2018-04-11
  • 1970-01-01
  • 2013-11-18
  • 2021-06-22
  • 1970-01-01
相关资源
最近更新 更多