/*
* Copyright (C) 2010-2014 JPEXS
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
package com.jpexs.helpers;
import com.jpexs.decompiler.flash.AppStrings;
import com.jpexs.decompiler.flash.configuration.Configuration;
import com.jpexs.decompiler.flash.helpers.Freed;
import com.jpexs.decompiler.flash.helpers.GraphTextWriter;
import com.jpexs.decompiler.flash.helpers.HilightedTextWriter;
import com.jpexs.decompiler.graph.GraphTargetItem;
import com.jpexs.decompiler.graph.model.LocalData;
import com.jpexs.helpers.utf8.Utf8Helper;
import java.awt.Component;
import java.awt.Rectangle;
import java.awt.Shape;
import java.awt.geom.Area;
import java.awt.image.BufferedImage;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.OutputStream;
import java.lang.reflect.Field;
import java.text.MessageFormat;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.BitSet;
import java.util.Collection;
import java.util.List;
import java.util.Scanner;
import java.util.Stack;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* Class with helper method
*
* @author JPEXS, Paolo Cancedda
*/
public class Helper {
public static String newLine = System.getProperty("line.separator");
/**
* Converts array of int values to string
*
* @param array Array of int values
* @return String representation of the array
*/
public static String intArrToString(int[] array) {
String s = "[";
for (int i = 0; i < array.length; i++) {
if (i > 0) {
s += ",";
}
s += array[i];
}
s += "]";
return s;
}
/**
* Converts array of byte values to string
*
* @param array Array of byte values
* @return String representation of the array
*/
public static String byteArrToString(byte[] array) {
String s = "[";
for (int i = 0; i < array.length; i++) {
if (i > 0) {
s += " ";
}
s += padZeros(Integer.toHexString(array[i] & 0xff), 2);
}
s += "]";
return s;
}
/**
* Adds zeros to beginning of the number to fill specified length. Returns
* as string
*
* @param number Number as string
* @param length Length of new string
* @return Number with added zeros
*/
public static String padZeros(String number, int length) {
int count = length - number.length();
for (int i = 0; i < count; i++) {
number = "0" + number;
}
return number;
}
/**
* Formats specified address to four numbers xxxx
*
* @param number Address to format
* @return String representation of the address
*/
public static String formatAddress(long number) {
if (Configuration.decimalAddress.get()) {
return padZeros(Long.toString(number), 4);
}
return padZeros(Long.toHexString(number), 4);
}
/**
* Adds space to text to fill specified width
*
* @param text Text to add spaces to
* @param width New width
* @return Text with appended spaces
*/
public static String padSpaceRight(String text, int width) {
int oldLen = text.length();
for (int i = oldLen; i < width; i++) {
text += " ";
}
return text;
}
/**
* Escapes string by adding backslashes
*
* @param s String to escape
* @return Escaped string
*/
public static String escapeString(String s) {
StringBuilder ret = new StringBuilder(s.length());
for (int i = 0; i < s.length(); i++) {
char c = s.charAt(i);
if (c == '\n') {
ret.append("\\n");
} else if (c == '\r') {
ret.append("\\r");
} else if (c == '\t') {
ret.append("\\t");
} else if (c == '\b') {
ret.append("\\b");
} else if (c == '\t') {
ret.append("\\t");
} else if (c == '\f') {
ret.append("\\f");
} else if (c == '\\') {
ret.append("\\\\");
} else if (c == '"') {
ret.append("\\\"");
} else if (c == '\'') {
ret.append("\\'");
} else {
ret.append(c);
}
}
return ret.toString();
}
public static String getValidHtmlId(String text) {
// ID and NAME tokens must begin with a letter ([A-Za-z]) and
// may be followed by any number of letters, digits ([0-9]),
// hyphens ("-"), underscores ("_"), colons (":"), and periods (".").
StringBuilder sb = new StringBuilder();
for (int i = 0; i < text.length(); i++) {
char ch = text.charAt(i);
if ((ch >= 'a' && ch <= 'z')
|| (ch >= 'A' && ch <= 'Z')
|| (i > 0 && ((ch >= '0' && ch <= '9')
|| ch == '-' || ch == '_' || ch == ':' || ch == '.'))) {
sb.append(ch);
} else {
sb.append('_');
}
}
return sb.toString();
}
private final static String SPACES12 = " ";
private final static String ZEROS8 = "00000000";
public static String formatHex(int value, int width) {
StringBuilder sb = new StringBuilder();
sb.append(Integer.toHexString(value));
if (width > sb.length()) {
sb.insert(0, ZEROS8, 0, width - sb.length());
}
return sb.toString();
}
public static String formatInt(int value, int width) {
StringBuilder sb = new StringBuilder();
sb.append(value);
if (width > sb.length()) {
sb.insert(0, SPACES12, 0, width - sb.length());
}
return sb.toString();
}
public static String indent(int level, String ss, String indentStr) {
StringBuilder sb = new StringBuilder();
for (int ii = 0; ii < level; ii++) {
sb.append(indentStr);
}
sb.append(ss);
return sb.toString();
}
public static String indentRows(int level, String ss, String indentStr) {
StringBuilder sb = new StringBuilder();
for (int ii = 0; ii < level; ii++) {
sb.append(indentStr);
}
ss = ss.replaceAll("(\r\n|\r|\n)", "\r\n");
ss = "\r\n" + ss;
String repl = "\r\n" + sb.toString();
ss = ss.replace("\r\n", repl);
if (ss.endsWith(repl)) {
ss = ss.substring(0, ss.length() - sb.toString().length());
}
ss = ss.substring(2);
return ss;
}
public static String unindentRows(int prefixLineCount, int level, String text) {
StringBuilder sb = new StringBuilder();
Scanner scanner = new Scanner(text);
String indentStr = "";
for (int i = 0; i < level; i++) {
indentStr += Configuration.getCodeFormatting().indentString;
}
int indentLength = indentStr.length();
for (int i = 0; i < prefixLineCount; i++) {
scanner.nextLine(); // ignore line
}
while (scanner.hasNextLine()) {
String line = scanner.nextLine();
if (line.startsWith(indentStr)) {
sb.append(line.substring(indentLength)).append(Configuration.getCodeFormatting().newLineChars);
} else {
return sb.toString();
}
}
return sb.toString();
}
public static int getLineCount(String s) {
if (s.endsWith("\r\n")) {
s = s.substring(0, s.length() - 2);
} else if (s.endsWith("\r")) {
s = s.substring(0, s.length() - 1);
} else if (s.endsWith("\n")) {
s = s.substring(0, s.length() - 1);
}
String[] parts = s.split("(\r\n|\r|\n)");
return parts.length;
}
public static String padZeros(long number, int length) {
String ret = "" + number;
while (ret.length() < length) {
ret = "0" + ret;
}
return ret;
}
public static String bytesToHexString(byte[] bytes) {
return bytesToHexString(bytes, 0);
}
public static String bytesToHexString(byte[] bytes, int start) {
StringBuilder sb = new StringBuilder();
if (start < bytes.length) {
for (int ii = start; ii < bytes.length; ii++) {
sb.append(formatHex(bytes[ii] & 0xff, 2));
sb.append(' ');
}
sb.setLength(sb.length() - 1);
}
return sb.toString();
}
public static String bytesToHexString(int maxByteCountInString, byte[] bytes, int start) {
if (bytes.length - start <= maxByteCountInString) {
return bytesToHexString(bytes, start);
}
byte[] trailingBytes = new byte[maxByteCountInString / 2];
byte[] headingBytes = new byte[maxByteCountInString - trailingBytes.length];
System.arraycopy(bytes, start, headingBytes, 0, headingBytes.length);
int startOfTrailingBytes = bytes.length - trailingBytes.length;
System.arraycopy(bytes, startOfTrailingBytes, trailingBytes, 0, trailingBytes.length);
StringBuilder sb = new StringBuilder();
sb.append(bytesToHexString(headingBytes, 0));
if (trailingBytes.length > 0) {
sb.append(" ... ");
sb.append(bytesToHexString(trailingBytes, 0));
}
return sb.toString();
}
public static String format(String str, int len) {
if (len <= str.length()) {
return str;
}
StringBuilder sb = new StringBuilder(str);
for (int ii = str.length(); ii < len; ii++) {
sb.append(' ');
}
return sb.toString();
}
public static String joinStrings(List arr, String glue) {
String ret = "";
boolean first = true;
for (String s : arr) {
if (!first) {
ret += glue;
} else {
first = false;
}
ret += s;
}
return ret;
}
public static String joinStrings(String[] arr, String glue) {
String ret = "";
boolean first = true;
for (String s : arr) {
if (!first) {
ret += glue;
} else {
first = false;
}
ret += s;
}
return ret;
}
public static String joinStrings(List arr, String formatString, String glue) {
String ret = "";
boolean first = true;
for (String s : arr) {
if (!first) {
ret += glue;
} else {
first = false;
}
ret += String.format(formatString, s);
}
return ret;
}
public static String joinStrings(String[] arr, String formatString, String glue) {
String ret = "";
boolean first = true;
for (String s : arr) {
if (!first) {
ret += glue;
} else {
first = false;
}
ret += String.format(formatString, s);
}
return ret;
}
@SuppressWarnings("unchecked")
public static E deepCopy(E o) {
try {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
try (ObjectOutputStream oos = new ObjectOutputStream(baos)) {
oos.writeObject(o);
oos.flush();
}
E copy;
try (ObjectInputStream ois = new ObjectInputStream(new ByteArrayInputStream(baos.toByteArray()))) {
copy = (E) ois.readObject();
}
return copy;
} catch (IOException | ClassNotFoundException ex) {
Logger.getLogger(Helper.class.getName()).log(Level.SEVERE, "Copy error", ex);
return null;
}
}
public static List