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 / 78899997
Accepted
dev_dev
dev_dev
Asked: 2024-08-22 14:06:33 +0800 CST2024-08-22 14:06:33 +0800 CST 2024-08-22 14:06:33 +0800 CST

Como interceptar o valor do NSResponder e usar essa string?

  • 772

Meu objetivo: tenho um aplicativo macOS criado e quero...

  1. Tem um botão para abrir o seletor de emojis (consegui isso)
  2. Quando o usuário seleciona um emoji, intercepte esse valor comNSResponder
  3. Em seguida, use-o na etiqueta do meu botão

O código abaixo é uma simplificação. Preciso de ajuda: Como posso interceptar o valor do NSResponder e imprimi-lo?

import SwiftUI
import AppKit

struct ContentView: View {
    private let emojiResponder = EmojiResponder()
    
    var body: some View {
        Button {
            // open the emoji picker
            if let window = NSApplication.shared.keyWindow {
                window.makeFirstResponder(emojiResponder)
                DispatchQueue.main.asyncAfter(deadline: .now() + 0.1) {
                    NSApp.orderFrontCharacterPalette(nil)
                }
            }
        } label: {
            Text("Emoji?")
        }
        .onAppear {
            emojiResponder.onEmojiSelected = { selectedEmoji in
                print(selectedEmoji)
            }
        }
    }
}

// Custom Emoji NSResponder
class EmojiResponder: NSResponder {
    var onEmojiSelected: ((String) -> Void)?
    
    // use insertTest method from NSResponder
    // I assume this allows me to get the input 
    override func insertText(_ insertString: Any) {
        guard let selectedEmoji = insertString as? String else { return }
        onEmojiSelected?(selectedEmoji)
        print(selectedEmoji)
    }
}

Alguém pode me dar uma dica sobre como resolver isso? O que acontece agora: o seletor de emoji é aberto, mas nada acontece depois de selecionar um emoji

Observação: estou usando a solução desta discussão - https://stackoverflow.com/a/77797594/5133585 . Mas espero uma solução sem um TextField oculto.


A razão pela qual não quero usar um TextField aqui:

Eu tenho uma lista que percorre um array. Em cada linha, existem 3 TextFields: [emoji] —— [nome] —— [data]

Quando seleciono uma linha e pressiono Enter no macOS, ela se concentra no campo de texto emoji. O que eu quero é focar no nome TextField. Portanto, espero remover este emoji TextField e resolver esse problema.

swift
  • 1 1 respostas
  • 60 Views

