Правильный способ привязки группы переключателей с использованием привязки данных JFace
Мне было интересно, может ли кто-нибудь объяснить мне, как я мог бы привязать группу переключателей к логической переменной в модели, используя привязку данных JFace.
Позвольте мне сначала объяснить ситуацию: я создал класс, представляющий группу кнопок SWT (со стилем, установленным в "SWT.RADIO" ), который состоит из трех элементов: метка с вопросом и две кнопки, один для ответа "да" и один для "нет". Я хотел бы создать привязку к логической переменной в модели таким образом, что, когда пользователь выбирает "да", кнопка boolean имеет значение true, а когда он выбирает кнопку "no", логическое значение установите значение false.
Здесь код моего класса:
private class YesOrNoRadioButtonGroup {
private static final String YES = "yes";
private static final String NO = "no";
private Button m_yesButton;
private Button m_noButton;
public YesOrNoRadioButtonGroup(final Composite p_parent,
final String p_questionText,
final IObservableValue p_modelProperty
final DataBindingContext p_dbContext)
{
Composite radioButtonGroupContainer = new Composite(p_parent, SWT.NONE);
radioButtonGroupContainer.setLayout(new GridLayout());
Label question = new Label(radioButtonGroupContainer, SWT.NONE);
question.setText(p_questionText);
m_yesButton = new Button(radioButtonGroupContainer, SWT.RADIO);
m_yesButton.setText(YES);
m_noButton = new Button(radioButtonGroupContainer, SWT.RADIO);
m_noButton.setText(NO);
m_noButton.setSelection(true);
Listener yesOrNoRadioGroupListener = new Listener() {
public void handleEvent(Event p_event) {
Button button = (Button) p_event.widget;
if (m_yesButton.equals(button)) {
m_yesButton.setSelection(true);
m_noButton.setSelection(false);
}
else {
m_yesButton.setSelection(false);
m_noButton.setSelection(true);
}
}
};
m_yesButton.addListener(SWT.Selection, yesOrNoRadioGroupListener);
m_noButton.addListener(SWT.Selection, yesOrNoRadioGroupListener);
p_dbContext.bindValue(SWTObservables.observeSelection(this.getYesButton()),
p_modelProperty, null, null);
}
public Button getYesButton() {
return m_yesButton;
}
public Button getNoButton() {
return m_noButton;
}
}
Теперь, как вы можете видеть, я привязываю свою кнопку "да" к логическому. В частности, значение будет привязано к событию SWT.selection. Это, кажется, единственное допустимое событие для привязки переключателя. Однако из-за этого, когда выбрана кнопка "Нет", значение логического значения остается неизменным (так как не было запущено событие SWT.selection на кнопке "да" ).
Что я могу сделать, чтобы сделать эту работу так, как она должна была бы, т.е. Иметь возможность изменять значение булева модели в зависимости от того, какую из кнопок выбирает пользователь?
Я пропустил что-то очевидное здесь?
Спасибо!
Ответы
Ответ 1
Похоже, что привязывает POJO к свойству.
Это означает, что у вас должен быть один объект с двумя кнопками, реализующий IObservable, а затем привяжите его к вашему свойству.
Как вы отметили в комментариях, вы должны расширить AbstractObservable.
У вас есть пример такого расширения здесь с этим испытанием класса о наблюдаемом списке (не обязательно, что вам нужно, но он может дать вам идеи об использовании Observable, когда дело доходит до обнаружения изменения при уведомлении)
Ответ 2
Думаю, я нашел для этого наиболее подходящую реализацию.
Вам нужно использовать org.eclipse.core.databinding.observable.value.SelectObservableValue.
вот код
class YesNoModel{
public static enum RESPONSE {YES,NO};
private RESPONSE value;
public RESPONSE getValue(){return value;}
public void setValue(RESPONSE value){this.value = value;}
}
class YouGraphicalClass {
YesNoModel yesNomodel = new YesNoModel();
void iniDataBinding(){
IObservableValue yesBtnSelection = SWTObservables.observeSelection(this.getYesButton());
IObservableValue noBtnSelection = SWTObservables.observeSelection(this.getNoButton());
SelectObservableValue featureRepoPolicyObservable = new SelectObservableValue(YesNoModel.RESPONSE.class);
featureRepoPolicyObservable.addOption(RESPONSE.YES, yesBtnSelection);
featureRepoPolicyObservable.addOption(RESPONSE.NO, noBtnSelection);
p_dbContext.bindValue(featureRepoPolicyObservable,
PojoObservables.observeValue(yesNomodel, "value"));
}
Это работает, конечно, для всех видов перечислений или других типов выбираемых значений. Вы могли бы, конечно, использовать вас String YES и NO, но я предпочитаю перечисления
Ответ 3
Радиокнопки внутри одного и того же композита будут действовать как группа, поэтому только одна кнопка будет иметь истинное значение, а все остальные должны иметь ложное значение. Наряду с этим, наблюдательный выбор на кнопке "Да" должен быть достаточным для отражения выбранного значения "да" или "нет" на стороне модели.
Или, другими словами, модифицированный конструктор будет выглядеть следующим образом.
public YesOrNoRadioButtonGroup(final Composite p_parent,
final String p_questionText,
final IObservableValue p_modelProperty,
final DataBindingContext p_dbContext)
{
Composite radioButtonGroupContainer = new Composite(p_parent, SWT.NONE);
radioButtonGroupContainer.setLayout(new GridLayout());
Label question = new Label(radioButtonGroupContainer, SWT.NONE);
question.setText(p_questionText);
m_yesButton = new Button(radioButtonGroupContainer, SWT.RADIO);
m_yesButton.setText(YES);
m_noButton = new Button(radioButtonGroupContainer, SWT.RADIO);
m_noButton.setText(NO);
m_noButton.setSelection(true);
p_dbContext.bindValue(SWTObservables.observeSelection(this.getYesButton()),
p_modelProperty, null, null);
}
Я пробовал это и нашел работу хорошо. Просто проверьте.
Ответ 4
Если вы используете управление Nebula RadioGroup, вы можете использовать RadioGroupViewer и привязать выбор зрителя, как и любой другой просмотрщик, например. ComboViewer
Ответ 5
Это то, что я закончил с отображением переключателей для перечисления (я продолжу и сделаю его многоразовым в нашем проекте). Использование ISWTObservableValue позволяет сэкономить усилие добавления слушателей:
final ISWTObservableValue upload = SWTObservables.observeSelection(btnUpload);
final ISWTObservableValue download = SWTObservables.observeSelection(btnDownload);
final ISWTObservableValue noTransfer = SWTObservables.observeSelection(btnNoTransfer);
final IObservableValue btns = new ComputedValue(ExecutableTransferModes.class) {
@Override
protected void doSetValue(final Object value) {
boolean u = false, d = false, n = false;
switch ((ExecutableTransferModes) value) {
case Upload:
u = true;
break;
case Download:
d = true;
break;
case Notransfer:
n = true;
break;
}
upload.setValue(u);
download.setValue(d);
noTransfer.setValue(n);
}
@Override
protected Object calculate() {
if (Boolean.TRUE.equals(upload.getValue())) {
return ExecutableTransferModes.Upload;
} else if (Boolean.TRUE.equals(download.getValue())) {
return ExecutableTransferModes.Download;
} else if (Boolean.TRUE.equals(noTransfer.getValue())) {
return ExecutableTransferModes.Notransfer;
} else {
return null;
}
}
};
bindingContext.bindValue(btns, uploadProperty);
Ответ 6
Это должно работать с минимальными усилиями, и все мы знаем, что лучший ответ правильный?
public static final void createAndBind(
final DataBindingContext dbc,
final Composite parent,
final Object bean,
final List<Object> options,
final String fieldName,
final Class<?> fieldType) {
//
List<Button> radios = new ArrayList<>();
Button b;
for (Object option : options) {
b = new Button(parent, SWT.RADIO);
b.setText(option.toString());
b.setLayoutData(new GridData(GridData.FILL_HORIZONTAL));
radios.add(b);
}
//
List<IObservableValue> iovs = new ArrayList<>();
IObservableValue iov;
for (Button radio : radios) {
iov = SWTObservables.observeSelection(radio);
iovs.add(iov);
}
SelectObservableValue observable = new SelectObservableValue(fieldType);
//
for (int i = 0; i < options.size(); i++) {
observable.addOption(options.get(i), iovs.get(i));
}
//
dbc.bindValue(observable, PojoObservables.observeValue(bean, fieldName));
}