mirror of
https://git.huckle.dev/Huckles-Minecraft-Archive/jpexs-decompiler.git
synced 2026-05-26 15:17:26 +00:00
317 lines
8.8 KiB
Java
317 lines
8.8 KiB
Java
/*
|
|
* Copyright (C) 2010-2021 JPEXS, All rights reserved.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 3.0 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library.
|
|
*/
|
|
package com.jpexs.decompiler.graph;
|
|
|
|
import com.jpexs.decompiler.flash.IdentifiersDeobfuscation;
|
|
import com.jpexs.helpers.Helper;
|
|
import java.io.File;
|
|
import java.io.Serializable;
|
|
import java.util.ArrayList;
|
|
import java.util.Arrays;
|
|
import java.util.List;
|
|
|
|
/**
|
|
*
|
|
* @author JPEXS
|
|
*/
|
|
public class DottedChain implements Serializable, Comparable<DottedChain> {
|
|
|
|
public static final String NO_SUFFIX = "";
|
|
|
|
public static final DottedChain EMPTY = new DottedChain(true);
|
|
|
|
public static final DottedChain TOPLEVEL = new DottedChain(new String[]{}, NO_SUFFIX);
|
|
|
|
public static final DottedChain BOOLEAN = new DottedChain(new String[]{"Boolean"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain STRING = new DottedChain(new String[]{"String"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain ARRAY = new DottedChain(new String[]{"Array"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain NUMBER = new DottedChain(new String[]{"Number"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain OBJECT = new DottedChain(new String[]{"Object"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain INT = new DottedChain(new String[]{"int"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain UINT = new DottedChain(new String[]{"uint"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain UNDEFINED = new DottedChain(new String[]{"Undefined"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain XML = new DottedChain(new String[]{"XML"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain NULL = new DottedChain(new String[]{"null"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain FUNCTION = new DottedChain(new String[]{"Function"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain VOID = new DottedChain(new String[]{"void"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain NAMESPACE = new DottedChain(new String[]{"Namespace"}, NO_SUFFIX);
|
|
|
|
public static final DottedChain ALL = new DottedChain(new String[]{"*"}, NO_SUFFIX);
|
|
|
|
private final String[] parts;
|
|
|
|
private final int length;
|
|
|
|
private final int hash;
|
|
|
|
private boolean isNull = false;
|
|
|
|
private String namespaceSuffix = "";
|
|
|
|
public String getNamespaceSuffix() {
|
|
return namespaceSuffix;
|
|
}
|
|
|
|
public static final DottedChain parseWithSuffix(String name) {
|
|
if (name == null) {
|
|
return DottedChain.EMPTY;
|
|
} else if (name.isEmpty()) {
|
|
return DottedChain.TOPLEVEL;
|
|
} else {
|
|
String nameNoSuffix = name;
|
|
String namespaceSuffix = "";
|
|
if (name.matches(".*#[0-9]+$")) {
|
|
nameNoSuffix = name.substring(0, name.lastIndexOf("#"));
|
|
namespaceSuffix = name.substring(name.lastIndexOf("#"));
|
|
}
|
|
return new DottedChain(nameNoSuffix.split("\\."), namespaceSuffix);
|
|
}
|
|
}
|
|
|
|
private DottedChain(boolean isNull) {
|
|
this.isNull = isNull;
|
|
this.parts = new String[0];
|
|
this.length = 0;
|
|
this.hash = 0;
|
|
}
|
|
|
|
public DottedChain(DottedChain src) {
|
|
this(src.parts, src.namespaceSuffix);
|
|
this.isNull = src.isNull;
|
|
}
|
|
|
|
public DottedChain(List<String> parts) {
|
|
length = parts.size();
|
|
this.parts = parts.toArray(new String[length]);
|
|
hash = calcHash();
|
|
}
|
|
|
|
/*public DottedChain(String onePart, String namespaceSuffix) {
|
|
this(new String[]{onePart}, namespaceSuffix);
|
|
}*/
|
|
public DottedChain(String[] parts, String namespaceSuffix) {
|
|
if (parts.length == 1 && parts[0].isEmpty()) {
|
|
length = 0;
|
|
this.parts = new String[0];
|
|
} else {
|
|
length = parts.length;
|
|
this.parts = parts;
|
|
}
|
|
hash = calcHash();
|
|
this.namespaceSuffix = namespaceSuffix;
|
|
}
|
|
|
|
private DottedChain(String[] parts, int length) {
|
|
this.length = length;
|
|
this.parts = parts;
|
|
hash = calcHash();
|
|
}
|
|
|
|
public boolean isTopLevel() {
|
|
return !isNull && length == 0;
|
|
}
|
|
|
|
public boolean isEmpty() {
|
|
return isNull;
|
|
}
|
|
|
|
public int size() {
|
|
return length;
|
|
}
|
|
|
|
public String get(int index) {
|
|
if (index >= length) {
|
|
throw new ArrayIndexOutOfBoundsException();
|
|
}
|
|
|
|
return parts[index];
|
|
}
|
|
|
|
public DottedChain subChain(int count) {
|
|
if (count > length) {
|
|
throw new ArrayIndexOutOfBoundsException();
|
|
}
|
|
|
|
return new DottedChain(parts, count);
|
|
}
|
|
|
|
public String getLast() {
|
|
if (isNull) {
|
|
return null;
|
|
}
|
|
if (length == 0) {
|
|
return "";
|
|
} else {
|
|
return parts[length - 1];
|
|
}
|
|
}
|
|
|
|
public DottedChain getWithoutLast() {
|
|
if (isNull) {
|
|
return null;
|
|
}
|
|
if (length < 2) {
|
|
return EMPTY;
|
|
}
|
|
|
|
return new DottedChain(parts, length - 1);
|
|
}
|
|
|
|
public DottedChain addWithSuffix(String name) {
|
|
String addedNameNoSuffix = name;
|
|
String addedNamespaceSuffix = "";
|
|
if (name != null && name.matches(".*#[0-9]+$")) {
|
|
addedNameNoSuffix = name.substring(0, name.lastIndexOf("#"));
|
|
addedNamespaceSuffix = name.substring(name.lastIndexOf("#"));
|
|
}
|
|
return add(addedNameNoSuffix, addedNamespaceSuffix);
|
|
}
|
|
|
|
public DottedChain add(String name, String namespaceSuffix) {
|
|
if (name == null) {
|
|
return new DottedChain(this);
|
|
}
|
|
String[] nparts = new String[length + 1];
|
|
if (length > 0) {
|
|
System.arraycopy(parts, 0, nparts, 0, length);
|
|
}
|
|
|
|
nparts[nparts.length - 1] = name;
|
|
return new DottedChain(nparts, namespaceSuffix);
|
|
}
|
|
|
|
protected String toString(boolean as3, boolean raw, boolean withSuffix) {
|
|
if (isNull) {
|
|
return "";
|
|
}
|
|
if (length == 0) {
|
|
return "";
|
|
}
|
|
|
|
StringBuilder ret = new StringBuilder();
|
|
for (int i = 0; i < length; i++) {
|
|
if (i > 0) {
|
|
ret.append(".");
|
|
}
|
|
|
|
String part = parts[i];
|
|
boolean lastStar = i == length - 1 && "*".equals(part);
|
|
ret.append((raw || lastStar) ? part : IdentifiersDeobfuscation.printIdentifier(as3, part));
|
|
}
|
|
if (withSuffix) {
|
|
ret.append(namespaceSuffix);
|
|
}
|
|
return ret.toString();
|
|
}
|
|
|
|
public String toFilePath() {
|
|
if (isNull) {
|
|
return "";
|
|
}
|
|
if (length == 0) {
|
|
return "";
|
|
}
|
|
|
|
StringBuilder ret = new StringBuilder();
|
|
for (int i = 0; i < length; i++) {
|
|
if (i > 0) {
|
|
ret.append(File.separator);
|
|
}
|
|
|
|
ret.append(Helper.makeFileName(IdentifiersDeobfuscation.printIdentifier(true, parts[i])));
|
|
}
|
|
return ret.toString();
|
|
}
|
|
|
|
public List<String> toList() {
|
|
return new ArrayList<>(Arrays.asList(parts));
|
|
}
|
|
|
|
public String toPrintableString(boolean as3) {
|
|
return toString(as3, false, true);
|
|
}
|
|
|
|
public String toRawString() { //Is SUFFIX correctly handled?
|
|
return toString(false/*ignored*/, true, true);
|
|
}
|
|
|
|
@Override
|
|
public String toString() {
|
|
return toRawString();
|
|
}
|
|
|
|
@Override
|
|
public int hashCode() {
|
|
return hash;
|
|
}
|
|
|
|
private int calcHash() {
|
|
if (isNull) {
|
|
return 0;
|
|
}
|
|
int result = 1;
|
|
for (int i = 0; i < length; i++) {
|
|
result = 31 * result + parts[i].hashCode();
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
@Override
|
|
public boolean equals(Object obj) {
|
|
if (obj == null) {
|
|
return false;
|
|
}
|
|
if (getClass() != obj.getClass()) {
|
|
return false;
|
|
}
|
|
final DottedChain other = (DottedChain) obj;
|
|
if (isNull && other.isNull) {
|
|
return true;
|
|
}
|
|
if (length != other.length) {
|
|
return false;
|
|
}
|
|
|
|
for (int i = 0; i < length; i++) {
|
|
String s1 = parts[i];
|
|
String s2 = other.parts[i];
|
|
if (!s1.equals(s2)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public int compareTo(DottedChain o) {
|
|
return toRawString().compareTo(o.toRawString());
|
|
}
|
|
}
|