mirror of
https://github.com/revanced/ARSCLib.git
synced 2025-04-30 22:34:24 +02:00
make general purpose Decoder for Value
This commit is contained in:
parent
fbf6b9ed6b
commit
c5c4a042a7
147
src/main/java/com/reandroid/arsc/decoder/Decoder.java
Normal file
147
src/main/java/com/reandroid/arsc/decoder/Decoder.java
Normal file
@ -0,0 +1,147 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2022 github.com/REAndroid
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package com.reandroid.arsc.decoder;
|
||||||
|
|
||||||
|
import com.reandroid.apk.AndroidFrameworks;
|
||||||
|
import com.reandroid.apk.FrameworkApk;
|
||||||
|
import com.reandroid.arsc.ApkFile;
|
||||||
|
import com.reandroid.arsc.chunk.MainChunk;
|
||||||
|
import com.reandroid.arsc.chunk.PackageBlock;
|
||||||
|
import com.reandroid.arsc.chunk.TableBlock;
|
||||||
|
import com.reandroid.arsc.chunk.xml.AndroidManifestBlock;
|
||||||
|
import com.reandroid.arsc.chunk.xml.ResXmlDocument;
|
||||||
|
import com.reandroid.arsc.group.EntryGroup;
|
||||||
|
import com.reandroid.arsc.value.AttributeValue;
|
||||||
|
import com.reandroid.arsc.value.Value;
|
||||||
|
import com.reandroid.arsc.value.ValueType;
|
||||||
|
import com.reandroid.common.EntryStore;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
public class Decoder {
|
||||||
|
private final EntryStore entryStore;
|
||||||
|
private int currentPackageId;
|
||||||
|
public Decoder(EntryStore entryStore, int currentPackageId){
|
||||||
|
this.entryStore = entryStore;
|
||||||
|
this.currentPackageId = currentPackageId;
|
||||||
|
}
|
||||||
|
public String decodeResourceName(int resourceId){
|
||||||
|
EntryGroup entryGroup = getEntryStore().getEntryGroup(resourceId);
|
||||||
|
if(entryGroup!=null){
|
||||||
|
return entryGroup.getSpecName();
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
public String decodeValue(Value value){
|
||||||
|
if(value==null){
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
ValueType valueType = value.getValueType();
|
||||||
|
if(valueType == ValueType.STRING){
|
||||||
|
return value.getValueAsString();
|
||||||
|
}
|
||||||
|
return ValueDecoder.decode(getEntryStore(), getCurrentPackageId(), value);
|
||||||
|
}
|
||||||
|
public String decodeAttributeValue(AttributeValue attributeValue){
|
||||||
|
if(attributeValue == null){
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return ValueDecoder.decode(getEntryStore(), getCurrentPackageId(), attributeValue);
|
||||||
|
}
|
||||||
|
private EntryStore getEntryStore() {
|
||||||
|
return entryStore;
|
||||||
|
}
|
||||||
|
public int getCurrentPackageId() {
|
||||||
|
return currentPackageId;
|
||||||
|
}
|
||||||
|
public void setCurrentPackageId(int currentPackageId) {
|
||||||
|
this.currentPackageId = currentPackageId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Decoder create(ResXmlDocument resXmlDocument){
|
||||||
|
MainChunk mainChunk = resXmlDocument.getMainChunk();
|
||||||
|
if(mainChunk == null){
|
||||||
|
return getNullEntryStoreDecoder();
|
||||||
|
}
|
||||||
|
ApkFile apkFile = mainChunk.getApkFile();
|
||||||
|
if(apkFile == null){
|
||||||
|
return getNullEntryStoreDecoder();
|
||||||
|
}
|
||||||
|
TableBlock tableBlock = apkFile.getTableBlock();
|
||||||
|
if(tableBlock == null){
|
||||||
|
return getNullEntryStoreDecoder();
|
||||||
|
}
|
||||||
|
AndroidManifestBlock manifestBlock = apkFile.getAndroidManifestBlock();
|
||||||
|
if(manifestBlock!=null){
|
||||||
|
int currentPackageId = manifestBlock.guessCurrentPackageId();
|
||||||
|
if(currentPackageId!=0){
|
||||||
|
return create(tableBlock, currentPackageId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return create(tableBlock);
|
||||||
|
}
|
||||||
|
public static Decoder create(TableBlock tableBlock){
|
||||||
|
if(tableBlock.getFrameWorks().size()==0){
|
||||||
|
tableBlock.addFramework(getFramework());
|
||||||
|
}
|
||||||
|
int currentPackageId;
|
||||||
|
PackageBlock packageBlock = tableBlock.pickOne();
|
||||||
|
if(packageBlock!=null){
|
||||||
|
currentPackageId = packageBlock.getId();
|
||||||
|
}else {
|
||||||
|
// 0x7f most common
|
||||||
|
currentPackageId = 0x7f;
|
||||||
|
}
|
||||||
|
return create(tableBlock, currentPackageId);
|
||||||
|
}
|
||||||
|
public static Decoder create(TableBlock tableBlock, int currentPackageId){
|
||||||
|
if(tableBlock.getFrameWorks().size()==0){
|
||||||
|
TableBlock framework = getFramework();
|
||||||
|
if(framework!=null){
|
||||||
|
PackageBlock packageBlock = framework.pickOne();
|
||||||
|
if(packageBlock!=null && packageBlock.getId() != currentPackageId){
|
||||||
|
tableBlock.addFramework(framework);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return new Decoder(tableBlock, currentPackageId);
|
||||||
|
}
|
||||||
|
private static TableBlock getFramework(){
|
||||||
|
try {
|
||||||
|
FrameworkApk frameworkApk = AndroidFrameworks.getCurrent();
|
||||||
|
if(frameworkApk == null){
|
||||||
|
frameworkApk = AndroidFrameworks.getLatest();
|
||||||
|
AndroidFrameworks.setCurrent(frameworkApk);
|
||||||
|
}
|
||||||
|
return frameworkApk.getTableBlock();
|
||||||
|
} catch (IOException ignored) {
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Decoder getNullEntryStoreDecoder(){
|
||||||
|
if(NULL_ENTRY_STORE_DECODER!=null){
|
||||||
|
return NULL_ENTRY_STORE_DECODER;
|
||||||
|
}
|
||||||
|
synchronized (Decoder.class){
|
||||||
|
TableBlock tableBlock = new TableBlock();
|
||||||
|
Decoder decoder = new Decoder(tableBlock, 0x7f);
|
||||||
|
NULL_ENTRY_STORE_DECODER = decoder;
|
||||||
|
return decoder;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
private static Decoder NULL_ENTRY_STORE_DECODER;
|
||||||
|
}
|
@ -69,6 +69,9 @@ import java.util.regex.Pattern;
|
|||||||
if("@null".equals(txt)){
|
if("@null".equals(txt)){
|
||||||
return new EncodeResult(ValueType.REFERENCE, 0);
|
return new EncodeResult(ValueType.REFERENCE, 0);
|
||||||
}
|
}
|
||||||
|
if("?null".equals(txt)){
|
||||||
|
return new EncodeResult(ValueType.ATTRIBUTE, 0);
|
||||||
|
}
|
||||||
EncodeResult result=encodeColor(txt);
|
EncodeResult result=encodeColor(txt);
|
||||||
if(result!=null){
|
if(result!=null){
|
||||||
return result;
|
return result;
|
||||||
@ -143,7 +146,7 @@ import java.util.regex.Pattern;
|
|||||||
return PATTERN_HEX_REFERENCE.matcher(txt).matches();
|
return PATTERN_HEX_REFERENCE.matcher(txt).matches();
|
||||||
}
|
}
|
||||||
private static boolean isNullReference(String txt){
|
private static boolean isNullReference(String txt){
|
||||||
if("@null".equals(txt)){
|
if("@null".equals(txt)||"?null".equals(txt)){
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if("@empty".equals(txt)){
|
if("@empty".equals(txt)){
|
||||||
@ -326,17 +329,6 @@ import java.util.regex.Pattern;
|
|||||||
}
|
}
|
||||||
return decode(valueType, data);
|
return decode(valueType, data);
|
||||||
}
|
}
|
||||||
public static String decodeIntEntry(EntryStore store, Entry entry){
|
|
||||||
if(entry ==null){
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
TableEntry<?, ?> tableEntry = entry.getTableEntry();
|
|
||||||
if(tableEntry == null || (tableEntry instanceof ResTableMapEntry)){
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
ResValue resValue =(ResValue) tableEntry.getValue();
|
|
||||||
return decodeIntEntry(store, resValue);
|
|
||||||
}
|
|
||||||
public static String decodeIntEntry(EntryStore store, ResValue resValue){
|
public static String decodeIntEntry(EntryStore store, ResValue resValue){
|
||||||
if(resValue ==null){
|
if(resValue ==null){
|
||||||
return null;
|
return null;
|
||||||
@ -406,6 +398,44 @@ import java.util.regex.Pattern;
|
|||||||
ValueType valueType= resValue.getValueType();
|
ValueType valueType= resValue.getValueType();
|
||||||
return buildReferenceValue(store, entry, valueType, resourceId);
|
return buildReferenceValue(store, entry, valueType, resourceId);
|
||||||
}
|
}
|
||||||
|
public static String decode(EntryStore entryStore, int currentPackageId, Value value){
|
||||||
|
|
||||||
|
ValueType valueType = value.getValueType();
|
||||||
|
if(valueType == ValueType.STRING){
|
||||||
|
return value.getValueAsString();
|
||||||
|
}
|
||||||
|
int data = value.getData();
|
||||||
|
if(valueType==ValueType.REFERENCE || valueType==ValueType.ATTRIBUTE){
|
||||||
|
String currentPackageName = getPackageName(entryStore, currentPackageId);
|
||||||
|
return buildReferenceValue(entryStore,
|
||||||
|
valueType, currentPackageName,
|
||||||
|
data);
|
||||||
|
}
|
||||||
|
return decode(valueType, data);
|
||||||
|
}
|
||||||
|
public static String decode(EntryStore entryStore, int currentPackageId, AttributeValue attributeValue){
|
||||||
|
ValueType valueType = attributeValue.getValueType();
|
||||||
|
if(valueType == ValueType.STRING){
|
||||||
|
return attributeValue.getValueAsString();
|
||||||
|
}
|
||||||
|
int data = attributeValue.getData();
|
||||||
|
if(valueType==ValueType.REFERENCE || valueType==ValueType.ATTRIBUTE){
|
||||||
|
String currentPackageName = getPackageName(entryStore, currentPackageId);
|
||||||
|
return buildReferenceValue(entryStore,
|
||||||
|
valueType, currentPackageName,
|
||||||
|
data);
|
||||||
|
}
|
||||||
|
if(valueType==ValueType.INT_DEC || valueType==ValueType.INT_HEX){
|
||||||
|
String result = decodeAttribute(entryStore,
|
||||||
|
attributeValue.getNameResourceID(),
|
||||||
|
attributeValue.getData());
|
||||||
|
if(result!=null){
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return decode(valueType, data);
|
||||||
|
}
|
||||||
|
@Deprecated
|
||||||
public static String decode(EntryStore entryStore, int currentPackageId, int nameResourceId, ValueType valueType, int rawVal){
|
public static String decode(EntryStore entryStore, int currentPackageId, int nameResourceId, ValueType valueType, int rawVal){
|
||||||
String currPackageName=getPackageName(entryStore, currentPackageId);
|
String currPackageName=getPackageName(entryStore, currentPackageId);
|
||||||
String result=buildReferenceValue(entryStore, valueType, currPackageName, rawVal);
|
String result=buildReferenceValue(entryStore, valueType, currPackageName, rawVal);
|
||||||
@ -506,7 +536,7 @@ import java.util.regex.Pattern;
|
|||||||
atOrQues='@';
|
atOrQues='@';
|
||||||
}else if(valueType==ValueType.ATTRIBUTE){
|
}else if(valueType==ValueType.ATTRIBUTE){
|
||||||
if(resourceId==0){
|
if(resourceId==0){
|
||||||
return "@empty";
|
return "?null";
|
||||||
}
|
}
|
||||||
atOrQues='?';
|
atOrQues='?';
|
||||||
}else {
|
}else {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user