2023-02-19 02:13:19 +01:00
|
|
|
//draw
|
2023-03-29 15:02:03 +02:00
|
|
|
function draw_surface_safe(surface, _x = 0, _y = 0) {
|
2022-11-22 14:25:39 +01:00
|
|
|
if(!is_surface(surface)) return;
|
2023-03-19 09:17:39 +01:00
|
|
|
|
|
|
|
__channel_pre(surface);
|
|
|
|
draw_surface(surface, _x, _y);
|
|
|
|
__channel_pos(surface);
|
2022-01-13 05:24:03 +01:00
|
|
|
}
|
2023-01-25 06:49:00 +01:00
|
|
|
function draw_surface_stretched_safe(surface, _x, _y, _w, _h) {
|
|
|
|
if(!is_surface(surface)) return;
|
2023-03-19 09:17:39 +01:00
|
|
|
|
|
|
|
__channel_pre(surface);
|
|
|
|
draw_surface_stretched(surface, _x, _y, _w, _h);
|
|
|
|
__channel_pos(surface);
|
2023-01-25 06:49:00 +01:00
|
|
|
}
|
2022-11-22 14:25:39 +01:00
|
|
|
function draw_surface_ext_safe(surface, _x, _y, _xs = 1, _ys = 1, _rot = 0, _col = c_white, _alpha = 1) {
|
|
|
|
if(!is_surface(surface)) return;
|
2023-03-19 09:17:39 +01:00
|
|
|
|
|
|
|
__channel_pre(surface);
|
|
|
|
draw_surface_ext(surface, _x, _y, _xs, _ys, _rot, _col, _alpha);
|
|
|
|
__channel_pos(surface);
|
|
|
|
}
|
|
|
|
function draw_surface_tiled_safe(surface, _x, _y) {
|
|
|
|
if(!is_surface(surface)) return;
|
|
|
|
|
|
|
|
__channel_pre(surface);
|
|
|
|
draw_surface_tiled(surface, _x, _y);
|
|
|
|
__channel_pos(surface);
|
2022-01-13 05:24:03 +01:00
|
|
|
}
|
2022-11-22 14:25:39 +01:00
|
|
|
function draw_surface_tiled_ext_safe(surface, _x, _y, _xs = 1, _ys = 1, _col = c_white, _alpha = 1) {
|
|
|
|
if(!is_surface(surface)) return;
|
2023-03-19 09:17:39 +01:00
|
|
|
|
|
|
|
__channel_pre(surface);
|
|
|
|
draw_surface_tiled_ext(surface, _x, _y, _xs, _ys, _col, _alpha);
|
|
|
|
__channel_pos(surface);
|
2022-01-13 05:24:03 +01:00
|
|
|
}
|
2022-11-22 14:25:39 +01:00
|
|
|
function draw_surface_part_ext_safe(surface, _l, _t, _w, _h, _x, _y, _xs = 1, _ys = 1, _rot = 0, _col = c_white, _alpha = 1) {
|
|
|
|
if(!is_surface(surface)) return;
|
2023-03-19 09:17:39 +01:00
|
|
|
|
|
|
|
__channel_pre(surface);
|
|
|
|
draw_surface_part_ext(surface, _l, _t, _w, _h, _x, _y, _xs, _ys, _col, _alpha);
|
|
|
|
__channel_pos(surface);
|
|
|
|
}
|
|
|
|
|
|
|
|
function surface_save_safe(surface, path) {
|
|
|
|
if(!is_surface(surface)) return;
|
|
|
|
var f = surface_get_format(surface);
|
2023-04-10 20:02:59 +02:00
|
|
|
|
|
|
|
if(f == surface_rgba8unorm) {
|
|
|
|
surface_save(surface, path);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
var w = surface_get_width(surface);
|
|
|
|
var h = surface_get_height(surface);
|
2023-03-31 06:59:08 +02:00
|
|
|
var s = surface_create(w, h, surface_rgba8unorm);
|
2023-03-19 09:17:39 +01:00
|
|
|
|
|
|
|
switch(f) {
|
|
|
|
case surface_rgba4unorm :
|
|
|
|
case surface_rgba8unorm :
|
|
|
|
case surface_rgba16float :
|
|
|
|
case surface_rgba32float :
|
2023-03-31 06:59:08 +02:00
|
|
|
surface_set_shader(s, sh_draw_normal);
|
|
|
|
draw_surface(surface, 0, 0);
|
|
|
|
surface_reset_shader();
|
|
|
|
surface_save(s, path);
|
2023-03-19 09:17:39 +01:00
|
|
|
return;
|
|
|
|
case surface_r8unorm :
|
|
|
|
s = surface_create(w, h, surface_rgba8unorm);
|
|
|
|
break;
|
|
|
|
case surface_r16float :
|
|
|
|
s = surface_create(w, h, surface_rgba16float);
|
|
|
|
break;
|
|
|
|
case surface_r32float :
|
|
|
|
s = surface_create(w, h, surface_rgba32float);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-03-31 06:59:08 +02:00
|
|
|
surface_set_shader(s, sh_draw_single_channel);
|
|
|
|
draw_surface(surface, 0, 0);
|
|
|
|
surface_reset_shader();
|
2023-03-19 09:17:39 +01:00
|
|
|
|
|
|
|
surface_save(s, path);
|
|
|
|
surface_free(s);
|
|
|
|
return;
|
2022-08-30 07:36:37 +02:00
|
|
|
}
|
2022-01-13 05:24:03 +01:00
|
|
|
|
2023-02-19 02:13:19 +01:00
|
|
|
//check
|
|
|
|
function is_surface(s) {
|
2023-04-16 16:24:17 +02:00
|
|
|
gml_pragma("forceinline");
|
|
|
|
|
2023-02-28 09:43:01 +01:00
|
|
|
if(is_undefined(s)) return false;
|
2023-02-19 02:13:19 +01:00
|
|
|
if(is_array(s)) return false;
|
|
|
|
if(!is_real(s)) return false;
|
|
|
|
if(!s) return false;
|
|
|
|
if(!surface_exists(s)) return false;
|
2022-01-13 05:24:03 +01:00
|
|
|
|
2023-02-19 02:13:19 +01:00
|
|
|
if(surface_get_width(s) <= 0) return false;
|
|
|
|
if(surface_get_height(s) <= 0) return false;
|
2022-01-13 05:24:03 +01:00
|
|
|
|
2022-11-22 14:25:39 +01:00
|
|
|
return true;
|
2022-01-13 05:24:03 +01:00
|
|
|
}
|
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
function surface_verify(surf, w, h, format = surface_rgba8unorm) {
|
2023-04-16 16:24:17 +02:00
|
|
|
gml_pragma("forceinline");
|
|
|
|
|
2023-02-19 02:13:19 +01:00
|
|
|
if(!is_surface(surf))
|
2023-03-19 09:17:39 +01:00
|
|
|
return surface_create_valid(w, h, format);
|
|
|
|
return surface_size_to(surf, w, h, format);
|
2022-12-10 05:06:01 +01:00
|
|
|
}
|
|
|
|
|
2023-03-31 06:59:08 +02:00
|
|
|
//get
|
|
|
|
function surface_get_pixel(surface, _x, _y) {
|
|
|
|
if(!is_surface(surface)) return;
|
|
|
|
var f = surface_get_format(surface);
|
|
|
|
var px = surface_getpixel(surface, _x, _y);
|
|
|
|
|
|
|
|
if(is_real(px)) return px;
|
|
|
|
return round(px[0] * (255 * power(256, 0))) + round(px[1] * (255 * power(256, 1))) + round(px[2] * (255 * power(256, 2)));
|
|
|
|
}
|
|
|
|
|
|
|
|
function surface_get_pixel_ext(surface, _x, _y) {
|
|
|
|
if(!is_surface(surface)) return;
|
|
|
|
var f = surface_get_format(surface);
|
|
|
|
var px = surface_getpixel_ext(surface, _x, _y);
|
|
|
|
|
|
|
|
if(is_real(px)) return px;
|
|
|
|
return round(px[0] * (255 * power(256, 0))) + round(px[1] * (255 * power(256, 1))) + round(px[2] * (255 * power(256, 2))) + round(px[3] * (255 * power(256, 3)));
|
|
|
|
}
|
|
|
|
|
2023-02-19 02:13:19 +01:00
|
|
|
//create
|
2023-03-19 09:17:39 +01:00
|
|
|
function surface_create_size(surface, format = surface_rgba8unorm) {
|
|
|
|
var s = surface_create_valid(surface_get_width(surface), surface_get_height(surface), format);
|
2022-01-13 05:24:03 +01:00
|
|
|
surface_set_target(s);
|
2023-03-19 09:17:39 +01:00
|
|
|
DRAW_CLEAR
|
2022-01-13 05:24:03 +01:00
|
|
|
surface_reset_target();
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
function surface_create_valid(w, h, format = surface_rgba8unorm) {
|
|
|
|
var s = surface_create(surface_valid_size(w), surface_valid_size(h), format);
|
2023-02-19 02:13:19 +01:00
|
|
|
surface_set_target(s);
|
2023-03-19 09:17:39 +01:00
|
|
|
DRAW_CLEAR
|
2022-01-13 05:24:03 +01:00
|
|
|
surface_reset_target();
|
2023-02-19 02:13:19 +01:00
|
|
|
return s;
|
2022-01-13 05:24:03 +01:00
|
|
|
}
|
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
function surface_create_from_buffer(w, h, buff, format = surface_rgba8unorm) {
|
|
|
|
var s = surface_create_valid(surface_valid_size(w), surface_valid_size(h), format);
|
2023-02-19 02:13:19 +01:00
|
|
|
buffer_set_surface(buff, s, 0);
|
|
|
|
return s;
|
2022-11-01 03:06:03 +01:00
|
|
|
}
|
|
|
|
|
2023-02-19 02:13:19 +01:00
|
|
|
function surface_create_from_sprite(spr) {
|
|
|
|
if(!sprite_exists(spr)) return noone;
|
2022-01-13 05:24:03 +01:00
|
|
|
|
2023-02-19 02:13:19 +01:00
|
|
|
if(sprite_get_number(spr) == 1)
|
|
|
|
return surface_create_from_sprite_ext(spr, 0);
|
2022-01-13 05:24:03 +01:00
|
|
|
|
2023-02-19 02:13:19 +01:00
|
|
|
var s = [];
|
|
|
|
for( var i = 0; i < sprite_get_number(spr); i++ ) {
|
|
|
|
array_push(s, surface_create_from_sprite_ext(spr, i));
|
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
2022-09-27 06:37:28 +02:00
|
|
|
}
|
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
function surface_create_from_sprite_ext(spr, ind, format = surface_rgba8unorm) {
|
2022-09-27 06:37:28 +02:00
|
|
|
if(!sprite_exists(spr)) return noone;
|
|
|
|
var sw = sprite_get_width(spr);
|
|
|
|
var sh = sprite_get_height(spr);
|
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
var s = surface_create_valid(sw, sh, format);
|
2022-09-27 06:37:28 +02:00
|
|
|
surface_set_target(s);
|
2023-02-19 02:13:19 +01:00
|
|
|
BLEND_OVERRIDE;
|
2023-03-19 09:17:39 +01:00
|
|
|
DRAW_CLEAR
|
2022-09-27 06:37:28 +02:00
|
|
|
draw_sprite(spr, ind, sprite_get_xoffset(spr), sprite_get_yoffset(spr));
|
|
|
|
BLEND_NORMAL
|
|
|
|
surface_reset_target();
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2023-03-21 03:01:53 +01:00
|
|
|
function surface_size_lim(surface, width, height) {
|
|
|
|
var sw = surface_get_width(surface);
|
|
|
|
var sh = surface_get_height(surface);
|
|
|
|
if(sw <= width && sh <= height) return surface;
|
|
|
|
|
|
|
|
var ss = min(width / sw, height / sh);
|
|
|
|
var s = surface_create(sw * ss, sh * ss);
|
|
|
|
surface_set_target(s);
|
|
|
|
DRAW_CLEAR;
|
|
|
|
draw_surface_ext(surface, 0, 0, ss, ss, 0, c_white, 1);
|
|
|
|
surface_reset_target();
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
function surface_size_to(surface, width, height, format = noone) {
|
|
|
|
if(!is_surface(surface)) return surface;
|
|
|
|
if(width < 1 && height < 1) return surface;
|
|
|
|
|
2023-05-03 21:42:17 +02:00
|
|
|
if(format != noone && format != surface_get_format(surface)) {
|
2023-03-19 09:17:39 +01:00
|
|
|
surface_free(surface);
|
|
|
|
return surface_create_valid(width, height, format);
|
|
|
|
}
|
2022-09-27 06:37:28 +02:00
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
width = surface_valid_size(width);
|
2023-02-19 02:13:19 +01:00
|
|
|
height = surface_valid_size(height);
|
2022-09-27 06:37:28 +02:00
|
|
|
|
2023-02-19 02:13:19 +01:00
|
|
|
var ww = surface_get_width(surface);
|
|
|
|
var hh = surface_get_height(surface);
|
2022-09-27 06:37:28 +02:00
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
if(ww == width && hh == height) return surface;
|
2023-02-19 02:13:19 +01:00
|
|
|
|
|
|
|
surface_resize(surface, width, height);
|
2023-03-19 09:17:39 +01:00
|
|
|
return surface;
|
2022-12-27 04:00:50 +01:00
|
|
|
}
|
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
function surface_copy_from(dst, src, format = noone) {
|
2023-02-19 02:13:19 +01:00
|
|
|
surface_set_target(dst);
|
2023-03-19 09:17:39 +01:00
|
|
|
DRAW_CLEAR
|
2023-02-19 02:13:19 +01:00
|
|
|
BLEND_OVERRIDE;
|
|
|
|
draw_surface_safe(src, 0, 0);
|
|
|
|
BLEND_NORMAL
|
|
|
|
surface_reset_target();
|
|
|
|
}
|
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
function surface_clone(surface, source = noone, format = noone) {
|
2023-02-19 02:13:19 +01:00
|
|
|
if(!is_surface(surface)) return noone;
|
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
source = surface_verify(source, surface_get_width(surface), surface_get_height(surface), format == noone? surface_get_format(surface) : format);
|
2023-02-19 02:13:19 +01:00
|
|
|
|
|
|
|
surface_set_target(source);
|
2023-03-19 09:17:39 +01:00
|
|
|
DRAW_CLEAR
|
2023-02-19 02:13:19 +01:00
|
|
|
BLEND_OVERRIDE;
|
|
|
|
draw_surface_safe(surface, 0, 0);
|
|
|
|
BLEND_NORMAL
|
|
|
|
surface_reset_target();
|
|
|
|
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
function surface_copy_size(dest, source, format = noone) {
|
2023-02-19 02:13:19 +01:00
|
|
|
if(!is_surface(dest)) return;
|
|
|
|
if(!is_surface(source)) return;
|
|
|
|
|
2023-03-19 09:17:39 +01:00
|
|
|
surface_size_to(dest, surface_get_width(source), surface_get_height(source), format);
|
2023-02-19 02:13:19 +01:00
|
|
|
surface_set_target(dest);
|
2023-03-19 09:17:39 +01:00
|
|
|
DRAW_CLEAR
|
2023-02-19 02:13:19 +01:00
|
|
|
surface_reset_target();
|
|
|
|
|
|
|
|
surface_copy_from(dest, source);
|
|
|
|
}
|
|
|
|
|
|
|
|
function surface_valid_size(s) {
|
|
|
|
if(is_infinity(s)) return 1;
|
|
|
|
return max(1, s);
|
2023-02-14 13:44:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function surface_array_free(arr) {
|
|
|
|
if(!is_array(arr)) {
|
|
|
|
if(is_surface(arr)) surface_free(arr);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for( var i = 0; i < array_length(arr); i++ )
|
|
|
|
surface_array_free(arr[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
function surface_array_clone(arr) {
|
|
|
|
if(!is_array(arr)) {
|
|
|
|
if(is_surface(arr))
|
|
|
|
return surface_clone(arr);
|
|
|
|
else
|
|
|
|
return arr;
|
|
|
|
}
|
|
|
|
|
|
|
|
var _arr = [];
|
|
|
|
|
|
|
|
for( var i = 0; i < array_length(arr); i++ )
|
|
|
|
_arr[i] = surface_array_clone(arr[i]);
|
|
|
|
|
2023-03-02 07:59:14 +01:00
|
|
|
return _arr;
|
|
|
|
}
|
|
|
|
|
|
|
|
function surface_array_serialize(arr) {
|
|
|
|
var _arr = __surface_array_serialize(arr);
|
|
|
|
return json_stringify(_arr);
|
|
|
|
}
|
|
|
|
|
|
|
|
function __surface_array_serialize(arr) {
|
|
|
|
if(!is_array(arr)) {
|
|
|
|
if(is_surface(arr)) {
|
|
|
|
var buff = buffer_create(surface_get_width(arr) * surface_get_height(arr) * 4, buffer_fixed, 1);
|
|
|
|
buffer_get_surface(buff, arr, 0);
|
|
|
|
var comp = buffer_compress(buff, 0, buffer_get_size(buff));
|
2023-03-13 10:45:56 +01:00
|
|
|
var enc = buffer_base64_encode(comp, 0, buffer_get_size(comp));
|
2023-03-02 07:59:14 +01:00
|
|
|
buffer_delete(buff);
|
|
|
|
return { width: surface_get_width(arr), height: surface_get_height(arr), buffer: enc };
|
|
|
|
} else
|
|
|
|
return arr;
|
|
|
|
}
|
|
|
|
|
|
|
|
var _arr = [];
|
|
|
|
|
|
|
|
for( var i = 0; i < array_length(arr); i++ )
|
|
|
|
_arr[i] = __surface_array_serialize(arr[i]);
|
|
|
|
|
|
|
|
return _arr;
|
|
|
|
}
|
|
|
|
|
|
|
|
function surface_array_deserialize(arr, index = -1) {
|
2023-03-05 07:16:44 +01:00
|
|
|
var _arr = json_try_parse(arr);
|
2023-03-13 10:45:56 +01:00
|
|
|
return index == -1? __surface_array_deserialize(_arr) : __surface_array_deserialize(_arr[index]);
|
2023-03-02 07:59:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function __surface_array_deserialize(arr) {
|
|
|
|
if(!is_array(arr)) {
|
|
|
|
var buff = buffer_base64_decode(arr.buffer);
|
2023-03-13 10:45:56 +01:00
|
|
|
buff = buffer_decompress(buff);
|
2023-03-02 07:59:14 +01:00
|
|
|
return surface_create_from_buffer(arr.width, arr.height, buff);
|
|
|
|
}
|
|
|
|
|
|
|
|
var _arr = [];
|
|
|
|
|
|
|
|
for( var i = 0; i < array_length(arr); i++ )
|
|
|
|
_arr[i] = __surface_array_deserialize(arr[i]);
|
|
|
|
|
2023-02-14 13:44:46 +01:00
|
|
|
return _arr;
|
2023-03-05 07:16:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function surface_encode(surface) {
|
|
|
|
if(!is_surface(surface)) return "";
|
|
|
|
|
|
|
|
var buff = buffer_create(surface_get_width(surface) * surface_get_height(surface) * 4, buffer_fixed, 1);
|
|
|
|
buffer_get_surface(buff, surface, 0);
|
|
|
|
var comp = buffer_compress(buff, 0, buffer_get_size(buff));
|
|
|
|
var enc = buffer_base64_encode(comp, 0, buffer_get_size(comp));
|
|
|
|
buffer_delete(buff);
|
|
|
|
var str = { width: surface_get_width(surface), height: surface_get_height(surface), buffer: enc };
|
|
|
|
return json_stringify(str);
|
|
|
|
}
|
|
|
|
|
|
|
|
function surface_decode(struct) {
|
|
|
|
var buff = buffer_base64_decode(struct.buffer);
|
|
|
|
var buff = buffer_decompress(buff);
|
|
|
|
return surface_create_from_buffer(struct.width, struct.height, buff);
|
2023-03-19 09:17:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function surface_bit_size(format) {
|
|
|
|
switch(format) {
|
|
|
|
case surface_rgba4unorm : return 4 * 0.5; break;
|
|
|
|
case surface_rgba8unorm : return 4 * 1; break;
|
|
|
|
case surface_rgba16float : return 4 * 2; break;
|
|
|
|
case surface_rgba32float : return 4 * 4; break;
|
|
|
|
|
|
|
|
case surface_r8unorm : return 1 * 1; break;
|
|
|
|
case surface_r16float : return 1 * 2; break;
|
|
|
|
case surface_r32float : return 1 * 3; break;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
function surface_get_size(surface) {
|
|
|
|
var sw = surface_get_width(surface);
|
|
|
|
var sh = surface_get_height(surface);
|
|
|
|
var sz = sw * sh * surface_bit_size(surface_get_format(surface));
|
|
|
|
return sz;
|
2022-01-13 05:24:03 +01:00
|
|
|
}
|