32 #define GETTEXT_DOMAIN "wesnoth-lib"
35 #define ERR_DP LOG_STREAM(err, log_display)
41 priorities_[mod->priority()].push_back(std::move(mod));
47 auto& mod_list = top_pair->second;
50 mod_list.erase(mod_list.begin());
53 if(mod_list.empty()) {
60 std::size_t count = 0;
77 using mod_parser = std::function<std::unique_ptr<modification>(std::string_view)>;
84 std::map<std::string, mod_parser, std::less<>> mod_parsers;
93 std::unique_ptr<modification> decode_modification(
const std::string& encoded_mod)
97 if(
split.size() != 2) {
98 ERR_DP <<
"error parsing image modifications: " << encoded_mod;
102 const std::string& mod_type =
split[0];
103 const std::string& args =
split[1];
105 if(
const auto parser = mod_parsers.find(mod_type); parser != mod_parsers.end()) {
106 return std::invoke(parser->second, args);
108 ERR_DP <<
"unknown image function in path: " << mod_type;
117 : message(message_stream.str())
142 if(
auto mod = decode_modification(encoded_mod)) {
143 mods.
push(std::move(mod));
157 if(horiz_ && vert_ ) {
170 const int normalized = degrees_ >= 0 ?
171 degrees_ - 360 * (degrees_ / 360) :
172 degrees_ + 360 * (1 + (-degrees_) / 360);
174 switch ( normalized )
202 if(src_rect.w ==
src->w && src_rect.h ==
src->h) {
209 ERR_DP <<
"Failed to either crop or scale the surface";
243 : color_callable(clr), coord(),
w(
w),
h(
h)
251 color_callable::get_inputs(inputs);
252 add_input(inputs,
"x");
253 add_input(inputs,
"y");
254 add_input(inputs,
"u");
255 add_input(inputs,
"v");
256 add_input(inputs,
"height");
257 add_input(inputs,
"width");
264 return variant(coord.x);
265 }
else if(key ==
"y") {
266 return variant(coord.y);
267 }
else if(key ==
"width") {
269 }
else if(key ==
"height") {
271 }
else if(key ==
"u") {
272 return variant(coord.x /
static_cast<float>(
w));
273 }
else if(key ==
"v") {
274 return variant(coord.y /
static_cast<float>(
h));
277 return color_callable::get_value(key);
291 std::size_t
index{0};
297 color.a = std::min<unsigned>(new_alpha.
evaluate(px).
as_int(), 255);
299 pixel = color.to_argb_bytes();
313 std::size_t
index{0};
324 pixel = color.to_argb_bytes();
346 std::stringstream sstr;
347 sstr <<
"~BLIT(): x-coordinate '"
348 << x_ <<
"' larger than destination image's width '"
349 <<
src->w <<
"' no blitting performed.\n";
355 std::stringstream sstr;
356 sstr <<
"~BLIT(): y-coordinate '"
357 << y_ <<
"' larger than destination image's height '"
358 <<
src->h <<
"' no blitting performed.\n";
363 if(surf_->w + x_ < 0) {
364 std::stringstream sstr;
365 sstr <<
"~BLIT(): offset and width '"
366 << x_ + surf_->w <<
"' less than zero no blitting performed.\n";
371 if(surf_->h + y_ < 0) {
372 std::stringstream sstr;
373 sstr <<
"~BLIT(): offset and height '"
374 << y_ + surf_->h <<
"' less than zero no blitting performed.\n";
379 rect r {x_, y_, 0, 0};
385 if(
src->w == mask_->w &&
src->h == mask_->h && x_ == 0 && y_ == 0) {
390 rect r {x_, y_, 0, 0};
392 sdl_blit(mask_,
nullptr, new_mask, &r);
398 if(
src ==
nullptr) {
return; }
401 if(surf_->w !=
src->w || surf_->h !=
src->h) {
414 }
else if(flags_ & X_BY_FACTOR) {
415 size.x =
src->w * (
static_cast<double>(
size.x) / 100);
420 }
else if(flags_ & Y_BY_FACTOR) {
421 size.y =
src->h * (
static_cast<double>(
size.y) / 100);
424 if(flags_ & PRESERVE_ASPECT_RATIO) {
425 const auto ratio = std::min(
426 static_cast<long double>(
size.x) /
src->w,
427 static_cast<long double>(
size.y) /
src->h
431 static_cast<int>(
src->w * ratio),
432 static_cast<int>(
src->h * ratio)
436 if(flags_ & SCALE_SHARP) {
453 const int new_w =
src->w + left_ + right_;
454 const int new_h =
src->h + top_ + bottom_;
467 SDL_BlendMode original_blend_mode;
468 SDL_GetSurfaceBlendMode(
src, &original_blend_mode);
470 SDL_SetSurfaceBlendMode(
src, SDL_BLENDMODE_NONE);
473 SDL_BlitSurface(
src,
nullptr, padded, &dstrect);
475 SDL_SetSurfaceBlendMode(padded, original_blend_mode);
490 uint32_t* beg = lock.
pixels();
491 uint32_t* end = beg +
src.
area();
494 uint8_t alpha = (*beg) >> 24;
503 *beg = (alpha << 24) + (r << 16) + (
g << 8) +
b;
513 if((r_ != 0 || g_ != 0 ||
b_ != 0)) {
531 SDL_FillRect(ret,
nullptr, SDL_MapRGBA(ret->format, color_.r, color_.g, color_.b, color_.a));
543 struct parse_mod_registration
545 parse_mod_registration(
const char* name, mod_parser parser)
547 mod_parsers[name] = std::move(parser);
560 #define REGISTER_MOD_PARSER(type, args_var) \
561 static std::unique_ptr<modification> parse_##type##_mod(std::string_view); \
562 static parse_mod_registration parse_##type##_mod_registration_aux(#type, &parse_##type##_mod); \
563 static std::unique_ptr<modification> parse_##type##_mod(std::string_view args_var) \
570 if(params.size() < 2) {
571 ERR_DP <<
"too few arguments passed to the ~TC() function";
576 const int side_n = utils::from_chars<int>(params[0]).value_or(-1);
578 ERR_DP <<
"Invalid side (" << params[0] <<
") passed to the ~TC() function";
586 ERR_DP <<
"could not load TC info for '" << params[1] <<
"' palette";
587 ERR_DP <<
"bailing out from TC";
599 ERR_DP <<
"caught config::error while processing TC: " <<
e.message;
600 ERR_DP <<
"bailing out from TC";
605 return std::make_unique<rc_modification>(rc_map);
613 if(recolor_params.size() <= 1) {
628 <<
"caught config::error while processing color-range RC: "
630 ERR_DP <<
"bailing out from RC";
634 return std::make_unique<rc_modification>(rc_map);
642 if(remap_params.size() < 2) {
643 ERR_DP <<
"not enough arguments passed to the ~PAL() function: " << args;
653 for(std::size_t
i = 0;
i < old_palette.size() &&
i < new_palette.size(); ++
i) {
654 rc_map[old_palette[
i]] = new_palette[
i];
657 return std::make_unique<rc_modification>(rc_map);
660 <<
"caught config::error while processing PAL function: "
663 <<
"bailing out from PAL";
672 bool horiz = (args.empty() || args.find(
"horiz") != std::string::npos);
673 bool vert = (args.find(
"vert") != std::string::npos);
675 return std::make_unique<fl_modification>(horiz, vert);
683 switch(slice_params.size()) {
685 return std::make_unique<rotate_modification>();
687 return std::make_unique<rotate_modification>(
688 utils::from_chars<int>(slice_params[0]).value_or(0));
690 return std::make_unique<rotate_modification>(
691 utils::from_chars<int>(slice_params[0]).value_or(0),
692 utils::from_chars<int>(slice_params[1]).value_or(0));
694 return std::make_unique<rotate_modification>(
695 utils::from_chars<int>(slice_params[0]).value_or(0),
696 utils::from_chars<int>(slice_params[1]).value_or(0),
697 utils::from_chars<int>(slice_params[2]).value_or(0));
705 return std::make_unique<gs_modification>();
711 return std::make_unique<crop_transparency_modification>();
715 bool in_range(
int val,
int min,
int max)
717 return min <= val && val <= max;
725 if(params.size() != 1) {
726 ERR_DP <<
"~BW() requires exactly one argument";
731 const auto threshold = utils::from_chars<int>(params[0]);
733 ERR_DP <<
"unsupported argument in ~BW() function";
737 if(!in_range(*threshold, 0, 255)) {
738 ERR_DP <<
"~BW() argument out of range 0 - 255";
742 return std::make_unique<bw_modification>(*threshold);
748 return std::make_unique<sepia_modification>();
756 switch(params.size()) {
760 return std::make_unique<negative_modification>(-1, -1, -1);
763 const auto threshold = utils::from_chars<int>(params[0]);
765 if(threshold && in_range(*threshold, -1, 255)) {
766 return std::make_unique<negative_modification>(*threshold, *threshold, *threshold);
768 ERR_DP <<
"unsupported argument value in ~NEG() function";
774 const auto thR = utils::from_chars<int>(params[0]);
775 const auto thG = utils::from_chars<int>(params[1]);
776 const auto thB = utils::from_chars<int>(params[2]);
778 if(thR && thG && thB && in_range(*thR, -1, 255) && in_range(*thG, -1, 255) && in_range(*thB, -1, 255)) {
779 return std::make_unique<negative_modification>(*thR, *thG, *thB);
781 ERR_DP <<
"unsupported argument value in ~NEG() function";
787 ERR_DP <<
"~NEG() requires 0, 1 or 3 arguments";
795 return std::make_unique<plot_alpha_modification>();
801 return std::make_unique<wipe_alpha_modification>();
811 if(params.size() != 1) {
812 ERR_DP <<
"~ADJUST_ALPHA() requires exactly 1 arguments";
816 return std::make_unique<adjust_alpha_modification>(params.at(0));
826 if(params.size() < 1 || params.size() > 4) {
827 ERR_DP <<
"~CHAN() requires 1 to 4 arguments";
831 return std::make_unique<adjust_channels_modification>(params);
838 const std::size_t
s = factors.size();
841 ERR_DP <<
"no arguments passed to the ~CS() function";
845 int r = 0,
g = 0,
b = 0;
847 r = utils::from_chars<int>(factors[0]).value_or(0);
850 g = utils::from_chars<int>(factors[1]).value_or(0);
853 b = utils::from_chars<int>(factors[2]).value_or(0);
856 return std::make_unique<cs_modification>(r,
g ,
b);
864 if(params.size() != 4) {
865 ERR_DP <<
"~BLEND() requires exactly 4 arguments";
869 float opacity = 0.0f;
870 const std::string_view& opacity_str = params[3];
871 const std::string_view::size_type p100_pos = opacity_str.find(
'%');
873 if(p100_pos == std::string::npos)
874 opacity = utils::from_chars<float>(opacity_str).value_or(0.0f);
877 const std::string_view parsed_field = opacity_str.substr(0, p100_pos);
878 opacity = utils::from_chars<float>(parsed_field).value_or(0.0f);
882 return std::make_unique<blend_modification>(
883 utils::from_chars<int>(params[0]).value_or(0),
884 utils::from_chars<int>(params[1]).value_or(0),
885 utils::from_chars<int>(params[2]).value_or(0),
893 const std::size_t
s = slice_params.size();
895 if(
s == 0 || (
s == 1 && slice_params[0].empty())) {
896 ERR_DP <<
"no arguments passed to the ~CROP() function";
900 rect slice_rect { 0, 0, 0, 0 };
902 slice_rect.x = utils::from_chars<int16_t>(slice_params[0]).value_or(0);
905 slice_rect.y = utils::from_chars<int16_t>(slice_params[1]).value_or(0);
908 slice_rect.w = utils::from_chars<uint16_t>(slice_params[2]).value_or(0);
911 slice_rect.h = utils::from_chars<uint16_t>(slice_params[3]).value_or(0);
914 return std::make_unique<crop_modification>(slice_rect);
920 message <<
" image not found: '" <<
img.get_filename() <<
"'\n";
929 const std::size_t
s = param.size();
931 if(
s == 0 || (
s == 1 && param[0].empty())){
932 ERR_DP <<
"no arguments passed to the ~BLIT() function";
937 ERR_DP <<
"too many arguments passed to the ~BLIT() function";
944 x = utils::from_chars<int>(param[1]).value_or(0);
945 y = utils::from_chars<int>(param[2]).value_or(0);
951 if(!check_image(
img, message))
955 return std::make_unique<blit_modification>(
surf, x, y);
962 const std::size_t
s = param.size();
964 if(
s == 0 || (
s == 1 && param[0].empty())){
965 ERR_DP <<
"no arguments passed to the ~MASK() function";
972 x = utils::from_chars<int>(param[1]).value_or(0);
973 y = utils::from_chars<int>(param[2]).value_or(0);
977 ERR_DP <<
"negative position arguments in ~MASK() function";
984 if(!check_image(
img, message))
988 return std::make_unique<mask_modification>(
surf, x, y);
995 ERR_DP <<
"no arguments passed to the ~L() function";
1000 return std::make_unique<light_modification>(
surf);
1005 std::pair<int, bool> parse_scale_value(std::string_view arg)
1007 if(
const std::size_t pos = arg.rfind(
'%'); pos != std::string_view::npos) {
1008 return { utils::from_chars<int>(arg.substr(0, pos)).value_or(0),
true };
1010 return { utils::from_chars<int>(arg).value_or(0),
false };
1015 utils::optional<std::pair<point, uint8_t>> parse_scale_args(std::string_view args)
1018 const std::size_t num_args = scale_params.size();
1020 if(num_args == 0 || (num_args == 1 && scale_params[0].empty())) {
1021 return utils::nullopt;
1025 std::array<int, 2> parsed_sizes{0,0};
1027 for(
unsigned i = 0; i < std::min<unsigned>(2, num_args); ++
i) {
1028 const auto& [
size, relative] = parse_scale_value(scale_params[
i]);
1031 ERR_DP <<
"Negative size passed to scaling IPF. Original image dimension will be used instead";
1035 parsed_sizes[
i] =
size;
1042 return std::pair{
point{parsed_sizes[0], parsed_sizes[1]}, flags};
1050 if(
auto params = parse_scale_args(args)) {
1052 return std::make_unique<scale_modification>(params->first, mode | params->second);
1054 ERR_DP <<
"no arguments passed to the ~SCALE() function";
1061 if(
auto params = parse_scale_args(args)) {
1063 return std::make_unique<scale_modification>(params->first, mode | params->second);
1065 ERR_DP <<
"no arguments passed to the ~SCALE_SHARP() function";
1072 if(
auto params = parse_scale_args(args)) {
1074 return std::make_unique<scale_modification>(params->first, mode | params->second);
1076 ERR_DP <<
"no arguments passed to the ~SCALE_INTO() function";
1083 if(
auto params = parse_scale_args(args)) {
1085 return std::make_unique<scale_modification>(params->first, mode | params->second);
1087 ERR_DP <<
"no arguments passed to the ~SCALE_INTO_SHARP() function";
1095 const int factor = std::clamp(utils::from_chars<int>(args).value_or(1), 1, 6);
1096 return std::make_unique<xbrz_modification>(factor);
1108 if(args.find(
'=') != std::string_view::npos) {
1113 const std::map<std::string, int*> alias_map = {
1118 {
"bottom", &bottom},
1125 for(
const auto& [key, value] : params) {
1126 auto it = alias_map.find(key);
1127 if(it != alias_map.end()) {
1128 if(utils::optional padding = utils::from_chars<int>(value)) {
1129 *it->second = padding.value();
1131 ERR_DP <<
"~PAD() keyword argument '" << key <<
"' requires a valid integer value. Received: '" << value <<
"'.";
1135 ERR_DP <<
"~PAD() found an unknown keyword: '" << key <<
"'. Valid keywords: top, t, right, r, bottom, b, left, l.";
1142 if(params.size() != 1) {
1143 ERR_DP <<
"~PAD() takes either 1 numeric argument for the padding on all sides or a comma separated list of '<keyword>=<number>' pairs with available keywords: top, t, right, r, bottom, b, left, l.";
1148 if(utils::optional padding = utils::from_chars<int>(params[0])) {
1149 top = right = bottom = left = padding.value();
1151 ERR_DP <<
"~PAD() numeric argument (pad all sides) requires a single valid integer. Received: '" << params[0] <<
"'.";
1156 return std::make_unique<pad_modification>(top, right, bottom, left);
1162 const int depth = std::max<int>(0, utils::from_chars<int>(args).value_or(0));
1163 return std::make_unique<bl_modification>(depth);
1169 const std::string::size_type p100_pos = args.find(
'%');
1171 if(p100_pos == std::string::npos) {
1172 num = utils::from_chars<float>(args).value_or(0.0f);
1175 const std::string_view parsed_field = args.substr(0, p100_pos);
1176 num = utils::from_chars<float>(parsed_field).value_or(0.0f);
1180 return std::make_unique<o_modification>(num);
1190 const int r = utils::from_chars<int>(args).value_or(0);
1191 return std::make_unique<cs_modification>(r, 0, 0);
1197 const int g = utils::from_chars<int>(args).value_or(0);
1198 return std::make_unique<cs_modification>(0,
g, 0);
1204 const int b = utils::from_chars<int>(args).value_or(0);
1205 return std::make_unique<cs_modification>(0, 0,
b);
1229 int c[4] { 0, 0, 0, SDL_ALPHA_OPAQUE };
1232 for(
int i = 0; i < std::min<int>(factors.size(), 4); ++
i) {
1233 c[
i] = utils::from_chars<int>(factors[
i]).value_or(0);
1236 return std::make_unique<background_modification>(
color_t(
c[0],
c[1],
c[2],
c[3]));
1245 if(params.size() != 3 && params.size() != 4) {
1246 ERR_DP <<
"incorrect number of arguments in ~SWAP() function, they must be 3 or 4";
1250 channel redValue, greenValue, blueValue, alphaValue;
1252 if(params[0] ==
"red") {
1254 }
else if(params[0] ==
"green") {
1256 }
else if(params[0] ==
"blue") {
1258 }
else if(params[0] ==
"alpha") {
1261 ERR_DP <<
"unsupported argument value in ~SWAP() function: " << params[0];
1266 if(params[1] ==
"red") {
1268 }
else if(params[1] ==
"green") {
1270 }
else if(params[1] ==
"blue") {
1272 }
else if(params[1] ==
"alpha") {
1275 ERR_DP <<
"unsupported argument value in ~SWAP() function: " << params[0];
1279 if(params[2] ==
"red") {
1281 }
else if(params[2] ==
"green") {
1283 }
else if(params[2] ==
"blue") {
1285 }
else if(params[2] ==
"alpha") {
1288 ERR_DP <<
"unsupported argument value in ~SWAP() function: " << params[0];
1294 if(params.size() == 3) {
1297 if(params[3] ==
"red") {
1299 }
else if(params[3] ==
"green") {
1301 }
else if(params[3] ==
"blue") {
1303 }
else if(params[3] ==
"alpha") {
1306 ERR_DP <<
"unsupported argument value in ~SWAP() function: " << params[3];
1311 return std::make_unique<swap_modification>(redValue, greenValue, blueValue, alphaValue);
A color range definition is made of four reference RGB colors, used for calculating conversions from ...
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
Generic locator abstracting the location of an image.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
A modified priority queue used to order image modifications.
void push(std::unique_ptr< modification > &&mod)
Adds mod to the queue.
const modification & top() const
Returns a const reference to the top element in the queue.
std::map< int, std::vector< std::unique_ptr< modification > >, std::greater< int > > priorities_
Map from a mod's priority() to the mods having that priority.
void pop()
Removes the top element from the queue.
std::size_t size() const
Returns the number of elements in the queue.
Base abstract class for an image-path modification.
static modification_queue decode(const std::string &)
Decodes modifications from a modification string.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
pixel_callable(std::size_t index, color_t clr, int w, int h)
wfl::variant get_value(const std::string &key) const override
void get_inputs(wfl::formula_input_vector &inputs) const override
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
void store() const noexcept
Stores a copy the current exception to be rethrown.
Helper class for pinning SDL surfaces into memory.
utils::span< pixel_t > pixel_span() const
static const color_range get_side_color_range(int side)
int as_int(int fallback=0) const
Returns the variant's value as an integer.
constexpr uint8_t color_multiply(uint8_t n1, uint8_t n2)
Multiply two 8-bit colour values as if in the range [0.0,1.0].
constexpr uint8_t float_to_color(double n)
Convert a double in the range [0.0,1.0] to an 8-bit colour value.
color_range_map recolor_range(const color_range &new_range, const std::vector< color_t > &old_rgb)
Converts a source palette using the specified color_range object.
std::unordered_map< color_t, color_t > color_range_map
Definitions for the interface to Wesnoth Markup Language (WML).
#define REGISTER_MOD_PARSER(type, args_var)
A macro for automatic modification parser registration.
static lg::log_domain log_display("display")
Standard logging facilities (interface).
const std::vector< color_t > & tc_info(std::string_view name)
const color_range & color_info(std::string_view name)
Functions to load and save images from/to disk.
bool exists(const image::locator &i_locator)
Returns true if the given image actually exists, without loading it.
surface get_surface(const image::locator &i_locator, TYPE type, bool skip_cache)
Returns an image surface suitable for software manipulation.
std::string img(const std::string &src, const std::string &align, bool floating)
Generates a Help markup tag corresponding to an image.
std::size_t size(std::string_view str)
Length in characters of a UTF-8 string.
std::size_t index(std::string_view str, const std::size_t index)
Codepoint index corresponding to the nth character in a UTF-8 string.
@ STRIP_SPACES
REMOVE_EMPTY: remove empty elements.
std::vector< std::string_view > split_view(std::string_view s, const char sep, const int flags)
std::map< std::string, std::string > map_split(const std::string &val, char major, char minor, int flags, const std::string &default_value)
Splits a string based on two separators into a map.
std::vector< std::string > parenthetical_split(std::string_view val, const char separator, std::string_view left, std::string_view right, const int flags)
Splits a string based either on a separator, except then the text appears within specified parenthesi...
std::vector< std::string > split(const config_attribute_value &val)
std::vector< formula_input > formula_input_vector
rect src
Non-transparent portion of the surface to compose.
The basic class for representing 8-bit RGB or RGBA colour values.
static constexpr color_t from_argb_bytes(uint32_t c)
Creates a new color_t object from a uint32_t variable.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
Exception thrown by the operator() when an error occurs.
imod_exception(const std::stringstream &message_stream)
Constructor.
virtual void operator()(surface &src) const override
Applies the image-path modification on the specified surface.
An abstract description of a rectangle with integer coordinates.
constexpr int area() const
The area of this rectangle, in square pixels.
static map_location::direction s
void mask_surface(surface &nsurf, const surface &nmask, bool *empty_result, const std::string &filename)
Applies a mask on a surface.
surface get_surface_portion(const surface &src, rect &area)
Get a portion of the screen.
rect get_non_transparent_portion(const surface &nsurf)
void alpha_to_greyscale(surface &nsurf)
surface scale_surface_legacy(const surface &surf, int w, int h)
Scale a surface using simple bilinear filtering (discarding rgb from source pixels with 0 alpha)
void recolor_image(surface &nsurf, const color_range_map &map_rgb)
Recolors a surface using a map with source and converted palette values.
void wipe_alpha(surface &nsurf)
surface cut_surface(const surface &surf, const rect &r)
Cuts a rectangle from a surface.
void sepia_image(surface &nsurf)
void blend_surface(surface &nsurf, const double amount, const color_t color)
Blends a surface with a color.
void swap_channels_image(surface &nsurf, channel r, channel g, channel b, channel a)
void adjust_surface_color(surface &nsurf, int red, int green, int blue)
void negative_image(surface &nsurf, const int thresholdR, const int thresholdG, const int thresholdB)
void light_surface(surface &nsurf, const surface &lightmap)
Light surf using lightmap.
surface scale_surface_xbrz(const surface &surf, std::size_t z)
Scale a surface using xBRZ algorithm.
surface scale_surface_sharp(const surface &surf, int w, int h)
Scale a surface using modified nearest neighbour algorithm.
void blur_alpha_surface(surface &res, int depth)
Cross-fades a surface with alpha channel.
void greyscale_image(surface &nsurf)
void flop_surface(surface &nsurf)
surface rotate_90_surface(const surface &surf, bool clockwise)
Rotates a surface 90 degrees.
surface rotate_180_surface(const surface &surf)
Rotates a surface 180 degrees.
void flip_surface(surface &nsurf)
surface scale_surface(const surface &surf, int w, int h)
Scale a surface using alpha-weighted modified bilinear filtering Note: causes artifacts with alpha gr...
surface rotate_any_surface(const surface &surf, float angle, int zoom, int offset)
Rotates a surface by any degrees.
void monochrome_image(surface &nsurf, const int threshold)
void sdl_blit(const surface &src, const SDL_Rect *src_rect, surface &dst, SDL_Rect *dst_rect)