-
-
Save erica/fb5005f625207e108836175ff201d8f2 to your computer and use it in GitHub Desktop.
import PlaygroundSupport | |
import SwiftUI | |
extension UIView { | |
var renderedImage: UIImage { | |
let image = UIGraphicsImageRenderer(size: self.bounds.size).image { context in | |
UIColor.lightGray.set(); UIRectFill(bounds) | |
context.cgContext.setAlpha(0.75) | |
self.layer.render(in: context.cgContext) | |
} | |
return image | |
} | |
} | |
extension View { | |
var renderedImage: UIImage { | |
let window = UIWindow(frame: CGRect(origin: .zero, size: CGSize(width: 320, height: 160))) | |
let hosting = UIHostingController(rootView: self) | |
hosting.view.frame = window.frame | |
window.rootViewController = hosting | |
window.makeKey() | |
return hosting.view.renderedImage | |
} | |
} | |
Text("Hello").renderedImage | |
Slider(value: .constant(0.5)).renderedImage | |
let img = ([Color.red, .orange, .yellow, .green, .blue, .purple] | |
.reduce(AnyView(Text("👭").font(.largeTitle) | |
.rotationEffect(Angle(radians: .pi)))) { | |
AnyView($0.padding() | |
.background($1) | |
.rotationEffect(Angle(radians: .pi / 6))) | |
}).renderedImage | |
img |
but I see you already know how to do that: https://ericasadun.com/2019/06/06/good-things-swiftui-on-mojave-in-ios-playgrounds/
But on the other hand I got to execute some swiftUI for the first time. Thanks!
I'm trying to render a SwiftUI View into a UIImage. This code works perfectly in Swift Playground. But in a simulator or real device, it generates a blank image. Any idea how could make it work please?
This is the code I'm using
let size = // Calcuate size //
let hosting = UIHostingController(rootView: self)
hosting.view.frame = CGRect(origin: .zero, size: size)
let image = hosting.view.renderedImage
I have a “report card” on screen which is a SwiftUI view. I want to use the export button to convert it to a UIImage then save to camera roll and / or send out the jpeg via email. Does SwiftUI view has a layer property or similar thing that I can work with CGContext?
I'm using XCode 11.3 on Mojave. None of these work. Could you please update the code? Thank you.
Leaving myself a note:
import PlaygroundSupport
import SwiftUI
extension UIView {
var renderedImage: UIImage {
let image = UIGraphicsImageRenderer(size: bounds.size).image { context in
layer.render(in: context.cgContext)
}
return image
}
var renderedPlaygroundImage: UIImage {
let image = UIGraphicsImageRenderer(size: bounds.size).image { context in
UIColor.lightGray.set(); UIRectFill(bounds)
context.cgContext.setAlpha(0.75)
self.layer.render(in: context.cgContext)
}
return image
}
}
extension View {
var renderedImage: UIImage {
let window = UIWindow(frame: CGRect(origin: .zero, size: CGSize(width: 320, height: 160)))
let hosting = UIHostingController(rootView: self)
hosting.view.frame = window.frame
window.rootViewController = hosting
window.makeKey()
return hosting.view.renderedImage
}
}
Text("Hello").renderedImage
Slider(value: .constant(0.5)).renderedImage
let img = ([Color.red, .orange, .yellow, .green, .blue, .purple]
.reduce(AnyView(Text("👭").font(.largeTitle)
.rotationEffect(Angle(radians: .pi)))) {
AnyView($0.padding()
.background($1)
.rotationEffect(Angle(radians: .pi / 6)))
}).renderedImage
img
If you add any effects to the views, such as .blur(radius: 10), layer.render(in: context.cgContext) will not render it to the image.
If you replace this:
layer.render(in: context.cgContext)
with this:
self.drawHierarchy(in: self.layer.bounds, afterScreenUpdates: true)
then all effects are included in the image... BUT then a new issue comes up. If you are dealing with high res images, for some reason the maximum size that can be drawn by self.drawHierarchy(in: self.layer.bounds, afterScreenUpdates: true) is CGSize(width: 2730, height: 2730). if you increase the image size to 2731 or higher, the image will not be drawn.
1: Since layer.render(in: context.cgContext) has no size limitation, would like to know if there is another way to use it with .blur and other effects.
2: or how to over come the size limitation of self.drawHierarchy(in: self.layer.bounds, afterScreenUpdates: true)
I have posted my code at: https://stackoverflow.com/questions/65163097/convert-swiftui-view-to-uiimage
Well, on Catalina I can do this. I haven't tried Mojave.