Search code examples
javaswingjtextarea

Column mode select in JTextArea?


Is there a way to implement a "column mode" select in a JTextArea, in the way many modern editors does (i.e. Alt + mouse select in Notepad++)?

I did a search around and couldn't find anything similar. Probably I didn't use the right key words....


Solution

  • I'm happy to say I got time (only two years after!!) to put together a successfully working class taking the code from Maneesh Godbole from coderanch pointed by @AxelH and also the code from Stanislav Lapitsky in http://java-sl.com . Thankyou very much!!

    Hope it's useful :)

    public class ColumnTextArea extends javax.swing.JTextArea{
    
        private static final long serialVersionUID = 1L;
    
        public ColumnTextArea(int rows, int columns){
            super(rows, columns);
            this.setCaret(new ColumnTextArea.ColumnSelectionCaret());
    
            //Disable ALT key presses when the textarea is focused
            this.getInputMap(javax.swing.JComponent.WHEN_FOCUSED).put(javax.swing.KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_ALT, 0, true), "ALTKEY");
            this.getActionMap().put("ALTKEY", new javax.swing.AbstractAction() {
                private static final long serialVersionUID = 1L;
                @Override
                public void actionPerformed(java.awt.event.ActionEvent e) {
                    System.out.println("Performed");
                }
            });
        }
    
        @Override
        public java.lang.String getSelectedText(){
            javax.swing.text.Highlighter.Highlight[] selections = this.getHighlighter().getHighlights();
            String text = "";
            int cnt = selections.length;
            if(cnt == 0){
                text = super.getSelectedText();
            }else{
                for (int i=0; i<cnt; i++) {
                    int start = selections[i].getStartOffset();
                    int end = selections[i].getEndOffset();
                    java.lang.String selectedText = "";
                    try {
                        selectedText = this.getDocument().getText(start,end-start);
                    } catch (javax.swing.text.BadLocationException e) {
                        e.printStackTrace();
                    }
                    if(i == 0){
                        text = selectedText;
                    }else{
                        text += '\n' + selectedText;
                    }
                }
            }
            return(text);
        }
    
    
        @Override
        public void copy(){
            java.awt.datatransfer.StringSelection ss = new java.awt.datatransfer.StringSelection(this.getSelectedText());
            java.awt.Toolkit.getDefaultToolkit().getSystemClipboard().setContents(ss,ss);        
        }
    
        class ColumnSelectionCaret extends javax.swing.text.DefaultCaret {
            private static final long serialVersionUID = 1L;
            java.awt.Point lastPoint = new java.awt.Point(0,0);
            boolean altDown = false;
            boolean moving = false;
    
            @Override
            public void mouseMoved(java.awt.event.MouseEvent e) {
                super.mouseMoved(e);
            }
    
            @Override
            public void mouseClicked(java.awt.event.MouseEvent e) {
                super.mouseClicked(e);
                if(e.getClickCount()==1){
                    this.getComponent().getHighlighter().removeAllHighlights();
                }
            }
    
            @Override
            public void mousePressed(java.awt.event.MouseEvent e){
                super.mousePressed(e);
                this.lastPoint = new java.awt.Point(e.getX(),e.getY());
                this.altDown = e.isAltDown();
            }
    
            @Override
            protected void moveCaret(java.awt.event.MouseEvent e) {
                if(!this.moving){
                    this.getComponent().getHighlighter().removeAllHighlights();
                    this.moving = true;
                }
                if(this.altDown){
                    java.awt.Point pt = new java.awt.Point(e.getX(), e.getY());
                    javax.swing.text.Position.Bias[] biasRet = new javax.swing.text.Position.Bias[1];
                    int pos = getComponent().getUI().viewToModel(getComponent(), pt, biasRet);
                    if(biasRet[0] == null)
                        biasRet[0] = javax.swing.text.Position.Bias.Forward;
                    if (pos >= 0) {
                        this.setDot(pos);
                        java.awt.Point start = new java.awt.Point(Math.min(this.lastPoint.x,pt.x), Math.min(this.lastPoint.y,pt.y));
                        java.awt.Point end = new java.awt.Point(Math.max(this.lastPoint.x,pt.x), Math.max(this.lastPoint.y,pt.y));
                        this.customHighlight(start,end);
                    }
                }else{
                    super.moveCaret(e);
                }
            }
            @Override
            public void mouseReleased(java.awt.event.MouseEvent e){
                this.moving = false;
                super.mouseReleased(e);
            }
    
            protected void customHighlight(java.awt.Point start, java.awt.Point end) {
                javax.swing.text.JTextComponent comp = this.getComponent();
    
                comp.getHighlighter().removeAllHighlights();
                int y = start.y;
                int firstX = start.x;
                int lastX = end.x;
    
                int pos1 = comp.getUI().viewToModel(comp, new java.awt.Point(firstX,y));
                int pos2 = comp.getUI().viewToModel(comp, new java.awt.Point(lastX,y));
                try {
                    comp.getHighlighter().addHighlight(pos1, pos2, this.getSelectionPainter());
                }
                catch (Exception ex) {
                    ex.printStackTrace();
                }
                y++;
                while (y<end.y) {
                    int pos1new = comp.getUI().viewToModel(comp, new java.awt.Point(firstX,y));
                    int pos2new = comp.getUI().viewToModel(comp, new java.awt.Point(lastX,y));
                    if (pos1!=pos1new)  {
                        pos1=pos1new;
                        pos2=pos2new;
                        try {
                            comp.getHighlighter().addHighlight(pos1, pos2, this.getSelectionPainter());
                        }
                        catch (Exception ex) {
                            ex.printStackTrace();
                        }
                    }
                    y++;
                }
            }
        }
    }