Pixel-Composer/scripts/ase_reader/ase_reader.gml

475 lines
No EOL
15 KiB
Text

/*
* ASE file reader
* File spec from: https://github.com/aseprite/aseprite/blob/main/docs/ase-file-specs.md
*/
enum _BIN_TYPE {
byte,
word,
short,
dword,
long,
fixed,
float,
double,
qword,
long64,
string,
point,
size,
rect,
color,
pixel,
}
//ASE blend mode
//Normal = 0
//Multiply = 1
//Screen = 2
//Overlay = 3
//Darken = 4
//Lighten = 5
//Color Dodge = 6
//Color Burn = 7
//Hard Light = 8
//Soft Light = 9
//Difference = 10
//Exclusion = 11
//Hue = 12
//Saturation = 13
//Color = 14
//Luminosity = 15
//Addition = 16
//Subtract = 17
//Divide = 18
globalvar __ase_format_header; __ase_format_header = [
[_BIN_TYPE.dword, "File size"],
[_BIN_TYPE.word, "Magic number"],
[_BIN_TYPE.word, "Frame amount"],
[_BIN_TYPE.word, "Width"],
[_BIN_TYPE.word, "Height"],
[_BIN_TYPE.word, "Color depth"], //32: RGBA, 16: Grey, 8: Index
[_BIN_TYPE.dword, "Flags"],
[_BIN_TYPE.word, "Speed"], //DEPRECATED
[_BIN_TYPE.dword, "0"],
[_BIN_TYPE.dword, "0"],
[_BIN_TYPE.byte, "Palette entry"], //For indexed sprite, index in palette that consider a transparent color.
[_BIN_TYPE.byte, "Ignore", 3],
[_BIN_TYPE.word, "Number of colors"],
[_BIN_TYPE.byte, "Pixel width"], //If zero, then pixel ratio is 1:1
[_BIN_TYPE.byte, "Pixel height"], //If zero, then pixel ratio is 1:1
[_BIN_TYPE.short, "Grid X"],
[_BIN_TYPE.short, "Grid Y"],
[_BIN_TYPE.word, "Grid width"], //If zero, no grid
[_BIN_TYPE.word, "Grid height"], //If zero, no grid
[_BIN_TYPE.byte, "Unused", 84],
];
globalvar __ase_format_frame; __ase_format_frame = [
[_BIN_TYPE.dword, "Length"],
[_BIN_TYPE.word, "Magic number"],
[_BIN_TYPE.word, "Chunk amount"], //If 0xFFFF, use "Chunk amount new"
[_BIN_TYPE.word, "Duration"], //In millisec
[_BIN_TYPE.byte, "Unused", 2],
[_BIN_TYPE.dword, "Chunk amount new"],
];
globalvar __ase_format_chunk; __ase_format_chunk = [
[_BIN_TYPE.dword, "Length"],
[_BIN_TYPE.word, "Type"],
];
globalvar __ase_format_chunk_old_palette; __ase_format_chunk_old_palette = [
[_BIN_TYPE.word, "Packet amount"],
];
globalvar __ase_format_chunk_old_palette_packet; __ase_format_chunk_old_palette_packet = [
[_BIN_TYPE.byte, "Entries skip index"],
[_BIN_TYPE.byte, "Color amount"],
[_BIN_TYPE.color, "Colors", "Color amount"],
];
globalvar __ase_format_chunk_layer; __ase_format_chunk_layer = [
[_BIN_TYPE.word, "Flag"], //1: Visible, 2: Editable, 4:Lock, 8:BG
[_BIN_TYPE.word, "Layer type"], //0: Normal, 1: Group, 2: Tilemap
[_BIN_TYPE.word, "Child level"],
[_BIN_TYPE.word, "Ignore"],
[_BIN_TYPE.word, "Ignore"],
[_BIN_TYPE.word, "Blend mode"],
[_BIN_TYPE.byte, "Opacity"],
[_BIN_TYPE.byte, "Unused", 3],
[_BIN_TYPE.string, "Name"],
[_BIN_TYPE.dword, "Tileset index", 1, function(chunk) { return chunk[$ "Layer type"] == 2; }],
];
globalvar __ase_format_chunk_cel; __ase_format_chunk_cel = [
[_BIN_TYPE.word, "Layer index"],
[_BIN_TYPE.short, "X"],
[_BIN_TYPE.short, "Y"],
[_BIN_TYPE.byte, "Opacity"],
[_BIN_TYPE.word, "Cel type"], //0: Raw image, 1: Linked, 2: Compressed image, 3: Compressed tilemap
[_BIN_TYPE.byte, "Unused", 7],
];
globalvar __ase_format_chunk_cel_raw_image; __ase_format_chunk_cel_raw_image = [
[_BIN_TYPE.word, "Width"],
[_BIN_TYPE.word, "Height"],
[_BIN_TYPE.pixel, "Pixels", function(chunk) { return chunk[$ "Width"] * chunk[$ "Width"]; }],
];
globalvar __ase_format_chunk_cel_linked; __ase_format_chunk_cel_linked = [
[_BIN_TYPE.word, "Frame position"],
];
globalvar __ase_format_chunk_cel_compress_image; __ase_format_chunk_cel_compress_image = [
[_BIN_TYPE.word, "Width"],
[_BIN_TYPE.word, "Height"],
//[_BIN_TYPE.long, "Raw cel", function(chunk) { return chunk[$ "Width"] * chunk[$ "Width"]; }],
];
globalvar __ase_format_chunk_cel_compress_tilemap; __ase_format_chunk_cel_compress_tilemap = [
[_BIN_TYPE.word, "Width"],
[_BIN_TYPE.word, "Height"],
[_BIN_TYPE.word, "Bits per tile"],
[_BIN_TYPE.dword, "Bitmask for tile ID"],
[_BIN_TYPE.dword, "X flip"],
[_BIN_TYPE.dword, "Y flip"],
[_BIN_TYPE.dword, "90CW rotation"],
[_BIN_TYPE.byte, "Unused", 10],
//[_BIN_TYPE.tile, "Tiles", function(chunk) { return chunk[$ "Width"] * chunk[$ "Width"]; }],
];
globalvar __ase_format_chunk_cel_extra; __ase_format_chunk_cel_extra = [
[_BIN_TYPE.dword, "Flag"],
[_BIN_TYPE.fixed, "X"],
[_BIN_TYPE.fixed, "Y"],
[_BIN_TYPE.fixed, "Width"],
[_BIN_TYPE.fixed, "Height"],
[_BIN_TYPE.byte, "Unused", 16],
];
globalvar __ase_format_chunk_color_profile; __ase_format_chunk_color_profile = [
[_BIN_TYPE.word, "Type"], //0: no profile, 1: sRGB, 2: ICC
[_BIN_TYPE.word, "Flag"], //1: Fix gamma
[_BIN_TYPE.fixed, "Fixed gamma"],
[_BIN_TYPE.byte, "Unused", 8],
[_BIN_TYPE.dword, "ICC Data length", 1, function(c) /*=>*/ {return c[$ "Type"] == 2}],
[_BIN_TYPE.byte, "ICC Data", "ICC Data length", function(c) /*=>*/ {return c[$ "Type"] == 2}],
];
globalvar __ase_format_chunk_file; __ase_format_chunk_file = [
[_BIN_TYPE.dword, "Entries"],
[_BIN_TYPE.byte, "Unused", 8],
];
globalvar __ase_format_chunk_file_entry; __ase_format_chunk_file_entry = [
[_BIN_TYPE.dword, "ID"],
[_BIN_TYPE.byte, "File type"], //0: External palette, 1: External tileset, 2: Extension anme
[_BIN_TYPE.byte, "Unused", 7],
[_BIN_TYPE.string, "File name"],
];
globalvar __ase_format_chunk_tag; __ase_format_chunk_tag = [
[_BIN_TYPE.word, "Tag amount"],
[_BIN_TYPE.byte, "Unused", 8],
];
globalvar __ase_format_chunk_tag_entry; __ase_format_chunk_tag_entry = [
[_BIN_TYPE.word, "Frame start"],
[_BIN_TYPE.word, "Frame end"],
[_BIN_TYPE.byte, "Loop"], //0: Forward, 1: Backward, 2: Ping pong, 3: Ping pong reverse
[_BIN_TYPE.word, "Repeat amount"], //0: Infinite, N: N-times
[_BIN_TYPE.byte, "Unused", 6],
[_BIN_TYPE.color, "Color"],
[_BIN_TYPE.byte, "Extra"],
[_BIN_TYPE.string, "Name"],
]
globalvar __ase_format_chunk_palette; __ase_format_chunk_palette = [
[_BIN_TYPE.dword, "Color amount"],
[_BIN_TYPE.dword, "First index"],
[_BIN_TYPE.dword, "Last index"],
[_BIN_TYPE.byte, "Unused", 8],
];
globalvar __ase_format_chunk_palette_entry; __ase_format_chunk_palette_entry = [
[_BIN_TYPE.word, "Flag"], //1: Has name
[_BIN_TYPE.byte, "Red"],
[_BIN_TYPE.byte, "Green"],
[_BIN_TYPE.byte, "Blue"],
[_BIN_TYPE.byte, "Alpha"],
[_BIN_TYPE.string, "Name", 1, function(c) /*=>*/ {return c[$ "Flag"] & (1 << 0)}],
];
globalvar __ase_format_chunk_user_data; __ase_format_chunk_user_data = [
[_BIN_TYPE.dword, "Flag"], //1: Text, 2: Color, 4: Properties
[_BIN_TYPE.string, "Name", 1, function(c) /*=>*/ {return c[$ "Flag"] & (1 << 0)} ],
[_BIN_TYPE.byte, "Red", 1, function(c) /*=>*/ {return c[$ "Flag"] & (1 << 1)} ],
[_BIN_TYPE.byte, "Green", 1, function(c) /*=>*/ {return c[$ "Flag"] & (1 << 1)} ],
[_BIN_TYPE.byte, "Blue", 1, function(c) /*=>*/ {return c[$ "Flag"] & (1 << 1)} ],
[_BIN_TYPE.byte, "Alpha", 1, function(c) /*=>*/ {return c[$ "Flag"] & (1 << 1)} ],
];
globalvar __ase_format_chunk_user_data_prop; __ase_format_chunk_user_data_prop = [
[_BIN_TYPE.dword, "Length"],
[_BIN_TYPE.dword, "Prop amount"],
]
/* TODO: Use data read */
globalvar __ase_format_chunk_slice; __ase_format_chunk_slice = [
[_BIN_TYPE.dword, "Slice key amount"],
[_BIN_TYPE.dword, "Flag"], //1: 9 slice, 2: pivot
[_BIN_TYPE.dword, "Reserved"],
[_BIN_TYPE.string, "Name"],
];
globalvar __ase_format_chunk_slice_key; __ase_format_chunk_slice_key = [
[_BIN_TYPE.dword, "Frame number"],
[_BIN_TYPE.long, "X"],
[_BIN_TYPE.long, "Y"],
[_BIN_TYPE.dword, "Width"],
[_BIN_TYPE.dword, "Height"],
];
globalvar __ase_format_chunk_slice_nine; __ase_format_chunk_slice_nine = [
[_BIN_TYPE.long, "Center X"],
[_BIN_TYPE.long, "Center Y"],
[_BIN_TYPE.dword, "Center width"],
[_BIN_TYPE.dword, "Center height"],
];
globalvar __ase_format_chunk_slice_pivot; __ase_format_chunk_slice_pivot = [
[_BIN_TYPE.long, "Pivot X"],
[_BIN_TYPE.long, "Pivot Y"],
];
globalvar __ase_format_chunk_tileset; __ase_format_chunk_tileset = [
[_BIN_TYPE.dword, "ID"],
[_BIN_TYPE.dword, "Flag"], //1: Link to external file, 2: Include tile in this file, 4: Use ID 0 as empty tiles.
[_BIN_TYPE.dword, "Tile amount"],
[_BIN_TYPE.word, "Tile width"],
[_BIN_TYPE.word, "Tile height"],
[_BIN_TYPE.short, "Base index"],
[_BIN_TYPE.byte, "Reserved", 14],
[_BIN_TYPE.string, "Name"],
[_BIN_TYPE.dword, "ID of external file", 1, function(c) /*=>*/ {return c[$ "Flag"] & (1 << 1)} ],
[_BIN_TYPE.dword, "Tileset ID", 1, function(c) /*=>*/ {return c[$ "Flag"] & (1 << 1)} ],
[_BIN_TYPE.dword, "Data length", 1, function(c) /*=>*/ {return c[$ "Flag"] & (1 << 2)} ],
[_BIN_TYPE.pixel, "Compressed image", "Data length", function(c) /*=>*/ {return c[$ "Flag"] & (1 << 2)} ],
];
function read_format_type(_bin, datType, outMap) {
switch(datType) {
case _BIN_TYPE.byte: return bin_read_byte(_bin);
case _BIN_TYPE.word: return bin_read_word(_bin);
case _BIN_TYPE.short: return bin_read_short(_bin);
case _BIN_TYPE.dword: return bin_read_dword(_bin);
case _BIN_TYPE.long: return bin_read_long(_bin);
case _BIN_TYPE.fixed: return bin_read_fixed(_bin);
case _BIN_TYPE.float: return bin_read_float(_bin);
case _BIN_TYPE.double: return bin_read_double(_bin);
case _BIN_TYPE.qword: return bin_read_qword(_bin);
case _BIN_TYPE.long64: return bin_read_long64(_bin);
case _BIN_TYPE.string: return bin_read_string(_bin);
case _BIN_TYPE.point: return bin_read_point(_bin);
case _BIN_TYPE.size: return bin_read_size(_bin);
case _BIN_TYPE.rect: return bin_read_rect(_bin);
case _BIN_TYPE.color: return bin_read_color(_bin);
case _BIN_TYPE.pixel: return bin_read_pixel(_bin, outMap[$ "Color depth"]);
}
return 0;
}
function read_format(_bin, format, outMap) {
var datType = array_safe_get_fast(format, 0, 0);
var key = array_safe_get_fast(format, 1, "");
var amount = array_safe_get_fast(format, 2, 1);
if(is_string(amount)) amount = struct_has(outMap, amount)? outMap[$ amount] : 1;
else if(is_method(amount)) amount = amount(outMap);
if(amount == 1) {
var val = read_format_type(_bin, datType, outMap);
outMap[$ key] = val;
return val;
} else {
var a = array_create(amount);
for( var i = 0; i < amount; i++ )
a[i] = read_format_type(_bin, datType, outMap);
outMap[$ key] = a;
return a;
}
}
function read_format_array(_bin, formatArr, outMap) {
for( var i = 0, n = array_length(formatArr); i < n; i++ ) {
var _form = formatArr[i];
if(array_length(_form) >= 4 && !_form[3](outMap))
continue;
var pos = file_bin_position(_bin);
var val = read_format(_bin, _form, outMap);
//printIf(global.FLAG.ase_import, "Pos " + dec_to_hex(pos) + " - " + dec_to_hex(file_bin_position(_bin)));
if(_form[1] == "Type") printIf(global.FLAG.ase_import, $"\t{_form[1]}:\t 0x{dec_to_hex(val, 4)}");
else printIf(global.FLAG.ase_import, $"\t{_form[1]}:\t {string(val)}");
}
}
function read_ase(path) { //// MAIN READ
printIf(global.FLAG.ase_import, $"===== Reading: {path} =====");
var file = file_bin_open(path, 0);
if(file < 0) { noti_warning($"ASE file read error."); return noone; }
file_bin_seek(file, 0);
fileMap = {};
read_format_array(file, __ase_format_header, fileMap);
var frames = [];
var frameAmo = struct_try_get(fileMap, "Frame amount", 0);
for( var i = 0; i < frameAmo; i++ ) {
printIf(global.FLAG.ase_import, $"\n=== Reading frame {i} ===");
array_push(frames, read_ase_frame(file));
}
fileMap[$ "Frames"] = frames;
file_bin_close(file);
return fileMap;
}
function read_ase_frame(file) {
var frame = {};
read_format_array(file, __ase_format_frame, frame);
var chunks = [];
var chunkAmo = struct_try_get(frame, "Chunk amount", 0);
if(chunkAmo == 0xFFFF)
chunkAmo = struct_try_get(frame, "Chunk amount new", chunkAmo);
for( var i = 0; i < chunkAmo; i++ ) {
printIf(global.FLAG.ase_import, $"\n=== Reading chunk {i} ===");
array_push(chunks, read_ase_chunk(file));
}
frame[$ "Chunks"] = chunks;
return frame;
}
function read_ase_chunk(file) {
var chunk = {};
var startPos = file_bin_position(file);
read_format_array(file, __ase_format_chunk, chunk);
var skipPos = startPos + chunk[$ "Length"];
switch(chunk[$ "Type"]) {
case 0x0004: //old palette
case 0x0011: //old palette
printIf(global.FLAG.ase_import, "\n -- Reading chunk [Old palette] -- ");
read_format_array(file, __ase_format_chunk_old_palette, chunk);
var cc = [];
for( var i = 0; i < chunk[$ "Packet amount"]; i++ ) {
cc[i] = {};
read_format_array(file, __ase_format_chunk_old_palette_packet, cc[i]);
}
chunk[$ "Packets"] = cc;
break;
case 0x2004: //layer
printIf(global.FLAG.ase_import, "\n -- Reading chunk [Layer] -- ");
read_format_array(file, __ase_format_chunk_layer, chunk);
break;
case 0x2005: //cel
printIf(global.FLAG.ase_import, "\n -- Reading chunk [Cel] -- ");
read_format_array(file, __ase_format_chunk_cel, chunk);
var type = chunk[$ "Cel type"];
switch(type) {
case 0 : read_format_array(file, __ase_format_chunk_cel_raw_image, chunk); break;
case 1 : read_format_array(file, __ase_format_chunk_cel_linked, chunk); break;
case 2 :
read_format_array(file, __ase_format_chunk_cel_compress_image, chunk);
chunk[$ "Surface"] = noone;
var compressLength = (skipPos - file_bin_position(file));
var _compBuff = buffer_create(compressLength * buffer_sizeof(buffer_u8), buffer_grow, 1);
buffer_seek(_compBuff, buffer_seek_start, 0);
repeat(compressLength) {
var byte = file_bin_read_byte(file);
buffer_write(_compBuff, buffer_u8, byte);
}
var _rawBuff = buffer_decompress(_compBuff);
if(_rawBuff != -1) chunk[$ "Buffer"] = _rawBuff;
printIf(global.FLAG.ase_import, $" Buffer size: {compressLength}");
buffer_delete(_compBuff);
break;
case 3 :
read_format_array(file, __ase_format_chunk_cel_compress_tilemap, chunk);
//TILE READ
break;
}
break;
case 0x2006: break; //cel extra
case 0x2007: //color profile
printIf(global.FLAG.ase_import, "\n -- Reading chunk [Color profile] -- ");
read_format_array(file, __ase_format_chunk_color_profile, chunk);
break;
case 0x2008: break; //external file
case 0x2009: break; //mask DEPRECATED
case 0x2017: break; //path
case 0x2018: //tag
printIf(global.FLAG.ase_import, "\n -- Reading chunk [Tag] -- ");
read_format_array(file, __ase_format_chunk_tag, chunk);
var amo = chunk[$ "Tag amount"]
var tags = [];
repeat(amo) {
var m = {};
read_format_array(file, __ase_format_chunk_tag_entry, m);
array_push(tags, m);
}
chunk[$ "Tags"] = tags;
break;
case 0x2019: //palette
printIf(global.FLAG.ase_import, "\n -- Reading chunk [Palette] -- ");
read_format_array(file, __ase_format_chunk_palette, chunk);
var cc = [];
for( var i = 0; i < chunk[$ "Color amount"]; i++ ) {
cc[i] = {};
read_format_array(file, __ase_format_chunk_palette_entry, cc[i]);
}
chunk[$ "Palette"] = cc;
break;
case 0x2020: break; //user data
case 0x2022: break; //slice
case 0x2023: break; //tileset
}
file_bin_seek(file, skipPos - file_bin_position(file));
return chunk;
}