All Downloads are FREE. Search and download functionalities are using the official Maven repository.

guru.nidi.text.transform.parse.AbstractCharReadingParser.scala Maven / Gradle / Ivy

/**
 * Copyright (C) 2013 Stefan Niederhauser ([email protected])
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *         http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package guru.nidi.text.transform.parse

import guru.nidi.text.transform.Name._
import guru.nidi.text.transform.parse.AbstractCharReadingParser.EOI
import guru.nidi.text.transform.{Segment, TransformContext}

import scala.collection.mutable.ListBuffer

/**
 *
 */
object AbstractCharReadingParser {
  val EOI: Char = -1.asInstanceOf[Char]
}

abstract class AbstractCharReadingParser(context: TransformContext) extends AbstractParser(context) {

  private var c: Char = 0
  private var pos: Int = 0
  private var savedC: Char = 0
  private var savedPos: Int = 0

  val result = ListBuffer[Segment]()

  override def parseImpl(): Segment = {
    pos = 0
    nextChar()
    reset()
    val root = ROOT()
    doParse()
    root(result: _*)
  }

  def doParse()

  protected def reset() {
    result.clear()
  }

  def addToResult(segment: Segment) {
    result += segment
  }

  protected def addToResult(pos: Int, segment: Segment) {
    result.insert(pos, segment)
  }

  def isCurrentCharOneOf(s: String): Boolean = s.indexOf(c) >= 0

  def nextChar(): Char = {
    c = fetchNextChar()
    c
  }

  private def fetchNextChar(): Char = {
    if (pos >= input.length) {
      EOI
    } else {
      if (c == '\n') {
        var d: Char = 0
        do {
          d = input(pos)
          pos += 1
        } while ((d == ' ' || d == '\t') && pos < input.length)
        if (d == ' ' || d == '\t') EOI else d
      } else {
        pos += 1
        input(pos - 1)
      }
    }
  }

  def currentChar = c

  def pushBack(n: Int) {
    //TODO take care of "space after newline" logic, see #nextChar
    pos -= n
    if (pos < input.length) {
      c = input(pos-1)
    }
  }

  def savePos() {
    savedPos = pos
    savedC = c
  }

  def restorePos() = {
    c = savedC
    pos = savedPos
    nextChar()
    savedC
  }

  //
  //  public String lookAhead(int n) {
  //    savePos();
  //    StringBuffer s = new StringBuffer();
  //    for (int i = 0; i < n; i++) {
  //      s.append(nextChar());
  //    }
  //    restorePos();
  //    return s.toString();
  //  }
  //
  //  public boolean isCurrentCharOneOf(String s) {
  //    return s.indexOf(c) >= 0;
  //  }
  //
  //
  //
  /**
   * Read the input until one of the given Strings occurs.
   *
   * @param ends the Strings to be searched
   * @return the input until but not including the first found String
   */
  def readUntil(ends: String*): String = {
    val res = new StringBuilder
    val endpos = new Array[Int](ends.length)
    var foundIndex = -1
    do {
      res.append(c)
      for (i <- 0 until ends.length) {
        if (c == ends(i).charAt(endpos(i))) {
          endpos(i) += 1
          if (endpos(i) == ends(i).length) {
            foundIndex = i
          }
        } else {
          endpos(i) = 0
        }
      }
      nextChar()
    } while (c != EOI && foundIndex == -1)
    if (foundIndex != -1) {
      res.delete(res.length - ends(foundIndex).length, res.length)
    }
    res.toString()
  }

  def readUntilChar(chars: String): String = {
    val res = new StringBuilder
    do {
      res.append(c)
      nextChar()
    } while (chars.indexOf(c) < 0 && c != EOI)
    res.toString()
  }

  def getCount(sym: Char) = {
    var level = 1
    while (currentChar == sym) {
      nextChar()
      level += 1
    }
    level
  }

  def skipWhitspaces() {
    while (currentChar <= ' ') {
      nextChar()
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy