Blame src/emitter.cpp

Packit bfcc33
#include "sass.hpp"
Packit bfcc33
#include "util.hpp"
Packit bfcc33
#include "context.hpp"
Packit bfcc33
#include "output.hpp"
Packit bfcc33
#include "emitter.hpp"
Packit bfcc33
#include "utf8_string.hpp"
Packit bfcc33
Packit bfcc33
namespace Sass {
Packit bfcc33
Packit bfcc33
  Emitter::Emitter(struct Sass_Output_Options& opt)
Packit bfcc33
  : wbuf(),
Packit bfcc33
    opt(opt),
Packit bfcc33
    indentation(0),
Packit bfcc33
    scheduled_space(0),
Packit bfcc33
    scheduled_linefeed(0),
Packit bfcc33
    scheduled_delimiter(false),
Packit bfcc33
    scheduled_mapping(0),
Packit bfcc33
    in_comment(false),
Packit bfcc33
    in_wrapped(false),
Packit bfcc33
    in_media_block(false),
Packit bfcc33
    in_declaration(false),
Packit bfcc33
    in_space_array(false),
Packit bfcc33
    in_comma_array(false)
Packit bfcc33
  { }
Packit bfcc33
Packit bfcc33
  // return buffer as string
Packit bfcc33
  std::string Emitter::get_buffer(void)
Packit bfcc33
  {
Packit bfcc33
    return wbuf.buffer;
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  Sass_Output_Style Emitter::output_style(void) const
Packit bfcc33
  {
Packit bfcc33
    return opt.output_style;
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  // PROXY METHODS FOR SOURCE MAPS
Packit bfcc33
Packit bfcc33
  void Emitter::add_source_index(size_t idx)
Packit bfcc33
  { wbuf.smap.source_index.push_back(idx); }
Packit bfcc33
Packit bfcc33
  std::string Emitter::render_srcmap(Context &ctx)
Packit bfcc33
  { return wbuf.smap.render_srcmap(ctx); }
Packit bfcc33
Packit bfcc33
  void Emitter::set_filename(const std::string& str)
Packit bfcc33
  { wbuf.smap.file = str; }
Packit bfcc33
Packit bfcc33
  void Emitter::schedule_mapping(const AST_Node_Ptr node)
Packit bfcc33
  { scheduled_mapping = node; }
Packit bfcc33
  void Emitter::add_open_mapping(const AST_Node_Ptr node)
Packit bfcc33
  { wbuf.smap.add_open_mapping(node); }
Packit bfcc33
  void Emitter::add_close_mapping(const AST_Node_Ptr node)
Packit bfcc33
  { wbuf.smap.add_close_mapping(node); }
Packit bfcc33
  ParserState Emitter::remap(const ParserState& pstate)
Packit bfcc33
  { return wbuf.smap.remap(pstate); }
Packit bfcc33
Packit bfcc33
  // MAIN BUFFER MANIPULATION
Packit bfcc33
Packit bfcc33
  // add outstanding delimiter
Packit bfcc33
  void Emitter::finalize(bool final)
Packit bfcc33
  {
Packit bfcc33
    scheduled_space = 0;
Packit bfcc33
    if (output_style() == SASS_STYLE_COMPRESSED)
Packit bfcc33
      if (final) scheduled_delimiter = false;
Packit bfcc33
    if (scheduled_linefeed)
Packit bfcc33
      scheduled_linefeed = 1;
Packit bfcc33
    flush_schedules();
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  // flush scheduled space/linefeed
Packit bfcc33
  void Emitter::flush_schedules(void)
Packit bfcc33
  {
Packit bfcc33
    // check the schedule
Packit bfcc33
    if (scheduled_linefeed) {
Packit bfcc33
      std::string linefeeds = "";
Packit bfcc33
Packit bfcc33
      for (size_t i = 0; i < scheduled_linefeed; i++)
Packit bfcc33
        linefeeds += opt.linefeed;
Packit bfcc33
      scheduled_space = 0;
Packit bfcc33
      scheduled_linefeed = 0;
Packit bfcc33
      append_string(linefeeds);
Packit bfcc33
Packit bfcc33
    } else if (scheduled_space) {
Packit bfcc33
      std::string spaces(scheduled_space, ' ');
Packit bfcc33
      scheduled_space = 0;
Packit bfcc33
      append_string(spaces);
Packit bfcc33
    }
Packit bfcc33
    if (scheduled_delimiter) {
Packit bfcc33
      scheduled_delimiter = false;
Packit bfcc33
      append_string(";");
Packit bfcc33
    }
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  // prepend some text or token to the buffer
Packit bfcc33
  void Emitter::prepend_output(const OutputBuffer& output)
Packit bfcc33
  {
Packit bfcc33
    wbuf.smap.prepend(output);
Packit bfcc33
    wbuf.buffer = output.buffer + wbuf.buffer;
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  // prepend some text or token to the buffer
Packit bfcc33
  void Emitter::prepend_string(const std::string& text)
Packit bfcc33
  {
Packit bfcc33
    wbuf.smap.prepend(Offset(text));
Packit bfcc33
    wbuf.buffer = text + wbuf.buffer;
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  // append some text or token to the buffer
Packit bfcc33
  void Emitter::append_string(const std::string& text)
Packit bfcc33
  {
Packit bfcc33
Packit bfcc33
    // write space/lf
Packit bfcc33
    flush_schedules();
Packit bfcc33
Packit bfcc33
    if (in_comment && output_style() == COMPACT) {
Packit bfcc33
      // unescape comment nodes
Packit bfcc33
      std::string out = comment_to_string(text);
Packit bfcc33
      // add to buffer
Packit bfcc33
      wbuf.buffer += out;
Packit bfcc33
      // account for data in source-maps
Packit bfcc33
      wbuf.smap.append(Offset(out));
Packit bfcc33
    } else {
Packit bfcc33
      // add to buffer
Packit bfcc33
      wbuf.buffer += text;
Packit bfcc33
      // account for data in source-maps
Packit bfcc33
      wbuf.smap.append(Offset(text));
Packit bfcc33
    }
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  // append some white-space only text
Packit bfcc33
  void Emitter::append_wspace(const std::string& text)
Packit bfcc33
  {
Packit bfcc33
    if (text.empty()) return;
Packit bfcc33
    if (peek_linefeed(text.c_str())) {
Packit bfcc33
      scheduled_space = 0;
Packit bfcc33
      append_mandatory_linefeed();
Packit bfcc33
    }
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  // append some text or token to the buffer
Packit bfcc33
  // this adds source-mappings for node start and end
Packit bfcc33
  void Emitter::append_token(const std::string& text, const AST_Node_Ptr node)
Packit bfcc33
  {
Packit bfcc33
    flush_schedules();
Packit bfcc33
    add_open_mapping(node);
Packit bfcc33
    // hotfix for browser issues
Packit bfcc33
    // this is pretty ugly indeed
Packit bfcc33
    if (scheduled_mapping) {
Packit bfcc33
      add_open_mapping(scheduled_mapping);
Packit bfcc33
      scheduled_mapping = 0;
Packit bfcc33
    }
Packit bfcc33
    append_string(text);
Packit bfcc33
    add_close_mapping(node);
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  // HELPER METHODS
Packit bfcc33
Packit bfcc33
  void Emitter::append_indentation()
Packit bfcc33
  {
Packit bfcc33
    if (output_style() == COMPRESSED) return;
Packit bfcc33
    if (output_style() == COMPACT) return;
Packit bfcc33
    if (in_declaration && in_comma_array) return;
Packit bfcc33
    if (scheduled_linefeed && indentation)
Packit bfcc33
      scheduled_linefeed = 1;
Packit bfcc33
    std::string indent = "";
Packit bfcc33
    for (size_t i = 0; i < indentation; i++)
Packit bfcc33
      indent += opt.indent;
Packit bfcc33
    append_string(indent);
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  void Emitter::append_delimiter()
Packit bfcc33
  {
Packit bfcc33
    scheduled_delimiter = true;
Packit bfcc33
    if (output_style() == COMPACT) {
Packit bfcc33
      if (indentation == 0) {
Packit bfcc33
        append_mandatory_linefeed();
Packit bfcc33
      } else {
Packit bfcc33
        append_mandatory_space();
Packit bfcc33
      }
Packit bfcc33
    } else if (output_style() != COMPRESSED) {
Packit bfcc33
      append_optional_linefeed();
Packit bfcc33
    }
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  void Emitter::append_comma_separator()
Packit bfcc33
  {
Packit bfcc33
    // scheduled_space = 0;
Packit bfcc33
    append_string(",");
Packit bfcc33
    append_optional_space();
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  void Emitter::append_colon_separator()
Packit bfcc33
  {
Packit bfcc33
    scheduled_space = 0;
Packit bfcc33
    append_string(":");
Packit bfcc33
    append_optional_space();
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  void Emitter::append_mandatory_space()
Packit bfcc33
  {
Packit bfcc33
    scheduled_space = 1;
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  void Emitter::append_optional_space()
Packit bfcc33
  {
Packit bfcc33
    if ((output_style() != COMPRESSED) && buffer().size()) {
Packit bfcc33
      unsigned char lst = buffer().at(buffer().length() - 1);
Packit bfcc33
      if (!isspace(lst) || scheduled_delimiter) {
Packit bfcc33
        append_mandatory_space();
Packit bfcc33
      }
Packit bfcc33
    }
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  void Emitter::append_special_linefeed()
Packit bfcc33
  {
Packit bfcc33
    if (output_style() == COMPACT) {
Packit bfcc33
      append_mandatory_linefeed();
Packit bfcc33
      for (size_t p = 0; p < indentation; p++)
Packit bfcc33
        append_string(opt.indent);
Packit bfcc33
    }
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  void Emitter::append_optional_linefeed()
Packit bfcc33
  {
Packit bfcc33
    if (in_declaration && in_comma_array) return;
Packit bfcc33
    if (output_style() == COMPACT) {
Packit bfcc33
      append_mandatory_space();
Packit bfcc33
    } else {
Packit bfcc33
      append_mandatory_linefeed();
Packit bfcc33
    }
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  void Emitter::append_mandatory_linefeed()
Packit bfcc33
  {
Packit bfcc33
    if (output_style() != COMPRESSED) {
Packit bfcc33
      scheduled_linefeed = 1;
Packit bfcc33
      scheduled_space = 0;
Packit bfcc33
      // flush_schedules();
Packit bfcc33
    }
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
  void Emitter::append_scope_opener(AST_Node_Ptr node)
Packit bfcc33
  {
Packit bfcc33
    scheduled_linefeed = 0;
Packit bfcc33
    append_optional_space();
Packit bfcc33
    flush_schedules();
Packit bfcc33
    if (node) add_open_mapping(node);
Packit bfcc33
    append_string("{");
Packit bfcc33
    append_optional_linefeed();
Packit bfcc33
    // append_optional_space();
Packit bfcc33
    ++ indentation;
Packit bfcc33
  }
Packit bfcc33
  void Emitter::append_scope_closer(AST_Node_Ptr node)
Packit bfcc33
  {
Packit bfcc33
    -- indentation;
Packit bfcc33
    scheduled_linefeed = 0;
Packit bfcc33
    if (output_style() == COMPRESSED)
Packit bfcc33
      scheduled_delimiter = false;
Packit bfcc33
    if (output_style() == EXPANDED) {
Packit bfcc33
      append_optional_linefeed();
Packit bfcc33
      append_indentation();
Packit bfcc33
    } else {
Packit bfcc33
      append_optional_space();
Packit bfcc33
    }
Packit bfcc33
    append_string("}");
Packit bfcc33
    if (node) add_close_mapping(node);
Packit bfcc33
    append_optional_linefeed();
Packit bfcc33
    if (indentation != 0) return;
Packit bfcc33
    if (output_style() != COMPRESSED)
Packit bfcc33
      scheduled_linefeed = 2;
Packit bfcc33
  }
Packit bfcc33
Packit bfcc33
}