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 / 79479981
Accepted
Andrew
Andrew
Asked: 2025-03-03 08:51:21 +0800 CST2025-03-03 08:51:21 +0800 CST 2025-03-03 08:51:21 +0800 CST

Como criar um CircleView "fixo" que fique preso às bordas de outras visualizações no SwiftUI?

  • 772

Como fazer com que algum ponto se torne arrastável através de um conjunto de arestas de visualização como no vídeo:

Conjunto de visualizações arrastável por pontos

Em vez de arrasto livre:

Pontos arrastáveis ​​livres

Nem sei por onde começar, pois não consigo medir os limites de uma visualização sem o GeometryReader.

Entretanto, o GeometryReader não é adequado neste caso porque são visualizações diferentes em camadas separadas.

Exemplo de visualização:

struct ContentView: View {
    @State var point: CGPoint = .zero
    
    var body: some View {
        ZStack {
            //foreach Nodes
            NodeView()
            
            NodeView()
                .offset(x:0, y:100)

            //foreach Points
            BezierPoint(p1: $point)
        }
    }
}

struct NodeView : View {
    // var nodeViewModel: NodeViewModel
    // with exact location in space

    var body: some View {
        Text("Business")
            .multilineTextAlignment(.center)
            .foregroundStyle(.red)
            .shadow(color: .black, radius: 2 )
            .frame(minHeight: 40)
            .padding( EdgeInsets(horizontal: 20, vertical: 14) )
            .background {
                // ANY Shape can be here
                RoundedRectangle(cornerRadius: 10)
            }
    }
}

struct BezierPoint: View {
    @Binding var p1: CGPoint
    
    let pointsSize: CGFloat = 15
    
    var body: some View {
        GeometryReader { reader in
            ControlPointHandle(size: pointsSize)
                .offset( CGSize(width: p1.x + reader.size.width/2, height: p1.y + reader.size.height/2) )
                .gesture(
                    DragGesture()
                        .onChanged { value in
                            self.p1 = value.location.relativeToCenter(of: reader.size, minus: true)
                        }
                )
        }
    }
}


private struct ControlPointHandle: View {
    let size: CGFloat
    
    var body: some View {
        Circle()
            .frame(width: size, height: size)
            .overlay(
                Circle()
                    .stroke(Color.blue, lineWidth: 2)
            )
            .offset(x: -size/2, y: -size/2)
    }
}


fileprivate extension CGPoint {
    func relativeToCenter(of size: CGSize, minus: Bool = false) -> CGPoint {
        let a: CGFloat = minus ? -1 : 1
        return CGPoint(x: x + a * size.width/2, y: y + a * size.height/2)
    }
}

swift
  • 1 1 respostas
  • 71 Views

