/**
 * DOM generation class
 */
exports.DOM = DOM;
exports.htmlEscape = htmlEscape;
exports.normalizeHeaderToId = normalizeHeaderToId;
//////////////////////////////////////////////////////////
function htmlEscape(text){
  return text
          .replace(/&/g, '&')
          .replace(//g, '>')
          .replace(/\{\{/g, '{{')
          .replace(/\}\}/g, '}}');
}
function nonEmpty(header) {
  return !!header;
}
function idFromCurrentHeaders(headers) {
  if (headers.length === 1) return headers[0];
  // Do not include the first level title, as that's the title of the page.
  return headers.slice(1).filter(nonEmpty).join('_');
}
function normalizeHeaderToId(header) {
  if (typeof header !== 'string') {
    return '';
  }
  return header.toLowerCase()
      .replace(/<.*>/g, '')         // html tags
      .replace(/[\!\?\:\.\']/g, '') // special characters
      .replace(/\d\d;/g, '')      // html entities
      .replace(/\(.*\)/mg, '')      // stuff in parenthesis
      .replace(/\s$/, '')           // trailing spaces
      .replace(/\s+/g, '-');        // replace whitespaces with dashes
}
function DOM() {
  this.out = [];
  this.headingDepth = 0;
  this.currentHeaders = [];
  this.anchors = [];
}
var INLINE_TAGS = {
    i: true,
    b: true,
    a: true
};
DOM.prototype = {
  toString: function() {
    return this.out.join('');
  },
  text: function(content) {
    if (typeof content == "string") {
      this.out.push(htmlEscape(content));
    } else if (typeof content == 'function') {
      content.call(this, this);
    } else if (content instanceof Array) {
      this.ul(content);
    }
  },
  html: function(html) {
    if (!html) return;
    var self = this;
    // rewrite header levels, add ids and collect the ids
    html = html.replace(/([\s\S]+?)<\/h\1>/gm, function(_, level, attrs, content) {
      level = parseInt(level, 10) + self.headingDepth; // change header level based on the context
      self.currentHeaders[level - 1] = normalizeHeaderToId(content);
      self.currentHeaders.length = level;
      var id = idFromCurrentHeaders(self.currentHeaders);
      self.anchors.push(id);
      return '' + content + '';
    });
    // collect anchors
    html = html.replace(//g, function(match, anchor) {
      self.anchors.push(anchor);
      return match;
    });
    this.out.push(html);
  },
  tag: function(name, attr, text) {
    if (!text) {
      text = attr;
      attr = {};
      if (name == 'code')
        attr['ng:non-bindable'] = '';
    }
    this.out.push('<' + name);
    for(var key in attr) {
      this.out.push(" " + key + '="' + attr[key] + '"');
    }
    this.out.push('>');
    this.text(text);
    this.out.push('' + name + '>');
    if (!INLINE_TAGS[name])
      this.out.push('\n');
  },
  code: function(text) {
    this.tag('pre', {'class':"prettyprint linenums"}, text);
  },
  div: function(attr, text) {
    this.tag('div', attr, text);
  },
  h: function(heading, content, fn){
    if (content==undefined || (content instanceof Array && content.length == 0)) return;
    this.headingDepth++;
    this.currentHeaders[this.headingDepth - 1] = normalizeHeaderToId(heading);
    this.currentHeaders.length = this.headingDepth;
    var className = null,
        anchor = null;
    if (typeof heading == 'string') {
      var id = idFromCurrentHeaders(this.currentHeaders);
      this.anchors.push(id);
      anchor = {'id': id};
      var classNameValue = this.currentHeaders[this.headingDepth - 1]
      if(classNameValue == 'hide') classNameValue = '';
      className = {'class': classNameValue};
    }
    this.tag('h' + this.headingDepth, anchor, heading);
    if (content instanceof Array) {
      this.ul(content, className, fn);
    } else if (fn) {
      this.tag('div', className, function() {
        fn.call(this, content);
      });
    } else {
      this.tag('div', className, content);
    }
    this.headingDepth--;
  },
  h1: function(attr, text) {
    this.tag('h1', attr, text);
  },
  h2: function(attr, text) {
    this.tag('h2', attr, text);
  },
  h3: function(attr, text) {
    this.tag('h3', attr, text);
  },
  p: function(attr, text) {
    this.tag('p', attr, text);
  },
  ul: function(list, attr, fn) {
    if (typeof attr == 'function') {
      fn = attr;
      attr = {};
    }
    this.tag('ul', attr, function(dom){
      list.forEach(function(item){
        dom.out.push('');
        dom.text(fn ? fn(item) : item);
        dom.out.push('\n');
      });
    });
  }
};