Files
lakebane/src/engine/wpak/PowersParser.java
T

310 lines
14 KiB
Java
Raw Normal View History

2024-08-13 12:04:18 -04:00
// • ▌ ▄ ·. ▄▄▄· ▄▄ • ▪ ▄▄· ▄▄▄▄· ▄▄▄· ▐▄▄▄ ▄▄▄ .
// ·██ ▐███▪▐█ ▀█ ▐█ ▀ ▪██ ▐█ ▌▪▐█ ▀█▪▐█ ▀█ •█▌ ▐█▐▌·
// ▐█ ▌▐▌▐█·▄█▀▀█ ▄█ ▀█▄▐█·██ ▄▄▐█▀▀█▄▄█▀▀█ ▐█▐ ▐▌▐▀▀▀
// ██ ██▌▐█▌▐█ ▪▐▌▐█▄▪▐█▐█▌▐███▌██▄▪▐█▐█ ▪▐▌██▐ █▌▐█▄▄▌
// ▀▀ █▪▀▀▀ ▀ ▀ ·▀▀▀▀ ▀▀▀·▀▀▀ ·▀▀▀▀ ▀ ▀ ▀▀ █▪ ▀▀▀
// Magicbane Emulator Project © 2013 - 2024
// www.magicbane.com
package engine.wpak;
2024-08-13 12:20:27 -04:00
import engine.gameManager.ConfigManager;
2024-08-17 15:58:48 -04:00
import engine.mbEnums;
2024-08-19 07:32:25 -04:00
import engine.wpak.data.*;
2024-08-15 16:31:04 -04:00
import org.pmw.tinylog.Logger;
2024-08-13 12:20:27 -04:00
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.ArrayList;
2024-08-18 15:47:22 -04:00
import java.util.Arrays;
2024-08-21 13:06:35 -04:00
import java.util.List;
2024-08-13 12:20:27 -04:00
import java.util.regex.Matcher;
2024-08-13 12:04:18 -04:00
import java.util.regex.Pattern;
public class PowersParser {
private static final Pattern POWER_REGEX = Pattern.compile("(?<=POWERBEGIN)(.+?)(?=POWEREND)", Pattern.DOTALL);
2024-08-13 12:20:27 -04:00
private static final Pattern STRSPLIT_REGEX = Pattern.compile("([^\"]\\S*|\"[^\"]*\")\\s*");
2024-08-19 05:36:51 -04:00
private static final Pattern CONDITION_REGEX = Pattern.compile("(?<=CONDITIONBEGIN)(.+?)(?=CONDITIONEND)", Pattern.DOTALL);
2024-08-19 11:10:55 -04:00
private static final String powersPath = ConfigManager.DEFAULT_DATA_DIR + "wpak/Powers.cfg";
2024-08-13 12:20:27 -04:00
2024-08-19 20:38:09 -04:00
public static void parseWpakFile() {
2024-08-13 12:20:27 -04:00
// Read .wpak file from disk
2024-08-19 20:38:38 -04:00
byte[] fileData;
2024-08-19 20:38:09 -04:00
try {
fileData = Files.readAllBytes(Paths.get(powersPath));
} catch (IOException e) {
throw new RuntimeException(e);
}
2024-08-13 12:20:27 -04:00
String fileContents = new String(fileData);
2024-08-14 16:27:51 -04:00
// Iterate over power entries from .wpak data
2024-08-13 12:20:27 -04:00
Matcher matcher = POWER_REGEX.matcher(fileContents);
while (matcher.find()) {
2024-08-14 16:46:02 -04:00
PowerEntry powerEntry = parsePowerEntry(matcher.group().trim());
2024-08-13 12:20:27 -04:00
}
}
2024-08-17 15:31:45 -04:00
private static PowerEntry parsePowerEntry(String powerData) {
2024-08-15 16:31:04 -04:00
2024-08-13 12:20:27 -04:00
PowerEntry powerEntry = new PowerEntry();
2024-08-21 13:23:22 -04:00
StringBuilder conditionBuilder = new StringBuilder();
StringBuilder powerBuilder = new StringBuilder();
String conditionString;
String powerString;
2024-08-19 05:28:23 -04:00
2024-08-20 12:06:35 -04:00
int endPos = 0;
2024-08-19 05:28:23 -04:00
2024-08-20 12:06:35 -04:00
// Separate out any conditions from the power data
2024-08-19 06:26:06 -04:00
2024-08-20 12:06:35 -04:00
Matcher matcher = CONDITION_REGEX.matcher(powerData);
2024-08-19 05:28:23 -04:00
2024-08-20 12:06:35 -04:00
while (matcher.find()) {
2024-08-21 13:23:22 -04:00
conditionBuilder.append(matcher.group().trim());
powerBuilder.append(powerData, endPos, matcher.start());
2024-08-20 12:06:35 -04:00
endPos = matcher.end();
}
2024-08-21 13:23:22 -04:00
powerBuilder.append(powerData.substring(endPos));
2024-08-19 05:28:23 -04:00
2024-08-20 12:06:35 -04:00
// Cleanup dangling tags and lines that contain a # and leading/trailing blank lines
2024-08-19 05:28:23 -04:00
2024-08-21 13:23:22 -04:00
powerString = powerBuilder.toString().replaceAll("CONDITIONBEGINCONDITIONEND", "")
.replaceAll("(?m)^(\\s*#.*|\\s*)\r?\n?", "");
conditionString = conditionBuilder.toString().replaceAll("(?m)^(\\s*#.*|\\s*)\r?\n?", "");
2024-08-19 06:25:03 -04:00
2024-08-20 12:06:35 -04:00
// Parse header line in power data
2024-08-19 06:25:03 -04:00
2024-08-21 13:23:22 -04:00
List<String> lineData = Arrays.asList(powerString.trim().split("\n"));
2024-08-21 13:06:35 -04:00
List<String> powerHeader = new ArrayList<>();
2024-08-19 06:26:59 -04:00
2024-08-21 13:06:35 -04:00
String headerString = lineData.get(0);
2024-08-20 12:06:35 -04:00
headerString = headerString.replace("\n", " ");
2024-08-15 16:31:04 -04:00
2024-08-20 12:06:35 -04:00
matcher = STRSPLIT_REGEX.matcher(headerString);
2024-08-15 16:31:04 -04:00
2024-08-20 12:06:35 -04:00
while (matcher.find())
powerHeader.add(matcher.group().trim());
2024-08-15 16:31:04 -04:00
2024-08-20 12:06:35 -04:00
java.util.Iterator<String> iterator = powerHeader.iterator();
2024-08-15 16:31:04 -04:00
2024-08-20 12:06:35 -04:00
powerEntry.power_id = iterator.next();
powerEntry.power = iterator.next().replaceAll("\"", "");
2024-08-16 12:01:13 -04:00
2024-08-20 12:06:35 -04:00
PowerData power = new PowerData();
power.power_type = mbEnums.PowerType.valueOf(iterator.next());
power.icon = Integer.parseInt(iterator.next());
power.powerBase = iterator.next().replaceAll("\"", "");
powerEntry.powers.add(power);
2024-08-16 12:01:13 -04:00
2024-08-20 12:06:35 -04:00
String nextValue = iterator.next();
// Account for second definition
if (nextValue.equals("SPELL") || nextValue.equals("SKILL")) {
power = new PowerData();
power.power_type = mbEnums.PowerType.valueOf(nextValue);
2024-08-17 15:13:32 -04:00
power.icon = Integer.parseInt(iterator.next());
power.powerBase = iterator.next().replaceAll("\"", "");
2024-08-17 14:51:18 -04:00
powerEntry.powers.add(power);
2024-08-20 12:06:35 -04:00
powerEntry.target_type = mbEnums.PowerTargetType.valueOf(iterator.next());
} else
powerEntry.target_type = mbEnums.PowerTargetType.valueOf(nextValue);
powerEntry.range = Integer.parseInt(iterator.next());
powerEntry.areaType = mbEnums.AreaType.valueOf(iterator.next());
powerEntry.areaRange = Integer.parseInt(iterator.next());
powerEntry.excludeType = mbEnums.ExcludeType.valueOf(iterator.next());
powerEntry.costType = mbEnums.CostType.valueOf(iterator.next());
powerEntry.cost = Float.parseFloat(iterator.next());
powerEntry.difficulty = Float.parseFloat(iterator.next());
powerEntry.precision = Float.parseFloat(iterator.next());
2024-08-21 13:06:35 -04:00
// Cleanup init_time in client data which is 0.35.1 or some such
2024-08-20 12:06:35 -04:00
powerEntry.init_time = Float.parseFloat(iterator.next().replaceAll("(\\.0)+$", ""));
powerEntry.release_time = Float.parseFloat(iterator.next());
powerEntry.recycle_time = Float.parseFloat(iterator.next());
powerEntry.hitRollYN = Integer.parseInt(iterator.next());
powerEntry.castingMode = mbEnums.CastingModeType.valueOf(iterator.next());
powerEntry.initAmin = Integer.parseInt(iterator.next());
powerEntry.releaseAnim = Integer.parseInt(iterator.next());
powerEntry.targetSelect = mbEnums.TargetSelectType.valueOf(iterator.next());
// Process key value pairs after header
2024-08-21 13:06:35 -04:00
iterator = lineData.iterator();
2024-08-20 12:06:35 -04:00
iterator.next(); // Ignore header
while (iterator.hasNext()) {
String lineValue = iterator.next();
2024-08-21 13:06:35 -04:00
List<String> lineValues = Arrays.asList(lineValue.split("="));
String key = lineValues.get(0).trim();
2024-08-20 12:06:35 -04:00
ActionEntry actionEntry;
2024-08-21 13:06:35 -04:00
List<String> arguments;
2024-08-20 12:06:35 -04:00
Matcher matcher1;
switch (key) {
case "ACTION":
actionEntry = new ActionEntry();
2024-08-21 13:06:35 -04:00
arguments = Arrays.asList(lineValues.get(1).trim().split("\\s+"));
actionEntry.effect_id = arguments.get(0);
actionEntry.minTrains = Integer.parseInt(arguments.get(1));
actionEntry.maxTrains = Integer.parseInt(arguments.get(2));
actionEntry.duration = Float.parseFloat(arguments.get(3));
actionEntry.curve = mbEnums.CompoundCurveType.valueOf(arguments.get(4));
actionEntry.stackingCategory = arguments.get(5);
actionEntry.stackingPriority = Integer.parseInt(arguments.get(6));
actionEntry.categoryToPower = mbEnums.CategoryToPowerType.valueOf(arguments.get(7));
2024-08-20 12:06:35 -04:00
powerEntry.actionEntries.add(actionEntry);
break;
case "MaxLevel":
2024-08-21 13:06:35 -04:00
powerEntry.maxLevel = Integer.parseInt(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "HateValue":
2024-08-21 13:06:35 -04:00
arguments = Arrays.asList(lineValues.get(1).trim().split("\\s+"));
powerEntry.hateValue = Integer.parseInt(arguments.get(0));
2024-08-20 12:06:35 -04:00
// Not all entries have a curve. Defaults to DefaultFlat;
2024-08-21 13:06:35 -04:00
if (arguments.size() > 1)
powerEntry.hateCurve = mbEnums.CompoundCurveType.valueOf(arguments.get(1));
2024-08-20 12:06:35 -04:00
break;
case "LOOPANIMID":
2024-08-21 13:06:35 -04:00
powerEntry.loopAnimID = Integer.parseInt(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "GRANTOVERRIDEVAR":
2024-08-21 13:06:35 -04:00
powerEntry.grantOverrideVar = lineValues.get(1).trim();
2024-08-20 12:06:35 -04:00
break;
case "DESCRIPTION":
2024-08-21 13:06:35 -04:00
powerEntry.description.add(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "CATEGORY":
2024-08-21 13:06:35 -04:00
powerEntry.category = lineValues.get(1).trim();
2024-08-20 12:06:35 -04:00
break;
case "CURVE":
2024-08-21 13:06:35 -04:00
arguments = Arrays.asList(lineValues.get(1).trim().split("\\s+"));
powerEntry.curves.put(arguments.get(0), mbEnums.CompoundCurveType.valueOf(arguments.get(1)));
2024-08-20 12:06:35 -04:00
break;
case "EQPREREQ":
EquipmentPreReq equipmentPreReq = new EquipmentPreReq();
2024-08-21 13:06:35 -04:00
matcher1 = STRSPLIT_REGEX.matcher(lineValues.get(1).trim());
2024-08-21 13:23:22 -04:00
arguments = new ArrayList<>();
2024-08-20 12:06:35 -04:00
while (matcher1.find())
2024-08-21 13:23:22 -04:00
arguments.add(matcher1.group().trim());
2024-08-20 12:06:35 -04:00
2024-08-21 13:23:22 -04:00
equipmentPreReq.slot = mbEnums.EquipSlotType.valueOf(arguments.get(0));
equipmentPreReq.skill = arguments.get(1).replaceAll("\"", "");
equipmentPreReq.level = Integer.parseInt(arguments.get(2));
2024-08-20 12:06:35 -04:00
powerEntry.equipmentPreReq = equipmentPreReq;
break;
case "CANCASTWHILEMOVING":
2024-08-21 13:06:35 -04:00
powerEntry.canCastWhileMoving = Boolean.parseBoolean(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "CANCASTWHILEFLYING":
2024-08-21 13:06:35 -04:00
powerEntry.canCastWhileFlying = Boolean.parseBoolean(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "BLADETRAILS":
2024-08-21 13:06:35 -04:00
powerEntry.bladeTrails = Boolean.parseBoolean(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "EFFECTPREREQ":
EffectDescription effectPreReq = new EffectDescription();
2024-08-21 13:06:35 -04:00
arguments = Arrays.asList(lineValues.get(1).trim().split("\\s+"));
effectPreReq.effect_id = arguments.get(9);
effectPreReq.level = Integer.parseInt(arguments.get(1));
effectPreReq.message = arguments.get(2);
2024-08-20 12:06:35 -04:00
powerEntry.effectPreReqs.add(effectPreReq);
break;
case "MONSTERTYPERESTRICTS":
2024-08-21 13:06:35 -04:00
arguments = Arrays.asList(lineValues.get(1).trim().split("\\s+"));
2024-08-20 12:06:35 -04:00
for (String restriction : arguments)
powerEntry.monsterRestricts.add(mbEnums.MonsterType.valueOf(restriction.trim()));
break;
case "MONSTERTYPEPREREQS":
2024-08-21 13:06:35 -04:00
arguments = Arrays.asList(lineValues.get(1).trim().split("\\s+"));
2024-08-20 12:06:35 -04:00
for (String restriction : arguments)
powerEntry.monsterPrereqs.add(mbEnums.MonsterType.valueOf(restriction.trim()));
break;
case "SHOULDCHECKPATH":
2024-08-21 13:06:35 -04:00
powerEntry.shouldCheckPath = Boolean.parseBoolean(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "STICKY":
2024-08-21 13:06:35 -04:00
powerEntry.sticky = Boolean.parseBoolean(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "PULSEINFO":
2024-08-21 13:06:35 -04:00
arguments = Arrays.asList(lineValues.get(1).trim().split("\\s+"));
powerEntry.pulseCycle = Integer.parseInt(arguments.get(0));
powerEntry.pulseDuration = Integer.parseInt(arguments.get(1));
2024-08-20 12:06:35 -04:00
break;
case "MAXNUMMOBTARGETS":
2024-08-21 13:06:35 -04:00
powerEntry.maxMobTargets = Integer.parseInt(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "MAXNUMPLAYERTARGETS":
2024-08-21 13:06:35 -04:00
powerEntry.maxPlayerTargets = Integer.parseInt(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "ISADMINPOWER":
2024-08-21 13:06:35 -04:00
powerEntry.isAdminPower = Boolean.parseBoolean(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "ISPROJECTILE":
2024-08-21 13:06:35 -04:00
powerEntry.isProjectile = Boolean.parseBoolean(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "CASTERSPULSEPARTICLE":
2024-08-21 13:06:35 -04:00
powerEntry.casterPulseParticle = Integer.parseInt(lineValues.get(1).trim());
2024-08-20 12:06:35 -04:00
break;
case "TARGETEFFECTPREREQS_ORED":
EffectDescription preReq = new EffectDescription();
2024-08-21 13:06:35 -04:00
arguments = Arrays.asList(lineValues.get(1).trim().split("\\s+"));
preReq.effect_id = arguments.get(0);
preReq.level = Integer.parseInt(arguments.get(1));
2024-08-20 12:06:35 -04:00
powerEntry.targetEffectPrereqs.add(preReq);
break;
case "SOUNDS": // Values not parsed
case "APPLYDAMAGESELF":
case "APPLYDAMAGECASTER":
case "APPLYDAMAGEOTHER":
case "APPLYDAMAGETARGET":
case "APPLYEFFECTSELF":
case "APPLYEFFECTOTHER":
case "APPLYEFFECTCASTER":
case "APPLYEFFECTTARGET":
case "FIZZLEOTHER":
case "FIZZLESELF":
case "INITSTRING":
case "SUCCESSOTHER":
case "SUCCESSSELF":
case "WEAROFFEFFECTOTHER":
case "WEAROFFEFFECTSELF":
break;
default:
Logger.error("Unhandled variable type:" + key + " for power: " + powerEntry.power_id);
2024-08-17 15:31:45 -04:00
}
2024-08-20 12:06:35 -04:00
}
2024-08-14 16:27:51 -04:00
2024-08-20 12:06:35 -04:00
// Parse power conditions
2024-08-19 11:10:55 -04:00
2024-08-21 13:23:22 -04:00
if (conditionString.isEmpty() == false) {
2024-08-21 13:06:35 -04:00
2024-08-21 13:23:22 -04:00
List<String> conditions = Arrays.asList(conditionString.split("\n"));
2024-08-19 11:10:55 -04:00
2024-08-20 12:06:35 -04:00
for (String condition : conditions) {
2024-08-21 13:06:35 -04:00
List<String> parameters = Arrays.asList(condition.trim().split("\\s+"));
powerEntry.conditions.put(parameters.get(0), Float.parseFloat(parameters.get(1)));
2024-08-19 11:10:55 -04:00
}
2024-08-15 16:31:04 -04:00
}
2024-08-13 12:20:27 -04:00
return powerEntry;
}
2024-08-13 12:04:18 -04:00
}
2024-08-13 12:20:27 -04:00