Android玩乐系列:修改汇编代码支持原生高清来电大头贴(三)

本文分三篇。本篇介绍更复杂的定制过程。

(接下来进一步细化上一篇的修改,前六节请参见:)

7、一些遗留问题的修改=======上面是最简版本,只是为了突出核心功能的实现,但实际留下的问题还是不少的。下面一一道来。1) 呀。是成功了呢,不过一会儿就被改回来了。———

[修改, 2012.09.04。方法名写错的,应该是updateInCallBackground(),而不是updateScreen()]这并不是普遍性的问题,有些拔号面板是有背景的,有些则是背景透明而直接显示桌面的。对于有背景的拔号面板,Phone.apk通常会在InCallScreen.smali中有一个updateInCallBackground()方法,找到它的几处调用,注释掉即可。或者干脆把updateInCallBackground()改成空函数就好了。如果找不到updateInCallBackground()函数,则尝试找一个setBackgroundResource()这个方法,看哪里重绘了mMainFrame的背景即可。2) 下一次电话呼入的时候,会残留上一个电话使用的大头贴———是的。这应该在此次电话结束时清理掉。这很简单,修改InCallScreen.smali,找到

.method private delayedCleanupAfterDisconnect()V这个方法。然后找到其中return-void这行代码,往上数几行找个地方插入如下代码即可: …## fixed by aimingoo## 重置背景const/4 v0, 0x0iget-object v1, p0, Lcom/android/phone/InCallScreen;->mMainFrame:Landroid/view/ViewGroup;invoke-virtual {v1, v0}, Landroid/view/ViewGroup;->setBackgroundResource(I)V## 这里参见本遗留问题第5项有关setPersonInfoStyle()的说明const/4 v0, 0x1iget-object v1, p0, Lcom/android/phone/InCallScreen;->mCallCard:Lcom/android/phone/CallCard;invoke-virtual {v1, v0}, Lcom/android/phone/CallCard;->setPersonInfoStyle(Z)V##end fix.:cond_3:goto_0return-void## <<- 注意从这行代码往上找…3) 好象原来的头像还是会闪一下?———原来的头像是这样的一个获得过程: – 首先开始接听或拔打电话 – 拔号程序显示面板,面板中头像位置显示为“无头像”的icon – 异步发起调用,从联系人数据库中读取头像 – 当上述异步调用返回时,更新显示上面的“无头像”icon为真正的头像图片所以事实上原生的应用在“显示头像”时都会是两步,这是为了更快地绘制出拨号面板以便用户操作。而们的代码事实上也是依赖这个原理,在上述的过程异步得到“高清大头贴”的数据之后,显示在背景上的。那么总的来说,事实上头像总会闪一下。并且在异步读取到头像之前,原生界面上就是会显示一个“无头像”的icon。尽管这个过程通常很短,多数时候在你抓过来电话之前就已经闪过去了,但是对于那些正盯着电话看效果的玩实来说,这还是不爽的。基本上来说,可以注释掉所有修改mPhoto的地方。如果必要,保留一些用于在没有大头贴的情况下显示原有的mPhoto的代码是可以的。作为一个示例,一个简单的地方就是修改showCachedImage()。这首先在CallCard.smail中找到showCachedImage(),然后注释掉其中下面的两行:## 不必在得到头像时更新cardCard中的头像#.method private static final showCachedImage(Landroid/widget/ImageView;Lcom/android/internal/telephony/CallerInfo;)Z…## iget-object v0, p1, Lcom/android/internal/telephony/CallerInfo;->cachedPhoto:Landroid/graphics/drawable/Drawable;#### invoke-static {p0, v0}, Lcom/android/phone/CallCard;->showImage(Landroid/widget/ImageView;Landroid/graphics/drawable/Drawable;)V…这里直接修改showCachedImage(),是因为Phone.apk只为mPhoto成员调用showCachedImage(),其它的会直接调用showImage()。4) 界面上大头贴显示不全,被一些元素遮住了。要是它们有透明度就好了。———有些时候,界面上的元素是通过贴图来绘制的,也就是在资源文件中,它的背景是一张图片。对于指定颜色的背景,例如#xxRRGGBB,我们可以在资源文件中通过指定xx值来使它透明化。但如果背景是图,那么在较低的android版本的资源文件中又不支持alpha属性,那么就只能在源代码中通过setAlpha()来使之透明了。后面这种情况(也包括前面这种设置color代码的情况)可以在CallCard.smali与InCallTouchUi.smali中添加代码来实现,某些情况下,你也可能要改到InCallScreen.smali中的代码的。但总的来说,都与具体的Phone.apk有关。下面是我在修改Mokee的Phone.apk中使用的代码。注意,这些代码都应该写在onFinishInflate()方法里,这里刚好初始化完界面,并将界面元素关联到Java对象的成员上。#———# 在完成初始化后,处理一些背景# – CallCard.smali#———.method protected onFinishInflate()V ……iput-object v0, p0, Lcom/android/phone/CallCard;->mPrimaryCallInfo:Landroid/view/ViewGroup;## fixed by aimingoo## for callCardPersonInfo.clild(0)const v0, 0x7f070020invoke-virtual {p0, v0}, Lcom/android/phone/CallCard;->findViewById(I)Landroid/view/View;move-result-object v0check-cast v0, Landroid/view/ViewGroup;const v1, 0x0invoke-virtual {v0, v1}, Landroid/view/ViewGroup;->getChildAt(I)Landroid/view/View;move-result-object v0invoke-virtual {v0}, Landroid/view/View;->getBackground()Landroid/graphics/drawable/Drawable;move-result-object v0const/16 v1, 0x40invoke-virtual {v0, v1}, Landroid/graphics/drawable/Drawable;->setAlpha(I)V## for phoneMsgContainerconst v0, 0x7f070028invoke-virtual {p0, v0}, Lcom/android/phone/CallCard;->findViewById(I)Landroid/view/View;move-result-object v0invoke-virtual {v0}, Landroid/view/View;->getBackground()Landroid/graphics/drawable/Drawable;move-result-object v0invoke-virtual {v0, v1}, Landroid/graphics/drawable/Drawable;->setAlpha(I)V## end fix.#———# 在完成初始化后,处理一些背景# – InCallTouchUi.smali#———.method protected onFinishInflate()V ……iput-object v1, p0, Lcom/android/phone/InCallTouchUi;->stop_layout:Landroid/widget/LinearLayout;## fixed by aimingoo## for bottomButtons @ mInCallControlsiget-object v1, p0, Lcom/android/phone/InCallTouchUi;->mInCallControls:Landroid/view/View;const v2, 0x7f070074invoke-virtual {v1, v2}, Landroid/view/View;->findViewById(I)Landroid/view/View;move-result-object v0const/16 v1, 0x40invoke-virtual {v0}, Landroid/view/View;->getBackground()Landroid/graphics/drawable/Drawable;move-result-object v0invoke-virtual {v0, v1}, Landroid/graphics/drawable/Drawable;->setAlpha(I)V## for endButton @ mEndButtoniget-object v1, p0, Lcom/android/phone/InCallTouchUi;->mEndButton:Landroid/widget/Button;invoke-virtual {v1}, Landroid/widget/Button;->getBackground()Landroid/graphics/drawable/Drawable;move-result-object v0const/16 v1, 0x60invoke-virtual {v0, v1}, Landroid/graphics/drawable/Drawable;->setAlpha(I)V## end fix.5) 上面那个mPhoto的确不显示了,但好象还占着位置,还是很难看。【话说,真的有必要通过汇编代码来调样式哇?GG,你直接改资源文件不好哇?】———如果我们真的要实现: – 有大头贴时,不显示小小的头像mPhoto – 没有大头贴时,显示一下“无头像”icon,或者 – 因为头像图片不够大,所以某些时候还是显示图片到头像mPhoto中间去更好看事实上,前面showCachedBackground()的实现代码中,还确实检查了头像图片的大小,当它长宽之一小于240px,我们就不作为全屏大头贴来显示了。所以,我们的确还是要将mPhoto处理成:有大头贴时隐藏,否则在必要时还得显示。这个,改资源文件还真不成。还得动代码。上面我们在showCachedBackground()中留下了一个setPersonInfoStyle()没做说明。那个方法,其实就是留给这里用的。传入参数toDefault。当toDefault为false时,就显示我们定制的大头贴界面,否则就切回原生界面(就是小头像)来显示。这个方法就与具体的Phone.apk有关了,因为每个Phone.apk的来电面板界面都不一样,显示哪些,不显示哪些,,其实都要靠程序员分析着资源文件一点点来改。尽管麻烦,但效果也确实惊人。下面是我为Lezo界面写的一个setPersonInfoStyle()方法:.method public setPersonInfoStyle(Z)V.locals 2.parameter "toDefault"if-nez p1, :cond_0## 42.0Fconst/high16 v0, 0x4228## CallCard.pA == mNameiget-object v1, p0, Lcom/android/phone/CallCard;->pA:Landroid/widget/TextView;invoke-virtual {v1, v0}, Landroid/widget/TextView;->setTextSize(F)V## 28.0Fconst/high16 v0, 0x41b8## CallCard.pC == mPhoneNumberiget-object v1, p0, Lcom/android/phone/CallCard;->pC:Landroid/widget/TextView;invoke-virtual {v1, v0}, Landroid/widget/TextView;->setTextSize(F)V## CallCard.pB == mLocationiget-object v1, p0, Lcom/android/phone/CallCard;->pB:Landroid/widget/TextView;invoke-virtual {v1, v0}, Landroid/widget/TextView;->setTextSize(F)V## bacgroundconst v0, 0x80CCCCCCinvoke-virtual {v1}, Landroid/widget/TextView;->getParent()Landroid/view/ViewParent;move-result-object v1check-cast v1, Landroid/view/ViewGroup;invoke-virtual {v1, v0}, Landroid/view/ViewGroup;->setBackgroundColor(I)V:goto_0return-void:cond_0## 25.0Fconst/high16 v0, 0x41c8## CallCard.pA == mNameiget-object v1, p0, Lcom/android/phone/CallCard;->pA:Landroid/widget/TextView;invoke-virtual {v1, v0}, Landroid/widget/TextView;->setTextSize(F)V## 18.0Fconst/high16 v0, 0x4190## CallCard.pC == mPhoneNumberiget-object v1, p0, Lcom/android/phone/CallCard;->pC:Landroid/widget/TextView;invoke-virtual {v1, v0}, Landroid/widget/TextView;->setTextSize(F)V## CallCard.pB == mLocationiget-object v1, p0, Lcom/android/phone/CallCard;->pB:Landroid/widget/TextView;invoke-virtual {v1, v0}, Landroid/widget/TextView;->setTextSize(F)V## bacgroundconst v0, 0x0invoke-virtual {v1}, Landroid/widget/TextView;->getParent()Landroid/view/ViewParent;move-result-object v1check-cast v1, Landroid/view/ViewGroup;invoke-virtual {v1, v0}, Landroid/view/ViewGroup;->setBackgroundResource(I)Vgoto :goto_0.end method这个setPersonInfoStyle()函数会在showCachedBackground()中调用并传入false值,另外也应该在InCallScreen.smali的delayedCleanupAfterDisconnect()方法中调用。后一种情况应传入true值,以使得“下一次”来电面板将以缺省形式打开。6) 全屏!!要真的全屏!!!———其实大多数拨号面板是并不支持“全屏”的,它通常会留下状态栏。既然我们这里说的是“全屏来电大头贴”,那么就加上下面这段代码好了:#———# 使拔号面板全屏# – in InCallScreen.smali# – 修改代码必须位于InCallScreen;->setContentView()调用之前!!!#———.method protected onCreate(Landroid/os/Bundle;)V ……## fixed by aimingoo.invoke-virtual {p0}, Lcom/android/phone/InCallScreen;->getWindow()Landroid/view/Window;move-result-object v2const/16 v1, 0x400invoke-virtual {v2, v1, v1}, Landroid/view/Window;->setFlags(II)V## fix end.……const v1, 0x7f030012invoke-virtual {p0, v1}, Lcom/android/phone/InCallScreen;->setContentView(I)V7) 好象你忘了说HD Contact Photos怎么改了!———嗯嗯。是的是的,不好意思。补过。其实很简单。反编译它,然后找到smali\com\jgarrison\hdcontacts\NewEntry.smali这个文件。将下面的代码注释掉,就可以了:##———## 注释掉下面的代码,使打开图片选取时显示一个“自由的”截图框## – .line xxx这样的代码可能与具体的反编译有关,不必在意。##———## .line 406## const-string v12, "outputX"#### const/16 v13, 0x100#### invoke-virtual {v8, v12, v13}, Landroid/content/Intent;->putExtra(Ljava/lang/String;I)Landroid/content/Intent;#### .line 407## const-string v12, "outputY"#### const/16 v13, 0x100#### invoke-virtual {v8, v12, v13}, Landroid/content/Intent;->putExtra(Ljava/lang/String;I)Landroid/content/Intent;#### .line 408## const-string v12, "aspectX"#### const/4 v13, 0x1#### invoke-virtual {v8, v12, v13}, Landroid/content/Intent;->putExtra(Ljava/lang/String;I)Landroid/content/Intent;#### .line 409## const-string v12, "aspectY"#### const/4 v13, 0x1#### invoke-virtual {v8, v12, v13}, Landroid/content/Intent;->putExtra(Ljava/lang/String;I)Landroid/content/Intent;然后重编译它,这样在用它设置大头贴时,我们可以自由选取图片大小。当然,为了得到“正好是一个全屏大小”的大头贴,我们也可以借助一下工具。这里强烈推荐“快图浏览”,它在截取时可以按大小(像素数)和长宽比来设置截取框。如果你按大小来设置,比如480×800的屏幕大小,那么无论你截选图片多大,最终都会等比缩放到这个大小——相当好用!n) 其它之其它———!强调!!!1:永远记住:插入代码的时候,要确认你在使用着合适的寄存器!2:不同的Phone.apk是不一样的,上面的代码主要基于CyanogenMod及其衍生版的ROM,大致在它们之间都是可以通用的。但要注意细节上的差异,尤其(再次强调)寄存器在反汇编代码中是可能不同的!3:非常多的ROM衍生自CyanogenMod,包括Lewa、Lezo、DianxinOS、Mokee、Shendu、Norma、Joyos,以及部分Miui的定制版。4:不同版本ROM中的Phone.apk,多数都是不能换在其它ROM中用的。主要的原因之一,是Phone.apk依赖framework-res.apk中的资源来实现了锁屏状态下的接听面板(TouchUi),而不用ROM的framework-res差异较大。另外,也可能是它们用到的TelephonyProvider.apk版本不一致,试试换个看,试试手气呵。5:一定要用platform.*的两个key来签名Phone.apk,它要求必须是这个权限的签名。6:没必要去尝试改原厂的Phone.apk,例如sesen原生界面的。因为你拿不到他们私有的platform keys。于是你签不了名,于是你改了也放不到原生ROM中去。用到别的ROM?你忘了,framework-res还不一样呢。7:写程序嘛,不过是汇编嘛,不怕不怕啦!

人生就像爬坡,要一步一步来。

Android玩乐系列:修改汇编代码支持原生高清来电大头贴(三)

相关文章:

你感兴趣的文章:

标签云: