不能在不可变值上使用可变成员:'self' 在 SwiftUI 中是不可变的

Sco*_*des 5 swiftui

对于以下代码,我收到以下错误。我不知道如何解决这个问题。如何调用volumeCheck()按钮单击?

struct ContentView: View {

    var player = AVAudioPlayer()

    var body: some View {
        HStack {
            Button(action: {
                self.volumeCheck()
            }) {
                Text("Click to test chimes volume")
            }
        }
    }

    mutating func volumeCheck() {
        guard let url = Bundle.main.url(
            forResource: "chimes",
            withExtension: "mp3"
            ) else { return }

        do {
            player = try AVAudioPlayer(contentsOf: url)
            player.prepareToPlay()
            player.volume = Float(self.sliderValue)
            player.play()
        } catch let error {
            print(error.localizedDescription)
        }
        print("volume checked print")
    }
}
Run Code Online (Sandbox Code Playgroud)

gla*_*oss 8

问题是它View是一个struct,它的body字段是一个具有非变异getter的计算属性。在您的代码中,会在非变异getter 中调用变异方法。所以你需要做的就是把你的播放器放到某种模型上:

class Model {
    var player: AVPlayerPlayer()
}

struct ContentView: View {

    var model = Model()

    // player can be changed from anywhere
}
Run Code Online (Sandbox Code Playgroud)

PS 在其他一些情况下,您可能希望模型中的更改反映在视图中,因此您必须在模型声明之前添加 @ObservedObject。

希望有帮助


LuL*_*aGa 1

您正尝试在volumeCheck() 中将player 设置为新对象。在初始化程序中设置播放器:

struct ContentView: View {

    private var player: AVAudioPlayer?

    public init() {
        if let url = Bundle.main.url(forResource: "chimes",
                                     withExtension: "mp3") {
            self.player = try? AVAudioPlayer(contentsOf: url)
        }
    }

    var body: some View {
        HStack {
            Button("Click to test chimes volume") {
                self.volumeCheck()
                }   .disabled(player == nil)
        }
    }

    private func volumeCheck() {
        player?.prepareToPlay()
        player?.volume = Float(self.sliderValue)
        player?.play()
        print("volume checked print")
    }
}
Run Code Online (Sandbox Code Playgroud)

请注意,您在代码中使用了 sliderValue,即使它没有在任何地方定义。