428 rindas
12 KiB

  1. //
  2. // Lol Engine
  3. //
  4. // Copyright: (c) 2004-2014 Sam Hocevar <sam@hocevar.net>
  5. // This program is free software; you can redistribute it and/or
  6. // modify it under the terms of the Do What The Fuck You Want To
  7. // Public License, Version 2, as published by Sam Hocevar. See
  8. // http://www.wtfpl.net/ for more details.
  9. //
  10. #if defined HAVE_CONFIG_H
  11. # include "config.h"
  12. #endif
  13. #include "core.h"
  14. /*
  15. * Stock kernels
  16. */
  17. namespace lol
  18. {
  19. Array2D<float> Image::BayerKernel(ivec2 size)
  20. {
  21. Array2D<float> ret(size);
  22. int n = 1;
  23. while (n < size.x || n < size.y)
  24. n *= 2;
  25. for (int j = 0; j < size.y; j++)
  26. for (int i = 0; i < size.x; i++)
  27. {
  28. int x = 0;
  29. for (int k = 1, l = n * n / 4; k < n; k *= 2, l /= 4)
  30. {
  31. if ((i & k) && (j & k))
  32. x += l;
  33. else if (i & k)
  34. x += 3 * l;
  35. else if (j & k)
  36. x += 2 * l;
  37. }
  38. ret[i][j] = (float)(x + 1) / (n * n + 1);
  39. }
  40. return ret;
  41. }
  42. Array2D<float> Image::HalftoneKernel(ivec2 size)
  43. {
  44. Array2D<float> ret(size);
  45. for (int y = 0; y < size.y; y++)
  46. for (int x = 0; x < size.x; x++)
  47. {
  48. float dx = 2.f * x / size.x - 0.5f;
  49. float dy = 2.f * (y + 0.07f) / size.y - 0.5f;
  50. bool flip = false;
  51. if (dx > 0.5f)
  52. {
  53. flip = !flip;
  54. dx -= 1.0f;
  55. }
  56. if (dy > 0.5f)
  57. {
  58. flip = !flip;
  59. dy -= 1.0f;
  60. }
  61. /* Using dx²+dy² here creates another interesting halftone. */
  62. float r = - lol::cos(F_PI * (dx - dy)) - lol::cos(F_PI * (dx + dy));
  63. ret[x][y] = flip ? 10.f - r : r;
  64. }
  65. return NormalizeKernel(ret);
  66. }
  67. Array2D<float> Image::BlueNoiseKernel(ivec2 size)
  68. {
  69. /* FIXME: this function could be faster if we didn't do the convolution
  70. * each time and instead work on the convoluted matrix. */
  71. Array2D<float> ret(size);
  72. float const epsilon = 1.f / (size.x * size.y + 1);
  73. /* Create a small Gaussian kernel for filtering */
  74. ivec2 const gsize = ivec2(9, 9);
  75. Array2D<float> gaussian(gsize);
  76. for (int j = 0; j < gsize.y; ++j)
  77. for (int i = 0; i < gsize.x; ++i)
  78. {
  79. ivec2 const distance = gsize / 2 - ivec2(i, j);
  80. gaussian[i][j] = lol::exp(-lol::sqlength(distance)
  81. / (0.05f * gsize.x * gsize.y));
  82. }
  83. /* Helper function to find voids and clusters */
  84. auto best = [&] (Array2D<float> const &p, float val, float mul) -> ivec2
  85. {
  86. float maxval = -size.x * size.y;
  87. ivec2 coord(0, 0);
  88. for (int y = 0; y < size.y; ++y)
  89. for (int x = 0; x < size.x; ++x)
  90. {
  91. if (p[x][y] != val)
  92. continue;
  93. float total = 0.0f;
  94. for (int j = 0; j < gsize.y; ++j)
  95. for (int i = 0; i < gsize.x; ++i)
  96. total += gaussian[i][j] *
  97. p[(x + i - gsize.x / 2 + size.x) % size.x]
  98. [(y + j - gsize.y / 2 + size.y) % size.y];
  99. if (total * mul > maxval)
  100. {
  101. maxval = total * mul;
  102. coord = ivec2(x, y);
  103. }
  104. }
  105. return coord;
  106. };
  107. /* Generate an array with about 10% random dots */
  108. Array2D<float> dots(size);
  109. int const ndots = (size.x * size.y + 9) / 10;
  110. memset(dots.Data(), 0, dots.Bytes());
  111. for (int n = 0; n < ndots; )
  112. {
  113. int x = lol::rand(size.x);
  114. int y = lol::rand(size.y);
  115. if (dots[x][y])
  116. continue;
  117. dots[x][y] = 1.0f;
  118. ++n;
  119. }
  120. /* Rearrange 1s so that they occupy the largest voids */
  121. for (;;)
  122. {
  123. ivec2 bestcluster = best(dots, 1.0f, 1.0f);
  124. dots[bestcluster] = 0.0f;
  125. ivec2 bestvoid = best(dots, 0.0f, -1.0f);
  126. dots[bestvoid] = 1.0f;
  127. if (bestcluster == bestvoid)
  128. break;
  129. }
  130. /* Reorder all 1s and replace them with 0.0001 */
  131. for (int n = ndots; n--; )
  132. {
  133. ivec2 bestcluster = best(dots, 1.0f, 1.0f);
  134. ret[bestcluster] = (n + 1.0f) * epsilon;
  135. dots[bestcluster] = 0.0001f;
  136. }
  137. /* Reorder all 0s and replace them with 0.0001 */
  138. for (int n = ndots; n < size.x * size.y; ++n)
  139. {
  140. ivec2 bestvoid = best(dots, 0.0f, -1.0f);
  141. ret[bestvoid] = (n + 1.0f) * epsilon;
  142. dots[bestvoid] = 0.0001f;
  143. }
  144. return ret;
  145. }
  146. struct Dot
  147. {
  148. int x, y;
  149. float val;
  150. };
  151. static int cmpdot(const void *p1, const void *p2)
  152. {
  153. return ((Dot const *)p1)->val > ((Dot const *)p2)->val;
  154. }
  155. Array2D<float> Image::NormalizeKernel(Array2D<float> const &kernel)
  156. {
  157. ivec2 size = kernel.GetSize();
  158. Array<Dot> tmp;
  159. tmp.Resize(size.x * size.y);
  160. for (int y = 0; y < size.y; y++)
  161. for (int x = 0; x < size.x; x++)
  162. {
  163. tmp[y * size.x + x].x = x;
  164. tmp[y * size.x + x].y = y;
  165. tmp[y * size.x + x].val = kernel[x][y];
  166. }
  167. std::qsort(tmp.Data(), size.x * size.y, sizeof(Dot), cmpdot);
  168. Array2D<float> dst(size);
  169. float const epsilon = 1.f / (size.x * size.y + 1);
  170. for (int n = 0; n < size.x * size.y; n++)
  171. {
  172. int x = tmp[n].x;
  173. int y = tmp[n].y;
  174. dst[x][y] = (float)(n + 1) * epsilon;
  175. }
  176. return dst;
  177. }
  178. Array2D<float> Image::EdiffKernel(EdiffAlgorithm algorithm)
  179. {
  180. Array2D<float> ret;
  181. switch(algorithm)
  182. {
  183. case EdiffAlgorithm::FloydSteinberg:
  184. ret.SetSize(ivec2(3, 2));
  185. {
  186. static float const myker[] =
  187. {
  188. 0., 1., 7./16,
  189. 3./16, 5./16, 1./16,
  190. };
  191. memcpy(&ret[0][0], myker, sizeof(myker));
  192. }
  193. return ret;
  194. case EdiffAlgorithm::JaJuNi:
  195. ret.SetSize(ivec2(5, 3));
  196. {
  197. static float const myker[] =
  198. {
  199. 0., 0., 1., 7./48, 5./48,
  200. 3./48, 5./48, 7./48, 5./48, 3./48,
  201. 1./48, 3./48, 5./48, 3./48, 1./48,
  202. };
  203. memcpy(&ret[0][0], myker, sizeof(myker));
  204. }
  205. return ret;
  206. case EdiffAlgorithm::Atkinson:
  207. ret.SetSize(ivec2(4, 3));
  208. {
  209. static float const myker[] =
  210. {
  211. 0., 1., 1./8, 1./8,
  212. 1./8, 1./8, 1./8, 0.,
  213. 0., 1./8, 0., 0.,
  214. };
  215. memcpy(&ret[0][0], myker, sizeof(myker));
  216. }
  217. return ret;
  218. case EdiffAlgorithm::Fan:
  219. ret.SetSize(ivec2(4, 2));
  220. {
  221. static float const myker[] =
  222. {
  223. 0., 0., 1., 7./16,
  224. 1./16, 3./16, 5./16, 0.,
  225. };
  226. memcpy(&ret[0][0], myker, sizeof(myker));
  227. }
  228. return ret;
  229. case EdiffAlgorithm::ShiauFan:
  230. ret.SetSize(ivec2(4, 2));
  231. {
  232. static float const myker[] =
  233. {
  234. 0., 0., 1., 1./2,
  235. 1./8, 1./8, 1./4, 0.,
  236. };
  237. memcpy(&ret[0][0], myker, sizeof(myker));
  238. }
  239. return ret;
  240. case EdiffAlgorithm::ShiauFan2:
  241. ret.SetSize(ivec2(5, 2));
  242. {
  243. static float const myker[] =
  244. {
  245. 0., 0., 0., 1., 1./2,
  246. 1./16, 1./16, 1./8, 1./4, 0.,
  247. };
  248. memcpy(&ret[0][0], myker, sizeof(myker));
  249. }
  250. return ret;
  251. case EdiffAlgorithm::Stucki:
  252. ret.SetSize(ivec2(5, 3));
  253. {
  254. static float const myker[] =
  255. {
  256. 0., 0., 1., 8./42, 4./42,
  257. 2./42, 4./42, 8./42, 4./42, 2./42,
  258. 1./42, 2./42, 4./42, 2./42, 1./42,
  259. };
  260. memcpy(&ret[0][0], myker, sizeof(myker));
  261. }
  262. return ret;
  263. case EdiffAlgorithm::Burkes:
  264. ret.SetSize(ivec2(5, 2));
  265. {
  266. static float const myker[] =
  267. {
  268. 0., 0., 1., 4./16, 2./16,
  269. 1./16, 2./16, 4./16, 2./16, 1./16,
  270. };
  271. memcpy(&ret[0][0], myker, sizeof(myker));
  272. }
  273. return ret;
  274. case EdiffAlgorithm::Sierra:
  275. ret.SetSize(ivec2(5, 3));
  276. {
  277. static float const myker[] =
  278. {
  279. 0., 0., 1., 5./32, 3./32,
  280. 2./32, 4./32, 5./32, 4./32, 2./32,
  281. 0., 2./32, 3./32, 2./32, 0.,
  282. };
  283. memcpy(&ret[0][0], myker, sizeof(myker));
  284. }
  285. return ret;
  286. case EdiffAlgorithm::Sierra2:
  287. ret.SetSize(ivec2(5, 2));
  288. {
  289. static float const myker[] =
  290. {
  291. 0., 0., 1., 4./16, 3./16,
  292. 1./16, 2./16, 3./16, 2./16, 1./16,
  293. };
  294. memcpy(&ret[0][0], myker, sizeof(myker));
  295. }
  296. return ret;
  297. case EdiffAlgorithm::Lite:
  298. ret.SetSize(ivec2(3, 2));
  299. {
  300. static float const myker[] =
  301. {
  302. 0., 1., 1./2,
  303. 1./4, 1./4, 0.,
  304. };
  305. memcpy(&ret[0][0], myker, sizeof(myker));
  306. }
  307. return ret;
  308. }
  309. return ret;
  310. }
  311. /* Any standard deviation below this value will be rounded up, in order
  312. * to avoid ridiculously low values. exp(-1/(2*0.2*0.2)) is < 10^-5 so
  313. * there is little chance that any value below 0.2 will be useful. */
  314. #define BLUR_EPSILON 0.2f
  315. Array2D<float> Image::GaussianKernel(vec2 radius, float angle, vec2 delta)
  316. {
  317. Array2D<float> kernel;
  318. if (radius.x < BLUR_EPSILON)
  319. radius.x = BLUR_EPSILON;
  320. if (radius.y < BLUR_EPSILON)
  321. radius.y = BLUR_EPSILON;
  322. float const sint = lol::sin(angle);
  323. float const cost = lol::cos(angle);
  324. /* Compute the final ellipse's bounding box */
  325. float const bbx = lol::sqrt(sq(radius.x * cost) + sq(radius.y * sint));
  326. float const bby = lol::sqrt(sq(radius.y * cost) + sq(radius.x * sint));
  327. /* FIXME: the kernel becomes far too big with large values of dx, because
  328. * we grow both left and right. Fix the growing direction. */
  329. int const krx = (int)(3.f * bbx + .99999f + lol::ceil(lol::abs(delta.x)));
  330. int const kry = (int)(3.f * bby + .99999f + lol::ceil(lol::abs(delta.y)));
  331. ivec2 size(2 * krx + 1, 2 * kry + 1);
  332. float const Kx = -1.f / (2.f * radius.x * radius.x);
  333. float const Ky = -1.f / (2.f * radius.y * radius.y);
  334. kernel.SetSize(size);
  335. float t = 0.f;
  336. for (int j = -kry; j <= kry; j++)
  337. {
  338. for (int i = -krx; i <= krx; i++)
  339. {
  340. /* FIXME: this level of interpolation sucks. We should
  341. * interpolate on the full NxN grid for better quality. */
  342. static vec3 const samples[] =
  343. {
  344. vec3( 0.0f, 0.0f, 1.0f),
  345. vec3(-0.4f, -0.4f, 0.8f),
  346. vec3(-0.3f, 0.0f, 0.9f),
  347. vec3(-0.4f, 0.4f, 0.8f),
  348. vec3( 0.0f, 0.3f, 0.9f),
  349. vec3( 0.4f, 0.4f, 0.8f),
  350. vec3( 0.3f, 0.0f, 0.9f),
  351. vec3( 0.4f, -0.4f, 0.8f),
  352. vec3( 0.0f, -0.3f, 0.9f),
  353. };
  354. float d = 0.f;
  355. for (auto p : samples)
  356. {
  357. float u = (i + p.x) * cost - (j + p.y) * sint + delta.x;
  358. float v = (i + p.x) * sint + (j + p.y) * cost + delta.y;
  359. float ex = Kx * u * u;
  360. float ey = Ky * v * v;
  361. d += p.z * lol::exp(ex + ey);
  362. /* Do not interpolate if this is a standard gaussian. */
  363. if (!delta.x && !delta.y && !angle)
  364. break;
  365. }
  366. kernel[i + krx][j + kry] = d;
  367. t += d;
  368. }
  369. }
  370. for (int j = 0; j < size.y; j++)
  371. for (int i = 0; i < size.x; i++)
  372. kernel[i][j] *= (1.f / t);
  373. return kernel;
  374. }
  375. } /* namespace lol */