Skip to content

Commit eff3c14

Browse files
committed
WIP: Support for Qihoo browser
Current support for Qihoo browser isn't perfect. For unkown reason, JavaScript codes might be evaluated in different contexts of WebView so that global objects aren't shared. However, this doesn't happen if we have a clean installation and try it for the first time. Since it is impossible to debug this browser, we simply choose not to solve this problem in the near future. Also, we fix version text in logs.
1 parent 45dbb14 commit eff3c14

File tree

4 files changed

+85
-9
lines changed

4 files changed

+85
-9
lines changed

app/src/main/java/org/matrix/chromext/Chrome.kt

+4-1
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@ object Chrome {
3838
var isDev = false
3939
var isEdge = false
4040
var isMi = false
41+
var isQihoo = false
4142
var isSamsung = false
4243
var isVivaldi = false
4344
var isCocCoc = false
@@ -60,11 +61,13 @@ object Chrome {
6061
isDev = packageName.endsWith("canary") || packageName.endsWith("dev")
6162
isEdge = packageName.startsWith("com.microsoft.emmx")
6263
isMi = packageName == "com.mi.globalbrowser" || packageName == "com.android.browser"
64+
isQihoo = packageName == "com.qihoo.contents"
6365
isSamsung = packageName.startsWith("com.sec.android.app.sbrowser")
6466
isVivaldi = packageName == "com.vivaldi.browser"
6567
@Suppress("DEPRECATION") val packageInfo = ctx.packageManager?.getPackageInfo(packageName, 0)
6668
version = packageInfo?.versionName
67-
Log.i("Package: ${packageName}, v${version}")
69+
version = (if (version?.startsWith("v") == true) "" else "v") + version
70+
Log.i("Package: ${packageName}, ${version}")
6871

6972
setupHttpCache(ctx)
7073
saveRedirectCookie()

app/src/main/java/org/matrix/chromext/MainHook.kt

+12-6
Original file line numberDiff line numberDiff line change
@@ -49,9 +49,7 @@ val supportedPackages =
4949
class MainHook : IXposedHookLoadPackage, IXposedHookZygoteInit {
5050
override fun handleLoadPackage(lpparam: XC_LoadPackage.LoadPackageParam) {
5151
Log.d(lpparam.processName + " started")
52-
if (lpparam.packageName == "org.matrix.chromext") {
53-
return
54-
}
52+
if (lpparam.packageName == "org.matrix.chromext") return
5553
if (supportedPackages.contains(lpparam.packageName)) {
5654
lpparam.classLoader
5755
.loadClass("org.chromium.ui.base.WindowAndroid")
@@ -75,12 +73,12 @@ class MainHook : IXposedHookLoadPackage, IXposedHookZygoteInit {
7573
Chrome.isMi =
7674
lpparam.packageName == "com.mi.globalbrowser" ||
7775
lpparam.packageName == "com.android.browser"
76+
Chrome.isQihoo = lpparam.packageName == "com.qihoo.contents"
77+
7878
if (ctx == null && Chrome.isMi) return
7979
// Wait to get the browser context of Mi Browser
8080

81-
if (ctx != null && lpparam.packageName != "android") {
82-
Chrome.init(ctx, ctx.packageName)
83-
}
81+
if (ctx != null && lpparam.packageName != "android") Chrome.init(ctx, ctx.packageName)
8482

8583
if (Chrome.isMi) {
8684
WebViewHook.WebView = Chrome.load("com.miui.webkit.WebView")
@@ -90,6 +88,14 @@ class MainHook : IXposedHookLoadPackage, IXposedHookZygoteInit {
9088
return
9189
}
9290

91+
if (Chrome.isQihoo) {
92+
WebViewHook.WebView = Chrome.load("com.qihoo.webkit.WebView")
93+
WebViewHook.ViewClient = Chrome.load("com.qihoo.webkit.WebViewClient")
94+
WebViewHook.ChromeClient = Chrome.load("com.qihoo.webkit.WebChromeClient")
95+
hookWebView()
96+
return
97+
}
98+
9399
WebViewClient::class.java.declaredConstructors[0].hookAfter {
94100
if (it.thisObject::class != WebViewClient::class) {
95101
WebViewHook.ViewClient = it.thisObject::class.java

app/src/main/java/org/matrix/chromext/hook/ContextMenu.kt

+57
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@ import android.view.View
88
import android.view.View.OnClickListener
99
import android.view.ViewGroup
1010
import android.widget.ImageView
11+
import android.widget.PopupWindow
12+
import android.widget.TextView
1113
import de.robv.android.xposed.XC_MethodHook.Unhook
1214
import java.lang.Class
1315
import org.matrix.chromext.Chrome
@@ -20,6 +22,7 @@ import org.matrix.chromext.utils.*
2022
object ContextMenuHook : BaseHook() {
2123

2224
val erudaMenuId = 31415926
25+
private var textView: TextView? = null
2326

2427
private fun openEruda(url: String) {
2528
if (WebViewHook.isInit) {
@@ -50,6 +53,7 @@ object ContextMenuHook : BaseHook() {
5053
}
5154

5255
private var actionModeFinder: Unhook? = null
56+
private var popupWindowFinder: Unhook? = null
5357

5458
private fun hookActionMode(cls: Class<*>) {
5559
actionModeFinder?.unhook()
@@ -73,9 +77,62 @@ object ContextMenuHook : BaseHook() {
7377
}
7478
}
7579

80+
private fun hookPopupWindow(cls: Class<*>) {
81+
popupWindowFinder?.unhook()
82+
cls.declaredConstructors.first().hookAfter {
83+
val ctx = it.args[0] as Context
84+
Resource.enrich(ctx)
85+
val popupWindow = it.thisObject as PopupWindow
86+
val view = popupWindow.contentView
87+
if (!(view is ViewGroup && view.getChildAt(0) is TextView)) return@hookAfter
88+
val sampleView = view.getChildAt(0) as TextView
89+
textView = TextView(ctx)
90+
textView!!.setHorizontallyScrolling(true)
91+
textView!!.setSingleLine(true)
92+
textView!!.ellipsize = sampleView.ellipsize
93+
textView!!.gravity = sampleView.gravity
94+
textView!!.transformationMethod = sampleView.transformationMethod
95+
textView!!.layoutParams = sampleView.layoutParams
96+
textView!!.typeface = sampleView.typeface
97+
textView!!.setOnClickListener {
98+
openEruda(Chrome.getUrl()!!)
99+
popupWindow.dismiss()
100+
}
101+
view.addView(textView!!, view.childCount)
102+
}
103+
}
104+
76105
override fun init() {
77106
if (Chrome.isSamsung) {
78107
hookActionMode(Chrome.load("com.sec.terrace.content.browser.TinActionModeCallback"))
108+
} else if (Chrome.isQihoo) {
109+
val WebViewExtensionClient = Chrome.load("com.qihoo.webkit.extension.WebViewExtensionClient")
110+
popupWindowFinder =
111+
WebViewExtensionClient.declaredConstructors.first().hookAfter {
112+
val selectionMenuWrapper = it.thisObject
113+
val showSelectionMenu =
114+
findMethodOrNull(selectionMenuWrapper::class.java) { name == "showSelectionMenu" }
115+
if (showSelectionMenu == null) return@hookAfter
116+
val selectionMenu =
117+
selectionMenuWrapper::class.java.declaredFields.first().get(selectionMenuWrapper)
118+
val horizontalCustomPopupDialog =
119+
selectionMenu::class
120+
.java
121+
.declaredFields
122+
.find { it.type.superclass == PopupWindow::class.java }!!
123+
.type
124+
hookPopupWindow(horizontalCustomPopupDialog)
125+
showSelectionMenu.hookAfter {
126+
val view = it.args[0]
127+
if (WebViewHook.WebView!!.isAssignableFrom(view::class.java)) Chrome.updateTab(view)
128+
val url = Chrome.getUrl()!!
129+
val titleId =
130+
if (isChromeXtFrontEnd(url)) R.string.main_menu_developer_tools
131+
else if (isUserScript(url)) R.string.main_menu_install_script
132+
else R.string.main_menu_eruda_console
133+
textView?.setText(titleId)
134+
}
135+
}
79136
} else if (Chrome.isMi) {
80137
val miuiFloatingSelectPopupWindow =
81138
Chrome.load("com.miui.org.chromium.content.browser.miui.MiuiFloatingSelectPopupWindow")

app/src/main/java/org/matrix/chromext/hook/WebView.kt

+12-2
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ import org.matrix.chromext.Listener
99
import org.matrix.chromext.script.Local
1010
import org.matrix.chromext.script.ScriptDbManager
1111
import org.matrix.chromext.utils.Log
12+
import org.matrix.chromext.utils.findField
1213
import org.matrix.chromext.utils.findMethod
1314
import org.matrix.chromext.utils.hookAfter
1415
import org.matrix.chromext.utils.hookBefore
@@ -53,7 +54,13 @@ object WebViewHook : BaseHook() {
5354
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
5455
records
5556
.find {
56-
it.get()?.invokeMethod { name == "getWebChromeClient" } == chromeClient
57+
if (Chrome.isQihoo) {
58+
val mProvider = findField(WebView!!) { name == "mProvider" }
59+
mProvider.get(it.get())
60+
} else {
61+
it.get()
62+
}
63+
?.invokeMethod { name == "getWebChromeClient" } == chromeClient
5764
}
5865
?.get()
5966
} else Chrome.getTab()
@@ -81,7 +88,10 @@ object WebViewHook : BaseHook() {
8188

8289
findMethod(ViewClient!!, true) { name == "onPageStarted" }
8390
// public void onPageStarted (WebView view, String url, Bitmap favicon)
84-
.hookAfter { onUpdateUrl(it.args[1] as String, it.args[0]) }
91+
.hookAfter {
92+
if (Chrome.isQihoo && it.thisObject::class.java.declaredMethods.size > 1) return@hookAfter
93+
onUpdateUrl(it.args[1] as String, it.args[0])
94+
}
8595

8696
findMethod(Activity::class.java) { name == "onStop" }
8797
.hookBefore { ScriptDbManager.updateScriptStorage() }

0 commit comments

Comments
 (0)