Skip to content

Commit edf9cca

Browse files
committed
quote & escape handling; fixes; atom list; select & search atom
1 parent 3503dc5 commit edf9cca

8 files changed

+1152
-41
lines changed

LICENSE.GPL2

+342
Large diffs are not rendered by default.

LICENSE.GPL3

+676
Large diffs are not rendered by default.

LispLexer.cpp

+34-6
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@
2727
#include <QtDebug>
2828
using namespace Lisp;
2929

30-
static QHash<QByteArray,QByteArray> d_symbols;
30+
static QHash<QByteArray,QByteArray> s_symbols;
3131

3232
bool Token::isValid() const
3333
{
@@ -77,21 +77,36 @@ QByteArray Token::getSymbol(const QByteArray& str)
7777
{
7878
if( str.isEmpty() )
7979
return str;
80-
QByteArray& sym = d_symbols[str];
80+
QByteArray& sym = s_symbols[str];
8181
if( sym.isEmpty() )
8282
sym = str;
8383
return sym;
8484
}
8585

86+
QStringList Token::getAllSymbols()
87+
{
88+
QHash<QByteArray,QByteArray>::const_iterator i;
89+
QStringList res;
90+
for( i = s_symbols.begin(); i != s_symbols.end(); ++i )
91+
res.append( QString::fromUtf8(i.key()) );
92+
res.sort(Qt::CaseInsensitive);
93+
return res;
94+
}
95+
8696
char Lexer::readc()
8797
{
98+
static char last = 0;
8899
char res;
89100
if( in && !in->atEnd() && in->getChar(&res) )
90101
{
91102
if( res == '\r' )
92-
res = '\n';
103+
res = '\n'; // immediatedly convert to \n
93104
else if( res == 0 || (!isprint(res) && !isspace(res)) )
105+
{
106+
if( last == '%' )
107+
return ' '; // TODO: we need another solution, maybe (CHARACTER res); TODO: sync with Navigator::decode
94108
return readc(); // ignore all clutter
109+
}
95110
if( res == '\n' )
96111
{
97112
pos.row++;
@@ -101,6 +116,7 @@ char Lexer::readc()
101116
if( res == 0 )
102117
res = -1;
103118
Q_ASSERT(isspace(res) || isprint(res));
119+
last = res;
104120
return res;
105121
}else
106122
{
@@ -119,8 +135,10 @@ void Lexer::ungetc(char c)
119135
return;
120136
Q_ASSERT(isspace(c) || isprint(c));
121137

122-
if( in && c && !isspace(c) ) // don't unget whitespace
138+
if( in && c )
123139
{
140+
if( c == '\n' )
141+
c = ' ';
124142
pos.col--;
125143
in->ungetChar(c);
126144
}
@@ -132,7 +150,7 @@ void Lexer::ungetstr(const QByteArray& str)
132150
ungetc(str[i]);
133151
}
134152

135-
Lexer::Lexer(QObject* parent):QObject(parent),in(0),emitComments(false),packed(true)
153+
Lexer::Lexer(QObject* parent):QObject(parent),in(0),emitComments(false),packed(true),inQuote(false)
136154
{
137155

138156
}
@@ -262,6 +280,16 @@ QList<Token> Lexer::tokens(const QByteArray& code, const QString& path)
262280
return res;
263281
}
264282

283+
void Lexer::startQuote()
284+
{
285+
inQuote = true;
286+
}
287+
288+
void Lexer::endQuote()
289+
{
290+
inQuote = false;
291+
}
292+
265293
Token Lexer::token(TokenType tt, int len, const QByteArray& val)
266294
{
267295
Token t( tt, start, len, val );
@@ -410,7 +438,7 @@ Token Lexer::atom()
410438
while( true )
411439
{
412440
char c = readc();
413-
if( c == '%' )
441+
if( !inQuote && c == '%' )
414442
{
415443
extra++;
416444
c = readc(); // escape

LispLexer.h

+4
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,7 @@ struct Token
6969
const char* getString() const;
7070

7171
static QByteArray getSymbol( const QByteArray& );
72+
static QStringList getAllSymbols();
7273
};
7374

7475
class Lexer : public QObject
@@ -87,6 +88,8 @@ class Lexer : public QObject
8788
void setEmitComments(bool on) { emitComments = on; }
8889
void setPacked(bool on) { packed = on; }
8990
RowCol getPos() const { return pos; }
91+
void startQuote();
92+
void endQuote();
9093

9194
static bool atom_delimiter(char);
9295

@@ -109,6 +112,7 @@ class Lexer : public QObject
109112
QList<Token> buffer;
110113
bool emitComments;
111114
bool packed;
115+
bool inQuote;
112116
};
113117

114118
}

LispNavigator.cpp

+65-5
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,9 @@
4040
#include <QtDebug>
4141
#include <QSettings>
4242
#include <QDesktopWidget>
43+
#include <QShortcut>
44+
#include <QInputDialog>
45+
#include <QListWidget>
4346

4447
static Navigator* s_this = 0;
4548
static void report(QtMsgType type, const QString& message )
@@ -311,6 +314,7 @@ Navigator::Navigator(QWidget *parent)
311314
createSourceTree();
312315
createXref();
313316
createLog();
317+
createAtomList();
314318

315319
s_this = this;
316320
s_oldHandler = qInstallMessageHandler(messageHander);
@@ -330,6 +334,8 @@ Navigator::Navigator(QWidget *parent)
330334

331335
new Gui::AutoShortcut( tr("ALT+Left"), this, this, SLOT(handleGoBack()) );
332336
new Gui::AutoShortcut( tr("ALT+Right"), this, this, SLOT(handleGoForward()) );
337+
new QShortcut(tr("CTRL+SHIFT+F"),this,SLOT(onSearchAtom()));
338+
new QShortcut(tr("CTRL+SHIFT+A"),this,SLOT(onSelectAtom()));
333339

334340
}
335341

@@ -361,6 +367,8 @@ static QByteArray decode(const QByteArray& source)
361367
bytes += "";
362368
else if( isprint(ch) || isspace(ch) )
363369
bytes += ch;
370+
else if( !bytes.isEmpty() && bytes[bytes.size()-1] == '%' )
371+
bytes += ' ';
364372
}
365373
return bytes;
366374
}
@@ -426,6 +434,7 @@ void Navigator::load(const QString& path)
426434
viewer->clear();
427435
asts.clear();
428436
xref.clear();
437+
atomList->clear();
429438
root = path;
430439
QStringList files = collectFiles(path);
431440
QMap<QString,QTreeWidgetItem*> dirs;
@@ -454,6 +463,7 @@ void Navigator::load(const QString& path)
454463
item->setText(0, debang(info.baseName()) );
455464
item->setIcon(0, fip.icon(QFileIconProvider::File));
456465
item->setData(0,Qt::UserRole, f);
466+
item->setToolTip(0,f);
457467

