2023-01-16 19:51:05 +01:00
|
|
|
import Account
|
2023-01-16 21:15:33 +01:00
|
|
|
import AppAccount
|
2023-01-17 11:36:01 +01:00
|
|
|
import DesignSystem
|
|
|
|
import Env
|
2023-01-25 21:18:34 +01:00
|
|
|
import Models
|
2023-01-27 20:36:40 +01:00
|
|
|
import SwiftUI
|
2024-01-04 16:21:15 +01:00
|
|
|
import SwiftUIIntrospect
|
2023-01-16 19:51:05 +01:00
|
|
|
|
2023-09-19 09:18:20 +02:00
|
|
|
@MainActor
|
2023-01-16 19:51:05 +01:00
|
|
|
struct SideBarView<Content: View>: View {
|
2023-10-23 19:12:25 +02:00
|
|
|
@Environment(\.openWindow) private var openWindow
|
2024-01-04 16:21:15 +01:00
|
|
|
@Environment(\.horizontalSizeClass) private var horizontalSizeClass
|
2023-11-01 18:58:44 +01:00
|
|
|
|
2023-09-18 07:01:23 +02:00
|
|
|
@Environment(AppAccountsManager.self) private var appAccounts
|
|
|
|
@Environment(CurrentAccount.self) private var currentAccount
|
2023-09-18 21:03:52 +02:00
|
|
|
@Environment(Theme.self) private var theme
|
2023-09-18 07:01:23 +02:00
|
|
|
@Environment(StreamWatcher.self) private var watcher
|
2023-09-19 09:18:20 +02:00
|
|
|
@Environment(UserPreferences.self) private var userPreferences
|
2023-09-18 07:01:23 +02:00
|
|
|
@Environment(RouterPath.self) private var routerPath
|
2024-02-14 12:48:14 +01:00
|
|
|
|
2024-09-10 06:53:19 +02:00
|
|
|
@Binding var selectedTab: AppTab
|
|
|
|
var tabs: [AppTab]
|
2023-01-16 21:15:33 +01:00
|
|
|
@ViewBuilder var content: () -> Content
|
2024-02-14 12:48:14 +01:00
|
|
|
|
2024-01-10 13:26:55 +01:00
|
|
|
@State private var sidebarTabs = SidebarTabs.shared
|
2023-01-17 11:36:01 +01:00
|
|
|
|
2024-09-10 06:53:19 +02:00
|
|
|
private func badgeFor(tab: AppTab) -> Int {
|
2023-09-16 14:15:03 +02:00
|
|
|
if tab == .notifications, selectedTab != tab,
|
2023-02-21 07:23:42 +01:00
|
|
|
let token = appAccounts.currentAccount.oauthToken
|
|
|
|
{
|
2023-09-19 08:44:11 +02:00
|
|
|
return watcher.unreadNotificationsCount + (userPreferences.notificationsCount[token] ?? 0)
|
2023-01-17 13:02:05 +01:00
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
2023-02-21 07:23:42 +01:00
|
|
|
|
2024-09-10 06:53:19 +02:00
|
|
|
private func makeIconForTab(tab: AppTab) -> some View {
|
2024-09-11 09:50:32 +02:00
|
|
|
ZStack(alignment: .topTrailing) {
|
|
|
|
HStack {
|
2024-05-08 11:51:28 +02:00
|
|
|
SideBarIcon(systemIconName: tab.iconName,
|
|
|
|
isSelected: tab == selectedTab)
|
2024-09-06 11:32:49 +02:00
|
|
|
if userPreferences.isSidebarExpanded {
|
|
|
|
Text(tab.title)
|
|
|
|
.font(.headline)
|
|
|
|
.foregroundColor(tab == selectedTab ? theme.tintColor : theme.labelColor)
|
|
|
|
.frame(maxWidth: .infinity, alignment: .leading)
|
2024-05-08 11:51:28 +02:00
|
|
|
}
|
|
|
|
}
|
2024-09-06 11:32:49 +02:00
|
|
|
.frame(width: (userPreferences.isSidebarExpanded ? .sidebarWidthExpanded : .sidebarWidth) - 24, height: 50)
|
|
|
|
.background(tab == selectedTab ? theme.primaryBackgroundColor : .clear,
|
|
|
|
in: RoundedRectangle(cornerRadius: 8))
|
|
|
|
.cornerRadius(8)
|
|
|
|
.shadow(color: tab == selectedTab ? .black.opacity(0.2) : .clear, radius: 5)
|
|
|
|
.overlay(
|
|
|
|
RoundedRectangle(cornerRadius: 8)
|
|
|
|
.stroke(tab == selectedTab ? theme.labelColor.opacity(0.1) : .clear, lineWidth: 1)
|
|
|
|
)
|
|
|
|
let badge = badgeFor(tab: tab)
|
|
|
|
if badge > 0 {
|
2024-09-11 09:50:32 +02:00
|
|
|
makeBadgeView(count: badge)
|
2023-01-17 13:02:05 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-02-21 07:23:42 +01:00
|
|
|
|
2023-02-18 15:36:18 +01:00
|
|
|
private func makeBadgeView(count: Int) -> some View {
|
|
|
|
ZStack {
|
|
|
|
Circle()
|
|
|
|
.fill(.red)
|
2023-02-23 18:42:34 +01:00
|
|
|
Text(count > 99 ? "99+" : String(count))
|
2023-02-18 15:36:18 +01:00
|
|
|
.foregroundColor(.white)
|
|
|
|
.font(.caption2)
|
|
|
|
}
|
2023-02-23 18:42:34 +01:00
|
|
|
.frame(width: 24, height: 24)
|
2024-09-11 09:50:32 +02:00
|
|
|
.offset(x: 5, y: -5)
|
2023-02-18 15:36:18 +01:00
|
|
|
}
|
2023-01-22 06:38:30 +01:00
|
|
|
|
2023-01-17 13:02:05 +01:00
|
|
|
private var postButton: some View {
|
|
|
|
Button {
|
2024-01-09 13:28:57 +01:00
|
|
|
#if targetEnvironment(macCatalyst) || os(visionOS)
|
2023-12-18 08:22:59 +01:00
|
|
|
openWindow(value: WindowDestinationEditor.newStatusEditor(visibility: userPreferences.postVisibility))
|
|
|
|
#else
|
|
|
|
routerPath.presentedSheet = .newStatusEditor(visibility: userPreferences.postVisibility)
|
|
|
|
#endif
|
2023-01-17 13:02:05 +01:00
|
|
|
} label: {
|
|
|
|
Image(systemName: "square.and.pencil")
|
|
|
|
.resizable()
|
|
|
|
.aspectRatio(contentMode: .fit)
|
|
|
|
.frame(width: 20, height: 30)
|
2024-01-05 08:50:51 +01:00
|
|
|
.offset(x: 2, y: -2)
|
2023-01-17 13:02:05 +01:00
|
|
|
}
|
|
|
|
.buttonStyle(.borderedProminent)
|
2024-09-10 06:53:19 +02:00
|
|
|
.help(AppTab.post.title)
|
2023-01-17 13:02:05 +01:00
|
|
|
}
|
2023-01-22 06:38:30 +01:00
|
|
|
|
2023-02-18 15:36:18 +01:00
|
|
|
private func makeAccountButton(account: AppAccount, showBadge: Bool) -> some View {
|
2023-01-17 19:41:46 +01:00
|
|
|
Button {
|
|
|
|
if account.id == appAccounts.currentAccount.id {
|
|
|
|
selectedTab = .profile
|
2023-11-07 11:22:36 +01:00
|
|
|
SoundEffectManager.shared.playSound(.tabSelection)
|
2023-01-17 19:41:46 +01:00
|
|
|
} else {
|
2023-02-09 18:48:31 +01:00
|
|
|
var transation = Transaction()
|
|
|
|
transation.disablesAnimations = true
|
|
|
|
withTransaction(transation) {
|
2023-01-17 19:41:46 +01:00
|
|
|
appAccounts.currentAccount = account
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} label: {
|
2023-02-18 15:36:18 +01:00
|
|
|
ZStack(alignment: .topTrailing) {
|
2024-05-08 11:51:28 +02:00
|
|
|
if userPreferences.isSidebarExpanded {
|
2024-08-01 08:58:54 +02:00
|
|
|
AppAccountView(viewModel: .init(appAccount: account,
|
2024-05-08 11:51:28 +02:00
|
|
|
isCompact: false,
|
|
|
|
isInSettings: false),
|
|
|
|
isParentPresented: .constant(false))
|
|
|
|
} else {
|
2024-08-01 08:58:54 +02:00
|
|
|
AppAccountView(viewModel: .init(appAccount: account,
|
2024-05-08 11:51:28 +02:00
|
|
|
isCompact: true,
|
|
|
|
isInSettings: false),
|
|
|
|
isParentPresented: .constant(false))
|
|
|
|
}
|
2024-05-13 09:27:24 +02:00
|
|
|
if !userPreferences.isSidebarExpanded,
|
|
|
|
showBadge,
|
2023-02-21 07:23:42 +01:00
|
|
|
let token = account.oauthToken,
|
2023-09-19 08:44:11 +02:00
|
|
|
let notificationsCount = userPreferences.notificationsCount[token],
|
|
|
|
notificationsCount > 0
|
2023-02-21 07:23:42 +01:00
|
|
|
{
|
2023-09-19 08:44:11 +02:00
|
|
|
makeBadgeView(count: notificationsCount)
|
2023-02-18 15:36:18 +01:00
|
|
|
}
|
|
|
|
}
|
2024-05-08 11:51:28 +02:00
|
|
|
.padding(.leading, userPreferences.isSidebarExpanded ? 16 : 0)
|
2023-01-17 19:41:46 +01:00
|
|
|
}
|
2024-05-08 10:38:27 +02:00
|
|
|
.help(accountButtonTitle(accountName: account.accountName))
|
2024-05-08 11:51:28 +02:00
|
|
|
.frame(width: userPreferences.isSidebarExpanded ? .sidebarWidthExpanded : .sidebarWidth, height: 50)
|
2023-01-17 19:41:46 +01:00
|
|
|
.padding(.vertical, 8)
|
|
|
|
.background(selectedTab == .profile && account.id == appAccounts.currentAccount.id ?
|
2023-01-22 06:38:30 +01:00
|
|
|
theme.secondaryBackgroundColor : .clear)
|
2023-01-17 19:41:46 +01:00
|
|
|
}
|
2023-01-22 06:38:30 +01:00
|
|
|
|
2024-05-08 10:38:27 +02:00
|
|
|
private func accountButtonTitle(accountName: String?) -> LocalizedStringKey {
|
|
|
|
if let accountName {
|
|
|
|
"tab.profile-account-\(accountName)"
|
|
|
|
} else {
|
2024-09-10 06:53:19 +02:00
|
|
|
AppTab.profile.title
|
2024-05-08 10:38:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-17 19:41:46 +01:00
|
|
|
private var tabsView: some View {
|
|
|
|
ForEach(tabs) { tab in
|
2024-01-10 13:26:55 +01:00
|
|
|
if tab != .profile && sidebarTabs.isEnabled(tab) {
|
2024-01-04 16:42:03 +01:00
|
|
|
Button {
|
|
|
|
// ensure keyboard is always dismissed when selecting a tab
|
|
|
|
hideKeyboard()
|
|
|
|
selectedTab = tab
|
|
|
|
SoundEffectManager.shared.playSound(.tabSelection)
|
|
|
|
if tab == .notifications {
|
|
|
|
if let token = appAccounts.currentAccount.oauthToken {
|
|
|
|
userPreferences.notificationsCount[token] = 0
|
|
|
|
}
|
|
|
|
watcher.unreadNotificationsCount = 0
|
2023-02-18 15:36:18 +01:00
|
|
|
}
|
2024-01-04 16:42:03 +01:00
|
|
|
} label: {
|
|
|
|
makeIconForTab(tab: tab)
|
2023-01-17 19:41:46 +01:00
|
|
|
}
|
2024-05-08 10:38:27 +02:00
|
|
|
.help(tab.title)
|
2023-01-17 19:41:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-01-22 06:38:30 +01:00
|
|
|
|
2023-01-16 19:51:05 +01:00
|
|
|
var body: some View {
|
2023-09-18 07:01:23 +02:00
|
|
|
@Bindable var routerPath = routerPath
|
2023-01-16 19:51:05 +01:00
|
|
|
HStack(spacing: 0) {
|
2024-01-04 16:21:15 +01:00
|
|
|
if horizontalSizeClass == .regular {
|
|
|
|
ScrollView {
|
|
|
|
VStack(alignment: .center) {
|
|
|
|
if appAccounts.availableAccounts.isEmpty {
|
|
|
|
tabsView
|
|
|
|
} else {
|
|
|
|
ForEach(appAccounts.availableAccounts) { account in
|
|
|
|
makeAccountButton(account: account,
|
|
|
|
showBadge: account.id != appAccounts.currentAccount.id)
|
|
|
|
if account.id == appAccounts.currentAccount.id {
|
|
|
|
tabsView
|
|
|
|
}
|
2023-01-16 19:51:05 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-05-08 11:51:28 +02:00
|
|
|
.frame(width: userPreferences.isSidebarExpanded ? .sidebarWidthExpanded : .sidebarWidth)
|
2024-01-04 16:21:15 +01:00
|
|
|
.scrollContentBackground(.hidden)
|
|
|
|
.background(.thinMaterial)
|
2024-01-16 18:43:09 +01:00
|
|
|
.safeAreaInset(edge: .bottom, content: {
|
2024-05-08 11:51:28 +02:00
|
|
|
HStack(spacing: 16) {
|
2024-01-16 18:43:09 +01:00
|
|
|
postButton
|
|
|
|
.padding(.vertical, 24)
|
2024-05-08 11:51:28 +02:00
|
|
|
.padding(.leading, userPreferences.isSidebarExpanded ? 18 : 0)
|
|
|
|
if userPreferences.isSidebarExpanded {
|
|
|
|
Text("menu.new-post")
|
|
|
|
.font(.subheadline)
|
|
|
|
.foregroundColor(theme.labelColor)
|
|
|
|
.frame(maxWidth: .infinity, alignment: .leading)
|
|
|
|
}
|
2024-01-16 18:43:09 +01:00
|
|
|
}
|
2024-05-08 11:51:28 +02:00
|
|
|
.frame(width: userPreferences.isSidebarExpanded ? .sidebarWidthExpanded : .sidebarWidth)
|
2024-01-16 18:43:09 +01:00
|
|
|
.background(.thinMaterial)
|
|
|
|
})
|
2024-02-14 12:48:14 +01:00
|
|
|
Divider().edgesIgnoringSafeArea(.all)
|
2023-01-16 19:51:05 +01:00
|
|
|
}
|
2023-01-16 21:15:33 +01:00
|
|
|
content()
|
2023-01-16 19:51:05 +01:00
|
|
|
}
|
|
|
|
.background(.thinMaterial)
|
2024-01-04 21:28:45 +01:00
|
|
|
.edgesIgnoringSafeArea(.bottom)
|
2023-01-17 13:02:05 +01:00
|
|
|
.withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
|
2023-01-16 19:51:05 +01:00
|
|
|
}
|
|
|
|
}
|
2023-01-19 11:59:25 +01:00
|
|
|
|
|
|
|
private struct SideBarIcon: View {
|
2023-09-18 21:03:52 +02:00
|
|
|
@Environment(Theme.self) private var theme
|
2023-01-22 06:38:30 +01:00
|
|
|
|
2023-01-19 11:59:25 +01:00
|
|
|
let systemIconName: String
|
|
|
|
let isSelected: Bool
|
2023-01-22 06:38:30 +01:00
|
|
|
|
2023-01-19 11:59:25 +01:00
|
|
|
@State private var isHovered: Bool = false
|
2023-01-22 06:38:30 +01:00
|
|
|
|
2023-01-19 11:59:25 +01:00
|
|
|
var body: some View {
|
|
|
|
Image(systemName: systemIconName)
|
2023-02-10 12:53:59 +01:00
|
|
|
.font(.title2)
|
|
|
|
.fontWeight(.medium)
|
2023-01-19 11:59:25 +01:00
|
|
|
.foregroundColor(isSelected ? theme.tintColor : theme.labelColor)
|
2024-01-10 08:56:35 +01:00
|
|
|
.symbolVariant(isSelected ? .fill : .none)
|
2023-01-19 11:59:25 +01:00
|
|
|
.scaleEffect(isHovered ? 0.8 : 1.0)
|
|
|
|
.onHover { isHovered in
|
|
|
|
withAnimation(.interpolatingSpring(stiffness: 300, damping: 15)) {
|
|
|
|
self.isHovered = isHovered
|
|
|
|
}
|
|
|
|
}
|
2024-05-08 11:51:28 +02:00
|
|
|
.frame(width: 50, height: 40)
|
2023-01-19 11:59:25 +01:00
|
|
|
}
|
|
|
|
}
|
2023-06-26 11:45:45 +02:00
|
|
|
|
|
|
|
extension View {
|
2023-09-14 11:04:14 +02:00
|
|
|
@MainActor func hideKeyboard() {
|
2023-07-19 07:46:25 +02:00
|
|
|
let resign = #selector(UIResponder.resignFirstResponder)
|
|
|
|
UIApplication.shared.sendAction(resign, to: nil, from: nil, for: nil)
|
|
|
|
}
|
2023-06-26 11:45:45 +02:00
|
|
|
}
|