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

时间:2021-02-16 13:26:30

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


(接下来进一步细化上一篇的修改,前六节请参见:http://blog.csdn.net/aimingoo/article/details/7939116


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, 0x0
    iget-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, 0x1
    iget-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_0
    return-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, 0x7f070020
    invoke-virtual {p0, v0}, Lcom/android/phone/CallCard;->findViewById(I)Landroid/view/View;
    move-result-object v0
    check-cast v0, Landroid/view/ViewGroup;


    const v1, 0x0
    invoke-virtual {v0, v1}, Landroid/view/ViewGroup;->getChildAt(I)Landroid/view/View;
    move-result-object v0


    invoke-virtual {v0}, Landroid/view/View;->getBackground()Landroid/graphics/drawable/Drawable;
    move-result-object v0


    const/16 v1, 0x40
    invoke-virtual {v0, v1}, Landroid/graphics/drawable/Drawable;->setAlpha(I)V


    ## for phoneMsgContainer
    const v0, 0x7f070028
    invoke-virtual {p0, v0}, Lcom/android/phone/CallCard;->findViewById(I)Landroid/view/View;
    move-result-object v0


    invoke-virtual {v0}, Landroid/view/View;->getBackground()Landroid/graphics/drawable/Drawable;
    move-result-object v0
    invoke-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 @ mInCallControls
    iget-object v1, p0, Lcom/android/phone/InCallTouchUi;->mInCallControls:Landroid/view/View;
    const v2, 0x7f070074
    invoke-virtual {v1, v2}, Landroid/view/View;->findViewById(I)Landroid/view/View;
    move-result-object v0


    const/16 v1, 0x40
    invoke-virtual {v0}, Landroid/view/View;->getBackground()Landroid/graphics/drawable/Drawable;
    move-result-object v0
    invoke-virtual {v0, v1}, Landroid/graphics/drawable/Drawable;->setAlpha(I)V




    ## for endButton @ mEndButton
    iget-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 v0
    const/16 v1, 0x60
    invoke-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.0F
    const/high16 v0, 0x4228


    ## CallCard.pA == mName
    iget-object v1, p0, Lcom/android/phone/CallCard;->pA:Landroid/widget/TextView;
    invoke-virtual {v1, v0}, Landroid/widget/TextView;->setTextSize(F)V


    ## 28.0F
    const/high16 v0, 0x41b8


    ## CallCard.pC == mPhoneNumber
    iget-object v1, p0, Lcom/android/phone/CallCard;->pC:Landroid/widget/TextView;
    invoke-virtual {v1, v0}, Landroid/widget/TextView;->setTextSize(F)V


    ## CallCard.pB == mLocation
    iget-object v1, p0, Lcom/android/phone/CallCard;->pB:Landroid/widget/TextView;
    invoke-virtual {v1, v0}, Landroid/widget/TextView;->setTextSize(F)V


    ## bacground
    const v0, 0x80CCCCCC
    invoke-virtual {v1}, Landroid/widget/TextView;->getParent()Landroid/view/ViewParent;
    move-result-object v1
    check-cast v1, Landroid/view/ViewGroup;
    invoke-virtual {v1, v0}, Landroid/view/ViewGroup;->setBackgroundColor(I)V


    :goto_0
    return-void


    :cond_0
    ## 25.0F
    const/high16 v0, 0x41c8


    ## CallCard.pA == mName
    iget-object v1, p0, Lcom/android/phone/CallCard;->pA:Landroid/widget/TextView;
    invoke-virtual {v1, v0}, Landroid/widget/TextView;->setTextSize(F)V


    ## 18.0F
    const/high16 v0, 0x4190


    ## CallCard.pC == mPhoneNumber
    iget-object v1, p0, Lcom/android/phone/CallCard;->pC:Landroid/widget/TextView;
    invoke-virtual {v1, v0}, Landroid/widget/TextView;->setTextSize(F)V


    ## CallCard.pB == mLocation
    iget-object v1, p0, Lcom/android/phone/CallCard;->pB:Landroid/widget/TextView;
    invoke-virtual {v1, v0}, Landroid/widget/TextView;->setTextSize(F)V


    ## bacground
    const v0, 0x0
    invoke-virtual {v1}, Landroid/widget/TextView;->getParent()Landroid/view/ViewParent;
    move-result-object v1
    check-cast v1, Landroid/view/ViewGroup;
    invoke-virtual {v1, v0}, Landroid/view/ViewGroup;->setBackgroundResource(I)V


    goto :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 v2
    const/16 v1, 0x400
    invoke-virtual {v2, v1, v1}, Landroid/view/Window;->setFlags(II)V
## fix end.


  ……
    const v1, 0x7f030012
    invoke-virtual {p0, v1}, Lcom/android/phone/InCallScreen;->setContentView(I)V



7) 好象你忘了说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;

然后重编译它,这样在用它设置大头贴时,我们可以*选取图片大小。当然,为了得到“正好是一个全屏大小”的大头贴,我们也可以借助一下工具。这里强烈推荐“快图浏览”,它在截取时可以按大小(像素数)和长宽比来设置截取框。如果你按大小来设置,比如480x800的屏幕大小,那么无论你截选图片多大,最终都会等比缩放到这个大小——相当好用!


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:写程序嘛,不过是汇编嘛,不怕不怕啦!