org.openide.NotifyDescriptor Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.openide;
import java.awt.Component;
import java.awt.Dimension;
import java.awt.Window;
import java.awt.event.ItemEvent;
import java.awt.event.ItemListener;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeSupport;
import java.lang.reflect.InvocationTargetException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.atomic.AtomicBoolean;
import javax.swing.BorderFactory;
import javax.swing.ButtonGroup;
import javax.swing.GroupLayout;
import javax.swing.GroupLayout.Alignment;
import javax.swing.JCheckBox;
import javax.swing.JLabel;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.JPasswordField;
import javax.swing.JRadioButton;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;
import javax.swing.JTextField;
import javax.swing.JToggleButton;
import javax.swing.LayoutStyle;
import javax.swing.UIManager;
import javax.swing.event.DocumentEvent;
import javax.swing.event.DocumentListener;
import org.openide.awt.Mnemonics;
import org.openide.util.NbBundle;
/**
* This class provides a description of a user notification to be displayed.
* Simple example of usage:
*
* NotifyDescriptor d =
* new NotifyDescriptor.Message("Hello...", NotifyDescriptor.INFORMATION_MESSAGE);
* DialogDisplayer.getDefault().notify(d);
*
* or to get a result:
*
* NotifyDescriptor d =
* new NotifyDescriptor.Confirmation("Really do this?!", "Dialog Title",
* NotifyDescriptor.OK_CANCEL_OPTION);
* if (DialogDisplayer.getDefault().notify(d) == NotifyDescriptor.OK_OPTION) {
* // really do it...
* }
*
* @see DialogDisplayer#notify
* @author David Peroutka, Jaroslav Tulach
*/
public class NotifyDescriptor extends Object {
// Property constants
/** Name of property for the message to be displayed. */
public static final String PROP_MESSAGE = "message"; // NOI18N
/** Name of property for the type of message to use. */
public static final String PROP_MESSAGE_TYPE = "messageType"; // NOI18N
/** Name of property for the style of options available. */
public static final String PROP_OPTION_TYPE = "optionType"; // NOI18N
/** Name of property for the exact list of options. */
public static final String PROP_OPTIONS = "options"; // NOI18N
/** Name of property for the value the user selected. */
public static final String PROP_VALUE = "value"; // NOI18N
/** Name of property for the dialog title. */
public static final String PROP_TITLE = "title"; // NOI18N
/** Name of property for the detail message reported. */
public static final String PROP_DETAIL = "detail"; // NOI18N
/** Name of property for the OK button validation. */
public static final String PROP_VALID = "valid"; // NOI18N
/**
* Name of property for toggling the ESC key and title frame button closing.
* @since 7.35
*/
public static final String PROP_NO_DEFAULT_CLOSE = "noDefaultClose"; // NOI18N
/** Name of property for the error message at the bottom of the wizard.
* To set such message use {@link #createNotificationLineSupport}
*
* @since 7.10
*/
public static final String PROP_ERROR_NOTIFICATION = "errorNotification"; // NOI18N
/** Name of property for the error message at the bottom of the wizard.
* To set such message use {@link #createNotificationLineSupport}
*
* @since 7.10
*/
public static final String PROP_WARNING_NOTIFICATION = "warningNotification"; // NOI18N
/** Name of property for the error message at the bottom of the wizard.
* To set such message use {@link #createNotificationLineSupport}
*
* @since 7.10
*/
public static final String PROP_INFO_NOTIFICATION = "infoNotification"; // NOI18N
//
// Return values
//
/** Return value if YES is chosen. */
public static final Object YES_OPTION = new Integer(JOptionPane.YES_OPTION);
/** Return value if NO is chosen. */
public static final Object NO_OPTION = new Integer(JOptionPane.NO_OPTION);
/** Return value if CANCEL is chosen. */
public static final Object CANCEL_OPTION = new Integer(JOptionPane.CANCEL_OPTION);
/** Return value if OK is chosen. */
public static final Object OK_OPTION = new Integer(JOptionPane.OK_OPTION);
/** Return value if user closes the window without pressing any button. */
public static final Object CLOSED_OPTION = new Integer(JOptionPane.CLOSED_OPTION);
//
// Option types
//
/** Option type used by default. */
public static final int DEFAULT_OPTION = JOptionPane.DEFAULT_OPTION;
/** Option type used for negatable confirmations. */
public static final int YES_NO_OPTION = JOptionPane.YES_NO_OPTION;
/** Option type used for negatable and cancellable confirmations. */
public static final int YES_NO_CANCEL_OPTION = JOptionPane.YES_NO_CANCEL_OPTION;
/** Option type used for cancellable confirmations. */
public static final int OK_CANCEL_OPTION = JOptionPane.OK_CANCEL_OPTION;
//
// Message types
//
/** Message type for error messages. */
public static final int ERROR_MESSAGE = JOptionPane.ERROR_MESSAGE;
/** Message type for information messages. */
public static final int INFORMATION_MESSAGE = JOptionPane.INFORMATION_MESSAGE;
/** Message type for warning messages. */
public static final int WARNING_MESSAGE = JOptionPane.WARNING_MESSAGE;
/** Message type for questions. */
public static final int QUESTION_MESSAGE = JOptionPane.QUESTION_MESSAGE;
/** Plain message type using no icon. */
public static final int PLAIN_MESSAGE = JOptionPane.PLAIN_MESSAGE;
/** Maximum text width to which the text is wrapped */
private static final int MAXIMUM_TEXT_WIDTH = 100;
/** preferred width of text area */
private static final int SIZE_PREFERRED_WIDTH = 350;
/** preferred height of text area */
private static final int SIZE_PREFERRED_HEIGHT = 150;
private Object message;
/** The message type. */
private int messageType = PLAIN_MESSAGE;
/** The option type specifying the user-selectable options. */
private int optionType;
/** The option object specifying the user-selectable options. */
private Object[] options;
/** The option object specifying the additional user-selectable options. */
private Object[] adOptions;
/** The user's choice value object. */
private Object value;
/** The default initial value. */
private Object defaultValue;
/** The title string for the report. */
private String title;
/** Is OK button valid (enabled). */
private boolean valid = true;
private NotificationLineSupport notificationLineSupport = null;
private String infoMsg;
private String warnMsg;
private String errMsg;
/** The object specifying the detail object. */
// private Object detail;
/** Property change support. */
private PropertyChangeSupport changeSupport;
/**
* When true the dialog can be closed only using one of its closing options,
* ESC key and the close button in its frame will not work.
*/
private boolean noDefaultClose = false;
/**
* Creates a new notify descriptor with specified information to report.
*
* If optionType
is {@link #YES_NO_OPTION} or {@link #YES_NO_CANCEL_OPTION}
* and the options
parameter is null
, then the options are
* supplied by the look and feel.
*
* The messageType
parameter is primarily used to supply a
* default icon from the look and feel.
*
* @param message the object to display
* @param title the title string for the dialog
* @param optionType indicates which options are available
* @param messageType indicates what type of message should be displayed
* @param options an array of objects indicating the possible choices
* @param initialValue the object that represents the default value
*
* @see #getMessage
* @see #getMessageType
* @see #getOptions
* @see #getOptionType
* @see #getValue
*/
public NotifyDescriptor(
Object message, String title, int optionType, int messageType, Object[] options, Object initialValue
) {
checkMessageValidity(message);
this.message = message;
this.messageType = messageType;
this.options = options;
this.optionType = optionType;
this.title = title;
this.value = initialValue;
this.defaultValue = initialValue;
}
/** Method that is called before a value is returned from any of
* getter methods in this object.
*
* Allows subclasses to do some additional initialization actions.
*/
protected void initialize() {
}
/** Checks for initialization.
*/
final void getterCalled() {
boolean init = false;
synchronized (this) {
if (changeSupport == null) {
changeSupport = new java.beans.PropertyChangeSupport(this);
init = true;
}
}
if (init) {
initialize();
}
}
//
// Getters/setters for properties.
//
/**
* Return true if OK button is valid (enabled), otherwise return false.
* @see #setValid
*
* @return validity status of OK button.
*/
public final boolean isValid() {
getterCalled();
return valid;
}
/** Set validity of OK button.
* @see #isValid
* @param newValid validity status of OK button
*/
public final void setValid(boolean newValid) {
boolean oldValid = valid;
valid = newValid;
firePropertyChange(
PROP_VALID, oldValid ? Boolean.TRUE : Boolean.FALSE, newValid ? Boolean.TRUE : Boolean.FALSE
);
}
/**
* Define a descriptive message to be reported. In the most common
* usage, the message is just a String
. However, the type
* of this parameter is actually Object
. Its interpretation depends on
* its type:
*
* Object[]
- A recursively interpreted series of messages.
*
- {@link Component}
- The
Component
is displayed in the dialog.
* - {@link javax.swing.Icon}
- The
Icon
is wrapped in a {@link JLabel} and displayed in the dialog.
* - anything else
- The {@link Object#toString string representation} of the object.
*
*
* @param newMessage the Object
to report
* @see #getMessage
*/
public void setMessage(Object newMessage) {
checkMessageValidity(newMessage);
Object oldMessage = message;
if (newMessage instanceof String) {
// bugfix #25457, use JTextArea for word-wrapping
JTextArea area = new JTextArea((String) newMessage);
area.setBackground(UIManager.getColor("Label.background")); // NOI18N
area.setBorder(BorderFactory.createEmptyBorder());
area.setLineWrap(true);
area.setWrapStyleWord(true);
area.setEditable(false);
area.setFocusable(true);
area.getAccessibleContext().setAccessibleName(NbBundle.getMessage(NotifyDescriptor.class, "ACN_NotifyDescriptor_MessageJTextArea")); // NOI18N
area.getAccessibleContext().setAccessibleDescription(NbBundle.getMessage(NotifyDescriptor.class, "ACD_NotifyDescriptor_MessageJTextArea")); // NOI18N
JScrollPane sp = new JScrollPane(area);
sp.setBorder(BorderFactory.createEmptyBorder());
sp.setPreferredSize(new Dimension(SIZE_PREFERRED_WIDTH, SIZE_PREFERRED_HEIGHT));
newMessage = sp;
}
message = newMessage;
firePropertyChange(PROP_MESSAGE, oldMessage, newMessage);
}
private void checkMessageValidity(Object message) {
if (message instanceof Window) {
// See https://netbeans.org/bugzilla/show_bug.cgi?id=267337
throw new IllegalArgumentException("The message must not be a window. message = "+message);
}
}
/**
* Get the message object.
* @see #setMessage
*
* @return the Object
that is to be reported
*/
public Object getMessage() {
getterCalled();
return message;
}
/**
* Define the style of the message. The look and feel manager may lay out
* the dialog differently depending on this value, and will often provide
* a default icon. The possible values are:
*
* - {@link #ERROR_MESSAGE}
*
- {@link #INFORMATION_MESSAGE}
*
- {@link #WARNING_MESSAGE}
*
- {@link #QUESTION_MESSAGE}
*
- {@link #PLAIN_MESSAGE}
*
*
* @param newType the kind of message
*
* @see #getMessageType
*/
public void setMessageType(int newType) {
if (
(newType != ERROR_MESSAGE) && (newType != INFORMATION_MESSAGE) && (newType != WARNING_MESSAGE) &&
(newType != QUESTION_MESSAGE) && (newType != PLAIN_MESSAGE)
) {
throw new IllegalArgumentException(
"Message type must be one of the following:" // NOI18N
+" ERROR_MESSAGE, INFORMATION_MESSAGE," // NOI18N
+" WARNING_MESSAGE, QUESTION_MESSAGE or PLAIN_MESSAGE." // NOI18N
);
}
int oldType = messageType;
messageType = newType;
firePropertyChange(PROP_MESSAGE_TYPE, new Integer(oldType), new Integer(messageType));
}
/**
* Get the message type.
*
* @return the message type
*
* @see #setMessageType
*/
public int getMessageType() {
getterCalled();
return messageType;
}
/**
* Define the set of options. The option type is used by the look and
* feel to determine what options to show (unless explicit options are supplied):
*
* - {@link #DEFAULT_OPTION}
*
- {@link #YES_NO_OPTION}
*
- {@link #YES_NO_CANCEL_OPTION}
*
- {@link #OK_CANCEL_OPTION}
*
*
* @param newType the options the look and feel is to display
*
* @see #getOptionType
* @see #setOptions
*/
public void setOptionType(int newType) {
if (
(newType != DEFAULT_OPTION) && (newType != YES_NO_OPTION) && (newType != YES_NO_CANCEL_OPTION) &&
(newType != OK_CANCEL_OPTION)
) {
throw new IllegalArgumentException(
"Option type must be one of the following:" // NOI18N
+" DEFAULT_OPTION, YES_NO_OPTION," // NOI18N
+" YES_NO_CANCEL_OPTION or OK_CANCEL_OPTION." // NOI18N
);
}
int oldType = optionType;
optionType = newType;
firePropertyChange(PROP_OPTION_TYPE, new Integer(oldType), new Integer(optionType));
}
/**
* Get the type of options that are to be displayed.
*
* @return the option type
*
* @see #setOptionType
*/
public int getOptionType() {
getterCalled();
return optionType;
}
/**
* Define an explicit description of the set of user-selectable options.
* The usual value for the options parameter is an array of
* String
s. But the parameter type is an array of Object
s. Its
* interpretation depends on its type:
*
* - {@link Component}
- The component is added to the button row directly.
*
- {@link javax.swing.Icon}
- A {@link javax.swing.JButton} is created with this icon as its label.
*
- anything else
- The
Object
is {@link Object#toString converted} to a string and the result is used to
* label a JButton
.
*
*
* @param newOptions an array of user-selectable options
*
* @see #getOptions
*/
public void setOptions(Object[] newOptions) {
Object[] oldOptions = options;
options = newOptions;
firePropertyChange(PROP_OPTIONS, oldOptions, newOptions);
}
/**
* Get the explicit choices the user can make.
* @return the array of Object
s that give the user's choices
*
* @see #setOptions
*/
public Object[] getOptions() {
getterCalled();
if (options != null) {
return options.clone ();
}
return options;
}
/**
* Define an explicit description of the set of additional user-selectable options.
* Additional options are supposed to be used for help button, etc.
*
* The usual value for the options parameter is an array of
* String
s. But the parameter type is an array of Object
s. Its
* interpretation depends on its type:
*
* - {@link Component}
- The component is added to the button row directly.
*
- {@link javax.swing.Icon}
- A {@link javax.swing.JButton} is created with this icon as its label.
*
- anything else
- The
Object
is {@link Object#toString converted} to a string and the result is used to
* label a JButton
.
*
*
* @param newOptions an array of user-selectable options
*
* @see #getOptions
*/
public void setAdditionalOptions(Object[] newOptions) {
Object[] oldOptions = adOptions;
adOptions = newOptions;
firePropertyChange(PROP_OPTIONS, oldOptions, newOptions);
}
/**
* Get the explicit additional choices the user can make.
* @return the array of Object
s that give the user's choices
*
* @see #setOptions
*/
public Object[] getAdditionalOptions() {
getterCalled();
if (adOptions != null) {
return adOptions.clone ();
}
return null;
}
/**
* Sets the value w/o firing property change. The caller this is responsible
* to notify this change.
*/
void setValueWithoutPCH(Object newValue) {
value = newValue;
}
/**
* Set the value the user has chosen and fires appropriate property change.
* You probably do not want to call this yourself, of course.
*
* @param newValue the chosen value
*
* @see #getValue
*/
public void setValue(Object newValue) {
Object oldValue = value;
setValueWithoutPCH(newValue);
firePropertyChange(PROP_VALUE, oldValue, newValue);
}
/**
* Get the value the user has selected.
*
* @return an Object
indicating the option selected by the user
*
* @see #setValue
*/
public Object getValue() {
getterCalled();
return value;
}
/**
* Get the default value of descriptor.
*
* @return an Object
that represents the default value
* @since 5.9
*/
public Object getDefaultValue() {
return defaultValue;
}
/**
* Set the title string for this report description.
*
* @param newTitle the title of this description
*
* @see #getTitle
*/
public void setTitle(String newTitle) {
Object oldTitle = title;
title = newTitle;
firePropertyChange(PROP_TITLE, oldTitle, newTitle);
}
/**
* Get the title string for this report description.
*
* @return the title of this description
*
* @see #setTitle
*/
public String getTitle() {
getterCalled();
return title;
}
/** Create {@link NotificationLineSupport} if you want to notify users
* using info/warning/error messages in designed line at the bottom
* of your dialog. These message will be labelled with appropriate icons.
*
* Note: Call this method before you call {@link DialogDisplayer#createDialog}
*
* @return NotificationLineSupport
* @since 7.10
*/
public final NotificationLineSupport createNotificationLineSupport() {
notificationLineSupport = new NotificationLineSupport (this);
return notificationLineSupport;
}
/** Returns NotificationLineSupport if it was created or null
if doesn't.
*
* Note: NotificationLineSupport will not be created by default, API client
* has to create this support purposely with the exception {@link WizardDescriptor}
* which has such capability longer.
*
* @see #createNotificationLineSupport()
* @return NotificationLineSupport or null if was not created yet
* @since 7.10
*/
public final NotificationLineSupport getNotificationLineSupport() {
return notificationLineSupport;
}
void setInformationMessage (String msg) {
if (notificationLineSupport == null) {
throw new IllegalStateException ("NotificationLineSupport wasn't created yet.");
}
infoMsg = msg;
warnMsg = null;
errMsg = null;
firePropertyChange (PROP_INFO_NOTIFICATION, null, msg);
}
void setWarningMessage (String msg) {
if (notificationLineSupport == null) {
throw new IllegalStateException ("NotificationLineSupport wasn't created yet.");
}
infoMsg = null;
warnMsg = msg;
errMsg = null;
firePropertyChange (PROP_WARNING_NOTIFICATION, null, msg);
}
void setErrorMessage (String msg) {
if (notificationLineSupport == null) {
throw new IllegalStateException ("NotificationLineSupport wasn't created yet.");
}
infoMsg = null;
warnMsg = null;
errMsg = msg;
firePropertyChange (PROP_ERROR_NOTIFICATION, null, msg);
}
String getInformationMessage () {
if (notificationLineSupport == null) {
throw new IllegalStateException ("NotificationLineSupport wasn't created yet.");
}
return infoMsg;
}
String getWarningMessage () {
if (notificationLineSupport == null) {
throw new IllegalStateException ("NotificationLineSupport wasn't created yet.");
}
return warnMsg;
}
String getErrorMessage () {
if (notificationLineSupport == null) {
throw new IllegalStateException ("NotificationLineSupport wasn't created yet.");
}
return errMsg;
}
void clearMessages () {
if (notificationLineSupport == null) {
throw new IllegalStateException ("NotificationLineSupport wasn't created yet.");
}
infoMsg = null;
warnMsg = null;
errMsg = null;
firePropertyChange (PROP_INFO_NOTIFICATION, null, null);
}
/**
* Define a detail message to be reported. In the most common usage,
* this message is just a String
. However, the type of this
* parameter is actually Object
. Its interpretation depends on its type:
*
* Object[]
- A recursively interpreted series of messages.
*
Throwable
- A stack trace is displayed.
*
- anything else
- The {@link Object#toString string representation} of the object is used.
*
*
* @param newDetail the detail object of this description
*
* @see #getDetail
*
public void setDetail(Object newDetail) {
Object oldDetail = detail;
detail = newDetail;
firePropertyChange(PROP_DETAIL, oldDetail, newDetail);
}
/**
* Get the detail object for this description.
*
* @return details of this description
*
* @see #setTitle
*
public Object getDetail() {
return detail;
}
*/
//
// Support for reporting bound property changes.
//
/**
* Add a {@link PropertyChangeListener} to the listener list.
*
* @param listener the PropertyChangeListener
to be added
*/
public void addPropertyChangeListener(PropertyChangeListener listener) {
getterCalled();
changeSupport.addPropertyChangeListener(listener);
}
/**
* Remove a {@link PropertyChangeListener} from the listener list.
*
* @param listener the PropertyChangeListener
to be removed
*/
public void removePropertyChangeListener(PropertyChangeListener listener) {
if (changeSupport != null) {
changeSupport.removePropertyChangeListener(listener);
}
}
/**
* Fire a {@link PropertyChangeEvent} to each listener.
*
* @param propertyName the programmatic name of the property that was changed
* @param oldValue the old value of the property
* @param newValue the new value of the property
*/
protected void firePropertyChange(String propertyName, Object oldValue, Object newValue) {
if (changeSupport != null) {
changeSupport.firePropertyChange(propertyName, oldValue, newValue);
}
}
/**
* Disable or enabled the closing of dialog window using ESC and the close button
* in dialog's title bar.
* @param noDefaultClose True to disabled ESC closing and to disable the close
* button in dialog's title bar.
* @since 7.35
* @see javax.swing.JDialog#DO_NOTHING_ON_CLOSE
*/
public void setNoDefaultClose( boolean noDefaultClose ) {
boolean oldValue = this.noDefaultClose;
this.noDefaultClose = noDefaultClose;
firePropertyChange(PROP_NO_DEFAULT_CLOSE, oldValue, noDefaultClose);
}
/**
* @return True if ESC key closing and close button in dialog's title bar are disabled,
* false otherwise.
* @since 7.35
*/
public boolean isNoDefaultClose() {
return noDefaultClose;
}
/**
* Get the title to use for the indicated type.
* @param messageType the type of message
* @return the title to use
*/
protected static String getTitleForType(int messageType) {
switch (messageType) {
case ERROR_MESSAGE:
return NbBundle.getMessage(NotifyDescriptor.class, "NTF_ErrorTitle");
case WARNING_MESSAGE:
return NbBundle.getMessage(NotifyDescriptor.class, "NTF_WarningTitle");
case QUESTION_MESSAGE:
return NbBundle.getMessage(NotifyDescriptor.class, "NTF_QuestionTitle");
case INFORMATION_MESSAGE:
return NbBundle.getMessage(NotifyDescriptor.class, "NTF_InformationTitle");
case PLAIN_MESSAGE:
return NbBundle.getMessage(NotifyDescriptor.class, "NTF_PlainTitle");
}
return ""; // NOI18N
}
/**
* Provides information about the results of a command. Offers
* no user choices; the user can only acknowledge the message.
*/
public static class Message extends NotifyDescriptor {
/**
* Create an informational report about the results of a command.
*
* @param message the message object
* @see NotifyDescriptor#NotifyDescriptor
*/
public Message(Object message) {
this(message, INFORMATION_MESSAGE);
}
/**
* Create a report about the results of a command.
*
* @param message the message object
* @param messageType the type of message to be displayed
* @see NotifyDescriptor#NotifyDescriptor
*/
public Message(Object message, int messageType) {
super(
message, NotifyDescriptor.getTitleForType(messageType), DEFAULT_OPTION, messageType,
new Object[] { OK_OPTION }, OK_OPTION
);
}
}
/**
* Provides a description of a possible action and requests confirmation from the user before proceeding.
* This should be used to alert the user to a condition
* or situation that requires the user's decision before proceeding, such
* as an impending action with potentially destructive or irreversible
* consequences. It is conventionally in the form of a question: for example,
* "Save changes to TestForm?"
*/
public static class Confirmation extends NotifyDescriptor {
/**
* Create a yes/no/cancel question with default title.
*
* @param message the message object
* @see NotifyDescriptor#NotifyDescriptor
*/
public Confirmation(Object message) {
this(message, YES_NO_CANCEL_OPTION);
}
/**
* Create a yes/no/cancel question.
*
* @param message the message object
* @param title the dialog title
* @see NotifyDescriptor#NotifyDescriptor
*/
public Confirmation(Object message, String title) {
this(message, title, YES_NO_CANCEL_OPTION);
}
/**
* Create a question with default title.
*
* @param message the message object
* @param optionType the type of options to display to the user
* @see NotifyDescriptor#NotifyDescriptor
*/
public Confirmation(Object message, int optionType) {
this(message, optionType, QUESTION_MESSAGE);
}
/**
* Create a question.
*
* @param message the message object
* @param title the dialog title
* @param optionType the type of options to display to the user
* @see NotifyDescriptor#NotifyDescriptor
*/
public Confirmation(Object message, String title, int optionType) {
this(message, title, optionType, QUESTION_MESSAGE);
}
/**
* Create a confirmation with default title.
*
* @param message the message object
* @param optionType the type of options to display to the user
* @param messageType the type of message to use
* @see NotifyDescriptor#NotifyDescriptor
*/
public Confirmation(Object message, int optionType, int messageType) {
super(
message, NotifyDescriptor.getTitleForType(messageType), optionType, messageType,
(optionType == DEFAULT_OPTION) ? new Object[] { OK_OPTION } : null, OK_OPTION
);
}
/**
* Create a confirmation.
*
* @param message the message object
* @param title the dialog title
* @param optionType the type of options to display to the user
* @param messageType the type of message to use
* @see NotifyDescriptor#NotifyDescriptor
*/
public Confirmation(Object message, String title, int optionType, int messageType) {
super(
message, title, optionType, messageType,
(optionType == DEFAULT_OPTION) ? new Object[] { OK_OPTION } : null, OK_OPTION
);
}
}
/**
* Provides a description of an exception that occurred during
* execution of the IDE. The best is to use this class together with
* {@link DialogDisplayer#notifyLater} as that allows an exception
* to be notified from any thread.
* @deprecated No longer used. Try {@link java.util.logging} or {@link org.openide.util.Exceptions}.
*/
@Deprecated
public static final class Exception extends Confirmation {
static final long serialVersionUID = -3387516993124229948L;
/**
* Create an exception report with default message.
*
* @param detail the detail object
*/
public Exception(Throwable detail) {
this(detail, detail.getMessage());
// handle InvocationTargetExceptions
if (detail instanceof InvocationTargetException) {
Throwable target = ((InvocationTargetException) detail).getTargetException();
this.setMessage(target);
Object msgObj = this.getMessage();
if ((msgObj == null) || "".equals(msgObj)) { // NOI18N
String msg = target.getMessage();
msg = org.openide.util.Utilities.wrapString(
msg, MAXIMUM_TEXT_WIDTH, java.text.BreakIterator.getCharacterInstance(), false
);
this.setMessage(msg);
}
}
Object obj = this.getMessage();
// emphasize user-non-friendly exceptions
if ((obj == null) || "".equals(obj)) { // NOI18N
this.setMessage(
NbBundle.getMessage(
NotifyDescriptor.class, "NTF_ExceptionalException", detail.getClass().getName(),
System.getProperty("netbeans.user") + java.io.File.separator + "system"
)
); // NOI18N
this.setTitle(NbBundle.getMessage(NotifyDescriptor.class, "NTF_ExceptionalExceptionTitle"));
}
}
/**
* Create an exception report.
*
* @param detail the detail object
* @param message the message object
*/
public Exception(Throwable detail, Object message) {
super(message, DEFAULT_OPTION, ERROR_MESSAGE);
// customize descriptor
// this.setDetail(detail);
this.setTitle(NbBundle.getMessage(NotifyDescriptor.class, "NTF_ExceptionTitle"));
}
}
/**
* Notification providing a password input.
*/
public static final class PasswordLine extends InputLine {
/** Construct dialog with the specified title and label text.
* @param text label text
* @param title title of the dialog
*/
public PasswordLine(String text, String title) {
super(text, title);
}
@Override
JTextField createTextField() {
return new JPasswordField(25);
}
}
/** Notification providing for a line of text input.
* @author Dafe Simonek
*/
public static class InputLine extends NotifyDescriptor {
/**
* Property whose value is the input text, an event is fired
* when the input text's value changes,
* if enabled using {@link #setInputTextEventsEnabled(boolean) }.
*
* @since 7.70
*/
public static final String PROP_INPUT_TEXT = "inputText"; // NOI18N
/**
* The text field used to enter the input.
*/
protected JTextField textField;
private final AtomicBoolean inputTextEventSuppressed = new AtomicBoolean();
private final AtomicBoolean inputTextEventEnabled = new AtomicBoolean();
/** Construct dialog with the specified title and label text.
* @param text label text
* @param title title of the dialog
*/
public InputLine(final String text, final String title) {
this(text, title, OK_CANCEL_OPTION, PLAIN_MESSAGE);
}
/** Construct dialog with the specified title, label text, option and
* message types.
* @param text label text
* @param title title of the dialog
* @param optionType option type (ok, cancel, ...)
* @param messageType message type (question, ...)
*/
public InputLine(final String text, final String title, final int optionType, final int messageType) {
super(null, title, optionType, messageType, null, null);
super.setMessage(createDesign(text));
}
/**
* Get the text which the user typed into the input line.
* @return the text entered by the user
*/
public String getInputText() {
return textField.getText();
}
/**
* Set the text on the input line.
* @param text the new text
*/
public void setInputText(final String text) {
inputTextEventSuppressed.set(true);
try {
textField.setText(text);
textField.selectAll();
if (inputTextEventEnabled.get()) {
firePropertyChange(PROP_INPUT_TEXT, null, null);
}
} finally {
inputTextEventSuppressed.set(false);
}
}
/**
* Enable the {@link #PROP_INPUT_TEXT} when the input text is changed.
*
* @param value {@code true} if the {@code PROP_INPUT_TEXT} even should be fired
* when the input text is modified, {@code false} otherwise.
*
* @since 7.70
*/
public void setInputTextEventEnabled(boolean value) {
inputTextEventEnabled.set(value);
}
/** Make a component representing the input line.
* @param text a label for the input line
* @return the component
*/
protected Component createDesign(final String text) {
JPanel panel = new JPanel();
panel.setOpaque (false);
JLabel textLabel = new JLabel();
Mnemonics.setLocalizedText(textLabel, text);
boolean longText = text.length () > 80;
textField = createTextField();
textField.getDocument().addDocumentListener(new DocumentListener() {
@Override
public void insertUpdate(DocumentEvent e) {
if (inputTextEventEnabled.get() && !inputTextEventSuppressed.get()) {
firePropertyChange(PROP_INPUT_TEXT, null, null);
}
}
@Override
public void removeUpdate(DocumentEvent e) {
if (inputTextEventEnabled.get() && !inputTextEventSuppressed.get()) {
firePropertyChange(PROP_INPUT_TEXT, null, null);
}
}
@Override
public void changedUpdate(DocumentEvent e) {}
});
textLabel.setLabelFor(textField);
textField.requestFocus();
GroupLayout layout = new GroupLayout(panel);
panel.setLayout(layout);
if (longText) {
layout.setHorizontalGroup(
layout.createParallelGroup(Alignment.LEADING)
.addGroup(layout.createSequentialGroup()
.addContainerGap()
.addGroup(layout.createParallelGroup(Alignment.LEADING)
.addGroup(layout.createSequentialGroup()
.addComponent(textLabel, GroupLayout.DEFAULT_SIZE, GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE)
.addGap(32, 32, 32))
.addComponent(textField))
.addContainerGap())
);
} else {
layout.setHorizontalGroup(
layout.createParallelGroup(Alignment.LEADING)
.addGroup(layout.createSequentialGroup()
.addContainerGap()
.addComponent(textLabel)
.addPreferredGap(LayoutStyle.ComponentPlacement.RELATED)
.addComponent(textField, GroupLayout.DEFAULT_SIZE, 207, Short.MAX_VALUE)
.addContainerGap())
);
}
if (longText) {
layout.setVerticalGroup(
layout.createParallelGroup(Alignment.LEADING)
.addGroup(layout.createSequentialGroup()
.addContainerGap()
.addComponent(textLabel)
.addPreferredGap(LayoutStyle.ComponentPlacement.RELATED)
.addComponent(textField, GroupLayout.PREFERRED_SIZE, GroupLayout.DEFAULT_SIZE, GroupLayout.PREFERRED_SIZE)
.addContainerGap(GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE))
);
} else {
layout.setVerticalGroup(
layout.createParallelGroup(Alignment.LEADING)
.addGroup(layout.createSequentialGroup()
.addContainerGap()
.addGroup(layout.createParallelGroup(Alignment.BASELINE)
.addComponent(textLabel)
.addComponent(textField, GroupLayout.PREFERRED_SIZE, GroupLayout.DEFAULT_SIZE, GroupLayout.PREFERRED_SIZE))
.addContainerGap(GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE))
);
}
javax.swing.KeyStroke enter = javax.swing.KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_ENTER, 0);
javax.swing.text.Keymap map = textField.getKeymap();
map.removeKeyStrokeBinding(enter);
/*
textField.addActionListener (new java.awt.event.ActionListener () {
public void actionPerformed (java.awt.event.ActionEvent evt) {
System.out.println("action: " + evt);
InputLine.this.setValue (OK_OPTION);
}
}
);
*/
panel.getAccessibleContext().setAccessibleDescription(
NbBundle.getMessage(NotifyDescriptor.class, "ACSD_InputPanel")
);
textField.getAccessibleContext().setAccessibleDescription(
NbBundle.getMessage(NotifyDescriptor.class, "ACSD_InputField")
);
return panel;
}
JTextField createTextField() {
return new JTextField(25);
}
}
// end of InputLine
/** Notification providing a selection list allowing multiple selections.
* @since 7.60
*/
public static final class QuickPick extends NotifyDescriptor {
private final String text;
private final List- items;
private final boolean multipleSelection;
/**
* Construct dialog with the specified title and label text.
* @param text label text
* @param title title of the dialog
* @param items a list of items
* @param multipleSelection true if multiple selection allowed
* @since 7.60
*/
public QuickPick(final String text, final String title, final List
- items, final boolean multipleSelection) {
super(null, title, OK_CANCEL_OPTION, PLAIN_MESSAGE, null, null);
this.text = text;
this.items = items;
this.multipleSelection = multipleSelection;
}
@Override
public Object getMessage() {
Object msg = super.getMessage();
if (msg != null) {
return msg;
}
JPanel panel = new JPanel();
panel.setOpaque (false);
javax.swing.GroupLayout layout = new javax.swing.GroupLayout(panel);
panel.setLayout(layout);
JLabel label = new JLabel();
Mnemonics.setLocalizedText(label, text);
GroupLayout.ParallelGroup hGroup = layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
.addComponent(label);
GroupLayout.SequentialGroup vGroup = layout.createSequentialGroup()
.addContainerGap()
.addComponent(label);
final Map
btn2items = new LinkedHashMap();
ItemListener listener = new ItemListener() {
@Override
public void itemStateChanged(ItemEvent e) {
JToggleButton btn = (JToggleButton) e.getItemSelectable();
Item item = btn2items.get(btn);
if (item != null) {
item.setSelected(btn.isSelected());
}
}
};
ButtonGroup buttonGroup = this.multipleSelection ? null : new ButtonGroup();
for (Item item : items) {
JToggleButton btn;
if (buttonGroup != null) {
btn = new JRadioButton();
buttonGroup.add(btn);
} else {
btn = new JCheckBox();
}
btn.setText(item.getLabel());
btn.setToolTipText(item.getDescription());
btn.setSelected(item.isSelected());
hGroup.addComponent(btn);
vGroup.addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.RELATED)
.addComponent(btn);
btn.addItemListener(listener);
btn2items.put(btn, item);
}
layout.setHorizontalGroup(layout.createParallelGroup(Alignment.LEADING)
.addGroup(layout.createSequentialGroup()
.addContainerGap()
.addGroup(hGroup)
.addContainerGap())
);
layout.setVerticalGroup(layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
.addGroup(vGroup.addContainerGap())
);
this.setMessage(panel);
return panel;
}
/**
* Get label text.
* @return label text
* @since 7.63
*/
public String getLabel() {
return text;
}
/**
* Get the list of selection items.
* @return unmodifiable list of items
* @since 7.60
*/
public List- getItems() {
return Collections.unmodifiableList(items);
}
/**
* Check if the picker accepts multiple selections.
* @return true if multiple selection allowed
* @since 7.60
*/
public boolean isMultipleSelection() {
return multipleSelection;
}
/**
* Item that can be selected from a list of items.
* @since 7.60
*/
public static final class Item {
private final String label;
private final String description;
private boolean selected;
/**
* Creates item that can be selected from a list of items.
* @param label item's label
* @param description item's description
* @since 7.60
*/
public Item(String label, String description) {
this.label = label;
this.description = description;
}
/**
* Item's label.
* @since 7.60
*/
public String getLabel() {
return label;
}
/**
* Item's description.
* @since 7.60
*/
public String getDescription() {
return description;
}
/**
* Flag indicating if this item is selected.
* @since 7.60
*/
public boolean isSelected() {
return selected;
}
/**
* Marks item as selected.
* @since 7.60
*/
public void setSelected(boolean selected) {
this.selected = selected;
}
}
}
/** Notification providing a composed input of multiple chained selection lists and/or input lines.
* @since 7.63
*/
public static final class ComposedInput extends NotifyDescriptor {
/** Name of property for the estimated number of chained inputs. */
public static final String PROP_ESTIMATED_NUMBER_OF_INPUTS = "estimatedNumberOfInputs"; // NOI18N
private final List
inputs = new ArrayList<>();
private final Callback callback;
private int estimatedNumberOfInputs;
/** Construct dialog with the specified title and chained inputs.
* @param title title of the dialog
* @param estimatedNumberOfInputs estimated number of chained inputs
* @param callback callback used to create chained inputs
* @since 7.63
*/
public ComposedInput(final String title, final int estimatedNumberOfInputs, final Callback callback) {
super(null, title, OK_CANCEL_OPTION, PLAIN_MESSAGE, null, null);
this.callback = callback;
this.estimatedNumberOfInputs = estimatedNumberOfInputs;
}
/**
* Estimated number of chained inputs.
* @since 7.63
*/
public int getEstimatedNumberOfInputs() {
return estimatedNumberOfInputs;
}
/**
* Set estimated number of chained inputs.
* @param estimatedNumberOfInputs estimated number of chained inputs
* @since 7.63
*/
public void setEstimatedNumberOfInputs(int estimatedNumberOfInputs) {
int oldNumber = this.estimatedNumberOfInputs;
this.estimatedNumberOfInputs = estimatedNumberOfInputs;
firePropertyChange(PROP_ESTIMATED_NUMBER_OF_INPUTS, oldNumber, estimatedNumberOfInputs);
}
/**
* Lazy creates chained input of the given ordinal.
* @param number input number from interval <1, totalInputs+1>
* @return nested selection list, input line, or null
* @since 7.63
*/
public NotifyDescriptor createInput(int number) {
NotifyDescriptor step = callback.createInput(this, number);
if (step != null) {
if (number - 1 < inputs.size()) {
inputs.set(number - 1, step);
} else if (number - 1 == inputs.size()) {
inputs.add(step);
} else {
return null;
}
if (number >= estimatedNumberOfInputs) {
estimatedNumberOfInputs = number;
}
}
return step;
}
/**
* Returns all created chained inputs.
* @since 7.63
*/
public NotifyDescriptor[] getInputs() {
return inputs.toArray(new NotifyDescriptor[0]);
}
@Override
public Object getMessage() {
Object msg = super.getMessage();
if (msg != null) {
return msg;
}
JPanel panel = new JPanel();
panel.setOpaque (false);
panel.setLayout(new java.awt.GridBagLayout());
NotifyDescriptor input;
int i = 0;
java.awt.GridBagConstraints gridBagConstraints = null;
while ((input = createInput(++i)) != null) {
gridBagConstraints = new java.awt.GridBagConstraints();
gridBagConstraints.gridx = 0;
gridBagConstraints.gridy = i - 1;
gridBagConstraints.gridwidth = java.awt.GridBagConstraints.RELATIVE;
gridBagConstraints.gridheight = java.awt.GridBagConstraints.RELATIVE;
gridBagConstraints.anchor = java.awt.GridBagConstraints.FIRST_LINE_START;
panel.add((JPanel)input.getMessage(), gridBagConstraints);
}
if (gridBagConstraints != null) {
gridBagConstraints.weighty = 1.0;
}
this.setMessage(panel);
return panel;
}
/**
* Callback used to lazy create chained inputs.
* @since 7.63
*/
public static interface Callback {
/**
* Lazy creates chained input of the given ordinal.
* @param input {@link ComposedInput} instance
* @param number input ordinal from interval <1, totalInputs+1>
* @return selection list, input line, or null
* @since 7.63
*/
public NotifyDescriptor createInput(ComposedInput input, int number);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy