首先,欢迎您为 QAuxiliary 这个项目做出贡献。
本项目由若干个功能组成,在大多数情况下,每个功能都是独立的,但是也有一些功能是依赖于其他功能的。 总体来说,每个功能都是由使用它的开发者维护的。这意味着,存在一些功能,由于没有开发者使用,而无人维护。 这很正常,本项目的初衷就是通过分享自己已经开发的功能,来省去其他有相同需求的人的重复劳动,而不是追求功能的多少。
由于本项目的特殊性,我并不推荐开发者在本项目上花费过多的时间,开发者只需要让自己需要的功能正常运行即可。 至于那些你自己都不用的功能,我并不推荐你去维护它,让真正需要它的人去维护就行了。
大可把开源项目当作一个并不重要的爱好,心情好就玩一玩,不想做了就不做,玩腻了就丢一边,没有人有资格去要求你做什么。 取悦自己,而不是为了没用的胜负欲去取悦别人。
- 如果你是第一次提交代码,先选一个 package, 通常是反写的域名加上一些其他信息,如
com.example.hook
. 然后将这个 package 加入到 proguard-rules.pro 中,以防止被 R8 优化掉。 - 在 package 下新建一个类,一个功能一个类,类名应该是一个名词,如
RemoveShakeAdExampleHook
, 按下面的模板编写功能代码。 写了类后,它就会自动被注册到功能列表中,如果用户启用了这个功能,那么这个类的 initOnce 方法将会自动调用。 - 测试功能,记得去模块里打开你写的功能。由于 Android Studio 在 Android 11+ 默认启用部署优化(deployment optimization),
使用 JVMTI(ARTTI) 来热加载代码,这会导致实际并没有更新 apk,因此需要在 Android Studio 中禁用部署优化。
在
Run
菜单中选择Edit Configurations...
,在Android App
标签页中,选择QAuxiliary.app
, 在General
标签页中,将Install Options
中的Always install with package manager (disable install optimization)
勾选上。 你也可以通过直接运行:app:installDebug
task 来安装应用,而不是通过 Android Studio 运行 app。
如果你倾向于使用 Kotlin 来编写功能,可以参考以下模板.
package com.example.hook
import cc.ioctl.util.hookBeforeIfEnabled
import io.github.qauxv.base.annotation.FunctionHookEntry
import io.github.qauxv.base.annotation.UiItemAgentEntry
import io.github.qauxv.dsl.FunctionEntryRouter
import io.github.qauxv.hook.CommonSwitchFunctionHook
import io.github.qauxv.util.Initiator
import io.github.qauxv.util.QQVersion
import io.github.qauxv.util.requireMinQQVersion
// FunctionHookEntry 和 UiItemAgentEntry 用于注册功能,这两个注解都是必要的
// 注意是 object,而不是 class
@FunctionHookEntry
@UiItemAgentEntry
object RemoveShakeAdExampleHook : CommonSwitchFunctionHook() {
override val name = "这里写功能名字"
override val description = "这里写功能描述"
override val uiItemLocation: Array<String> = FunctionEntryRouter.Locations.这里写功能位置
// isAvailable 可选,可以不写
override val isAvailable: Boolean get() = requireMinQQVersion(QQVersion.最低支持版本)
override fun initOnce(): Boolean {
// 在这里写功能初始化代码,如果用户启用了这个功能,那么这里的代码将会被执行一次
// initOnce 里可以随便 throw 异常, throw 的异常会显示在日志和故障排除方便定位问题
val klass = Initiator.loadClass("com.tencent.mobileqq.example.SomeClassManager")
// Initiator.loadClass 用于加载 QQ 的类,如果类不存在,将会 throw ClassNotFoundException
val someMethod = kTroopInfo.getDeclaredMethod("someMethod", Long::class.java)
// hookBeforeIfEnabled 只有在用户启用了这个功能的情况下才会执行 hook 回调
// 避免使用 XposedBridge 和 XposedHelpers,因为这些类在 new Xposed API 中不复存在
hookBeforeIfEnabled(someMethod) {
// 做一些操作,其中 it 是一个 HookParam 对象,可以通过 it.args 获取方法参数
it.result = 0L
}
// return true 表示初始化成功,false 或者抛异常表示初始化失败
return true
}
}
如果你倾向于使用 Java 来编写功能,可以参考以下模板.
package com.example.hook;
import cc.ioctl.util.hookBeforeIfEnabled;
import io.github.qauxv.base.annotation.FunctionHookEntry;
import io.github.qauxv.base.annotation.UiItemAgentEntry;
import io.github.qauxv.dsl.FunctionEntryRouter;
import io.github.qauxv.hook.CommonSwitchFunctionHook;
import io.github.qauxv.util.Initiator;
import io.github.qauxv.util.QQVersion;
import io.github.qauxv.util.requireMinQQVersion;
// FunctionHookEntry 和 UiItemAgentEntry 用于注册功能,这两个注解都是必要的
@FunctionHookEntry
@UiItemAgentEntry
public final class RemoveShakeAdExampleHook extends CommonSwitchFunctionHook {
// INSTANCE 是必须的,因为 Java 没有 object,所以我们需要一个单例
public static final RemoveShakeAdExampleHook INSTANCE = new RemoveShakeAdExampleHook();
@Override
public String getName() {
return "这里写功能名字";
}
@Override
public String getDescription() {
return "这里写功能描述";
}
@Override
public String[] getUiItemLocation() {
return FunctionEntryRouter.Locations.这里写功能位置;
}
// isAvailable 可选,可以不写
@Override
public boolean isAvailable() {
return requireMinQQVersion(QQVersion.最低支持版本);
}
@Override
public boolean initOnce() throws Exception {
// 在这里写功能初始化代码,如果用户启用了这个功能,那么这里的代码将会被执行一次
// initOnce 里可以随便 throw 异常, throw 的异常会显示在日志和故障排除方便定位问题
Class<?> klass = Initiator.loadClass("com.tencent.mobileqq.example.SomeClassManager");
// Initiator.loadClass 用于加载 QQ 的类,如果类不存在,将会 throw ClassNotFoundException
Method someMethod = kTroopInfo.getDeclaredMethod("someMethod", Long.class);
// hookBeforeIfEnabled 只有在用户启用了这个功能的情况下才会执行 hook 回调
// 避免使用 XposedBridge 和 XposedHelpers,因为这些类在 new Xposed API 中不复存在
HookUtils.hookBeforeIfEnabled(this, someMethod, param -> {
// 做一些操作,其中 param 是一个 HookParam 对象,可以通过 param.args 获取方法参数
param.setResult(0L);
});
// return true 表示初始化成功,false 或者抛异常表示初始化失败
return true;
}
}
- 类名、变量名、方法名禁止中文/拼音(例外: 如果 QQ 的 API 本身就是拼音的, 那么就只能用拼音了)
- 简洁明了
- 一个 commit 做一件事情
- 请勿在 commit 附上任何有关 [skip ci] 的字段
- 在 commit 之前请先更新到最新的 main 分支, 以方便我们进行快速合并(fast-forward merge).
- 每个 commit 都必须附着有效的GPG签名,如果您不知道如何使用 GPG 签名,请参阅 这里. 如果你实在不会配置 GPG 签名,你仍然可以提交 PR, 但由于 main 分支要求所有 commit 必须附着有效的 GPG 签名,就只好由我们来代替你签名了.
- 请勿在修改会被编译分发至用户的部分时在 PR 标题添加 [skip ci];请务必在文档、模板等不会影响编译流程和实际分发的目标生成,或完全无法编译但出于必要目的必须提交的 PR 标题添加 [skip ci]
-
请确认您的编辑器支持 EditorConfig,否则请注意您的编码、行位序列等事项。
-
代码风格建议遵循 Google Java Style 中文翻译
-
每位开发者自己的代码风格应保持一致
-
以 UTF-8 编码,以 LF 作为行位序列
-
命名方面: 禁止拼音 (参考上面的例外)
-
使用 4 个空格缩进 (Java/Kotlin/C++)
-
弃用或注释的代码应删除,若需重复使用请翻阅
git log
-
大括号放应同一行上 (Java/Kotlin/C++)
-
代码请务必格式化
-
将自己的代码放在自己的包里,另外,应注意的是,如果你创建了自己的包,一定要记得修改 proguard-rules.pro
-
除例外情况外,原则上要求添加代码头 (例外情况: 反编译的代码、自动生成的代码、非贡献者编写的代码、由于文件格式或其它原因不适合添加代码头等)
如还有疑问,可直接在 Telegram 群聊询问