| 1 | // License: GPL. Copyright 2007 by Immanuel Scholz and others
|
|---|
| 2 | package org.openstreetmap.josm.actions.search;
|
|---|
| 3 |
|
|---|
| 4 | import java.io.IOException;
|
|---|
| 5 | import java.io.PushbackReader;
|
|---|
| 6 | import java.util.LinkedList;
|
|---|
| 7 |
|
|---|
| 8 | public class PushbackTokenizer {
|
|---|
| 9 | private PushbackReader search;
|
|---|
| 10 |
|
|---|
| 11 | private LinkedList<String> pushBackBuf = new LinkedList<String>();
|
|---|
| 12 |
|
|---|
| 13 | public PushbackTokenizer(PushbackReader search) {
|
|---|
| 14 | this.search = search;
|
|---|
| 15 | }
|
|---|
| 16 |
|
|---|
| 17 | /**
|
|---|
| 18 | * The token returned is <code>null</code> or starts with an identifier character:
|
|---|
| 19 | * - for an '-'. This will be the only character
|
|---|
| 20 | * : for an key. The value is the next token
|
|---|
| 21 | * | for "OR"
|
|---|
| 22 | * ' ' for anything else.
|
|---|
| 23 | * @return The next token in the stream.
|
|---|
| 24 | */
|
|---|
| 25 | public String nextToken() {
|
|---|
| 26 | if (!pushBackBuf.isEmpty()) {
|
|---|
| 27 | return pushBackBuf.removeLast();
|
|---|
| 28 | }
|
|---|
| 29 |
|
|---|
| 30 | try {
|
|---|
| 31 | int next;
|
|---|
| 32 | char c = ' ';
|
|---|
| 33 | while (c == ' ' || c == '\t' || c == '\n') {
|
|---|
| 34 | next = search.read();
|
|---|
| 35 | if (next == -1)
|
|---|
| 36 | return null;
|
|---|
| 37 | c = (char)next;
|
|---|
| 38 | }
|
|---|
| 39 | StringBuilder s;
|
|---|
| 40 | switch (c) {
|
|---|
| 41 | case ':':
|
|---|
| 42 | next = search.read();
|
|---|
| 43 | c = (char) next;
|
|---|
| 44 | if (next == -1 || c == ' ' || c == '\t') {
|
|---|
| 45 | pushBack(" ");
|
|---|
| 46 | } else {
|
|---|
| 47 | search.unread(next);
|
|---|
| 48 | }
|
|---|
| 49 | return ":";
|
|---|
| 50 | case '-':
|
|---|
| 51 | return "-";
|
|---|
| 52 | case '(':
|
|---|
| 53 | return "(";
|
|---|
| 54 | case ')':
|
|---|
| 55 | return ")";
|
|---|
| 56 | case '"':
|
|---|
| 57 | s = new StringBuilder(" ");
|
|---|
| 58 | for (int nc = search.read(); nc != -1 && nc != '"'; nc = search.read())
|
|---|
| 59 | s.append((char)nc);
|
|---|
| 60 | return s.toString();
|
|---|
| 61 | default:
|
|---|
| 62 | s = new StringBuilder();
|
|---|
| 63 | for (;;) {
|
|---|
| 64 | s.append(c);
|
|---|
| 65 | next = search.read();
|
|---|
| 66 | if (next == -1) {
|
|---|
| 67 | if (s.toString().equals("OR"))
|
|---|
| 68 | return "|";
|
|---|
| 69 | return " "+s.toString();
|
|---|
| 70 | }
|
|---|
| 71 | c = (char)next;
|
|---|
| 72 | if (c == ' ' || c == '\t' || c == '"' || c == ':' || c == '(' || c == ')') {
|
|---|
| 73 | search.unread(next);
|
|---|
| 74 | if (s.toString().equals("OR"))
|
|---|
| 75 | return "|";
|
|---|
| 76 | return " "+s.toString();
|
|---|
| 77 | }
|
|---|
| 78 | }
|
|---|
| 79 | }
|
|---|
| 80 | } catch (IOException e) {
|
|---|
| 81 | throw new RuntimeException(e.getMessage(), e);
|
|---|
| 82 | }
|
|---|
| 83 | }
|
|---|
| 84 |
|
|---|
| 85 | public boolean readIfEqual(String tok) {
|
|---|
| 86 | String nextTok = nextToken();
|
|---|
| 87 | if (nextTok == null ? tok == null : nextTok.equals(tok))
|
|---|
| 88 | return true;
|
|---|
| 89 | pushBack(nextTok);
|
|---|
| 90 | return false;
|
|---|
| 91 | }
|
|---|
| 92 |
|
|---|
| 93 | public String readText() {
|
|---|
| 94 | String nextTok = nextToken();
|
|---|
| 95 | if (nextTok != null && nextTok.startsWith(" "))
|
|---|
| 96 | return nextTok.substring(1);
|
|---|
| 97 | pushBack(nextTok);
|
|---|
| 98 | return null;
|
|---|
| 99 | }
|
|---|
| 100 |
|
|---|
| 101 | public void pushBack(String tok) {
|
|---|
| 102 | pushBackBuf.addLast(tok);
|
|---|
| 103 | }
|
|---|
| 104 | }
|
|---|