-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathAbstractButton.java
2652 lines (2430 loc) · 80.7 KB
/
AbstractButton.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
/* AbstractButton.java -- Provides basic button functionality.
Copyright (C) 2002, 2004, 2006, Free Software Foundation, Inc.
This file is part of GNU Classpath.
GNU Classpath is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2, or (at your option)
any later version.
GNU Classpath is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received a copy of the GNU General Public License
along with GNU Classpath; see the file COPYING. If not, write to the
Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
02110-1301 USA.
Linking this library statically or dynamically with other modules is
making a combined work based on this library. Thus, the terms and
conditions of the GNU General Public License cover the whole
combination.
As a special exception, the copyright holders of this library give you
permission to link this library with independent modules to produce an
executable, regardless of the license terms of these independent
modules, and to copy and distribute the resulting executable under
terms of your choice, provided that you also meet, for each linked
independent module, the terms and conditions of the license of that
module. An independent module is a module which is not derived from
or based on this library. If you modify this library, you may extend
this exception to your version of the library, but you are not
obligated to do so. If you do not wish to do so, delete this
exception statement from your version. */
package javax.swing;
import gnu.java.lang.CPStringBuilder;
import java.awt.Component;
import java.awt.Graphics;
import java.awt.Image;
import java.awt.Insets;
import java.awt.ItemSelectable;
import java.awt.LayoutManager;
import java.awt.Point;
import java.awt.Rectangle;
import java.awt.Shape;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.ItemEvent;
import java.awt.event.ItemListener;
import java.awt.image.ImageObserver;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.io.Serializable;
import java.util.Enumeration;
import javax.accessibility.Accessible;
import javax.accessibility.AccessibleAction;
import javax.accessibility.AccessibleContext;
import javax.accessibility.AccessibleIcon;
import javax.accessibility.AccessibleRelation;
import javax.accessibility.AccessibleRelationSet;
import javax.accessibility.AccessibleState;
import javax.accessibility.AccessibleStateSet;
import javax.accessibility.AccessibleText;
import javax.accessibility.AccessibleValue;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import javax.swing.plaf.ButtonUI;
import javax.swing.plaf.basic.BasicHTML;
import javax.swing.text.AttributeSet;
import javax.swing.text.BadLocationException;
import javax.swing.text.Document;
import javax.swing.text.Element;
import javax.swing.text.Position;
import javax.swing.text.StyledDocument;
import javax.swing.text.View;
/**
* Provides an abstract implementation of common button behaviour,
* data model and look & feel.
*
* <p>This class is supposed to serve as a base class for
* several kinds of buttons with similar but non-identical semantics:
* toggle buttons (radio buttons and checkboxes), simple push buttons,
* menu items, etc.</p>
*
* <p>Buttons have many properties, some of which are stored in this class
* while others are delegated to the button's model. The following properties
* are available:</p>
*
* <table>
* <tr><th>Property </th><th>Stored in</th><th>Bound?</th></tr>
*
* <tr><td>action </td><td>button</td> <td>no</td></tr>
* <tr><td>actionCommand </td><td>model</td> <td>no</td></tr>
* <tr><td>borderPainted </td><td>button</td> <td>yes</td></tr>
* <tr><td>contentAreaFilled </td><td>button</td> <td>yes</td></tr>
* <tr><td>disabledIcon </td><td>button</td> <td>yes</td></tr>
* <tr><td>disabledSelectedIcon </td><td>button</td> <td>yes</td></tr>
* <tr><td>displayedMnemonicIndex </td><td>button</td> <td>no</td></tr>
* <tr><td>enabled </td><td>model</td> <td>no</td></tr>
* <tr><td>focusPainted </td><td>button</td> <td>yes</td></tr>
* <tr><td>horizontalAlignment </td><td>button</td> <td>yes</td></tr>
* <tr><td>horizontalTextPosition </td><td>button</td> <td>yes</td></tr>
* <tr><td>icon </td><td>button</td> <td>yes</td></tr>
* <tr><td>iconTextGap </td><td>button</td> <td>no</td></tr>
* <tr><td>label (same as text) </td><td>model</td> <td>yes</td></tr>
* <tr><td>margin </td><td>button</td> <td>yes</td></tr>
* <tr><td>multiClickThreshold </td><td>button</td> <td>no</td></tr>
* <tr><td>pressedIcon </td><td>button</td> <td>yes</td></tr>
* <tr><td>rolloverEnabled </td><td>button</td> <td>yes</td></tr>
* <tr><td>rolloverIcon </td><td>button</td> <td>yes</td></tr>
* <tr><td>rolloverSelectedIcon </td><td>button</td> <td>yes</td></tr>
* <tr><td>selected </td><td>model</td> <td>no</td></tr>
* <tr><td>selectedIcon </td><td>button</td> <td>yes</td></tr>
* <tr><td>selectedObjects </td><td>button</td> <td>no</td></tr>
* <tr><td>text </td><td>model</td> <td>yes</td></tr>
* <tr><td>UI </td><td>button</td> <td>yes</td></tr>
* <tr><td>verticalAlignment </td><td>button</td> <td>yes</td></tr>
* <tr><td>verticalTextPosition </td><td>button</td> <td>yes</td></tr>
*
* </table>
*
* <p>The various behavioral aspects of these properties follows:</p>
*
* <ul>
*
* <li>When non-bound properties stored in the button change, the button
* fires ChangeEvents to its ChangeListeners.</li>
*
* <li>When bound properties stored in the button change, the button fires
* PropertyChangeEvents to its PropertyChangeListeners</li>
*
* <li>If any of the model's properties change, it fires a ChangeEvent to
* its ChangeListeners, which include the button.</li>
*
* <li>If the button receives a ChangeEvent from its model, it will
* propagate the ChangeEvent to its ChangeListeners, with the ChangeEvent's
* "source" property set to refer to the button, rather than the model. The
* the button will request a repaint, to paint its updated state.</li>
*
* <li>If the model's "selected" property changes, the model will fire an
* ItemEvent to its ItemListeners, which include the button, in addition to
* the ChangeEvent which models the property change. The button propagates
* ItemEvents directly to its ItemListeners.</li>
*
* <li>If the model's armed and pressed properties are simultaneously
* <code>true</code>, the model will fire an ActionEvent to its
* ActionListeners, which include the button. The button will propagate
* this ActionEvent to its ActionListeners, with the ActionEvent's "source"
* property set to refer to the button, rather than the model.</li>
*
* </ul>
*
* @author Ronald Veldema (rveldema@cs.vu.nl)
* @author Graydon Hoare (graydon@redhat.com)
*/
public abstract class AbstractButton extends JComponent
implements ItemSelectable, SwingConstants
{
private static final long serialVersionUID = -937921345538462020L;
/**
* An extension of ChangeListener to be serializable.
*/
protected class ButtonChangeListener
implements ChangeListener, Serializable
{
private static final long serialVersionUID = 1471056094226600578L;
/**
* The spec has no public/protected constructor for this class, so do we.
*/
ButtonChangeListener()
{
// Nothing to do here.
}
/**
* Notified when the target of the listener changes its state.
*
* @param ev the ChangeEvent describing the change
*/
public void stateChanged(ChangeEvent ev)
{
getEventHandler().stateChanged(ev);
}
}
/**
* The combined event handler for ActionEvent, ChangeEvent and
* ItemEvent. This combines ButtonChangeListener, ActionListener
*/
private class EventHandler
implements ActionListener, ChangeListener, ItemListener
{
public void actionPerformed(ActionEvent ev)
{
fireActionPerformed(ev);
}
public void stateChanged(ChangeEvent ev)
{
fireStateChanged();
repaint();
}
public void itemStateChanged(ItemEvent ev)
{
fireItemStateChanged(ev);
}
}
/** The icon displayed by default. */
Icon default_icon;
/** The icon displayed when the button is pressed. */
Icon pressed_icon;
/** The icon displayed when the button is disabled. */
Icon disabledIcon;
/** The icon displayed when the button is selected. */
Icon selectedIcon;
/** The icon displayed when the button is selected but disabled. */
Icon disabledSelectedIcon;
/** The icon displayed when the button is rolled over. */
Icon rolloverIcon;
/** The icon displayed when the button is selected and rolled over. */
Icon rolloverSelectedIcon;
/** The icon currently displayed. */
Icon current_icon;
/** The text displayed in the button. */
String text;
/**
* The gap between icon and text, if both icon and text are
* non-<code>null</code>.
*/
int iconTextGap;
/** The vertical alignment of the button's text and icon. */
int verticalAlignment;
/** The horizontal alignment of the button's text and icon. */
int horizontalAlignment;
/** The horizontal position of the button's text relative to its icon. */
int horizontalTextPosition;
/** The vertical position of the button's text relative to its icon. */
int verticalTextPosition;
/** Whether or not the button paints its border. */
boolean borderPainted;
/** Whether or not the button paints its focus state. */
boolean focusPainted;
/** Whether or not the button fills its content area. */
boolean contentAreaFilled;
/** Whether rollover is enabled. */
boolean rollOverEnabled;
/** The action taken when the button is clicked. */
Action action;
/** The button's current state. */
protected ButtonModel model;
/** The margin between the button's border and its label. */
Insets margin;
/**
* A hint to the look and feel class, suggesting which character in the
* button's label should be underlined when drawing the label.
*/
int mnemonicIndex;
/**
* Listener the button uses to receive ActionEvents from its model.
*/
protected ActionListener actionListener;
/**
* Listener the button uses to receive ItemEvents from its model.
*/
protected ItemListener itemListener;
/**
* Listener the button uses to receive ChangeEvents from its model.
*/
protected ChangeListener changeListener;
/**
* The event handler for ActionEvent, ItemEvent and ChangeEvent.
* This replaces the above three handlers and combines them
* into one for efficiency.
*/
private EventHandler eventHandler;
/**
* The time in milliseconds in which clicks get coalesced into a single
* <code>ActionEvent</code>.
*/
long multiClickThreshhold;
/**
* Listener the button uses to receive PropertyChangeEvents from its
* Action.
*/
PropertyChangeListener actionPropertyChangeListener;
/** ChangeEvent that is fired to button's ChangeEventListeners */
protected ChangeEvent changeEvent = new ChangeEvent(this);
/**
* Indicates if the borderPainted property has been set by a client
* program or by the UI.
*
* @see #setUIProperty(String, Object)
* @see LookAndFeel#installProperty(JComponent, String, Object)
*/
private boolean clientBorderPaintedSet = false;
/**
* Indicates if the rolloverEnabled property has been set by a client
* program or by the UI.
*
* @see #setUIProperty(String, Object)
* @see LookAndFeel#installProperty(JComponent, String, Object)
*/
private boolean clientRolloverEnabledSet = false;
/**
* Indicates if the iconTextGap property has been set by a client
* program or by the UI.
*
* @see #setUIProperty(String, Object)
* @see LookAndFeel#installProperty(JComponent, String, Object)
*/
private boolean clientIconTextGapSet = false;
/**
* Indicates if the contentAreaFilled property has been set by a client
* program or by the UI.
*
* @see #setUIProperty(String, Object)
* @see LookAndFeel#installProperty(JComponent, String, Object)
*/
private boolean clientContentAreaFilledSet = false;
/**
* Fired in a PropertyChangeEvent when the "borderPainted" property changes.
*/
public static final String BORDER_PAINTED_CHANGED_PROPERTY = "borderPainted";
/**
* Fired in a PropertyChangeEvent when the "contentAreaFilled" property
* changes.
*/
public static final String CONTENT_AREA_FILLED_CHANGED_PROPERTY =
"contentAreaFilled";
/**
* Fired in a PropertyChangeEvent when the "disabledIcon" property changes.
*/
public static final String DISABLED_ICON_CHANGED_PROPERTY = "disabledIcon";
/**
* Fired in a PropertyChangeEvent when the "disabledSelectedIcon" property
* changes.
*/
public static final String DISABLED_SELECTED_ICON_CHANGED_PROPERTY =
"disabledSelectedIcon";
/**
* Fired in a PropertyChangeEvent when the "focusPainted" property changes.
*/
public static final String FOCUS_PAINTED_CHANGED_PROPERTY = "focusPainted";
/**
* Fired in a PropertyChangeEvent when the "horizontalAlignment" property
* changes.
*/
public static final String HORIZONTAL_ALIGNMENT_CHANGED_PROPERTY =
"horizontalAlignment";
/**
* Fired in a PropertyChangeEvent when the "horizontalTextPosition" property
* changes.
*/
public static final String HORIZONTAL_TEXT_POSITION_CHANGED_PROPERTY =
"horizontalTextPosition";
/**
* Fired in a PropertyChangeEvent when the "icon" property changes. */
public static final String ICON_CHANGED_PROPERTY = "icon";
/** Fired in a PropertyChangeEvent when the "margin" property changes. */
public static final String MARGIN_CHANGED_PROPERTY = "margin";
/** Fired in a PropertyChangeEvent when the "mnemonic" property changes. */
public static final String MNEMONIC_CHANGED_PROPERTY = "mnemonic";
/** Fired in a PropertyChangeEvent when the "model" property changes. */
public static final String MODEL_CHANGED_PROPERTY = "model";
/** Fired in a PropertyChangeEvent when the "pressedIcon" property changes. */
public static final String PRESSED_ICON_CHANGED_PROPERTY = "pressedIcon";
/**
* Fired in a PropertyChangeEvent when the "rolloverEnabled" property
* changes.
*/
public static final String ROLLOVER_ENABLED_CHANGED_PROPERTY =
"rolloverEnabled";
/**
* Fired in a PropertyChangeEvent when the "rolloverIcon" property changes.
*/
public static final String ROLLOVER_ICON_CHANGED_PROPERTY = "rolloverIcon";
/**
* Fired in a PropertyChangeEvent when the "rolloverSelectedIcon" property
* changes.
*/
public static final String ROLLOVER_SELECTED_ICON_CHANGED_PROPERTY =
"rolloverSelectedIcon";
/**
* Fired in a PropertyChangeEvent when the "selectedIcon" property changes.
*/
public static final String SELECTED_ICON_CHANGED_PROPERTY = "selectedIcon";
/** Fired in a PropertyChangeEvent when the "text" property changes. */
public static final String TEXT_CHANGED_PROPERTY = "text";
/**
* Fired in a PropertyChangeEvent when the "verticalAlignment" property
* changes.
*/
public static final String VERTICAL_ALIGNMENT_CHANGED_PROPERTY =
"verticalAlignment";
/**
* Fired in a PropertyChangeEvent when the "verticalTextPosition" property
* changes.
*/
public static final String VERTICAL_TEXT_POSITION_CHANGED_PROPERTY =
"verticalTextPosition";
/**
* A Java Accessibility extension of the AbstractButton.
*/
protected abstract class AccessibleAbstractButton
extends AccessibleJComponent implements AccessibleAction, AccessibleValue,
AccessibleText
{
private static final long serialVersionUID = -5673062525319836790L;
protected AccessibleAbstractButton()
{
// Nothing to do here yet.
}
/**
* Returns the accessible state set of this object. In addition to the
* superclass's states, the <code>AccessibleAbstractButton</code>
* supports the following states: {@link AccessibleState#ARMED},
* {@link AccessibleState#FOCUSED}, {@link AccessibleState#PRESSED} and
* {@link AccessibleState#CHECKED}.
*
* @return the current state of this accessible object
*/
public AccessibleStateSet getAccessibleStateSet()
{
AccessibleStateSet state = super.getAccessibleStateSet();
if (getModel().isArmed())
state.add(AccessibleState.ARMED);
if (getModel().isPressed())
state.add(AccessibleState.PRESSED);
if (isSelected())
state.add(AccessibleState.CHECKED);
return state;
}
/**
* Returns the accessible name for the button.
*/
public String getAccessibleName()
{
String result = super.getAccessibleName();
if (result == null)
result = text;
return result;
}
/**
* Returns the accessible icons of this object. If the AbstractButton's
* icon is an Accessible, and it's AccessibleContext is an AccessibleIcon,
* then this AccessibleIcon is returned, otherwise <code>null</code>.
*
* @return the accessible icons of this object, or <code>null</code> if
* there is no accessible icon
*/
public AccessibleIcon[] getAccessibleIcon()
{
AccessibleIcon[] ret = null;
Icon icon = getIcon();
if (icon instanceof Accessible)
{
AccessibleContext ctx = ((Accessible) icon).getAccessibleContext();
if (ctx instanceof AccessibleIcon)
{
ret = new AccessibleIcon[]{ (AccessibleIcon) ctx };
}
}
return ret;
}
/**
* Returns the accessible relations of this AccessibleAbstractButton.
* If the AbstractButton is part of a ButtonGroup, then all the buttons
* in this button group are added as targets in a MEMBER_OF relation,
* otherwise an empty relation set is returned (from super).
*
* @return the accessible relations of this AccessibleAbstractButton
*/
public AccessibleRelationSet getAccessibleRelationSet()
{
AccessibleRelationSet relations = super.getAccessibleRelationSet();
ButtonModel model = getModel();
if (model instanceof DefaultButtonModel)
{
ButtonGroup group = ((DefaultButtonModel) model).getGroup();
if (group != null)
{
Object[] target = new Object[group.getButtonCount()];
Enumeration els = group.getElements();
for (int index = 0; els.hasMoreElements(); ++index)
{
target[index] = els.nextElement();
}
AccessibleRelation rel =
new AccessibleRelation(AccessibleRelation.MEMBER_OF);
rel.setTarget(target);
relations.add(rel);
}
}
return relations;
}
/**
* Returns the accessible action associated with this object. For buttons,
* this will be <code>this</code>.
*
* @return <code>this</code>
*/
public AccessibleAction getAccessibleAction()
{
return this;
}
/**
* Returns the accessible value of this AccessibleAbstractButton, which
* is always <code>this</code>.
*
* @return the accessible value of this AccessibleAbstractButton, which
* is always <code>this</code>
*/
public AccessibleValue getAccessibleValue()
{
return this;
}
/**
* Returns the number of accessible actions that are supported by this
* object. Buttons support one action by default ('press button'), so this
* method always returns <code>1</code>.
*
* @return <code>1</code>, the number of supported accessible actions
*/
public int getAccessibleActionCount()
{
return 1;
}
/**
* Returns a description for the action with the specified index or
* <code>null</code> if such action does not exist.
*
* @param actionIndex the zero based index to the actions
*
* @return a description for the action with the specified index or
* <code>null</code> if such action does not exist
*/
public String getAccessibleActionDescription(int actionIndex)
{
String descr = null;
if (actionIndex == 0)
{
// FIXME: Supply localized descriptions in the UIDefaults.
descr = UIManager.getString("AbstractButton.clickText");
}
return descr;
}
/**
* Performs the acccessible action with the specified index on this object.
* Since buttons have only one action by default (which is to press the
* button), this method performs a 'press button' when the specified index
* is <code>0</code> and nothing otherwise.
*
* @param actionIndex a zero based index into the actions of this button
*
* @return <code>true</code> if the specified action has been performed
* successfully, <code>false</code> otherwise
*/
public boolean doAccessibleAction(int actionIndex)
{
boolean retVal = false;
if (actionIndex == 0)
{
doClick();
retVal = true;
}
return retVal;
}
/**
* Returns the current value of this object as a number. This
* implementation returns an <code>Integer(1)</code> if the button is
* selected, <code>Integer(0)</code> if the button is not selected.
*
* @return the current value of this object as a number
*/
public Number getCurrentAccessibleValue()
{
Integer retVal;
if (isSelected())
retVal = new Integer(1);
else
retVal = new Integer(0);
return retVal;
}
/**
* Sets the current accessible value as object. If the specified number
* is 0 the button will be deselected, otherwise the button will
* be selected.
*
* @param value 0 for deselected button, other for selected button
*
* @return <code>true</code> if the value has been set, <code>false</code>
* otherwise
*/
public boolean setCurrentAccessibleValue(Number value)
{
boolean retVal = false;
if (value != null)
{
if (value.intValue() == 0)
setSelected(false);
else
setSelected(true);
retVal = true;
}
return retVal;
}
/**
* Returns the minimum accessible value for the AccessibleAbstractButton,
* which is <code>0</code>.
*
* @return the minimimum accessible value for the AccessibleAbstractButton,
* which is <code>0</code>
*/
public Number getMinimumAccessibleValue()
{
return new Integer(0);
}
/**
* Returns the maximum accessible value for the AccessibleAbstractButton,
* which is <code>1</code>.
*
* @return the maximum accessible value for the AccessibleAbstractButton,
* which is <code>1</code>
*/
public Number getMaximumAccessibleValue()
{
return new Integer(1);
}
/**
* Returns the accessible text for this AccessibleAbstractButton. This
* will be <code>null</code> if the button has a non-HTML label, otherwise
* <code>this</code>.
*
* @return the accessible text for this AccessibleAbstractButton
*/
public AccessibleText getAccessibleText()
{
AccessibleText accessibleText = null;
if (getClientProperty(BasicHTML.propertyKey) != null)
accessibleText = this;
return accessibleText;
}
/**
* Returns the index of the label's character at the specified point,
* relative to the local bounds of the button. This only works for
* HTML labels.
*
* @param p the point, relative to the buttons local bounds
*
* @return the index of the label's character at the specified point
*/
public int getIndexAtPoint(Point p)
{
int index = -1;
View view = (View) getClientProperty(BasicHTML.propertyKey);
if (view != null)
{
Rectangle shape = new Rectangle(0, 0, getWidth(), getHeight());
index = view.viewToModel(p.x, p.y, shape, new Position.Bias[1]);
}
return index;
}
/**
* Returns the bounds of the character at the specified index of the
* button's label. This will only work for HTML labels.
*
* @param i the index of the character of the label
*
* @return the bounds of the character at the specified index of the
* button's label
*/
public Rectangle getCharacterBounds(int i)
{
Rectangle rect = null;
View view = (View) getClientProperty(BasicHTML.propertyKey);
if (view != null)
{
Rectangle shape = new Rectangle(0, 0, getWidth(), getHeight());
try
{
Shape s = view.modelToView(i, shape, Position.Bias.Forward);
rect = s.getBounds();
}
catch (BadLocationException ex)
{
rect = null;
}
}
return rect;
}
/**
* Returns the number of characters in the button's label.
*
* @return the bounds of the character at the specified index of the
* button's label
*/
public int getCharCount()
{
int charCount;
View view = (View) getClientProperty(BasicHTML.propertyKey);
if (view != null)
{
charCount = view.getDocument().getLength();
}
else
{
charCount = getAccessibleName().length();
}
return charCount;
}
/**
* This always returns <code>-1</code> since there is no caret in a button.
*
* @return <code>-1</code> since there is no caret in a button
*/
public int getCaretPosition()
{
return -1;
}
/**
* Returns the character, word or sentence at the specified index. The
* <code>part</code> parameter determines what is returned, the character,
* word or sentence after the index.
*
* @param part one of {@link AccessibleText#CHARACTER},
* {@link AccessibleText#WORD} or
* {@link AccessibleText#SENTENCE}, specifying what is returned
* @param index the index
*
* @return the character, word or sentence after <code>index</code>
*/
public String getAtIndex(int part, int index)
{
String result = "";
int startIndex = -1;
int endIndex = -1;
switch(part)
{
case AccessibleText.CHARACTER:
result = String.valueOf(text.charAt(index));
break;
case AccessibleText.WORD:
startIndex = text.lastIndexOf(' ', index);
endIndex = text.indexOf(' ', startIndex + 1);
if (endIndex == -1)
endIndex = startIndex + 1;
result = text.substring(startIndex + 1, endIndex);
break;
case AccessibleText.SENTENCE:
default:
startIndex = text.lastIndexOf('.', index);
endIndex = text.indexOf('.', startIndex + 1);
if (endIndex == -1)
endIndex = startIndex + 1;
result = text.substring(startIndex + 1, endIndex);
break;
}
return result;
}
/**
* Returns the character, word or sentence after the specified index. The
* <code>part</code> parameter determines what is returned, the character,
* word or sentence after the index.
*
* @param part one of {@link AccessibleText#CHARACTER},
* {@link AccessibleText#WORD} or
* {@link AccessibleText#SENTENCE}, specifying what is returned
* @param index the index
*
* @return the character, word or sentence after <code>index</code>
*/
public String getAfterIndex(int part, int index)
{
String result = "";
int startIndex = -1;
int endIndex = -1;
switch(part)
{
case AccessibleText.CHARACTER:
result = String.valueOf(text.charAt(index + 1));
break;
case AccessibleText.WORD:
startIndex = text.indexOf(' ', index);
endIndex = text.indexOf(' ', startIndex + 1);
if (endIndex == -1)
endIndex = startIndex + 1;
result = text.substring(startIndex + 1, endIndex);
break;
case AccessibleText.SENTENCE:
default:
startIndex = text.indexOf('.', index);
endIndex = text.indexOf('.', startIndex + 1);
if (endIndex == -1)
endIndex = startIndex + 1;
result = text.substring(startIndex + 1, endIndex);
break;
}
return result;
}
/**
* Returns the character, word or sentence before the specified index. The
* <code>part</code> parameter determines what is returned, the character,
* word or sentence before the index.
*
* @param part one of {@link AccessibleText#CHARACTER},
* {@link AccessibleText#WORD} or
* {@link AccessibleText#SENTENCE}, specifying what is returned
* @param index the index
*
* @return the character, word or sentence before <code>index</code>
*/
public String getBeforeIndex(int part, int index)
{
String result = "";
int startIndex = -1;
int endIndex = -1;
switch(part)
{
case AccessibleText.CHARACTER:
result = String.valueOf(text.charAt(index - 1));
break;
case AccessibleText.WORD:
endIndex = text.lastIndexOf(' ', index);
if (endIndex == -1)
endIndex = 0;
startIndex = text.lastIndexOf(' ', endIndex - 1);
result = text.substring(startIndex + 1, endIndex);
break;
case AccessibleText.SENTENCE:
default:
endIndex = text.lastIndexOf('.', index);
if (endIndex == -1)
endIndex = 0;
startIndex = text.lastIndexOf('.', endIndex - 1);
result = text.substring(startIndex + 1, endIndex);
break;
}
return result;
}
/**
* Returns the text attribute for the character at the specified character
* index.
*
* @param i the character index
*
* @return the character attributes for the specified character or
* <code>null</code> if the character has no attributes
*/
public AttributeSet getCharacterAttribute(int i)
{
AttributeSet atts = null;
View view = (View) getClientProperty(BasicHTML.propertyKey);
if (view != null)
{
Document doc = view.getDocument();
if (doc instanceof StyledDocument)
{
StyledDocument sDoc = (StyledDocument) doc;
Element charEl = sDoc.getCharacterElement(i);
if (charEl != null)
atts = charEl.getAttributes();
}
}
return atts;
}
/**
* This always returns <code>-1</code> since
* button labels can't be selected.
*
* @return <code>-1</code>, button labels can't be selected
*/
public int getSelectionStart()
{
return -1;
}
/**
* This always returns <code>-1</code> since
* button labels can't be selected.
*
* @return <code>-1</code>, button labels can't be selected
*/
public int getSelectionEnd()
{
return -1;
}
/**
* Returns the selected text. This always returns <code>null</code> since
* button labels can't be selected.
*
* @return <code>null</code>, button labels can't be selected
*/
public String getSelectedText()
{
return null;
}
}
/**
* Creates a new AbstractButton object. Subclasses should call the following
* sequence in their constructor in order to initialize the button correctly:
* <pre>
* super();
* init(text, icon);
* </pre>