Module: PSDNative::Renderer::ClippingMask

Defined in:
ext/psd_native/psd_native_ext.c

Instance Method Summary collapse

Instance Method Details

#apply!Object



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
# File 'ext/psd_native/clipping_mask.c', line 3

VALUE psd_native_clipping_mask_apply_bang(VALUE self) {
  VALUE node = rb_iv_get(self, "@node");
  if (rb_funcall(node, rb_intern("clipped?"), 0) == Qfalse) {
    return Qnil;
  }

  psd_logger("debug", "Applying clipping mask with native code");

  VALUE canvas = rb_iv_get(self, "@canvas");
  VALUE canvas_pixels = psd_canvas_to_pixel_array(canvas);
  VALUE mask = rb_iv_get(self, "@mask");
  VALUE mask_pixels = psd_canvas_to_pixel_array(mask);

  uint32_t canvas_width = FIX2UINT(rb_funcall(canvas, rb_intern("width"), 0));
  uint32_t canvas_height = FIX2UINT(rb_funcall(canvas, rb_intern("height"), 0));
  int canvas_left = FIX2INT(rb_funcall(canvas, rb_intern("left"), 0));
  int canvas_top = FIX2INT(rb_funcall(canvas, rb_intern("top"), 0));

  uint32_t mask_width = FIX2UINT(rb_funcall(mask, rb_intern("width"), 0));
  uint32_t mask_height = FIX2UINT(rb_funcall(mask, rb_intern("height"), 0));
  int mask_left = FIX2INT(rb_funcall(mask, rb_intern("left"), 0));
  int mask_top = FIX2INT(rb_funcall(mask, rb_intern("top"), 0));
  int mask_pixel_length = mask_width * mask_height;

  int x, y, doc_x, doc_y, mask_x, mask_y;
  uint32_t alpha, color, loc;
  VALUE pixel;
  for (y = 0; y < canvas_height; y++) {
    for (x = 0; x < canvas_width; x++) {
      doc_x = canvas_left + x;
      doc_y = canvas_top + y;

      mask_x = doc_x - mask_left;
      mask_y = doc_y - mask_top;

      if (mask_x < 0 || mask_x > mask_width || mask_y < 0 || mask_y > mask_height) {
        alpha = 0;
      } else {
        loc = mask_y * mask_width + mask_x;

        if (loc > mask_pixel_length) {
          alpha = 0;
        } else {
          pixel = rb_ary_entry(mask_pixels, loc);
          alpha = pixel == Qnil ? 0 : A(FIX2UINT(pixel));
        }
      }

      color = FIX2UINT(rb_ary_entry(canvas_pixels, y * canvas_width + x));
      rb_ary_store(canvas_pixels, y * canvas_width + x, INT2FIX((color & 0xffffff00) | (A(color) * alpha / 255)));
    }
  }

  return Qnil;
}