feat(kakaotalk): add layout handling for deleted and hidden messages in chat log

This is the fourth commit to make deleted, hidden messages visible.
This commit is contained in:
2025-05-28 01:29:08 +09:00
parent 1e91a5cb52
commit 11bf82640d

View File

@ -3,10 +3,15 @@ package app.revanced.patches.kakaotalk.chatlog
import app.revanced.patcher.extensions.InstructionExtensions.addInstructions
import app.revanced.patcher.extensions.InstructionExtensions.addInstructionsWithLabels
import app.revanced.patcher.extensions.InstructionExtensions.getInstruction
import app.revanced.patcher.extensions.InstructionExtensions.instructions
import app.revanced.patcher.patch.bytecodePatch
import app.revanced.patcher.util.proxy.mutableTypes.MutableField.Companion.toMutable
import app.revanced.patcher.util.smali.ExternalLabel
import app.revanced.util.getReference
import com.android.tools.smali.dexlib2.AccessFlags
import com.android.tools.smali.dexlib2.Opcode
import com.android.tools.smali.dexlib2.builder.instruction.BuilderInstruction35c
import com.android.tools.smali.dexlib2.iface.reference.FieldReference
import com.android.tools.smali.dexlib2.immutable.ImmutableField
import com.android.tools.smali.dexlib2.immutable.value.ImmutableBooleanEncodedValue
@ -23,6 +28,9 @@ val addDeletedOrHiddenLayoutPatch = bytecodePatch(
val makeLayoutMethod = chatInfoViewClass.methods.firstOrNull {
it.name == "makeLayout"
} ?: error("makeLayout method not found in ChatInfoView class")
val onDrawMethod = chatInfoViewClass.methods.firstOrNull {
it.name == "onDraw"
} ?: error("onDraw method not found in ChatInfoView class")
println("ChatInfoView class: $chatInfoViewClass")
println("makeLayout method: $makeLayoutMethod")
@ -172,48 +180,160 @@ val addDeletedOrHiddenLayoutPatch = bytecodePatch(
).toMutable()
)
// TODO: Add logic to initialize these fields in the makeLayout method
val cls = chatInfoViewClass.type
makeLayoutMethod.addInstructionsWithLabels(
val deletedMakeLayoutBlock = """
iget-boolean v0, p0, $cls->isDeleted:Z
if-eqz v0, :skip_deleted
const-string v0, "[Deleted]"
iget-object v1, p0, $cls->deletedPaint:Landroid/text/TextPaint;
invoke-static {v0, v1}, Landroid/text/BoringLayout;->isBoring(Ljava/lang/CharSequence;Landroid/text/TextPaint;)Landroid/text/BoringLayout${'$'}Metrics;
move-result-object v8
if-nez v8, :boring_deleted
new-instance v1, Landroid/text/StaticLayout;
iget-object v4, p0, $cls->deletedPaint:Landroid/text/TextPaint;
invoke-virtual {v4, v0}, Landroid/graphics/Paint;->measureText(Ljava/lang/String;)F
move-result v2
float-to-int v5, v2
sget-object v6, Landroid/text/Layout${'$'}Alignment;->ALIGN_NORMAL:Landroid/text/Layout${'$'}Alignment;
const/4 v9, 0x1
const/high16 v7, 0x3f800000 # 1.0f
const/4 v8, 0x0
move-object v2, v1
move-object v3, v0
invoke-direct/range {v2 .. v9}, Landroid/text/StaticLayout;-><init>(Ljava/lang/CharSequence;Landroid/text/TextPaint;ILandroid/text/Layout${'$'}Alignment;FFZ)V
goto :set_deleted
:boring_deleted
iget-object v3, p0, $cls->deletedPaint:Landroid/text/TextPaint;
invoke-virtual {v3, v0}, Landroid/graphics/Paint;->measureText(Ljava/lang/String;)F
move-result v1
float-to-int v4, v1
sget-object v5, Landroid/text/Layout${'$'}Alignment;->ALIGN_NORMAL:Landroid/text/Layout${'$'}Alignment;
const/4 v9, 0x0
const/high16 v6, 0x3f800000 # 1.0f
const/4 v7, 0x0
move-object v2, v0
invoke-static/range {v2 .. v9}, Landroid/text/BoringLayout;->make(Ljava/lang/CharSequence;Landroid/text/TextPaint;ILandroid/text/Layout${'$'}Alignment;FFLandroid/text/BoringLayout${'$'}Metrics;Z)Landroid/text/BoringLayout;
move-result-object v1
:set_deleted
iput-object v1, p0, $cls->deletedLayout:Landroid/text/Layout;
:skip_deleted
""".trimIndent()
val hiddenMakeLayoutBlock = """
iget-boolean v0, p0, $cls->isHidden:Z
if-eqz v0, :skip_hidden
const-string v0, "[Hidden]"
iget-object v1, p0, $cls->hiddenPaint:Landroid/text/TextPaint;
invoke-static {v0, v1}, Landroid/text/BoringLayout;->isBoring(Ljava/lang/CharSequence;Landroid/text/TextPaint;)Landroid/text/BoringLayout${'$'}Metrics;
move-result-object v8
if-nez v8, :boring_hidden
new-instance v1, Landroid/text/StaticLayout;
iget-object v4, p0, $cls->hiddenPaint:Landroid/text/TextPaint;
invoke-virtual {v4, v0}, Landroid/graphics/Paint;->measureText(Ljava/lang/String;)F
move-result v2
float-to-int v5, v2
sget-object v6, Landroid/text/Layout${'$'}Alignment;->ALIGN_NORMAL:Landroid/text/Layout${'$'}Alignment;
const/4 v9, 0x1
const/high16 v7, 0x3f800000 # 1.0f
const/4 v8, 0x0
move-object v2, v1
move-object v3, v0
invoke-direct/range {v2 .. v9}, Landroid/text/StaticLayout;-><init>(Ljava/lang/CharSequence;Landroid/text/TextPaint;ILandroid/text/Layout${'$'}Alignment;FFZ)V
goto :set_hidden
:boring_hidden
iget-object v3, p0, $cls->hiddenPaint:Landroid/text/TextPaint;
invoke-virtual {v3, v0}, Landroid/graphics/Paint;->measureText(Ljava/lang/String;)F
move-result v1
float-to-int v4, v1
sget-object v5, Landroid/text/Layout${'$'}Alignment;->ALIGN_NORMAL:Landroid/text/Layout${'$'}Alignment;
const/4 v9, 0x0
const/high16 v6, 0x3f800000 # 1.0f
const/4 v7, 0x0
move-object v2, v0
invoke-static/range {v2 .. v9}, Landroid/text/BoringLayout;->make(Ljava/lang/CharSequence;Landroid/text/TextPaint;ILandroid/text/Layout${'$'}Alignment;FFLandroid/text/BoringLayout${'$'}Metrics;Z)Landroid/text/BoringLayout;
move-result-object v1
:set_hidden
iput-object v1, p0, $cls->hiddenLayout:Landroid/text/Layout;
:skip_hidden
""".trimIndent()
makeLayoutMethod.addInstructions(
0,
"""
iget-boolean v0, p0, Lcom/kakao/talk/widget/chatlog/ChatInfoView;->isDeleted:Z
if-eqz v0, :not_deleted
const-string v0, "[Deleted]"
iget-object v1, p0, Lcom/kakao/talk/widget/chatlog/ChatInfoView;->deletedPaint:Landroid/text/TextPaint;
invoke-static {v0, v1}, Landroid/text/BoringLayout;->isBoring(Ljava/lang/CharSequence;Landroid/text/TextPaint;)Landroid/text/BoringLayout${'$'}Metrics;
move-result-object v8
if-nez v8, :boring_deleted
new-instance v1, Landroid/text/StaticLayout;
iget-object v4, p0, Lcom/kakao/talk/widget/chatlog/ChatInfoView;->deletedPaint:Landroid/text/TextPaint;
invoke-virtual { v4, v0 }, Landroid/graphics/Paint;->measureText(Ljava/lang/String;)F
move-result v2
float-to-int v5, v2
sget-object v6, Landroid/text/Layout${'$'}Alignment;->ALIGN_NORMAL:Landroid/text/Layout${'$'}Alignment;
const/4 v9, 0x1
const/high16 v7, 0x3f800000 # 1.0f
const/4 v8, 0x0
move-object v2, v1
move-object v3, v0
invoke-direct/range {v2 .. v9}, Landroid/text/StaticLayout;-><init>(Ljava/lang/CharSequence;Landroid/text/TextPaint;ILandroid/text/Layout${'$'}Alignment;FFZ)V
goto :set_deleted_layout
:boring_deleted
iget-object v3, p0, Lcom/kakao/talk/widget/chatlog/ChatInfoView;->deletedPaint:Landroid/text/TextPaint;
invoke-virtual {v3, v0}, Landroid/graphics/Paint;->measureText(Ljava/lang/String;)F
move-result v1
float-to-int v4, v1
sget-object v5, Landroid/text/Layout${'$'}Alignment;->ALIGN_NORMAL:Landroid/text/Layout${'$'}Alignment;
const/4 v9, 0x0
const/high16 v6, 0x3f800000 # 1.0f
const/4 v7, 0x0
move-object v2, v0
invoke-static/range {v2 .. v9}, Landroid/text/BoringLayout;->make(Ljava/lang/CharSequence;Landroid/text/TextPaint;ILandroid/text/Layout${'$'}Alignment;FFLandroid/text/BoringLayout${'$'}Metrics;Z)Landroid/text/BoringLayout;
move-result-object v1
:set_deleted_layout
iput-object v1, p0, Lcom/kakao/talk/widget/chatlog/ChatInfoView;->deletedLayout:Landroid/text/Layout;
""".trimIndent(),
ExternalLabel("not_deleted", makeLayoutMethod.getInstruction(0))
deletedMakeLayoutBlock + "\n\n" + hiddenMakeLayoutBlock,
)
val superIndex = onDrawMethod.instructions.indexOfFirst {
it.opcode == Opcode.INVOKE_SUPER
}
val deletedOnDrawBlock = """
iget-object v0, p0, $cls->deletedLayout:Landroid/text/Layout;
if-eqz v0, :skip_deleted_draw
iget-boolean v0, p0, $cls->isDeleted:Z
if-eqz v0, :skip_deleted_draw
invoke-virtual {p1}, Landroid/graphics/Canvas;->save()I
iget-object v0, p0, $cls->deletedRect:Landroid/graphics/Rect;
invoke-virtual {p1, v0}, Landroid/graphics/Canvas;->clipRect(Landroid/graphics/Rect;)Z
iget-object v0, p0, $cls->deletedRect:Landroid/graphics/Rect;
iget v0, v0, Landroid/graphics/Rect;->left:I
int-to-float v0, v0
iget-object v1, p0, $cls->deletedRect:Landroid/graphics/Rect;
iget v1, v1, Landroid/graphics/Rect;->top:I
int-to-float v1, v1
invoke-virtual {p1, v0, v1}, Landroid/graphics/Canvas;->translate(FF)V
iget-object v0, p0, $cls->deletedLayout:Landroid/text/Layout;
.line 1
if-eqz v0, :fuck
.line 2
invoke-virtual {v0, p1}, Landroid/text/Layout;->draw(Landroid/graphics/Canvas;)V
.line 3
:fuck
invoke-virtual {p1}, Landroid/graphics/Canvas;->restore()V
""".trimIndent()
val hiddenOnDrawBlock = """
iget-object v0, p0, $cls->hiddenLayout:Landroid/text/Layout;
if-eqz v0, :skip_hidden_draw
iget-boolean v0, p0, $cls->isHidden:Z
if-eqz v0, :skip_hidden_draw
invoke-virtual {p1}, Landroid/graphics/Canvas;->save()I
iget-object v0, p0, $cls->hiddenRect:Landroid/graphics/Rect;
invoke-virtual {p1, v0}, Landroid/graphics/Canvas;->clipRect(Landroid/graphics/Rect;)Z
iget-object v0, p0, $cls->hiddenRect:Landroid/graphics/Rect;
iget v0, v0, Landroid/graphics/Rect;->left:I
int-to-float v0, v0
iget-object v1, p0, $cls->hiddenRect:Landroid/graphics/Rect;
iget v1, v1, Landroid/graphics/Rect;->top:I
int-to-float v1, v1
invoke-virtual {p1, v0, v1}, Landroid/graphics/Canvas;->translate(FF)V
iget-object v0, p0, $cls->hiddenLayout:Landroid/text/Layout;
.line 4
if-eqz v0, :nullCheck_2
.line 5
invoke-virtual {v0, p1}, Landroid/text/Layout;->draw(Landroid/graphics/Canvas;)V
.line 6
:nullCheck_2
invoke-virtual {p1}, Landroid/graphics/Canvas;->restore()V
""".trimIndent()
val instructionFollowingBothBlocks = onDrawMethod.getInstruction(superIndex + 1)
val skipHiddenLabel = ExternalLabel("skip_hidden_draw", instructionFollowingBothBlocks)
onDrawMethod.addInstructionsWithLabels(
superIndex + 1,
hiddenOnDrawBlock,
skipHiddenLabel
)
val firstInstructionOfHiddenBlock = onDrawMethod.getInstruction(superIndex + 1)
val skipDeletedLabel = ExternalLabel("skip_deleted_draw", firstInstructionOfHiddenBlock)
onDrawMethod.addInstructionsWithLabels(
superIndex + 1,
deletedOnDrawBlock,
skipDeletedLabel
)
}
}