458468
QFile in(f);
459469
if( !in.open(QFile::ReadOnly) )
@@ -466,7 +476,8 @@ void Navigator::load(const QString& path)
466476
if( !r.read(&in, f) )
467477
{
468478
qCritical() << "ERROR " << info.baseName() << r.getPos().row << r.getError();
469-
}else
479+
}
480+
// else
470481
{
471482
Lisp::Reader::Object ast = r.getAst();
472483
asts.insert(f, ast);
@@ -482,9 +493,17 @@ void Navigator::load(const QString& path)
482493
out.setDevice(&file);
483494
ast.print(out);
484495
}
496+
#endif
497+
#if 0
498+
QFile file(f + ".lisp");
499+
if( file.open(QFile::WriteOnly) )
500+
{
501+
in.reset();
502+
QByteArray code = decode(in.readAll());
503+
file.write( code );
504+
}
485505
#endif
486506
}
487-
//return; // TEST
488507

489508
#if 0
490509
Lisp::Lexer lex;
@@ -499,6 +518,8 @@ void Navigator::load(const QString& path)
499518
qCritical() << t.getName() << t.pos.row << t.pos.col << t.val;
500519
#endif
501520
}
521+
522+
atomList->addItems(Lisp::Token::getAllSymbols());
502523
}
503524

504525
void Navigator::logMessage(const QString& str)
@@ -560,6 +581,31 @@ void Navigator::onUpdateLocation(int line, int col)
560581
pushLocation(Location(viewer->getPath(), line,col,viewer->verticalScrollBar()->value()));
561582
}
562583

