#ifndef MM_ALGOS_H
#define MM_ALGOS_H

/*************************************************************************/
/*  mm_algos.h                                                           */
/*************************************************************************/
/*                         This file is part of:                         */
/*                          PANDEMONIUM ENGINE                           */
/*             https://github.com/Relintai/pandemonium_engine            */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar.                              */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md).   */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur.                 */
/*                                                                       */
/* Permission is hereby granted, free of charge, to any person obtaining */
/* a copy of this software and associated documentation files (the       */
/* "Software"), to deal in the Software without restriction, including   */
/* without limitation the rights to use, copy, modify, merge, publish,   */
/* distribute, sublicense, and/or sell copies of the Software, and to    */
/* permit persons to whom the Software is furnished to do so, subject to */
/* the following conditions:                                             */
/*                                                                       */
/* The above copyright notice and this permission notice shall be        */
/* included in all copies or substantial portions of the Software.       */
/*                                                                       */
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,       */
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF    */
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY  */
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,  */
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE     */
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                */
/*************************************************************************/

#include "core/containers/pool_vector.h"
#include "core/io/image.h"
#include "core/math/color.h"
#include "core/math/vector2.h"
#include "core/math/vector3.h"
#include "core/variant/variant.h"

#include "core/object/reference.h"

