如何处理小部件上的点击手势?

时间:2020-09-26 11:26:58

标签: swiftui widgetkit

我正在使用WidgetKit编写小部件,并且希望使小部件的内容可单击。例如,如果用户单击排名,我想在应用程序激活时打开“排名”标签。

我尝试在应用程序和窗口小部件之间使用通知,但点击手势不起作用,我在点击手势内添加了打印,但未在控制台中显示。另外,我向他们两个都添加了相同的应用组。

WidgetView:

struct LargeWidget : View {
    
    @State var standings : [StandingsTable]
    
    var body: some View {
        VStack(alignment:.leading){
            if standings.count > 0{
                HStack(spacing:5){
                    Text("#").foregroundColor(.gray)
                        .frame(width: 30)
                    Text("Team".localized).foregroundColor(.gray)
                    Spacer()
                    Text("_D".localized).foregroundColor(.gray)
                        .frame(width: 30)
                    Text("_L".localized).foregroundColor(.gray)
                        .frame(width: 30)
                    Text("_W".localized).foregroundColor(.gray)
                        .frame(width: 30)
                    Text("_P".localized).foregroundColor(.gray)
                        .frame(width: 30)
                }
                Divider()
                ForEach(0..<5, id: \.self) { i in
                    HStack(spacing:5){
                        Text(standings[i].rank)
                            .font(.system(size: 15))
                            .padding(.vertical, 3)
                            .frame(width: 30)
                            .background(Color(UIColor.systemBackground))
                            .cornerRadius(4)
                        Text(standings[i].name)
                            .lineLimit(1)
                            .padding(.leading, 5)
                        Spacer()
                        Text(standings[i].drawn)
                            .frame(width: 30)
                        Text(standings[i].lost)
                            .frame(width: 30)
                        Text(standings[i].won)
                            .frame(width: 30)
                        Text(standings[i].points)
                            .frame(width: 30)
                    }
                    .padding(.vertical, 5)
                    .background(standings[i].name == "Besiktas" ? Color(UIColor.systemGray6) : Color.clear)
                    .cornerRadius(8)
                    
                }
                Spacer(minLength: 0)
            }else{
                Text("Large")
                    .padding()
            }
        }.padding()
        .onTapGesture {
            print("clicked to standings")
            DispatchQueue.main.async(execute: {
                NotificationCenter.default.post(name: NSNotification.Name("standings"), object: nil, userInfo: nil)
            })
        }
    }
}

,这里是应用程序中的ContentView:

import SwiftUI

extension NSNotification {
    static let openStandings = NSNotification.Name.init("standings")
}

struct ContentView: View {
    
    @State var show: Bool = false
    
    var body: some View {
        NavigationView{
            Text("Hello, world!")
                .padding()
        }.sheet(isPresented: self.$show) {
            VStack{
                Text("Notification")
                    .padding()
            }
        }
        .onReceive(NotificationCenter.default.publisher(for: NSNotification.openStandings))
        { obj in
            self.show.toggle()
        }
    }
}

Screenshot of Widget

1 个答案:

答案 0 :(得分:0)

好的,我用SwiftUI环境(不是旧的AppDelegate和SceneDelegate)创建了一个新项目。

然后,我使用Link进行点击操作,并在ContentView中使用.onOpenURL修饰符得到它。它有效:)

ContentView:

import SwiftUI

enum SelectedTab: Hashable {
    case home
    case standings
    case options
}

struct ContentView: View {
    
    @State var selectedTab: SelectedTab = .home
    
    var body: some View {
        TabView(selection: self.$selectedTab) {
            Text("Home")
                .tabItem {
                    Image(systemName: "house")
                        .renderingMode(.template)
                    Text("Home")
                }.tag(SelectedTab.home)
            Text("Standings")
                .tabItem {
                    Image(systemName: "list.number")
                        .renderingMode(.template)
                    Text("Standings")
                }.tag(SelectedTab.standings)
            Text("Options")
                .tabItem {
                    Image(systemName: "gear")
                        .renderingMode(.template)
                    Text("Options")
                }.tag(SelectedTab.options)
                .onOpenURL { (url) in
                    if url.absoluteString == "widget-deeplink://standings"{
                        self.selectedTab = .standings
                    }
                }
        }
    }
}

小部件中的链接用法示例:

Link(destination: URL(string: "widget-deeplink://standings")!) {
    Text("Link Test")
}