584+
void Navigator::onSearchAtom()
585+
{
586+
const QString pname = QInputDialog::getText(this, "Search Atom", "Enter an atom pname (case sensitive)");
587+
if( pname.isEmpty() )
588+
return;
589+
const char* atom = Lisp::Token::getSymbol(pname.toUtf8());
590+
fillXrefForAtom(atom, Lisp::RowCol());
591+
}
592+
593+
void Navigator::onSelectAtom()
594+
{
595+
const QString pname = QInputDialog::getItem(this, "Select Atom", "Select an atom from the list:",
596+
Lisp::Token::getAllSymbols() );
597+
if( pname.isEmpty() )
598+
return;
599+
const char* atom = Lisp::Token::getSymbol(pname.toUtf8());
600+
fillXrefForAtom(atom, Lisp::RowCol());
601+
}
602+
603+
void Navigator::onAtomDblClicked(QListWidgetItem* item)
604+
{
605+
const char* atom = Lisp::Token::getSymbol(item->text().toUtf8());
606+
fillXrefForAtom(atom, Lisp::RowCol());
607+
}
608+
563609
void Navigator::pushLocation(const Navigator::Location& loc)
564610
{
565611
if( d_pushBackLock )
@@ -643,7 +689,7 @@ void Navigator::createXref()
643689
d_xref->setRootIsDecorated(false);
644690
vbox->addWidget(d_xref);
645691
dock->setWidget(pane);
646-
addDockWidget( Qt::LeftDockWidgetArea, dock );
692+
addDockWidget( Qt::RightDockWidgetArea, dock );
647693
connect(d_xref, SIGNAL(itemDoubleClicked(QTreeWidgetItem*,int)), this, SLOT(onXrefDblClicked()) );
648694
}
649695

@@ -661,6 +707,20 @@ void Navigator::createLog()
661707
new QShortcut(tr("ESC"), dock, SLOT(close()) );
662708
}
663709

710+
void Navigator::createAtomList()
711+
{
712+
QDockWidget* dock = new QDockWidget( tr("Atoms"), this );
713+
dock->setObjectName("AtomList");
714+
dock->setAllowedAreas( Qt::AllDockWidgetAreas );
715+
dock->setFeatures( QDockWidget::DockWidgetMovable );
716+
atomList = new QListWidget(dock);
717+
atomList->setAlternatingRowColors(true);
718+
atomList->setSortingEnabled(false);
719+
dock->setWidget(atomList);
720+
addDockWidget( Qt::LeftDockWidgetArea, dock );
721+
connect( atomList,SIGNAL(itemDoubleClicked(QListWidgetItem*)),this,SLOT(onAtomDblClicked(QListWidgetItem*)));
722+
}
723+
664724
void Navigator::closeEvent(QCloseEvent* event)
665725
{
666726
QSettings s;
@@ -730,7 +790,7 @@ void Navigator::fillXrefForAtom(const char* atom, const Lisp::RowCol& rc)
730790
QFont f = d_xref->font();
731791
f.setBold(true);
732792

733-
d_xrefTitle->setText(atom);
793+
d_xrefTitle->setText(tr("Atom: %1").arg(atom));
734794
const QString curMod = viewer->getPath();
735795

736796
QTreeWidgetItem* black = 0;
@@ -821,7 +881,7 @@ int main(int argc, char *argv[])
821881
a.setOrganizationName("[email protected]");
822882
a.setOrganizationDomain("github.com/rochus-keller/Interlisp");
823883
a.setApplicationName("InterlispNavigator");
824-
a.setApplicationVersion("0.2.0");
884+
a.setApplicationVersion("0.2.1");
825885
a.setStyle("Fusion");
826886
Navigator w;
827887
if( a.arguments().size() > 1 )

LispNavigator.h

+7
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,8 @@ class QLabel;
2929
class QTreeWidgetItem;
3030
class CodeEditor;
3131
class QPlainTextEdit;
32+
class QListWidget;
33+
class QListWidgetItem;
3234

3335
class Navigator : public QMainWindow
3436
{
@@ -48,6 +50,9 @@ protected slots:
4850
void onXrefDblClicked();
4951
void onCursor();
5052
void onUpdateLocation(int line, int col);
53+
void onSearchAtom();
54+
void onSelectAtom();
55+
void onAtomDblClicked(QListWidgetItem*);
5156

5257
protected:
5358
struct Location
@@ -69,6 +74,7 @@ protected slots:
6974
void createSourceTree();
7075
void createXref();
7176
void createLog();
77+
void createAtomList();
7278
void closeEvent(QCloseEvent* event);
7379
void fillXref();
7480
void fillXrefForAtom(const char* atom, const Lisp::RowCol& rc);
@@ -81,6 +87,7 @@ protected slots:
8187
QLabel* d_xrefTitle;
8288
QTreeWidget* d_xref;
8389
QPlainTextEdit* d_msgLog;
90+
QListWidget* atomList;
8491
class Viewer;
8592
Viewer* viewer;
8693
QString root;

0 commit comments

Comments
 (0)