API Basic Configs

The basic configuration method of YukiHookAPI is introduced here.

Function Configs

Either Use as Xposed Module Configs or Use as Hook API Configs, you can specify YukiHookAPI for configuration.

configs Method

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.

Hooker Configs

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.

Created by lambda

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.

encase Method

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") {
        "$packageName.DemoClass".toClass()
            .method { 
                // Your code here.
            }.hook {
                // Your code here.
            }
    }
}

Sample Code 2

encase {
    loadApp(name = "com.example.demo") {
        "$packageName.DemoClass".toClass()
            .method { 
                // Your code here.
            }.hook {
                // Your code here.
            }
    }
}

Do your Hook operations in the encase method.

Created by Custom Hooker

This scheme is more suitable for large-scale projects, such as the need to classify Hooker or classify the role of Hook.

encase Method

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 Usage

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 to use singleton object to create, you can also use class but it is generally 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") {
            "$packageName.DemoClass".toClass()
                .method { 
                    // Your code here.
                }.hook {
                    // Your code here.
                }
        }
        loadApp(name = "com.example.demo2") {
            "$packageName.CustomClass".toClass()
                .method { 
                    // Your code here.
                }.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

object HookEntry : IYukiHookXposedInit {

    override fun onHook() = encase {
        loadApp(name = "com.example.demo", ChildCustomHooker)
    }
}

object ChildCustomHooker : YukiBaseHooker() {

    override fun onHook() {
        "$packageName.DemoClass".toClass()
            .method { 
                // Your code here.
            }.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() {
        "$packageName.DemoClass".toClass()
            .method { 
                // Your code here.
            }.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 Hook entry class.

The following example

object HookEntry : IYukiHookXposedInit {

    override fun onHook() =
        YukiHookAPI.encase(FirstHooker, SecondHooker, ThirdHooker ...)
}

Of course, we can also abbreviate it.

The following example

object HookEntry : IYukiHookXposedInit {

    override fun onHook() = encase(FirstHooker, SecondHooker, ThirdHooker ...)
}

Special Case

As we mentioned above, it is generally not recommended to use class to create child Hookers, but there is a special case where it may still be necessary to keep your Hooker supporting multiple instantiations.

There is a rare possibility that there are multiple package names in a process.

In this case, when YukiHookAPI finds that the child Hooker is a singleton, it will ignore it and print a warning message.

This Hooker "HOOKER" is singleton or reused, but the current process has multiple package name "NAME", the original is "NAME"
Make sure your Hooker supports multiple instances for this situation
The process with package name "NAME" will be ignored

In this case, we only need to modify object to class or determine the package name during loading and then load the child Hooker.

For example, in the above cases, the following forms can be used to load.

The following example

encase {
    // Assume this is the app package name and child Hooker you need to load
    loadApp("com.example.demo", YourCustomHooker)
}

Expansion Features

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") {
        "$packageName.DemoClass".toClass()
            .method { 
                // Your code here.
            }.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") {
        "$packageName.DemoClass".toClass()
            .method { 
                // Your code here.
            }.hook {
                // Your code here.
            }
        // Create a Resources Hook in the app
        resources().hook {
            // Your code here.
        }
    }
}

Precautions

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 initZygoteencase

  • load handleLoadPackageencase

  • load handleInitPackageResourcesencase

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 {
    //  Wrong usage, can't start Hook directly
    "com.example.demo.DemoClass".toClass()
        .method { 
            // Your code here.
        }.hook {
            // Your code here.
        }
    //  Wrong usage, can't start Hook directly
    resources().hook {
        // ...
    }
}

Sample Code 2

object HookEntry : IYukiHookXposedInit {

    override fun onHook() {
        // <Scenario 1>
        encase {
            loadHooker(CustomHooker)
        }
        // <Scenario 2>
        encase(CustomHooker)
    }
}

object CustomHooker : YukiBaseHooker() {

    override fun onHook() {
        //  Wrong method of use
        // Because there is no judgment object in the outer layer, you cannot start Hook directly
        "com.example.demo.DemoClass".toClass()
            .method { 
                // Your code here.
            }.hook {
                // Your code here.
            }
    }
}

Below is a correct example of the wrong example above.

Sample Code 1

encase {
    // ✅ Correct usage, load in Zygote
    loadZygote(CustomHooker)
     // ✅ Correct usage, load in Zygote
    loadZygote {
        // ✅ Correct usage, Hook in Zygote
        resources().hook {
            // ...
        }
    }
    // ✅ The correct way to use it, use the app scope to load
    loadApp(/** name parameter optional */, hooker = CustomHooker)
    // ✅ The correct way to use it, load the Hooker after judging the scope of the app
    loadApp(/** name parameter optional */) {
        loadHooker(CustomHooker)
         // ✅ Correct usage, Hook in app scope
        "com.example.demo.DemoClass".toClass()
            .method { 
                // Your code here.
            }.hook {
                // Your code here.
            }
        // ✅ Correct usage, Hook in app scope
        resources().hook {
            // ...
        }
    }
}

Sample Code 2

object HookEntry : IYukiHookXposedInit {

    override fun onHook() {
        encase(CustomHooker)
    }
}

object CustomHooker : YukiBaseHooker() {

    override fun onHook() {
        // ✅ 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 */) {
            "com.example.demo.DemoClass".toClass()
                .method { 
                    // Your code here.
                }.hook {
                    // Your code here.
                }
        }
    }
}

Precautions when using as Hook API

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.