This Bugzilla instance is a read-only archive of historic NetBeans bug reports. To report a bug in NetBeans please follow the project's instructions for reporting issues.

View | Details | Raw Unified | Return to bug 155167
Collapse All | Expand All

(-)a/db.sql.editor/nbproject/project.xml (+9 lines)
Lines 102-107 Link Here
102
                    </run-dependency>
102
                    </run-dependency>
103
                </dependency>
103
                </dependency>
104
                <dependency>
104
                <dependency>
105
                    <code-name-base>org.netbeans.modules.editor.bracesmatching</code-name-base>
106
                    <build-prerequisite/>
107
                    <compile-dependency/>
108
                    <run-dependency>
109
                        <release-version>0</release-version>
110
                        <specification-version>1.22</specification-version>
111
                    </run-dependency>
112
                </dependency>
113
                <dependency>
105
                    <code-name-base>org.netbeans.modules.editor.completion</code-name-base>
114
                    <code-name-base>org.netbeans.modules.editor.completion</code-name-base>
106
                    <build-prerequisite/>
115
                    <build-prerequisite/>
107
                    <compile-dependency/>
116
                    <compile-dependency/>
(-)a/db.sql.editor/src/org/netbeans/modules/db/sql/editor/SQLBracesMatcher.java (+166 lines)
Line 0 Link Here
1
/*
2
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
3
 *
4
 * Copyright 2012 Oracle and/or its affiliates. All rights reserved.
5
 *
6
 * Oracle and Java are registered trademarks of Oracle and/or its affiliates.
7
 * Other names may be trademarks of their respective owners.
8
 *
9
 * The contents of this file are subject to the terms of either the GNU
10
 * General Public License Version 2 only ("GPL") or the Common
11
 * Development and Distribution License("CDDL") (collectively, the
12
 * "License"). You may not use this file except in compliance with the
13
 * License. You can obtain a copy of the License at
14
 * http://www.netbeans.org/cddl-gplv2.html
15
 * or nbbuild/licenses/CDDL-GPL-2-CP. See the License for the
16
 * specific language governing permissions and limitations under the
17
 * License.  When distributing the software, include this License Header
18
 * Notice in each file and include the License file at
19
 * nbbuild/licenses/CDDL-GPL-2-CP.  Oracle designates this
20
 * particular file as subject to the "Classpath" exception as provided
21
 * by Oracle in the GPL Version 2 section of the License file that
22
 * accompanied this code. If applicable, add the following below the
23
 * License Header, with the fields enclosed by brackets [] replaced by
24
 * your own identifying information:
25
 * "Portions Copyrighted [year] [name of copyright owner]"
26
 *
27
 * If you wish your version of this file to be governed by only the CDDL
28
 * or only the GPL Version 2, indicate your decision by adding
29
 * "[Contributor] elects to include this software in this distribution
30
 * under the [CDDL or GPL Version 2] license." If you do not indicate a
31
 * single choice of license, a recipient has the option to distribute
32
 * your version of this file under either the CDDL, the GPL Version 2 or
33
 * to extend the choice of license to its licensees as provided above.
34
 * However, if you add GPL Version 2 code and therefore, elected the GPL
35
 * Version 2 license, then the option applies only if the new code is
36
 * made subject to such option by the copyright holder.
37
 *
38
 * Contributor(s):
39
 *
40
 * Portions Copyrighted 2012 Sun Microsystems, Inc.
41
 */