1 respostas

  • Voted
  1. Best Answer
    Sweeper
    2024-08-22T17:45:55+08:002024-08-22T17:45:55+08:00

    O seletor de emoji deve se concentrar em algo e deve ser algo sobre o qual você tenha controle para obter o que está selecionado.

    Se você não gosta de focar em um arquivo oculto TextField, pode focar no botão que abre o seletor de emojis. Crie uma NSButtonsubclasse que esteja em conformidade com NSTextInputClient, como nesta resposta , e envolva-a em um arquivo NSViewRepresentable.

    struct EmojiButton: NSViewRepresentable {
        // I chose to use a callback closure here
        // but you can easily make this into a @Binding if you prefer that
        let onSelectEmoji: (String) -> Void
        
        func makeNSView(context: Context) -> TextReceiverButton {
            context.coordinator.button
        }
        
        func updateNSView(_ nsView: TextReceiverButton, context: Context) {
            nsView.receiveText = onSelectEmoji
        }
        
        func makeCoordinator() -> Coordinator {
            Coordinator()
        }
        
        @MainActor
        class Coordinator: NSObject {
            let button = TextReceiverButton(title: "Emoji", target: nil, action: nil)
            
            override init() {
                super.init()
                button.target = self
                button.action = #selector(displayEmojiInButton(_:))
            }
            
            @objc func displayEmojiInButton(_ sender: Any) {
                NSApp.orderFrontCharacterPalette(nil)
                Task {
                    // this has to be done asynchronously, as in your attempt
                    button.window?.makeFirstResponder(button)
                }
            }
        }
    }
    
    class TextReceiverButton: NSButton, NSTextInputClient {
        var receiveText: ((String) -> Void)?
        
        nonisolated func setMarkedText(_ string: Any, selectedRange: NSRange, replacementRange: NSRange) {
            
        }
        
        nonisolated func unmarkText() {
            
        }
        
        nonisolated func selectedRange() -> NSRange {
            .init()
        }
        
        nonisolated func markedRange() -> NSRange {
            .init()
        }
        
        nonisolated func hasMarkedText() -> Bool {
            false
        }
        
        nonisolated func attributedSubstring(forProposedRange range: NSRange, actualRange: NSRangePointer?) -> NSAttributedString? {
            nil
        }
        
        nonisolated func firstRect(forCharacterRange range: NSRange, actualRange: NSRangePointer?) -> NSRect {
            .zero
        }
        
        nonisolated func characterIndex(for point: NSPoint) -> Int {
            0
        }
    
        nonisolated func insertText(_ string: Any, replacementRange: NSRange) {
            if let receivedText = string as? String {
                Task { @MainActor in
                    receiveText?(receivedText)
                }
            }
        }
        
    
        nonisolated func validAttributesForMarkedText() -> [NSAttributedString.Key] {
            return [.font, .paragraphStyle, .writingDirection]
        }
    }
    

    Como você mencionou um selecionável Listcom linhas contendo campos de texto, aqui está um exemplo de uso em uma situação semelhante. Isso simplesmente imprime o emoji selecionado.

    struct ContentView: View {
        @State var text = ""
        @State var selection: Int?
        
        var body: some View {
            List(selection: $selection) {
                ForEach(0..<10) { i in Row().tag(i) }
            }
        }
    }
    
    struct Row: View {
        @State var name = ""
        @State var date = ""
        var body: some View {
            HStack {
                EmojiButton { 
                    print("Selected", $0)
                }
                TextField("Name", text: $name)
                TextField("Date", text: $date)
            }
        }
    }
    

    Para que o seletor de emoji apareça como um pop-up, a visualização focada anteriormente antes de fazer isso orderFrontCharacterPalettedeve ser um campo de texto. Então se você quiser que o pop-up apareça no botão, você ainda terá que colocar um invisível NSTextViewno centro do botão.

    Talvez como este campo de texto esteja no lado do AppKit, o SwiftUI não saiba disso e não terá o comportamento de foco indesejável ao pressionar a tecla Enter.

    Quando o botão for pressionado, defina o socorrista para a visualização de texto invisível e, em seguida, chame de forma assíncrona orderFrontCharacterPalette.

    Aqui está um exemplo de implementação.

    struct EmojiButton: NSViewRepresentable {
        let onSelectEmoji: (String) -> Void
        
        func makeNSView(context: Context) -> NSView {
            context.coordinator.view
        }
        
        func updateNSView(_ nsView: NSView, context: Context) {
            context.coordinator.textView.receiveText = onSelectEmoji
        }
        
        func makeCoordinator() -> Coordinator {
            Coordinator()
        }
        
        @MainActor
        class Coordinator: NSObject {
            let button = NSButton(title: "Emoji", target: nil, action: nil)
            let textView = TextReceiverView(frame: .init(x: 0, y: 0, width: 1, height: 1))
            let view = NSView()
            
            override init() {
                super.init()
                
                button.target = self
                button.action = #selector(displayEmojiInButton(_:))
                
                // remove the cursor
                textView.insertionPointColor = .clear
                
                button.translatesAutoresizingMaskIntoConstraints = false
                textView.translatesAutoresizingMaskIntoConstraints = false
                // add the textView first so the button goes above it
                view.addSubview(textView)
                view.addSubview(button)
                NSLayoutConstraint.activate([
                    button.leftAnchor.constraint(equalTo: view.leftAnchor),
                    button.rightAnchor.constraint(equalTo: view.rightAnchor),
                    button.topAnchor.constraint(equalTo: view.topAnchor),
                    button.bottomAnchor.constraint(equalTo: view.bottomAnchor),
                    textView.widthAnchor.constraint(equalToConstant: 1),
                    textView.heightAnchor.constraint(equalToConstant: 1),
                    textView.centerXAnchor.constraint(equalTo: view.centerXAnchor),
                    textView.centerYAnchor.constraint(equalTo: view.centerYAnchor),
                ])
            }
            
            @objc func displayEmojiInButton(_ sender: Any) {
                button.window?.makeFirstResponder(textView)
                Task {
                    try await Task.sleep(for: .milliseconds(100))
                    NSApp.orderFrontCharacterPalette(nil)
                }
            }
        }
    }
    
    class TextReceiverView: NSTextView {
        
        var receiveText: ((String) -> Void)?
    
        override func insertText(_ string: Any, replacementRange: NSRange) {
            if let receivedText = string as? String {
                receiveText?(receivedText)
            }
        }
    }
    
    • 1

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