#include "common.h"
#include "scanner.h"
#include "stdlib.h"
#include "string.h"

void initScanner(Scanner* scanner, const char* source) {
    scanner->start = source;
    scanner->current = source;
    scanner->line = 1;
}

static int isAtEnd(Scanner* scanner) {
    return *scanner->current == '\0';
}


static Token makeToken(Scanner* scanner, TokenType type) {
    Token token;
    token.type = type;
    token.start = scanner->start;
    token.length = scanner->current - scanner->start;
    token.line = scanner->line;
    return token;
}

static Token errorToken(const char* message) {
    Token token;
    token.type = ERROR_TOKEN;
    token.start = message;
    token.length = strlen(message);
    token.line = 1;
    return token;
}


static char advance(Scanner* scanner) {
    scanner->current++;
    return scanner->current[-1];
}

static char peek(Scanner* scanner) {
    return *scanner->current;
}

static char peekNext(Scanner* scanner) {
    if (isAtEnd(scanner)) return '\0';
    return scanner->current[1];
}

static int match(Scanner* scanner, char expected) {
    if (isAtEnd(scanner)) return 0;
    if (*scanner->current != expected) return 0;
    scanner->current++;
    return 1;
}

static void skipWhitespace(Scanner* scanner) {
    while (1){
        char c = peek(scanner);
        if (c == ' ' || c == '\r' || c == '\t') {
            advance(scanner);
        } else if (c == '\n') {
            scanner->line++;
            advance(scanner);
        } else if (c == ';') { // Skip comments
            while (peek(scanner) != '\n' && !isAtEnd(scanner)) {
                    advance(scanner);
            }
        } else {
            break;
        }
    }
}

static Token string(Scanner* scanner) {
    while (peek(scanner) != '"' && !isAtEnd(scanner)) {
        if (peek(scanner) == '\n') scanner->line++;
        advance(scanner);
    }

    if (isAtEnd(scanner)) return errorToken("Unterminated string.");

    // The closing "
    advance(scanner);

    return makeToken(scanner, STRING);
}

static int isDigit(char c) {
    return c >= '0' && c <= '9';
}
static Token number(Scanner* scanner) {
    while (isDigit(peek(scanner))) advance(scanner);
    if (peek(scanner) == '.' && isDigit(peekNext(scanner))) {
        advance(scanner);
        while (isDigit(peek(scanner))) advance(scanner);
    }
    return makeToken(scanner, NUMBER);
}

static int isAlpha(char c) {
    return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || c == '_';
}



static TokenType checkKeyword(Scanner* scanner, int start, int length,
const char* rest, TokenType type) {
    if (scanner->current - scanner->start == start + length && 
        memcmp(scanner->start + start, rest, length) == 0) {
        return type;
    }
    return IDENTIFIER;
}

static TokenType identifierType(Scanner* scanner) {
    switch (scanner->start[0]) {
        case 'a': return checkKeyword(scanner, 1, 2, "nd", AND);
        case 'e': return checkKeyword(scanner, 1, 3, "lse", ELSE);
        case 'f': {
            // can be either fn, false, or, object
            switch (scanner->start[1]) {
                case 'n': return checkKeyword(scanner, 2, 0, "", FUN);
                case 'u': return checkKeyword(scanner, 2, 1, "n", FALSE);
                case 'a': return checkKeyword(scanner, 2, 3, "lse", FALSE);
            }
        }
        case 'i': {
            switch (scanner->start[1]) {
                case 'f': return checkKeyword(scanner, 2, 0, "", IF);
                case 'n': return checkKeyword(scanner, 2, 6, "itnext", INITNEXT);
            }
        }
        case 'l': return checkKeyword(scanner, 1, 2, "et", LET);
        case 'n': return checkKeyword(scanner, 1, 2, "il", NIL);
        case 'o': {
            // can be either on, or, object
            switch (scanner->start[1]) {
                case 'n': return checkKeyword(scanner, 2, 0, "", ON);
                case 'r': return checkKeyword(scanner, 2, 0, "", OR);
                case 'b': return checkKeyword(scanner, 2, 4, "ject", OBJECT);
            }
        }
        case 'p': return checkKeyword(scanner, 1, 3, "rev", PREV);
        case 't': {
            switch (scanner->start[1]) {
                case 'h': return checkKeyword(scanner, 2, 2, "en", THEN);
                case 'r': return checkKeyword(scanner, 2, 2, "ue", TRUE);
            }
        }
    }
    return IDENTIFIER;
}

static Token identifier(Scanner* scanner) {
    while (isAlpha(peek(scanner)) || isDigit(peek(scanner))) advance(scanner);
    return makeToken(scanner, identifierType(scanner));
}
Token scanToken(Scanner* scanner) {
    skipWhitespace(scanner);
    scanner->start = scanner->current;
    if (isAtEnd(scanner)) return makeToken(scanner, EOF_TOKEN);

    char c = advance(scanner);
    if (isAlpha(c)) return identifier(scanner);
    if (isDigit(c)) return number(scanner);
    switch (c) {
        case '(': return makeToken(scanner, LEFT_PAREN);
        case ')': return makeToken(scanner, RIGHT_PAREN);
        case ',': return makeToken(scanner, COMMA);
        case '.':
            return makeToken(scanner, peek(scanner) == '.' ? DOTDOT: DOT);
        case '-':
            if (match(scanner, '-')){
                if (match(scanner, '>')){
                    return makeToken(scanner, MAPTO);
                } else{
                    return errorToken("Expected '>' after '-'");
                }
            }
            return makeToken(scanner, MINUS);
        case '+':
            return makeToken(scanner, PLUS);
        case '/':
            return makeToken(scanner, SLASH);
        case '*':
            return makeToken(scanner, STAR);
        case '&':
            return makeToken(scanner, AND);
        case '|':
            return makeToken(scanner, OR);
        case ':':
            return makeToken(scanner, COLON);
        case '%':
            return makeToken(scanner, MODULO);
        case '!':
            return match(scanner, '=') ? makeToken(scanner, BANG_EQUAL) : makeToken(scanner, BANG);
        case '=':
            return match(scanner, '=') ? makeToken(scanner, EQUAL_EQUAL) : makeToken(scanner, EQUAL);
        case '<':
            return match(scanner, '=') ? makeToken(scanner, LESS_EQUAL) : makeToken(scanner, LESS);
        case '>':
            return match(scanner, '=') ? makeToken(scanner, GREATER_EQUAL) : makeToken(scanner, GREATER);
        case '"': return string(scanner);
    }
}
