A Simple X Image Viewer
Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
13 лет назад
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346
  1. /* sxiv: image.c
  2. * Copyright (c) 2011 Bert Muennich <muennich at informatik.hu-berlin.de>
  3. *
  4. * This program is free software; you can redistribute it and/or modify
  5. * it under the terms of the GNU General Public License as published by
  6. * the Free Software Foundation; either version 2 of the License, or
  7. * (at your option) any later version.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with this program; if not, write to the Free Software
  16. * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  17. */
  18. #include <unistd.h>
  19. #include <Imlib2.h>
  20. #include "config.h"
  21. #include "icon.h"
  22. #include "image.h"
  23. #include "options.h"
  24. #include "util.h"
  25. int zl_cnt;
  26. float zoom_min;
  27. float zoom_max;
  28. Imlib_Image *im_broken;
  29. void img_init(img_t *img, win_t *win) {
  30. zl_cnt = sizeof(zoom_levels) / sizeof(zoom_levels[0]);
  31. zoom_min = zoom_levels[0] / 100.0;
  32. zoom_max = zoom_levels[zl_cnt - 1] / 100.0;
  33. im_broken = imlib_create_image_using_data(32, 32, icon_broken);
  34. if (img) {
  35. img->zoom = options->zoom;
  36. img->zoom = MAX(img->zoom, zoom_min);
  37. img->zoom = MIN(img->zoom, zoom_max);
  38. img->aa = options->aa;
  39. }
  40. if (win) {
  41. imlib_context_set_display(win->env.dpy);
  42. imlib_context_set_visual(win->env.vis);
  43. imlib_context_set_colormap(win->env.cmap);
  44. }
  45. }
  46. void img_free(img_t* img) {
  47. if (img && img->valid && imlib_context_get_image())
  48. imlib_free_image();
  49. imlib_context_set_image(im_broken);
  50. imlib_free_image();
  51. }
  52. int _imlib_load_image(const char *filename) {
  53. Imlib_Image *im;
  54. if (!filename)
  55. return 0;
  56. if (access(filename, F_OK) || !(im = imlib_load_image(filename))) {
  57. warn("could not open file: %s", filename);
  58. return 0;
  59. }
  60. imlib_context_set_image(im);
  61. imlib_image_set_changes_on_disk();
  62. return 1;
  63. }
  64. int img_check(const char *filename) {
  65. int ret;
  66. if ((ret = _imlib_load_image(filename)))
  67. imlib_free_image();
  68. return ret;
  69. }
  70. int img_load(img_t *img, const char *filename) {
  71. if (!img || !filename)
  72. return 0;
  73. if (img->valid && imlib_context_get_image())
  74. imlib_free_image();
  75. if ((img->valid = _imlib_load_image(filename))) {
  76. imlib_context_set_anti_alias(img->aa);
  77. img->scalemode = options->scalemode;
  78. } else {
  79. imlib_context_set_image(im_broken);
  80. imlib_context_set_anti_alias(0);
  81. img->scalemode = SCALE_DOWN;
  82. }
  83. img->re = 0;
  84. img->checkpan = 0;
  85. img->w = imlib_image_get_width();
  86. img->h = imlib_image_get_height();
  87. return 1;
  88. }
  89. void img_check_pan(img_t *img, win_t *win) {
  90. if (!img || !win)
  91. return;
  92. if (img->w * img->zoom > win->w) {
  93. if (img->x > 0 && img->x + img->w * img->zoom > win->w)
  94. img->x = 0;
  95. if (img->x < 0 && img->x + img->w * img->zoom < win->w)
  96. img->x = win->w - img->w * img->zoom;
  97. } else {
  98. img->x = (win->w - img->w * img->zoom) / 2;
  99. }
  100. if (img->h * img->zoom > win->h) {
  101. if (img->y > 0 && img->y + img->h * img->zoom > win->h)
  102. img->y = 0;
  103. if (img->y < 0 && img->y + img->h * img->zoom < win->h)
  104. img->y = win->h - img->h * img->zoom;
  105. } else {
  106. img->y = (win->h - img->h * img->zoom) / 2;
  107. }
  108. }
  109. int img_fit(img_t *img, win_t *win) {
  110. float oz, zw, zh;
  111. if (!img || !win)
  112. return 0;
  113. oz = img->zoom;
  114. zw = (float) win->w / (float) img->w;
  115. zh = (float) win->h / (float) img->h;
  116. img->zoom = MIN(zw, zh);
  117. img->zoom = MAX(img->zoom, zoom_min);
  118. img->zoom = MIN(img->zoom, zoom_max);
  119. return oz != img->zoom;
  120. }
  121. void img_render(img_t *img, win_t *win) {
  122. int sx, sy, sw, sh;
  123. int dx, dy, dw, dh;
  124. if (!img || !win || !imlib_context_get_image())
  125. return;
  126. if (img->scalemode != SCALE_ZOOM) {
  127. img_fit(img, win);
  128. if (img->scalemode == SCALE_DOWN && img->zoom > 1.0)
  129. img->zoom = 1.0;
  130. }
  131. if (!img->re) {
  132. /* rendered for the first time */
  133. img->re = 1;
  134. img_center(img, win);
  135. }
  136. if (img->checkpan) {
  137. img_check_pan(img, win);
  138. img->checkpan = 0;
  139. }
  140. /* calculate source and destination offsets */
  141. if (img->x < 0) {
  142. sx = -img->x / img->zoom;
  143. sw = win->w / img->zoom;
  144. dx = 0;
  145. dw = win->w;
  146. } else {
  147. sx = 0;
  148. sw = img->w;
  149. dx = img->x;
  150. dw = img->w * img->zoom;
  151. }
  152. if (img->y < 0) {
  153. sy = -img->y / img->zoom;
  154. sh = win->h / img->zoom;
  155. dy = 0;
  156. dh = win->h;
  157. } else {
  158. sy = 0;
  159. sh = img->h;
  160. dy = img->y;
  161. dh = img->h * img->zoom;
  162. }
  163. win_clear(win);
  164. imlib_context_set_drawable(win->pm);
  165. imlib_render_image_part_on_drawable_at_size(sx, sy, sw, sh, dx, dy, dw, dh);
  166. win_draw(win);
  167. }
  168. int img_fit_win(img_t *img, win_t *win) {
  169. if (!img || !img->valid || !win)
  170. return 0;
  171. img->scalemode = SCALE_FIT;
  172. return img_fit(img, win);
  173. }
  174. int img_center(img_t *img, win_t *win) {
  175. int ox, oy;
  176. if (!img || !win)
  177. return 0;
  178. ox = img->x;
  179. oy = img->y;
  180. img->x = (win->w - img->w * img->zoom) / 2;
  181. img->y = (win->h - img->h * img->zoom) / 2;
  182. return ox != img->x || oy != img->y;
  183. }
  184. int img_zoom(img_t *img, float z) {
  185. if (!img || !img->valid)
  186. return 0;
  187. z = MAX(z, zoom_min);
  188. z = MIN(z, zoom_max);
  189. img->scalemode = SCALE_ZOOM;
  190. if (z != img->zoom) {
  191. img->x -= (img->w * z - img->w * img->zoom) / 2;
  192. img->y -= (img->h * z - img->h * img->zoom) / 2;
  193. img->zoom = z;
  194. img->checkpan = 1;
  195. return 1;
  196. } else {
  197. return 0;
  198. }
  199. }
  200. int img_zoom_in(img_t *img) {
  201. int i;
  202. if (!img || !img->valid)
  203. return 0;
  204. for (i = 1; i < zl_cnt; ++i) {
  205. if (zoom_levels[i] > img->zoom * 100.0)
  206. return img_zoom(img, zoom_levels[i] / 100.0);
  207. }
  208. return 0;
  209. }
  210. int img_zoom_out(img_t *img) {
  211. int i;
  212. if (!img || !img->valid)
  213. return 0;
  214. for (i = zl_cnt - 2; i >= 0; --i) {
  215. if (zoom_levels[i] < img->zoom * 100.0)
  216. return img_zoom(img, zoom_levels[i] / 100.0);
  217. }
  218. return 0;
  219. }
  220. int img_move(img_t *img, win_t *win, int dx, int dy) {
  221. int ox, oy;
  222. if (!img || !img->valid || !win)
  223. return 0;
  224. ox = img->x;
  225. oy = img->y;
  226. img->x += dx;
  227. img->y += dy;
  228. img_check_pan(img, win);
  229. return ox != img->x || oy != img->y;
  230. }
  231. int img_pan(img_t *img, win_t *win, pandir_t dir) {
  232. if (!img || !img->valid || !win)
  233. return 0;
  234. switch (dir) {
  235. case PAN_LEFT:
  236. return img_move(img, win, win->w / 5, 0);
  237. case PAN_RIGHT:
  238. return img_move(img, win, win->w / 5 * -1, 0);
  239. case PAN_UP:
  240. return img_move(img, win, 0, win->h / 5);
  241. case PAN_DOWN:
  242. return img_move(img, win, 0, win->h / 5 * -1);
  243. }
  244. return 0;
  245. }
  246. void img_rotate(img_t *img, win_t *win, int d) {
  247. int ox, oy, tmp;
  248. if (!img || !img->valid || !win)
  249. return;
  250. ox = d == 1 ? img->x : win->w - img->x - img->w * img->zoom;
  251. oy = d == 3 ? img->y : win->h - img->y - img->h * img->zoom;
  252. imlib_image_orientate(d);
  253. img->x = oy + (win->w - win->h) / 2;
  254. img->y = ox + (win->h - win->w) / 2;
  255. tmp = img->w;
  256. img->w = img->h;
  257. img->h = tmp;
  258. img->checkpan = 1;
  259. }
  260. void img_rotate_left(img_t *img, win_t *win) {
  261. img_rotate(img, win, 3);
  262. }
  263. void img_rotate_right(img_t *img, win_t *win) {
  264. img_rotate(img, win, 1);
  265. }
  266. void img_toggle_antialias(img_t *img) {
  267. if (!img || !img->valid)
  268. return;
  269. img->aa ^= 1;
  270. imlib_context_set_anti_alias(img->aa);
  271. }