42
package org.netbeans.modules.db.sql.editor;
43
44
import javax.swing.text.AbstractDocument;
45
import javax.swing.text.BadLocationException;
46
import org.netbeans.api.editor.mimelookup.MimeRegistration;
47
import org.netbeans.api.editor.mimelookup.MimeRegistrations;
48
import org.netbeans.api.lexer.Token;
49
import org.netbeans.api.lexer.TokenId;
50
import org.netbeans.api.lexer.TokenSequence;
51
import org.netbeans.editor.BaseDocument;
52
import org.netbeans.modules.csl.api.OffsetRange;
53
import org.netbeans.modules.db.sql.lexer.LexerUtilities;
54
import org.netbeans.modules.db.sql.lexer.SQLLexer;
55
import org.netbeans.modules.db.sql.lexer.SQLTokenId;
56
import org.netbeans.spi.editor.bracesmatching.BracesMatcher;
57
import org.netbeans.spi.editor.bracesmatching.BracesMatcherFactory;
58
import org.netbeans.spi.editor.bracesmatching.MatcherContext;
59
60
public class SQLBracesMatcher implements BracesMatcher {
61
62
    @MimeRegistrations({
63
        @MimeRegistration(mimeType = "text/x-sql", service = BracesMatcherFactory.class)
64
    })
65
    public static class Factory implements BracesMatcherFactory {
66
67
        @Override
68
        public SQLBracesMatcher createMatcher(MatcherContext context) {
69
            return new SQLBracesMatcher(context);
70
        }
71
    }
72
    private MatcherContext context;
73
74
    public SQLBracesMatcher(MatcherContext context) {
75
        this.context = context;
76
    }
77
78
    @Override
79
    public int[] findOrigin() throws InterruptedException, BadLocationException {
80
        int[] ret = null;
81
        ((AbstractDocument) context.getDocument()).readLock();
82
        try {
83
            BaseDocument doc = (BaseDocument) context.getDocument();
84
            int offset = context.getSearchOffset();
85
            TokenSequence<? extends SQLTokenId> ts = LexerUtilities.getTokenSequence(doc, offset);
86
87
            if (ts != null) {
88
                ts.move(offset);
89
90
                if (ts.moveNext()) {
91
92
                    Token<? extends SQLTokenId> token = ts.token();
93
94
                    if (token != null) {
95
                        TokenId id = token.id();
96
97
                        if (id == SQLTokenId.LPAREN || id == SQLTokenId.RPAREN) {
98
                            ret = new int[]{ts.offset(), ts.offset() + token.length()};
99
                        } else if (id == SQLTokenId.IDENTIFIER || id == SQLTokenId.STRING) {
100
                            int startOffset = ts.offset();
101
                            int endOffset = ts.offset() + token.length() - 1;
102
                            char startChar = token.text().charAt(0);
103
                            char endChar = token.text().charAt(token.length() - 1);
104
                            if (offset == startOffset && SQLLexer.isEndIdentifierQuoteChar(startChar, endChar)) {
105
                                ret = new int[]{startOffset, startOffset + 1};
106
                            } else if (offset == endOffset && SQLLexer.isEndIdentifierQuoteChar(startChar, endChar)) {
107
                                ret = new int[]{endOffset, endOffset + 1};
108
                            }
109
                        }
110
                    }
111
                }
112
            }
113
114
        } finally {
115
            ((AbstractDocument) context.getDocument()).readUnlock();
116
        }
117
        return ret;
118
    }
119
120
    @Override
121
    public int[] findMatches() throws InterruptedException, BadLocationException {
122
        int[] ret = null;
123
        ((AbstractDocument) context.getDocument()).readLock();
124
        try {
125
            BaseDocument doc = (BaseDocument) context.getDocument();
126
            int offset = context.getSearchOffset();
127
            TokenSequence<? extends SQLTokenId> ts = LexerUtilities.getTokenSequence(doc, offset);
128
129
            if (ts != null) {
130
                ts.move(offset);
131
132
                if (ts.moveNext()) {
133
134
                    Token<? extends SQLTokenId> token = ts.token();
135
136
                    if (token != null) {
137
                        TokenId id = token.id();
138
139
                        if (id == SQLTokenId.LPAREN) {
140
                            OffsetRange r = LexerUtilities.findFwd(ts, SQLTokenId.LPAREN.ordinal(),
141
                                    SQLTokenId.RPAREN.ordinal());
142
                            ret = new int[]{r.getStart(), r.getEnd()};
143
                        } else if (id == SQLTokenId.RPAREN) {
144
                            OffsetRange r = LexerUtilities.findBwd(ts, SQLTokenId.LPAREN.ordinal(),
145
                                    SQLTokenId.RPAREN.ordinal());
146
                            ret = new int[]{r.getStart(), r.getEnd()};
147
                        } else if (id == SQLTokenId.IDENTIFIER || id == SQLTokenId.STRING) {
148
                            int startOffset = ts.offset();
149
                            int endOffset = ts.offset() + token.length() - 1;
150
                            char startChar = token.text().charAt(0);
151
                            char endChar = token.text().charAt(token.length() - 1);
152
                            if (offset == startOffset && SQLLexer.isEndIdentifierQuoteChar(startChar, endChar)) {
153
                                ret = new int[]{endOffset, endOffset + 1};
154
                            } else if (offset == endOffset && SQLLexer.isEndIdentifierQuoteChar(startChar, endChar)) {
155
                                ret = new int[]{startOffset, startOffset + 1};
156
                            }
157
                        }
158
                    }
159
                }
160
            }
161
        } finally {
162
            ((AbstractDocument) context.getDocument()).readUnlock();
163
        }
164
        return ret;
165
    }
166
}
(-)a/db.sql.editor/src/org/netbeans/modules/db/sql/lexer/LexerUtilities.java (+159 lines)
Line 0 Link Here
1
/*
2
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
3
 *
4
 * Copyright 2012 Oracle and/or its affiliates. All rights reserved.
5
 *
6
 * Oracle and Java are registered trademarks of Oracle and/or its affiliates.
7
 * Other names may be trademarks of their respective owners.
8
 *
9
 * The contents of this file are subject to the terms of either the GNU
10
 * General Public License Version 2 only ("GPL") or the Common
11
 * Development and Distribution License("CDDL") (collectively, the
12
 * "License"). You may not use this file except in compliance with the
13
 * License. You can obtain a copy of the License at
14
 * http://www.netbeans.org/cddl-gplv2.html
15
 * or nbbuild/licenses/CDDL-GPL-2-CP. See the License for the
16
 * specific language governing permissions and limitations under the
17
 * License.  When distributing the software, include this License Header
18
 * Notice in each file and include the License file at
19
 * nbbuild/licenses/CDDL-GPL-2-CP.  Oracle designates this
20
 * particular file as subject to the "Classpath" exception as provided
21
 * by Oracle in the GPL Version 2 section of the License file that
22
 * accompanied this code. If applicable, add the following below the
23
 * License Header, with the fields enclosed by brackets [] replaced by
24
 * your own identifying information:
25
 * "Portions Copyrighted [year] [name of copyright owner]"
26
 *
27
 * If you wish your version of this file to be governed by only the CDDL
28
 * or only the GPL Version 2, indicate your decision by adding
29
 * "[Contributor] elects to include this software in this distribution
30
 * under the [CDDL or GPL Version 2] license." If you do not indicate a
31
 * single choice of license, a recipient has the option to distribute
32
 * your version of this file under either the CDDL, the GPL Version 2 or
33
 * to extend the choice of license to its licensees as provided above.
34
 * However, if you add GPL Version 2 code and therefore, elected the GPL
35
 * Version 2 license, then the option applies only if the new code is
36
 * made subject to such option by the copyright holder.
37
 *
38
 * Contributor(s):
39
 *
40
 * Portions Copyrighted 2012 Sun Microsystems, Inc.
41
 */
