-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathforced-random-dither.cpp
116 lines (95 loc) · 3.02 KB
/
forced-random-dither.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
/*!
force-random-dither.cpp -- Generate a dither matrix using the force-random-dither method from:
W. Purgathofer, R. F. Tobler and M. Geiler.
"Forced random dithering: improved threshold matrices for ordered dithering"
Image Processing, 1994. Proceedings. ICIP-94., IEEE International Conference,
Austin, TX, 1994, pp. 1032-1035 vol.2.
doi: 10.1109/ICIP.1994.413512
*/
//
// Copyright (C) Wojciech Jarosz <wjarosz@gmail.com>. All rights reserved.
// Use of this source code is governed by a BSD-style license that can
// be found in the LICENSE.txt file.
//
#include <iostream>
#include <Eigen/Dense>
#include <vector>
#include <algorithm> // std::random_shuffle
#include <ctime> // std::time
#include <cstdlib> // std::rand, std::srand
using Eigen::MatrixXd;
using Eigen::ArrayXXd;
using Eigen::Vector2i;
using namespace std;
const int Sm = 128;
const int Smk = Sm*Sm;
double toroidalMinimumDistance(const Vector2i & a, const Vector2i & b)
{
int x0 = min(a.x(), b.x());
int x1 = max(a.x(), b.x());
int y0 = min(a.y(), b.y());
int y1 = max(a.y(), b.y());
double deltaX = min(x1-x0, x0+Sm-x1);
double deltaY = min(y1-y0, y0+Sm-y1);
return sqrt(deltaX*deltaX + deltaY*deltaY);
}
double force(double r)
{
return exp(-sqrt(2*r));
}
int main(int, char **)
{
srand(unsigned ( std::time(0) ) );
vector<Vector2i> freeLocations;
ArrayXXd M = ArrayXXd::Zero(Sm,Sm);
ArrayXXd forceField = ArrayXXd::Zero(Sm,Sm);
// initialize free locations
for (int y = 0; y < Sm; ++y)
for (int x = 0; x < Sm; ++x)
freeLocations.push_back(Vector2i(x,y));
for (int ditherValue = 0; ditherValue < Smk; ++ditherValue)
{
random_shuffle(freeLocations.begin(), freeLocations.end());
double minimum = 1e20f;
Vector2i minimumLocation(0,0);
// int halfP = freeLocations.size();
int halfP = min(max(1, (int)sqrt(freeLocations.size()*3/4)), (int)freeLocations.size());
// int halfP = min(10, (int)freeLocations.size());
for (int i = 0; i < halfP; ++i)
{
const Vector2i & location = freeLocations[i];
if (forceField(location.x(), location.y()) < minimum)
{
minimum = forceField(location.x(), location.y());
minimumLocation = location;
}
}
Vector2i cell(0,0);
for (cell.y() = 0; cell.y() < Sm; ++cell.y())
for (cell.x() = 0; cell.x() < Sm; ++cell.x())
{
double r = toroidalMinimumDistance(cell, minimumLocation);
forceField(cell.x(), cell.y()) += force(r);
}
freeLocations.erase(remove(freeLocations.begin(), freeLocations.end(), minimumLocation), freeLocations.end());
M(minimumLocation.x(), minimumLocation.y()) = ditherValue;
// if (ditherValue % 16 == 0)
// {
// std::stringstream ss;
// ss << ditherValue;
// writeEXR("forceField-" + ss.str() + ".exr", forceField/(ditherValue+1));
// }
}
std::cout << M << std::endl;
cout << "unsigned dither_matrix[" << Smk << "] = \n{\n ";
printf("%5d", (int)M(0));
for (int i = 1; i < M.size(); ++i)
{
cout << ", ";
if (i % Sm == 0)
cout << endl << " ";
printf("%5d", (int)M(i));
}
cout << "\n};" << endl;
return 0;
}