-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathimage-factory.cpp
121 lines (115 loc) · 3.83 KB
/
image-factory.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
#include <CLI/CLI.hpp>
#include <cmath>
#include <fstream>
#include <iostream>
#include <opencv2/opencv.hpp>
#include <sstream>
#include <string>
#include <vector>
using namespace cv;
using namespace std;
typedef Point3_<uint8_t> Pixel;
class Color {
int r;
int g;
int b;
float r_scale;
float g_scale;
float b_scale;
public:
int relative_dist(Pixel &pixel) {
// this uses the naive algorithm, there are better (more complicated) ways
// omits square root from distance formula for speed
// RGB seems to be more effective than HSV for this use-case
return r_scale * pow(r - pixel.z, 2) + g_scale * pow(g - pixel.y, 2) +
b_scale * pow(b - pixel.x, 2);
};
void set_pixel_color(Pixel &pixel) {
pixel.x = b;
pixel.y = g;
pixel.z = r;
};
Color(string hexstring, int r_scale, int g_scale, int b_scale) {
int hex = std::stoul(hexstring, nullptr, 16);
r = ((hex >> 16) & 0xFF); // Extract the RR byte
g = ((hex >> 8) & 0xFF); // Extract the GG byte
b = ((hex)&0xFF); // Extract the BB byte
this->r_scale = r_scale;
this->g_scale = g_scale;
this->b_scale = b_scale;
}
};
void snap_pixel(Pixel &pixel, vector<Color> colors) {
vector<int> dist;
for (auto color : colors) {
dist.push_back(color.relative_dist(pixel));
};
auto it = min_element(begin(dist), end(dist));
auto index = distance(begin(dist), it);
colors[index].set_pixel_color(pixel);
}
void convert_image(string input_image_path, string palette_path,
string output_image_path, int r_scale, int g_scale,
int b_scale, int average, int blur, float blend) {
Mat image;
image = imread(input_image_path, 1);
auto ogImage = image.clone();
if (average > 1) {
auto kernel =
Mat::ones(average, average, CV_32F) / (float)(average * average);
filter2D(image, image, -1, kernel, Point(-1, -1), 0, BORDER_DEFAULT);
}
vector<Color> colors;
ifstream palette(palette_path);
if (palette.is_open()) {
string hex_color;
while (getline(palette, hex_color)) {
colors.push_back(Color(hex_color, r_scale, g_scale, b_scale));
};
}
image.forEach<Pixel>([colors](Pixel &pixel, const int *position) -> void {
snap_pixel(pixel, colors);
});
if (blend < 1.0) {
addWeighted(image, blend, ogImage, 1.0 - blend, 0.0, image);
}
if (blur > 1) {
auto kernel = Mat::ones(blur, blur, CV_32F) / (float)(blur * blur);
filter2D(image, image, -1, kernel, Point(-1, -1), 0, BORDER_DEFAULT);
}
imwrite(output_image_path, image);
}
int main(int argc, char **argv) {
CLI::App app{"apply color palette to image"};
string palette_path;
app.add_option("-p,--palette", palette_path, "path to palette file")
->required()
->check(CLI::ExistingFile);
string input_image_path;
app.add_option("-i,--input", input_image_path, "Path to input image")
->required()
->check(CLI::ExistingFile);
string output_image_path;
app.add_option("-o,--output", output_image_path, "Path to output image")
->required();
int r_scale;
app.add_option("-r,--red", r_scale, "Red scale factor")->default_val(2);
int g_scale;
app.add_option("-g,--green", g_scale, "Green scale factor")->default_val(4);
int b_scale;
app.add_option("-b,--blue", b_scale, "Blue scale factor")->default_val(3);
int average = 1;
app.add_option("-a, --average", average,
"Size of pixel square to average over")
->default_val(1);
float blend = 1;
app.add_option("--blend", blend,
"Blend ratio between original and final image")
->default_val(1);
int blur = 1;
app.add_option("-u, --blur", blur, "Size of pixel square to blur over")
->default_val(1);
CLI11_PARSE(app, argc, argv)
convert_image(input_image_path, palette_path, output_image_path, r_scale,
g_scale, b_scale, average, blur, blend);
};