Files
prestonbane/src/engine/wpak/PowerActionParser.java
T

306 lines
14 KiB
Java
Raw Normal View History

2024-08-19 13:15:33 -04:00
// • ▌ ▄ ·. ▄▄▄· ▄▄ • ▪ ▄▄· ▄▄▄▄· ▄▄▄· ▐▄▄▄ ▄▄▄ .
// ·██ ▐███▪▐█ ▀█ ▐█ ▀ ▪██ ▐█ ▌▪▐█ ▀█▪▐█ ▀█ •█▌ ▐█▐▌·
// ▐█ ▌▐▌▐█·▄█▀▀█ ▄█ ▀█▄▐█·██ ▄▄▐█▀▀█▄▄█▀▀█ ▐█▐ ▐▌▐▀▀▀
// ██ ██▌▐█▌▐█ ▪▐▌▐█▄▪▐█▐█▌▐███▌██▄▪▐█▐█ ▪▐▌██▐ █▌▐█▄▄▌
// ▀▀ █▪▀▀▀ ▀ ▀ ·▀▀▀▀ ▀▀▀·▀▀▀ ·▀▀▀▀ ▀ ▀ ▀▀ █▪ ▀▀▀
// Magicbane Emulator Project © 2013 - 2024
// www.magicbane.com
package engine.wpak;
2024-08-19 20:41:30 -04:00
import engine.gameManager.ConfigManager;
2024-08-20 16:35:16 -04:00
import engine.mbEnums;
2024-08-19 21:11:53 -04:00
import engine.wpak.data.EffectDescription;
2024-08-19 20:49:43 -04:00
import engine.wpak.data.PowerActionEntry;
2024-08-20 16:35:16 -04:00
import engine.wpak.data.StatTransfer;
2024-08-20 18:18:21 -04:00
import engine.wpak.data.TrackEntry;
2024-08-20 11:58:53 -04:00
import org.pmw.tinylog.Logger;
2024-08-19 20:41:30 -04:00
2024-08-19 20:49:43 -04:00
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
2024-08-19 21:02:13 -04:00
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
2024-08-19 20:49:43 -04:00
import java.util.regex.Matcher;
2024-08-19 20:41:30 -04:00
import java.util.regex.Pattern;
2024-08-19 13:17:44 -04:00
2024-08-19 20:41:30 -04:00
public class PowerActionParser {
private static final Pattern STRSPLIT_REGEX = Pattern.compile("([^\"]\\S*|\"[^\"]*\")\\s*");
private static final Pattern POWER_ACTION_REGEX = Pattern.compile("(?<=POWERACTIONBEGIN)(.+?)(?=POWERACTIONEND)", Pattern.DOTALL);
private static final String powerActionPath = ConfigManager.DEFAULT_DATA_DIR + "wpak/PowerActions.cfg";
2024-08-19 20:49:43 -04:00
public static void parseWpakFile() {
// Read .wpak file from disk
byte[] fileData;
try {
fileData = Files.readAllBytes(Paths.get(powerActionPath));
2024-08-20 15:01:28 -04:00
} catch (IOException e) {
2024-08-19 20:49:43 -04:00
throw new RuntimeException(e);
}
String fileContents = new String(fileData);
// Iterate over power entries from .wpak data
Matcher matcher = POWER_ACTION_REGEX.matcher(fileContents);
while (matcher.find()) {
2024-08-19 20:50:45 -04:00
PowerActionEntry powerActionEntry = parsePowerActionEntry(matcher.group().trim());
2024-08-19 20:49:43 -04:00
}
}
2024-08-19 20:50:45 -04:00
private static PowerActionEntry parsePowerActionEntry(String powerActionData) {
2024-08-19 20:49:43 -04:00
PowerActionEntry powerActionEntry = new PowerActionEntry();
2024-08-19 21:11:53 -04:00
EffectDescription effectDescription;
2024-08-20 16:35:16 -04:00
StatTransfer statTransfer;
2024-08-20 18:18:21 -04:00
TrackEntry trackEntry;
2024-08-19 21:11:53 -04:00
2024-08-20 19:30:33 -04:00
// Remove all lines that contain a # and leading/trailing blank lines
2024-08-19 21:02:13 -04:00
2024-08-20 19:30:33 -04:00
powerActionData = powerActionData.replaceAll("(?m)^(\\s*#.*|\\s*)\r?\n?", "").trim();
2024-08-19 21:02:13 -04:00
2024-08-20 19:30:33 -04:00
String[] lineData = powerActionData.split("\n");
2024-08-19 21:02:13 -04:00
2024-08-20 19:30:33 -04:00
// Parse effect entry header
2024-08-19 21:02:13 -04:00
2024-08-20 19:30:33 -04:00
Iterator<String> entryIterator = Arrays.stream(lineData).iterator();
2024-08-19 21:02:13 -04:00
2024-08-20 19:30:33 -04:00
String headerLine = entryIterator.next();
ArrayList<String> headerData = new ArrayList<>();
2024-08-19 21:02:13 -04:00
2024-08-20 19:30:33 -04:00
Matcher matcher = STRSPLIT_REGEX.matcher(headerLine.trim());
2024-08-19 21:03:34 -04:00
2024-08-20 19:30:33 -04:00
while (matcher.find())
headerData.add(matcher.group().trim());
2024-08-19 20:49:43 -04:00
2024-08-20 19:30:33 -04:00
Iterator<String> headerIterator = headerData.iterator();
powerActionEntry.action_id = headerIterator.next();
powerActionEntry.action_type = headerIterator.next();
2024-08-19 21:11:53 -04:00
2024-08-20 19:30:33 -04:00
switch (powerActionEntry.action_type) {
case "RemoveEffect":
effectDescription = new EffectDescription();
effectDescription.effect_id = headerIterator.next();
powerActionEntry.effects.add(effectDescription);
break;
case "CreateMob":
powerActionEntry.petLevel = Integer.parseInt(headerIterator.next());
powerActionEntry.petRace = Integer.parseInt(headerIterator.next());
break;
case "DamageOverTime":
effectDescription = new EffectDescription();
effectDescription.effect_id = headerIterator.next();
effectDescription.cycleDuration = Integer.parseInt(headerIterator.next());
effectDescription.cycleDelay = Integer.parseInt(headerIterator.next());
powerActionEntry.effects.add(effectDescription);
break;
case "ApplyEffects":
int level = Integer.parseInt(headerIterator.next());
while (headerIterator.hasNext()) {
2024-08-20 15:41:26 -04:00
effectDescription = new EffectDescription();
2024-08-20 19:30:33 -04:00
effectDescription.level = level;
2024-08-20 15:41:26 -04:00
effectDescription.effect_id = headerIterator.next();
powerActionEntry.effects.add(effectDescription);
2024-08-20 19:30:33 -04:00
}
break;
case "Transform":
case "Invis":
case "ApplyEffect":
case "DeferredPower":
case "DirectDamage":
case "SpireDisable":
while (headerIterator.hasNext()) {
2024-08-20 15:41:26 -04:00
effectDescription = new EffectDescription();
effectDescription.effect_id = headerIterator.next();
2024-08-20 19:30:33 -04:00
// Some applyEffect entries are naked withot a level
if (headerData.size() > 3)
effectDescription.level = Integer.parseInt(headerIterator.next());
2024-08-20 15:41:26 -04:00
powerActionEntry.effects.add(effectDescription);
2024-08-20 19:30:33 -04:00
}
break;
case "TransferStat":
statTransfer = new StatTransfer();
statTransfer.fromStat = mbEnums.CostType.valueOf(headerIterator.next());
statTransfer.toStat = mbEnums.CostType.valueOf(headerIterator.next());
statTransfer.fromStatValue = Float.parseFloat(headerIterator.next());
statTransfer.fromCurve = mbEnums.CompoundCurveType.valueOf(headerIterator.next());
statTransfer.toStatValue = Float.parseFloat(headerIterator.next());
statTransfer.toCurve = mbEnums.CompoundCurveType.valueOf(headerIterator.next());
statTransfer.fromStatBool = Boolean.parseBoolean(headerIterator.next());
statTransfer.toStatBool = Boolean.parseBoolean(headerIterator.next());
powerActionEntry.statTransfer = statTransfer;
break;
case "TransferStatOT":
statTransfer = new StatTransfer();
statTransfer.fromStat = mbEnums.CostType.valueOf(headerIterator.next());
statTransfer.toStat = mbEnums.CostType.valueOf(headerIterator.next());
statTransfer.fromStatValue = Float.parseFloat(headerIterator.next());
statTransfer.fromCurve = mbEnums.CompoundCurveType.valueOf(headerIterator.next());
statTransfer.toStatValue = Float.parseFloat(headerIterator.next());
statTransfer.toCurve = mbEnums.CompoundCurveType.valueOf(headerIterator.next());
statTransfer.fromStatBool = Boolean.parseBoolean(headerIterator.next());
statTransfer.toStatBool = Boolean.parseBoolean(headerIterator.next());
statTransfer.transfer_action = headerIterator.next();
statTransfer.transfer_ticks = Integer.parseInt(headerIterator.next());
powerActionEntry.statTransfer = statTransfer;
break;
case "Charm":
effectDescription = new EffectDescription();
effectDescription.effect_id = headerIterator.next();
effectDescription.level = Integer.parseInt(headerIterator.next());
effectDescription.type = headerIterator.next();
powerActionEntry.effects.add(effectDescription);
break;
case "Block":
effectDescription = new EffectDescription();
effectDescription.effect_id = headerIterator.next();
effectDescription.level = Integer.parseInt(headerIterator.next());
powerActionEntry.effects.add(effectDescription);
break;
case "Resurrect":
powerActionEntry.levelCap = Integer.parseInt(headerIterator.next());
break;
case "SetItemFlag":
powerActionEntry.itemFlag = mbEnums.ItemFlags.valueOf(headerIterator.next());
break;
case "Track":
trackEntry = new TrackEntry();
trackEntry.action_id = headerIterator.next();
trackEntry.unknown1 = Boolean.parseBoolean(headerIterator.next());
trackEntry.unknown2 = Boolean.parseBoolean(headerIterator.next());
trackEntry.type = headerIterator.next();
trackEntry.min = Integer.parseInt(headerIterator.next());
trackEntry.max = Integer.parseInt(headerIterator.next());
powerActionEntry.trackEntry = trackEntry;
break;
case "Recall": // No arguments for these tags or not parsed
case "Summon":
case "Teleport":
case "TreeChoke":
case "SimpleDamage":
case "MobRecall": // One argument always zero
case "ClearAggro":
case "ClearNearbyAggro":
case "Peek":
case "ClaimMine":
case "RunegateTeleport":
case "Steal":
break;
default:
Logger.error("Unhandled type " + powerActionEntry.action_type + " for Pow4erAction: " + powerActionEntry.action_id);
break;
}
// Process key value pairs after header
while (entryIterator.hasNext()) {
String lineValue = entryIterator.next();
String[] lineValues = lineValue.split("=");
String key = lineValues[0].trim();
String[] arguments;
switch (key) {
case "BODYPARTS":
arguments = lineValues[1].trim().split("\\s+");
for (String bodyPart : arguments)
powerActionEntry.bodyparts.add(Integer.parseInt(bodyPart));
2024-08-20 11:58:53 -04:00
break;
2024-08-20 19:30:33 -04:00
case "FEMALEBODYPARTS":
arguments = lineValues[1].trim().split("\\s+");
for (String bodyPart : arguments)
powerActionEntry.femaleBodyParts.add(Integer.parseInt(bodyPart));
2024-08-20 18:24:05 -04:00
break;
2024-08-20 19:30:33 -04:00
case "SCALEFACTOR":
arguments = lineValues[1].trim().split("\\s+");
for (String scaleFactor : arguments)
powerActionEntry.scaleFactor.add(Float.parseFloat(scaleFactor));
2024-08-20 12:05:03 -04:00
break;
2024-08-20 19:30:33 -04:00
case "ISRESISTABLE":
powerActionEntry.isResistible = Boolean.parseBoolean(lineValues[1].trim());
2024-08-20 16:35:16 -04:00
break;
2024-08-20 19:30:33 -04:00
case "ISAGGRESSIVE":
powerActionEntry.isAggressive = Boolean.parseBoolean(lineValues[1].trim());
2024-08-20 16:35:16 -04:00
break;
2024-08-20 19:30:33 -04:00
case "BLADETRAILS":
powerActionEntry.bladeTrails = Boolean.parseBoolean(lineValues[1].trim());
2024-08-20 16:49:51 -04:00
break;
2024-08-20 19:30:33 -04:00
case "SHOULDSHOWWEAPONS":
powerActionEntry.shouldShowWeapons = Boolean.parseBoolean(lineValues[1].trim());
break;
case "SHOULDSHOWARMOR":
powerActionEntry.shouldShowArmor = Boolean.parseBoolean(lineValues[1].trim());
break;
case "APPLYEFFECTBLANK":
powerActionEntry.applyEffectBlank = Boolean.parseBoolean(lineValues[1].trim());
break;
case "WEAROFFEFFECTBLANK":
powerActionEntry.wearOffEffectBlank = Boolean.parseBoolean(lineValues[1].trim());
break;
case "ATTACKANIMS":
arguments = lineValues[1].trim().split("\\s+");
for (String animation : arguments)
powerActionEntry.attackAnimations.add(Integer.parseInt(animation));
break;
case "REMOVEALL":
powerActionEntry.removeAll = Boolean.parseBoolean(lineValues[1].trim());
break;
case "EFFECTID":
2024-08-20 17:14:26 -04:00
effectDescription = new EffectDescription();
2024-08-20 19:30:33 -04:00
effectDescription.effect_id = lineValues[1].trim();
2024-08-20 17:14:26 -04:00
powerActionEntry.effects.add(effectDescription);
break;
2024-08-20 19:30:33 -04:00
case "LEVELCAP":
arguments = lineValues[1].trim().split("\\s+");
powerActionEntry.levelCap = Integer.parseInt(arguments[0]);
if (arguments.length > 1) // Not all level caps have a curve
powerActionEntry.levelCurve = mbEnums.CompoundCurveType.valueOf(arguments[1]);
break;
case "CLEARAGGRO":
powerActionEntry.clearAggro = Boolean.parseBoolean(lineValues[1].trim());
2024-08-20 17:28:10 -04:00
break;
2024-08-20 19:30:33 -04:00
case "TARGETBECOMESPET":
powerActionEntry.targetBecomesPet = Boolean.parseBoolean(lineValues[1].trim());
2024-08-20 17:37:17 -04:00
break;
2024-08-20 19:30:33 -04:00
case "DESTROYOLDPET":
powerActionEntry.destroyOldPet = Boolean.parseBoolean(lineValues[1].trim());
2024-08-20 18:18:21 -04:00
break;
2024-08-20 19:30:33 -04:00
case "DAMAGETYPE":
powerActionEntry.damageType = mbEnums.DamageType.valueOf(lineValues[1].trim().toUpperCase());
2024-08-20 15:18:02 -04:00
break;
2024-08-20 19:30:33 -04:00
case "ROOTFSMID":
powerActionEntry.rootFsmID = mbEnums.MobBehaviourType.valueOf(lineValues[1].trim());
2024-08-20 15:01:28 -04:00
break;
2024-08-20 19:30:33 -04:00
case "SPLASHDAMAGE":
arguments = lineValues[1].trim().split("\\s+");
powerActionEntry.splashDamageMin = Integer.parseInt(arguments[0]);
powerActionEntry.splashDamageMax = Integer.parseInt(arguments[1]);
break;
case "APPLYEFFECTOTHER":
case "APPLYEFFECTSELF":
case "WEAROFFEFFECTOTHER": // Keys not parsed go here.
case "WEAROFFEFFECTSELF":
break;
default:
Logger.error("Unhandled variable type:" + key + " for powerAction: " + powerActionEntry.action_id);
2024-08-20 11:58:53 -04:00
}
2024-08-20 19:30:33 -04:00
}
2024-08-19 20:49:43 -04:00
return powerActionEntry;
}
2024-08-19 13:15:33 -04:00
}