/* 
 * File:   TextReader.cpp
 * Author: lennyn
 * 
 * Created on May 28, 2014, 11:43 AM
 */

#include "TextReader.hpp"
#include "charset_utils.hpp"

using namespace std;

namespace morfeusz {

TextReader::TextReader(
    const char* inputStart, 
    const char* inputEnd, 
    const Environment& env)
: codepointsNum(0),
    wordStartPtr(inputStart),
    currPtr(inputStart),
        inputEnd(inputEnd), 
        env(env),
        knowsAboutWhitespace(false),
        atWhitespace(false),
        peekIsRead(false),
        thePeek(0x00),
        theNormalizedPeek(0x00),
        ptrAfterThePeek(NULL) {}

void TextReader::markWordStartsHere() {
    codepointsNum = 0;
    wordStartPtr = currPtr;
}

const char* TextReader::getWordStartPtr() const {
    return wordStartPtr;
}

const char* TextReader::getCurrPtr() const {
    return currPtr;
}

const char* TextReader::getNextPtr() {
    if (!peekIsRead) {
        peek();
    }
    return ptrAfterThePeek;
}

const char* TextReader::getEndPtr() const {
    return inputEnd;
}

int TextReader::getCodepointsRead() const {
    return codepointsNum;
}

bool TextReader::isAtEnd() const {
    return currPtr == inputEnd;
}

bool TextReader::isAtWhitespace() {
    if (isAtEnd()) {
        return true;
    }
    else {
        if (!peekIsRead) {
            peek();
        }
        return atWhitespace;
    }
}

bool TextReader::isInsideAWord() {
    return !isAtEnd() && !isAtWhitespace();
}

uint32_t TextReader::peek() {
    if (peekIsRead) {
        return thePeek;
    }
    else {
        ptrAfterThePeek = currPtr;
        thePeek = env.getCharsetConverter().next(ptrAfterThePeek, inputEnd);
        theNormalizedPeek = env.getProcessorType() == ANALYZER
                ? env.getCaseConverter().toLower(thePeek)
                : thePeek;
        atWhitespace = isWhitespace(thePeek);
        peekIsRead = true;
        return thePeek;
    }
}

uint32_t TextReader::normalizedPeek() {
    if (!peekIsRead) {
        peek();
    }
    return theNormalizedPeek;
}

uint32_t TextReader::next() {
    if (!peekIsRead) {
        peek();
    }
    currPtr = ptrAfterThePeek;
    peekIsRead = false;
    knowsAboutWhitespace = false;
    codepointsNum++;
    return thePeek;
}

void TextReader::skipWhitespaces() {
    while (!isAtEnd() && isAtWhitespace()) {
        next();
    }
}

string TextReader::readWhitespacesChunk() {
    const char* startPtr = currPtr;
    while (!isAtEnd() && isAtWhitespace()) {
        next();
    }
    string res(startPtr, currPtr);
    return res;
}

void TextReader::proceedToEnd() {
    while (!isAtEnd()) {
        next();
    }
}

TextReader::~TextReader() {
    
}

}