Created
July 9, 2020 07:10
-
-
Save agelessman/c8ef0a21b59ddb2199ad47af909e1b78 to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// | |
// ContentView.swift | |
// PreferenceKeyDemo | |
// | |
// Created by MC on 2020/7/9. | |
// | |
import SwiftUI | |
struct ContentView: View { | |
var body: some View { | |
Example3() | |
} | |
} | |
struct Example4: View { | |
@Environment(\.myEnvironmentTestValue) var value: Double | |
var body: some View { | |
Text("\(value)") | |
} | |
} | |
struct Example3: View { | |
@State private var activeNumber: Int = 1 | |
@State private var rects: [CGRect] = Array<CGRect>(repeating: CGRect(), count: 9) | |
var body: some View { | |
ZStack(alignment: .topLeading) { | |
Circle() | |
.stroke(Color.green, lineWidth: 5) | |
.frame(width: rects[activeNumber - 1].width, height: rects[activeNumber - 1].height) | |
.offset(x: rects[activeNumber - 1].minX, y: rects[activeNumber - 1].minY) | |
.animation(.easeInOut) | |
VStack { | |
Spacer() | |
HStack { | |
NumberView(activeNumber: $activeNumber, number: 1) | |
NumberView(activeNumber: $activeNumber, number: 2) | |
NumberView(activeNumber: $activeNumber, number: 3) | |
} | |
HStack { | |
NumberView(activeNumber: $activeNumber, number: 4) | |
NumberView(activeNumber: $activeNumber, number: 5) | |
NumberView(activeNumber: $activeNumber, number: 6) | |
} | |
HStack { | |
NumberView(activeNumber: $activeNumber, number: 7) | |
NumberView(activeNumber: $activeNumber, number: 8) | |
NumberView(activeNumber: $activeNumber, number: 9) | |
} | |
Spacer() | |
} | |
} | |
.onPreferenceChange(NumberPreferenceKey.self) { preferences in | |
for pre in preferences { | |
self.rects[pre.viewIdx] = pre.rect | |
} | |
} | |
.coordinateSpace(name: "ZStackSpace") | |
} | |
struct NumberView: View { | |
@Binding var activeNumber: Int | |
let number: Int | |
var body: some View { | |
Text("\(number)") | |
.font(.system(size: 40, weight: .heavy, design: .rounded)) | |
.padding(20) | |
.background(NumberPreferenceViewSetter(idx: number - 1)) | |
.onTapGesture { | |
self.activeNumber = number | |
} | |
} | |
} | |
struct NumberPreferenceViewSetter: View { | |
let idx: Int | |
var body: some View { | |
GeometryReader { proxy in | |
Circle() | |
.stroke(Color.clear, lineWidth: 5) | |
.preference(key: NumberPreferenceKey.self, value: [NumberPreferenceValue(viewIdx: idx, rect: proxy.frame(in: .named("ZStackSpace")))]) | |
} | |
} | |
} | |
struct NumberPreferenceValue: Equatable { | |
let viewIdx: Int | |
let rect: CGRect | |
} | |
struct NumberPreferenceKey: PreferenceKey { | |
typealias Value = [NumberPreferenceValue] | |
static var defaultValue: [NumberPreferenceValue] = [] | |
static func reduce(value: inout [NumberPreferenceValue], nextValue: () -> [NumberPreferenceValue]) { | |
value.append(contentsOf: nextValue()) | |
} | |
} | |
} | |
struct Example2: View { | |
@State private var activeNumber: Int = 1 | |
var body: some View { | |
VStack { | |
Spacer() | |
HStack { | |
NumberView(activeNumber: $activeNumber, number: 1) | |
NumberView(activeNumber: $activeNumber, number: 2) | |
NumberView(activeNumber: $activeNumber, number: 3) | |
} | |
HStack { | |
NumberView(activeNumber: $activeNumber, number: 4) | |
NumberView(activeNumber: $activeNumber, number: 5) | |
NumberView(activeNumber: $activeNumber, number: 6) | |
} | |
HStack { | |
NumberView(activeNumber: $activeNumber, number: 7) | |
NumberView(activeNumber: $activeNumber, number: 8) | |
NumberView(activeNumber: $activeNumber, number: 9) | |
} | |
Spacer() | |
} | |
} | |
struct NumberView: View { | |
@Binding var activeNumber: Int | |
let number: Int | |
var body: some View { | |
Text("\(number)") | |
.font(.system(size: 40, weight: .heavy, design: .rounded)) | |
.padding(20) | |
.background(NumberBorder(show: activeNumber == number)) | |
.onTapGesture { | |
self.activeNumber = number | |
} | |
} | |
} | |
struct NumberBorder: View { | |
let show: Bool | |
var body: some View { | |
Circle() | |
.stroke(show ? Color.green : Color.clear, lineWidth: 5) | |
.animation(.easeInOut) | |
} | |
} | |
} | |
struct Example1: View { | |
var body: some View { | |
NavigationView { | |
Text("Hello, world!") | |
.padding() | |
.navigationBarTitle("😯🔟⌚️🦶", displayMode: .inline) | |
} | |
} | |
} | |
struct ContentView_Previews: PreviewProvider { | |
static var previews: some View { | |
ContentView() | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment