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
|
// CardLayout.java - Card-based layout engine
/* Copyright (C) 2000 Free Software Foundation
This file is part of libgcj.
This software is copyrighted work licensed under the terms of the
Libgcj License. Please consult the file "LIBGCJ_LICENSE" for
details. */
package java.awt;
import java.util.Enumeration;
import java.util.Hashtable;
import java.io.Serializable;
/** This class implements a card-based layout scheme. Each included
* component is treated as a card. Only one card can be shown at a
* time. This class includes methods for changing which card is
* shown.
*
* @verson 0.0
* @author Tom Tromey <tromey@redhat.com>
* @date December 2, 2000
*/
public class CardLayout implements LayoutManager2, Serializable
{
/** Create a new CardLayout object with both gaps zero. */
public CardLayout ()
{
this (0, 0);
}
/** Create a new CardLayout object with the specified horizontal and
* vertical gaps.
* @param hgap The horizontal gap
* @param vgap The vertical gap
*/
public CardLayout (int hgap, int vgap)
{
this.hgap = hgap;
this.vgap = vgap;
this.map = new Hashtable ();
}
/** Add a new component to the layout. The constraint must be a
* string which is used to name the component. This string can
* later be used to refer to the particular component.
* @param comp The component to add
* @param constraints The name by which the component can later be called
* @exception IllegalArgumentException If `constraints' is not a string
*/
public void addLayoutComponent (Component comp, Object constraints)
{
if (! (constraints instanceof String))
throw new IllegalArgumentException ("Object " + constraints
+ " is not a string");
map.put (constraints, comp);
}
/** Add a new component to the layout. The name can be used later
* to refer to the component.
* @param name The name by which the component can later be called
* @param comp The component to add
* @deprecated
*/
public void addLayoutComponent (String name, Component comp)
{
addLayoutComponent (comp, name);
}
/** Cause the first component in the container to be displayed.
* @param parent The parent container
*/
public void first (Container parent)
{
gotoComponent (parent, FIRST, null);
}
/** Return this layout manager's horizontal gap. */
public int getHgap ()
{
return hgap;
}
/** Return this layout manager's x alignment. This method always
* returns Component.CENTER_ALIGNMENT.
* @param parent Container using this layout manager instance
*/
public float getLayoutAlignmentX (Container parent)
{
return Component.CENTER_ALIGNMENT;
}
/** Returns this layout manager's y alignment. This method always
* returns Component.CENTER_ALIGNMENT.
* @param parent Container using this layout manager instance
*/
public float getLayoutAlignmentY (Container parent)
{
return Component.CENTER_ALIGNMENT;
}
/** Return this layout manager's vertical gap. */
public int getVgap ()
{
return vgap;
}
/** Invalidate this layout manager's state. */
public void invalidateLayout (Container target)
{
// Do nothing.
}
/** Cause the last component in the container to be displayed.
* @param parent The parent container
*/
public void last (Container parent)
{
gotoComponent (parent, LAST, null);
}
/** Lay out the container's components based on the current
* settings.
* @param parent The parent container
*/
public void layoutContainer (Container parent)
{
// FIXME: can we just use the width and height fields of parent?
// Or will that break with subclassing?
Dimension d = parent.getSize ();
Insets ins = parent.getInsets ();
int num = parent.getComponentCount ();
// This is more efficient than calling getComponents().
Component[] comps = parent.component;
for (int i = 0; i < num; ++i)
{
if (comps[i].isVisible ())
{
// Only resize the one we care about.
comps[i].setBounds (hgap + ins.left, vgap + ins.top,
d.width - 2 * hgap - ins.left - ins.right,
d.height - 2 * vgap - ins.top - ins.bottom);
break;
}
}
}
/** Get the maximum layout size of the container.
* @param target The parent container
*/
public Dimension maximumLayoutSize (Container target)
{
// The JCL says that this returns Integer.MAX_VALUE for both
// dimensions. But that just seems wrong to me.
return getSize (target, MAX);
}
/** Get the minimum layout size of the container.
* @param target The parent container
*/
public Dimension minimumLayoutSize (Container target)
{
return getSize (target, MIN);
}
/** Cause the next component in the container to be displayed.
* @param parent The parent container
*/
public void next (Container parent)
{
gotoComponent (parent, NEXT, null);
}
/** Get the preferred layout size of the container.
* @param target The parent container
*/
public Dimension preferredLayoutSize (Container parent)
{
return getSize (parent, PREF);
}
/** Cause the previous component in the container to be displayed.
* @param parent The parent container
*/
public void previous (Container parent)
{
gotoComponent (parent, PREV, null);
}
/** Remove the indicated component from this layout manager.
* @param comp The component to remove
*/
public void removeLayoutComponent (Component comp)
{
Enumeration e = map.keys ();
while (e.hasMoreElements ())
{
Object key = e.nextElement ();
if (map.get (key) == comp)
{
map.remove (key);
break;
}
}
}
/** Set this layout manager's horizontal gap.
* @param hgap The new gap
*/
public void setHgap (int hgap)
{
this.hgap = hgap;
}
/** Set this layout manager's vertical gap.
* @param vgap The new gap
*/
public void setVgap (int vgap)
{
this.vgap = vgap;
}
/** Cause the named component to be shown. If the component name is
* unknown, this method does nothing.
* @param parent The parent container
* @param name The name of the component to show
*/
public void show (Container parent, String name)
{
Object target = map.get (name);
if (target != null)
gotoComponent (parent, NONE, (Component) target);
}
public String toString ()
{
return getClass ().getName () + "[" + hgap + "," + vgap + "]";
}
// This implements first(), last(), next(), and previous().
private void gotoComponent (Container parent, int what,
Component target)
{
int num = parent.getComponentCount ();
// This is more efficient than calling getComponents().
Component[] comps = parent.component;
int choice = -1;
if (what == FIRST)
choice = 0;
else if (what == LAST)
choice = num;
else if (what >= 0)
choice = what;
for (int i = 0; i < num; ++i)
{
// If TARGET is set then we are looking for a specific
// component.
if (target != null)
{
if (target == comps[i])
choice = i;
}
if (comps[i].isVisible ())
{
if (what == NEXT)
{
choice = i + 1;
if (choice == num)
choice = num - 1;
}
else if (what == PREV)
{
choice = i - 1;
if (choice < 0)
choice = 0;
}
else
{
// Do nothing if we're already looking at the right
// component.
if (choice == i)
return;
}
comps[i].setVisible (false);
if (choice >= 0)
break;
}
}
comps[choice].setVisible (true);
}
// Compute the size according to WHAT.
private Dimension getSize (Container parent, int what)
{
int w = 0, h = 0, num = parent.getComponentCount ();
// This is more efficient than calling getComponents().
Component[] comps = parent.component;
for (int i = 0; i < num; ++i)
{
// FIXME: can we just directly read the fields in Component?
// Or will that not work with subclassing?
Dimension d;
if (what == MIN)
d = comps[i].getMinimumSize ();
else if (what == MAX)
d = comps[i].getMaximumSize ();
else
d = comps[i].getPreferredSize ();
w = Math.max (d.width, w);
h = Math.max (d.height, h);
}
Insets i = parent.getInsets ();
w += 2 * hgap + i.right + i.left;
h += 2 * vgap + i.bottom + i.top;
// Handle overflow.
if (w < 0)
w = Integer.MAX_VALUE;
if (h < 0)
h = Integer.MAX_VALUE;
return new Dimension (w, h);
}
// The gaps.
private int hgap;
private int vgap;
// This hashtable maps a name to a component.
private Hashtable map;
// These constants are used by the private gotoComponent method.
private int FIRST = 0;
private int LAST = 1;
private int NEXT = 2;
private int PREV = 3;
private int NONE = 4;
// These constants are used by the private getSize method.
private int MIN = 0;
private int MAX = 1;
private int PREF = 2;
}
|