PlayTools is an essential part of PlayCover. PlayTools implements core functions of PlayCover, including display control, key mapping and bypassing.
PlayTools allows you to control:
- Resolution: Supports 1080p, 4k, 1440p and custom resolution
- Aspect Ratio: Supports 16:9, 16:10, 4:3 and custom aspect ratio
- Scale Factor: Supports custom scale factor (e.g. 1.0, 1.5, 2.0)
- Display Orientation: Supports manually rotating the game window during game play.
- Application type (most useful case is changing to type
game
and the type reflects in screen time usage) - Custom discord activity
- Device type
PlayTools provides a key mapping tool to map game actions to the keyboard, mouse, trackpad or controller.
Supported input devices include:
- Keyboard
- Mouse
- Trackpad
- Controller
Input from these devices can be mapped to these in-game actions:
- Button
- Joystick (e.g. WASD)
- Camera Control (usually by mouse)
- Draggable Button (e.g. wheel menu selection)
More mapping options such as swiping will be added in the future.
Games designed for iOS devices may not work properly on macOS. PlayTools provides a solution to bypass these issues.
Some games refuse to work on macOS because they detect the environment as a jailbroken device. PlayTools hijacks the detection process of popular jailbreak detection tools, so that the game will think it is running on a non-jailbroken device.
PlayTools also allows per-game configuration for more advanced jailbreak bypassing techniques.
PlayChain is a tool that solves key chain issues by replacing the game's key chain with a custom one. Key chain issues usually prevent the game from logging in.
Some games only work under debug environment. By inserting the Introspection Library, PlayTools tricks the game into thinking it is running under debug environment.
PlayTools is shipped with PlayCover, and is installed together with the game. You do not need to install PlayTools manually.
PlayTools provides in-game menus mostly for key mapping setup. Other features are accessible through the PlayCover settings menu.
Localization is handled in Weblate.
PlayTools runs alongside of the game. During IPA installation, PlayTools is inserted into the game's executable. When the game is launched, PlayTools will be launched together.
PlayTools uses swizzle techniques to replace framework methods and system calls with the versions provided by PlayTools. During game play, PlayTools intercepts game input events and translates them to the corresponding touch events and feed them to the game.
PlayTools is built using Xcode.
In release builds, the building script of PlayCover will automatically fetch the latest version of PlayTools from the official repository and build it. Generally, you do not need to build PlayTools manually.
To build PlayCover with a specific version of PlayTools, mostly for testing purposes, change the Cartfile of PlayCover to point to the specific version of PlayTools. You can specify which branch/tag of which repository to build from.
You can also edit the Cartfile to build from a local directory. To do this, edit the Cartfile to be:
git "file:///path/to/playtools" "branch or tag"
See Cartfile format
In most cases, Cartfile.resolved
would be automatically updated based on the Cartfile. In rare cases, you may need to manually edit the resolved file.
PlayTools can also be built separately. This is useful when you want to modify the source code.
To do so,
-
Clone the PlayTools repository.
-
Open the PlayTools project in Xcode.
-
Set the development team in the Xcode project settings. Both
PlayTools
andAKInterface
targets need to be set. You may need to create a development team first. -
Build PlayTools towards iOS platform. This will create a
PlayTools.framework
in the build directory. -
Find the build path. This can be done by right clicking on
PlayTools.framework
inProduct
, and selectingShow in Finder
. -
Deploy the build. Replace the
BUILD_PATH
in the following script with your build path and run:
#!/bin/sh
BUILD_PATH=~/Library/Developer/Xcode/DerivedData/PlayTools-<YOUR-UUID>/Build/Products/Debug-iphoneos
echo "Converting to maccatalyst"
vtool \
-set-build-version maccatalyst 11.0 14.0 \
-replace -output \
"$BUILD_PATH/PlayTools.framework/PlayTools" \
"$BUILD_PATH/PlayTools.framework/PlayTools"
echo "Codesigning PlayTools"
codesign -fs- "$BUILD_PATH/PlayTools.framework/PlayTools"
echo "Copying to PlayCover"
rm -r "/Applications/PlayCover.app/Contents/Frameworks/PlayTools.framework"
cp -r "$BUILD_PATH/PlayTools.framework" "/Applications/PlayCover.app/Contents/Frameworks/"
This script transforms the target platform to Mac Catalyst, codesigns PlayTools and copies the binaries into the PlayCover App.
- Relaunch PlayCover.
If you are debugging and testing your own code, relaunching PlayCover every time you make a change is a bit annoying.
To avoid this, run this script instead:
#!/bin/sh
BUILD_PATH=~/Library/Developer/Xcode/DerivedData/PlayTools-<YOUR-UUID>/Build/Products/Debug-iphoneos
echo "Converting to maccatalyst"
vtool \
-set-build-version maccatalyst 11.0 14.0 \
-replace -output \
"$BUILD_PATH/PlayTools.framework/PlayTools" \
"$BUILD_PATH/PlayTools.framework/PlayTools"
echo "Codesigning PlayTools"
codesign -fs- "$BUILD_PATH/PlayTools.framework/PlayTools"
echo "Copying to frameworks"
cp "$BUILD_PATH/PlayTools.framework/PlayTools" "~/Library/Frameworks/PlayTools.framework/"
This only copies PlayTools.framework/PlayTools
to ~/Library/Frameworks/PlayTools.framework/
, instead of the whole PlayTools.framework
directory into PlayCover. Changes take effect immediately, no PlayCover relaunch needed. Changes will be lost when you relaunch PlayCover.
However, If you modified AKInerface
or added localization strings, the temporary deploy method may not work for you. You may copy the whole PlayTools.framework
as described above, or directly copy them into the game you're testing on:
#!/bin/sh
BUILD_PATH=~/Library/Developer/Xcode/DerivedData/PlayTools-<YOUR-UUID>/Build/Products/Debug-iphoneos
cp "$BUILD_PATH/PlayTools.framework/PlugIns/AKInterface.bundle" "~/Library/Containers/io.playcover.PlayCover/Applications/<YOUR-GAME-NAME>.app/PlugIns"
cp "$BUILD_PATH/PlayTools.framework/*.lproj" "~/Library/Containers/io.playcover.PlayCover/Applications/<YOUR-GAME-NAME>.app/"
This will be overwritten when you launch the game through PlayCover. You can launch the game from Finder to avoid this.
The main part of PlayTools. This ends up as a dynamic library. This part is built towards iOS platform because it uses iOS APIs that are not available on Mac Catalyst.
A bridge that encapsulates native macOS APIs to expose an interface for PlayTools. This includes manipulating mouse and keyboard events, controlling cursor and application state, and reading window information.
Localizations for PlayTools are tricky. As PlayTools runs as a dynamic library inside the game, localizable strings must be copied into the game to take effect. This is done during IPA installation and every launch of the game through PlayCover.
To avoid conflicting with the game's own localizable strings, PlayTools' localizable strings are renamed to PlayTools.strings
, instead of the default Localizable.strings
.