Situation:
I'm developing an iPad app using SwiftUI
, and I have an external screen connected to the iPad. In my app, I want to provide a drawing feature using PKCanvasView
. The goal is to allow users to draw on the iPad screen and have the drawing display in real-time on the external screen. How can I achieve this synchronization between the PKCanvasView
on the iPad and the external screen?
I have tried using an ObservableObject
and then accessing the canvas property to pass it to the ExternalCanvas View
When I try this, I can draw on the iPad but when I connect an external monitor it removes the PKCanvasView
off the iPad screen and puts it on the external monitor.
EDIT: I have also been back and forward with ChatGPT for almost 3 hours trying to get a resolution, but it was unable to suggest anything, mainly because it didn't seem to grasp the concept of what I was trying to achieve and they more it tried the further it got from the desired outcome
class SharedData : ObservableObject {
@Published var canvas : PKCanvasView = PKCanvasView()
}
import SwiftUI
import PencilKit
struct ExternalView: View {
@EnvironmentObject var shared : SharedData
var body: some View {
ExternalCanvas(canvasView: shared.canvas)
}
}
struct ExternalCanvas: UIViewRepresentable {
@State var canvasView: PKCanvasView
func makeUIView(context: Context) -> PKCanvasView {
canvasView.drawingPolicy = .anyInput
canvasView.tool = PKInkingTool(.pen, color: .black, width: 15)
return canvasView
}
func updateUIView(_ canvasView: PKCanvasView, context: Context) { }
}
PKCanvasView
is a subclass ofUIView
. AUIView
can only have a single superview. So you cannot use the same instance ofPKCanvasView
in two places (built-in screen and external screen) at the same time. You need onePKCanvasView
for each place where you want to show the drawing, and you need to keep thedrawing
property of all thePKCanvasView
s in sync.PKCanvasView
only updates itsdrawing
when a stroke ends, so you cannot quite get all the views to synchronize in "real-time". This is what you get:First, I wrote a wrapper for
PKCanvasView
:I like to keep my
UIViewControllerRepresentable
types private, so that wrapper is just aView
, and it uses a private type namedRep
:The
Controller
type is aUIViewController
that manages thePKCanvasView
:The
Controller
type conforms toPKCanvasViewDelegate
so that the canvas can notify it when the drawing changes:In a real app, you might want to add more properties to the
PKCanvasViewWrapper
type to specify how to configure thePKCanvasView
. Then, in theController
'supdate(to:)
method, you'd need to use those properties to configure thePKCanvasView
.Anyway, I then wrote an
ObservableObject
type to hold the sharedPKDrawing
:I added an
appModel
property to myAppDelegate
to make it available to all scenes:Then I wrote a SwiftUI view to appear on the internal screen:
and I wrote a scene delegate to create that view and show it in a window:
I copied the view and scene delegate for the external screen, just changing "Internal" to "External" everywhere.
Finally, I updated the "Application Scene Manifest" in the Info tab of my app target:
You can find the full project in this github repo.