2020-12-19 15:19:32 -05:00
|
|
|
//
|
|
|
|
// ToolbarView.swift
|
|
|
|
// Gemini-iOS
|
|
|
|
//
|
|
|
|
// Created by Shadowfacts on 12/19/20.
|
|
|
|
//
|
|
|
|
|
|
|
|
import UIKit
|
|
|
|
import BrowserCore
|
|
|
|
import Combine
|
|
|
|
|
|
|
|
class ToolbarView: UIView {
|
|
|
|
|
|
|
|
let navigator: NavigationManager
|
|
|
|
|
2020-12-20 13:45:22 -05:00
|
|
|
var showTableOfContents: (() -> Void)?
|
2020-12-19 15:19:32 -05:00
|
|
|
var showShareSheet: ((UIView) -> Void)?
|
|
|
|
var showPreferences: (() -> Void)?
|
|
|
|
|
|
|
|
private var border: UIView!
|
2021-10-16 00:16:53 -04:00
|
|
|
private var buttonsStack: UIStackView!
|
|
|
|
private var toolbarButtons: [ToolbarItem: UIButton] = [:]
|
2020-12-19 15:19:32 -05:00
|
|
|
|
|
|
|
private var cancellables = [AnyCancellable]()
|
|
|
|
|
|
|
|
init(navigator: NavigationManager) {
|
|
|
|
self.navigator = navigator
|
|
|
|
|
|
|
|
super.init(frame: .zero)
|
|
|
|
|
|
|
|
backgroundColor = .systemBackground
|
|
|
|
|
|
|
|
border = UIView()
|
|
|
|
border.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
border.backgroundColor = UIColor(white: traitCollection.userInterfaceStyle == .dark ? 0.25 : 0.75, alpha: 1)
|
|
|
|
addSubview(border)
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
border.leadingAnchor.constraint(equalTo: leadingAnchor),
|
|
|
|
border.trailingAnchor.constraint(equalTo: trailingAnchor),
|
|
|
|
border.topAnchor.constraint(equalTo: topAnchor),
|
|
|
|
border.heightAnchor.constraint(equalToConstant: 1),
|
|
|
|
])
|
|
|
|
|
2021-10-16 00:16:53 -04:00
|
|
|
buttonsStack = UIStackView()
|
|
|
|
buttonsStack.axis = .horizontal
|
|
|
|
buttonsStack.distribution = .fillEqually
|
|
|
|
buttonsStack.alignment = .fill
|
|
|
|
buttonsStack.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
addSubview(buttonsStack)
|
|
|
|
let safeAreaConstraint = buttonsStack.bottomAnchor.constraint(equalTo: safeAreaLayoutGuide.bottomAnchor)
|
2020-12-20 14:37:31 -05:00
|
|
|
safeAreaConstraint.priority = .defaultHigh
|
2020-12-19 15:19:32 -05:00
|
|
|
NSLayoutConstraint.activate([
|
2021-10-16 00:16:53 -04:00
|
|
|
buttonsStack.leadingAnchor.constraint(equalTo: leadingAnchor),
|
|
|
|
buttonsStack.trailingAnchor.constraint(equalTo: trailingAnchor),
|
|
|
|
buttonsStack.topAnchor.constraint(equalTo: topAnchor, constant: 5),
|
2020-12-20 14:37:31 -05:00
|
|
|
safeAreaConstraint,
|
2021-10-16 00:16:53 -04:00
|
|
|
buttonsStack.bottomAnchor.constraint(lessThanOrEqualTo: bottomAnchor, constant: -8)
|
2020-12-19 15:19:32 -05:00
|
|
|
])
|
|
|
|
|
2020-12-20 14:03:38 -05:00
|
|
|
updateNavigationButtons()
|
|
|
|
|
2020-12-20 13:47:49 -05:00
|
|
|
navigator.navigationOperation
|
2021-10-16 00:16:53 -04:00
|
|
|
.sink { [unowned self] (_) in
|
2020-12-20 14:03:38 -05:00
|
|
|
self.updateNavigationButtons()
|
2020-12-19 15:19:32 -05:00
|
|
|
}
|
|
|
|
.store(in: &cancellables)
|
2021-10-16 00:16:53 -04:00
|
|
|
|
|
|
|
Preferences.shared.$toolbar
|
|
|
|
.sink { [unowned self] (newValue) in
|
|
|
|
self.createToolbarButtons(newValue)
|
|
|
|
}
|
|
|
|
.store(in: &cancellables)
|
2020-12-19 15:19:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
required init?(coder: NSCoder) {
|
|
|
|
fatalError("init(coder:) has not been implemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
|
|
|
super.traitCollectionDidChange(previousTraitCollection)
|
|
|
|
|
|
|
|
border.backgroundColor = UIColor(white: traitCollection.userInterfaceStyle == .dark ? 0.25 : 0.75, alpha: 1)
|
|
|
|
}
|
|
|
|
|
2021-10-16 00:16:53 -04:00
|
|
|
private func createToolbarButtons(_ items: [ToolbarItem] = Preferences.shared.toolbar) {
|
|
|
|
toolbarButtons = [:]
|
|
|
|
buttonsStack.arrangedSubviews.forEach { $0.removeFromSuperview() }
|
2020-12-20 14:03:38 -05:00
|
|
|
|
2021-10-16 00:16:53 -04:00
|
|
|
for item in items {
|
|
|
|
let button = createButton(item)
|
|
|
|
toolbarButtons[item] = button
|
|
|
|
buttonsStack.addArrangedSubview(button)
|
|
|
|
}
|
|
|
|
|
|
|
|
updateNavigationButtons()
|
|
|
|
}
|
|
|
|
|
|
|
|
private func createButton(_ item: ToolbarItem) -> UIButton {
|
|
|
|
let button = UIButton()
|
|
|
|
let symbolConfig = UIImage.SymbolConfiguration(pointSize: 24)
|
|
|
|
button.setImage(UIImage(systemName: item.imageName, withConfiguration: symbolConfig)!, for: .normal)
|
|
|
|
button.accessibilityLabel = item.displayName
|
|
|
|
button.isPointerInteractionEnabled = true
|
|
|
|
|
|
|
|
switch item {
|
|
|
|
case .back:
|
|
|
|
button.addTarget(navigator, action: #selector(NavigationManager.goBack), for: .touchUpInside)
|
|
|
|
// fallback for when UIButton.menu isn't available
|
|
|
|
if #available(iOS 14.0, *) {
|
|
|
|
} else {
|
|
|
|
button.addInteraction(UIContextMenuInteraction(delegate: self))
|
|
|
|
}
|
|
|
|
|
|
|
|
case .forward:
|
|
|
|
button.addTarget(navigator, action: #selector(NavigationManager.goForward), for: .touchUpInside)
|
|
|
|
if #available(iOS 14.0, *) {
|
|
|
|
} else {
|
|
|
|
button.addInteraction(UIContextMenuInteraction(delegate: self))
|
|
|
|
}
|
|
|
|
|
|
|
|
case .reload:
|
|
|
|
button.addTarget(navigator, action: #selector(NavigationManager.reload), for: .touchUpInside)
|
|
|
|
|
|
|
|
case .share:
|
|
|
|
button.addTarget(self, action: #selector(sharePressed), for: .touchUpInside)
|
|
|
|
|
|
|
|
case .home:
|
|
|
|
button.addTarget(self, action: #selector(homePressed), for: .touchUpInside)
|
|
|
|
|
|
|
|
case .tableOfContents:
|
|
|
|
button.addTarget(self, action: #selector(tableOfContentsPressed), for: .touchUpInside)
|
|
|
|
|
|
|
|
case .preferences:
|
|
|
|
button.addTarget(self, action: #selector(prefsPressed), for: .touchUpInside)
|
|
|
|
}
|
|
|
|
|
|
|
|
return button
|
|
|
|
}
|
|
|
|
|
|
|
|
private func updateNavigationButtons() {
|
|
|
|
if let backButton = toolbarButtons[.back] {
|
|
|
|
backButton.isEnabled = navigator.backStack.count > 0
|
|
|
|
if #available(iOS 14.0, *) {
|
|
|
|
let back = navigator.backStack.suffix(5).enumerated().reversed().map { (index, entry) -> UIAction in
|
|
|
|
let backCount = min(5, navigator.backStack.count) - index
|
|
|
|
if #available(iOS 15.0, *),
|
|
|
|
let title = entry.title {
|
|
|
|
return UIAction(title: title, subtitle: BrowserHelper.urlForDisplay(entry.url)) { [unowned self] (_) in
|
|
|
|
self.navigator.back(count: backCount)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return UIAction(title: BrowserHelper.urlForDisplay(entry.url)) { [unowned self] (_) in
|
|
|
|
self.navigator.back(count: backCount)
|
|
|
|
}
|
2021-09-28 22:13:28 -04:00
|
|
|
}
|
2020-12-20 14:03:38 -05:00
|
|
|
}
|
2021-10-16 00:16:53 -04:00
|
|
|
backButton.menu = UIMenu(children: back)
|
2020-12-20 14:03:38 -05:00
|
|
|
}
|
2021-10-16 00:16:53 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if let forwardsButton = toolbarButtons[.forward] {
|
|
|
|
forwardsButton.isEnabled = navigator.forwardStack.count > 0
|
|
|
|
if #available(iOS 14.0, *) {
|
|
|
|
let forward = navigator.forwardStack.prefix(5).enumerated().map { (index, entry) -> UIAction in
|
|
|
|
let forwardCount = index + 1
|
|
|
|
if #available(iOS 15.0, *),
|
|
|
|
let title = entry.title {
|
|
|
|
return UIAction(title: title, subtitle: BrowserHelper.urlForDisplay(entry.url)) { [unowned self] (_) in
|
|
|
|
self.navigator.forward(count: forwardCount)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return UIAction(title: BrowserHelper.urlForDisplay(entry.url)) { [unowned self] (_) in
|
|
|
|
self.navigator.forward(count: forwardCount)
|
|
|
|
}
|
2021-09-28 22:13:28 -04:00
|
|
|
}
|
2020-12-20 14:03:38 -05:00
|
|
|
}
|
2021-10-16 00:16:53 -04:00
|
|
|
forwardsButton.menu = UIMenu(children: forward)
|
2020-12-20 14:03:38 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-20 13:45:22 -05:00
|
|
|
@objc private func tableOfContentsPressed() {
|
|
|
|
showTableOfContents?()
|
|
|
|
}
|
|
|
|
|
2020-12-19 15:19:32 -05:00
|
|
|
@objc private func sharePressed() {
|
2021-10-16 00:16:53 -04:00
|
|
|
showShareSheet?(toolbarButtons[.share]!)
|
2020-12-19 15:19:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
@objc private func prefsPressed() {
|
|
|
|
showPreferences?()
|
|
|
|
}
|
2021-10-16 00:16:53 -04:00
|
|
|
|
|
|
|
@objc private func homePressed() {
|
|
|
|
navigator.changeURL(Preferences.shared.homepage)
|
|
|
|
}
|
2020-12-19 15:19:32 -05:00
|
|
|
|
|
|
|
}
|
2020-12-20 14:03:38 -05:00
|
|
|
|
|
|
|
extension ToolbarView: UIContextMenuInteractionDelegate {
|
|
|
|
func contextMenuInteraction(_ interaction: UIContextMenuInteraction, configurationForMenuAtLocation location: CGPoint) -> UIContextMenuConfiguration? {
|
2021-09-28 22:13:28 -04:00
|
|
|
// this path is only used on <iOS 14, on >=iOS 14, we don't create a UIContextMenuInteraction
|
2021-10-16 00:16:53 -04:00
|
|
|
if let backButton = toolbarButtons[.back],
|
|
|
|
interaction.view == backButton {
|
2020-12-20 14:03:38 -05:00
|
|
|
return UIContextMenuConfiguration(identifier: nil, previewProvider: { nil }) { (_) -> UIMenu? in
|
2021-09-28 22:13:28 -04:00
|
|
|
let children = self.navigator.backStack.suffix(5).enumerated().map { (index, entry) in
|
2021-09-30 11:05:35 -04:00
|
|
|
UIAction(title: BrowserHelper.urlForDisplay(entry.url)) { (_) in
|
2020-12-20 14:03:38 -05:00
|
|
|
self.navigator.back(count: min(5, self.navigator.backStack.count) - index)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UIMenu(title: "", image: nil, identifier: nil, options: [], children: children)
|
|
|
|
}
|
2021-10-16 00:16:53 -04:00
|
|
|
} else if let forwardsButton = toolbarButtons[.forward],
|
|
|
|
interaction.view == forwardsButton {
|
2020-12-20 14:03:38 -05:00
|
|
|
return UIContextMenuConfiguration(identifier: nil, previewProvider: { nil }) { (_) -> UIMenu? in
|
2021-09-28 22:13:28 -04:00
|
|
|
let children = self.navigator.forwardStack.prefix(5).enumerated().map { (index, entry) -> UIAction in
|
2020-12-20 14:03:38 -05:00
|
|
|
let forwardCount = index + 1
|
2021-09-30 11:05:35 -04:00
|
|
|
return UIAction(title: BrowserHelper.urlForDisplay(entry.url)) { (_) in
|
2020-12-20 14:03:38 -05:00
|
|
|
self.navigator.forward(count: forwardCount)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UIMenu(title: "", image: nil, identifier: nil, options: [], children: children)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|