Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add AppleTV support #2067

Open
wants to merge 29 commits into
base: main
Choose a base branch
from

Conversation

maxphillipsdev
Copy link

@maxphillipsdev maxphillipsdev commented Sep 26, 2024

Proposed changes

🎉 Working now, ready for review!

This PR adds AppleTV support to Maestro by extending the iOS driver.
When a tvOS simulator is running, the iOS driver will now install a tvOS build of the maestro-ios-driver instead of the normal ios one. From there on the flow is largely the same.

Sorry for the big PR, the xcode project and tvos build files made the diff pretty big. The logic is contained to the kotlin, swift and bash scripts.

Feature additions

  • When running tests against a tvOS device, the Maestro iOS driver will now install a tvOS compatible driver
  • A subset of the DPAD commands that are available on AndroidTV through pressKey are now available on AppleTV as well. I only included the ones that had a clear mapping to something from XCUIRemote
  • Maestro studio also works (see screenshot)

Implementation

  • Migrated maestro-ios-xctest-runner from Storyboard to SwiftUI so iOS and tvOS could share the same XCode target.
  • Updated XCode build settings to support tvOS as a compilation and testing target.
  • Added platform check compiler directives in several Swift files to provide tvOS support.
  • Added bindings for Remote Dpad commands in PressButtonRequest.swift and passed them through from the iOS driver.
  • Updated the maestro-ios-xctest-runner build script to build separate versions for iOS and tvOS and copy them into the resources folder under an ios/ or tvos/ prefix.
  • Updated the iOS driver to detect when a device is using tvOS and install the correct driver.

Limitations

  • Certain commands and features of Studio are not available on tvOS since there are no XCUIRemote mappings (tapOn, swipe for example)
  • Since tvOS is a different compile target than iOS, I had to include binaries for both as part of the iOS driver resources. I haven't measured the size difference of the CLI build yet but I expect it to be larger.

To Do

  • When launching in Studio on tvOS the driver app goes to the foreground and you have to close it manually or launch another app with launchApp. UPDATE: Haven't found a fix for this yet, open to suggestions 🙏
  • Still need to test the normal Maestro commands more fully to see which need modification for tvOS
  • Studio fails when uninstalling the driver app on tvOS, still investigating

Screenshots

image

image image

Testing

  • Tested locally with tvOS and iOS simulators.
  • Ran and tested Studio locally.
  • Ran e2e tests on iOS, investigating adding separate tests for tvOS.

Issues fixed

#1515 – Add support for testing tvOS apps

private static let springboardBundleId = "com.apple.springboard"

#if os(tvOS)
private static let homescreenBundleId = "com.apple.HeadBoard"
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

On AppleTV the home screen is made up of a process called HeadBoard (the UI and cards) which is layered over another process called PineBoard (the top-right dropdown and background image). We can use the HeadBoard bundleId to allow interacting with cards inside of Studio and we can use PineBoard to get the screen dimensions

case remoteMediaFastForward = "Remote Media Fast Forward"
case RemoteSystemNavigationUp = "Remote System Navigation Up"
case RemoteSystemNavigationDown = "Remote System Navigation Down"
**/
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not all of these have bindings on tvOS, will do some more research and remove

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is it possible to take the ones without tvOS bindings, and make them no-ops on tvOS?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@douglowder Already done 👍 The file includes a compiler directive to exclude .Home and .Lock cases from the Button enum for tvOS. Anything unrecognised that isn't included in the enum will fall through the switch block in PressButtonHandler and no-op.

@timrijckaert
Copy link

Ooh! Very nice would be very helpful for our situation

@douglowder
Copy link

@maxphillipsdev I really appreciate your work on this -- it would be game-changing for testing the React Native TV repo. I will look through this, and try it out!

@douglowder
Copy link

@maxphillipsdev I created a branch in my fork where I cherry-picked all your commits against current main, and resolved conflicts. Testing this out today... https://github.com/douglowder/maestro/tree/tvos-with-main-merge

@maxphillipsdev
Copy link
Author

maxphillipsdev commented Jan 7, 2025

PR is rebased and CLI correctly installs / uninstalls the test runner when using studio / test commands. You can try it out locally by gradle syncing, launching a tvOS simulator and running ./maestro --verbose test some-test.yml or ./maestro --verbose studio. Here's an example yml I've been working with:

appId: com.apple.TVSettings
---
- launchApp
- assertVisible: 'General'
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Down

To do

  • Fix runner app not dismissing when launching studio on tvOS
  • Fix CI (the xcode proj buildstep seems to be failing silently due to an xcode version mismatch with the github agent) Had to convert a folder to a group in xcode due to a weird backwards compatibility issue with CI agent.
  • Investigate adding support for the focused selector of the assertVisible command, which would allow for making assertions about the TV's focus state. Done, thanks Doug!

@douglowder
Copy link

douglowder commented Jan 8, 2025

To do

  • Fix CI (the xcode proj buildstep seems to be failing silently due to an xcode version mismatch with the github agent)
  • Fix runner app not dismissing when launching studio on tvOS
  • Investigate adding support for the focused selector of the assertVisible command, which would allow for making assertions about the TV's focus state.

@maxphillipsdev it appears that focused support is already working, at least for RNTester in the latest RNTV build (0.77.0-0rc5 prerelease). Example:

appId: com.meta.RNTester.localDevelopment
---
- launchApp #RNTester app launch
- assertVisible: 'Components'
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Center # Arrow key down to Button example and launch
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Down # First example button should be focused
- assertVisible:
    id: 'button_default_styling'
    focused: true
- assertVisible:
    id: 'cancel_button'
    focused: false
- pressKey: Remote Dpad Down # Down arrow to the cancel button, which should now be focused
- assertVisible:
    id: 'button_default_styling'
    focused: false
- assertVisible:
    id: 'cancel_button'
    focused: true

I decided to try the same test on RNTester for Android TV, and it works, but requires some changes because the focus engine is a little different, so a different series of down and right arrow presses are needed to navigate to the expected buttons for the test.

appId: com.facebook.react.uiapp
---
- launchApp #RNTester app launch
- assertVisible: 'Components'
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Right
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Center #Arrow key down to Button example and launch
- pressKey: Remote Dpad Down
- pressKey: Remote Dpad Right
- pressKey: Remote Dpad Down #First example button should be focused
- assertVisible:
    id: 'button_default_styling'
    focused: true
- assertVisible:
    id: 'cancel_button'
    focused: false
- pressKey: Remote Dpad Down #Down arrow to the cancel button, which should now be focused
- assertVisible:
    id: 'button_default_styling'
    focused: false
- assertVisible:
    id: 'cancel_button'
    focused: true

@maxphillipsdev maxphillipsdev marked this pull request as ready for review January 11, 2025 18:43
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants