papermario/tools/splat_ext/PaperMarioMessages.py

466 lines
12 KiB
Python
Raw Normal View History

2021-01-24 15:45:43 +01:00
from segtypes.n64.segment import N64Segment
2020-11-11 22:21:25 +01:00
from pathlib import Path
2021-02-10 01:12:28 +01:00
import re
2020-11-11 22:21:25 +01:00
CHARSET = {
2021-02-10 01:12:28 +01:00
0x00: "[note]",
2020-11-11 22:21:25 +01:00
0x01: "!",
0x02: '"',
0x03: "#",
0x04: "$",
0x05: "%",
0x06: "&",
0x07: "'",
0x08: "(",
0x09: ")",
0x0A: "*",
0x0B: "+",
0x0C: ",",
0x0D: "-",
0x0E: ".",
0x0F: "/",
0x10: "0",
0x11: "1",
0x12: "2",
0x13: "3",
0x14: "4",
0x15: "5",
0x16: "6",
0x17: "7",
0x18: "8",
0x19: "9",
0x1A: ":",
0x1B: ";",
0x1C: "<",
0x1D: "=",
0x1E: ">",
0x1F: "?",
0x20: "@",
0x21: "A",
0x22: "B",
0x23: "C",
0x24: "D",
0x25: "E",
0x26: "F",
0x27: "G",
0x28: "H",
0x29: "I",
0x2A: "J",
0x2B: "K",
0x2C: "L",
0x2D: "M",
0x2E: "N",
0x2F: "O",
0x30: "P",
0x31: "Q",
0x32: "R",
0x33: "S",
0x34: "T",
0x35: "U",
0x36: "V",
0x37: "W",
0x38: "X",
0x39: "Y",
0x3A: "Z",
0x3B: "\\[",
0x3C: "¥",
0x3D: "]",
0x3E: "^",
0x3F: "_",
0x40: "`",
0x41: "a",
0x42: "b",
0x43: "c",
0x44: "d",
0x45: "e",
0x46: "f",
0x47: "g",
0x48: "h",
0x49: "i",
0x4A: "j",
0x4B: "k",
0x4C: "l",
0x4D: "m",
0x4E: "n",
0x4F: "o",
0x50: "p",
0x51: "q",
0x52: "r",
0x53: "s",
0x54: "t",
0x55: "u",
0x56: "v",
0x57: "w",
0x58: "x",
0x59: "y",
0x5A: "z",
0x5B: "{",
0x5C: "|",
0x5D: "}",
0x5E: "~",
0x5F: "°",
0x60: "À",
0x61: "Á",
0x62: "Â",
0x63: "Ä",
0x64: "Ç",
0x65: "È",
0x66: "É",
0x67: "Ê",
0x68: "Ë",
0x69: "Ì",
0x6A: "Í",
0x6B: "Î",
0x6C: "Ï",
0x6D: "Ñ",
0x6E: "Ò",
0x6F: "Ó",
0x70: "Ô",
0x71: "Ö",
0x72: "Ù",
0x73: "Ú",
0x74: "Û",
0x75: "Ü",
0x76: "ß",
0x77: "à",
0x78: "á",
0x79: "â",
0x7A: "ä",
0x7B: "ç",
0x7C: "è",
0x7D: "é",
0x7E: "ê",
0x7F: "ë",
0x80: "ì",
0x81: "í",
0x82: "î",
0x83: "ï",
0x84: "ñ",
0x85: "ò",
0x86: "ó",
0x87: "ô",
0x88: "ö",
0x89: "ù",
0x8A: "ú",
0x8B: "û",
0x8C: "ü",
0x8D: "¡",
0x8E: "¿",
0x8F: "ª",
2021-02-10 01:12:28 +01:00
0x90: "[heart]",
0x91: "[star]",
0x92: "[up]",
0x93: "[down]",
0x94: "[left]",
0x95: "[right]",
0x96: "[circle]",
0x97: "[cross]",
0x98: "[a]",
0x99: "[b]",
0x9A: "[l]",
0x9B: "[r]",
0x9C: "[z]",
0x9D: "[c-up]",
0x9E: "[c-down]",
0x9F: "[c-left]",
0xA0: "[c-right]",
0xA1: "[start]",
2020-11-11 22:21:25 +01:00
0xA2: "",
0xA3: "",
0xA4: "",
0xA5: "",
0xF7: " ",
0xF0: "[br]\n",
2021-02-10 01:12:28 +01:00
0xF1: "[wait]",
0xF2: {None: lambda d: (f"[pause:{d[0]}]", 1)},
2020-11-11 22:21:25 +01:00
0xFB: "[next]\n",
0xFC: {
2021-02-10 01:12:28 +01:00
0x01: "[style:right]\n",
0x02: "[style:left]\n",
0x03: "[style:center]\n",
0x04: "[style:tattle]\n",
0x05: {None: lambda d: (f"[style:choice:{d[0]}:{d[1]}:{d[2]}:{d[3]}]\n", 4)},
0x06: "[style:inspect]\n",
0x07: "[style:sign]\n",
0x08: {None: lambda d: (f"[style:lamppost:{d[0]}]\n", 1)},
0x09: {None: lambda d: (f"[style:postcard:{d[0]}]\n", 1)},
0x0A: "[style:popup]\n",
0x0C: {None: lambda d: (f"[style:upgrade:{d[0]}:{d[1]}:{d[2]}:{d[3]}]\n", 4)},
0x0D: "[style:narrate]\n",
0x0E: "[style:epilogue]\n",
2020-11-11 22:21:25 +01:00
},
0xFF: {
0x00: {
2021-02-10 01:12:28 +01:00
0: "[font:normal]",
3: "[font:title]\n",
4: "[font:subtitle]\n",
2020-11-11 22:21:25 +01:00
},
2021-02-10 01:12:28 +01:00
0x04: "[func_04]",
2020-11-11 22:21:25 +01:00
0x05: {
2021-02-10 01:12:28 +01:00
# 0x0A: "[color:normal]",
# 0x20: "[color:red]",
# 0x21: "[color:pink]",
# 0x22: "[color:purple]",
# 0x23: "[color:blue]",
# 0x24: "[color:cyan]",
# 0x25: "[color:green]",
# 0x26: "[color:yellow]",
# 0x00: "[color=normal ctx=diary]",
# 0x07: "[color=red ctx=diary]",
# 0x17: "[color=dark ctx=inspect]",
# 0x18: "[color=normal ctx=sign]",
# 0x19: "[color=red ctx=sign]",
# 0x1A: "[color=blue ctx=sign]",
# 0x1B: "[color=green ctx=sign]",
# 0x28: "[color=red ctx=popup]",
# 0x29: "[color=pink ctx=popup]",
# 0x2A: "[color=purple ctx=popup]",
# 0x2B: "[color=blue ctx=popup]",
# 0x2C: "[color=teal ctx=popup]",
# 0x2D: "[color=green ctx=popup]",
# 0x2E: "[color=yellow ctx=popup]",
# 0x2F: "[color=normal ctx=popup]",
None: lambda d: (f"[color:0x{d[0]:X}]", 1),
2020-11-11 22:21:25 +01:00
},
2021-02-10 01:12:28 +01:00
0x07: "[inputOff]\n",
0x08: "[inputOn]\n",
0x09: "[delayOff]\n",
0x0A: "[delayOn]\n",
0x0B: {None: lambda d: (f"[kerning:{d[0]}]", 1)},
0x0C: {None: lambda d: (f"[scroll:{d[0]}]", 1)},
0x0D: {None: lambda d: (f"[size:{d[0]}:{d[0]}]\n", 2)},
0x0E: "[sizeReset]\n",
0x0F: {None: lambda d: (f"[speed:{d[0]}:{d[1]}]", 2)},
0x10: {None: lambda d: (f"[setPrintPos:{d[0]}:{d[1]}]", 2)},
0x11: {None: lambda d: (f"[setPrintY:{d[0]}]", 1)},
0x12: {None: lambda d: (f"[indent:{d[0]}]", 1)},
0x13: {None: lambda d: (f"[down:{d[0]}]", 1)},
0x14: {None: lambda d: (f"[up:{d[0]}]", 1)},
0x15: {None: lambda d: (f"[image1:{d[0]}]\n", 1)},
0x16: {None: lambda d: (f"[sprite:{d[0]}:{d[1]}:{d[2]}]\n", 3)},
0x17: {None: lambda d: (f"[item:{d[0]}:{d[1]}]\n", 2)},
0x18: {None: lambda d: (f"[image7:{d[0]}:{d[1]}:{d[2]}:{d[3]}:{d[4]}:{d[5]}:{d[6]}]\n", 7)},
0x1A: {None: lambda d: (f"[func_1A:{d[0]}:{d[1]}:{d[2]}]", 3)},
0x1B: {None: lambda d: (f"[func_1B:{d[0]}:{d[1]}]", 2)},
0x1C: {None: lambda d: (f"[func_1C:{d[0]}]", 1)},
0x1E: {None: lambda d: (f"[cursor:{d[0]}]", 1)},
0x1F: {None: lambda d: (f"[endChoice:{d[0]}]", 1)},
0x20: {None: lambda d: (f"[setCancel:{d[0]}]", 1)},
0x21: {None: lambda d: (f"[option:{d[0]}]", 1)},
0x22: "[startAnim]",
0x23: "[endAnim]",
# 0x24: {0xFF: {0x05: {
# 0x10: {0x98: {0xFF: {0x25: "Ⓐ"}}},
# 0x11: {0x99: {0xFF: {0x25: "Ⓑ"}}},
# 0x12: {0xA1: {0xFF: {0x25: "Ⓢ"}}},
# 0x13: {
# 0x9D: {0xFF: {0x25: "▲"}},
# 0x9E: {0xFF: {0x25: "▼"}},
# 0x9F: {0xFF: {0x25: "◀"}},
# 0xA0: {0xFF: {0x25: "▶"}},
# },
# 0x14: {0x9C: {0xFF: {0x25: "Ⓩ"}}},
# }}},
0x24: "[pushColor]",
0x25: "[popColor]",
2020-11-11 22:21:25 +01:00
0x26: {
2021-02-10 01:12:28 +01:00
0x00: "[startFX:jitter]",
0x01: "[startFX:wavy]",
0x03: {None: lambda d: (f"[startFX:fadedNoise:{d[0]}]", 1)},
0x05: {None: lambda d: (f"[startFX:fadedJitter:{d[0]}]", 1)},
0x07: {None: lambda d: (f"[startFX:faded:{d[0]}]", 1)},
0x0A: "[startFX:shrinking]",
0x0B: "[startFX:growing]",
0x0C: "[startFX:sizeJitter]",
0x0D: "[startFX:sizeWave]",
0x0E: "[startFX:dropShadow]",
2020-11-11 22:21:25 +01:00
},
0x27: {
2021-02-10 01:12:28 +01:00
0x00: "[endFX:jitter]",
0x01: "[endFX:wavy]",
0x03: "[endFX:fadedNoise]",
0x05: "[endFX:fadedJitter]",
0x07: "[endFX:faded]",
0x0A: "[endFX:shrinking]",
0x0B: "[endFX:growing]",
0x0C: "[endFX:sizeJitter]",
0x0D: "[endFX:sizeWave]",
0x0E: "[endFX:dropShadow]",
2020-11-11 22:21:25 +01:00
},
2021-02-10 01:12:28 +01:00
0x28: {None: lambda d: (f"[var:{d[0]}]", 1)},
0x29: {None: lambda d: (f"[func_29:{d[0]}]", 1)},
0x2B: "[func_2B]",
0x2E: {None: lambda d: (f"[volume:{d[0]}]", 1)},
2020-11-11 22:21:25 +01:00
0x2F: {
2021-02-10 01:12:28 +01:00
#1: "[speechSound:bowser]\n",
#2: "[speechSound:star]\n",
None: lambda d: (f"[speechSound:{d[0]}]\n", 1),
2020-11-11 22:21:25 +01:00
},
2021-02-10 01:12:28 +01:00
#None: lambda d: (f"[func_{d[0]:02X}]", 1),
2020-11-11 22:21:25 +01:00
},
2021-02-10 01:12:28 +01:00
None: lambda d: (f"[raw:0x{d[0]:02X}]", 1),
2020-11-11 22:21:25 +01:00
}
CHARSET_CREDITS = {
**CHARSET,
0x00: "A",
0x01: "B",
0x02: "C",
0x03: "D",
0x04: "E",
0x05: "F",
0x06: "G",
0x07: "H",
0x08: "I",
0x09: "J",
0x0A: "K",
0x0B: "L",
0x0C: "M",
0x0D: "N",
0x0E: "O",
0x0F: "P",
0x10: "Q",
0x11: "R",
0x12: "S",
0x13: "T",
0x14: "U",
0x15: "V",
0x16: "W",
0x17: "X",
0x18: "Y",
0x19: "Z",
0x1A: "'",
0x1B: ".",
0x1C: ",",
0x1D: "0",
0x1E: "1",
0x1F: "2",
0x20: "3",
0x21: "4",
0x22: "5",
0x23: "6",
0x24: "7",
0x25: "8",
0x26: "9",
0x27: "©",
0x28: "&",
0xF7: " ",
}
class N64SegPaperMarioMessages(N64Segment):
def __init__(self, segment, next_segment, options):
super().__init__(segment, next_segment, options)
self.files = segment.get("files", []) if type(segment) is dict else []
def split(self, rom_bytes, base_path):
data = rom_bytes[self.rom_start: self.rom_end]
section_offsets = []
pos = 0
while True:
offset = int.from_bytes(data[pos:pos+4], byteorder="big")
if offset == 0:
break
section_offsets.append(offset)
pos += 4
2021-02-10 01:12:28 +01:00
msg_dir = Path(base_path, self.options["assets_dir"], self.name)
msg_dir.mkdir(parents=True, exist_ok=True)
# delete existing files
self.delete_dir_childs(msg_dir)
2020-11-11 22:21:25 +01:00
for i, section_offset in enumerate(section_offsets):
name = f"{i:02X}"
if len(self.files) >= i:
name = self.files[i]
msg_offsets = []
pos = section_offset
while True:
offset = int.from_bytes(data[pos:pos+4], byteorder="big")
if offset == section_offset:
break
msg_offsets.append(offset)
pos += 4
self.log(f"Reading {len(msg_offsets)} messages in section {name} (0x{i:02X})")
2021-02-10 01:12:28 +01:00
path = msg_dir / Path(name + ".msg")
2020-11-18 11:37:02 +01:00
with open(path, "w") as self.f:
2020-11-11 22:21:25 +01:00
for j, msg_offset in enumerate(msg_offsets):
if j != 0:
2020-11-18 11:37:02 +01:00
self.f.write("\n")
2021-02-10 01:12:28 +01:00
self.f.write(f"#message:{i:02X}:{j:03X} {{\n ")
2020-11-18 11:37:02 +01:00
self.write_message_markup(data[msg_offset:])
2021-02-10 01:12:28 +01:00
self.f.write("\n}\n")
@staticmethod
def delete_dir_childs(path):
for f in path.iterdir():
if f.is_dir():
N64SegPaperMarioMessages.delete_dir_childs(f)
f.rmdir()
else:
f.unlink()
2020-11-11 22:21:25 +01:00
def get_ld_files(self):
2021-02-06 04:36:46 +01:00
return [(self.options["assets_dir"], self.name, ".data", self.rom_start)]
2020-11-11 22:21:25 +01:00
@staticmethod
def get_default_name(addr):
return "msg"
2020-11-18 11:37:02 +01:00
def write_message_markup(self, data):
2020-11-11 22:21:25 +01:00
pos = 0
2020-11-18 11:56:56 +01:00
self.root_charset = CHARSET
2020-11-11 22:21:25 +01:00
while data[pos] != 0xFD:
2020-11-18 11:56:56 +01:00
self.charset = self.root_charset
2020-11-11 22:21:25 +01:00
2020-11-18 11:37:02 +01:00
while True:
char = data[pos]
2020-11-11 22:21:25 +01:00
2020-11-18 11:37:02 +01:00
if char in self.charset:
value = self.charset[char]
elif None in self.charset:
value = self.charset[None]
2020-11-11 22:21:25 +01:00
2020-11-18 11:37:02 +01:00
if value is None:
value = fallback
2020-11-11 22:21:25 +01:00
2020-11-18 11:37:02 +01:00
if isinstance(value, str):
self.write_markup(value)
pos += 1
break
elif callable(value):
2020-11-18 12:16:23 +01:00
markup, delta = value(data[pos:])
2020-11-18 11:37:02 +01:00
self.write_markup(markup)
pos += delta
break
elif isinstance(value, dict):
if None in self.charset:
fallback = self.charset[None]
self.charset = value
pos += 1
else:
raise ValueError(value)
2020-11-11 22:21:25 +01:00
2020-11-18 11:37:02 +01:00
def write_markup(self, markup):
2021-02-10 01:12:28 +01:00
self.f.write(re.sub("\n", "\n ", markup))
2020-11-11 22:21:25 +01:00
2021-02-10 01:12:28 +01:00
if markup == "[font:title]\n" or markup == "[font:subtitle]\n":
2020-11-18 11:56:56 +01:00
self.root_charset = CHARSET_CREDITS
2021-02-10 01:12:28 +01:00
elif markup == "[font:normal]":
2020-11-18 11:56:56 +01:00
self.root_charset = CHARSET