42
package org.netbeans.modules.db.sql.lexer;
43
44
import java.util.List;
45
import javax.swing.text.Document;
46
import org.netbeans.api.lexer.Token;
47
import org.netbeans.api.lexer.TokenHierarchy;
48
import org.netbeans.api.lexer.TokenId;
49
import org.netbeans.api.lexer.TokenSequence;
50
import org.netbeans.modules.csl.api.OffsetRange;
51
52
/**
53
 * Based on documentation on netbeans.org: http://wiki.netbeans.org/Netbeans_Antlr_BracesMatching
54
 */
55
public class LexerUtilities {
56
57
    @SuppressWarnings("unchecked")
58
    public static TokenSequence<SQLTokenId> getTokenSequence(Document doc, int offset) {
59
        TokenHierarchy<Document> th = TokenHierarchy.get(doc);
60
        TokenSequence<SQLTokenId> ts = th == null ? null : th.tokenSequence(SQLTokenId.language());
61
62
        if (ts == null) {
63
            // Possibly an embedding scenario such as an RHTML file
64
            // First try with backward bias true
65
            List<TokenSequence<?>> list = th.embeddedTokenSequences(offset, true);
66
67
            for (TokenSequence<? extends TokenId> t : list) {
68
                if (t.language() == SQLTokenId.language()) {
69
                    ts = (TokenSequence<SQLTokenId>) t;
70
                    break;
71
                }
72
            }
73
74
            if (ts == null) {
75
                list = th.embeddedTokenSequences(offset, false);
76
                for (TokenSequence<? extends TokenId> t : list) {
77
                    if (t.language() == SQLTokenId.language()) {
78
                        ts = (TokenSequence<SQLTokenId>) t;
79
                        break;
80
                    }
81
                }
82
            }
83
        }
84
85
        return ts;
86
    }
87
88
    /**
89
     * Search forwards in the token sequence until a matching closing token is
90
     * found so keeps track of nested pairs of up-down eg (()) is ignored if
91
     * we're searching for a )
92
     *
93
     * @param ts the TokenSequence set to the position after an up
94
     * @param up the opening token eg { or [
95
     * @param down the closing token eg } or ]
96
     * @return the Range of closing token in our case 1 char
97
     */
98
    public static OffsetRange findFwd(TokenSequence<? extends SQLTokenId> ts, int up, int down) {
99
        int balance = 0;
100
101
        while (ts.moveNext()) {
102
            Token<? extends SQLTokenId> token = ts.token();
103
104
            if (token.id().ordinal() == up) {
105
                balance++;
106
            } else if (token.id().ordinal() == down) {
107
                if (balance == 0) {
108
                    return new OffsetRange(ts.offset(), ts.offset() + token.length());
109
                }
110
                balance--;
111
            }
112
        }
113
114
        return OffsetRange.NONE;
115
    }
116
117
    /**
118
     * Search forwards in the token sequence until a matching closing token is
119
     * found so keeps track of nested pairs of up-down eg (()) is ignored if
120
     * we're searching for a )
121
     *
122
     * @param ts the TokenSequence set to the position after an up
123
     * @param up the opening token eg { or [
124
     * @param down the closing token eg } or ]
125
     * @return the Range of closing token in our case 1 char
126
     */
127
    public static OffsetRange findBwd(TokenSequence<? extends SQLTokenId> ts, int up, int down) {
128
        int balance = 0;
129
130
        while (ts.movePrevious()) {
131
            Token<? extends SQLTokenId> token = ts.token();
132
133
            if (token.id().ordinal() == up) {
134
                if (balance == 0) {
135
                    return new OffsetRange(ts.offset(), ts.offset() + token.length());
136
                }
137
138
                balance++;
139
            } else if (token.id().ordinal() == down) {
140
                balance--;
141
            }
142
        }
143
144
        return OffsetRange.NONE;
145
    }
146
147
    public static boolean textEquals(CharSequence text1, char... text2) {
148
        int len = text1.length();
149
        if (len == text2.length) {
150
            for (int i = len - 1; i >= 0; i--) {
151
                if (text1.charAt(i) != text2[i]) {
152
                    return false;
153
                }
154
            }
155
            return true;
156
        }
157
        return false;
158
    }
159
}

Return to bug 155167