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

ammonite.interp.script.PositionOffsetConversion.scala Maven / Gradle / Ivy

The newest version!
package ammonite.interp.script

import scala.collection.mutable

object PositionOffsetConversion {

  private def lineStartIndices(content: String): Array[Int] = {

    val content0 = content.toCharArray

    // adapted from scala/scala SourceFile.scala

    val length = content0.length
    val CR = '\r'
    val LF = '\n'

    def charAtIsEOL(idx: Int)(p: Char => Boolean) = {
      // don't identify the CR in CR LF as a line break, since LF will do.
      def notCRLF0 =
        content0(idx) != CR ||
          !content0.isDefinedAt(idx + 1) ||
          content0(idx + 1) != LF

      idx < length && notCRLF0 && p(content0(idx))
    }

    def isAtEndOfLine(idx: Int) = charAtIsEOL(idx) {
      case CR | LF => true
      case _       => false
    }

    val buf = new mutable.ArrayBuffer[Int]
    buf += 0
    for (i <- 0 until content0.length if isAtEndOfLine(i))
      buf += i + 1
    buf.toArray
  }

  /** Converts a character offset from beginning of `content` to a `Position` */
  def offsetToPos(content: String): Int => Position = {

    val lineStartIndices0 = lineStartIndices(content)

    def offsetToLine(offset: Int): Int = {

      assert(lineStartIndices0.nonEmpty)

      if (offset >= lineStartIndices0.last) lineStartIndices0.length - 1
      else {
        def find(a: Int, b: Int): Int =
          if (a + 1 >= b) a
          else {
            val c = (a + b) / 2
            val idx = lineStartIndices0(c)
            if (idx == offset) c
            else if (idx < offset) find(c, b)
            else find(a, c)
          }
        find(0, lineStartIndices0.length - 1)
      }
    }

    offset =>
      assert(offset >= 0)
      assert(offset <= content.length)
      val line = offsetToLine(offset)
      Position(line, offset - lineStartIndices0(line))
  }

  private def indicesOf(input: String, elem: String): Array[Int] = {
    val b = new mutable.ArrayBuffer[Int]
    var idx = 0
    while (idx >= 0 && idx < input.length) {
      val nextIdx = input.indexOf(elem, idx)
      if (nextIdx >= 0) {
        b += nextIdx
        idx = nextIdx + 1
      } else
        idx = -1
    }
    b.toArray
  }

  /** Returns the start and end indices of sections delimited by `start` and `end` in `input` */
  def sections(input: String, start: String, end: String): Array[(Int, Int)] = {
    val b = new mutable.ArrayBuffer[(Int, Int)]
    var idx = 0
    while (idx < input.length) {
      val startIdx = input.indexOf(start, idx)
      if (startIdx >= 0) {
        val endIdx = input.indexOf(end, startIdx + start.length)
        if (endIdx >= 0) {
          idx = endIdx + end.length
          b.append((startIdx, idx))
        } else
          idx = input.length
      } else
        idx = input.length
    }
    b.toArray
  }

  /** Removes sections delimited by `ignoreSections` from offset `idx` */
  def extraOffset(ignoreSections: Seq[(Int, Int)], idx: Int): Int =
    ignoreSections
      .iterator
      .takeWhile(_._2 <= idx)
      .map {
        case (start, end) =>
          end - start
      }
      .sum

  private val firstLineWrapperPrefix = "/*