7
votes

I have a view which uses Geometry Reader to calculate how large the image area should be:

GeometryReader { metrics in
    ZStack{
        self.image
           .resizable()
           .frame(width:  (metrics.size.height - 10) * 0.561403509 , height: metrics.size.height - 10, alignment: .top)
           .clipped()
    }
}

But I have a function where I want to use the frame height and width calculated by the GeometryReader in order to crop the image.

I have a separate function which crops the image when a button is pressed:

 DownloadImageButton(prepareImagefunction: { self.prepareImage() })

Which then calls a prepareImage function:

func prepareImage( ) {
 // In order for image cropping to work, we need the equivalent of view.bounds in order to adjust the crop so that it correctly measures the crop area. We need metrics.width and metrics.height 
 var adjustmentScaleForDisplaySize = targetSize.width / metrics.width

Note that the GeometryReader is a child of the parent view where prepareImage is called. Therefore, ideally, the child would save the metrics values in an EnvironmentObject or Binding to the parent.

2
Can you add an example of what you're trying to accomplish, even if it doesn't work? metric can be passed into a function just like any other variable, but it's not clear what you're trying to do. - arsenius
Thanks for your comment. I have edited the answer. When the image is presented on screen I need to save the metrics.size in order to later use it in an image cropping function in the parent view. If you think about image cropping, usually you use view.bounds to determine crop area. It's the equivalent requirement. - dot3

2 Answers

10
votes

Is there a way to pass the value calculated by GeometryReader to a function?

Yes, you can do this. The var metrics is a GeometryProxy, so function signature should be like in the following example

private func resizedImage(for metrics: GeometryProxy) -> some View {
    self.image
       .resizable()
       .frame(width:  (metrics.size.height - 10) * 0.561403509 , height: metrics.size.height - 10, alignment: .top)
       .clipped()
}

so your usage

GeometryReader { metrics in
    ZStack{
       self.resizedImage(for: metrics)
    }
}

Update: for changed request

Assuming you have in child view binding as

@Binding var cropSize: CGSize

the above function can be modified as follow

private func resizedImage(for metrics: GeometryProxy) -> some View {
    let size = CGSize(width: (metrics.size.height - 10) * 0.561403509, 
                           height: metrics.size.height - 10)
    self.cropSize = size
    return self.image
       .resizable()
       .frame(width: size.width, height: size.height, alignment: .top)
       .clipped()
}
2
votes

Since I actually ONLY needed the crop size, and didn't necessarily need an image returned, I did the following which also helped to avoid throwing an error which can occur in the accepted answer:

[SwiftUI] Modifying state during view update, this will cause undefined behavior.

I added an .onAppear function call which helps to avoid the above error, and then properly passes the crop dimension to the parent:

GeometryReader { metrics in
  ZStack{
   self.image
   .resizable()
   .frame(width:  (metrics.size.height - 10) * 0.561403509 , height: metrics.size.height - 10, alignment: .top)
   .clipped()
  }.onAppear{
    self.saveCropSize(for: metrics)
  }....

private func saveCropSize(for metrics: GeometryProxy){
    let size = CGSize(width: (metrics.size.height - 10) * 0.561403509, 
                           height: metrics.size.height - 10)
    self.cropSize = size
}