SwiftUI-根据滚动视图的位置更改视图的不透明度

时间:2020-01-01 21:08:06

标签: scrollview swiftui

这更多是“是否可能”的问题。

我目前有一个浮动的自定义导航栏,它位于其自己的父容器中,该容器具有白色背景,不透明度为0%。在scrollView滚动到scrollview中的英雄图像之后,我想将该父容器的不透明度从0%更改为50%。

var body: some View {
    ZStack {
        ScrollView(.vertical, showsIndicators: false) {
            GeometryReader { geometry in
                ZStack{
                    if geometry.frame(in: .global).minY <= 0 {
                        Image(self.venueImage)
                        .resizable()
                        .aspectRatio(contentMode: .fill)
                        .frame(width: geometry.size.width, height: geometry.size.height)
                        .offset(y: geometry.frame(in: .global).minY/9)
                        .clipped()
                    } else {
                        Image(self.venueImage)
                        .resizable()
                        .aspectRatio(contentMode: .fill)
                        .frame(width: geometry.size.width, height: geometry.size.height + geometry.frame(in: .global).minY)
                        .clipped()
                        .offset(y: -geometry.frame(in: .global).minY)
                    }
                }
            }.frame(height: 300)
            VStack {
                HStack {
                    Text(self.title)
                        .font(.title)
                        .fontWeight(.bold)
                    Spacer()
                }.padding(.horizontal, 24)
                HStack {
                    Text(self.venueArea)
                        .foregroundColor(Color("bodyText"))
                    Spacer()
                }.padding(.horizontal, 24)

                VStack {
                    ForEach(self.venueProd) { gc in

                        VStack {
                            HStack {
                                    Text(gc.title)
                                        .font(.system(size: 17))
                                        .fontWeight(.semibold)
                                    Spacer()
                            }
                            HStack {
                                Text(gc.productDescription)
                                    .foregroundColor(Color("bodyText"))
                                    .font(.system(size: 15))
                                    .padding(.top, 5)
                                Spacer()
                            }
                            HStack {
                                Text("$\(gc.productPrice, specifier: "%.2f")")
                                Spacer()
                            }.padding(.top, 8)
                            Divider()
                        }.padding(.bottom, 8)

                    }
                    Spacer()
                }.padding(.horizontal, 24)
                    .padding(.top, 24)
                    .padding(.bottom, 60)
                Spacer()
            }
        }.edgesIgnoringSafeArea(.top)

       // AREA I'M TRYING TO MODIFY OPACITY ON SCROLL
        ZStack {

            GeometryReader { geometry in
            ZStack {
                    VStack {
                       Text("")
                           .frame(width: geometry.size.width, height: 120)
                        .background(Color.white)
                        .opacity(0.3                     )
                    Spacer()
                    } 
            }.edgesIgnoringSafeArea(.top) // Ends ZStack
            } // Ends Geometry Reader

            VStack {
                GeometryReader { gr in
                    HStack {
                        Button(action: {self.presentationMode.wrappedValue.dismiss()}) {
                            Image(systemName: "chevron.left")
                                .foregroundColor(.black)
                                .padding(.leading, 16)
                            HStack {
                                Text("Venues · Venue Details")
                                    .font(.system(size: 15))
                                    .fontWeight(.bold)
                                    .foregroundColor(Color.black)

                                    .padding()
                                Spacer()
                            }
                        }.frame(width: gr.size.width * 0.92, height: 48)
                            .background(Color.white)
                            .cornerRadius(8)

                    }.padding(.leading, 16)
                    Spacer()
                }
            }
            .padding(.top, 50)
            .edgesIgnoringSafeArea(.top)
        } // Ends Floating Menu ZStack

    } // Second ZStack
}

}

1 个答案:

答案 0 :(得分:1)

这是一个想法演示。

enter image description here

该代码仅显示了该方法,并且由于简单性,它可​​以轻松集成/传输。所有背景颜色都只是为了获得更好的可见性,并且计算可能会更复杂并包含动画,但是这个想法保持不变-可以使用GeometryProxy并根据更改来读取全局坐标中的图像通过相应的@State使其他视图不透明。

struct TestScrollDependentOpacity: View {

    @State private var barOpacity: Double = 0

    var body: some View {
        ZStack(alignment: .top) {
            mainContent
            floatingBar
        }
    }

    private var mainContent: some View {
        ScrollView {
            GeometryReader { g -> AnyView in
                let rect = g.frame(in: .global)
                DispatchQueue.main.async {
                    self.barOpacity = rect.maxY < 0 ? 0.5 : 0.0
                }
                return AnyView(Image(systemName: "sun.max")
                    .resizable()
                    .aspectRatio(contentMode: .fit)
                    .position(x: g.size.width / 2.0, y: g.size.height / 2.0))
            }
            .frame(height: 100).background(Color.yellow)

            VStack(alignment: .leading) {
                ForEach (0..<40) { i in
                    HStack {
                        Text("Text \(i)").padding()
                        Spacer()
                    }
                }
            }.background(Color.green)
        }
    }

    private var floatingBar: some View {
        HStack {
            Button("Left") { }
            Spacer()
            Button("Right") { }
        }
        .padding()
        .background(Color.white.opacity(barOpacity))
        .edgesIgnoringSafeArea(.top)
    }
}

struct TestScrollDependentOpacity_Previews: PreviewProvider {
    static var previews: some View {
        TestScrollDependentOpacity()
    }
}