#include <bu/sio.h> #include <bu/optparser.h> #include <bu/csvreader.h> #include <bu/file.h> #include <bu/newline.h> #include <bu/buffer.h> #include <bu/util.h> #include <bu/regex.h> #include <ncurses.h> using namespace Bu; class Options : public Bu::OptParser { public: Options( int argc, char *argv[] ) : bHeader( true ) { addOption( bHeader, "no-header", "Don't use the first line as a header row. This behaviour can " "also be toggled while running with 'h'." ); setNonOption( slot( this, &Options::onNonOption ) ); addHelpOption(); setOverride( "no-header", "true" ); parse( argc, argv ); } virtual ~Options() { } int onNonOption( StrArray aParams ) { //sio << aParams << sio.nl; sFileIn = aParams[0]; return 0; } Bu::FString sFileIn; bool bHeader; }; typedef Bu::Array<StrArray> StrGrid; typedef Bu::Array<int> IntArray; class CsvDoc { public: CsvDoc() : iMaxCols( 0 ) { } virtual ~CsvDoc() { } void addRow( StrArray aStr ) { sgData.append( aStr ); if( iMaxCols < aStr.getSize() ) iMaxCols = aStr.getSize(); while( aWidths.getSize() < iMaxCols ) { aWidths.append( 0 ); } for( int j = 0; j < aStr.getSize(); j++ ) { if( aWidths[j] < aStr[j].getSize() ) aWidths[j] = aStr[j].getSize(); } } int iMaxCols; StrGrid sgData; IntArray aWidths; }; class CsvView { public: CsvView( CsvDoc &doc ) : doc( doc ), iXOff( 0 ), iYOff( 0 ), bHeaderRow( false ) { } virtual ~CsvView() { } void render() { erase(); int maxx, maxy; getmaxyx( stdscr, maxy, maxx ); int iRows = min( (int)doc.sgData.getSize(), maxy-((bHeaderRow)?(4):(3)) ); int iCols = min( doc.iMaxCols, (int)maxx-1 ); int iHdrHeight = 1; if( bHeaderRow ) iHdrHeight++; // Draw the headers for( int iRow = 0; iRow < iRows; iRow++ ) { if( iRow+iYOff >= doc.sgData.getSize() ) break; char buf[6]; snprintf( buf, 6, "%5d", iRow+iYOff ); mvaddnstr( iRow+iHdrHeight+1, 0, buf, 5 ); mvaddch( iRow+iHdrHeight+1, 6, ACS_VLINE ); } int iXPos = 6; try { for( int iCol = 0; iCol < iCols; iCol++ ) { if( iXPos >= maxx ) break; int iWidth = min( doc.aWidths[iCol+iXOff], maxx-iXPos-1 ); char buf[6]; snprintf( buf, 6, "%d", iCol+iXOff ); mvaddch( 0, iXPos, ACS_VLINE ); mvaddch( iHdrHeight, iXPos, ACS_PLUS ); mvaddnstr( 0, iXPos+1, buf, 5 ); if( bHeaderRow ) { mvaddnstr( 1, iXPos+1, doc.sgData[0][iCol+iXOff].getStr(), iWidth ); mvaddch( 1, iXPos, ACS_VLINE ); } for( int j = 0; j < iWidth; j++ ) { mvaddch( iHdrHeight, iXPos+j+1, ACS_HLINE ); } iXPos += iWidth+1; } } catch(...) { } for( int j = 0; j < 6; j++ ) { mvaddch( iHdrHeight, j, ACS_HLINE ); } // Draw some data for( int iRow = 0; iRow < iRows; iRow++ ) { try { int iXPos = 6; for( int iCol = 0; iCol < iCols; iCol++ ) { if( iXPos >= maxx ) break; int iWidth = min( doc.aWidths[iCol+iXOff], maxx-iXPos-1 ); mvaddch( iRow+iHdrHeight+1, iXPos, ACS_VLINE ); mvaddnstr( iRow+iHdrHeight+1, iXPos+1, doc.sgData[iRow+iYOff][iCol+iXOff].getStr(), iWidth ); iXPos += iWidth+1; } } catch(...) { } } attron( A_REVERSE ); for( int j = 0; j < maxx; j++ ) { mvaddch( maxy-1, j, ' ' ); } mvaddstr( maxy-1, 1, "q) quit h) toggle header row" ); char buf[30]; int iWidth = sprintf( buf, "[%dx%ld]", doc.iMaxCols, doc.sgData.getSize() ); mvaddstr( maxy-1, maxx-iWidth-1, buf ); attroff( A_REVERSE ); } void move( int iX, int iY ) { iXOff += iX; iYOff += iY; if( iXOff < 0 ) iXOff = 0; if( bHeaderRow ) { if( iYOff < 1 ) iYOff = 1; } else { if( iYOff < 0 ) iYOff = 0; } if( iYOff >= doc.sgData.getSize() ) iYOff = doc.sgData.getSize()-1; if( iXOff >= doc.iMaxCols ) iXOff = doc.iMaxCols-1; } void pageDown() { int maxx, maxy; getmaxyx( stdscr, maxy, maxx ); move( 0, maxy-((bHeaderRow)?(4):(3)) ); } void pageUp() { int maxx, maxy; getmaxyx( stdscr, maxy, maxx ); move( 0, -(maxy-((bHeaderRow)?(4):(3))) ); } void home() { iYOff = 0; if( bHeaderRow ) iYOff++; } void end() { iYOff = doc.sgData.getSize()-1; } void setHeaderRow( bool bOn ) { if( bHeaderRow == bOn ) return; bHeaderRow = bOn; move( 0, ((bOn)?(1):(-1)) ); } void toggleHeaderRow() { setHeaderRow( !bHeaderRow ); } Bu::FString prompt( const Bu::FString &sPrompt ) { int maxx, maxy; Bu::FString sStr; RegEx re( sPrompt ); curs_set( 1 ); for(;;) { getmaxyx( stdscr, maxy, maxx ); for( int j = 0; j < maxx; j++ ) { mvaddch( maxy-1, j, ' ' ); } mvaddstr( maxy-1, 0, sPrompt.getStr() ); mvaddstr( maxy-1, sPrompt.getSize(), sStr.getStr() ); int iCh = getch(); switch( iCh ) { case '\n': case '\r': case KEY_ENTER: curs_set( 0 ); return sStr; break; case KEY_BACKSPACE: if( sStr.getSize() > 0 ) sStr.resize( sStr.getSize()-1 ); break; default: if( iCh < 127 ) sStr += (char)iCh; break; } } } void resetCaret() { sysCaret.reset(); } void findNext( const Bu::FString &sTerm ) { RegEx re( sTerm ); int y = sysCaret.iRow; if( y < 0 ) y = 0; int x = sysCaret.iCol+1; for( ; y < doc.sgData.getSize(); y++ ) { StrArray &aRow = doc.sgData[y]; for( ; x < aRow.getSize(); x++ ) { if( re.execute( aRow[x] ) ) //aRow[x].find( sTerm ) ) { sysCaret.iRow = y; sysCaret.iCol = x; scrollToCaret(); return; } } x = 0; } } void scrollToCaret() { iXOff = sysCaret.iCol; iYOff = sysCaret.iRow; } CsvDoc &doc; int iXOff; int iYOff; bool bHeaderRow; class Caret { public: Caret() : iRow( -1 ), iCol( -1 ) { } virtual ~Caret() { } void reset() { iRow = iCol = -1; } bool isSet() { if( iRow < 0 || iCol < 0 ) return false; return true; } int iRow; int iCol; }; Caret sysCaret; }; int main( int argc, char *argv[] ) { Options opt( argc, argv ); if( !opt.sFileIn.isSet() ) { sio << "No file specified." << sio.nl; return 1; } CsvDoc doc; { File fIn( opt.sFileIn, File::Read ); NewLine nlIn( fIn ); Buffer bIn( nlIn ); CsvReader cr( bIn ); while( !fIn.isEos() ) { StrArray sa = cr.readLine(); if( fIn.isEos() ) break; doc.addRow( sa ); } } initscr(); cbreak(); noecho(); nonl(); intrflush( stdscr, FALSE ); keypad( stdscr, TRUE ); curs_set( 0 ); CsvView view( doc ); view.setHeaderRow( opt.bHeader ); Bu::FString sSearchTerm; bool bRun = true; do { view.render(); int ch = getch(); switch( ch ) { case 'q': bRun = false; break; case KEY_DOWN: view.move( 0, 1 ); break; case KEY_UP: view.move( 0, -1 ); break; case KEY_LEFT: view.move( -1, 0 ); break; case KEY_RIGHT: view.move( 1, 0 ); break; case KEY_NPAGE: view.pageDown(); break; case KEY_PPAGE: view.pageUp(); break; case KEY_HOME: view.home(); break; case KEY_END: view.end(); break; case '/': sSearchTerm = view.prompt("find: "); view.resetCaret(); view.findNext( sSearchTerm ); break; case 'n': view.findNext( sSearchTerm ); break; case 'h': view.toggleHeaderRow(); break; } } while( bRun ); endwin(); return 0; }