Annotation of parser3/src/main/pa_table.C, revision 1.54

1.14      paf         1: /** @file
1.15      paf         2:        Parser: table class.
                      3: 
1.53      paf         4:        Copyright (c) 2001, 2003 ArtLebedev Group (http://www.artlebedev.com)
1.44      paf         5:        Author: Alexandr Petrosian <paf@design.ru> (http://paf.design.ru)
1.48      paf         6: */
1.15      paf         7: 
1.54    ! paf         8: static const char* IDENT_TABLE_C="$Date: 2003/01/21 15:51:15 $";
1.1       paf         9: 
1.48      paf        10: //#include <stdlib.h>
1.2       paf        11: 
1.1       paf        12: #include "pa_table.h"
                     13: #include "pa_pool.h"
1.25      paf        14: #include "pa_exception.h"
1.1       paf        15: 
1.12      paf        16: Table::Table(Pool& apool, 
1.18      paf        17:                         const String *aorigin_string,
1.35      parser     18:                         const Array *acolumns, 
1.1       paf        19:                         int initial_rows) :
1.12      paf        20:        Array(apool, initial_rows),
1.13      paf        21: 
1.18      paf        22:        forigin_string(aorigin_string),
1.5       paf        23:        fcurrent(0),
                     24:        fcolumns(acolumns), 
1.47      paf        25:        name2number(*NEW Hash(pool())) {
1.1       paf        26: 
1.5       paf        27:        if(fcolumns)
                     28:                for(int i=0; i<fcolumns->size(); i++) {
1.12      paf        29:                        const String& name=*fcolumns->get_string(i);
1.2       paf        30:                        name2number.put(name, i+1);
1.1       paf        31:                }
1.42      paf        32: }
                     33: 
1.54    ! paf        34: Table::Table(Pool& apool, const Table& source, Action_options& options) :
        !            35:        Array(apool, options.limit/*may be more than needed, no harm done*/),
1.42      paf        36: 
                     37:        forigin_string(source.forigin_string),
1.51      paf        38:        fcurrent(0),
1.42      paf        39:        fcolumns(source.fcolumns),
                     40:        name2number(source.name2number) {
                     41: 
1.54    ! paf        42:        append_array(source, options.offset, options.limit, options.reverse);
1.2       paf        43: }
                     44: 
1.29      paf        45: int Table::column_name2index(const String& column_name, bool bark) const {
                     46:        if(fcolumns) {// named
                     47:                int result=name2number.get_int(column_name)-1; // -1 = column not found
                     48:                if(bark && result<0)
1.45      paf        49:                        throw Exception("parser.runtime",
1.29      paf        50:                                &column_name,
                     51:                                "column not found");
                     52:                return result;
                     53:        } else { // nameless
1.36      parser     54:                char *error_pos;
                     55:                const char *cstr=column_name.cstr();
                     56:                int result=(int)strtol(cstr, &error_pos, 0);
1.37      parser     57:                if(*error_pos/*not EOS*/) {
                     58:                        result=-1;
1.30      paf        59:                        if(bark)
1.45      paf        60:                                throw Exception("parser.runtime",
1.30      paf        61:                                        &column_name,
                     62:                                        "invalid column number");
                     63:                }
1.24      paf        64:                return result;
                     65:        }
1.21      paf        66: }
                     67: 
1.22      paf        68: const String *Table::item(int column) const {
1.21      paf        69:        if(valid(fcurrent)) {
1.7       paf        70:                const Array& row=at(fcurrent);
1.22      paf        71:                if(column>=0 && column<row.size()) // proper index?
                     72:                        return row.get_string(column);
1.21      paf        73:        }
                     74:        return 0; // it's OK we don't have row|column, just return nothing
                     75: }
                     76: 
1.54    ! paf        77: void Table::set_current(int acurrent) { 
        !            78:        if(acurrent<0 || acurrent>size())
        !            79:                throw Exception(0,
        !            80:                        0,
        !            81:                        "table row (%d) out of range [0..%d]", acurrent, size()-1);
        !            82:        fcurrent=acurrent; 
        !            83: }
        !            84: bool Table::locate(Table::locate_func func, void *info, Table::Action_options& o) {
        !            85:        int size=this->size();
        !            86:        if(!size || !o.limit)
        !            87:                return false;
        !            88:        if(o.limit<0)
        !            89:                o.limit=size;
        !            90:        int row=o.offset;
        !            91: 
        !            92:        int saved_current=current();
        !            93:        if(o.reverse) { // reverse
        !            94:                int to=max(0, row-o.limit);
        !            95:                for(; row>=to; --row) {
        !            96:                        set_current(row);
        !            97: 
        !            98:                        if(func(*this, info))
        !            99:                                return true;
        !           100:                }
        !           101:        } else { // forward
        !           102:                int to=min(row+o.limit, size);
        !           103:                for(; row<to; row++) {
        !           104:                        set_current(row);
        !           105: 
        !           106:                        if(func(*this, info))
        !           107:                                return true;
        !           108:                }
1.2       paf       109:        }
1.54    ! paf       110:        set_current(saved_current);
1.2       paf       111: 
1.21      paf       112:        return false;
1.28      paf       113: }
                    114: 
1.54    ! paf       115: #ifndef DOXYGEN
        !           116: struct Locate_int_string_info {
        !           117:        int column;
        !           118:        const String* value;
        !           119: };
        !           120: #endif
        !           121: bool locate_int_string(Table& self, void* ainfo) {
        !           122:        Locate_int_string_info& info=*static_cast<Locate_int_string_info*>(ainfo);
        !           123: 
        !           124:        const String *item_value=self.item(info.column);
        !           125:        return item_value && *item_value==*info.value;
        !           126: }
        !           127: 
        !           128: bool Table::locate(int column, const String& value,
        !           129:                   Table::Action_options& options) {
        !           130:        Locate_int_string_info info={column, &value};
        !           131:        return locate(locate_int_string, &info, options);
        !           132: }
        !           133: 
        !           134: bool Table::locate(const String& column, const String& value, 
        !           135:                   Table::Action_options& options) {
        !           136:        return locate(column_name2index(column, true), value, options);
1.22      paf       137: }
                    138: 
1.41      paf       139: void Table::offset(bool absolute, int offset) {
1.22      paf       140:        if(size())
1.41      paf       141:                fcurrent=((absolute?0:fcurrent)+offset+size())%size();
1.1       paf       142: }

E-mail: