| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108 |
- import SwiftUI
- struct TagCloudView: View {
- var tags: [String]
- @State private var totalHeight
- // = CGFloat.zero // << variant for ScrollView/List
- = CGFloat.infinity // << variant for VStack
- var body: some View {
- VStack {
- GeometryReader { geometry in
- self.generateContent(in: geometry)
- }
- }
- // .frame(height: totalHeight)// << variant for ScrollView/List
- .frame(maxHeight: totalHeight) // << variant for VStack
- }
- private func generateContent(in g: GeometryProxy) -> some View {
- var width = CGFloat.zero
- var height = CGFloat.zero
- return ZStack(alignment: .topLeading) {
- ForEach(self.tags, id: \.self) { tag in
- self.item(for: tag)
- .padding([.horizontal, .vertical], 2)
- .alignmentGuide(.leading, computeValue: { d in
- if abs(width - d.width) > g.size.width
- {
- width = 0
- height -= d.height
- }
- let result = width
- if tag == self.tags.last! {
- width = 0 // last item
- } else {
- width -= d.width
- }
- return result
- })
- .alignmentGuide(.top, computeValue: { _ in
- let result = height
- if tag == self.tags.last! {
- height = 0 // last item
- }
- return result
- })
- }
- }.background(viewHeightReader($totalHeight))
- }
- private func item(for textTag: String) -> some View {
- var colorOfTag: Color {
- if textTag.contains("Not Floating") {
- return .loopYellow } else {
- switch textTag {
- case "Floating Carbs:":
- return .loopPink
- case "autoISF":
- return .zt
- case "SMB Delivery Ratio:":
- return .uam
- case "Parabolic Fit":
- return .loopRed
- case "Autosens":
- return .loopGreen
- case "Standard":
- return .darkerBlue
- case "TDD":
- return .loopYellow
- default:
- return .insulin
- }
- }
- }
- return ZStack { Text(textTag)
- .padding(.vertical, 2)
- .padding(.horizontal, 4)
- .font(.caption)
- .background(colorOfTag.opacity(0.8))
- .foregroundColor(Color.white)
- .cornerRadius(5) }
- }
- private func viewHeightReader(_ binding: Binding<CGFloat>) -> some View {
- GeometryReader { geometry -> Color in
- let rect = geometry.frame(in: .local)
- DispatchQueue.main.async {
- binding.wrappedValue = rect.size.height
- }
- return .clear
- }
- }
- }
- struct TestTagCloudView: View {
- var body: some View {
- VStack {
- Text("Header").font(.largeTitle)
- TagCloudView(tags: ["Ninetendo", "XBox", "PlayStation", "PlayStation 2", "PlayStation 3", "PlayStation 4"])
- Text("Some other text")
- Divider()
- Text("Some other cloud")
- TagCloudView(tags: ["Apple", "Google", "Amazon", "Microsoft", "Oracle", "Facebook"])
- }
- }
- }
|