1 respostas

  • Voted
  1. Best Answer
    Benzy Neez
    2025-03-03T21:52:59+08:002025-03-03T21:52:59+08:00

    A técnica mostrada na resposta para É possível detectar qual View atualmente cai sob o local de um DragGesture? pode ser usada para detectar quando uma forma está sob o ponto de arrasto (foi minha resposta). Isso usa um GeometryReaderno fundo da forma, o que deve funcionar mesmo se você tiver uma view multicamadas.

    Para encontrar o ponto ao longo da borda da forma que está mais próximo do ponto de arrasto, eu sugeriria a seguinte abordagem:

    • Determine se o ponto de arrasto está próximo da forma usando a técnica de ponto no quadro descrita na outra resposta.
    • Se o ponto de arrasto estiver próximo à forma, crie dois caminhos:
      1. Um caminho que representa o contorno da forma.
      2. Um caminho que consiste em uma linha que vai do meio da forma, passa pelo ponto de arrasto e depois vai além.
    • Use a Pathfunção lineIntersection(_:eoFill:)para encontrar a intersecção da linha com a forma.
    • O último ponto na intersecção será um ponto ao longo da borda da forma.

    Anteriormente, você estava envolvendo cada ponto com um GeometryReader. A GeometryReaderé ganancioso e consome todo o espaço disponível, então isso estava aumentando o tamanho de cada ponto para o tamanho total da visualização pai. Em vez de fazer dessa forma, eu sugeriria usar .onGeometryChangepara medir a posição de cada ponto.

    Aqui está o exemplo atualizado para mostrar que ele funciona. Ele inclui um segundo ponto, para que a independência dos pontos também possa ser testada.

    import SwiftUI
    
    struct ContentView: View {
        @State private var dragLocation: CGPoint?
        @State private var contactPoint: CGPoint?
        @State private var nearestNodeId: Int?
        @State private var previousNodeId: Int?
        
        var body: some View {
            ZStack {
                ForEach(1...5, id: \.self) { i in
                    NodeView()
                        .background {
                            SplineContactDetector(nodeId: i,
                                                  shape:  .rect(cornerRadius: 10),
                                                  dragLocation: $dragLocation,
                                                  contactPoint: $contactPoint,
                                                  nearestNodeId: $nearestNodeId,
                                                  previousNodeId: $previousNodeId
                            )
                        }
                        .offset(x:0, y: CGFloat(i) * 100 - 400)
                }
                
                //foreach Points
                BezierPoint(dragLocation: $dragLocation, contactPoint: contactPoint)
                BezierPoint(dragLocation: $dragLocation, contactPoint: contactPoint)
                    .offset(x:0, y:100)
            }
            .frame(maxWidth: .infinity, maxHeight: .infinity)
            .background(Color(red: 0.99, green: 0.94, blue: 0.76))
            .onChange(of: dragLocation) { _, newVal in
                if newVal == nil {
                    contactPoint = nil
                    nearestNodeId = nil
                    previousNodeId = nil
                }
            }
        }
    }
    
    /// ///////////////////
    /// Basic Views
    /// /////////////////
    
    struct BezierPoint: View {
        @Binding var dragLocation: CGPoint?
        let contactPoint: CGPoint?
        
        @State private var dragOffset: CGSize?
        @State private var currentOffset = CGSize.zero
        @State private var defaultFrame: CGRect?
        let pointsSize: CGFloat = 15
        
        private var offsetForContactPoint: CGSize? {
            if let contactPoint, let defaultFrame {
                CGSize(
                    width: contactPoint.x - defaultFrame.midX,
                    height: contactPoint.y - defaultFrame.midY
                )
            } else {
                nil
            }
        }
        
        private var offset: CGSize {
            let result: CGSize
            if let dragOffset {
                if let offsetForContactPoint {
                    result = offsetForContactPoint
                } else {
                    result = CGSize(
                        width: currentOffset.width + dragOffset.width,
                        height: currentOffset.height + dragOffset.height
                    )
                }
            } else {
                result = currentOffset
            }
            return result
        }
        
        var body: some View {
            Circle()
                .fill(.blue)
                .stroke(.primary, lineWidth: 2)
                .frame(width: pointsSize, height: pointsSize)
                .offset(offset)
                .gesture(
                    DragGesture(minimumDistance: 1, coordinateSpace: .global)
                        .onChanged { value in
                            dragOffset = value.translation
                            dragLocation = value.location
                        }
                        .onEnded { value in
                            if let offsetForContactPoint {
                                currentOffset = offsetForContactPoint
                            }
                            dragOffset = nil
                            dragLocation = nil
                        }
                )
                .onGeometryChange(for: CGRect.self) { proxy in
                    proxy.frame(in: .global)
                } action: { frame in
                    defaultFrame = frame
                }
        }
    }
    
    struct NodeView : View {
        var body: some View {
            Text("Business")
                .multilineTextAlignment(.center)
                .foregroundStyle(.red)
                .shadow(color: .black, radius: 2 )
                .frame(minHeight: 40)
                .padding( EdgeInsets(horizontal: 20, vertical: 14) )
                .background {
                    // ANY Shape can be here
                    RoundedRectangle(cornerRadius: 10)
                }
        }
    }
    
    /// ///////////////////
    /// Helpers
    /// /////////////////
    
    struct SplineContactDetector<S: Shape> : View {
        let nodeId: Int
        let shape: S
        
        @Binding var dragLocation: CGPoint?
        @Binding var contactPoint: CGPoint?
        @Binding var nearestNodeId: Int?
        @Binding var previousNodeId: Int?
        
        private let proximityMargin: CGFloat = 10
        
        var body: some View {
            GeometryReader { proxy in
                let frame = proxy.frame(in: .global)
                let proximity = proximity(nodeId: nodeId, frame: frame, shape: shape)
                
                Color.clear
                    .onChange(of: proximity) { _, newVal in
                        if newVal.isNearby {
                            if nearestNodeId != nodeId {
                                nearestNodeId = nodeId
                            }
                        } else if nearestNodeId == nodeId {
                            previousNodeId = nodeId
                            nearestNodeId = nil
                        }
                        if let nearestPoint = newVal.nearestPoint {
                            contactPoint = nearestPoint
                        }
                    }
            }
        }
            
        private func proximity(nodeId: Int, frame: CGRect, shape: S) -> ProximityInfo {
            let result: ProximityInfo
            if let dragLocation {
                let isNearby = frame
                    .insetBy(dx: -proximityMargin, dy: -proximityMargin)
                    .contains(dragLocation)
                if isNearby || (nearestNodeId == nil && previousNodeId == nodeId) {
                    let shapePath = shape.path(in: frame)
                    let joiningLine = Path { path in
                        path.move(to: CGPoint(x: frame.midX, y: frame.midY))
                        let dx = dragLocation.x - frame.midX
                        let dy = dragLocation.y - frame.midY
                        path.addLine(to: CGPoint(x: dx * 1000, y: dy * 1000))
                    }
                    let intersection = joiningLine.lineIntersection(shapePath)
                    result = ProximityInfo(isNearby: isNearby, nearestPoint: intersection.currentPoint)
                } else {
                    result = ProximityInfo(isNearby: false, nearestPoint: nil)
                }
            } else {
                result = ProximityInfo(isNearby: false, nearestPoint: nil)
            }
            return result
        }
        
        private struct ProximityInfo: Equatable {
            let isNearby: Bool
            let nearestPoint: CGPoint?
        }
    }
    

    Animação

    • 3

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

    Reformatar números, inserindo separadores em posições fixas

    • 6 respostas
  • Marko Smith

    Por que os conceitos do C++20 causam erros de restrição cíclica, enquanto o SFINAE antigo não?

    • 2 respostas
  • Marko Smith

    Problema com extensão desinstalada automaticamente do VScode (tema Material)

    • 2 respostas
  • Marko Smith

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

    • 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

    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
  • Martin Hope
    Fantastic Mr Fox Somente o tipo copiável não é aceito na implementação std::vector do MSVC 2025-04-23 06:40:49 +0800 CST
  • Martin Hope
    Howard Hinnant Encontre o próximo dia da semana usando o cronógrafo 2025-04-21 08:30:25 +0800 CST
  • Martin Hope
    Fedor O inicializador de membro do construtor pode incluir a inicialização de outro membro? 2025-04-15 01:01:44 +0800 CST
  • Martin Hope
    Petr Filipský Por que os conceitos do C++20 causam erros de restrição cíclica, enquanto o SFINAE antigo não? 2025-03-23 21:39:40 +0800 CST
  • Martin Hope
    Catskul O C++20 mudou para permitir a conversão de `type(&)[N]` de matriz de limites conhecidos para `type(&)[]` de matriz de limites desconhecidos? 2025-03-04 06:57:53 +0800 CST
  • Martin Hope
    Stefan Pochmann Como/por que {2,3,10} e {x,3,10} com x=2 são ordenados de forma diferente? 2025-01-13 23:24:07 +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

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