function MaskElement(data, element, globalData) {
  this.data = data;
  this.element = element;
  this.globalData = globalData;
  this.storedData = [];
  this.masksProperties = this.data.masksProperties || [];
  this.maskElement = null;
  var defs = this.globalData.defs;
  var i, len = this.masksProperties ? this.masksProperties.length : 0;
  this.viewData = createSizedArray(len);
  this.solidPath = '';

  var path, properties = this.masksProperties;
  var count = 0;
  var currentMasks = [];
  var j, jLen;
  var layerId = createElementID();
  var rect, expansor, feMorph, x;
  var maskType = 'clipPath', maskRef = 'clip-path';
  for (i = 0; i < len; i++) {
    if ((properties[i].mode !== 'a' && properties[i].mode !== 'n') || properties[i].inv || properties[i].o.k !== 100 || properties[i].o.x) {
      maskType = 'mask';
      maskRef = 'mask';
    }

    if ((properties[i].mode == 's' || properties[i].mode == 'i') && count === 0) {
      rect = createNS('rect');
      rect.setAttribute('fill', '#ffffff');
      rect.setAttribute('width', this.element.comp.data.w || 0);
      rect.setAttribute('height', this.element.comp.data.h || 0);
      currentMasks.push(rect);
    } else {
      rect = null;
    }

    path = createNS('path');
    if (properties[i].mode == 'n') {
      // TODO move this to a factory or to a constructor
      this.viewData[i] = {
        op: PropertyFactory.getProp(this.element, properties[i].o, 0, 0.01, this.element),
        prop: ShapePropertyFactory.getShapeProp(this.element, properties[i], 3),
        elem: path,
        lastPath: '',
      };
      defs.appendChild(path);
      continue;
    }
    count += 1;

    path.setAttribute('fill', properties[i].mode === 's' ? '#000000' : '#ffffff');
    path.setAttribute('clip-rule', 'nonzero');
    var filterID;

    if (properties[i].x.k !== 0) {
      maskType = 'mask';
      maskRef = 'mask';
      x = PropertyFactory.getProp(this.element, properties[i].x, 0, null, this.element);
      filterID = createElementID();
      expansor = createNS('filter');
      expansor.setAttribute('id', filterID);
      feMorph = createNS('feMorphology');
      feMorph.setAttribute('operator', 'erode');
      feMorph.setAttribute('in', 'SourceGraphic');
      feMorph.setAttribute('radius', '0');
      expansor.appendChild(feMorph);
      defs.appendChild(expansor);
      path.setAttribute('stroke', properties[i].mode === 's' ? '#000000' : '#ffffff');
    } else {
      feMorph = null;
      x = null;
    }

    // TODO move this to a factory or to a constructor
    this.storedData[i] = {
      elem: path,
      x: x,
      expan: feMorph,
      lastPath: '',
      lastOperator: '',
      filterId: filterID,
      lastRadius: 0,
    };
    if (properties[i].mode == 'i') {
      jLen = currentMasks.length;
      var g = createNS('g');
      for (j = 0; j < jLen; j += 1) {
        g.appendChild(currentMasks[j]);
      }
      var mask = createNS('mask');
      mask.setAttribute('mask-type', 'alpha');
      mask.setAttribute('id', layerId + '_' + count);
      mask.appendChild(path);
      defs.appendChild(mask);
      g.setAttribute('mask', 'url(' + locationHref + '#' + layerId + '_' + count + ')');

      currentMasks.length = 0;
      currentMasks.push(g);
    } else {
      currentMasks.push(path);
    }
    if (properties[i].inv && !this.solidPath) {
      this.solidPath = this.createLayerSolidPath();
    }
    // TODO move this to a factory or to a constructor
    this.viewData[i] = {
      elem: path,
      lastPath: '',
      op: PropertyFactory.getProp(this.element, properties[i].o, 0, 0.01, this.element),
      prop: ShapePropertyFactory.getShapeProp(this.element, properties[i], 3),
      invRect: rect,
    };
    if (!this.viewData[i].prop.k) {
      this.drawPath(properties[i], this.viewData[i].prop.v, this.viewData[i]);
    }
  }

  this.maskElement = createNS(maskType);

  len = currentMasks.length;
  for (i = 0; i < len; i += 1) {
    this.maskElement.appendChild(currentMasks[i]);
  }

  if (count > 0) {
    this.maskElement.setAttribute('id', layerId);
    this.element.maskedElement.setAttribute(maskRef, 'url(' + locationHref + '#' + layerId + ')');
    defs.appendChild(this.maskElement);
  }
  if (this.viewData.length) {
    this.element.addRenderableComponent(this);
  }
}

