--- parser3/src/main/pa_table.C 2001/03/19 17:42:16 1.15 +++ parser3/src/main/pa_table.C 2007/04/23 10:30:32 1.62 @@ -1,58 +1,86 @@ /** @file Parser: table class. - Copyright (c) 2001 ArtLebedev Group (http://www.artlebedev.com) - - Author: Alexander Petrosyan (http://design.ru/paf) - - $Id: pa_table.C,v 1.15 2001/03/19 17:42:16 paf Exp $ + Copyright (c) 2001-2005 ArtLebedev Group (http://www.artlebedev.com) + Author: Alexandr Petrosian (http://paf.design.ru) */ -#include +static const char * const IDENT_TABLE_C="$Date: 2007/04/23 10:30:32 $"; #include "pa_table.h" -#include "pa_pool.h" -Table::Table(Pool& apool, - const String& asource, - Array *acolumns, - int initial_rows) : - Array(apool, initial_rows), +#include "pa_exception.h" + +Table::Table(columns_type acolumns, size_t initial_rows): + Array(initial_rows), - fsource(asource), fcurrent(0), fcolumns(acolumns), - name2number(pool(), false) { + name2number(new name2number_hash_class) { - if(fcolumns) - for(int i=0; isize(); i++) { - const String& name=*fcolumns->get_string(i); - name2number.put(name, i+1); + if(fcolumns) { + size_t number=1; + for(Array_iterator i(*fcolumns); i.has_next(); ) { + const String& name=*i.next(); + name2number->put(name, number++); } + } } -const Array &Table::at(int index) { - return *static_cast(get(index)); +Table::Table(const Table& src, Action_options& options) : + Array(options.limit==ARRAY_OPTION_LIMIT_ALL?0:options.limit/*may be more than needed, no harm done*/), + + fcurrent(0), + fcolumns(src.fcolumns), + name2number(src.name2number) { + + append(src, options.offset, options.limit, options.reverse); } -const String *Table::item(const String& column_name) { - int column_index; - if(fcolumns) { // named - int found_index=name2number.get_int(column_name); - if(found_index) - column_index=found_index-1; - else - THROW(0, 0, - &column_name, +int Table::column_name2index(const String& column_name, bool bark) const { + if(fcolumns) {// named + int result=name2number->get(column_name)-1; // -1 = column not found + if(bark && result<0) + throw Exception(PARSER_RUNTIME, + &column_name, "column not found"); - } else { // nameless - column_index=atoi(column_name.cstr()); - if(!valid(fcurrent)) - return 0; // it's OK we don't have row, just return nothing - const Array& row=at(fcurrent); - if(column_index<0 || column_index>=row.size()) // read past proper index? - return 0; // it's OK we don't have column, just return nothing + return result; + } else // nameless + return column_name.as_int(); +} + +const String* Table::item(size_t column) { + if(valid(fcurrent)) { + element_type row=get(fcurrent); + if(columncount()) // proper index? + return row->get(column); } + return 0; // it's OK we don't have row|column, just return nothing +} + +#ifndef DOXYGEN +struct Locate_int_string_info { + int column; + const String* value; +}; +#endif +bool locate_int_string(Table& self, Locate_int_string_info* info) { + const String *item_value=self.item(info->column); + return item_value && *item_value==*info->value; +} + +bool Table::locate(int column, const String& value, + Table::Action_options& options) { + Locate_int_string_info info={column, &value}; + return table_first_that(locate_int_string, &info, options); +} + +bool Table::locate(const String& column, const String& value, + Table::Action_options& options) { + return locate(column_name2index(column, true), value, options); +} - return item(column_index); +void Table::offset(bool absolute, int offset) { + if(size_t lcount=count()) + fcurrent=((absolute?0:fcurrent)+offset+lcount)%lcount; }