passepartout-apple/Library/Sources/UILibrary/Views/UI/EditableListSection.swift

271 lines
6.9 KiB
Swift

//
// EditableListSection.swift
// Passepartout
//
// Created by Davide De Rosa on 8/19/24.
// Copyright (c) 2024 Davide De Rosa. All rights reserved.
//
// https://github.com/passepartoutvpn
//
// This file is part of Passepartout.
//
// Passepartout is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Passepartout is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Passepartout. If not, see <http://www.gnu.org/licenses/>.
//
#if !os(tvOS)
import SwiftUI
public protocol EditableValue: Hashable, CustomStringConvertible {
static var emptyValue: Self { get }
var isEmptyValue: Bool { get }
}
extension String: EditableValue {
public static var emptyValue: String {
""
}
public var isEmptyValue: Bool {
trimmingCharacters(in: .whitespaces) == ""
}
}
public struct EditableListSection<ItemView: View, RemoveView: View, EditView: View, T: EditableValue>: View {
private let title: String
private let addTitle: String
@Binding
private var originalItems: [T]
private let emptyValue: (() async -> T)?
private let itemLabel: (Bool, Binding<T>) -> ItemView
private let removeLabel: (@escaping () -> Void) -> RemoveView
private let editLabel: () -> EditView
@State
private var items: [Item] = []
@State
private var draggingItem: Item?
@State
private var isEditing = false
public init(
_ title: String,
addTitle: String,
originalItems: Binding<[T]>,
emptyValue: (() async -> T)? = nil,
@ViewBuilder itemLabel: @escaping (Bool, Binding<T>) -> ItemView,
@ViewBuilder removeLabel: @escaping (@escaping () -> Void) -> RemoveView,
@ViewBuilder editLabel: @escaping () -> EditView
) {
self.title = title
self.addTitle = addTitle
_originalItems = originalItems
self.emptyValue = emptyValue
self.itemLabel = itemLabel
self.removeLabel = removeLabel
self.editLabel = editLabel
}
public var body: some View {
ForEach(items, id: \.id) { item in
RemovableItemRow(isEditing: isEditing) {
itemView(for: item)
} removeView: {
removeView(for: item)
}
.onDrag {
draggingItem = item
return NSItemProvider(object: item.value.description as NSString)
}
.onDrop(of: [.text], delegate: ItemDropDelegate(
item: item,
items: $items,
draggingItem: $draggingItem
))
}
.onMove {
items.move(fromOffsets: $0, toOffset: $1)
}
.onDelete {
items.remove(atOffsets: $0)
}
.onChange(of: items, perform: exportItems)
.asSectionWithHeader(title) {
#if os(iOS)
addButton
#elseif os(macOS)
editButton
addButton
#endif
}
.onLoad(perform: importItems)
}
}
private extension EditableListSection {
struct Item: Identifiable, Hashable {
let id = UUID()
var value: T
var isEmpty: Bool {
value.isEmptyValue
}
}
struct ItemDropDelegate: DropDelegate {
let item: Item
@Binding
var items: [Item]
@Binding
var draggingItem: Item?
func performDrop(info: DropInfo) -> Bool {
draggingItem = nil
return true
}
func dropEntered(info: DropInfo) {
guard let draggingItem = draggingItem, draggingItem != item else {
return
}
guard let fromIndex = items.firstIndex(of: draggingItem) else {
return
}
guard let toIndex = items.firstIndex(of: item) else {
return
}
guard fromIndex != toIndex else {
return
}
withAnimation {
items.move(fromOffsets: IndexSet(integer: fromIndex), toOffset: toIndex > fromIndex ? toIndex + 1 : toIndex)
}
}
}
}
private extension EditableListSection {
var canAdd: Bool {
if let lastItem = items.last {
return !lastItem.isEmpty
}
return true
}
var canEdit: Bool {
!items.isEmpty
}
func itemView(for item: Item) -> some View {
itemLabel(isEditing, itemValueBinding(for: item))
}
func itemValueBinding(for item: Item) -> Binding<T> {
Binding {
item.value
} set: {
guard let itemIndex = items.firstIndex(where: { $0.id == item.id }) else {
return
}
items[itemIndex].value = $0
}
}
func removeView(for item: Item) -> some View {
removeLabel {
withAnimation {
items.removeAll {
$0.id == item.id
}
}
}
}
var addButton: some View {
Button(addTitle) {
Task {
let newValue = await emptyValue?() ?? T.emptyValue
withAnimation {
items.append(Item(value: newValue))
}
}
}
.disabled(!canAdd)
}
var editButton: some View {
Toggle(isOn: $isEditing, label: editLabel)
.toggleStyle(.button)
.disabled(!canEdit)
}
}
private extension EditableListSection {
func importItems() {
items = originalItems.map(Item.init)
}
func exportItems(_ newItems: [Item]) {
let newOriginalItems = newItems.map(\.value)
guard newOriginalItems != originalItems else {
return
}
originalItems = newOriginalItems
}
}
#Preview {
struct ContentView: View {
@State
private var originalItems = ["One", "Two", "Three"]
var body: some View {
Form {
EditableListSection(
"Title",
addTitle: "Add item",
originalItems: $originalItems
) {
if $0 {
Text($1.wrappedValue)
} else {
TextField("", text: $1)
}
} removeLabel: { action in
Button("Remove", action: action)
} editLabel: {
Image(systemName: "arrow.up.arrow.down")
}
}
}
}
return ContentView()
}
#endif