MaskElement.prototype.getMaskProperty = function (pos) {
  return this.viewData[pos].prop;
};

MaskElement.prototype.renderFrame = function (isFirstFrame) {
  var finalMat = this.element.finalTransform.mat;
  var i, len = this.masksProperties.length;
  for (i = 0; i < len; i++) {
    if (this.viewData[i].prop._mdf || isFirstFrame) {
      this.drawPath(this.masksProperties[i], this.viewData[i].prop.v, this.viewData[i]);
    }
    if (this.viewData[i].op._mdf || isFirstFrame) {
      this.viewData[i].elem.setAttribute('fill-opacity', this.viewData[i].op.v);
    }
    if (this.masksProperties[i].mode !== 'n') {
      if (this.viewData[i].invRect && (this.element.finalTransform.mProp._mdf || isFirstFrame)) {
        this.viewData[i].invRect.setAttribute('transform', finalMat.getInverseMatrix().to2dCSS());
      }
      if (this.storedData[i].x && (this.storedData[i].x._mdf || isFirstFrame)) {
        var feMorph = this.storedData[i].expan;
        if (this.storedData[i].x.v < 0) {
          if (this.storedData[i].lastOperator !== 'erode') {
            this.storedData[i].lastOperator = 'erode';
            this.storedData[i].elem.setAttribute('filter', 'url(' + locationHref + '#' + this.storedData[i].filterId + ')');
          }
          feMorph.setAttribute('radius', -this.storedData[i].x.v);
        } else {
          if (this.storedData[i].lastOperator !== 'dilate') {
            this.storedData[i].lastOperator = 'dilate';
            this.storedData[i].elem.setAttribute('filter', null);
          }
          this.storedData[i].elem.setAttribute('stroke-width', this.storedData[i].x.v * 2);
        }
      }
    }
  }
};

MaskElement.prototype.getMaskelement = function () {
  return this.maskElement;
};

MaskElement.prototype.createLayerSolidPath = function () {
  var path = 'M0,0 ';
  path += ' h' + this.globalData.compSize.w;
  path += ' v' + this.globalData.compSize.h;
  path += ' h-' + this.globalData.compSize.w;
  path += ' v-' + this.globalData.compSize.h + ' ';
  return path;
};

MaskElement.prototype.drawPath = function (pathData, pathNodes, viewData) {
  var pathString = ' M' + pathNodes.v[0][0] + ',' + pathNodes.v[0][1];
  var i, len;
  len = pathNodes._length;
  for (i = 1; i < len; i += 1) {
    // pathString += " C"+pathNodes.o[i-1][0]+','+pathNodes.o[i-1][1] + " "+pathNodes.i[i][0]+','+pathNodes.i[i][1] + " "+pathNodes.v[i][0]+','+pathNodes.v[i][1];
    pathString += ' C' + pathNodes.o[i - 1][0] + ',' + pathNodes.o[i - 1][1] + ' ' + pathNodes.i[i][0] + ',' + pathNodes.i[i][1] + ' ' + pathNodes.v[i][0] + ',' + pathNodes.v[i][1];
  }
  // pathString += " C"+pathNodes.o[i-1][0]+','+pathNodes.o[i-1][1] + " "+pathNodes.i[0][0]+','+pathNodes.i[0][1] + " "+pathNodes.v[0][0]+','+pathNodes.v[0][1];
  if (pathNodes.c && len > 1) {
    pathString += ' C' + pathNodes.o[i - 1][0] + ',' + pathNodes.o[i - 1][1] + ' ' + pathNodes.i[0][0] + ',' + pathNodes.i[0][1] + ' ' + pathNodes.v[0][0] + ',' + pathNodes.v[0][1];
  }
  // pathNodes.__renderedString = pathString;

  if (viewData.lastPath !== pathString) {
    var pathShapeValue = '';
    if (viewData.elem) {
      if (pathNodes.c) {
        pathShapeValue = pathData.inv ? this.solidPath + pathString : pathString;
      }
      viewData.elem.setAttribute('d', pathShapeValue);
    }
    viewData.lastPath = pathString;
  }
};

MaskElement.prototype.destroy = function () {
  this.element = null;
  this.globalData = null;
  this.maskElement = null;
  this.data = null;
  this.masksProperties = null;
};
