You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
279 lines
11 KiB
279 lines
11 KiB
// • ▌ ▄ ·. ▄▄▄· ▄▄ • ▪ ▄▄· ▄▄▄▄· ▄▄▄· ▐▄▄▄ ▄▄▄ . |
|
// ·██ ▐███▪▐█ ▀█ ▐█ ▀ ▪██ ▐█ ▌▪▐█ ▀█▪▐█ ▀█ •█▌ ▐█▐▌· |
|
// ▐█ ▌▐▌▐█·▄█▀▀█ ▄█ ▀█▄▐█·██ ▄▄▐█▀▀█▄▄█▀▀█ ▐█▐ ▐▌▐▀▀▀ |
|
// ██ ██▌▐█▌▐█ ▪▐▌▐█▄▪▐█▐█▌▐███▌██▄▪▐█▐█ ▪▐▌██▐ █▌▐█▄▄▌ |
|
// ▀▀ █▪▀▀▀ ▀ ▀ ·▀▀▀▀ ▀▀▀·▀▀▀ ·▀▀▀▀ ▀ ▀ ▀▀ █▪ ▀▀▀ |
|
// Magicbane Emulator Project © 2013 - 2024 |
|
// www.magicbane.com |
|
|
|
package engine.wpak; |
|
|
|
import engine.gameManager.ConfigManager; |
|
import engine.mbEnums; |
|
import engine.wpak.data.ConditionEntry; |
|
import engine.wpak.data.EffectEntry; |
|
import engine.wpak.data.EffectModifier; |
|
import org.pmw.tinylog.Logger; |
|
|
|
import java.io.IOException; |
|
import java.nio.file.Files; |
|
import java.nio.file.Paths; |
|
import java.util.*; |
|
import java.util.regex.Matcher; |
|
import java.util.regex.Pattern; |
|
|
|
public class EffectsParser { |
|
|
|
public static String effectsPath = ConfigManager.DEFAULT_DATA_DIR + "wpak/Effects.cfg"; |
|
public static HashMap<String, EffectEntry> effect_data = new HashMap<>(); |
|
private static final Pattern EFFECT_REGEX = Pattern.compile("(?<=EFFECTBEGIN)(.+?)(?=EFFECTEND)", Pattern.DOTALL); |
|
private static final Pattern SOURCE_REGEX = Pattern.compile("(?<=SOURCEBEGIN)(.+?)(?=SOURCEEND)", Pattern.DOTALL); |
|
private static final Pattern MODS_REGEX = Pattern.compile("(?<=MODSBEGIN)(.+?)(?=MODSEND)", Pattern.DOTALL); |
|
private static final Pattern CONDITIONS_REGEX = Pattern.compile("(?<=CONDITIONBEGIN)(.+?)(?=CONDITIONEND)", Pattern.DOTALL); |
|
private static final Pattern STRSPLIT_REGEX = Pattern.compile("([^\"]\\S*|\"[^\"]*\")\\s*"); // Regex ignores spaces within quotes |
|
|
|
public static void parseWpakFile() { |
|
|
|
// Read .wpak file from disk |
|
|
|
byte[] fileData; |
|
|
|
try { |
|
fileData = Files.readAllBytes(Paths.get(effectsPath)); |
|
} catch (IOException e) { |
|
throw new RuntimeException(e); |
|
} |
|
|
|
String fileContents = new String(fileData); |
|
|
|
// Iterate over effect entries from .wpak data |
|
|
|
Matcher matcher = EFFECT_REGEX.matcher(fileContents); |
|
|
|
while (matcher.find()) { |
|
EffectEntry effectEntry = parseEffectEntry(matcher.group()); |
|
effect_data.put(effectEntry.effect_id, effectEntry); |
|
} |
|
} |
|
|
|
private static EffectEntry parseEffectEntry(String effectData) { |
|
|
|
EffectEntry effectEntry = new EffectEntry(); |
|
|
|
// Parse fields that lie outside the other tags |
|
|
|
effectEntry.isItemEffect = effectData.contains("IsItemEffect"); |
|
effectEntry.isSpireEffect = effectData.contains("IsSpireEffect"); |
|
effectEntry.ignoreNoMod = effectData.contains("IgnoreNoMod"); |
|
effectEntry.dontSave = effectData.contains("DontSave"); |
|
|
|
// Remove all lines that contain a # and leading/trailing blank lines |
|
|
|
effectData = effectData.replaceAll("(?m)^(\\s*#.*|\\s*)\r?\n?", ""); |
|
effectData = effectData.trim(); |
|
|
|
// Parse effect entry header |
|
|
|
String firstLine; |
|
ArrayList<String> effectHeader = new ArrayList<>(); |
|
|
|
// Some effects exist without sources/mods or conditions |
|
// (ACID "MOB" 0) |
|
|
|
if (effectData.indexOf('\n') > 0) |
|
firstLine = effectData.substring(0, effectData.indexOf('\n')); |
|
else |
|
firstLine = effectData; |
|
|
|
Matcher matcher = STRSPLIT_REGEX.matcher(firstLine); |
|
|
|
while (matcher.find()) |
|
effectHeader.add(matcher.group().trim()); |
|
|
|
effectEntry.effect_id = effectHeader.get(0); |
|
effectEntry.effect_name = effectHeader.get(1); |
|
effectEntry.effect_name = effectEntry.effect_name.replaceAll("\"", ""); |
|
|
|
// Some effect mods have no icon |
|
// (SEEINVIS-SHADE "See Invis") |
|
|
|
if (effectHeader.size() == 3) |
|
effectEntry.icon = Integer.parseInt(effectHeader.get(2)); |
|
else |
|
effectEntry.icon = 0; |
|
|
|
// Parse source entries |
|
|
|
matcher = SOURCE_REGEX.matcher(effectData); |
|
|
|
while (matcher.find()) |
|
effectEntry.sources.add(matcher.group().trim()); |
|
|
|
// Parse modifier entries |
|
|
|
matcher = MODS_REGEX.matcher(effectData); |
|
|
|
// Iterate effect entries from .wpak config data |
|
|
|
while (matcher.find()) { |
|
EffectModifier effectModifier = parseModEntry(matcher.group()); |
|
effectEntry.mods.add(effectModifier); |
|
} |
|
|
|
// Parse Conditions |
|
|
|
matcher = CONDITIONS_REGEX.matcher(effectData); |
|
|
|
while (matcher.find()) { |
|
String[] conditions = matcher.group().trim().split("\n"); |
|
|
|
for (String condition : conditions) { |
|
List<String> parameters = Arrays.asList(condition.trim().split("\\s+")); |
|
Iterator<String> iterator = parameters.iterator(); |
|
|
|
ConditionEntry conditionEntry = new ConditionEntry(); |
|
conditionEntry.condition = iterator.next(); |
|
conditionEntry.arg = Integer.parseInt(iterator.next()); |
|
|
|
if (iterator.hasNext()) |
|
conditionEntry.curveType = mbEnums.CompoundCurveType.valueOf(iterator.next()); |
|
|
|
while (iterator.hasNext()) |
|
conditionEntry.damageTypes.add(mbEnums.DamageType.valueOf(iterator.next().toUpperCase())); |
|
|
|
effectEntry.conditions.add(conditionEntry); |
|
} |
|
} |
|
|
|
return effectEntry; |
|
} |
|
|
|
private static EffectModifier parseModEntry(String modData) { |
|
|
|
EffectModifier effectModifier = new EffectModifier(); |
|
|
|
String[] modEntries = modData.trim().split("\n"); |
|
|
|
for (String modEntry : modEntries) { |
|
|
|
ArrayList<String> modValues = new ArrayList<>(); |
|
Matcher matcher = STRSPLIT_REGEX.matcher(modEntry.trim()); |
|
|
|
while (matcher.find()) |
|
modValues.add(matcher.group().trim()); |
|
|
|
effectModifier.type = mbEnums.ModType.valueOf(modValues.get(0).trim()); |
|
|
|
switch (effectModifier.type) { |
|
case BladeTrails: // No parm modifiers |
|
case ImmuneToAttack: |
|
case ImmuneToPowers: |
|
case Ambidexterity: |
|
case Silenced: |
|
case IgnorePassiveDefense: |
|
case Stunned: |
|
case PowerCostHealth: |
|
case Charmed: |
|
case Fly: |
|
case CannotMove: |
|
case CannotTrack: |
|
case CannotAttack: |
|
case CannotCast: |
|
case SpireBlock: |
|
case Invisible: |
|
case SeeInvisible: |
|
break; |
|
case AnimOverride: |
|
effectModifier.min = Float.parseFloat(modValues.get(1).trim()); |
|
effectModifier.max = Float.parseFloat(modValues.get(2).trim()); |
|
break; |
|
case Health: |
|
case Mana: |
|
case Stamina: |
|
effectModifier.min = Float.parseFloat(modValues.get(1).trim()); |
|
effectModifier.max = Float.parseFloat(modValues.get(2).trim()); |
|
effectModifier.scale = Float.parseFloat(modValues.get(3).trim()); |
|
// Parameter 4 is always 0. |
|
effectModifier.compoundCurveType = mbEnums.CompoundCurveType.valueOf(modValues.get(5).trim()); |
|
effectModifier.arg1 = modValues.get(6).trim(); |
|
break; |
|
case Attr: |
|
case Resistance: |
|
case Skill: |
|
case HealthRecoverRate: |
|
case ManaRecoverRate: |
|
case StaminaRecoverRate: |
|
case DamageShield: |
|
case HealthFull: |
|
case ManaFull: |
|
case StaminaFull: |
|
case Slay: |
|
case Fade: |
|
case Durability: |
|
effectModifier.min = Float.parseFloat(modValues.get(1).trim()); |
|
effectModifier.scale = Float.parseFloat(modValues.get(2).trim()); |
|
effectModifier.compoundCurveType = mbEnums.CompoundCurveType.valueOf(modValues.get(3).trim()); |
|
|
|
if (modValues.size() > 4) |
|
effectModifier.arg1 = modValues.get(4).trim(); // Some HeathFull entries do not have an argument |
|
break; |
|
case MeleeDamageModifier: |
|
case OCV: |
|
case DCV: |
|
case AttackDelay: |
|
case AdjustAboveDmgCap: |
|
case DamageCap: |
|
case ArmorPiercing: |
|
case Speed: |
|
case PowerDamageModifier: |
|
case DR: |
|
case PassiveDefense: |
|
case MaxDamage: |
|
case Value: |
|
case WeaponSpeed: |
|
case MinDamage: |
|
case PowerCost: |
|
case Block: |
|
case Parry: |
|
case Dodge: |
|
case ScanRange: |
|
case ScaleHeight: |
|
case ScaleWidth: |
|
case WeaponRange: |
|
effectModifier.min = Float.parseFloat(modValues.get(1).trim()); |
|
effectModifier.scale = Float.parseFloat(modValues.get(2).trim()); |
|
effectModifier.compoundCurveType = mbEnums.CompoundCurveType.valueOf(modValues.get(3).trim()); |
|
break; |
|
case ItemName: |
|
case BlockedPowerType: |
|
case ImmuneTo: |
|
case BlackMantle: |
|
effectModifier.arg1 = modValues.get(1).trim(); |
|
|
|
// Some BlockedPowerType entries have only one argument |
|
|
|
if (modValues.size() > 2) |
|
effectModifier.arg2 = modValues.get(2).trim(); |
|
break; |
|
case NoMod: |
|
case ConstrainedAmbidexterity: |
|
case ProtectionFrom: |
|
case ExclusiveDamageCap: |
|
case IgnoreDamageCap: |
|
effectModifier.arg1 = modValues.get(1).trim(); |
|
break; |
|
case WeaponProc: |
|
effectModifier.min = Float.parseFloat(modValues.get(1).trim()); |
|
effectModifier.arg1 = modValues.get(2).trim(); |
|
effectModifier.scale = Float.parseFloat(modValues.get(3).trim()); |
|
break; |
|
default: |
|
Logger.error("Unhandled type: " + effectModifier.type); |
|
break; |
|
} |
|
} |
|
|
|
return effectModifier; |
|
} |
|
|
|
}
|
|
|