Streaming is available in most browsers,
and in the Developer app.
-
Prototype with Xcode Playgrounds
Speed up feature development by prototyping new code with Xcode Playgrounds, eliminating the need to keep rebuilding and relaunching your project to verify your changes. We'll show you how using a playground in your project or package can help you try out your code in various scenarios and take a close look at the returned values, including complex structures and user interface elements, so you can quickly iterate on a feature before integrating it into your project.
Chapters
- 3:00 - Exploring the data model and inspecting UI snapshots
- 14:26 - Becoming familiar with a new package
- 18:07 - Using the Live View
- 22:03 - Running the updated project
Resources
Related Videos
WWDC20
-
Download
♪ ♪ Dariusz: Hi, I'm Dariusz, and I'm an engineer on the Xcode Playgrounds team. I would like to show you how the improvements in Xcode Playgrounds make it easier to prototype new features in your project. There are many scenarios where Xcode Playgrounds can improve your workflow. First of all, they allow you to skip rebuilding and relaunching the project whenever you want to prototype a new feature or simply try out a small change in your code. They also make it much easier to execute code that otherwise would be quite difficult to reach, like the logic responsible for generating an order summary in a shopping app. Of course, such code can and should be verified with tests, but Playgrounds can be a great starting point for developing them as well. It's also a perfect environment for trying out code before introducing new dependencies to your project.
I'd like to show you how Xcode Playgrounds are helping me in my project. I've been working on a small app that's meant to help me with my wildlife photography. At the moment, my app helps me with keeping track of the species that I've already found and photographed. Now, I'd like this app to help me with one quite important aspect of wildlife photography, which is actually finding the animals in the first place. I want to introduce a new tab for a checklist view. I already started implementing the view, and I'm quite happy with this user interface. Each bird has a checkbox, which will allow me to track my progress. However, with over 2,000 birds on the list, I probably won't feel like I'm advancing much. It would be great to narrow this list down a bit. To do that, I will have to adjust the birdsToShow computed property in my custom ChecklistView. At the moment, it simply creates my BirdProvider type configured for North America and returns all the bird species found on the whole continent. To avoid the frequent rebuilding and relaunching the project and then navigating to the ChecklistView to see my changes, I will try to adjust this code in an Xcode Playground. I will start by adding a new playground to my project.
This is an iOS app, so I'll stick with the iOS templates. I'll type "playground" in the filter bar and select a blank playground.
In this case, I will probably get rid of the playground once I try out a few things, so I can keep the default "MyPlayground" name.
I'll get rid of the default contents of the playground.
To iterate even faster, I'll switch to "Automatically run" in the menu that shows up when I long click the run button on the bottom bar.
This causes the playground to automatically execute the whole code whenever I stop making changes. Notice that the playgrounds added to projects have two settings enabled by default: Build Active Scheme and Import App Types. They will ensure that the active scheme is built before each playground execution and that the app target modules are automatically imported. This makes it much easier to work with the types defined within your project. I can close the inspector to give some more room for the playground.
Let's start by declaring a BirdProvider instance like we saw in the birdsToShow property of my ChecklistView.
The result sidebar to the right of the editor shows me that this declaration has generated a playground result. I can use the inline result toggle to see more details.
The inline result shows the details of this BirdProvider instance along with its two properties: the birds array and the provided region. Now in Xcode 15, each row also has a type information label, which shows a short summary of the type, and I can use the tooltips for each row to see more details.
For example, the tooltip tells me that the BirdProvider type comes from my app module and that the region enum was defined within that struct. Let's expand the array row to see more details about the birds.
Notice that when I start to interact with the inline result view, Xcode 15 highlights the source code that produced the result. In this case, the view displays the value assigned to the birdProvider constant. This makes it much easier to understand the displayed values. Let's take a closer look at the array elements.
We saw the nice summaries for the region and birds array properties, but by default, the rows representing each bird only tell us about the array indexes. That's because the custom Bird type has no description defined. We can improve this by making the Bird type conform to CustomStringConvertible protocol. I could add an extension here in a playground or in its Sources directory. Both these options would only affect the Bird type in the playground's scope, and we wouldn't get the nice new descriptions in other places that rely on them, like the debugger. That's why I'll add the extension in the file that defines the Bird type.
With the new description definition, each row should show me the common and scientific name of the bird. Let's go back to the playground and see the new description in action.
In the automatic mode of playground execution, the playground will automatically re-execute when you reopen it. Let's expand the birds array row again to check the descriptions.
This is much better. Now it's clear what the rows represent without expanding them, but let's take a look at other properties of the Bird type.
I have already found and photographed some of these birds, and you can see that some of them already have the photo property, like this Atlantic puffin.
Once I click on its row, the photo is displayed in the new split view-based user interface, which allows me to see the structure of the object along with the preview.
By default, there is no such preview for my custom Bird type when I click on its row. It would be great to already see the photo here, as I'm not that familiar with all the species names. To achieve that, I can use the CustomPlaygroundDisplayConvertible protocol. As the name of the protocol suggests, this conformance only affects the playground representation, so this time, I'll add the extension in the playground's Sources directory.
I'll import the app module to access the Bird type and add a simple extension that returns the photo property as the playgroundDescription.
Notice that I'm explicitly casting photo to Any in the return statement. Without it, the compiler would warn me that we are losing an important piece of information about the value being an optional. This is fine in this case, as Xcode Playgrounds handles optionals by only creating a custom description for objects that don't return nil in the playgroundDescription property. Let's go back to the playground again to check the new descriptions.
In Xcode 15, the playgroundDescription returned by types conforming to CustomPlaygroundDisplayConvertible will be displayed in the split view along the object's structure.
Now, the birds that already have a photo will quickly show it without the need of expanding the row. This will make it much easier to work with large collections of this custom type. But today, I'd like to focus on birds that don't have the photos yet. Let's close this inline result and filter out all the birds that have a photo already.
You can see that those two sidebar annotations look a bit different. That's because there are multiple expressions on the new line. I can click on the new control to see the familiar summaries for each expression.
Hovering over the inline result toggles highlights their source code ranges. This makes it clear that the array is the result assigned to the birdsToFind constant and true is the latest value produced by the closure passed to the filter function.
The result sidebar tells me that the number of all birds that I'm yet to photograph is still over 1,800, which is quite intimidating. While this might ultimately be the goal, I'd like to make it lower by focusing on a smaller group of birds, for example owls. June is a great month to go owling, but that's a topic for another session or two. Let's filter out the birds from other families.
The array only has five elements now, which is way more encouraging. I'd like to try out my custom ChecklistView with this small group of birds. To do that, I'll create a ChecklistView instance and add each bird one by one.
Let's open the inline result for the ChecklistView.
As a UIView subclass, it now also shows a few properties along with the snapshot. I can switch to the Value history mode, which now also uses the new split-view-based user interface.
This allows me to see what the ChecklistView looked like in each loop iteration.
It actually helped me find a problem with my view, which incorrectly says "birds" in the header for just one bird. I should be able to fix this by adjusting the strings defined in the new String Catalog.
The first row seems to be the one that is used in the header of my Checklist View. I'll bring up the context menu and select Vary By Plural.
Once I do that, the affected rows change their state to "Needs review." I'll adjust the singular form of this string.
To learn more about the new String Catalogs, make sure to check the "Discover String Catalogs" session. Let's go back to the playground and see what the checklist looks like now with this change.
The headers look correct in each loop iteration. I can close this inline result now.
I think my ChecklistView is ready to use. I'll quickly bring the code that we used to narrow down the list of birds to my project.
I'll copy these three lines to the birdsToShow property in my ChecklistView.
I will also add a return statement with the new, much smaller array of birds.
Before I rerun my project, I'd like to prototype one more feature. You could see that each row in my custom Checklist View had a disclosure indicator. Once I select a row in the list, it opens a simple map view. It's not that useful yet, but I would like to fetch the data about the most recent sighting of the selected bird and show it on the map. To achieve that, I will have to adjust the sightingsToShow(for bird:) function in my ChecklistView. To help with that, I have already added one dependency to my project. The BirdSightings package makes it easy to fetch the data from one of the citizen-science websites, where people report their sightings. I haven't used that package before, so I'm not that familiar with its API yet. Luckily, the package includes documentation in form of a playground that shows a few examples.
This is a great way to allow the clients of your package to try out the provided API. It looks like we will need to provide two arguments to the fetchSightings function: the code of the bird to look for and the location to look around. I can execute this playground to learn more about the expected results.
Let's go back to our playground and try to use it.
I can close the navigator to give me more room in the editor area.
Before we call the function, I need to add two import statements. I'll import the CoreLocation framework to be able to work with coordinates and the BirdSighting framework to use its API.
For the function arguments, we can simply start with the first bird from the list.
Looks like we will be searching for the short-eared owl. Notice that I used the force-unwrapping here. I don't have to worry that much about error handling in the Playgrounds environment, but it's important to keep that in mind when bringing the code to your project.
As for location, I will probably use my current location most of the time, but the ability to provide specific coordinates will be great for two things: testing my code and planning all the road trips. Let's try to see what we can find around Apple Park.
Before I introduce a network call, I will switch to the manual mode of playground execution to make sure that I can avoid unnecessary calls. Such requests can take a while, and I want to iterate on my new feature as fast as possible. To do that, I will once again bring up the menu on the bottom bar and select Manually Run.
Now, I have full control over the parts of the code that should be executed. Let's add the fetching code.
The control in the source editor gutter shows me that executing those two lines would not re-execute all the lines we already executed above. Let's execute the new lines and see if we get any data.
Luckily, there are some sightings. Let's take a look at the first one, which should be the most recent.
Looks like short-eared owls were most recently seen at Changmen Cliffs Reserve. I'm not that familiar yet with all the birdwatching hotspots in the area, so it will be great to see the data about the sighting in my SightingMapView. Let's initialize it with the fetched sighting data.
For such complex user interface elements like map views, we can use the playgrounds live view to see the large, fully interactive preview. To use it, I first need to import the PlaygroundSupport framework.
Now, I'm ready to set up the live view and execute the playground.
I modified the already executed part of the playground by adding the import statement at the top of the file, so the control in the source editor gutter tells me that it will need to re-execute the whole file. I will lose results from the last execution, but that shouldn't be a problem in this case, so I'll go ahead and execute the playground.
I don't recall any islands too close to Apple Park. Since the live view is fully interactive, I can zoom out a little like I would in an iPhone simulator to see where we are.
Looks like we are way too far east. Let's close the live view in the Editor Options and try to see where the problem was introduced.
My SightingMapView was initialized with the mostRecentSighting constant, so let's check its value. Instead of opening the inline result, I can take a quick look at the value by clicking the eye icon in the sidebar.
In Xcode 15, we improved the playground results for some of the MapKit and CoreLocation types. Playgrounds can now show a preview for CLLocationCoordinate2D, so let's take a look at the location property.
This seems to be pointing at the same location, so this doesn't seem to be caused by my SightingMapView. We got the wrong location from the BirdSightings package, so it could either be caused by a problem in the package, or I passed a wrong location to start with. Let's verify the latter.
This doesn't seem to be close to Apple Park either. Since we are way too far east, I think this might just be a matter of mixing west with east. Let's try to fix this by adding the minus sign in front of the longitude and re-executing the playground.
Now this is definitely Apple Park. Let's execute the remaining part of the playground with the updated location and reopen the live view in the Editor Options.
That's more like it. Now I know that Coyote Hills might be the best place to photograph new species. Let's quickly bring the fetching code to my ChecklistView.
I'll copy these three lines to the sightingsToShow function.
Instead of always using the hardcoded Apple Park location here, I'll replace it with the lastCurrentLocation that I get from CLLocationManager.
I will also add the return statement with the new mostRecentSighting.
It's time to finally run the project with our changes.
The checklist looks much more realistic now that we focused on a small group of birds. Let's see what happens when I click on a row now.
That's great. My app can now show me the most recent sighting for the selected bird. Of course there is still a lot of room for improvements. For example, I should be showing a progress indicator while the sightings are fetched in the background, but this is a great starting point. This app will already help me a lot, and Xcode Playgrounds made these improvements so much easier to work on. In this session, we used Xcode Playgrounds to quickly prototype new features of a project. We used CustomStringConvertible and CustomPlaygroundDisplayConvertible protocols to customize the representations of our custom types. We saw how adjusting the playground execution mode can speed up your workflow. Using Value History mode allowed us to quickly see how our classes react to multiple inputs. Finally, we used the playground's live view to take a closer look at complex user interface elements. Thank you for watching.
-
-
2:04 - birdsToShow computed property before the required changes
var birdsToShow: [Bird] { // TODO: Narrow down the list of birds to find. let birdProvider = BirdProvider(region: .northAmerica) return birdProvider.birds }
-
4:15 - birdProvider instance
let birdProvider = BirdProvider(region: .northAmerica)
-
6:31 - CustomStringConvertible
extension Bird: CustomStringConvertible { public var description: String { return "\(commonName) (\(scientificName))" } }
-
8:17 - CustomPlaygroundDisplayConvertible
extension Bird: CustomPlaygroundDisplayConvertible { public var playgroundDescription: Any { return photo as Any } }
-
9:45 - Birds without a photo
let birdsToFind = birdProvider.birds.filter { $0.photo == nil }
-
10:52 - Owls without a photo
let owlsToFind = birdsToFind.filter { $0.family == .owls }
-
11:15 - Verifying the ChecklistView
let checklist = ChecklistView() for bird in owlsToFind { checklist.add(bird) }
-
13:41 - birdsToShow computed property with the required changes
var birdsToShow: [Bird] { let birdProvider = BirdProvider(region: .northAmerica) let birdsToFind = birdProvider.birds.filter { $0.photo == nil } let owlsToFind = birdsToFind.filter { $0.family == .owls } return owlsToFind }
-
14:21 - sightingToShow function before the required changes
func sightingToShow(for bird: Bird) -> Sighting? { // TODO: Use BirdSightings package to fetch the most recent sighting. return nil }
-
15:04 - BirdSightings package example
let barnOwlCode = "BNOW" let centralParkLocation = CLLocationCoordinate2D(latitude: 40.785091, longitude: -73.968285) let sightingsProvider = SightingsProvider() sightingsProvider.fetchSightings(of: barnOwlCode, around: centralParkLocation)
-
16:08 - Parameters for sightings fetching
let bird = owlsToFind.first! let appleParkLocation = CLLocationCoordinate2D(latitude: 37.3348655, longitude: 122.0089409)
-
17:31 - Sightings fetching
let sightingsProvider = SightingsProvider() let sightings = sightingsProvider.fetchSightings(of: bird.speciesCode, around: appleParkLocation)
-
18:23 - Initializing the SightingMapView
let mostRecentSighting = sightings.first let sightingMapView = SightingMapView(sighting: mostRecentSighting)
-
18:55 - Setting up the playground's live view
import PlaygroundSupport PlaygroundPage.current.liveView = sightingMapView
-
22:20 - sightingToShow function with the required changes
func sightingToShow(for bird: Bird) -> Sighting? { let sightingsProvider = SightingsProvider() let sightings = sightingsProvider.fetchSightings(of: bird.speciesCode, around: lastCurrentLocation) let mostRecentSighting = sightings.first return mostRecentSighting }
-
-
Looking for something specific? Enter a topic above and jump straight to the good stuff.
An error occurred when submitting your query. Please check your Internet connection and try again.