class MMAlgos : public Reference {
	GDCLASS(MMAlgos, Reference);

public:
	static Vector3 clampv3(const Vector3 &v, const Vector3 &mi, const Vector3 &ma);
	static Color floorc(const Color &a);
	static Vector2 floorv2(const Vector2 &a);
	static Vector3 floorv3(const Vector3 &a);
	static Vector2 smoothstepv2(const float a, const float b, const Vector2 &c);
	static Vector2 maxv2(const Vector2 &a, const Vector2 &b);
	static Vector3 maxv3(const Vector3 &a, const Vector3 &b);
	static Vector2 absv2(const Vector2 &v);
	static Vector3 absv3(const Vector3 &v);
	static Vector2 cosv2(const Vector2 &v);
	static Vector3 cosv3(const Vector3 &v);
	static Vector2 powv2(const Vector2 &x, const Vector2 &y);
	static Vector3 modv3(const Vector3 &a, const Vector3 &b);
	static Vector2 modv2(const Vector2 &a, const Vector2 &b);
	static float modf(const float x, const float y);
	static Vector2 fractv2(const Vector2 &v);
	static Vector3 fractv3(const Vector3 &v);
	static float fract(const float f);
	static Vector2 clampv2(const Vector2 &v, const Vector2 &pmin, const Vector2 &pmax);
	static Vector2 minv2(const Vector2 &v1, const Vector2 &v2);
	static Vector3 minv3(const Vector3 &v1, const Vector3 &v2);
	static float rand(const Vector2 &x);
	static Vector2 rand2(const Vector2 &x);
	static Vector3 rand3(const Vector2 &x);
	static float step(const float edge, const float x);
	static Vector2 stepv2(const Vector2 &edge, const Vector2 &x);
	static Color stepc(const Color &edge, const Color &x);
	static Color mixc(const Color &a, const Color &b, const Color &t);
	static Vector2 signv2(const Vector2 &x);
	static Vector2 transform(const Vector2 &uv, const Vector2 &translate, const float rotate, const Vector2 &scale, const bool repeat);
	static float fractf(const float x);
	static float mix_mul(const float x, const float y);
	static float mix_add(const float x, const float y);
	static float mix_max(const float x, const float y);
	static float mix_min(const float x, const float y);
	static float mix_xor(const float x, const float y);
	static float mix_pow(const float x, const float y);
	static float wave_constant(const float x);
	static float wave_sine(const float x);
	static float wave_triangle(const float x);
	static float wave_sawtooth(const float x);
	static float wave_square(const float x);
	static float wave_bounce(const float x);
	static Color sinewave(const Vector2 &uv, const float amplitude, const float frequency, const float phase);
	static float ThickLine(const Vector2 &uv, const Vector2 &posA, const Vector2 &posB, const float radiusInv);
	static float curve(const float x, const PoolRealArray &points);
	static Color flood_fill_preprocess(const Vector2 &uv, const float c, const float s);
	static Vector3 fill_to_uv_stretch(const Vector2 &coord, const Color &bb, const float pseed);
	static Vector3 fill_to_uv_square(const Vector2 &coord, const Color &bb, const float pseed);
	static Vector3 rgb_to_hsv(const Vector3 &c);
	static Vector3 hsv_to_rgb(const Vector3 &c);
	static Color adjust_hsv(const Color &color, const float hue, const float saturation, const float value);
	static Color adjust_levels(const Color &color, const Color &in_min, const Color &in_mid, const Color &in_max, const Color &out_min, const Color &out_max);
	static Color brightness_contrast(const Color &color, const float brightness, const float contrast);
	static float grayscale_min(const Vector3 &c);
	static float grayscale_max(const Vector3 &c);
	static float grayscale_lightness(const Vector3 &c);
	static float grayscale_average(const Vector3 &c);
	static float grayscale_luminosity(const Vector3 &c);
	static Color invert(const Color &color);
	static Vector3 blend_normal(const Vector2 &uv, const Vector3 &c1, const Vector3 &c2, const float opacity);
	static Vector3 blend_dissolve(const Vector2 &uv, const Vector3 &c1, const Vector3 &c2, const float opacity);
	static Vector3 blend_multiply(const Vector2 &uv, const Vector3 &c1, const Vector3 &c2, const float opacity);
	static Vector3 blend_screen(const Vector2 &uv, const Vector3 &c1, const Vector3 &c2, const float opacity);
	static float blend_overlay_f(const float c1, const float c2);
	static Vector3 blend_overlay(const Vector2 &uv, const Vector3 &c1, const Vector3 &c2, const float opacity);
	static Vector3 blend_hard_light(const Vector2 &uv, const Vector3 &c1, const Vector3 &c2, const float opacity);
	static float blend_soft_light_f(const float c1, const float c2);
	static Vector3 blend_soft_light(const Vector2 &uv, const Vector3 &c1, const Vector3 &c2, const float opacity);
	static float blend_burn_f(const float c1, const float c2);
	static Vector3 blend_burn(const Vector2 &uv, const Vector3 &c1, const Vector3 &c2, const float opacity);
	static float blend_dodge_f(const float c1, const float c2);
	static Vector3 blend_dodge(const Vector2 &uv, const Vector3 &c1, const Vector3 &c2, const float opacity);
	static Vector3 blend_lighten(const Vector2 &uv, const Vector3 &c1, const Vector3 &c2, const float opacity);
	static Vector3 blend_darken(const Vector2 &uv, const Vector3 &c1, const Vector3 &c2, const float opacity);
	static Vector3 blend_difference(const Vector2 &uv, const Vector3 &c1, const Vector3 &c2, const float opacity);
	static Color radial_gradient_type_1(const Vector2 &uv, const float repeat, const PoolRealArray &data);
	static Color radial_gradient_type_2(const Vector2 &uv, const float repeat, const PoolRealArray &data);
	static Color radial_gradient_type_3(const Vector2 &uv, const float repeat, const PoolRealArray &data);
	static Color radial_gradient_type_4(const Vector2 &uv, const float repeat, const PoolRealArray &data);
	static Color normal_gradient_type_1(const Vector2 &uv, const float repeat, const float rotate, const PoolRealArray &data);
	static Color normal_gradient_type_2(const Vector2 &uv, const float repeat, const float rotate, const PoolRealArray &data);
	static Color normal_gradient_type_3(const Vector2 &uv, const float repeat, const float rotate, const PoolRealArray &data);
	static Color normal_gradient_type_4(const Vector2 &uv, const float repeat, const float rotate, const PoolRealArray &data);
	static Color circular_gradient_type_1(const Vector2 &uv, const float repeat, const PoolRealArray &data);
	static Color circular_gradient_type_2(const Vector2 &uv, const float repeat, const PoolRealArray &data);
	static Color circular_gradient_type_3(const Vector2 &uv, const float repeat, const PoolRealArray &data);
	static Color circular_gradient_type_4(const Vector2 &uv, const float repeat, const PoolRealArray &data);
	static Color gradient_type_1(const float x, const PoolRealArray &data);
	static Color gradient_type_2(const float x, const PoolRealArray &data);
	static Color gradient_type_3(const float x, const PoolRealArray &data);
	static Color get_data_color(const int index, const PoolRealArray &data);
	static float get_data_pos(const int index, const PoolRealArray &data);
	static Color gradient_type_4(const float x, const PoolRealArray &data);
	static Color gradient_type_5(const float x, const PoolRealArray &data);
	static float dots(const Vector2 &uv, const float size, const float density, const float pseed);
	static Color anisotropicc(const Vector2 &uv, const Vector2 &size, const float pseed, const float smoothness, const float interpolation);
	static float anisotropic(const Vector2 &uv, const Vector2 &size, const float pseed, const float smoothness, const float interpolation);
	static Vector3 color_dots(const Vector2 &uv, const float size, const float pseed);
	static Color noise_color(const Vector2 &uv, const float size, const float pseed);
	static Color fbmval(const Vector2 &uv, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static Color perlin(const Vector2 &uv, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static Color perlinabs(const Vector2 &uv, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static Color simplex(const Vector2 &uv, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static Color cellular(const Vector2 &uv, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static Color cellular2(const Vector2 &uv, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static Color cellular3(const Vector2 &uv, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static Color cellular4(const Vector2 &uv, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static Color cellular5(const Vector2 &uv, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static Color cellular6(const Vector2 &uv, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static float fbmf(const Vector2 &coord, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static float perlinf(const Vector2 &coord, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static float perlinabsf(const Vector2 &coord, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static float fbm_simplexf(const Vector2 &coord, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static float cellularf(const Vector2 &coord, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static float cellular2f(const Vector2 &coord, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static float cellular3f(const Vector2 &coord, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static float cellular4f(const Vector2 &coord, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static float cellular5f(const Vector2 &coord, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static float cellular6f(const Vector2 &coord, const Vector2 &size, const int folds, const int octaves, const float persistence, const float pseed);
	static float fbm_value(const Vector2 &coord, const Vector2 &size, const float pseed);
	static float fbm_perlin(const Vector2 &coord, const Vector2 &size, const float pseed);
	static float fbm_perlinabs(const Vector2 &coord, const Vector2 &size, const float pseed);
	static Vector2 rgrad2(const Vector2 &p, const float rot, const float pseed);
	static float fbm_simplex(const Vector2 &coord, const Vector2 &size, const float pseed);
	static float fbm_cellular(const Vector2 &coord, const Vector2 &size, const float pseed);
	static float fbm_cellular2(const Vector2 &coord, const Vector2 &size, const float pseed);
	static float fbm_cellular3(const Vector2 &coord, const Vector2 &size, const float pseed);
	static float fbm_cellular4(const Vector2 &coord, const Vector2 &size, const float pseed);
	static float fbm_cellular5(const Vector2 &coord, const Vector2 &size, const float pseed);
	static float fbm_cellular6(const Vector2 &coord, const Vector2 &size, const float pseed);
	static Color perlinc(const Vector2 &uv, const Vector2 &size, const int iterations, const float persistence, const int pseed);
	static float perlin2c(const Vector2 &uv, const Vector2 &size, const int iterations, const float persistence, const int pseed);
	static Vector3 perlin_color(const Vector2 &uv, const Vector2 &size, const int iterations, const float persistence, const int pseed);
	static Color perlin_colorc(const Vector2 &uv, const Vector2 &size, const int iterations, const float persistence, const int pseed);
	static Color perlin_warp_1(const Vector2 &uv, const Vector2 &size, const int iterations, const float persistence, const int pseed, const Vector2 &translate, const float rotate, const Vector2 &size2);
	static Color perlin_warp_2(const Vector2 &uv, const Vector2 &size, const int iterations, const float persistence, const int pseed, const Vector2 &translate, const float rotate, const Vector2 &size2);
	static Color voronoi(const Vector2 &uv, const Vector2 &size, const Vector2 &stretch, const float intensity, const float randomness, const int pseed);
	static Color voronoi_1(const Vector2 &uv, const Vector2 &size, const Vector2 &stretch, const float intensity, const float randomness, const int pseed);
	static Color voronoi_2(const Vector2 &uv, const Vector2 &size, const Vector2 &stretch, const float intensity, const float randomness, const int pseed);
	static Color voronoi_3(const Vector2 &uv, const Vector2 &size, const Vector2 &stretch, const float intensity, const float randomness, const int pseed);

	enum CombinerAxisType {
		COMBINER_AXIS_TYPE_SINE = 0,
		COMBINER_AXIS_TYPE_TRIANGLE,
		COMBINER_AXIS_TYPE_SQUARE,
		COMBINER_AXIS_TYPE_SAWTOOTH,
		COMBINER_AXIS_TYPE_CONSTANT,
		COMBINER_AXIS_TYPE_BOUNCE
	};

	enum CombinerType {
		COMBINER_TYPE_MULTIPLY = 0,
		COMBINER_TYPE_ADD,
		COMBINER_TYPE_MAX,
		COMBINER_TYPE_MIN,
		COMBINER_TYPE_XOR,
		COMBINER_TYPE_POW
	};

	static float pattern(const Vector2 &uv, const float x_scale, const float y_scale, const int ct, const int catx, const int caty);
	static Color truchet1c(const Vector2 &uv, const float size, const float pseed);
	static float truchet1(const Vector2 &uv, const Vector2 &pseed);
	static Color truchet2c(const Vector2 &uv, const float size, const float pseed);
	static float truchet2(const Vector2 &uv, const Vector2 &pseed);
	static Color weavec(const Vector2 &uv, const Vector2 &count, const float width);
	static float weave(const Vector2 &uv, const Vector2 &count, const float width);
	static Vector3 weave2(const Vector2 &uv, const Vector2 &count, const float stitch, const float width_x, const float width_y);
	static Color sinewavec(const Vector2 &uv, const float amplitude, const float frequency, const float phase);
	static float sinewavef(const Vector2 &uv, const float amplitude, const float frequency, const float phase);
	static float scratch(const Vector2 &uv, const Vector2 &size, const float waviness, const float angle, const float randomness, const Vector2 &pseed);
	static float scratches(const Vector2 &uv, const int layers, const Vector2 &size, const float waviness, const float angle, const float randomness, const Vector2 &pseed);
	static Color scratchesc(const Vector2 &uv, const int layers, const Vector2 &size, const float waviness, const float angle, const float randomness, const Vector2 &pseed);
	static Color runesc(const Vector2 &uv, const Vector2 &col_row, const float pseed);
	static float runesf(const Vector2 &uv, const Vector2 &col_row, const float pseed);
	static float rune(const Vector2 &uv, const float pseed);
	static Color IChingc(const Vector2 &uv, const Vector2 &row_col, const int pseed);
	static float IChing(const Vector2 &uv, const float pseed);
	static Color beehive_1c(const Vector2 &uv, const Vector2 &size, const int pseed);
	static Color beehive_2c(const Vector2 &uv, const Vector2 &size, const int pseed);
	static Color beehive_3c(const Vector2 &uv, const Vector2 &size, const int pseed);
	static float beehive_dist(const Vector2 &p);
	static Color beehive_center(const Vector2 &p);
	static Vector3 brick_corner_uv(const Vector2 &uv, const Vector2 &bmin, const Vector2 &bmax, const float mortar, const float corner, const float pseed);
	static Color brick(const Vector2 &uv, const Vector2 &bmin, const Vector2 &bmax, const float mortar, const float pround, const float bevel);
	static Vector3 brick_random_color(const Vector2 &bmin, const Vector2 &bmax, const float pseed);
	static Vector3 brick_uv(const Vector2 &uv, const Vector2 &bmin, const Vector2 &bmax, const float pseed);
	static Color bricks_rb(const Vector2 &uv, const Vector2 &count, const float repeat, const float offset);
	static Color bricks_rb2(const Vector2 &uv, const Vector2 &count, const float repeat, const float offset);
	static Color bricks_hb(const Vector2 &uv, const Vector2 &count, const float repeat, const float offset);
	static Color bricks_bw(const Vector2 &uv, const Vector2 &count, const float repeat, const float offset);
	static Color bricks_sb(const Vector2 &uv, const Vector2 &count, const float repeat, const float offset);
	static Color brick2(const Vector2 &uv, const Vector2 &bmin, const Vector2 &bmax, const float mortar, const float pround, const float bevel);
	static Color bricks_uneven(const Vector2 &uv, const int iterations, const float min_size, const float randomness, const float pseed);
	static Vector2 truchet_generic_uv(const Vector2 &uv, const float pseed);
	static Color sdf_show(const float val, const float bevel);
	static float sdf_circle(const Vector2 &uv, const Vector2 &c, const float r);
	static float sdf_box(const Vector2 &uv, const Vector2 &c, const Vector2 &wh);
	static Vector2 sdf_line(const Vector2 &uv, const Vector2 &a, const Vector2 &b, const float r);
	static float sdf_rhombus(const Vector2 &uv, const Vector2 &c, const Vector2 &wh);
	static float sdf_arc(const Vector2 &uv, const Vector2 &a, const Vector2 &r);
	static float sdr_ndot(const Vector2 &a, const Vector2 &b);
	static float sdRhombus(const Vector2 &p, const Vector2 &b);
	static float sdArc(const Vector2 &p, const float a1, const float a2, const float ra, const float rb);
	static float sdf_boolean_union(const float a, const float b);
	static float sdf_boolean_substraction(const float a, const float b);
	static float sdf_boolean_intersection(const float a, const float b);
	static float sdf_smooth_boolean_union(const float d1, const float d2, const float k);
	static float sdf_smooth_boolean_substraction(const float d1, const float d2, const float k);
	static float sdf_smooth_boolean_intersection(const float d1, const float d2, const float k);
	static float sdf_rounded_shape(const float a, const float r);
	static float sdf_annular_shape(const float a, const float r);
	static float sdf_morph(const float a, const float b, const float amount);
	static Vector2 sdLine(const Vector2 &p, const Vector2 &a, const Vector2 &b);
	static Vector2 sdf2d_rotate(const Vector2 &uv, const float a);
	static Vector2 sdBezier(const Vector2 &pos, const Vector2 &A, const Vector2 &B, const Vector2 &C);
	static Vector2 circle_repeat_transform_2d(const Vector2 &p, const float count);
	static Vector2 sdNgon(const Vector2 &pos, const float r, const float n);
	static Vector2 repeat_2d(const Vector2 &p, const Vector2 &r, const float pseed, const float randomness);
	static float sdSmoothUnion(const float d1, const float d2, const float k);
	static float sdSmoothSubtraction(const float d1, const float d2, const float k);
	static float sdSmoothIntersection(const float d1, const float d2, const float k);
	static float sdRipples(const float d, const float w, const int r);
	static float sdPolygon(const Vector2 &p, const PoolVector2Array &v);
	static Color raymarch(const Vector2 &uv);
	static Color raymarch2(const Vector2 &uv);
	static Color raymarch3(const Vector2 &uv);
	static Vector2 sdf3d_sphere(const Vector3 &p, const float r);
	static Vector2 sdf3d_box(const Vector3 &p, const float sx, const float sy, const float sz, const float r);
	static Vector2 sdf3d_cylinder_y(const Vector3 &p, const float r, const float l);
	static Vector2 sdf3d_cylinder_x(const Vector3 &p, const float r, const float l);
	static Vector2 sdf3d_cylinder_z(const Vector3 &p, const float r, const float l);
	static Vector2 sdf3d_capsule_y(const Vector3 &p, const float r, const float l);
	static Vector2 sdf3d_capsule_x(const Vector3 &p, const float r, const float l);
	static Vector2 sdf3d_capsule_z(const Vector3 &p, const float r, const float l);
	static Vector2 sdf3d_cone_px(const Vector3 &p, const float a);
	static Vector2 sdf3d_cone_nx(const Vector3 &p, const float a);
	static Vector2 sdf3d_cone_py(const Vector3 &p, const float a);
	static Vector2 sdf3d_cone_ny(const Vector3 &p, const float a);
	static Vector2 sdf3d_cone_pz(const Vector3 &p, const float a);
	static Vector2 sdf3d_cone_nz(const Vector3 &p, const float a);
	static Vector2 sdf3d_torus_x(const Vector3 &p, const float R, const float r);
	static Vector2 sdf3d_torus_y(const Vector3 &p, const float R, const float r);
	static Vector2 sdf3d_torus_z(const Vector3 &p, const float R, const float r);
	static Vector2 sdf3d_raymarch(const Vector2 &uv);
	static Vector3 sdf3d_normal(const Vector3 &p);
	static Vector2 sdf3dc_union(const Vector2 &a, const Vector2 &b);
	static Vector2 sdf3dc_sub(const Vector2 &a, const Vector2 &b);
	static Vector2 sdf3dc_inter(const Vector2 &a, const Vector2 &b);
	static Vector2 sdf3d_smooth_union(const Vector2 &d1, const Vector2 &d2, const float k);
	static Vector2 sdf3d_smooth_subtraction(const Vector2 &d1, const Vector2 &d2, const float k);
	static Vector2 sdf3d_smooth_intersection(const Vector2 &d1, const Vector2 &d2, const float k);
	static Vector2 sdf3d_rounded(const Vector2 &v, const float r);
	static Vector3 sdf3d_elongation(const Vector3 &p, const Vector3 &v);
	static Vector3 sdf3d_repeat(const Vector3 &p, const Vector2 &r, const float randomness, const int pseed);
	static Vector3 repeat(const Vector3 &p, const Vector3 &r, const float pseed, const float randomness);
	static Vector3 rotate3d(const Vector3 &p, const Vector3 &a);
	static Vector3 circle_repeat_transform(const Vector3 &p, const float count);
	static Vector2 sdf3d_input(const Vector3 &p);
	static float sphere(const Vector2 &uv, const Vector2 &c, const float r);
	static float shape_circle(const Vector2 &uv, const float sides, const float size, const float edge);
	static float shape_polygon(const Vector2 &uv, const float sides, const float size, const float edge);
	static float shape_star(const Vector2 &uv, const float sides, const float size, const float edge);
	static float shape_curved_star(const Vector2 &uv, const float sides, const float size, const float edge);
	static float shape_rays(const Vector2 &uv, const float sides, const float size, const float edge);
	static Vector2 transform2_clamp(const Vector2 &uv);
	static Vector2 transform2(const Vector2 &uv, const Vector2 &translate, const float rotate, const Vector2 &scale);
	static Vector2 rotate(const Vector2 &uv, const Vector2 &center, const float rotate);
	static Vector2 scale(const Vector2 &uv, const Vector2 &center, const Vector2 &scale);
	static Vector2 uvmirror_h(const Vector2 &uv, const float offset);
	static Vector2 uvmirror_v(const Vector2 &uv, const float offset);
	static Vector2 kal_rotate(const Vector2 &uv, const float count, const float offset);
	static Vector2 get_from_tileset(const float count, const float pseed, const Vector2 &uv);
	static Vector2 custom_uv_transform(const Vector2 &uv, const Vector2 &cst_scale, const float rnd_rotate, const float rnd_scale, const Vector2 &pseed);

	static Ref<Image> generate_histogram(const Ref<Image> &input, const int texture_size = 256);

	static void register_node_class(const String &category, const String &cls);
	static void unregister_node_class(const String &category, const String &cls);

	static void register_node_script(const String &category, const String &file_path);
	static void unregister_node_script(const String &category, const String &file_path);

	enum MMNodeRegistryType {
		MMNODE_REGISTRY_TYPE_CLASS = 0,
		MMNODE_REGISTRY_TYPE_SCRIPT
	};

	struct MMNodeRegistryEntry {
		MMNodeRegistryType type;
		String data;
	};

	struct MMNodeRegistryCategory {
		String category_name;
		Vector<MMNodeRegistryEntry> entries;
	};

	static Vector<MMNodeRegistryCategory> mm_node_registry;

	MMAlgos();
	~MMAlgos();

protected:
	static void _bind_methods();
};

#endif