IceCubes/IceCubesApp/App/SideBarView.swift

171 lines
4.9 KiB
Swift
Raw Normal View History

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
import Models
2023-01-27 20:36:40 +01:00
import SwiftUI
2023-01-16 19:51:05 +01:00
struct SideBarView<Content: View>: View {
@EnvironmentObject private var appAccounts: AppAccountsManager
2023-01-16 19:51:05 +01:00
@EnvironmentObject private var currentAccount: CurrentAccount
@EnvironmentObject private var theme: Theme
@EnvironmentObject private var watcher: StreamWatcher
@EnvironmentObject private var userPreferences: UserPreferences
2023-01-22 06:38:30 +01:00
2023-01-16 19:51:05 +01:00
@Binding var selectedTab: Tab
@Binding var popToRootTab: Tab
var tabs: [Tab]
@ObservedObject var routerPath = RouterPath()
2023-01-16 21:15:33 +01:00
@ViewBuilder var content: () -> Content
2023-01-17 11:36:01 +01:00
private func badgeFor(tab: Tab) -> Int {
if tab == .notifications && selectedTab != tab,
2023-02-21 07:23:42 +01:00
let token = appAccounts.currentAccount.oauthToken
{
return watcher.unreadNotificationsCount + userPreferences.getNotificationsCount(for: token)
}
return 0
}
2023-02-21 07:23:42 +01:00
private func makeIconForTab(tab: Tab) -> some View {
ZStack(alignment: .topTrailing) {
2023-01-19 11:59:25 +01:00
SideBarIcon(systemIconName: tab.iconName,
isSelected: tab == selectedTab)
if let badge = badgeFor(tab: tab), badge > 0 {
makeBadgeView(count: badge)
}
}
.contentShape(Rectangle())
.frame(width: .sidebarWidth, height: 50)
}
2023-02-21 07:23:42 +01:00
private func makeBadgeView(count: Int) -> some View {
ZStack {
Circle()
.fill(.red)
Text(String(count))
.foregroundColor(.white)
.font(.caption2)
}
.frame(width: 20, height: 20)
.offset(x: 10, y: -10)
}
2023-01-22 06:38:30 +01:00
private var postButton: some View {
Button {
routerPath.presentedSheet = .newStatusEditor(visibility: userPreferences.postVisibility)
} label: {
Image(systemName: "square.and.pencil")
.resizable()
.aspectRatio(contentMode: .fit)
.frame(width: 20, height: 30)
}
.buttonStyle(.borderedProminent)
.keyboardShortcut("n", modifiers: .command)
}
2023-01-22 06:38:30 +01:00
private func makeAccountButton(account: AppAccount, showBadge: Bool) -> some View {
Button {
if account.id == appAccounts.currentAccount.id {
selectedTab = .profile
} else {
var transation = Transaction()
transation.disablesAnimations = true
withTransaction(transation) {
appAccounts.currentAccount = account
}
}
} label: {
ZStack(alignment: .topTrailing) {
AppAccountView(viewModel: .init(appAccount: account, isCompact: true))
if showBadge,
2023-02-21 07:23:42 +01:00
let token = account.oauthToken,
userPreferences.getNotificationsCount(for: token) > 0
{
makeBadgeView(count: userPreferences.getNotificationsCount(for: token))
}
}
}
.frame(width: .sidebarWidth, height: 50)
.padding(.vertical, 8)
.background(selectedTab == .profile && account.id == appAccounts.currentAccount.id ?
2023-01-22 06:38:30 +01:00
theme.secondaryBackgroundColor : .clear)
}
2023-01-22 06:38:30 +01:00
private var tabsView: some View {
ForEach(tabs) { tab in
Button {
if tab == selectedTab {
popToRootTab = .other
DispatchQueue.main.asyncAfter(deadline: .now() + 0.01) {
popToRootTab = tab
}
}
selectedTab = tab
if tab == .notifications {
if let token = appAccounts.currentAccount.oauthToken {
userPreferences.setNotification(count: 0, token: token)
}
watcher.unreadNotificationsCount = 0
}
} label: {
makeIconForTab(tab: tab)
}
.background(tab == selectedTab ? theme.secondaryBackgroundColor : .clear)
}
}
2023-01-22 06:38:30 +01:00
2023-01-16 19:51:05 +01:00
var body: some View {
HStack(spacing: 0) {
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
}
}
}
postButton
.padding(.top, 12)
Spacer()
2023-01-16 19:51:05 +01:00
}
}
.frame(width: .sidebarWidth)
.scrollContentBackground(.hidden)
.background(.thinMaterial)
2023-01-16 19:51:05 +01:00
Divider()
.edgesIgnoringSafeArea(.top)
2023-01-16 21:15:33 +01:00
content()
2023-01-16 19:51:05 +01:00
}
.background(.thinMaterial)
.withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
2023-01-16 19:51:05 +01:00
}
}
2023-01-19 11:59:25 +01:00
private struct SideBarIcon: View {
@EnvironmentObject private var theme: 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)
.scaleEffect(isHovered ? 0.8 : 1.0)
.onHover { isHovered in
withAnimation(.interpolatingSpring(stiffness: 300, damping: 15)) {
self.isHovered = isHovered
}
}
}
}