import{_ as s,o as n,c as a,a as e}from"./app.0df13dd9.js";const o={},l=e(`
The basic configuration method of
YukiHookAPI
is introduced here.
Either Use as Xposed Module Configs or Use as Hook API Configs, you can specify
YukiHookAPI
for configuration.
fun configs(initiate: Configs.() -> Unit)
The configs
method implements a lambda
method body on the Configs
class, which you can easily call for configuration.
Tips
For more functions, please refer to the YukiHookAPI.configs method.
The most important part of an Xposed Module or Hook API is the creation and use of Hooker.
YukiHookAPI
provides two ways to use it.
This solution is the simplest. If your module has few functions and a small amount of code, and does not need to be classified, it is recommended to create it in this way.
fun encase(initiate: PackageParam.() -> Unit)
The encase
method is the beginning of all Hook life. In a Module App or a Hook process, the encase
method can only be used once to create a Hooker.
PackageParam
is an important instance object of the Host App, and PackageParam
is used to implement all Hook operations on the current Hook object.
Tips
For more functions, please refer to PackageParam.
The encase
method can be created in the onHook
method using two schemes.
Sample Code 1
YukiHookAPI.encase {
loadApp(name = "com.example.demo") {
findClass(name = "$packageName.DemoClass").hook {
// Your code here.
}
}
}
Sample Code 2
encase {
loadApp(name = "com.example.demo") {
findClass(name = "$packageName.DemoClass").hook {
// Your code here.
}
}
}
Do your Hook operations in the encase
method.
This scheme is more suitable for large-scale projects, such as the need to classify Hooker or classify the role of Hook.
fun encase(vararg hooker: YukiBaseHooker)
Also for the encase
method, the variable array parameter hooker
of the method here provides an object for creating an entry, you can load all Hookers extends YukiBaseHooker
at one time.
YukiBaseHooker
extends PackageParam
, you need to extends your child Hooker from YukiBaseHooker
.
Tips
For more functions, please refer to YukiBaseHooker.
The following example
object CustomHooker : YukiBaseHooker() {
override fun onHook() {
// Your code here.
}
}
Child Hooker recommended singleton object
creation, you can also use class
but not recommended.
Notice
You don't need to re-call encase in the onHook method extends YukiBaseHooker, this is wrong and will not take effect, you should start writing your Hook code directly .
The following example
object CustomHooker : YukiBaseHooker() {
override fun onHook() {
loadApp(name = "com.example.demo1") {
findClass(name = "$packageName.DemoClass").hook {
// Your code here.
}
}
loadApp(name = "com.example.demo2") {
findClass(name = "$packageName.CustomClass").hook {
// Your code here.
}
}
}
}
As a child hooker, you can also call the loadApp
method externally, and then directly start the Hook internally.
The following example
class HookEntry : IYukiHookXposedInit {
override fun onHook() = encase {
loadApp(name = "com.example.demo", ChildCustomHooker)
}
}
object ChildCustomHooker : YukiBaseHooker() {
override fun onHook() {
findClass(name = "$packageName.DemoClass").hook {
// Your code here.
}
}
}
You can use the loadHooker
method to load another Hooker in multiple layers in the child Hooker, please do as you like.
The following example
object FirstHooker : YukiBaseHooker() {
override fun onHook() {
findClass(name = "$packageName.DemoClass").hook {
// Your code here.
}
loadHooker(SecondHooker)
loadHooker(ThirdHooker)
}
}
Once all Hookers are set up, you can load your Hooker in the onHook
method of your HookEntryClass
.
The following example
class HookEntry : IYukiHookXposedInit {
override fun onHook() =
YukiHookAPI.encase(FirstHooker, SecondHooker, ThirdHooker ...)
}
Of course, we can also abbreviate it.
The following example
class HookEntry : IYukiHookXposedInit {
override fun onHook() = encase(FirstHooker, SecondHooker, ThirdHooker ...)
}
If your current Hook Framework supports and enables the Resources Hook feature, you can now create Resources Hooks directly in encase
.
You don't need to separate the initZygote
, handleLoadPackage
, handleInitPackageResources
methods to perform different functions as before using the Xposed API.
In YukiHookAPI
, these functions are seamless.
The following example
encase {
loadApp(name = "com.example.demo") {
findClass(name = "$packageName.DemoClass").hook {
// Your code here.
}
// Create a Resources Hook (fixed usage)
resources().hook {
// Your code here.
}
}
}
You can also use the loadZygote
method to load the first event initZygote
after a new process has been forked.
The following example
encase {
loadZygote {
ActivityClass.hook {
// Your code here.
}
// Create a Resources Hook in Zygote
resources().hook {
// Your code here.
}
}
loadApp(name = "com.example.demo") {
findClass(name = "$packageName.DemoClass").hook {
// Your code here.
}
// Create a Resources Hook in the app
resources().hook {
// Your code here.
}
}
}
It is wrong to load Hooker directly or start Hook directly, encase
event will go through three callbacks after being loaded by Hook Framework.
load initZygote
\u2192 encase
load handleLoadPackage
\u2192 encase
load handleInitPackageResources
\u2192 encase
In this process, you need to use loadApp
, loadSystem
, loadZygote
to distinguish the calling domain of each loading code, otherwise your code will be executed multiple times and cause errors.
Notice
Whether you use encase to create the lambda method body or use the Hooker form directly, you should not directly load the Hooker or start the Hook directly in the first onHook event.
Below are two error examples.
Sample Code 1
encase {
// \u2757 Wrong usage, can't start Hook directly
findClass(name = "com.example.demo.DemoClass").hook {
// ...
}
// \u2757 Wrong usage, can't start Hook directly
resources().hook {
// ...
}
}
Sample Code 2
class HookEntry : IYukiHookXposedInit {
override fun onHook() {
// <Scenario 1>
encase {
loadHooker(CustomHooker)
}
// <Scenario 2>
encase(CustomHooker)
}
}
object CustomHooker : YukiBaseHooker() {
override fun onHook() {
// \u2757 Wrong method of use
// Because there is no judgment object in the outer layer, you cannot start Hook directly
findClass(name = "com.example.demo.DemoClass").hook {
// ...
}
}
}
Below is a correct example of the wrong example above.
Sample Code 1
encase {
// \u2705 Correct usage, load in Zygote
loadZygote(CustomHooker)
// \u2705 Correct usage, load in Zygote
loadZygote {
// \u2705 Correct usage, Hook in Zygote
resources().hook {
// ...
}
}
// \u2705 The correct way to use it, use the app scope to load
loadApp(/** name parameter optional */, hooker = CustomHooker)
// \u2705 The correct way to use it, load the Hooker after judging the scope of the app
loadApp(/** name parameter optional */) {
loadHooker(CustomHooker)
// \u2705 Correct usage, Hook in app scope
findClass(name = "com.example.demo.DemoClass").hook {
// ...
}
// \u2705 Correct usage, Hook in app scope
resources().hook {
// ...
}
}
}
Sample Code 2
class HookEntry : IYukiHookXposedInit {
override fun onHook() {
encase(CustomHooker)
}
}
object CustomHooker : YukiBaseHooker() {
override fun onHook() {
// \u2705 The correct method of use, since there is no judgment object in the outer layer
// it is necessary to judge the scope of the app before performing Hook
loadApp(/** name parameter optional */) {
findClass(name = "com.example.demo.DemoClass").hook {
// ...
}
}
}
}
If you are using it as a Hook API, then you only need to differentiate the encase
method at the entry point.
Notice
The encase method provides two identical methods for use as a Hook API, but with only one more parameter baseContext than the previous two.
Method 1
fun encase(baseContext: Context?, initiate: PackageParam.() -> Unit)
Method 2
fun encase(baseContext: Context?, vararg hooker: YukiBaseHooker)
The baseContext
here only needs to fill in the Context
you got at attachBaseContext
, and other usages are exactly the same as the above.
Pay Attention
Never use the encase method in an Xposed way without omitting the baseContext parameter, this will lead to your Hook not work at all.
The Resources Hook feature is not supported as Hook API.