444 lines
16 KiB
Swift
444 lines
16 KiB
Swift
import SwiftUI
|
|
import Bedrock
|
|
|
|
/// A sheet that displays header layout options as a live preview carousel.
|
|
struct HeaderLayoutPickerView: View {
|
|
@Environment(\.dismiss) private var dismiss
|
|
|
|
@Binding var selectedLayout: CardHeaderLayout
|
|
let photoData: Data?
|
|
let coverPhotoData: Data?
|
|
let logoData: Data?
|
|
let avatarSystemName: String
|
|
let theme: CardTheme
|
|
let displayName: String
|
|
let role: String
|
|
let company: String
|
|
|
|
@State private var currentLayout: CardHeaderLayout
|
|
|
|
init(
|
|
selectedLayout: Binding<CardHeaderLayout>,
|
|
photoData: Data?,
|
|
coverPhotoData: Data?,
|
|
logoData: Data?,
|
|
avatarSystemName: String,
|
|
theme: CardTheme,
|
|
displayName: String,
|
|
role: String,
|
|
company: String
|
|
) {
|
|
self._selectedLayout = selectedLayout
|
|
self.photoData = photoData
|
|
self.coverPhotoData = coverPhotoData
|
|
self.logoData = logoData
|
|
self.avatarSystemName = avatarSystemName
|
|
self.theme = theme
|
|
self.displayName = displayName
|
|
self.role = role
|
|
self.company = company
|
|
self._currentLayout = State(initialValue: selectedLayout.wrappedValue)
|
|
}
|
|
|
|
private var suggestedLayout: CardHeaderLayout {
|
|
CardHeaderLayout.suggested(
|
|
hasProfile: photoData != nil,
|
|
hasCover: coverPhotoData != nil,
|
|
hasLogo: logoData != nil
|
|
)
|
|
}
|
|
|
|
var body: some View {
|
|
NavigationStack {
|
|
VStack(spacing: 0) {
|
|
ScrollViewReader { proxy in
|
|
ScrollView(.horizontal, showsIndicators: false) {
|
|
HStack(spacing: Design.Spacing.large) {
|
|
ForEach(CardHeaderLayout.allCases) { layout in
|
|
LayoutPreviewCard(
|
|
layout: layout,
|
|
isSelected: currentLayout == layout,
|
|
isSuggested: layout == suggestedLayout,
|
|
photoData: photoData,
|
|
coverPhotoData: coverPhotoData,
|
|
logoData: logoData,
|
|
avatarSystemName: avatarSystemName,
|
|
theme: theme
|
|
) {
|
|
withAnimation(.snappy(duration: Design.Animation.quick)) {
|
|
currentLayout = layout
|
|
}
|
|
}
|
|
.id(layout)
|
|
}
|
|
}
|
|
.padding(.horizontal, Design.Spacing.xLarge)
|
|
.padding(.vertical, Design.Spacing.large)
|
|
}
|
|
.onAppear {
|
|
proxy.scrollTo(currentLayout, anchor: .center)
|
|
}
|
|
}
|
|
.scrollClipDisabled()
|
|
|
|
Spacer()
|
|
|
|
Button {
|
|
selectedLayout = currentLayout
|
|
dismiss()
|
|
} label: {
|
|
Text("Confirm layout")
|
|
.typography(.heading)
|
|
.foregroundStyle(.white)
|
|
.frame(maxWidth: .infinity)
|
|
.padding(.vertical, Design.Spacing.large)
|
|
.background(.black)
|
|
.clipShape(.rect(cornerRadius: Design.CornerRadius.medium))
|
|
}
|
|
.padding(.horizontal, Design.Spacing.xLarge)
|
|
.padding(.bottom, Design.Spacing.xLarge)
|
|
}
|
|
.background(Color.AppBackground.base)
|
|
.navigationTitle("Choose a layout")
|
|
.navigationBarTitleDisplayMode(.inline)
|
|
.toolbar {
|
|
ToolbarItem(placement: .cancellationAction) {
|
|
Button {
|
|
dismiss()
|
|
} label: {
|
|
Image(systemName: "xmark")
|
|
.typography(.body)
|
|
.foregroundStyle(Color.Text.primary)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: - Layout Preview Card
|
|
|
|
private struct LayoutPreviewCard: View {
|
|
let layout: CardHeaderLayout
|
|
let isSelected: Bool
|
|
let isSuggested: Bool
|
|
let photoData: Data?
|
|
let coverPhotoData: Data?
|
|
let logoData: Data?
|
|
let avatarSystemName: String
|
|
let theme: CardTheme
|
|
let onSelect: () -> Void
|
|
|
|
private let cardWidth: CGFloat = 200
|
|
private let cardHeight: CGFloat = 280
|
|
private let bannerHeight: CGFloat = 100
|
|
private let avatarSize: CGFloat = 56
|
|
private let logoRectWidth: CGFloat = 84 // 56 * 1.5 aspect ratio
|
|
|
|
private var needsMoreImages: Bool {
|
|
!layout.hasAllRequiredImages(
|
|
hasProfile: photoData != nil,
|
|
hasCover: coverPhotoData != nil,
|
|
hasLogo: logoData != nil
|
|
)
|
|
}
|
|
|
|
private var hasOverlappingContent: Bool {
|
|
layout.hasOverlappingContent
|
|
}
|
|
|
|
var body: some View {
|
|
Button(action: onSelect) {
|
|
VStack(spacing: 0) {
|
|
ZStack(alignment: .top) {
|
|
VStack(spacing: 0) {
|
|
bannerContent
|
|
.frame(height: bannerHeight)
|
|
.clipped()
|
|
|
|
contentArea
|
|
.offset(y: hasOverlappingContent ? -avatarSize / 2 : 0)
|
|
.padding(.bottom, hasOverlappingContent ? -avatarSize / 2 : 0)
|
|
}
|
|
.frame(width: cardWidth, height: cardHeight)
|
|
.background(Color.AppBackground.elevated)
|
|
.clipShape(.rect(cornerRadius: Design.CornerRadius.large))
|
|
.overlay(
|
|
RoundedRectangle(cornerRadius: Design.CornerRadius.large)
|
|
.stroke(isSelected ? theme.primaryColor : .clear, lineWidth: Design.LineWidth.thick)
|
|
)
|
|
.shadow(color: Color.Text.secondary.opacity(Design.Opacity.subtle), radius: Design.Shadow.radiusMedium, y: Design.Shadow.offsetMedium)
|
|
|
|
badgeOverlay
|
|
.offset(y: -Design.Spacing.small)
|
|
}
|
|
}
|
|
}
|
|
.buttonStyle(.plain)
|
|
.accessibilityLabel(layout.displayName)
|
|
.accessibilityValue(isSelected ? String(localized: "Selected") : "")
|
|
.accessibilityHint(layout.description)
|
|
.accessibilityAddTraits(isSelected ? [.isSelected] : [])
|
|
}
|
|
|
|
// MARK: - Banner Content
|
|
|
|
@ViewBuilder
|
|
private var bannerContent: some View {
|
|
switch layout.bannerContent {
|
|
case .profile:
|
|
profileBannerPreview
|
|
case .logo:
|
|
logoBannerPreview
|
|
case .cover:
|
|
coverBannerPreview
|
|
}
|
|
}
|
|
|
|
private var profileBannerPreview: some View {
|
|
ZStack {
|
|
if let photoData, let uiImage = UIImage(data: photoData) {
|
|
Image(uiImage: uiImage)
|
|
.resizable()
|
|
.scaledToFill()
|
|
} else {
|
|
LinearGradient(colors: [theme.primaryColor, theme.secondaryColor], startPoint: .topLeading, endPoint: .bottomTrailing)
|
|
|
|
VStack(spacing: Design.Spacing.xxSmall) {
|
|
Image(systemName: "person.fill")
|
|
.typography(.title3)
|
|
Text("Profile")
|
|
.typography(.caption)
|
|
.bold()
|
|
}
|
|
.foregroundStyle(theme.textColor.opacity(Design.Opacity.medium))
|
|
}
|
|
}
|
|
}
|
|
|
|
private var logoBannerPreview: some View {
|
|
ZStack {
|
|
LinearGradient(colors: [theme.primaryColor, theme.secondaryColor], startPoint: .topLeading, endPoint: .bottomTrailing)
|
|
|
|
if let logoData, let uiImage = UIImage(data: logoData) {
|
|
Image(uiImage: uiImage)
|
|
.resizable()
|
|
.scaledToFill()
|
|
} else {
|
|
VStack(spacing: Design.Spacing.xxSmall) {
|
|
Image(systemName: "building.2.fill")
|
|
.typography(.title3)
|
|
Text("Logo")
|
|
.typography(.caption)
|
|
.bold()
|
|
}
|
|
.foregroundStyle(theme.textColor.opacity(Design.Opacity.medium))
|
|
}
|
|
}
|
|
}
|
|
|
|
private var coverBannerPreview: some View {
|
|
Group {
|
|
if let coverData = coverPhotoData, let uiImage = UIImage(data: coverData) {
|
|
Image(uiImage: uiImage)
|
|
.resizable()
|
|
.scaledToFill()
|
|
} else {
|
|
ZStack {
|
|
LinearGradient(colors: [theme.primaryColor.opacity(Design.Opacity.light), theme.secondaryColor.opacity(Design.Opacity.light)], startPoint: .topLeading, endPoint: .bottomTrailing)
|
|
|
|
VStack(spacing: Design.Spacing.xxSmall) {
|
|
Image(systemName: "photo.fill")
|
|
.typography(.title3)
|
|
Text("Cover")
|
|
.typography(.caption)
|
|
.bold()
|
|
}
|
|
.foregroundStyle(Color.Text.tertiary)
|
|
}
|
|
}
|
|
}
|
|
.clipped()
|
|
}
|
|
|
|
// MARK: - Content Area
|
|
|
|
private var contentArea: some View {
|
|
VStack(alignment: .leading, spacing: Design.Spacing.small) {
|
|
contentOverlay
|
|
placeholderTextLines
|
|
Spacer()
|
|
}
|
|
.padding(.horizontal, Design.Spacing.medium)
|
|
.padding(.top, Design.Spacing.medium)
|
|
.padding(.bottom, Design.Spacing.small)
|
|
}
|
|
|
|
@ViewBuilder
|
|
private var contentOverlay: some View {
|
|
switch layout.contentOverlay {
|
|
case .none:
|
|
EmptyView()
|
|
case .avatar:
|
|
HStack {
|
|
profileAvatar
|
|
Spacer()
|
|
}
|
|
case .logoRectangle:
|
|
HStack {
|
|
logoRectangle
|
|
Spacer()
|
|
}
|
|
case .avatarAndLogo:
|
|
HStack {
|
|
profileAvatar
|
|
Spacer()
|
|
logoRectangle
|
|
}
|
|
}
|
|
}
|
|
|
|
private var placeholderTextLines: some View {
|
|
VStack(alignment: .leading, spacing: Design.Spacing.xSmall) {
|
|
RoundedRectangle(cornerRadius: Design.CornerRadius.small)
|
|
.fill(Color.Text.tertiary.opacity(Design.Opacity.hint))
|
|
.frame(height: Design.Spacing.medium)
|
|
.frame(maxWidth: .infinity, alignment: .leading)
|
|
.padding(.trailing, Design.Spacing.xLarge)
|
|
|
|
RoundedRectangle(cornerRadius: Design.CornerRadius.small)
|
|
.fill(Color.Text.tertiary.opacity(Design.Opacity.subtle))
|
|
.frame(height: Design.Spacing.small)
|
|
.frame(maxWidth: .infinity, alignment: .leading)
|
|
.padding(.trailing, Design.Spacing.xxLarge)
|
|
|
|
RoundedRectangle(cornerRadius: Design.CornerRadius.small)
|
|
.fill(Color.Text.tertiary.opacity(Design.Opacity.subtle))
|
|
.frame(height: Design.Spacing.small)
|
|
.frame(maxWidth: .infinity, alignment: .leading)
|
|
.padding(.trailing, Design.Spacing.xLarge)
|
|
}
|
|
}
|
|
|
|
// MARK: - Overlay Components
|
|
|
|
private var profileAvatar: some View {
|
|
Group {
|
|
if let photoData, let uiImage = UIImage(data: photoData) {
|
|
Image(uiImage: uiImage)
|
|
.resizable()
|
|
.scaledToFill()
|
|
} else {
|
|
Image(systemName: avatarSystemName)
|
|
.font(.system(size: avatarSize / 2.5))
|
|
.foregroundStyle(theme.textColor)
|
|
.frame(maxWidth: .infinity, maxHeight: .infinity)
|
|
.background(theme.accentColor)
|
|
}
|
|
}
|
|
.frame(width: avatarSize, height: avatarSize)
|
|
.clipShape(.circle)
|
|
.overlay(Circle().stroke(Color.AppBackground.elevated, lineWidth: Design.LineWidth.medium))
|
|
}
|
|
|
|
private var logoBadge: some View {
|
|
Group {
|
|
if let logoData, let uiImage = UIImage(data: logoData) {
|
|
Image(uiImage: uiImage)
|
|
.resizable()
|
|
.scaledToFill()
|
|
.clipped()
|
|
} else {
|
|
VStack(spacing: Design.Spacing.xxSmall) {
|
|
Image(systemName: "building.2")
|
|
.typography(.caption)
|
|
Text("Logo")
|
|
.typography(.caption2)
|
|
}
|
|
.foregroundStyle(theme.textColor)
|
|
.frame(maxWidth: .infinity, maxHeight: .infinity)
|
|
}
|
|
}
|
|
.frame(width: avatarSize, height: avatarSize)
|
|
.background(theme.accentColor)
|
|
.clipShape(.rect(cornerRadius: Design.CornerRadius.medium))
|
|
.overlay(RoundedRectangle(cornerRadius: Design.CornerRadius.medium).stroke(Color.AppBackground.elevated, lineWidth: Design.LineWidth.medium))
|
|
}
|
|
|
|
private var logoRectangle: some View {
|
|
Group {
|
|
if let logoData, let uiImage = UIImage(data: logoData) {
|
|
Image(uiImage: uiImage)
|
|
.resizable()
|
|
.scaledToFill()
|
|
.clipped()
|
|
} else {
|
|
VStack(spacing: Design.Spacing.xxSmall) {
|
|
Image(systemName: "building.2")
|
|
.typography(.caption)
|
|
Text("Logo")
|
|
.typography(.caption2)
|
|
}
|
|
.foregroundStyle(theme.textColor)
|
|
.frame(maxWidth: .infinity, maxHeight: .infinity)
|
|
}
|
|
}
|
|
.frame(width: logoRectWidth, height: avatarSize)
|
|
.background(theme.accentColor)
|
|
.clipShape(.rect(cornerRadius: Design.CornerRadius.medium))
|
|
.overlay(RoundedRectangle(cornerRadius: Design.CornerRadius.medium).stroke(Color.AppBackground.elevated, lineWidth: Design.LineWidth.medium))
|
|
}
|
|
|
|
// MARK: - Badges
|
|
|
|
@ViewBuilder
|
|
private var badgeOverlay: some View {
|
|
if needsMoreImages {
|
|
LayoutBadge(text: "More images required", iconName: "lock.fill", backgroundColor: Color.Text.primary.opacity(Design.Opacity.strong))
|
|
} else if isSuggested && !isSelected {
|
|
LayoutBadge(text: "Suggested", iconName: "star.fill", backgroundColor: Color.Badge.star)
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: - Layout Badge
|
|
|
|
private struct LayoutBadge: View {
|
|
let text: String
|
|
let iconName: String
|
|
let backgroundColor: Color
|
|
|
|
var body: some View {
|
|
HStack(spacing: Design.Spacing.xSmall) {
|
|
Image(systemName: iconName)
|
|
.typography(.caption2)
|
|
Text(text)
|
|
.typography(.caption2)
|
|
.bold()
|
|
}
|
|
.padding(.horizontal, Design.Spacing.small)
|
|
.padding(.vertical, Design.Spacing.xSmall)
|
|
.background(backgroundColor)
|
|
.foregroundStyle(.white)
|
|
.clipShape(.capsule)
|
|
}
|
|
}
|
|
|
|
// MARK: - Preview
|
|
|
|
#Preview {
|
|
@Previewable @State var selectedLayout: CardHeaderLayout = .profileBanner
|
|
|
|
HeaderLayoutPickerView(
|
|
selectedLayout: $selectedLayout,
|
|
photoData: nil,
|
|
coverPhotoData: nil,
|
|
logoData: nil,
|
|
avatarSystemName: "person.crop.circle",
|
|
theme: .coral,
|
|
displayName: "John Doe",
|
|
role: "Developer",
|
|
company: "Acme Inc"
|
|
)
|
|
}
|