info monitoring disable
This commit is contained in:
parent
0e528a3c86
commit
663662d50e
21 changed files with 309 additions and 192 deletions
|
@ -8,6 +8,7 @@
|
||||||
|
|
||||||
package ch.ethz.seb.sebserver.gbl.model.session;
|
package ch.ethz.seb.sebserver.gbl.model.session;
|
||||||
|
|
||||||
|
import java.util.EnumSet;
|
||||||
import java.util.function.Predicate;
|
import java.util.function.Predicate;
|
||||||
|
|
||||||
import com.fasterxml.jackson.annotation.JsonCreator;
|
import com.fasterxml.jackson.annotation.JsonCreator;
|
||||||
|
@ -234,4 +235,14 @@ public final class ClientConnection implements GrantEntity {
|
||||||
return connection -> connection.status == status;
|
return connection -> connection.status == status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static Predicate<ClientConnection> getStatusPredicate(final ConnectionStatus... status) {
|
||||||
|
final EnumSet<ConnectionStatus> stati = EnumSet.allOf(ConnectionStatus.class);
|
||||||
|
if (status != null) {
|
||||||
|
for (int i = 0; i < status.length; i++) {
|
||||||
|
stati.add(status[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return connection -> stati.contains(connection.status);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -110,11 +110,13 @@ public class InstitutionForm implements TemplateComposer {
|
||||||
.addField(FormBuilder.text(
|
.addField(FormBuilder.text(
|
||||||
Domain.INSTITUTION.ATTR_NAME,
|
Domain.INSTITUTION.ATTR_NAME,
|
||||||
FORM_NAME_TEXT_KEY,
|
FORM_NAME_TEXT_KEY,
|
||||||
institution.name))
|
institution.name)
|
||||||
|
.withInfoLeft(new LocTextKey("some info text here!\nwith line brake")))
|
||||||
.addField(FormBuilder.text(
|
.addField(FormBuilder.text(
|
||||||
Domain.INSTITUTION.ATTR_URL_SUFFIX,
|
Domain.INSTITUTION.ATTR_URL_SUFFIX,
|
||||||
FORM_URL_SUFFIX_TEXT_KEY,
|
FORM_URL_SUFFIX_TEXT_KEY,
|
||||||
institution.urlSuffix))
|
institution.urlSuffix)
|
||||||
|
.withInfoRight(new LocTextKey("some info text here!\nwith line brake")))
|
||||||
.addField(FormBuilder.imageUpload(
|
.addField(FormBuilder.imageUpload(
|
||||||
Domain.INSTITUTION.ATTR_LOGO_IMAGE,
|
Domain.INSTITUTION.ATTR_LOGO_IMAGE,
|
||||||
FORM_LOGO_IMAGE_TEXT_KEY,
|
FORM_LOGO_IMAGE_TEXT_KEY,
|
||||||
|
|
|
@ -9,11 +9,9 @@
|
||||||
package ch.ethz.seb.sebserver.gui.content;
|
package ch.ethz.seb.sebserver.gui.content;
|
||||||
|
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.Set;
|
|
||||||
import java.util.function.BooleanSupplier;
|
import java.util.function.BooleanSupplier;
|
||||||
import java.util.function.Consumer;
|
import java.util.function.Consumer;
|
||||||
import java.util.function.Function;
|
import java.util.function.Function;
|
||||||
import java.util.function.Predicate;
|
|
||||||
|
|
||||||
import org.eclipse.swt.SWT;
|
import org.eclipse.swt.SWT;
|
||||||
import org.eclipse.swt.layout.GridData;
|
import org.eclipse.swt.layout.GridData;
|
||||||
|
@ -31,7 +29,6 @@ import ch.ethz.seb.sebserver.gbl.model.Domain;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.EntityKey;
|
import ch.ethz.seb.sebserver.gbl.model.EntityKey;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
|
import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.exam.Indicator;
|
import ch.ethz.seb.sebserver.gbl.model.exam.Indicator;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection;
|
|
||||||
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection.ConnectionStatus;
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection.ConnectionStatus;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnectionData;
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnectionData;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.user.UserRole;
|
import ch.ethz.seb.sebserver.gbl.model.user.UserRole;
|
||||||
|
@ -71,6 +68,8 @@ public class MonitoringRunningExam implements TemplateComposer {
|
||||||
new LocTextKey("sebserver.monitoring.exam.connection.action.instruction.quit.selected.confirm");
|
new LocTextKey("sebserver.monitoring.exam.connection.action.instruction.quit.selected.confirm");
|
||||||
private static final LocTextKey CONFIRM_QUIT_ALL =
|
private static final LocTextKey CONFIRM_QUIT_ALL =
|
||||||
new LocTextKey("sebserver.monitoring.exam.connection.action.instruction.quit.all.confirm");
|
new LocTextKey("sebserver.monitoring.exam.connection.action.instruction.quit.all.confirm");
|
||||||
|
private static final LocTextKey CONFIRM_DISABLE_SELECTED =
|
||||||
|
new LocTextKey("sebserver.monitoring.exam.connection.action.instruction.disable.selected.confirm");
|
||||||
|
|
||||||
private final ServerPushService serverPushService;
|
private final ServerPushService serverPushService;
|
||||||
private final PageService pageService;
|
private final PageService pageService;
|
||||||
|
@ -183,7 +182,19 @@ public class MonitoringRunningExam implements TemplateComposer {
|
||||||
action -> this.quitSebClients(action, clientTable, false),
|
action -> this.quitSebClients(action, clientTable, false),
|
||||||
EMPTY_SELECTION_TEXT_KEY)
|
EMPTY_SELECTION_TEXT_KEY)
|
||||||
.noEventPropagation()
|
.noEventPropagation()
|
||||||
.publishIf(privilege);
|
.publishIf(privilege)
|
||||||
|
|
||||||
|
.newAction(ActionDefinition.MONITOR_EXAM_DISABLE_SELECTED_CONNECTION)
|
||||||
|
.withEntityKey(entityKey)
|
||||||
|
.withConfirm(() -> CONFIRM_DISABLE_SELECTED)
|
||||||
|
.withSelect(
|
||||||
|
clientTable::getSelection,
|
||||||
|
action -> this.disableSebClients(action, clientTable, false),
|
||||||
|
EMPTY_SELECTION_TEXT_KEY)
|
||||||
|
.noEventPropagation()
|
||||||
|
.publishIf(privilege)
|
||||||
|
|
||||||
|
;
|
||||||
|
|
||||||
clientTable.hideStatus(ConnectionStatus.DISABLED);
|
clientTable.hideStatus(ConnectionStatus.DISABLED);
|
||||||
|
|
||||||
|
@ -228,6 +239,7 @@ public class MonitoringRunningExam implements TemplateComposer {
|
||||||
|
|
||||||
return action -> {
|
return action -> {
|
||||||
clientTable.showStatus(status);
|
clientTable.showStatus(status);
|
||||||
|
clientTable.removeSelection();
|
||||||
return action;
|
return action;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -238,6 +250,7 @@ public class MonitoringRunningExam implements TemplateComposer {
|
||||||
|
|
||||||
return action -> {
|
return action -> {
|
||||||
clientTable.hideStatus(status);
|
clientTable.hideStatus(status);
|
||||||
|
clientTable.removeSelection();
|
||||||
return action;
|
return action;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -247,20 +260,27 @@ public class MonitoringRunningExam implements TemplateComposer {
|
||||||
final ClientConnectionTable clientTable,
|
final ClientConnectionTable clientTable,
|
||||||
final boolean all) {
|
final boolean all) {
|
||||||
|
|
||||||
final Predicate<ClientConnection> activePredicate = ClientConnection
|
|
||||||
.getStatusPredicate(ConnectionStatus.ACTIVE);
|
|
||||||
|
|
||||||
final Set<String> connectionTokens = clientTable.getConnectionTokens(
|
|
||||||
activePredicate,
|
|
||||||
!all);
|
|
||||||
|
|
||||||
if (connectionTokens.isEmpty()) {
|
|
||||||
return action;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.instructionProcessor.propagateSebQuitInstruction(
|
this.instructionProcessor.propagateSebQuitInstruction(
|
||||||
clientTable.getExam().id,
|
clientTable.getExam().id,
|
||||||
connectionTokens,
|
statesPredicate -> clientTable.getConnectionTokens(
|
||||||
|
statesPredicate,
|
||||||
|
!all),
|
||||||
|
action.pageContext());
|
||||||
|
|
||||||
|
clientTable.removeSelection();
|
||||||
|
return action;
|
||||||
|
}
|
||||||
|
|
||||||
|
private PageAction disableSebClients(
|
||||||
|
final PageAction action,
|
||||||
|
final ClientConnectionTable clientTable,
|
||||||
|
final boolean all) {
|
||||||
|
|
||||||
|
this.instructionProcessor.disableConnection(
|
||||||
|
clientTable.getExam().id,
|
||||||
|
statesPredicate -> clientTable.getConnectionTokens(
|
||||||
|
statesPredicate,
|
||||||
|
!all),
|
||||||
action.pageContext());
|
action.pageContext());
|
||||||
|
|
||||||
clientTable.removeSelection();
|
clientTable.removeSelection();
|
||||||
|
|
|
@ -26,15 +26,8 @@ public class CheckboxFieldBuilder extends FieldBuilder<String> {
|
||||||
@Override
|
@Override
|
||||||
void build(final FormBuilder builder) {
|
void build(final FormBuilder builder) {
|
||||||
final boolean readonly = builder.readonly || this.readonly;
|
final boolean readonly = builder.readonly || this.readonly;
|
||||||
final Label lab = (this.label != null)
|
final Label titleLabel = createTitleLabel(builder.formParent, builder, this);
|
||||||
? builder.labelLocalized(
|
final Composite fieldGrid = createFieldGrid(builder.formParent, this.spanInput);
|
||||||
builder.formParent,
|
|
||||||
this.label,
|
|
||||||
this.defaultLabel,
|
|
||||||
this.spanLabel)
|
|
||||||
: null;
|
|
||||||
|
|
||||||
final Composite fieldGrid = Form.createFieldGrid(builder.formParent, this.spanInput);
|
|
||||||
final Button checkbox = builder.widgetFactory.buttonLocalized(
|
final Button checkbox = builder.widgetFactory.buttonLocalized(
|
||||||
fieldGrid,
|
fieldGrid,
|
||||||
SWT.CHECK,
|
SWT.CHECK,
|
||||||
|
@ -48,7 +41,7 @@ public class CheckboxFieldBuilder extends FieldBuilder<String> {
|
||||||
checkbox.setEnabled(false);
|
checkbox.setEnabled(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
builder.form.putField(this.name, lab, checkbox);
|
builder.form.putField(this.name, titleLabel, checkbox);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,17 +10,31 @@ package ch.ethz.seb.sebserver.gui.form;
|
||||||
|
|
||||||
import java.util.function.BooleanSupplier;
|
import java.util.function.BooleanSupplier;
|
||||||
|
|
||||||
|
import org.apache.commons.lang3.StringUtils;
|
||||||
|
import org.eclipse.rap.rwt.RWT;
|
||||||
|
import org.eclipse.swt.SWT;
|
||||||
|
import org.eclipse.swt.layout.GridData;
|
||||||
|
import org.eclipse.swt.layout.GridLayout;
|
||||||
|
import org.eclipse.swt.widgets.Composite;
|
||||||
|
import org.eclipse.swt.widgets.Label;
|
||||||
|
|
||||||
import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey;
|
import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey;
|
||||||
|
import ch.ethz.seb.sebserver.gui.widget.WidgetFactory;
|
||||||
|
import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.CustomVariant;
|
||||||
|
|
||||||
public abstract class FieldBuilder<T> {
|
public abstract class FieldBuilder<T> {
|
||||||
int spanLabel = -1;
|
int spanLabel = -1;
|
||||||
int spanInput = -1;
|
int spanInput = -1;
|
||||||
int spanEmptyCell = -1;
|
int spanEmptyCell = -1;
|
||||||
|
int titleValign = SWT.TOP;
|
||||||
Boolean autoEmptyCellSeparation = null;
|
Boolean autoEmptyCellSeparation = null;
|
||||||
String group = null;
|
String group = null;
|
||||||
boolean readonly = false;
|
boolean readonly = false;
|
||||||
boolean visible = true;
|
boolean visible = true;
|
||||||
String defaultLabel = null;
|
String defaultLabel = null;
|
||||||
|
LocTextKey infoText;
|
||||||
|
boolean infoLeft = false;
|
||||||
|
boolean infoRight = false;
|
||||||
|
|
||||||
final String name;
|
final String name;
|
||||||
final LocTextKey label;
|
final LocTextKey label;
|
||||||
|
@ -42,6 +56,18 @@ public abstract class FieldBuilder<T> {
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public FieldBuilder<T> withInfoLeft(final LocTextKey infoText) {
|
||||||
|
this.infoText = infoText;
|
||||||
|
this.infoLeft = true;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public FieldBuilder<T> withInfoRight(final LocTextKey infoText) {
|
||||||
|
this.infoText = infoText;
|
||||||
|
this.infoRight = true;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
public FieldBuilder<T> withInputSpan(final int span) {
|
public FieldBuilder<T> withInputSpan(final int span) {
|
||||||
this.spanInput = span;
|
this.spanInput = span;
|
||||||
return this;
|
return this;
|
||||||
|
@ -79,4 +105,102 @@ public abstract class FieldBuilder<T> {
|
||||||
|
|
||||||
abstract void build(FormBuilder builder);
|
abstract void build(FormBuilder builder);
|
||||||
|
|
||||||
|
protected static Label createTitleLabel(
|
||||||
|
final Composite parent,
|
||||||
|
final FormBuilder builder,
|
||||||
|
final FieldBuilder<?> fieldBuilder) {
|
||||||
|
|
||||||
|
final Composite infoGrid = new Composite(parent, SWT.NONE);
|
||||||
|
final GridLayout gridLayout = new GridLayout(2, false);
|
||||||
|
gridLayout.verticalSpacing = 0;
|
||||||
|
gridLayout.marginHeight = 0;
|
||||||
|
gridLayout.marginWidth = 0;
|
||||||
|
gridLayout.marginRight = 0;
|
||||||
|
infoGrid.setLayout(gridLayout);
|
||||||
|
|
||||||
|
final GridData gridData = new GridData(SWT.LEFT, SWT.TOP, false, false);
|
||||||
|
infoGrid.setLayoutData(gridData);
|
||||||
|
|
||||||
|
if (fieldBuilder.infoText != null && fieldBuilder.infoLeft) {
|
||||||
|
final Label info = builder.widgetFactory.imageButton(
|
||||||
|
WidgetFactory.ImageIcon.HELP,
|
||||||
|
infoGrid,
|
||||||
|
fieldBuilder.infoText);
|
||||||
|
info.setLayoutData(new GridData(SWT.LEFT, SWT.TOP, false, false));
|
||||||
|
}
|
||||||
|
final Label lab = (fieldBuilder.label != null)
|
||||||
|
? labelLocalized(
|
||||||
|
builder.widgetFactory,
|
||||||
|
infoGrid,
|
||||||
|
fieldBuilder.label,
|
||||||
|
fieldBuilder.defaultLabel,
|
||||||
|
(fieldBuilder.spanLabel > 0) ? fieldBuilder.spanLabel : 1,
|
||||||
|
fieldBuilder.titleValign)
|
||||||
|
: null;
|
||||||
|
|
||||||
|
if (fieldBuilder.infoText != null && fieldBuilder.infoRight) {
|
||||||
|
final Label info = builder.widgetFactory.imageButton(
|
||||||
|
WidgetFactory.ImageIcon.HELP,
|
||||||
|
infoGrid,
|
||||||
|
fieldBuilder.infoText);
|
||||||
|
info.setLayoutData(new GridData(SWT.LEFT, SWT.TOP, false, false));
|
||||||
|
}
|
||||||
|
|
||||||
|
return lab;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Label labelLocalized(
|
||||||
|
final WidgetFactory widgetFactory,
|
||||||
|
final Composite parent,
|
||||||
|
final LocTextKey locTextKey,
|
||||||
|
final String defaultText,
|
||||||
|
final int hspan) {
|
||||||
|
|
||||||
|
return labelLocalized(widgetFactory, parent, locTextKey, defaultText, hspan, SWT.CENTER);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Label labelLocalized(
|
||||||
|
final WidgetFactory widgetFactory,
|
||||||
|
final Composite parent,
|
||||||
|
final LocTextKey locTextKey,
|
||||||
|
final String defaultText,
|
||||||
|
final int hspan,
|
||||||
|
final int verticalAlignment) {
|
||||||
|
|
||||||
|
final Label label = widgetFactory.labelLocalized(
|
||||||
|
parent,
|
||||||
|
locTextKey,
|
||||||
|
(StringUtils.isNotBlank(defaultText) ? defaultText : locTextKey.name));
|
||||||
|
final GridData gridData = new GridData(SWT.LEFT, verticalAlignment, true, true, hspan, 1);
|
||||||
|
gridData.heightHint = FormBuilder.FORM_ROW_HEIGHT;
|
||||||
|
label.setLayoutData(gridData);
|
||||||
|
label.setData(RWT.CUSTOM_VARIANT, CustomVariant.TITLE_LABEL.key);
|
||||||
|
return label;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Composite createFieldGrid(final Composite parent, final int hspan) {
|
||||||
|
final Composite fieldGrid = new Composite(parent, SWT.NONE);
|
||||||
|
final GridLayout gridLayout = new GridLayout();
|
||||||
|
gridLayout.verticalSpacing = 0;
|
||||||
|
gridLayout.marginHeight = 0;
|
||||||
|
gridLayout.marginWidth = 0;
|
||||||
|
gridLayout.marginRight = 0;
|
||||||
|
fieldGrid.setLayout(gridLayout);
|
||||||
|
|
||||||
|
final GridData gridData = new GridData(SWT.FILL, SWT.FILL, true, true);
|
||||||
|
gridData.horizontalSpan = hspan;
|
||||||
|
fieldGrid.setLayoutData(gridData);
|
||||||
|
|
||||||
|
return fieldGrid;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Label createErrorLabel(final Composite innerGrid) {
|
||||||
|
final Label errorLabel = new Label(innerGrid, SWT.NONE);
|
||||||
|
final GridData gridData = new GridData(SWT.FILL, SWT.CENTER, true, true);
|
||||||
|
errorLabel.setLayoutData(gridData);
|
||||||
|
errorLabel.setVisible(false);
|
||||||
|
errorLabel.setData(RWT.CUSTOM_VARIANT, CustomVariant.ERROR.key);
|
||||||
|
return errorLabel;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -34,14 +34,8 @@ public class FileUploadFieldBuilder extends FieldBuilder<String> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
void build(final FormBuilder builder) {
|
void build(final FormBuilder builder) {
|
||||||
|
final Label titleLabel = createTitleLabel(builder.formParent, builder, this);
|
||||||
final Label lab = builder.labelLocalized(
|
final Composite fieldGrid = createFieldGrid(builder.formParent, this.spanInput);
|
||||||
builder.formParent,
|
|
||||||
this.label,
|
|
||||||
this.defaultLabel,
|
|
||||||
1);
|
|
||||||
|
|
||||||
final Composite fieldGrid = Form.createFieldGrid(builder.formParent, this.spanInput);
|
|
||||||
final FileUploadSelection fileUpload = builder.widgetFactory.fileUploadSelection(
|
final FileUploadSelection fileUpload = builder.widgetFactory.fileUploadSelection(
|
||||||
fieldGrid,
|
fieldGrid,
|
||||||
builder.readonly || this.readonly,
|
builder.readonly || this.readonly,
|
||||||
|
@ -50,8 +44,8 @@ public class FileUploadFieldBuilder extends FieldBuilder<String> {
|
||||||
fileUpload.setLayoutData(gridData);
|
fileUpload.setLayoutData(gridData);
|
||||||
fileUpload.setFileName(this.value);
|
fileUpload.setFileName(this.value);
|
||||||
|
|
||||||
final Label errorLabel = Form.createErrorLabel(fieldGrid);
|
final Label errorLabel = createErrorLabel(fieldGrid);
|
||||||
builder.form.putField(this.name, lab, fileUpload, errorLabel);
|
builder.form.putField(this.name, titleLabel, fileUpload, errorLabel);
|
||||||
builder.setFieldVisible(this.visible, this.name);
|
builder.setFieldVisible(this.visible, this.name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -21,12 +21,8 @@ import java.util.function.Predicate;
|
||||||
import org.apache.commons.lang3.BooleanUtils;
|
import org.apache.commons.lang3.BooleanUtils;
|
||||||
import org.apache.commons.lang3.StringUtils;
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.eclipse.rap.rwt.RWT;
|
import org.eclipse.rap.rwt.RWT;
|
||||||
import org.eclipse.swt.SWT;
|
|
||||||
import org.eclipse.swt.graphics.Color;
|
import org.eclipse.swt.graphics.Color;
|
||||||
import org.eclipse.swt.layout.GridData;
|
|
||||||
import org.eclipse.swt.layout.GridLayout;
|
|
||||||
import org.eclipse.swt.widgets.Button;
|
import org.eclipse.swt.widgets.Button;
|
||||||
import org.eclipse.swt.widgets.Composite;
|
|
||||||
import org.eclipse.swt.widgets.Control;
|
import org.eclipse.swt.widgets.Control;
|
||||||
import org.eclipse.swt.widgets.Label;
|
import org.eclipse.swt.widgets.Label;
|
||||||
import org.eclipse.swt.widgets.Text;
|
import org.eclipse.swt.widgets.Text;
|
||||||
|
@ -512,29 +508,4 @@ public final class Form implements FormBinding {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Composite createFieldGrid(final Composite parent, final int hspan) {
|
|
||||||
final Composite fieldGrid = new Composite(parent, SWT.NONE);
|
|
||||||
final GridLayout gridLayout = new GridLayout();
|
|
||||||
gridLayout.verticalSpacing = 0;
|
|
||||||
gridLayout.marginHeight = 0;
|
|
||||||
gridLayout.marginWidth = 0;
|
|
||||||
gridLayout.marginRight = 0;
|
|
||||||
fieldGrid.setLayout(gridLayout);
|
|
||||||
|
|
||||||
final GridData gridData = new GridData(SWT.FILL, SWT.FILL, true, true);
|
|
||||||
gridData.horizontalSpan = hspan;
|
|
||||||
fieldGrid.setLayoutData(gridData);
|
|
||||||
|
|
||||||
return fieldGrid;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static Label createErrorLabel(final Composite innerGrid) {
|
|
||||||
final Label errorLabel = new Label(innerGrid, SWT.NONE);
|
|
||||||
final GridData gridData = new GridData(SWT.FILL, SWT.CENTER, true, true);
|
|
||||||
errorLabel.setLayoutData(gridData);
|
|
||||||
errorLabel.setVisible(false);
|
|
||||||
errorLabel.setData(RWT.CUSTOM_VARIANT, CustomVariant.ERROR.key);
|
|
||||||
return errorLabel;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,7 +15,6 @@ import java.util.function.BooleanSupplier;
|
||||||
import java.util.function.Supplier;
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
import org.apache.commons.lang3.StringUtils;
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.eclipse.rap.rwt.RWT;
|
|
||||||
import org.eclipse.swt.SWT;
|
import org.eclipse.swt.SWT;
|
||||||
import org.eclipse.swt.layout.GridData;
|
import org.eclipse.swt.layout.GridData;
|
||||||
import org.eclipse.swt.widgets.Composite;
|
import org.eclipse.swt.widgets.Composite;
|
||||||
|
@ -24,7 +23,6 @@ import org.eclipse.swt.widgets.TabItem;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
import ch.ethz.seb.sebserver.gbl.Constants;
|
|
||||||
import ch.ethz.seb.sebserver.gbl.model.Entity;
|
import ch.ethz.seb.sebserver.gbl.model.Entity;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.exam.Indicator;
|
import ch.ethz.seb.sebserver.gbl.model.exam.Indicator;
|
||||||
import ch.ethz.seb.sebserver.gbl.util.Tuple;
|
import ch.ethz.seb.sebserver.gbl.util.Tuple;
|
||||||
|
@ -35,11 +33,10 @@ import ch.ethz.seb.sebserver.gui.service.page.PageService;
|
||||||
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall;
|
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall;
|
||||||
import ch.ethz.seb.sebserver.gui.widget.Selection;
|
import ch.ethz.seb.sebserver.gui.widget.Selection;
|
||||||
import ch.ethz.seb.sebserver.gui.widget.WidgetFactory;
|
import ch.ethz.seb.sebserver.gui.widget.WidgetFactory;
|
||||||
import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.CustomVariant;
|
|
||||||
|
|
||||||
public class FormBuilder {
|
public class FormBuilder {
|
||||||
|
|
||||||
private static final int FORM_ROW_HEIGHT = 25;
|
public static final int FORM_ROW_HEIGHT = 25;
|
||||||
|
|
||||||
private static final Logger log = LoggerFactory.getLogger(FormBuilder.class);
|
private static final Logger log = LoggerFactory.getLogger(FormBuilder.class);
|
||||||
|
|
||||||
|
@ -283,54 +280,4 @@ public class FormBuilder {
|
||||||
(supportedFiles != null) ? Arrays.asList(supportedFiles) : Collections.emptyList());
|
(supportedFiles != null) ? Arrays.asList(supportedFiles) : Collections.emptyList());
|
||||||
}
|
}
|
||||||
|
|
||||||
Label labelLocalized(
|
|
||||||
final Composite parent,
|
|
||||||
final LocTextKey locTextKey,
|
|
||||||
final String defaultText,
|
|
||||||
final int hspan) {
|
|
||||||
|
|
||||||
return labelLocalized(parent, locTextKey, defaultText, hspan, SWT.CENTER);
|
|
||||||
}
|
|
||||||
|
|
||||||
Label labelLocalized(
|
|
||||||
final Composite parent,
|
|
||||||
final LocTextKey locTextKey,
|
|
||||||
final String defaultText,
|
|
||||||
final int hspan,
|
|
||||||
final int verticalAlignment) {
|
|
||||||
|
|
||||||
final Label label = this.widgetFactory.labelLocalized(
|
|
||||||
parent,
|
|
||||||
locTextKey,
|
|
||||||
(StringUtils.isNotBlank(defaultText) ? defaultText : locTextKey.name));
|
|
||||||
final GridData gridData = new GridData(SWT.LEFT, verticalAlignment, true, true, hspan, 1);
|
|
||||||
gridData.heightHint = FORM_ROW_HEIGHT;
|
|
||||||
label.setLayoutData(gridData);
|
|
||||||
label.setData(RWT.CUSTOM_VARIANT, CustomVariant.TITLE_LABEL.key);
|
|
||||||
return label;
|
|
||||||
}
|
|
||||||
|
|
||||||
Label valueLabel(
|
|
||||||
final Composite parent,
|
|
||||||
final String value,
|
|
||||||
final int hspan,
|
|
||||||
final boolean centered) {
|
|
||||||
|
|
||||||
final Label label = new Label(parent, SWT.NONE);
|
|
||||||
label.setText((StringUtils.isNotBlank(value)) ? value : Constants.EMPTY_NOTE);
|
|
||||||
final GridData gridData = new GridData(
|
|
||||||
SWT.FILL,
|
|
||||||
(centered) ? SWT.CENTER : SWT.TOP,
|
|
||||||
true, false,
|
|
||||||
hspan, 1);
|
|
||||||
|
|
||||||
if (centered) {
|
|
||||||
label.setAlignment(SWT.CENTER);
|
|
||||||
label.setData(RWT.CUSTOM_VARIANT, CustomVariant.FORM_CENTER.key);
|
|
||||||
}
|
|
||||||
|
|
||||||
label.setLayoutData(gridData);
|
|
||||||
return label;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -37,14 +37,8 @@ public final class ImageUploadFieldBuilder extends FieldBuilder<String> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
void build(final FormBuilder builder) {
|
void build(final FormBuilder builder) {
|
||||||
|
final Label titleLabel = createTitleLabel(builder.formParent, builder, this);
|
||||||
final Label lab = builder.labelLocalized(
|
final Composite fieldGrid = createFieldGrid(builder.formParent, this.spanInput);
|
||||||
builder.formParent,
|
|
||||||
this.label,
|
|
||||||
this.defaultLabel,
|
|
||||||
1);
|
|
||||||
|
|
||||||
final Composite fieldGrid = Form.createFieldGrid(builder.formParent, this.spanInput);
|
|
||||||
final ImageUploadSelection imageUpload = builder.widgetFactory.imageUploadLocalized(
|
final ImageUploadSelection imageUpload = builder.widgetFactory.imageUploadLocalized(
|
||||||
fieldGrid,
|
fieldGrid,
|
||||||
new LocTextKey("sebserver.overall.upload"),
|
new LocTextKey("sebserver.overall.upload"),
|
||||||
|
@ -55,8 +49,8 @@ public final class ImageUploadFieldBuilder extends FieldBuilder<String> {
|
||||||
imageUpload.setLayoutData(gridData);
|
imageUpload.setLayoutData(gridData);
|
||||||
imageUpload.setImageBase64(this.value);
|
imageUpload.setImageBase64(this.value);
|
||||||
|
|
||||||
final Label errorLabel = Form.createErrorLabel(fieldGrid);
|
final Label errorLabel = createErrorLabel(fieldGrid);
|
||||||
builder.form.putField(this.name, lab, imageUpload, errorLabel);
|
builder.form.putField(this.name, titleLabel, imageUpload, errorLabel);
|
||||||
builder.setFieldVisible(this.visible, this.name);
|
builder.setFieldVisible(this.visible, this.name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -55,25 +55,18 @@ public final class SelectionFieldBuilder extends FieldBuilder<String> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
void build(final FormBuilder builder) {
|
void build(final FormBuilder builder) {
|
||||||
final Label lab = (this.label != null)
|
final Label titleLabel = createTitleLabel(builder.formParent, builder, this);
|
||||||
? builder.labelLocalized(
|
|
||||||
builder.formParent,
|
|
||||||
this.label,
|
|
||||||
this.defaultLabel,
|
|
||||||
this.spanLabel,
|
|
||||||
SWT.TOP)
|
|
||||||
: null;
|
|
||||||
|
|
||||||
if (builder.readonly || this.readonly) {
|
if (builder.readonly || this.readonly) {
|
||||||
buildReadOnly(builder, lab);
|
buildReadOnly(builder, titleLabel);
|
||||||
} else {
|
} else {
|
||||||
buildInput(builder, lab);
|
buildInput(builder, titleLabel);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void buildInput(final FormBuilder builder, final Label lab) {
|
private void buildInput(final FormBuilder builder, final Label titleLabel) {
|
||||||
|
|
||||||
final Composite fieldGrid = Form.createFieldGrid(builder.formParent, this.spanInput);
|
final Composite fieldGrid = createFieldGrid(builder.formParent, this.spanInput);
|
||||||
final String actionKey = (this.label != null) ? this.label.name + ".action" : null;
|
final String actionKey = (this.label != null) ? this.label.name + ".action" : null;
|
||||||
final Selection selection = builder.widgetFactory.selectionLocalized(
|
final Selection selection = builder.widgetFactory.selectionLocalized(
|
||||||
this.type,
|
this.type,
|
||||||
|
@ -87,8 +80,8 @@ public final class SelectionFieldBuilder extends FieldBuilder<String> {
|
||||||
((Control) selection).setLayoutData(gridData);
|
((Control) selection).setLayoutData(gridData);
|
||||||
selection.select(this.value);
|
selection.select(this.value);
|
||||||
|
|
||||||
final Label errorLabel = Form.createErrorLabel(fieldGrid);
|
final Label errorLabel = createErrorLabel(fieldGrid);
|
||||||
builder.form.putField(this.name, lab, selection, errorLabel);
|
builder.form.putField(this.name, titleLabel, selection, errorLabel);
|
||||||
|
|
||||||
if (this.selectionListener != null) {
|
if (this.selectionListener != null) {
|
||||||
((Control) selection).addListener(SWT.Selection, e -> {
|
((Control) selection).addListener(SWT.Selection, e -> {
|
||||||
|
@ -100,7 +93,7 @@ public final class SelectionFieldBuilder extends FieldBuilder<String> {
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Build the read-only representation of the selection field */
|
/* Build the read-only representation of the selection field */
|
||||||
private void buildReadOnly(final FormBuilder builder, final Label lab) {
|
private void buildReadOnly(final FormBuilder builder, final Label titleLabel) {
|
||||||
if (this.type == Type.MULTI || this.type == Type.MULTI_COMBO) {
|
if (this.type == Type.MULTI || this.type == Type.MULTI_COMBO) {
|
||||||
final Composite composite = new Composite(builder.formParent, SWT.NONE);
|
final Composite composite = new Composite(builder.formParent, SWT.NONE);
|
||||||
final GridLayout gridLayout = new GridLayout(1, true);
|
final GridLayout gridLayout = new GridLayout(1, true);
|
||||||
|
@ -127,7 +120,7 @@ public final class SelectionFieldBuilder extends FieldBuilder<String> {
|
||||||
} else {
|
} else {
|
||||||
builder.form.putReadonlyField(
|
builder.form.putReadonlyField(
|
||||||
this.name,
|
this.name,
|
||||||
lab,
|
titleLabel,
|
||||||
buildReadonlyLabel(builder.formParent, this.value, this.spanInput));
|
buildReadonlyLabel(builder.formParent, this.value, this.spanInput));
|
||||||
builder.setFieldVisible(this.visible, this.name);
|
builder.setFieldVisible(this.visible, this.name);
|
||||||
}
|
}
|
||||||
|
|
|
@ -52,6 +52,7 @@ public final class TextFieldBuilder extends FieldBuilder<String> {
|
||||||
|
|
||||||
public TextFieldBuilder asArea() {
|
public TextFieldBuilder asArea() {
|
||||||
this.isArea = true;
|
this.isArea = true;
|
||||||
|
this.titleValign = SWT.CENTER;
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -63,16 +64,8 @@ public final class TextFieldBuilder extends FieldBuilder<String> {
|
||||||
@Override
|
@Override
|
||||||
void build(final FormBuilder builder) {
|
void build(final FormBuilder builder) {
|
||||||
final boolean readonly = builder.readonly || this.readonly;
|
final boolean readonly = builder.readonly || this.readonly;
|
||||||
final Label lab = (this.label != null)
|
final Label titleLabel = createTitleLabel(builder.formParent, builder, this);
|
||||||
? builder.labelLocalized(
|
final Composite fieldGrid = createFieldGrid(builder.formParent, this.spanInput);
|
||||||
builder.formParent,
|
|
||||||
this.label,
|
|
||||||
this.defaultLabel,
|
|
||||||
this.spanLabel,
|
|
||||||
(this.isArea) ? SWT.TOP : SWT.CENTER)
|
|
||||||
: null;
|
|
||||||
|
|
||||||
final Composite fieldGrid = Form.createFieldGrid(builder.formParent, this.spanInput);
|
|
||||||
final Text textInput = (this.isNumber)
|
final Text textInput = (this.isNumber)
|
||||||
? builder.widgetFactory.numberInput(fieldGrid, this.numberCheck, readonly)
|
? builder.widgetFactory.numberInput(fieldGrid, this.numberCheck, readonly)
|
||||||
: (this.isArea)
|
: (this.isArea)
|
||||||
|
@ -95,12 +88,13 @@ public final class TextFieldBuilder extends FieldBuilder<String> {
|
||||||
|
|
||||||
if (readonly) {
|
if (readonly) {
|
||||||
textInput.setEditable(false);
|
textInput.setEditable(false);
|
||||||
builder.form.putReadonlyField(this.name, lab, textInput);
|
builder.form.putReadonlyField(this.name, titleLabel, textInput);
|
||||||
} else {
|
} else {
|
||||||
final Label errorLabel = Form.createErrorLabel(fieldGrid);
|
final Label errorLabel = createErrorLabel(fieldGrid);
|
||||||
builder.form.putField(this.name, lab, textInput, errorLabel);
|
builder.form.putField(this.name, titleLabel, textInput, errorLabel);
|
||||||
builder.setFieldVisible(this.visible, this.name);
|
builder.setFieldVisible(this.visible, this.name);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -39,18 +39,13 @@ public class ThresholdListBuilder extends FieldBuilder<Collection<Threshold>> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
void build(final FormBuilder builder) {
|
void build(final FormBuilder builder) {
|
||||||
final Label lab = builder.labelLocalized(
|
final Label titleLabel = createTitleLabel(builder.formParent, builder, this);
|
||||||
builder.formParent,
|
|
||||||
this.label,
|
|
||||||
this.defaultLabel,
|
|
||||||
this.spanLabel);
|
|
||||||
|
|
||||||
if (builder.readonly || this.readonly) {
|
if (builder.readonly || this.readonly) {
|
||||||
// No read-only view needed for this so far?
|
// No read-only view needed for this so far?
|
||||||
return;
|
return;
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
final Composite fieldGrid = Form.createFieldGrid(builder.formParent, this.spanInput);
|
final Composite fieldGrid = createFieldGrid(builder.formParent, this.spanInput);
|
||||||
|
|
||||||
final ThresholdList thresholdList = builder.widgetFactory.thresholdList(
|
final ThresholdList thresholdList = builder.widgetFactory.thresholdList(
|
||||||
fieldGrid,
|
fieldGrid,
|
||||||
|
@ -68,8 +63,8 @@ public class ThresholdListBuilder extends FieldBuilder<Collection<Threshold>> {
|
||||||
final GridData gridData = new GridData(SWT.FILL, SWT.FILL, true, false);
|
final GridData gridData = new GridData(SWT.FILL, SWT.FILL, true, false);
|
||||||
thresholdList.setLayoutData(gridData);
|
thresholdList.setLayoutData(gridData);
|
||||||
|
|
||||||
final Label errorLabel = Form.createErrorLabel(fieldGrid);
|
final Label errorLabel = createErrorLabel(fieldGrid);
|
||||||
builder.form.putField(this.name, lab, thresholdList, errorLabel);
|
builder.form.putField(this.name, titleLabel, thresholdList, errorLabel);
|
||||||
builder.setFieldVisible(this.visible, this.name);
|
builder.setFieldVisible(this.visible, this.name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -29,7 +29,7 @@ import ch.ethz.seb.sebserver.gbl.model.sebconfig.AttributeType;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationAttribute;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationAttribute;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Orientation;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Orientation;
|
||||||
import ch.ethz.seb.sebserver.gbl.profile.GuiProfile;
|
import ch.ethz.seb.sebserver.gbl.profile.GuiProfile;
|
||||||
import ch.ethz.seb.sebserver.gui.form.Form;
|
import ch.ethz.seb.sebserver.gui.form.FieldBuilder;
|
||||||
import ch.ethz.seb.sebserver.gui.service.examconfig.InputField;
|
import ch.ethz.seb.sebserver.gui.service.examconfig.InputField;
|
||||||
import ch.ethz.seb.sebserver.gui.service.examconfig.InputFieldBuilder;
|
import ch.ethz.seb.sebserver.gui.service.examconfig.InputFieldBuilder;
|
||||||
import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey;
|
import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey;
|
||||||
|
@ -82,7 +82,7 @@ public class PassworFieldBuilder implements InputFieldBuilder {
|
||||||
orientation,
|
orientation,
|
||||||
passwordInput,
|
passwordInput,
|
||||||
confirmInput,
|
confirmInput,
|
||||||
Form.createErrorLabel(innerGrid));
|
FieldBuilder.createErrorLabel(innerGrid));
|
||||||
|
|
||||||
if (viewContext.readonly) {
|
if (viewContext.readonly) {
|
||||||
passwordInput.setEditable(false);
|
passwordInput.setEditable(false);
|
||||||
|
|
|
@ -19,7 +19,7 @@ import ch.ethz.seb.sebserver.gbl.model.sebconfig.AttributeType;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationAttribute;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationAttribute;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Orientation;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Orientation;
|
||||||
import ch.ethz.seb.sebserver.gbl.profile.GuiProfile;
|
import ch.ethz.seb.sebserver.gbl.profile.GuiProfile;
|
||||||
import ch.ethz.seb.sebserver.gui.form.Form;
|
import ch.ethz.seb.sebserver.gui.form.FieldBuilder;
|
||||||
import ch.ethz.seb.sebserver.gui.service.examconfig.ExamConfigurationService;
|
import ch.ethz.seb.sebserver.gui.service.examconfig.ExamConfigurationService;
|
||||||
import ch.ethz.seb.sebserver.gui.service.examconfig.InputField;
|
import ch.ethz.seb.sebserver.gui.service.examconfig.InputField;
|
||||||
import ch.ethz.seb.sebserver.gui.service.examconfig.InputFieldBuilder;
|
import ch.ethz.seb.sebserver.gui.service.examconfig.InputFieldBuilder;
|
||||||
|
@ -74,7 +74,7 @@ public class SingleSelectionFieldBuilder extends SelectionFieldBuilder implement
|
||||||
attribute,
|
attribute,
|
||||||
orientation,
|
orientation,
|
||||||
selection,
|
selection,
|
||||||
Form.createErrorLabel(innerGrid));
|
FieldBuilder.createErrorLabel(innerGrid));
|
||||||
|
|
||||||
if (viewContext.readonly) {
|
if (viewContext.readonly) {
|
||||||
selection.setEnabled(false);
|
selection.setEnabled(false);
|
||||||
|
|
|
@ -25,7 +25,7 @@ import ch.ethz.seb.sebserver.gbl.model.sebconfig.AttributeType;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationAttribute;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationAttribute;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Orientation;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Orientation;
|
||||||
import ch.ethz.seb.sebserver.gbl.profile.GuiProfile;
|
import ch.ethz.seb.sebserver.gbl.profile.GuiProfile;
|
||||||
import ch.ethz.seb.sebserver.gui.form.Form;
|
import ch.ethz.seb.sebserver.gui.form.FieldBuilder;
|
||||||
import ch.ethz.seb.sebserver.gui.service.examconfig.ExamConfigurationService;
|
import ch.ethz.seb.sebserver.gui.service.examconfig.ExamConfigurationService;
|
||||||
import ch.ethz.seb.sebserver.gui.service.examconfig.InputField;
|
import ch.ethz.seb.sebserver.gui.service.examconfig.InputField;
|
||||||
import ch.ethz.seb.sebserver.gui.service.examconfig.InputFieldBuilder;
|
import ch.ethz.seb.sebserver.gui.service.examconfig.InputFieldBuilder;
|
||||||
|
@ -102,7 +102,7 @@ public class TextFieldBuilder implements InputFieldBuilder {
|
||||||
attribute,
|
attribute,
|
||||||
orientation,
|
orientation,
|
||||||
text,
|
text,
|
||||||
Form.createErrorLabel(innerGrid));
|
FieldBuilder.createErrorLabel(innerGrid));
|
||||||
|
|
||||||
if (viewContext.readonly) {
|
if (viewContext.readonly) {
|
||||||
text.setEditable(false);
|
text.setEditable(false);
|
||||||
|
|
|
@ -10,6 +10,11 @@ package ch.ethz.seb.sebserver.gui.service.session;
|
||||||
|
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
import java.util.function.Function;
|
||||||
|
import java.util.function.Predicate;
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
import org.apache.commons.lang3.StringUtils;
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
|
@ -23,14 +28,17 @@ import ch.ethz.seb.sebserver.gbl.Constants;
|
||||||
import ch.ethz.seb.sebserver.gbl.api.API;
|
import ch.ethz.seb.sebserver.gbl.api.API;
|
||||||
import ch.ethz.seb.sebserver.gbl.api.APIMessage;
|
import ch.ethz.seb.sebserver.gbl.api.APIMessage;
|
||||||
import ch.ethz.seb.sebserver.gbl.api.JSONMapper;
|
import ch.ethz.seb.sebserver.gbl.api.JSONMapper;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.Domain;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection.ConnectionStatus;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.session.ClientInstruction;
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientInstruction;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.session.ClientInstruction.InstructionType;
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientInstruction.InstructionType;
|
||||||
import ch.ethz.seb.sebserver.gbl.profile.GuiProfile;
|
import ch.ethz.seb.sebserver.gbl.profile.GuiProfile;
|
||||||
import ch.ethz.seb.sebserver.gbl.util.Utils;
|
|
||||||
import ch.ethz.seb.sebserver.gui.service.page.PageContext;
|
import ch.ethz.seb.sebserver.gui.service.page.PageContext;
|
||||||
import ch.ethz.seb.sebserver.gui.service.page.PageService;
|
import ch.ethz.seb.sebserver.gui.service.page.PageService;
|
||||||
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCallError;
|
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCallError;
|
||||||
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService;
|
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService;
|
||||||
|
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.session.DisableClientConnection;
|
||||||
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.session.PropagateInstruction;
|
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.session.PropagateInstruction;
|
||||||
|
|
||||||
@Lazy
|
@Lazy
|
||||||
|
@ -53,15 +61,23 @@ public class InstructionProcessor {
|
||||||
final String connectionToken,
|
final String connectionToken,
|
||||||
final PageContext pageContext) {
|
final PageContext pageContext) {
|
||||||
|
|
||||||
propagateSebQuitInstruction(examId, Utils.immutableSetOf(connectionToken), pageContext);
|
propagateSebQuitInstruction(
|
||||||
|
examId,
|
||||||
|
p -> Stream.of(connectionToken).collect(Collectors.toSet()),
|
||||||
|
pageContext);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void propagateSebQuitInstruction(
|
public void propagateSebQuitInstruction(
|
||||||
final Long examId,
|
final Long examId,
|
||||||
final Set<String> connectionTokens,
|
final Function<Predicate<ClientConnection>, Set<String>> selectionFunction,
|
||||||
final PageContext pageContext) {
|
final PageContext pageContext) {
|
||||||
|
|
||||||
if (examId == null || connectionTokens == null || connectionTokens.isEmpty()) {
|
final Set<String> connectionTokens = selectionFunction
|
||||||
|
.apply(ClientConnection.getStatusPredicate(ConnectionStatus.ACTIVE));
|
||||||
|
|
||||||
|
if (connectionTokens.isEmpty()) {
|
||||||
|
// TODO message
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -75,15 +91,59 @@ public class InstructionProcessor {
|
||||||
null,
|
null,
|
||||||
examId,
|
examId,
|
||||||
InstructionType.SEB_QUIT,
|
InstructionType.SEB_QUIT,
|
||||||
StringUtils.join(connectionTokens, Constants.COMMA),
|
StringUtils.join(connectionTokens, Constants.LIST_SEPARATOR),
|
||||||
null);
|
null);
|
||||||
|
|
||||||
try {
|
processInstruction(() -> {
|
||||||
final String response = this.restService.getBuilder(PropagateInstruction.class)
|
return this.restService.getBuilder(PropagateInstruction.class)
|
||||||
.withURIVariable(API.PARAM_MODEL_ID, String.valueOf(examId))
|
.withURIVariable(API.PARAM_MODEL_ID, String.valueOf(examId))
|
||||||
.withBody(clientInstruction)
|
.withBody(clientInstruction)
|
||||||
.call()
|
.call()
|
||||||
.getOrThrow();
|
.getOrThrow();
|
||||||
|
},
|
||||||
|
pageContext);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void disableConnection(
|
||||||
|
final Long examId,
|
||||||
|
final Function<Predicate<ClientConnection>, Set<String>> selectionFunction,
|
||||||
|
final PageContext pageContext) {
|
||||||
|
|
||||||
|
final Set<String> connectionTokens = selectionFunction
|
||||||
|
.apply(ClientConnection.getStatusPredicate(
|
||||||
|
ConnectionStatus.CONNECTION_REQUESTED,
|
||||||
|
ConnectionStatus.UNDEFINED,
|
||||||
|
ConnectionStatus.CLOSED,
|
||||||
|
ConnectionStatus.AUTHENTICATED));
|
||||||
|
|
||||||
|
if (connectionTokens.isEmpty()) {
|
||||||
|
// TOOD message
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Disable SEB client connections for exam: {} and connections: {}",
|
||||||
|
examId,
|
||||||
|
connectionTokens);
|
||||||
|
}
|
||||||
|
|
||||||
|
processInstruction(() -> {
|
||||||
|
return this.restService.getBuilder(DisableClientConnection.class)
|
||||||
|
.withURIVariable(API.PARAM_MODEL_ID, String.valueOf(examId))
|
||||||
|
.withFormParam(
|
||||||
|
Domain.CLIENT_CONNECTION.ATTR_CONNECTION_TOKEN,
|
||||||
|
StringUtils.join(connectionTokens, Constants.LIST_SEPARATOR))
|
||||||
|
.call()
|
||||||
|
.getOrThrow();
|
||||||
|
},
|
||||||
|
pageContext);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private void processInstruction(final Supplier<String> apiCall, final PageContext pageContext) {
|
||||||
|
try {
|
||||||
|
final String response = apiCall.get();
|
||||||
|
|
||||||
if (StringUtils.isNotBlank(response)) {
|
if (StringUtils.isNotBlank(response)) {
|
||||||
try {
|
try {
|
||||||
|
@ -93,7 +153,7 @@ public class InstructionProcessor {
|
||||||
});
|
});
|
||||||
|
|
||||||
pageContext.notifyUnexpectedError(new RestCallError(
|
pageContext.notifyUnexpectedError(new RestCallError(
|
||||||
"Failed to propagate SEB quit instruction: ",
|
"Failed to propagate SEB client instruction: ",
|
||||||
errorMessage));
|
errorMessage));
|
||||||
|
|
||||||
} catch (final Exception e) {
|
} catch (final Exception e) {
|
||||||
|
@ -101,7 +161,7 @@ public class InstructionProcessor {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (final Exception e) {
|
} catch (final Exception e) {
|
||||||
log.error("Failed to propagate SEB quit instruction: ", e);
|
log.error("Failed to propagate SEB client instruction: ", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -100,7 +100,8 @@ public class WidgetFactory {
|
||||||
INDICATOR("indicator.png"),
|
INDICATOR("indicator.png"),
|
||||||
TEMPLATE("template.png"),
|
TEMPLATE("template.png"),
|
||||||
DISABLE("disable.png"),
|
DISABLE("disable.png"),
|
||||||
SEND_QUIT("send-quit.png");
|
SEND_QUIT("send-quit.png"),
|
||||||
|
HELP("help.png");
|
||||||
|
|
||||||
private String fileName;
|
private String fileName;
|
||||||
private ImageData image = null;
|
private ImageData image = null;
|
||||||
|
@ -502,6 +503,14 @@ public class WidgetFactory {
|
||||||
return label;
|
return label;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Label imageButton(
|
||||||
|
final ImageIcon type,
|
||||||
|
final Composite parent,
|
||||||
|
final LocTextKey toolTip) {
|
||||||
|
|
||||||
|
return this.imageButton(type, parent, toolTip, null);
|
||||||
|
}
|
||||||
|
|
||||||
public Label imageButton(
|
public Label imageButton(
|
||||||
final ImageIcon type,
|
final ImageIcon type,
|
||||||
final Composite parent,
|
final Composite parent,
|
||||||
|
|
|
@ -11,6 +11,7 @@ package ch.ethz.seb.sebserver.webservice.servicelayer.session.impl;
|
||||||
import java.security.Principal;
|
import java.security.Principal;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
import java.util.function.Predicate;
|
||||||
|
|
||||||
import org.apache.commons.lang3.StringUtils;
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
|
@ -374,6 +375,13 @@ public class SebClientConnectionServiceImpl implements SebClientConnectionServic
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static final Predicate<ClientConnection> DISABLE_STATE_PREDICATE = ClientConnection
|
||||||
|
.getStatusPredicate(
|
||||||
|
ConnectionStatus.UNDEFINED,
|
||||||
|
ConnectionStatus.CONNECTION_REQUESTED,
|
||||||
|
ConnectionStatus.AUTHENTICATED,
|
||||||
|
ConnectionStatus.CLOSED);
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Result<ClientConnection> disableConnection(final String connectionToken, final Long institutionId) {
|
public Result<ClientConnection> disableConnection(final String connectionToken, final Long institutionId) {
|
||||||
return Result.tryCatch(() -> {
|
return Result.tryCatch(() -> {
|
||||||
|
@ -390,9 +398,7 @@ public class SebClientConnectionServiceImpl implements SebClientConnectionServic
|
||||||
.getOrThrow();
|
.getOrThrow();
|
||||||
|
|
||||||
ClientConnection updatedClientConnection;
|
ClientConnection updatedClientConnection;
|
||||||
if (clientConnection.status == ConnectionStatus.CONNECTION_REQUESTED ||
|
if (DISABLE_STATE_PREDICATE.test(clientConnection)) {
|
||||||
clientConnection.status == ConnectionStatus.UNDEFINED ||
|
|
||||||
clientConnection.status == ConnectionStatus.AUTHENTICATED) {
|
|
||||||
|
|
||||||
updatedClientConnection = saveInState(
|
updatedClientConnection = saveInState(
|
||||||
clientConnection,
|
clientConnection,
|
||||||
|
|
|
@ -231,7 +231,8 @@ public class ExamMonitoringController {
|
||||||
required = true,
|
required = true,
|
||||||
defaultValue = UserService.USERS_INSTITUTION_AS_DEFAULT) final Long institutionId,
|
defaultValue = UserService.USERS_INSTITUTION_AS_DEFAULT) final Long institutionId,
|
||||||
@PathVariable(name = API.PARAM_MODEL_ID, required = true) final Long examId,
|
@PathVariable(name = API.PARAM_MODEL_ID, required = true) final Long examId,
|
||||||
@PathVariable(name = Domain.CLIENT_CONNECTION.ATTR_CONNECTION_TOKEN,
|
@RequestParam(
|
||||||
|
name = Domain.CLIENT_CONNECTION.ATTR_CONNECTION_TOKEN,
|
||||||
required = true) final String connectionToken) {
|
required = true) final String connectionToken) {
|
||||||
|
|
||||||
if (connectionToken.contains(Constants.LIST_SEPARATOR)) {
|
if (connectionToken.contains(Constants.LIST_SEPARATOR)) {
|
||||||
|
@ -242,7 +243,8 @@ public class ExamMonitoringController {
|
||||||
.onError(error -> log.error("Failed to disable SEB client connection: {}", token));
|
.onError(error -> log.error("Failed to disable SEB client connection: {}", token));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
this.sebClientConnectionService.disableConnection(connectionToken, institutionId)
|
this.sebClientConnectionService
|
||||||
|
.disableConnection(connectionToken, institutionId)
|
||||||
.getOrThrow();
|
.getOrThrow();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1065,12 +1065,14 @@ sebserver.monitoring.connection.list.column.vdiAddress=IP Address (VDI)
|
||||||
sebserver.monitoring.exam.connection.emptySelection=Please select a connection from the list
|
sebserver.monitoring.exam.connection.emptySelection=Please select a connection from the list
|
||||||
sebserver.monitoring.exam.connection.title=SEB Client Connection
|
sebserver.monitoring.exam.connection.title=SEB Client Connection
|
||||||
sebserver.monitoring.exam.connection.list.actions=Selected Connection
|
sebserver.monitoring.exam.connection.list.actions=Selected Connection
|
||||||
sebserver.monitoring.exam.connection.action.view=View Details
|
sebserver.monitoring.exam.connection.action.view=Back To Monitoring
|
||||||
sebserver.monitoring.exam.connection.action.instruction.quit=Send SEB Quit
|
sebserver.monitoring.exam.connection.action.instruction.quit=Send SEB Quit
|
||||||
sebserver.monitoring.exam.connection.action.instruction.quit.all=Send SEB Quit
|
sebserver.monitoring.exam.connection.action.instruction.quit.all=Send SEB Quit
|
||||||
sebserver.monitoring.exam.connection.action.instruction.quit.all.confirm=Are you sure to quit this SEB client connection?
|
sebserver.monitoring.exam.connection.action.instruction.quit.confirm=Are you sure to quit this SEB client connection?
|
||||||
sebserver.monitoring.exam.connection.action.instruction.quit.selected.confirm=Are you sure to quit all selected, active SEB client connections?
|
sebserver.monitoring.exam.connection.action.instruction.quit.selected.confirm=Are you sure to quit all selected, active SEB client connections?
|
||||||
sebserver.monitoring.exam.connection.action.instruction.quit.all.confirm=Are you sure to quit all active SEB client connections?
|
sebserver.monitoring.exam.connection.action.instruction.quit.all.confirm=Are you sure to quit all active SEB client connections?
|
||||||
|
sebserver.monitoring.exam.connection.action.instruction.disable.selected.confirm=Are you sure to disable all selected SEB client connections?
|
||||||
|
sebserver.monitoring.exam.connection.action.instruction.disable.all.confirm=Are you sure to disable all active SEB client connections?
|
||||||
sebserver.monitoring.exam.connection.action.disable=Mark As Disabled
|
sebserver.monitoring.exam.connection.action.disable=Mark As Disabled
|
||||||
sebserver.monitoring.exam.connection.action.hide.requested=Hide Requested
|
sebserver.monitoring.exam.connection.action.hide.requested=Hide Requested
|
||||||
sebserver.monitoring.exam.connection.action.show.requested=Show Requested
|
sebserver.monitoring.exam.connection.action.show.requested=Show Requested
|
||||||
|
|
BIN
src/main/resources/static/images/help.png
Normal file
BIN
src/main/resources/static/images/help.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 643 B |
Loading…
Reference in a new issue