AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / coding / Perguntas / 78843288
Accepted
john smith
john smith
Asked: 2024-08-07 18:57:04 +0800 CST2024-08-07 18:57:04 +0800 CST 2024-08-07 18:57:04 +0800 CST

Como limpar um TextField quando ele está vinculado a um objeto SwiftData que não é opcional

  • 772

Digamos que você tenha um objeto SwiftData que não permite opcionais:

@Model
class MySet: Identifiable {
    var id: UUID
    var weight: Int
    var reps: Int
    var isCompleted: Bool
    var exercise: Exercise

Então você obtém dos dados do MySet uma lista desses MySets e os anexa a um estado:

  @State var sets: [MySet]

            if let newSets = exercise.sets {    //unwrapping the passed exercises sets
                if (!newSets.isEmpty) {     //we passed actual sets in so set them to our set
                    sets = newSets
                }

E você usa essa lista de estados em um ForEach e vincula o Textfield diretamente aos dados assim:

 ForEach($sets){ $set  in

 TextField("\(set.weight)", value: $set.weight, formatter: NumberFormatter())
                                        .keyboardType(.decimalPad)

   TextField("\(set.reps)", value: $set.reps,formatter: NumberFormatter())
                                        .keyboardType(.decimalPad)
                                    


}

Isso vinculará tudo o que estiver escrito no TextField diretamente ao objeto SwiftData, o que é um problema porque, digamos que você tenha 50 escritos e queira alterá-lo para 60, você precisa limpar o 50 para escrever o 60 e já que o objeto SwiftData não permite nulo, permitirá que você remova o 0, mas não o 5. Existe alguma maneira de remediar isso sem que o objeto swiftData permita opcionais, por exemplo, capturar a limpeza do TextField e torná-lo 0 em vez de nulo quando for limpo?

Tento consertar isso vinculando TextField a um novo estado e usando onChanged:

@State private var reps: Int = 0
@State private var weight: Int = 0

                            TextField("\(myset.weight)", value: $weight, formatter: NumberFormatter())
                                .keyboardType(.decimalPad)
                             
                                .onChange(of: weight) { newWeight in
                                    // Code to run when propertyToWatch changes to newValue
                                    if newWeight != 0 {
                                        myset.weight = newWeight
                                    }
                                }

mas agora ele está mudando cada repetição e peso de todas as séries quando eu digito uma em vez de apenas aquela em que estou digitando

swift
  • 2 2 respostas
  • 50 Views

2 respostas

  • Voted
  1. LoVo
    2024-08-07T20:53:37+08:002024-08-07T20:53:37+08:00

    Você pode criar uma ligação personalizada que deve resolver o problema:

    private func weightBinding(for set: MySet) -> Binding<Int?> {
        Binding<Int?>(
             get: { set.weight },
             set: { value in
                 if let value {
                    set.weight = value
                 }
              }
          )
    }
    
    ForEach(sets) { set in
       TextField("\(set.weight)", value: weightBinding(for: set), formatter: NumberFormatter())
    }
    
    // or as extension if you prefer:
    extension MySet {
        var weightBinding: Binding<Int?> {
            Binding<Int?>(
                get: { self.weight },
                set: { value in
                    if let value {
                        self.weight = value
                    }
                }
            )
        }
    }
    
    TextField("\(set.weight)", value: set.weightBinding, formatter: NumberFormatter())
    
    

    Assim você pode limpar o campo de texto, mas se não inserir um novo valor e parar de editar, ele usará o último valor.

    EDITAR:

    Se você deseja definir o valor como 0 quando o campo de texto estiver vazio, você precisa alterar o Binding para isto:

    
     Binding<Int?>(
             get: { set.weight },
             set: { value in
                 self.weight = value ?? 0
              }
          )
    
    
    • 1
  2. Best Answer
    Benzy Neez
    2024-08-07T20:11:46+08:002024-08-07T20:11:46+08:00

    Não sei por que seu código não está funcionando. Ajudaria ter um exemplo mínimo reproduzível para trabalhar.

    No entanto, uma maneira de chegar a uma solução seria criar um wrapper genérico para um arquivo TextField. O wrapper pode conter a variável de estado de sombra e realizar a cópia do opcional para o não opcional.

    Aqui está uma tentativa de implementar tal wrapper. Eu optei pelo Textfieldinicializador que leva ParseableFormatStyle:

    struct ValueField<F>: View
    where F: ParseableFormatStyle, F.FormatInput: Equatable, F.FormatOutput == String {
        private let titleKey: LocalizedStringKey
        @Binding private var value: F.FormatInput
        private let format: F
        private let prompt: Text?
        @State private var fieldVal: F.FormatInput?
        @FocusState private var isFocused: Bool
    
        init(
            _ titleKey: LocalizedStringKey,
            value: Binding<F.FormatInput>,
            format: F,
            prompt: Text? = nil
        ) {
            self.titleKey = titleKey
            self._value = value
            self.format = format
            self.prompt = prompt
        }
    
        var body: some View {
            TextField(titleKey, value: $fieldVal, format: format, prompt: prompt)
                .focused($isFocused)
                .onChange(of: fieldVal) { oldVal, newVal in
                    if let newVal, newVal != value {
                        value = newVal
                    }
                }
                .onChange(of: isFocused, initial: true) { oldVal, newVal in
                    if newVal {
                        fieldVal = nil
                    } else if fieldVal == nil {
                        fieldVal = value
                    }
                }
        }
    }
    

    Exemplo de uso:

    struct ContentView: View {
        @State private var val: Int = 0
    
        var body: some View {
            VStack(spacing: 40) {
                ValueField(
                    "Weight",
                    value: $val,
                    format: .number,
                    prompt: Text("Weight")
                )
                .keyboardType(.decimalPad)
    
                TextField("Dummy", text: dummy)
            }
            .textFieldStyle(.roundedBorder)
            .padding()
        }
    
        private var dummy: Binding<String> {
            Binding<String>(
                get: { "\(val)" },
                set: { _ in /* NOP */ }
            )
        }
    }
    

    Animação

    • 0

relate perguntas

  • IOS (simulador) --> Local Vapor POST Image/png: Abort.413: Payload Too Large

  • Redimensione a imagem antes de salvar no Core Data

  • Como lidar com dois manipuladores de conclusão em uma função swift

  • Por que um dicionário de chave e valores codificáveis ​​não é codificável?

  • Existe uma maneira de incorporar um assertionFailure em '?' expressão

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    Vue 3: Erro na criação "Identificador esperado, mas encontrado 'import'" [duplicado]

    • 1 respostas
  • Marko Smith

    Por que esse código Java simples e pequeno roda 30x mais rápido em todas as JVMs Graal, mas não em nenhuma JVM Oracle?

    • 1 respostas
  • Marko Smith

    Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores?

    • 1 respostas
  • Marko Smith

    Como faço para corrigir um erro MODULE_NOT_FOUND para um módulo que não importei manualmente?

    • 6 respostas
  • Marko Smith

    `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso?

    • 3 respostas
  • Marko Smith

    Quando devo usar um std::inplace_vector em vez de um std::vector?

    • 3 respostas
  • Marko Smith

    Um programa vazio que não faz nada em C++ precisa de um heap de 204 KB, mas não em C

    • 1 respostas
  • Marko Smith

    PowerBI atualmente quebrado com BigQuery: problema de driver Simba com atualização do Windows

    • 2 respostas
  • Marko Smith

    AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos

    • 1 respostas
  • Marko Smith

    Estou tentando fazer o jogo pacman usando apenas o módulo Turtle Random e Math

    • 1 respostas
  • Martin Hope
    Aleksandr Dubinsky Por que a correspondência de padrões com o switch no InetAddress falha com 'não cobre todos os valores de entrada possíveis'? 2024-12-23 06:56:21 +0800 CST
  • Martin Hope
    Phillip Borge Por que esse código Java simples e pequeno roda 30x mais rápido em todas as JVMs Graal, mas não em nenhuma JVM Oracle? 2024-12-12 20:46:46 +0800 CST
  • Martin Hope
    Oodini Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores? 2024-12-12 06:27:11 +0800 CST
  • Martin Hope
    sleeptightAnsiC `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso? 2024-11-09 07:18:53 +0800 CST
  • Martin Hope
    The Mad Gamer Quando devo usar um std::inplace_vector em vez de um std::vector? 2024-10-29 23:01:00 +0800 CST
  • Martin Hope
    Chad Feller O ponto e vírgula agora é opcional em condicionais bash com [[ .. ]] na versão 5.2? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench Por que um traço duplo (--) faz com que esta cláusula MariaDB seja avaliada como verdadeira? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng Por que `dict(id=1, **{'id': 2})` às vezes gera `KeyError: 'id'` em vez de um TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos 2024-03-20 03:12:31 +0800 CST
  • Martin Hope
    MarkB Por que o GCC gera código que executa condicionalmente uma implementação SIMD? 2024-02-17 06:17:14 +0800 CST

Hot tag

python javascript c++ c# java typescript sql reactjs html

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve