#include "HeadSpin.h"
#include "pfTextFile.h"
#include <stdio.h>
#include <ctype.h>

//-----------------------------------------------------------------------------

namespace dev
{


class TextFile::TextFileImpl
{
public:
    TextFile::ErrorType     err;
    int                     line;

    explicit TextFileImpl( const char* filename )
    {
        err             = TextFile::ERROR_NONE;
        line            = 1;
        m_peeked        = false;
        m_peekedChar    = 0;
        m_file          = fopen( filename, "rt" );

        if ( !m_file )
            err = TextFile::ERROR_OPEN;
    }

    ~TextFileImpl()
    {
        if ( m_file )
        {
            fclose( m_file );
            m_file = 0;
        }
    }

    bool eof() const
    {
        if ( err )
            return true;
        return 0 != feof(m_file);
    }

    bool peekChar( char* ch )
    {
        if ( err )
            return false;

        if ( !m_peeked )
        {
            int c = getc( m_file );
            if ( EOF != c )
            {
                m_peeked = true;
                m_peekedChar = (char)c;
            }
            else
            {
                if ( ferror(m_file) )
                    err = TextFile::ERROR_READ;
            }
        }

        if ( m_peeked )
            *ch = m_peekedChar;
        return m_peeked;
    }

    bool readChar( char* ch )
    {
        if ( err )
            return false;

        bool more = peekChar( ch );
        m_peeked = false;
        if ( more && *ch == '\n' )
            ++line;
        return more;
    }

    bool skipWhitespace()
    {
        if ( err )
            return false;

        char ch;
        while ( peekChar(&ch) )
        {
            if ( !isspace(ch) )
                break;
            readChar( &ch );
        }
        return !eof();
    }

    bool readString( char* buf, int size )
    {
        if ( err )
            return false;

        skipWhitespace();

        int count = 0;
        char ch;
        while ( peekChar(&ch) )
        {
            if ( isspace(ch) )
                break;
            if ( count+1 < size )
                buf[count++] = ch;
            readChar( &ch );
        }
        if ( size > 0 )
            buf[count] = 0;
        return count > 0;
    }

    void skipLine()
    {
        if ( err )
            return;

        char ch;
        while ( readChar(&ch) )
        {
            if ( ch == '\n' )
                break;
        }
    }

    long readHex()
    {
        if ( err )
            return 0;

        skipWhitespace();

        // hex must start with alphanumeric character
        char ch;
        if ( !peekChar(&ch) || !isalnum(ch) )
        {
            err = TextFile::ERROR_PARSE;
            return 0;
        }

        long x = 0;
        while ( peekChar(&ch) )
        {
            switch ( ch )
            {
            case '0':   x <<= 4; x += 0; break;
            case '1':   x <<= 4; x += 1; break;
            case '2':   x <<= 4; x += 2; break;
            case '3':   x <<= 4; x += 3; break;
            case '4':   x <<= 4; x += 4; break;
            case '5':   x <<= 4; x += 5; break;
            case '6':   x <<= 4; x += 6; break;
            case '7':   x <<= 4; x += 7; break;
            case '8':   x <<= 4; x += 8; break;
            case '9':   x <<= 4; x += 9; break;
            case 'a':
            case 'A':   x <<= 4; x += 0xA; break;
            case 'b':
            case 'B':   x <<= 4; x += 0xB; break;
            case 'c':
            case 'C':   x <<= 4; x += 0xC; break;
            case 'd':
            case 'D':   x <<= 4; x += 0xD; break;
            case 'e':
            case 'E':   x <<= 4; x += 0xE; break;
            case 'f':
            case 'F':   x <<= 4; x += 0xF; break;
            default:    return x;
            }
            readChar( &ch );
        }
        return x;
    }

private:
    bool    m_peeked;
    char    m_peekedChar;
    FILE*   m_file;

    TextFileImpl( const TextFileImpl& );
    TextFileImpl& operator=( const TextFileImpl& );
};

//-----------------------------------------------------------------------------

TextFile::TextFile( const char* filename )
{
    m_this = TRACKED_NEW TextFileImpl( filename );
}

TextFile::~TextFile()
{
    delete m_this;
}

bool TextFile::readString( char* buf, int size )
{
    return m_this->readString( buf, size );
}

void TextFile::skipLine()
{
    m_this->skipLine();
}

long TextFile::readHex()
{
    return m_this->readHex();
}

bool TextFile::skipWhitespace()
{
    return m_this->skipWhitespace();
}

TextFile::ErrorType TextFile::error() const
{
    return m_this->err;
}

bool TextFile::readChar( char* ch )
{
    return m_this->readChar( ch );
}

bool TextFile::peekChar( char* ch )
{
    return m_this->peekChar( ch );
}

bool TextFile::eof() const
{
    return m_this->eof();
}

int TextFile::line() const
{
    return m_this->line;
}


} // dev
/*
 * Copyright (c) 2001 Jani Kajala
 *
 * Permission to use, copy, modify, distribute and sell this
 * software and its documentation for any purpose is hereby
 * granted without fee, provided that the above copyright notice
 * appear in all copies and that both that copyright notice and
 * this permission notice appear in supporting documentation.
 * Jani Kajala makes no representations about the suitability 
 * of this software for any purpose. It is provided "as is" 
 * without express or implied warranty.
 */