This posting is based on Kavsoft video:
https://youtu.be/HKnJ_95zisI?si=ejS_RflEYLm9YMGG
(1) VisonOSStyleView.swift
import SwiftUI
struct VisionOSStyleView<Content: View>: View {
var cornerRadius: CGFloat = 30
@ViewBuilder var content: Content
/// View Properties
@State private var viewSize: CGSize = .zero
var body: some View {
content
.foregroundStyle(.black)
.clipShape(.rect(cornerRadius: cornerRadius, style: .continuous))
.contentShape(.rect(cornerRadius: cornerRadius, style: .continuous))
.background {
BackgroundView()
}
.compositingGroup()
.shadow(color: .black.opacity(0.15), radius: 8, x: 8, y: 8)
.shadow(color: .black.opacity(0.1), radius: 5, x: -5, y: -5)
.onGeometryChange(for: CGSize.self) {
$0.size
} action: { newValue in
viewSize = newValue
}
}
/// VisionOS Style Background
@ViewBuilder
private func BackgroundView() -> some View {
ZStack {
RoundedRectangle(cornerRadius: cornerRadius, style: .continuous)
.stroke(
.thinMaterial,
style: .init(
lineWidth: 3,
lineCap: .round,
lineJoin: .round
)
)
RoundedRectangle(cornerRadius: cornerRadius, style: .continuous)
.fill(
.ultraThinMaterial.shadow(
.inner(
color: .black.opacity(0.2),
radius: 10
)
)
)
}
.environment(\.colorScheme, .light)
}
}
(2) ContentView.swift
import SwiftUI
extension ColorScheme {
var currentColor: Color {
switch self {
case .light:
return .white
case .dark:
return .black
default:
return .clear
}
}
}
struct ContentView: View {
/// View Properties
@State private var isExpanded: Bool = false
@State private var menuPosition: CGRect = .zero
@Environment(\.colorScheme) private var colorScheme
var body: some View {
VStack {
HeaderView()
DummyContentView()
Spacer(minLength: 0)
}
.overlay(alignment: .topLeading) {
ZStack(alignment: .topLeading) {
Rectangle()
.foregroundStyle(.clear)
.contentShape(.rect)
.onTapGesture {
withAnimation(.snappy(duration: 0.3, extraBounce: 0)) {
isExpanded = false
}
}
.allowsHitTesting(isExpanded)
ZStack {
if isExpanded {
VisionOSStyleView {
MenuBarControls()
.frame(width: 220, height: 270)
}
.transition(.blurReplace)
}
}
.offset(
x: menuPosition.minX - 220 + menuPosition.width,
y: menuPosition.maxY + 10
)
}
.ignoresSafeArea()
}
}
/// Header View
@ViewBuilder
func HeaderView() -> some View {
HStack {
Text("Notes")
.font(.largeTitle.bold())
Spacer(minLength: 0)
/// Menu Button
Button(action: {
withAnimation(.smooth) {
isExpanded = true
}
}, label: {
Image(systemName: "ellipsis")
.font(.title3)
.foregroundStyle(
isExpanded
? colorScheme.currentColor
: Color.primary
)
.frame(width: 45, height: 45)
.background {
ZStack {
Rectangle()
.fill(.ultraThinMaterial)
Rectangle()
.fill(Color.primary.opacity(isExpanded ? 1 : 0.03))
}
.clipShape(.circle)
}
})
.onGeometryChange(for: CGRect.self) {
$0.frame(in: .global)
} action: { newValue in
menuPosition = newValue
}
}
.padding(15)
}
@ViewBuilder
func DummyContentView() -> some View {
VStack(alignment: .leading, spacing: 5) {
GeometryReader { geometry in
Image(.castle)
.resizable()
.aspectRatio(contentMode: .fit)
.frame(width: geometry.size.width, height: geometry.size.height)
.clipped()
}
.frame(height: 320)
VStack(alignment: .leading, spacing: 12) {
Text("Neuschwanstein Castle")
.font(.title.bold())
}
.padding(15)
}
}
}
struct MenuBarControls: View {
var body: some View {
VStack(spacing: 12) {
HStack(spacing: 15) {
ForEach(["document.viewfinder", "pin.fill", "lock.fill"], id: \.self) { image in
Button {
} label: {
Image(systemName: image)
.font(.title3)
.frame(maxWidth: .infinity)
.padding(.vertical, 12)
}
}
}
/// Custom Divider
Rectangle()
.fill(.black.opacity(0.1))
.frame(height: 1)
/// Custom Buttons
CustomButton(title: "Find in Note", image: "magnifyingglass")
CustomButton(title: "Move Note", image: "folder")
CustomButton(title: "Lines & Grids", image: "squareshape.split.3x3")
CustomButton(title: "Delete", image: "trash")
}
.padding(20)
}
@ViewBuilder
private func CustomButton(
title: String,
image: String,
action: @escaping () -> () = {}
) -> some View {
Button(action: action) {
HStack {
Text(title)
.font(.system(size: 13))
Spacer(minLength: 0)
Image(systemName: image)
.frame(width: 20)
}
.frame(maxHeight: .infinity)
}
}
}
#Preview {
ContentView()
}
you can download source code here:
'iOS > SwiftUI' 카테고리의 다른 글
[iOS / SwiftUI] Staggered Animation View (0) | 2025.04.01 |
---|---|
[iOS / SwiftUI] DownsizedImage (0) | 2025.03.03 |
[iOS / SwiftUI] SwiftUI Notification Deep Linking (0) | 2025.02.16 |
[iOS / SwiftUI] SwiftUI Custom Alerts (0) | 2025.02.08 |
[iOS / SwiftUI] Use Deep Link in SwiftUI (0) | 2022.01.11 |