2022-10-15 18:09:25 +04:00
|
|
|
//
|
|
|
|
// GroupLinkView.swift
|
|
|
|
// SimpleX (iOS)
|
|
|
|
//
|
|
|
|
// Created by JRoberts on 15.10.2022.
|
|
|
|
// Copyright © 2022 SimpleX Chat. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import SwiftUI
|
|
|
|
import SimpleXChat
|
|
|
|
|
|
|
|
struct GroupLinkView: View {
|
2025-04-14 21:25:32 +01:00
|
|
|
@EnvironmentObject var theme: AppTheme
|
2022-10-15 18:09:25 +04:00
|
|
|
var groupId: Int64
|
2025-04-14 21:25:32 +01:00
|
|
|
@Binding var groupLink: CreatedConnLink?
|
2023-03-06 13:54:43 +00:00
|
|
|
@Binding var groupLinkMemberRole: GroupMemberRole
|
2023-10-26 18:51:45 +04:00
|
|
|
var showTitle: Bool = false
|
|
|
|
var creatingGroup: Bool = false
|
|
|
|
var linkCreatedCb: (() -> Void)? = nil
|
2025-04-14 21:25:32 +01:00
|
|
|
@State private var showShortLink = true
|
2022-12-13 17:15:45 +00:00
|
|
|
@State private var creatingLink = false
|
2022-10-15 18:09:25 +04:00
|
|
|
@State private var alert: GroupLinkAlert?
|
2024-01-21 10:09:32 +00:00
|
|
|
@State private var shouldCreate = true
|
2022-10-15 18:09:25 +04:00
|
|
|
|
|
|
|
private enum GroupLinkAlert: Identifiable {
|
|
|
|
case deleteLink
|
2024-07-28 17:54:58 +01:00
|
|
|
case error(title: LocalizedStringKey, error: LocalizedStringKey?)
|
2022-10-15 18:09:25 +04:00
|
|
|
|
|
|
|
var id: String {
|
|
|
|
switch self {
|
|
|
|
case .deleteLink: return "deleteLink"
|
|
|
|
case let .error(title, _): return "error \(title)"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var body: some View {
|
2023-10-26 18:51:45 +04:00
|
|
|
if creatingGroup {
|
2024-08-05 12:58:24 +01:00
|
|
|
groupLinkView()
|
|
|
|
.navigationBarBackButtonHidden()
|
|
|
|
.toolbar {
|
|
|
|
ToolbarItem(placement: .navigationBarTrailing) {
|
|
|
|
Button ("Continue") { linkCreatedCb?() }
|
2023-10-26 18:51:45 +04:00
|
|
|
}
|
2024-08-05 12:58:24 +01:00
|
|
|
}
|
2023-10-26 18:51:45 +04:00
|
|
|
} else {
|
|
|
|
groupLinkView()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func groupLinkView() -> some View {
|
2023-05-24 15:29:27 +02:00
|
|
|
List {
|
2023-10-26 18:51:45 +04:00
|
|
|
Group {
|
|
|
|
if showTitle {
|
|
|
|
Text("Group link")
|
|
|
|
.font(.largeTitle)
|
|
|
|
.bold()
|
|
|
|
.fixedSize(horizontal: false, vertical: true)
|
|
|
|
}
|
|
|
|
Text("You can share a link or a QR code - anybody will be able to join the group. You won't lose members of the group if you later delete it.")
|
|
|
|
}
|
|
|
|
.listRowBackground(Color.clear)
|
|
|
|
.listRowSeparator(.hidden)
|
|
|
|
.listRowInsets(EdgeInsets(top: 0, leading: 0, bottom: 0, trailing: 0))
|
|
|
|
|
2023-05-24 15:29:27 +02:00
|
|
|
Section {
|
2022-10-15 18:09:25 +04:00
|
|
|
if let groupLink = groupLink {
|
2023-05-24 15:29:27 +02:00
|
|
|
Picker("Initial role", selection: $groupLinkMemberRole) {
|
|
|
|
ForEach([GroupMemberRole.member, GroupMemberRole.observer]) { role in
|
|
|
|
Text(role.text)
|
2023-03-22 21:38:12 +00:00
|
|
|
}
|
|
|
|
}
|
2023-05-24 15:29:27 +02:00
|
|
|
.frame(height: 36)
|
2025-04-14 21:25:32 +01:00
|
|
|
SimpleXCreatedLinkQRCode(link: groupLink, short: $showShortLink)
|
|
|
|
.id("simplex-qrcode-view-for-\(groupLink.simplexChatUri(short: showShortLink))")
|
2023-05-24 15:29:27 +02:00
|
|
|
Button {
|
2025-04-14 21:25:32 +01:00
|
|
|
showShareSheet(items: [groupLink.simplexChatUri(short: showShortLink)])
|
2023-05-24 15:29:27 +02:00
|
|
|
} label: {
|
|
|
|
Label("Share link", systemImage: "square.and.arrow.up")
|
|
|
|
}
|
2022-10-15 18:09:25 +04:00
|
|
|
|
2023-10-26 18:51:45 +04:00
|
|
|
if !creatingGroup {
|
|
|
|
Button(role: .destructive) { alert = .deleteLink } label: {
|
|
|
|
Label("Delete link", systemImage: "trash")
|
|
|
|
}
|
2022-10-15 18:09:25 +04:00
|
|
|
}
|
|
|
|
} else {
|
2022-12-13 17:15:45 +00:00
|
|
|
Button(action: createGroupLink) {
|
|
|
|
Label("Create link", systemImage: "link.badge.plus")
|
|
|
|
}
|
|
|
|
.disabled(creatingLink)
|
|
|
|
if creatingLink {
|
|
|
|
ProgressView()
|
|
|
|
.scaleEffect(2)
|
|
|
|
.frame(maxWidth: .infinity)
|
|
|
|
}
|
2022-10-15 18:09:25 +04:00
|
|
|
}
|
2025-04-14 21:25:32 +01:00
|
|
|
} header: {
|
|
|
|
if let groupLink, groupLink.connShortLink != nil {
|
|
|
|
ToggleShortLinkHeader(text: Text(""), link: groupLink, short: $showShortLink)
|
|
|
|
}
|
2022-10-15 18:09:25 +04:00
|
|
|
}
|
|
|
|
.alert(item: $alert) { alert in
|
|
|
|
switch alert {
|
|
|
|
case .deleteLink:
|
|
|
|
return Alert(
|
|
|
|
title: Text("Delete link?"),
|
|
|
|
message: Text("All group members will remain connected."),
|
|
|
|
primaryButton: .destructive(Text("Delete")) {
|
|
|
|
Task {
|
|
|
|
do {
|
|
|
|
try await apiDeleteGroupLink(groupId)
|
2022-12-13 17:15:45 +00:00
|
|
|
await MainActor.run { groupLink = nil }
|
2022-10-15 18:09:25 +04:00
|
|
|
} catch let error {
|
|
|
|
logger.error("GroupLinkView apiDeleteGroupLink: \(responseError(error))")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}, secondaryButton: .cancel()
|
|
|
|
)
|
|
|
|
case let .error(title, error):
|
2024-07-28 17:54:58 +01:00
|
|
|
return mkAlert(title: title, message: error)
|
2022-10-15 18:09:25 +04:00
|
|
|
}
|
|
|
|
}
|
2023-03-06 13:54:43 +00:00
|
|
|
.onChange(of: groupLinkMemberRole) { _ in
|
|
|
|
Task {
|
|
|
|
do {
|
|
|
|
_ = try await apiGroupLinkMemberRole(groupId, memberRole: groupLinkMemberRole)
|
|
|
|
} catch let error {
|
|
|
|
let a = getErrorAlert(error, "Error updating group link")
|
|
|
|
alert = .error(title: a.title, error: a.message)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-12-13 17:15:45 +00:00
|
|
|
.onAppear {
|
2024-01-21 10:09:32 +00:00
|
|
|
if groupLink == nil && !creatingLink && shouldCreate {
|
2022-12-13 17:15:45 +00:00
|
|
|
createGroupLink()
|
|
|
|
}
|
2024-01-21 10:09:32 +00:00
|
|
|
shouldCreate = false
|
2022-12-13 17:15:45 +00:00
|
|
|
}
|
|
|
|
}
|
2024-07-03 22:42:13 +01:00
|
|
|
.modifier(ThemedBackground(grouped: true))
|
2022-12-13 17:15:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private func createGroupLink() {
|
|
|
|
Task {
|
|
|
|
do {
|
|
|
|
creatingLink = true
|
|
|
|
let link = try await apiCreateGroupLink(groupId)
|
|
|
|
await MainActor.run {
|
|
|
|
creatingLink = false
|
2023-03-06 13:54:43 +00:00
|
|
|
(groupLink, groupLinkMemberRole) = link
|
2022-12-13 17:15:45 +00:00
|
|
|
}
|
|
|
|
} catch let error {
|
|
|
|
logger.error("GroupLinkView apiCreateGroupLink: \(responseError(error))")
|
|
|
|
await MainActor.run {
|
|
|
|
creatingLink = false
|
|
|
|
let a = getErrorAlert(error, "Error creating group link")
|
|
|
|
alert = .error(title: a.title, error: a.message)
|
|
|
|
}
|
|
|
|
}
|
2022-10-15 18:09:25 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct GroupLinkView_Previews: PreviewProvider {
|
|
|
|
static var previews: some View {
|
2025-04-14 21:25:32 +01:00
|
|
|
@State var groupLink: CreatedConnLink? = CreatedConnLink(connFullLink: "https://simplex.chat/contact#/?v=1&smp=smp%3A%2F%2FPQUV2eL0t7OStZOoAsPEV2QYWt4-xilbakvGUGOItUo%3D%40smp6.simplex.im%2FK1rslx-m5bpXVIdMZg9NLUZ_8JBm8xTt%23MCowBQYDK2VuAyEALDeVe-sG8mRY22LsXlPgiwTNs9dbiLrNuA7f3ZMAJ2w%3D", connShortLink: nil)
|
|
|
|
@State var noGroupLink: CreatedConnLink? = nil
|
2022-10-15 18:09:25 +04:00
|
|
|
|
|
|
|
return Group {
|
2023-03-06 13:54:43 +00:00
|
|
|
GroupLinkView(groupId: 1, groupLink: $groupLink, groupLinkMemberRole: Binding.constant(.member))
|
|
|
|
GroupLinkView(groupId: 1, groupLink: $noGroupLink, groupLinkMemberRole: Binding.constant(.member))
|
2022-10-15 18:09:25 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|