Jakub Niedermertl has uploaded a new change for review.

Change subject: VM Icons - editor
......................................................................

VM Icons - editor

Change-Id: Ibd5e8c120282e35d5940a66286d600b128c57753
Signed-off-by: Jakub Niedermertl <jnied...@redhat.com>
---
M 
backend/manager/modules/common/src/main/java/org/ovirt/engine/core/common/action/VmManagementParametersBase.java
M 
backend/manager/modules/common/src/main/java/org/ovirt/engine/core/common/queries/OsQueryParameters.java
M 
frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/CommonApplicationConstants.java
M 
frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/CommonApplicationTemplates.java
M 
frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/gin/CommonGinUiBinderWidgets.java
M 
frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/HasAccess.java
A 
frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/editor/CompositeHandlerRegistration.java
A 
frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/editor/IconEditorWidget.java
A 
frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/editor/IconEditorWidget.ui.xml
M 
frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/uicommon/popup/AbstractVmPopupWidget.java
M 
frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/uicommon/popup/AbstractVmPopupWidget.ui.xml
M 
frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/Linq.java
M 
frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/TabName.java
A 
frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/vms/IconWithDefault.java
M 
frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/vms/UnitVmModel.java
A 
frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/IconValidation.java
A 
frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/IconWithDefaultValidation.java
M 
frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/ValidationResult.java
M 
frontend/webadmin/modules/uicompat/src/main/java/org/ovirt/engine/ui/uicompat/UIConstants.java
M 
frontend/webadmin/modules/uicompat/src/main/java/org/ovirt/engine/ui/uicompat/UIMessages.java
20 files changed, 830 insertions(+), 9 deletions(-)


  git pull ssh://gerrit.ovirt.org:29418/ovirt-engine refs/changes/85/37985/1

diff --git 
a/backend/manager/modules/common/src/main/java/org/ovirt/engine/core/common/action/VmManagementParametersBase.java
 
b/backend/manager/modules/common/src/main/java/org/ovirt/engine/core/common/action/VmManagementParametersBase.java
index bc9d9a8..c02e05e 100644
--- 
a/backend/manager/modules/common/src/main/java/org/ovirt/engine/core/common/action/VmManagementParametersBase.java
+++ 
b/backend/manager/modules/common/src/main/java/org/ovirt/engine/core/common/action/VmManagementParametersBase.java
@@ -34,6 +34,7 @@
     private boolean copyTemplatePermissions;
     private boolean applyChangesLater;
     private boolean updateNuma;
+    // TODO jakub add private String vmIcon;
 
     /*
      * This parameter is needed at update to make sure that when we get a null 
watchdog from rest-api it is not meant to
diff --git 
a/backend/manager/modules/common/src/main/java/org/ovirt/engine/core/common/queries/OsQueryParameters.java
 
b/backend/manager/modules/common/src/main/java/org/ovirt/engine/core/common/queries/OsQueryParameters.java
index 4a807f3..0dab93a 100644
--- 
a/backend/manager/modules/common/src/main/java/org/ovirt/engine/core/common/queries/OsQueryParameters.java
+++ 
b/backend/manager/modules/common/src/main/java/org/ovirt/engine/core/common/queries/OsQueryParameters.java
@@ -57,5 +57,6 @@
         GetOsNames,
         GetOsArchitectures,
         GetDefaultOSes
+        // TODO jakub add 'GetOsIcons'
     }
 }
diff --git 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/CommonApplicationConstants.java
 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/CommonApplicationConstants.java
index 6ede749..7123c41 100644
--- 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/CommonApplicationConstants.java
+++ 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/CommonApplicationConstants.java
@@ -748,6 +748,27 @@
     @DefaultStringValue("Clone Name")
     String clonedVmName();
 
+    @DefaultStringValue("Icon")
+    String iconTabVmPopup();
+
+    @DefaultStringValue("New Icon")
+    String newIconVmPopup();
+
+    @DefaultStringValue("Current Icon")
+    String currentIconVmPopup();
+
+    @DefaultStringValue("Upload")
+    String uploadIconVmPopup();
+
+    @DefaultStringValue("Use default")
+    String useDefaultIconVmPopup();
+
+    @DefaultStringValue("Discard changes")
+    String discardChangesIconVmPopup();
+
+    @DefaultStringValue("Icons limitations: max dimensions: width 150px, 
heigth 120px; max size 24kB; supported formats: jpg, png, gif")
+    String iconLimitationsIconVmPopup();
+
     // Permissions
     @DefaultStringValue("Inherited Permission")
     String inheritedPermission();
diff --git 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/CommonApplicationTemplates.java
 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/CommonApplicationTemplates.java
index fb94750..f054bcd 100644
--- 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/CommonApplicationTemplates.java
+++ 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/CommonApplicationTemplates.java
@@ -47,9 +47,15 @@
     @Template("<ul style='margin-top:0'>{0}</ul>")
     SafeHtml unsignedList(SafeHtml list);
 
+    @Template("<ul>{0}</ul>")
+    SafeHtml unorderedList(SafeHtml items);
+
     @Template("<li>{0}</li>")
     SafeHtml listItem(SafeHtml item);
 
+    @Template("<li>{0}</li>")
+    SafeHtml listItem(String item);
+
     @Template("{0} <sub>{1}</sub>")
     SafeHtml sub(String main, String sub);
 
diff --git 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/gin/CommonGinUiBinderWidgets.java
 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/gin/CommonGinUiBinderWidgets.java
index 0d0c7ed..19f68cb 100644
--- 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/gin/CommonGinUiBinderWidgets.java
+++ 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/gin/CommonGinUiBinderWidgets.java
@@ -1,25 +1,29 @@
 package org.ovirt.engine.ui.common.gin;
 
 import com.google.gwt.inject.client.Ginjector;
+import org.ovirt.engine.ui.common.widget.editor.IconEditorWidget;
 import 
org.ovirt.engine.ui.common.widget.uicommon.popup.vm.SerialNumberPolicyWidget;
 
 /**
  * Ginjector extension containing views that make use of the GWTP GinUiBinder.
  *
  * Every view that makes use of GIN dependency injection and is to be embedded 
in a .ui.xml file --ex:
+ * <pre>
  * <code>
- *     <w:MyWidget />
+ *     &lt;w:MyWidget />
  *
  *     public class MyWidget {
- *         @Inject
+ *         {@literal @}Inject
  *         public MyWidget(ApplicationResources resources, MyOtherDependency 
dep) {
  *             // ...
  *         }
  *     }
  * </code>
+ * </pre>
  *
  * Must be registered in this interface.
  */
 public interface CommonGinUiBinderWidgets extends Ginjector {
     SerialNumberPolicyWidget getSerialNumberPolicyWidget();
+    IconEditorWidget getIconEditorWidget();
 }
diff --git 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/HasAccess.java
 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/HasAccess.java
index df4eb9b..71512b7 100644
--- 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/HasAccess.java
+++ 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/HasAccess.java
@@ -2,6 +2,10 @@
 
 /**
  * Widgets that implement this interface have a user access policy associated 
with them.
+ * <p>
+ *     Inaccessible widgets should be hidden. Usually using
+ *     {@link com.google.gwt.user.client.ui.Widget#setVisible(boolean)}
+ * </p>
  */
 public interface HasAccess {
 
diff --git 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/editor/CompositeHandlerRegistration.java
 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/editor/CompositeHandlerRegistration.java
new file mode 100644
index 0000000..fcc8b6d
--- /dev/null
+++ 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/editor/CompositeHandlerRegistration.java
@@ -0,0 +1,26 @@
+package org.ovirt.engine.ui.common.widget.editor;
+
+import com.google.gwt.event.shared.HandlerRegistration;
+
+import java.util.Arrays;
+import java.util.List;
+
+public class CompositeHandlerRegistration implements HandlerRegistration {
+
+    private final List<HandlerRegistration> registrations;
+
+    private CompositeHandlerRegistration(List<HandlerRegistration> 
registrations) {
+        this.registrations = registrations;
+    }
+
+    public static CompositeHandlerRegistration of(HandlerRegistration... 
registrations) {
+        return new CompositeHandlerRegistration(Arrays.asList(registrations));
+    }
+
+    @Override
+    public void removeHandler() {
+        for(HandlerRegistration registration : registrations) {
+            registration.removeHandler();
+        }
+    }
+}
diff --git 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/editor/IconEditorWidget.java
 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/editor/IconEditorWidget.java
new file mode 100644
index 0000000..04dd68f
--- /dev/null
+++ 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/editor/IconEditorWidget.java
@@ -0,0 +1,349 @@
+package org.ovirt.engine.ui.common.widget.editor;
+
+import com.google.gwt.core.client.GWT;
+import com.google.gwt.dom.client.Element;
+import com.google.gwt.editor.client.LeafValueEditor;
+import com.google.gwt.event.dom.client.ChangeEvent;
+import com.google.gwt.event.dom.client.ChangeHandler;
+import com.google.gwt.event.dom.client.ClickEvent;
+import com.google.gwt.event.dom.client.KeyCodes;
+import com.google.gwt.event.dom.client.KeyDownHandler;
+import com.google.gwt.event.dom.client.KeyPressEvent;
+import com.google.gwt.event.dom.client.KeyPressHandler;
+import com.google.gwt.event.dom.client.KeyUpHandler;
+import com.google.gwt.event.logical.shared.HasValueChangeHandlers;
+import com.google.gwt.event.logical.shared.ValueChangeEvent;
+import com.google.gwt.event.logical.shared.ValueChangeHandler;
+import com.google.gwt.event.shared.HandlerRegistration;
+import com.google.gwt.resources.client.CssResource;
+import com.google.gwt.safehtml.shared.SafeHtml;
+import com.google.gwt.safehtml.shared.SafeHtmlBuilder;
+import com.google.gwt.safehtml.shared.SafeHtmlUtils;
+import com.google.gwt.uibinder.client.UiBinder;
+import com.google.gwt.uibinder.client.UiField;
+import com.google.gwt.uibinder.client.UiHandler;
+import com.google.gwt.user.client.ui.Button;
+import com.google.gwt.user.client.ui.FileUpload;
+import com.google.gwt.user.client.ui.HTML;
+import com.google.gwt.user.client.ui.HTMLPanel;
+import com.google.gwt.user.client.ui.Image;
+import com.google.gwt.user.client.ui.UIObject;
+import com.google.gwt.user.client.ui.Widget;
+import com.google.inject.Inject;
+import org.ovirt.engine.ui.common.CommonApplicationConstants;
+import org.ovirt.engine.ui.common.CommonApplicationResources;
+import org.ovirt.engine.ui.common.CommonApplicationTemplates;
+import org.ovirt.engine.ui.common.editor.UiCommonEditor;
+import org.ovirt.engine.ui.common.widget.AbstractValidatedWidget;
+import org.ovirt.engine.ui.common.widget.dialog.InfoIcon;
+import org.ovirt.engine.ui.uicommonweb.models.vms.IconWithDefault;
+import org.ovirt.engine.ui.uicommonweb.validation.IconValidation;
+import org.ovirt.engine.ui.uicommonweb.validation.ValidationResult;
+
+import java.util.List;
+
+/**
+ * Icon editor. It allows to set custom icon to VM-like entities.
+ */
+public class IconEditorWidget extends AbstractValidatedWidget
+        implements LeafValueEditor<IconWithDefault>,
+                   HasValueChangeHandlers<IconWithDefault>,
+                   UiCommonEditor<IconWithDefault> {
+
+    interface ViewUiBinder extends UiBinder<HTMLPanel, IconEditorWidget> {
+        ViewUiBinder uiBinder = GWT.create(ViewUiBinder.class);
+    }
+
+    protected interface Style extends CssResource {
+        String grey();
+        String iconImageDisabled();
+    }
+
+    @UiField
+    protected Style style;
+
+    @UiField
+    protected Image image;
+
+    @UiField
+    protected Button uploadButton;
+
+    @UiField(provided = true)
+    protected InfoIcon uploadInfoIcon;
+
+    @UiField
+    protected FileUpload fileUpload;
+
+    @UiField
+    protected Button defaultButton;
+
+    @UiField
+    protected HTML errorMessageHtml;
+
+    protected final CommonApplicationTemplates templates;
+
+    /**
+     * current value, the visible image <br/>
+     * in dataUri format
+     */
+    private String icon;
+
+    /**
+     * default value (given by OS of VM) <br/>
+     * in dataUri format
+     */
+    private String defaultIcon;
+
+    /**
+     * relates to {@link com.google.gwt.user.client.ui.HasEnabled} 
implementation
+     */
+    private boolean enabled;
+
+    /**
+     * relates to {@link org.ovirt.engine.ui.common.widget.HasAccess} 
implementation
+     */
+    private boolean accessible;
+
+    @Inject
+    public IconEditorWidget(CommonApplicationConstants constants,
+                            CommonApplicationTemplates templates,
+                            CommonApplicationResources resources) {
+        this.templates = templates;
+        uploadInfoIcon = new InfoIcon(
+                
SafeHtmlUtils.fromTrustedString(constants.iconLimitationsIconVmPopup()), 
resources);
+        initWidget(ViewUiBinder.uiBinder.createAndBindUi(this));
+        fileUpload.getElement().setAttribute("accept", 
"image/gif,image/jpeg,image/png"); //$NON-NLS-1$ //$NON-NLS-2$
+        fileUpload.addChangeHandler(new ChangeHandler() {
+            @Override
+            public void onChange(ChangeEvent event) {
+                IconEditorWidget.this.readUploadedIconFile();
+            }
+        });
+        fileUpload.getElement().setTabIndex(-1); // can be moved to *.ui.xml 
file in form of attribute `tabIndex="-1"` since GWT 2.7.0
+
+        KeyPressHandler preventEnterKeyPressHandler = 
createPreventEnterKeyPressHandler();
+        uploadButton.addKeyPressHandler(preventEnterKeyPressHandler);
+        defaultButton.addKeyPressHandler(preventEnterKeyPressHandler);
+
+        setEnabled(true);
+        setAccessible(true);
+    }
+
+    private KeyPressHandler createPreventEnterKeyPressHandler() {
+        return new KeyPressHandler() {
+            @Override
+            public void onKeyPress(KeyPressEvent event) {
+                if (!event.isAnyModifierKeyDown()
+                        && event.getNativeEvent().getKeyCode() == 
KeyCodes.KEY_ENTER
+                        && event.getUnicodeCharCode() == 0) {
+                    event.preventDefault();
+                }
+            }
+        };
+    }
+
+    @Override
+    public void setValue(IconWithDefault value) {
+        setIcon(value.getIcon());
+        defaultIcon = value.getDefaultIcon();
+    }
+
+    @Override
+    public IconWithDefault getValue() {
+        return new IconWithDefault(icon, defaultIcon);
+    }
+
+    @Override
+    protected Widget getValidatedWidget() {
+        return this;
+    }
+
+    @UiHandler("uploadButton")
+    void onUploadIconButton(ClickEvent event) {
+        clickElement(fileUpload.getElement());
+    }
+
+    @UiHandler("defaultButton")
+    void onDefaultIconButton(ClickEvent event) {
+        setIcon(defaultIcon);
+    }
+
+    /**
+     * There is FileUpload#click() method since GWT 2.7.0.
+     */
+    native void clickElement(Element element) /*-{
+        element.click();
+    }-*/;
+
+    protected void setIcon(String icon) {
+        final String oldIcon = this.icon;
+        this.icon = icon;
+        final ValidationResult validation = (new 
IconValidation()).validate(icon);
+        updateErrorIconLabel(validation);
+        image.getElement().setAttribute("src", icon); //$NON-NLS-1$
+        ValueChangeEvent.fireIfNotEqual(this,
+                new IconWithDefault(oldIcon, defaultIcon),
+                new IconWithDefault(icon, defaultIcon));
+    }
+
+    private void updateErrorIconLabel(ValidationResult validation) {
+        if (!validation.getSuccess() && validation.getReasons().isEmpty()) {
+            throw new IllegalArgumentException("Unsuccessful validation 
without any reason not allowed."); //$NON-NLS-1$
+        }
+        updateErrorIconLabel(validation.getReasons());
+    }
+
+    private void updateErrorIconLabel(List<String> reasons) {
+        if (reasons.isEmpty()) {
+            errorMessageHtml.setHTML(SafeHtmlUtils.EMPTY_SAFE_HTML);
+        } else {
+            final SafeHtml htmlReasons = toUnorderedList(reasons);
+            errorMessageHtml.setHTML(htmlReasons);
+        }
+    }
+
+    private SafeHtml toUnorderedList(List<String> stringItems) {
+        SafeHtmlBuilder builder = new SafeHtmlBuilder();
+        for (String stringItem : stringItems) {
+            builder.append(templates.listItem(stringItem));
+        }
+        return templates.unorderedList(builder.toSafeHtml());
+    }
+
+    native void readUploadedIconFile() /*-{
+        var inputFileElement = 
th...@org.ovirt.engine.ui.common.widget.editor.IconEditorWidget::fileuplo...@com.google.gwt.user.client.ui.FileUpload::getElement()();
+        var self = this;
+        var javaCallback = $entry(function (dataUri) {
+            return 
se...@org.ovirt.engine.ui.common.widget.editor.IconEditorWidget::setIcon(Ljava/lang/String;)(dataUri);
+        });
+        if (inputFileElement.files.length > 0) {
+            var file = inputFileElement.files[0];
+            var fileReader = new FileReader();
+            fileReader.onload = onFileRead;
+            fileReader.readAsDataURL(file);
+        }
+
+        function onFileRead(event) {
+            var iconDataUri = event.target.result;
+            javaCallback(iconDataUri);
+        }
+    }-*/;
+
+    native void preventDefaultEnterBehavior(Element element) /*-{
+        element.addEvent
+    }-*/;
+
+
+    // see com.google.gwt.user.client.ui.ValueListBox.addValueChangeHandler()
+    @Override
+    public HandlerRegistration 
addValueChangeHandler(ValueChangeHandler<IconWithDefault> handler) {
+        return this.addHandler(handler, ValueChangeEvent.getType());
+    }
+
+    @Override
+    public void markAsValid() {
+        super.markAsValid();
+        getValidatedWidgetStyle().setBorderColor("transparent"); //$NON-NLS-1$
+    }
+
+    @Override
+    public void markAsInvalid(List<String> validationHints) {
+        super.markAsInvalid(validationHints);
+        updateErrorIconLabel(validationHints);
+    }
+
+    @Override
+    public LeafValueEditor<IconWithDefault> getActualEditor() {
+        return this;
+    }
+
+    @Override
+    public int getTabIndex() {
+        return uploadButton.getTabIndex();
+    }
+
+    @Override
+    public void setAccessKey(char key) {
+        uploadButton.setAccessKey(key);
+    }
+
+    @Override
+    public void setFocus(boolean focused) {
+        uploadButton.setFocus(focused);
+    }
+
+    @Override
+    public void setTabIndex(int index) {
+        uploadButton.setTabIndex(index);
+        defaultButton.setTabIndex(index);
+    }
+
+    @Override
+    public void disable(String disabilityHint) {
+        setEnabled(false, disabilityHint);
+    }
+
+    @Override
+    public boolean isEnabled() {
+        return enabled;
+    }
+
+    @Override
+    public void setEnabled(boolean enabled) {
+        setEnabled(true, ""); //$NON-NLS-1$
+    }
+
+    protected void setEnabled(boolean enabled, String hint) {
+        this.enabled = enabled;
+        uploadButton.setEnabled(enabled);
+        uploadButton.setTitle(hint);
+        defaultButton.setEnabled(enabled);
+        defaultButton.setTitle(hint);
+        ensureStyleNamePresent(errorMessageHtml, !enabled, style.grey());
+        errorMessageHtml.setTitle(hint);
+        ensureStyleNamePresent(image, !enabled, style.iconImageDisabled());
+        image.setTitle(hint);
+    }
+
+    private static void ensureStyleNamePresent(UIObject object, boolean 
styleNameExists, String styleName) {
+        if (styleNameExists) {
+            object.addStyleName(styleName);
+        } else {
+            object.removeStyleName(styleName);
+        }
+    }
+
+    @Override
+    public boolean isAccessible() {
+        return accessible;
+    }
+
+    @Override
+    public void setAccessible(boolean accessible) {
+        this.accessible = accessible;
+        setVisible(accessible);
+    }
+
+    @Override
+    public HandlerRegistration addKeyDownHandler(KeyDownHandler handler) {
+        return CompositeHandlerRegistration.of(
+                uploadButton.addKeyDownHandler(handler),
+                defaultButton.addKeyDownHandler(handler));
+    }
+
+    @Override
+    public HandlerRegistration addKeyPressHandler(KeyPressHandler handler) {
+        return CompositeHandlerRegistration.of(
+                uploadButton.addKeyPressHandler(handler),
+                defaultButton.addKeyPressHandler(handler));
+    }
+
+    @Override
+    public HandlerRegistration addKeyUpHandler(KeyUpHandler handler) {
+        return CompositeHandlerRegistration.of(
+                uploadButton.addKeyUpHandler(handler),
+                defaultButton.addKeyUpHandler(handler));
+    }
+
+
+}
diff --git 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/editor/IconEditorWidget.ui.xml
 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/editor/IconEditorWidget.ui.xml
new file mode 100644
index 0000000..ec7142f
--- /dev/null
+++ 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/editor/IconEditorWidget.ui.xml
@@ -0,0 +1,90 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!DOCTYPE ui:UiBinder SYSTEM "http://dl.google.com/gwt/DTD/xhtml.ent";>
+<ui:UiBinder xmlns:ui="urn:ui:com.google.gwt.uibinder"
+             xmlns:g="urn:import:com.google.gwt.user.client.ui"
+             xmlns:d="urn:import:org.ovirt.engine.ui.common.widget.dialog">
+
+    <ui:with field='resources' 
type='org.ovirt.engine.ui.common.CommonApplicationResources'/>
+    <ui:with field='constants' 
type='org.ovirt.engine.ui.common.CommonApplicationConstants'/>
+
+    <ui:style 
type="org.ovirt.engine.ui.common.widget.editor.IconEditorWidget.Style">
+
+        .iconImage {
+            display: block;
+            height: 120px;
+            width: 150px;
+            border: thin solid rgb(211, 211, 211);
+        }
+
+        .iconImageDisabled {
+            opacity: 0.4;
+        }
+
+        .horizontal-spacing > tbody > tr > td {
+            padding-left: 10px;
+        }
+
+        .inline-block {
+            display: inline-block
+        }
+
+        .iconButton {
+            margin: 0px 0px 10px;
+            min-width: 110px;
+        }
+
+        .iconButton:focus {
+            outline: thin black dotted;
+        }
+
+        .hidden {
+            display: none;
+        }
+
+        .grey {
+            color: grey;
+        }
+
+        .iconInfoIcon {
+            margin: 0px 7px;
+        }
+
+        .iconErrorHtml {
+            position: relative;
+            bottom: 0px;
+            left: 0px;
+            margin: 10px;
+        }
+
+        .iconErrorHtml ul {
+            padding-left: 13px;
+        }
+
+        .no-border {
+            border: none;
+        }
+
+    </ui:style>
+
+
+    <g:HTMLPanel addStyleNames="{style.no-border}">
+        <g:HorizontalPanel addStyleNames="{style.horizontal-spacing}">
+            <g:Image ui:field="image" addStyleNames="{style.iconImage}"/>
+            <g:VerticalPanel>
+                <g:FlowPanel>
+                    <g:Button ui:field="uploadButton" 
text="{constants.uploadIconVmPopup}"
+                              addStyleNames="{style.inline-block} 
{style.iconButton}"/>
+                    <d:InfoIcon ui:field="uploadInfoIcon"
+                                addStyleNames="{style.inline-block} 
{style.iconInfoIcon}"/>
+                    <g:HTMLPanel addStyleNames="{style.hidden}">
+                        <g:FileUpload ui:field="fileUpload"/>
+                    </g:HTMLPanel>
+                </g:FlowPanel>
+                <g:Button ui:field="defaultButton" 
text="{constants.useDefaultIconVmPopup}"
+                          addStyleNames="{style.iconButton}"/>
+            </g:VerticalPanel>
+        </g:HorizontalPanel>
+        <g:HTML ui:field="errorMessageHtml" 
addStyleNames="{style.iconErrorHtml}"/>
+    </g:HTMLPanel>
+
+</ui:UiBinder>
diff --git 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/uicommon/popup/AbstractVmPopupWidget.java
 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/uicommon/popup/AbstractVmPopupWidget.java
index d3b2867..16319c4 100644
--- 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/uicommon/popup/AbstractVmPopupWidget.java
+++ 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/uicommon/popup/AbstractVmPopupWidget.java
@@ -27,6 +27,7 @@
 import com.google.gwt.user.client.ui.RadioButton;
 import com.google.gwt.user.client.ui.ValueLabel;
 import com.google.gwt.user.client.ui.Widget;
+
 import java.util.ArrayList;
 import java.util.Arrays;
 import java.util.HashMap;
@@ -71,6 +72,7 @@
 import org.ovirt.engine.ui.common.widget.dialog.tab.DialogTab;
 import org.ovirt.engine.ui.common.widget.dialog.tab.DialogTabPanel;
 import org.ovirt.engine.ui.common.widget.editor.EntityModelCellTable;
+import org.ovirt.engine.ui.common.widget.editor.IconEditorWidget;
 import org.ovirt.engine.ui.common.widget.editor.ListModelListBoxEditor;
 import org.ovirt.engine.ui.common.widget.editor.ListModelListBoxOnlyEditor;
 import 
org.ovirt.engine.ui.common.widget.editor.ListModelTypeAheadChangeableListBoxEditor;
@@ -864,6 +866,15 @@
     @Ignore
     protected DialogTabPanel mainTabPanel;
 
+    // ==Icon Tab==
+    @UiField
+    @Ignore
+    protected DialogTab iconTab;
+
+    @UiField
+    @Path("icon.entity")
+    protected IconEditorWidget iconEditorWidget;
+
     private UnitVmModel unitVmModel;
 
     private final Driver driver = GWT.create(Driver.class);
@@ -993,6 +1004,7 @@
         getTabNameMapping().put(TabName.POOL_TAB, this.poolTab);
         getTabNameMapping().put(TabName.RESOURCE_ALLOCATION_TAB, 
this.resourceAllocationTab);
         getTabNameMapping().put(TabName.SYSTEM_TAB, this.systemTab);
+        getTabNameMapping().put(TabName.ICON_TAB, this.iconTab);
     }
 
     private void initDetachableFields() {
@@ -1501,6 +1513,9 @@
         numOfSocketsEditorWithDetachable.setLabel(constants.numOfSockets());
         emulatedMachine.setLabel(constants.emulatedMachineLabel());
         customCpu.setLabel(constants.cpuModelLabel());
+
+        // Icon tab
+        iconTab.setLabel(constants.iconTabVmPopup());
     }
 
     protected void applyStyles() {
@@ -2004,6 +2019,10 @@
         // ==Custom Properties Tab==
         nextTabIndex = customPropertiesTab.setTabIndexes(nextTabIndex);
 
+        // ==Icon Tab==
+        nextTabIndex = iconTab.setTabIndexes(nextTabIndex);
+        iconEditorWidget.setTabIndex(nextTabIndex++);
+
         return nextTabIndex;
     }
 
@@ -2063,7 +2082,8 @@
                 rngDeviceTab,
                 highAvailabilityTab,
                 poolTab,
-                systemTab);
+                systemTab,
+                iconTab);
     }
 
     protected List<Widget> adancedFieldsFromGeneralTab() {
@@ -2119,7 +2139,8 @@
                 highAvailabilityTab,
                 resourceAllocationTab,
                 customPropertiesTab,
-                rngDeviceTab
+                rngDeviceTab,
+                iconTab
         );
     }
 
@@ -2151,7 +2172,8 @@
             bootOptionsTab,
             customPropertiesTab,
             systemTab,
-            rngDeviceTab
+            rngDeviceTab,
+            iconTab
         );
     }
 
diff --git 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/uicommon/popup/AbstractVmPopupWidget.ui.xml
 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/uicommon/popup/AbstractVmPopupWidget.ui.xml
index c6c8901..ae89f1f 100644
--- 
a/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/uicommon/popup/AbstractVmPopupWidget.ui.xml
+++ 
b/frontend/webadmin/modules/gwt-common/src/main/java/org/ovirt/engine/ui/common/widget/uicommon/popup/AbstractVmPopupWidget.ui.xml
@@ -426,6 +426,42 @@
         .overrideMigrationDowntimeContentEditor {
             width: 20px;
         }
+
+        .iconImage {
+            height: 120px;
+            width: 150px;
+            border: thin solid rgb(211, 211, 211);
+            display: block;
+        }
+
+        .horizontal-spacing > tbody > tr > td {
+            margin-left: 10px;
+        }
+
+        .inline-block {
+            display: inline-block
+        }
+
+        .iconButton {
+            margin: 0px 0px 5px;
+            min-width: 110px;
+        }
+
+        .hidden {
+            display: none;
+        }
+
+        .iconInfoIcon {
+            margin: 0px 7px;
+        }
+
+        .iconErrorLabel {
+            position: relative;
+            bottom: 0px;
+            left: 0px;
+            margin: 10px;
+        }
+
     </ui:style>
 
     <t:DialogTabPanel width="100%" height="100%" ui:field="mainTabPanel">
@@ -784,6 +820,13 @@
                 </t:content>
             </t:DialogTab>
         </t:tab>
+        <t:tab>
+            <t:DialogTab ui:field="iconTab">
+                <t:content>
+                    <e:IconEditorWidget ui:field="iconEditorWidget" />
+                </t:content>
+            </t:DialogTab>
+        </t:tab>
     </t:DialogTabPanel>
 
 </ui:UiBinder>
diff --git 
a/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/Linq.java
 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/Linq.java
index f330cda..30a8eef 100644
--- 
a/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/Linq.java
+++ 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/Linq.java
@@ -770,6 +770,10 @@
         return result;
     }
 
+    public static <T> List<T> concatTypesafe(List<T>... lists) {
+        return concat(lists);
+    }
+
     public static <T> ArrayList<T> union(ArrayList<ArrayList<T>> lists)
     {
         HashSet<T> set = new HashSet<T>();
diff --git 
a/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/TabName.java
 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/TabName.java
index ccbd4d9..02e81b1 100644
--- 
a/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/TabName.java
+++ 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/TabName.java
@@ -24,6 +24,7 @@
     RESOURCE_ALLOCATION_TAB,
     SYSTEM_TAB,
     TAB_RNG,
-    FIRST_RUN;
+    FIRST_RUN,
+    ICON_TAB;
 
 }
diff --git 
a/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/vms/IconWithDefault.java
 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/vms/IconWithDefault.java
new file mode 100644
index 0000000..fcb9731
--- /dev/null
+++ 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/vms/IconWithDefault.java
@@ -0,0 +1,66 @@
+package org.ovirt.engine.ui.uicommonweb.models.vms;
+
+/**
+ * Bean holding two icons in dataUri format.
+ *
+ * <p>
+ * {@link #icon} carries data in both directions between {@link UnitVmModel} 
and
+ * {@link IconEditor}
+ * </p>
+ * <p>
+ *     {@link #defaultIcon} is used to one-direction communication {@link 
UnitVmModel} -> {@link IconEditor}
+ * </p>
+ */
+public class IconWithDefault {
+
+    /**
+     * Icon - this value changes over time during model-view communication
+     */
+    private final String icon;
+
+    /**
+     * Default (VM OS based) icon. It allows user to reset custom icon 
settings.
+     */
+    private final String defaultIcon;
+
+    public IconWithDefault(String icon, String defaultIcon) {
+        this.icon = icon;
+        this.defaultIcon = defaultIcon;
+    }
+
+    public String getIcon() {
+        return icon;
+    }
+
+    public String getDefaultIcon() {
+        return defaultIcon;
+    }
+
+    @Override
+    public boolean equals(Object o) {
+        if (this == o) return true;
+        if (!(o instanceof IconWithDefault)) return false;
+
+        IconWithDefault that = (IconWithDefault) o;
+
+        if (defaultIcon != null ? !defaultIcon.equals(that.defaultIcon) : 
that.defaultIcon != null) return false;
+        if (icon != null ? !icon.equals(that.icon) : that.icon != null) return 
false;
+
+        return true;
+    }
+
+    @Override
+    public int hashCode() {
+        int result = icon != null ? icon.hashCode() : 0;
+        result = 31 * result + (defaultIcon != null ? defaultIcon.hashCode() : 
0);
+        return result;
+    }
+
+    @Override
+    public String toString() {
+        return "IconWithDefault{" + //$NON-NLS-1$
+                "icon='" + icon.substring(0, 30) + "…'" + //$NON-NLS-1$ 
//$NON-NLS-2$
+                ", defaultIcon='" + defaultIcon.substring(0, 30) + "'" + 
//$NON-NLS-1$ //$NON-NLS-2$
+                '}';
+    }
+}
diff --git 
a/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/vms/UnitVmModel.java
 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/vms/UnitVmModel.java
index d1cef78..360558f 100644
--- 
a/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/vms/UnitVmModel.java
+++ 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/models/vms/UnitVmModel.java
@@ -67,6 +67,7 @@
 import 
org.ovirt.engine.ui.uicommonweb.validation.I18NExtraNameOrNoneValidation;
 import org.ovirt.engine.ui.uicommonweb.validation.I18NNameValidation;
 import org.ovirt.engine.ui.uicommonweb.validation.IValidation;
+import org.ovirt.engine.ui.uicommonweb.validation.IconWithDefaultValidation;
 import org.ovirt.engine.ui.uicommonweb.validation.IntegerValidation;
 import org.ovirt.engine.ui.uicommonweb.validation.LengthValidation;
 import 
org.ovirt.engine.ui.uicommonweb.validation.NoTrimmingWhitespacesValidation;
@@ -200,6 +201,20 @@
         this.nicsWithLogicalNetworks = nicsWithLogicalNetworks;
     }
 
+    //TODO jakub add initialization, validation, saving
+    /**
+     * VM icon
+     */
+    private EntityModel<IconWithDefault> icon;
+
+    public EntityModel<IconWithDefault> getIcon() {
+        return icon;
+    }
+
+    public void setIcon(EntityModel<IconWithDefault> icon) {
+        this.icon = icon;
+    }
+
     /**
      * Note: We assume that this method is called only once, on the creation 
stage of the model. if this assumption is
      * changed (i.e the VM can attached/detached from a pool after the model 
is created), this method should be modified
@@ -284,6 +299,9 @@
             // ==Custom Properties Tab==
             getCustomProperties().setIsChangable(false);
             getCustomPropertySheet().setIsChangable(false);
+
+            // ==Icon Tab==
+            getIcon().setIsChangable(false);
 
             vmAttachedToPool = true;
         }
@@ -1713,6 +1731,11 @@
         getAutoConverge().setItems(Arrays.asList(null, true, false));
         setMigrateCompressed(new NotChangableForVmInPoolListModel<Boolean>());
         getMigrateCompressed().setItems(Arrays.asList(null, true, false));
+        setIcon(new EntityModel<IconWithDefault>());
+        // TODO jakub delete
+        this.getIcon().setEntity(new IconWithDefault(
+                
"!
 
buyn7izqMWwosGymt+vZbK4wca+daS+476+u83DtNTErLzL4tXPq97Om8bR3+TNsN3Uw+TUu9Pa5e3m0PTnsvDnxPn35iwAAAAAlgB4AAAI/gDvCRw4cJ/BgwgRFtx3jx48cODgwduGLZnFZMqUXbyYUdk2cN68SQwJblvHjslsqVypcpXLlzBFyZwpKibNm5RyisrJk2chSoWCCh0qVI7Ro0iTEiSYsKlCgvQcQoTnDSK2jLSy0srIseO2jxDBXcVocWVKlithqn15U+1Nmjp39pybk6jdpHiRLmXqNCHUqBLDhvXGixSpQ4i3dj2J7aPJlBfNop1say3bt6veapbJUy4lz0DtFs2bV+DBe+5SL2TotCHgwCXBelNFihHiQ4rYqNKIESXGxiU1qixbljJLyzFdbsYZtydon6ILyZFOWqnBe/vIkUKX2iG9fVHp/mFn6FriQ6oQv2IDR5tUbja425DiRcui74zrwXHdeNY4ZeRrLdfZTpzNVYghdRVlCHVDVWfUafcURkwv7kwEj1QXhvcQWF9xiE1jtDHiXhtW0MEIHbIQw4t+/ClDizcm+QaZf/4BaNNyns2kkyGBZIVgUEBNF5SQRJGG3UDzySILOTA69pF5DlH04ZTqgZgiMbKMskUbjnRJjCnq9MKifYfI4s42GIXEW3E00gggjgXKNZMhe/iATjPk/BhadKPhdaFrhQW6opMdplfllIjSMkoz3XTTzDGjqAGHI4yccow83LgDjn202AGJOpsq40g85KzZX5v/IQenTKXQguBM/ntIwcMxv8RTSlA8GrLggqKRNpVEhhnWiyy8VNXYoYgmWxEtbByzzjrqxLOJOo6YSMcopuTDDjqbYsQGHfGgUyotkMiDjje8KTMcqpNZVlNlq1RGky0y7bEdLaIYIkogVkghBSTPyOMNj02w4c0euvJq1JB4BRXbR7QNS80xsnBX0rHHdogousocYkc3+OCzTjzczEONG!
 
ic6Qow63DiyGy2HtMFIN7LIQ84hdMBRSzzoEncqu2vB61JaNGVWCBuQdENOKfk20YOssmwCiauB+ODDKG3sEYh0BzbodSGHqBIWbdR0s043l+pzZsYdHkqOO7zEDAc3Ic+jzjzzdKOG/h05U0NNL2q0wYYVWaB8DJj5OKKGGpDkk4+mtmBUGbvHwTR5KaqgozS9NwXSRjObsDPwHv3yIMUWp0jRhCh78GC662xsbRevRB2GGG2ymA1tyf7kQ4/GGn8IDjnqrCPPKXY4QrfIYOqDzhZqXEuMI20IboUVbGxBBzHdcKNPN1LYwQg6+uiTjzeRR0a50EOvUgobjLAzDy+lZBZvTXts0YwpiZCzBw4+iIKsfAAHNTSBTjyIQhSsJgUrBOKBuuKaaG6DGJXRbR7rSMc8/MEPfZDDUBnzRjzkIbJ0xMMOo1hHyNTxBmqYC3p24AU16GAHNkQhC1FoQ/RWxg155OMN/nAgBffmwY/zbSR9qBoa+0rRBik8qxf1s0Up6MUvRzQDEjnghRV6EMAF+kAKBuRRFHpwPTL2AAdR4IED+QQ2xNhBEbxARzrwIQ91pIMfHTQfL4olFuDwooN2m9AhRqGOkHEDEmWTRRbagBtG2EFwasgC4VBWNm7kAx3HeIOlLoXHffTsiOuDly32ULhnEaMJgSBdIAyBAyno7A1SoEMUcMBFH3AxfFYwhNNwYIVDDK4HPTDdFgJBJEMsLCi+jBkjZKGODrJjHfPQRx75EY9RyEJsX6FFL/iBj3z0wnaE7CA3qHGKUzBikbghxSPZsEjC2UEW8ajFPNLxiH1Q4RTE/jhGNP3hD0/a52c0qkm8pMhOLHDjG9SQAg5o2YQ6bWFScHBlFFKQAmA+jQckCsQuyYg9L/IAC3tgoyqqpwhGENJxGMSbNPXRu1PIohdiYw81OpgPWdRGZujoYC265AhFVA83hckCG0hkhejJgh/d0Ic8ILGPZjwCEsfQxwb9oY99hAoyAKVRKfZgQywcY39bSMFCU9CvKKgBeq7MggXESksBzpKtXMyCXBUYQCkcYldBWaWQ!
 
YoYbJcWjGfmQB96imQ98+CMejhiFYnmhilF0w3HqOAwjSiqLfPADEont6U/5tk42DE5Sp/hHN9RxvEvV4g2mmIdh/TGOfNzDVFlF/gtX6/e+LURhC44wBSToYIWxcrEHLNhS4dRKS2BuwbYUPSMXbzjXMfJyD5RYkBxwEFJkIkYN6PCbI0jruO4SER2kkIUjbDqKU9hxhorAzbXMi1RKOYIUJE2vLz372WsFFh3EgAMdqJGPRzRDpfz4BjtsphHYUiYZpMtCEwiaheN2CQ5b6K1FLWpWO0SYosW1gnNpudDryRWHwMSBUHWl0QBqDWyepUN2R7Gd7kqziPxAh0khkaVyHqMb1KjeG5dZtnz8gxqTTa8d3oiYwbUzCiiMRz4mBAc7RJUb3GDHN1TLDvmpo1QcKcVZ+pOMUmg4Ck1olVwdHNEo9JaXWwRm/hbsoIYc4iC5CgwxLXt7PcJJMs0kakITUvDFLfDAB3uIGRtIEY9esJi068gHEeWBR3QooryjoMMpEHmMuDFSEYrIUo8TR0FMK+KRhCORD+igsnhMqBeS1kc6uJGOZkATU/JAKC32kBVVsEHL/HmfW3HQhENADw5YcAQcxnjGLVqtBzc8Ky7XKkAp2FKsIeboXNNshRRYwALB9Bey7SrUUcRDFtRARw9zKs1m/CPG8hEvbiFhaFJ4NmaK6FLZSEsNOyDG05iOmVzZMOounYIao+gFbuMBKWqYohmMKuf+SLHILChCDbfeSIKz8OwUnFVSuI2oLXlZSwUGzqx0yMKb/jVsyx5U9IbKJRzKLWrtFChwrACMmRp6gQ7H1lEd6pDm8f7xYzUoQp1RqFSWbMeGUTDi6POOFnzdaAf3PFKobTDr0RMroig4ouZS6IUpykkMOmBhFNUihcfbvIeLeNlfIn9zCixMh9MpztlnpCULeHDcLeBwcWqo9vX4bHI0ShKYXUT5m6/NcYoavnpqqFgv5BEP/kZTH7Wo2T+e!
 
dwihRkHSSvomI9UhCxGVrRksqzcjPYbpIbdhkTc8Og1NdHmAS8ERp1AsKdoORikcxgd7g7hFvOxWkVP08trjQfK2MGGLns62tz1um4FJUR6YHNlmBrwCb1htwsuZotf2ZRZG/hFufjQjp/ykJzrk8Q91bKHybcBt7BX76TYgVrGN4l6rGRFfOGJah23PwtE/3fT0uxSMWfJedmA1aiAFPxdJgWMFGbEH0CdJI7cFQLQFuOcIY2RLdVY4WyAFCtRgW2AHW5QFFLVxcycFFgViZjZ40HZGhmdtgjNoK4NUx8APLQUHvVAu+UAHp5d+sGdTh/Fz8mAijkUxx5AOx+BTa4Zv8JV1WzJZjyQ+dpAF+mVh40VDiqBmW8AIlCIFbWAHPbBVX3ZDs4QD0VNAfyYpY7RAc1U4keRxx1WBbVZRJvdFJdcDcpVmaidWG3eHKUAi25dz3xNV/tANdFBSJ0UNMERq/rH3c4eAhfIwWYtCDKNwDJynb21QepjWYCezhabHNwhIQ9FjGG/0gdpjdFKwZlvgWSSIbLbVA5UYSRSnPbw1fRsIRpHkA1ngL87GAsDHVizgAyywUMBEfXGndnSogthnAZ8FT+XDDd3AD/PwCCZCCoRkLtVDByZiCiyWXr3wBugAB5HYDaYgC8dwDHTgMYvkaW+UBY5ADdSjiZ9mjdCTM9bYJbVhB8l3LeI1ZI90erY0OAD0aZJ0W9aoBtBnVn5Wdx5nOl4EixjGAsBlUQFkd32ndsn3ZmqHjFbwaEoWWEmVDqcQjaOQD+mQD6SAMtYYe4ahCAaHarnzKJA2iPuI/m9sBgdZomNvxDd0AD1q8GDCFixnBT3j1VP6GDMb2FsOZ48gZ41naFaL4y8ZqIoLtECno18i54vHJn0aOJHY9lBqYJGDNzgq6UNKJQ+r1nks5m13ZIiYRmruoQh0oA6C+E4IN2lZcpP1N2TRI4APh5cmEo80mElBpFiRtjjjNQolRYVnJVfANGQ49FDwyECkVkAZ6Gdx1Yt1dTp7M3cJRGxWaTV5iG08EFFxh33Vw0zRBE3z8A0t83P!
 
SWDbm4ghNR4+ehnOUAm6QQmNt+Ui4MWT8d3S8GTiVaI2MMIbUwChvMApf4lKOgAXeaJjWWHrHRXFIVosQdTpW81Bdclyn/pOKY+SQX+QvCpRAARR3wPVFaeQDFoAD6flFf7aCFuBwxBBYSqVBdtNT0sh98yYL4nN0kxVe3XMK40UNnRdku2lvbIaX+zmPwzlkJ2JhD3UM7JAObnAKtdAMtWAKFwoJ69cl1hRpaKV8DSZsDmZLkgIJkIAFBZiBc9iLYJSTCpSBplNy3ulFtuSQ0PeLKLiHhJYOfihgrUZDJuUIx0AMoNcL+3l0hmFezTA9LyWUlYhpe/Nwe3OThgF2nWegn/aTdCBP87A/zcANFvoMz7A/pzAsJlpOAAcHjFB3QKkGPtAlpRhvjgBEP+lsCuSdYJSBhcNmauA6vcgCgGo13umZ/lajhwv1TiLZDfKgVGgDoOIDdsSQTzRTG9JoIiZlcNNjaEFWevp4oJ5mGEg6IltIQ9FJB28wR612UOzwDAeVDmgzpMRQTpbCjooFbNrzBliAW0Y3htkpgThEqAGkbIsjPoOYRrJimYAKqBiAAbroixZgo2P1Tj7EKIlKDZAYmypjKd1gmCLCU4mVIh3qSDTkMUMGim/kdFNqjSYSSaoXPdDjBnTzDetwUN8gplDmpY8ypKZgClgyKb3AU29QLS+FJcI2KcNJd7JCo7hIhdYiga7zZw+brMkqqMDkkIZnUo91OPIwr2jDrZRSTqbAjsHCoV2iJIoFm/r4qTalegMZ/jjaAwdYsjjqmjNYgAWP4AzfIK84yw0VWgvMuAlgWgu1MI77CqDH+a88hY+9YGjVEo0nQgeL47Cns67CeSJ+ZjpSoJl/5pm9OH2AynwpoFj8NaTsAJfWOqD4WF4oyQsBWC3j5VJGR0MM2iWQ8Ahz+gZv4AZ4q6akkFv8sFTx+GtYIAWmIKbj8A3OAGVCWwvO0A0VOqaLyyhDegpv0LbWWKU!
 
meyJnpV8IKUAdmKVQa6mzpz2y8rBbW6Pb5os2ygJ6AArqkA/HIEetNqQDalKygE+x1wt+A3b6BXvH4FL8yQjARgVYQAVUkANUoANVEAZhoAd64AaY1Q3+gA/dUC1s/no6byC0zjAO4wC5m3BwFtoNz+C4+1oLo/Uo4Yi0lKKumeUIhZEzVqt8UapOeCecoJqTpes6FKuLzaa6gFoEfhAKrpoOEaoOSypeR4ec+UQM1IBzOMeONtYNpxBeYIcFOcADyhqoc0cFVbC8zvsItSBV6TA9OcmmWCC0mzALs/AMtQAKjwAKtUAM2Tu+x2Cim2C+8WC27MhTowAHk+KthfG0Hchm+Jai8SgioBo9bfewngmGspKLgCoEZVAHuBAKuCA/eXMMAFcb+BSy6EBajycPB0cM0zLBjmC8xnvBzAqoJ5ADJ4ACxru8zKsHz8CjkBi/wFa3F1q+s1AL37tT/rmFcNPjCG7wCJvgauqQXZNGY2DHww+WJcOpxPyJrjwwnKEqImtqLVD7Z3nqLzEKqFEgBELABXVQyuOgWq3WDCipWOiQw4FlWfhgPq3Grz1svDpwwRiMATxwArucxserA0gwx3oQBm5Qwz0lKVjwBo/gBoW8r6AAwy1st767pJFGyMwcu5ozPR06p46cMwMaPU+LxLRnv8Giesd1dMqGVlZDd3+GAT4QAzFQBF7gBXXAB5+AC+PAo9QgjcSQwz5URCvFUvygDtTwBjmAvMebyxfAAm28y7t80Mi7A8E8x8yrt3UKbMyZAzULBwaNt7nqCJBgCkPqtm9QBW5wTwxM/rAP5sMzm4Em9bRKXM6uo65IXBsPloHWeHxtZ5nHuqxC4AdTIARFoARcUNR1EArPoCTPEFh3VD54A0jzBApVgMa3fAJzx8sn0Ma4DKg8kAM6wMFy3LzDXAUXLAXLetZonda6DHuQQLCM8AZojAWZJ17JnKtampM8AM5Wa6mCKStVC3a868M168lgBGGWSXcswKw//QlTUAMvAARO4AVlUARl4LwCfEd4JD9kaaGPIAZPkLxpjNWi3dWJjdYX0M!
 
bKG9ZiXQVWjQHX5trXdm0PYAEPsKwP8AAsICmJZXTA1sHEqyRKgtGDHY8pijKzp9ffCmw82ciBLdw5MJVZy9X+/oIBF3AGZ/ALuCAENRDPRdDdZTDPZRDA44BH+dANmyAGyXvQoE0FKJDVXs3LGKysaJ3YWq28zDvRzDvVJwDbso0Bt/3fAI7bD2V0W4AFVaC8VHCc1iQLNdvgG93go6h6maXg3JzM4hWwk+bRDq7RWJDVWc0DwpYDMSAEn1DiUxADSsDdRUAETvDdkv0M8zALYuDbEA3R7H3BBy3fFrCsO57VF/DjWp0DSDDk973aOeDftB3gSn7bujx7porGxPutsVrXDl6zOsCcJKuho6BJsapJCX4Ky+wGIg0JzMzMCc7hEI0CCE0FI+4H9xwKQT0CI77ilE3ZZRDeZYAEUJ4D/u3N56C9364926+9rHxe6CdwAQd9AsobzDqwA8N853rA2v+d5Ev+3w7gADzwcF7dwcz5Bm1NDI/Qwb4dx09QBVDVzM8MCoWMcEPrBptg3puQBq/ODTIOCpuwCQe+ASiQAReQASsAAiuwARsAzwEMBX4ABTXgAR4gykBQBEIAAkrw3VxQBkJQBU8QBlN9vAdtwfu944PuAAuNAYnuxqft4yewAjtQBSiAAlUgA8xbBsxLBSfgAEuuALdt7w+gAPquACyQA8urBE9Q5kLruGLg2QVfBTuwA2JwyKkuBq2rB5vAqtzwDCpcx88QChX/DJvQ2U8Q7CuwAijQ679uAsIO/s8/bQRuLgQysOwx0OwvsAEgUARO4ARFMAVFUNRFgARejQI8cNa17QDLytDsfgEOEPLlzgKnnQGHjgISve46wAASfedhAO85QO+Vru/5vu8PAO7+/gRi4MeJsMLc4AyJkAYO3/Eo8AQU7/BiMAVmvwNPMAs5S/ZhP/G14AYRhQENkAEoIAM78PEfz/cb0PEbwAAdEAM14OZnQAQx8AIgAAIyAAIvUA!
 
QMIPndTQTeXdTwvgOAbtvLetpubLyHjtBanQEZ4NUrIANIoO4o0Oh/P+R3Du/q7t/3vu+2f/sKAO7oDvFKEAazsL3j8AuJ8AR60Oh6kA6moPQ7oARTIAY6/qADeuAMOIvrQH4AW68AB2AABNAA22/6pn8B2z8A4g8Aj/8CU+AHRWDyJvD4lg/5L6Di6R8Dmg/vh/7fGJDV687nan7oOqDmCb8CAKGjyg4dBVFkyKBDRpWCSJSEgagnzI4TFxwowJhR48YDDhqgqMIwTKhfJXGJGYgkFDtuJxxcOFHlCQoUOnY8caOjwQGMB3wSABp0wNABBIgOBQAABIgOSz28gArVxNKoL2JAVYK1CBcuTspUsYgBQw4dJw7mMIv2AgqzKwYOXIHwhFkUOTLkoLkDyY4dMsIoeVgFBcYHGzlufHlCRxg/oUL5kVgFSZlx7GZVOdHgRIYLDRo4/vDZ8aKCA0CPnj6aVLUMqCAiPH0qFYQMDzFsx1jK+kWZIk6cAAFSZu8OmmVzHJ9Lc27ntSvKLoepg61mtih2ON/QFwlEJC4VOABN2rBhnxd2hClTxk8ZPWXQz3LmJ8zez6FJG/BoQL9p1KlV/7fKAxAYyO2FDqwCwaqrrHphqRiKgNCJIryIkIsylCBOB7x0oAKvucADrwGYNMvgoBV24OyCCzZQboUVDspgBRkW4uwlEH0aLzQdHcjgPN7+Yqyx9SDSATQDQtuvqP78+081AQR4AYgRQJBAggRt80BABWOorUHWgHiQiAkhLMO3C0+8gIccVlgORPA686yBGPna/uGztl5MDoW4YjwRBRV3Cm00HAfVMbQGNkAipAwaONEm4mosdD+jliSqyf+e7ABLEDzI9EENBLzStiJiiECCF4oYAcIxUwViKwvLkGGHKlZQETzRNIuTz4LiYkFEPVfw1aYVNtDBRedQqO+AI0ErtFkk9fPsoo58Gu2BSA0QitIBLHXyydpiOPA2JyjEjalWHwQzghFG0KCIH1Rl90EnuKrDCy/OzMBNX5krCyEV//3IxYP05EvgDWTU4QJnm9VPP5+Od!
 
BbiQK8NatLTuMVYAN0W8CDVH5z4QcwXPNAATA5+4GLUByFMFd5TfZPQXi/q8EOJRT8TMa4LEvo3RDhL/iTWxYOve/FgGdhktlBbj2wYYoeVfXrhhyXN9mKML/X2KgkWiJIGVGN417YWOhjBtzApFOIDVSV09+OYubC3jjpgfcKzEznjuaLP/vRs5xX22iBoYyGFGj/wnpZYR6cRlxrqio3ir9KMA6BcAMo9KMIDCRI4lQYibKMhhhFEf80J0YUAHcIfPuYCZQlR5uKLOmKW2486jnYxzwwqAjjOnJVAwtjr/G2AaWh3MhLxppdnvvmGH7f6agAsp7z6ADwQIoECFhidCCJoGL2GGDQQ4jVUIxBihA9imJcr2bmooysn6tACjz+cIMKLrrjIY+4NSpSOn3gHp77ZDTDAewES/jZAAM8YL2rKKpzzCEC1oEgKW1Rb0tWoZz0OLqUABZBA97zwPfW9QG0SwFypahABEMTgC2DQQv3AgIf6faF+dcBDDuXGBSKgrH9KyA4IWDQXvcWpVox6QhHKAATgbYBphYOiBB83RSpSETWXstQGOVi9AjDggxHogLpoMC7bEAF7VeKSa1j4Af05QXZ1AAMY8jDHONZRh++CX/yUeCadETBXegFMDNIDvJ04z2HME4rFilIxbWWQW0960harRz0QJoABlkxABTQQAxrQ4GMLqEGVPBCBF0QgARKIAMvmB0c85OELeIgFLDQBC1gwgxn1IgJXxvWxucVACbTynWdQ/rABv2BFCUUATFwM2bQJNpORRIFeI6/ILUlukXoVIIIECoAAARRgdJsMmRMSUAMQgDBLVALBAj4wggiE4Ah/AMMXjvCFPLzCF/f0hStsqYk6yO5dLfBNJ+ZVhh38D2eMkpEMkCCDAx4zDE6U4vOseBpF9gcA26rUtrAWyWpaj3of8F4EPoCABZSBC6J7EBcgMCoEFCABCIDp9j7wAQ1owAk5bF89kYGMZSwjG!
 
D/96TJiUYn7wWCVcpubDG7GAGLORkYOkYEQCHoBAxRveVO8YDSlySTpdbSaH0XZD/JghBFw4QhFiEAEbCqC70WgANVraW1oCoEfaAEMKeNfJmKx/tdYLAOfxliGNAS7DKLODn79K8P/EgqCvmwABA5ZqAKtyszIVQxbz9wqUqSnGq9KknrucuMd5kAEeRYhAQmIQA9JyMUHrVOtFXCXFpwwgh/QE5aVqEQsXLFXWTIjGNCAhjSE2gmkyq0IHVCC3ISg0KwowZcL3UADKajIiUbvohbd7PQ621EukAFlZCBtHo6ggbQuIAZEqEAMhPDWAEggl1xKACdpAAMgtKCT39MCM8DQh0y4ohW8tWcwejrgZbCip6yIxR/+kMNRFaFmrJGBEBkQJ2dWDZoVfVJ2NSyApGhxu9bzQh4+dgTa1oHEtklAx1BVBAkEwKVFgMGDIsC+/tDVVHTq+wEr+/tTX8TCF8FoBSuEjInfQmOnRg6sLTvBlR8ggQFCY8CEKaxVSk0vwxrWYOU+XE0VnMFzdQgd/GibzQRwjwMj8ABMOQcBDXzBCescHwRG6YEKVAACHLgpHjLxW8FKI6hCtgQrgPpTZATDF+EIx2+FWoQCBE7KQGkA5JzZH0hWGpJYntyWO2oEJpy3DpuEHwyI4IQWXiWtHljAAiBQSrD9QAMp/tQHT7uABGggCHHEg4GFuotbDNiWsTDGoIHqC8ECt6exAMKAGuDFCUeZARYTAFGeNIArTw9r192opS2taa/WgAgi+AJ4wcZDNC9gxmU2d6o5NmNS/mmTeyOFaboX4AQtJKGGfMiDNsyBCCMIAQq6sMc7wuFnYQcDGcCFhjV4ygotdMCLBWBgA6aNmolz+JEWb5K2IcntD9dgBBWobQwqMMYipBqVtEbt+CJAa/OiGgETUGcMTosAFNI6Ak5wAcpoEApNcMIevzBBCRLRj3YEvMh9NnhQEb5ToZ50QFHuYrRPY+mMcbjaVtY4x7c7OnXNdsZEGEECRofWAKSYdNpDQAfMXYISjFICLS17WhEQgBi0gANB6AoOmfGJT/Dh!
 
F/awhzlg0QqEI1yw9/ypNIBrcJ5WImUxYMBQCiB1aV96cgXI4pU3rnWv0sALtCXDF1Q24/39/uCUo+MSzWsAARCcskpz5+IpGVD2Bdi0E0GggSYyUQlb8r4cn/gCK/o8/Gkg2vh93qmghVyJOgDhBR+seMUngHnLWZxyJqgB5Trs0QBwlPNcRq/3QmYbDXABDB/Dwxd+ELoQOKEAU/iEW9/6wQ9ykb2UyyQM/qAFDTDDE5ioBFZIhVhgBlioBDyoBFdIPMEqPuNzQERDOiCLhU7ohDKIABnwookbAPeSAA5zsdlDgPiqgwgwge/7vg+ogPX5AHPTgDKYLy34ARjQgj4gAyeoABGogyIgAliggZf6oJi6PxejHJeCgFcCM1jYPdzCLU0whkrQgiOYId0KBmvoM2OY/oYrDIdqeMAHnAZmiAV+ioG3CwAASAAa+IJPmAAAaK8Y+AAh6JjZkoFTmjsrM8Ht+gCReoFZEx0IgACQoYDacjPzcrMv0AQy4AIaULdUayn2eis1gwD4kS0yyIQcWsI5MsQRMCoozARkGD6CCwZXAMVMeAVjIEVSvMIGDAdxUEUla6U86IQFQIARYCsmGIEFWJ1TgpC587A6lKQpaLu0igA+rIAyQxkYCwHZIQJv+x4yOALfgIBE1B5GhKkQXAARQBkc+kJN0IQ82EYn8JwKWAAlWIAQAIM9kwZrQEdrAK6dQoZUcEd3BMUgE7JWaAVMwARXyIQkxINRnMAPoIE5/ugCPqgBEeACBPAAIhiCGODFD9ukj6uA8uIY7oGBIJCtI8iDOZiDGTACJ3AlEWAz8lq504KrWdOeBIAAGIgfszoCz7mB9WsXETCvakwCBMyEHyu8dASGdkwFVsAtAUwFS7CEA1wwPBCE8yODPjhAPGDDLiAEQ5ynERCCF+KCNFTDzqpKr/JADrKEP6AfGFydHHOCIzgCQJwhTbgEgMwDBNSCILA7DoCAkBzCbVK3kaOA!
 
svoBTPwBPIOBUWGzMDQB7jmCIRiCGwCDBCyydVw6d8SEoLSEVjjAtcw5GPiBGIABMriBJIgnJ1iCQiSDJVgCODKDPtAEPjiD1EsK6+Gs/sq5Mo/aRcoRhNd8TaCUTUsYqhyiRG18BU3ILVawhLVsSz5EudNCpRkTqzrIMQ0QgfmiqxBwl9CJAXWqtR8YOQ34gTiqhFY4OCPbqVT4rZ36rwMEgyH4ASAwpUxRqyOwxhDopCXAAzLgzHL0hFeoJWPosU44AxOAve7jIMuZABPwzydIAxOYgI56zT+ITUGwBNiMg9dUQiWkxwfFBEFgSw7ogAqlUKbYJDb0pCM4QPEyK1eDgb2srUNMlQRwF+4BOTIggzjCTnc8uHrUs1aQBmRoBTBIAifQgAWgHLmLgSEIAS64AS6AAcy00SV4oXxshf5SwCqEBfuMgenbIhWA/oISSIN8+Ad20IUSqB7t6r4DhE3GBMovbdBKAMoHTQV5/IMg4AAOoAAKtdAW0AAVSK8jWNEZ0jNNCLfXscjYqYMKmBfVogHCJFJYygQw+AOejAMzQMDAcoLKHEYleMQfEIEhAIP2E0xKDU/MVNEb0DPc0itQDAZgoEJfWAYvrE25oUDi6oR6+Id/4Ic06D4TqAAPcDEToCVXaNAE1dWgzFXZBLQY+gPZXLIK7QAOeA0NuAQvoIEZuARJUFEyqARNOIL8SZkbpIEkcLPVETnTCQEYgIAQCBlcMwMzMMP++YIkGILa+4G361Ea+oGZNKsQtashgAFKTYIQhQHPtKNM/nhQehS2VtitPsADS7hHWlIy9NqaF7ieGKiGatiFXXiFiH0F3OLVXvXV5dMCF9BYF7A7C22dlVOBGfACJpCEc+gCJiCDV8gEG4yBJeACb6pXNeUADdicFXypEKy1I1iCPNCFS4ACKOiCS2CGTtCCmXMpGMCD8Gy/G6gfGBABkPlKLYCBj1OXH1hRzKyEOJiEfuVaewRKTDjTeEQFVPAFPHgpKrnBhtWGhm3YhyXT5cstBJ1NoHxHSwACYq1QDTjGPCCCKRCCEDuCXZAETpg!
 
jwv2CTjoCLeCe1alFo71ZNSuz/KkBTjOCM/iESygHJkgz/UyAEJjXChjSGcLRrwwC/icYAhcgr5eigSUAA8KMozgAW8W0x9ldTIKN3TNVzCQwLzWLgIa9BrZ1WIglWAHkTYKdzeJ1x61M07bkABiAGy8YLTNsJVh4hTogAjLIAxUFGW/VpBF4xq1JtZUDxrSqs5mSHZlLACiwh+l7KzWMxcHE0RDATDPgggrASyhMAheAgNMqQ8wUzNel3Xd0x0nwWti83UoIghAYOZj8IGNgW21Y210gxVeQRyGLR66FUEEYgiCoLyA4ghkgrc9Jq+3NBE8oxDn6ghT+ATjVgA5gswrlwxiW4bcMRj7UALCrACjohynIzxAUgSMwqgTGVDB4SEw8AswMAbdKtRE4AsIc/uLYtF0Blt0AboUfeMb8Q90CuIOe2wW2fdhRnOBX0K18MjgzrVsJbYEW0AI9goHUnbHa0DNRlGA8Xb+21IA1XVMNmOE9pmFg/AAd1gUuQoCRC9EfoNfLjKOHpAAYOGIwEAFtOq0I+AGxdN0ZQlADtsd3JGDaxQQ8eMtZ4wAXgIEEYMYvqIRXKEBx6IQ8sCVxKIdyYAZ+DTJLAIMvTdD9m1AY4IBS2t9Ze7kK4B9NeFhYyAPTqbsWuLGamlk+3uNSOa0PmIJ+sIcAmDtq/IAfCAHB1Nh6A4Mi0AAKcKc4ogFIlgB/bOLLHALMtM2kRVDYtUdB2L8NptRR9kHUkt8K/tAEGuBQZiQDZtgKWIgFVy4HcWiF5JVNLcBjCkWCUVoAO3vGaaS1ADgDI4gAJmCG3ATiNE7jM1sX0akpZuZD8VXEU5oCe+iHCajmSBYBIk2CG8BUGvhmRk6CJAjD9lIrF2jdmQ4CF2jiSyXSMQDqMYChIYghFwiBBdgm2BvHJBABMhCBPNCrZqyDJDhAVxi+dRy0VBAEyHTLlUu1GF65FETqObiFCVABJsgD3Fqy22DrdUnojwZ!
 
rcwvGldumEijpfvgEOaSx2lpRM4gjzLRBvb1Mp4a9MqzXdE5nPFhJGtCkel1hiizKBNYC2GyB05o/DQhREYABz8mhGwhL3MID/k8AhilUvIRbx+xcR97UYBdYZnNDLVNaALKOANAdWEvAgzoIAg4GjmNugdFZl5qiAApw6Gf06pstAUToB7z+gDoQAREAgyMA10pOWhhqwxm4ASMVAZii5iH1zJkG6hsQAamNwZYsSdemOVTSnBCEu3a6ARqggCIw5zqAgVcCz0oAA//CznRER2gAslbISdRehj9o40RMtW+7syAYgziIg9kkqrXM7fpC5htr3hCoKc0JRkVEABPQ4X7QBRqwzCHlguSkAcGc6emegYleAmakgSmYgBKQAPm9ASB2Jy1g7Fyj1yGogw+IxkmiHsvRbmoOAaM+yZliF322b/G+qUwg/jJ1XLoc2jGDc9FgoOXTdUs+DB0XuDWgFoQxQFdKLVAF04INZss09twgoIAYqIEJkICaioCXS4RQ8Ad/mAW1KSMhKIFQANJ6nV8wmIEZoNPOVAEVMAIV+GMmWAKxTE4jqBI+iJ0fIEwn+OQg1DJdFAAEoACjjgAaeI20MkNcy4MZqM7rPEzgSgX+ykd2RPWdkvKtrlcExLV1TlONhYEgx/IgAHMHx2ktUE8auMPZbvMJSIRZ4Id8EAP/NIEX8M8p4AdcCNF03q874ARCuIM7IARCONk+v4JhaIQrAIM76IIuGPQSEAIj8IIfWII7QGAbdKshDEHUBAAEwOxRZkM6/hs581NRTZDUWm6F/T5tx6wEsE31gE+w/coEPNUfPOCnM/ADL+CDT8jLNQVnjSXq3CYDPGjUI5jQrq4Ay52FTTD2j3+Gf6iHb7uBFcUDTsiFlE/5W5gDMxAtYYD5RsAD0bqES2ACz2GrFOcECk6/yWzzsss+7fJx5JwX62XsTnICMnhYfB8BLYiDfddvdQyGGQL4IzvtpIsFG+0D0fQDPziDORhNP0iEobOHT/AeIkCdB8njmvKnLwgCGGBeDahcMRADD/h4/+QHV8WFH86DOCKDLp!
 
iDXciFXegC5+6DO8iFRoD5OSADJmCCOWBsYgQD7N3iO5jp8AxDGtAFEygA/v+cADW3RhoQhFaSnW0Mt1dIxSWggBsYRWnIhqiXhhw6x6hXRypUx7KdoYLv+jPIgyblAz5gwmrQBkTQBb4r/lB4zgWgkg9wJSLwVlj8wSnggycQAxL8+FpoVZEPy1eafCYgghlIcUnVWTKYgytQ/O9vbsZeoQoIZW8VASPgghDI5vYLSybg4c///BLgAhjcTt6cBIEFCC1JNL2qJJCZJjKxrGVjmC1bsGAPrVGsKE0aNGvIMIEhQ+aOpk9nPn26xEeIkyV5QH7iU/LWLUQ1IEBAEEACkzxLjlQQM+vnr1+fxDyRYOKoiRL5/jH9h2vnl45M5sxRIYIGViZk8lxp/nTLhQgNGiBUiFEBhgsXYIZUSJCApoYINPJwIcJlyoQACCbgIrMWGTRkgpFFbGX4sOFUhjGlSiVYWkSMFCdCg5wKU6U6ISpU+LHVjx8+U4QQOXIED5gkYC4Nu3JN2BoVFCAkKIDgw5Rx49glSuMsaNBZUxKBDo2IX9N/zI7EgHFkhogunCoFoaFixowld5YsucH9xo8QC2LMofHjBtsIC2oHSKBBRB4KR/LgmmWiwAQ+R7iQEVxYMSsBsuJYRAUCyEorlvwRBAxBBKFFHZ38EUtEwAQTixNuubUAEXlM8cQUU9SggRNcqAYGGK+0M4w27yBSQU0BQAHFBBNM8c03/uwI54xu5ZQzji6h4DLkJ/zw448/Ts3wBRHWqXDFHETAAIMIXszARBdMwFjBCCPQENcIizABww1aVGBTAGkuQIEKTIgAg0dMGFHDLOyM1AUyjemZoCWCCGIJoIEKiiAmrSwTyx9cgFZGgzBo0BYCCUSqIWk1HPWCBxW8eUMSecCySz/+2POLHhHEuIg2KpRQQhrfmPOqOf4Y+Y8uLX0iZCL+OGOPPf38ogIufgjxQZtzjEgGdljO0QUNs0!
 
laQAEgfNDIIjQckQQZTKhwZgATqKCCJHPMkMccklyiySX1BHWJYHqmEgyAgfqZ2hBBDAEGHvL6+Ucn/f4hiBZDDPED/g20uYUAAhEgIMELZn0ghHk3gJHHLrfUY887QRpBQwkmNBLNIh2XgIg57bRjTj1NiQQFSZ8goi6v/YwDBY+JTEHDF3ec8cMS2XmUByd3gHEElWfWwEQui1RAgwjEMqFtCY00AoUkucxxyS3a3ALLJbpcolVjiBlWYCtkW4IH2multXYLa7vtgr1a/BBBAhF4sIAEU5QgRBkVEPfmvbCUU4095kQTDSGXzDHFGajeAsUUJJs8DDuyMpXIjKC1VA8/9aRsziLj6FMPLiJkRwgssHDCiSZ55KHJHWaYgUe2Uy2yCI01rIowsVELc0UXiyiujTZdeKqJJ0swYZgrzR8W/szYhRWG2NlDpBUC9hxwQAH33dO0Ht4J2GjCFHq8UENpWqBdjj8Xa3P4LnfMUULjWs+4yDDm9GOPkUk6ZUQJQsSHK9TDH/vrx6tiZg8fleMdiuPEuZawi2rsYheAQAMG79AFQmhLBUZQQR2IgIAKfHARuQjXGc7ABz5AgQbjygMeaLC8VmSihq9oHg5z2Dxf+MIVfTADJlyBIFZYwnoUCEH3uPe9BYQvTQEogAn0IAQ+lIMMldAELNLBFHtoIxeHe8cnoDAHmNyCKu8oHCLeMQdOEM5XUChBIsZRDj6QwQi74MTt8niLd7xDHNWoRuvwkIcvkEETuyCEJK5xw1iQ/qsLu5gDE2agApUkoAJnUIENRGCEOXlAAhr6QAVCgYAadKEVzXvFKzLhCVTeEIeeeGUmmheLWVaiD33ABCZYMUs8aCEILkgiTSDwgQ9IIFILE0Mi6lGOTzCja1rURzkIkQthROMW5WDGLiQhCUKQgRBa+wUJEOEFMOyiHK/ShY+6sIQgzE6VmSDX6yqRiUm0AhjAUIwrytYKTNTBC1i6RFa0lQsIeoEISyDD0oZZAwHopZJOLMACELCJCZRghs2r4SpfqdFXeEKVGgVEDWvovBrioRKVsIRJe!
 
RmEH8BgCF9wQpOo9IEafIsIRPgAFPjwC3+U4wqXkMThrrELX7zi/g5MmA8hbrELMjhhBj/AAxZhAotPzOEMTrgBJwhhwTis5A41pGcrUoEKxIi1FejigzgKQjubyu8DImASDWKwANtMAAB6kYAEAmDXAuQ1ESQgwRUKBL1TZqIPsAxpYW3ZB7ThYbGpOUJUTPqHSqCGsVr4Q0lj6YpYUPYIqqrRXqagiyxVzYvR2IUxODEDGtjgDpyY4DSIWsPVvWJ1mtBEJchwUCN44Qj4siUuMTEJXB7GFb4whjHEIY5wGKMVqDiuMaZhjGIgFxaty4MXdGeCGtWoAE7Ui038CgUVSKMi9ryoRztaQ8WmDUWmmZIIknhE9EDoB7OhCQdaWlIudGwC/nsBbQn82YVcEDga10BuHmhwUE1wIrrT+COEKVjBO+4ilZStQx1eB4vohiMcyv3wh6thDFRCF7nVuAaKUxzhP054F9b9QkFjoDsJJMIEOgjARSpiDchArxWTmERI2YsaFKlmK3kgww2oFBYlQmABS2Sik1UFXgkQEynW6UI0THuODh/hB0cggyqRO+I7AO0SLW6xmFGJ3Ad3uM0dlq6YTSzhM1cwwnSu8CtQy8pU1qF1TqiBLwRbXvNCD5eJVawZ8IUvoLEuD1IJJqSd/L0E6AWAHxhBBCTggQ+UQALfAqppb8EELkDWI314RTGuIeJK/MF1XrVhno0hjWnQuta2/j6uradRjAdDmNcrZjGa74xnV6DSE8ROZSbs+a7j2hMY0zBvKlyRCUz0IQ627Mii70CI1W1FE18IAaSbLKkEGOUoJfCA3Y6yaZo+yYQnBFkFFhABMoGhD4BwhUZdKbZ8hvUwYy0bDwPu4FnPWroG56ExShznhTM8zaw89ivdZZhmX2TQO46IK4ar2GwDwhN9uINGNcEHJUA5AiCQAQgkMFdyI6AE432jCqCAiGUl7RagFgETNQCD13kCE2MYxCB+/!
 
ONUTEKsqDg60YUeCaEzfRJLX3rTEZPPHQa86lYf8cP33MMc4lJPR0eFPYMRaIr4whO+mIY99xkHFL3uDtrG/lqLyqGLM3iBC70MAQeaDGUPkO+DM5I5IhxJCOFxIpGa+EF2bmCDPPRhDGOIQxyCHvSif73yqIjE052Oecw3XfJAh/wgMIgGoE8erP4+OjCM20Me6jCHah+DINwVNoG/8hXTtZC0Bfm6S/iIeHKHRUgn8Uo8OAEGISD1TgqqgmJ6ekZtkk7hd5ELTwACEHfQ7RJQ5PjHk/7Hlv/65jE/CPFHwvNAnwTp0096yD++/ZEXrunFVtzmeeIwlxlCEpIQe8W0AuHIfWWDFQN1GUPq6IIupM5tcdRw/Rj8Ud8Q0IQILMG2ScJ23IAIbIkKDBMmldYJEYL1kdkSnAgYmIHjgkWe5E3C9x1d+KUf+g0C6Klf+kGeDMbB9tVgDZ4f00Fd0xUdRziIC8SBWDnbNVCXR3HUKzyY3B2g62hCY3nC5aEC0+GSIPzS6SCSNgVNalDJdRjBVFzDCW3TR7jd9YXgCNagCUJdJKTg5bVgDL7g+s0gHLKfDdqgCY5f+X0e0LVfQAAAOw==",
 //$NON-NLS-1$
+                
"!
 
z98TMqZFlRpb+pRVdpTV/Puh49J6mtRlfanGNW++lp0P2xMw5kWpvJPrEUdDXimmj2/9Vr1RP5aPFLRS/fDg9O5llTUGdNN1BMVdbilgL+KljV8+eewQVCHutQmd9Rxt4HmVGlVReuqtaNaB7r+OO7hs9rx1dd2VLSuSqu8oAwAAAAAAAAAAAAAAAAAAAAAQDv9AeKA+xpCWfFnAAAAAElFTkSuQmCC"));
 //$NON-NLS-1$
     }
 
     public void initialize(SystemTreeItemModel SystemTreeSelectedItem)
@@ -2709,6 +2732,9 @@
         boolean vmInitIsValid = getVmInitModel().validate();
         setValidTab(TabName.FIRST_RUN, vmInitIsValid);
 
+        getIcon().validateEntity(new IValidation[]{new 
IconWithDefaultValidation()});
+        setValidTab(TabName.ICON_TAB, getIcon().getIsValid());
+
         boolean isValid = hwPartValid && vmInitIsValid && allTabsValid();
         getValid().setEntity(isValid);
         ValidationCompleteEvent.fire(getEventBus(), this);
@@ -2797,6 +2823,7 @@
         setValidTab(TabName.CONSOLE_TAB, true);
         setValidTab(TabName.INITIAL_RUN_TAB, true);
         setValidTab(TabName.GENERAL_TAB, true);
+        setValidTab(TabName.ICON_TAB, true);
         getValid().setEntity(true);
     }
 
diff --git 
a/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/IconValidation.java
 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/IconValidation.java
new file mode 100644
index 0000000..036265e
--- /dev/null
+++ 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/IconValidation.java
@@ -0,0 +1,112 @@
+package org.ovirt.engine.ui.uicommonweb.validation;
+
+import com.google.gwt.user.client.ui.Image;
+import org.ovirt.engine.ui.uicommonweb.Linq;
+import org.ovirt.engine.ui.uicompat.ConstantsManager;
+
+import java.util.List;
+
+/**
+ * It validates icons size, dimensions and file format.
+ */
+public class IconValidation implements IValidation {
+
+    @Override
+    public ValidationResult validate(Object iconObject) {
+        if (iconObject instanceof String) {
+            final String iconString = (String) iconObject;
+            return validate(iconString);
+        }
+        throw new IllegalArgumentException(
+                "Illegal argument type: " + iconObject == null ? "null" : 
iconObject.getClass().toString()); //$NON-NLS-1$ //$NON-NLS-2$
+    }
+
+    private ValidationResult validate(String icon) {
+        ValidationResult typeValidation = validateType(icon);
+        if (!typeValidation.getSuccess()) {
+            return typeValidation;
+        }
+        ValidationResult dimensionsValidation = validateDimensions(icon);
+        ValidationResult sizeValidation = validateSize(icon);
+        if (dimensionsValidation.getSuccess() && sizeValidation.getSuccess()) {
+            return ValidationResult.ok();
+        }
+        final List<String> reasons = 
Linq.concatTypesafe(dimensionsValidation.getReasons(), 
sizeValidation.getReasons());
+        return new ValidationResult(false, reasons);
+    }
+
+    /**
+     * Max width 150px, max height 120px
+     */
+    private ValidationResult validateDimensions(String icon) {
+        final int maxWidth = 150;
+        final int maxHeight = 120;
+        final Image image = new Image(icon);
+        boolean valid = image.getWidth() <= maxWidth
+                && image.getHeight() <= maxHeight;
+        if (valid) {
+            return ValidationResult.ok();
+        }
+        return ValidationResult.fail(
+                
ConstantsManager.getInstance().getMessages().iconDimensionsTooLarge(
+                        image.getWidth(), image.getWidth(), maxWidth, 
maxHeight));
+    }
+
+    /**
+     * The dataUri string has to fit in 32kB.
+     * Ratio base64encoded/raw data is approx. 4/3.
+     */
+    private ValidationResult validateSize(String icon) {
+        final int maxEncodedSize = 32 * 1024;
+        final int maxRawSize = new Double(maxEncodedSize * (3d/4d) / 
1024).intValue(); // just estimate for users, in kB
+        boolean valid = maxEncodedSize > icon.length();
+        if (valid) {
+            return ValidationResult.ok();
+        }
+        return ValidationResult.fail(
+                
ConstantsManager.getInstance().getMessages().iconFileTooLarge(maxRawSize));
+    }
+
+    /**
+     * Magic numbers
+     * <pre>
+     *     png 89 50 4e 47 0d 0a 1a 0a
+     *     jpg ff d8 ff
+     *     gif 'GIF87a' or 'GIF89a'
+     * </pre>
+     */
+    private ValidationResult validateType(String icon) {
+        final String iconBase64Data = dataUriToBase64Data(icon);
+        final String iconRawData = atob(iconBase64Data);
+        final String[] magicNumbers = new String[] {
+                "\u0089\u0050\u004e\u0047\r\n\u001a\n", //$NON-NLS-1$ // png
+                "\u00ff\u00d8\u00ff", //$NON-NLS-1$ // png
+                "GIF87a", //$NON-NLS-1$ // gif
+                "GIF89a" //$NON-NLS-1$ // gif
+        };
+        for (String magicNumber : magicNumbers) {
+            if (iconRawData.startsWith(magicNumber)) {
+                return ValidationResult.ok();
+            }
+        }
+        return 
ValidationResult.fail(ConstantsManager.getInstance().getMessages().invalidIconFormat("png,
 jpg, gif")); //$NON-NLS-1$
+    }
+
+    private native String atob(String encodedString) /*-{
+        return atob(encodedString);
+    }-*/;
+
+
+    /**
+     * Datauri format:
+     * <pre>
+     *     data:[&lt;MIME-type>][;charset=<encoding>][;base64],&lt;data>
+     * </pre>
+     * @param dataUri
+     * @return base46 part of datauri
+     */
+    private String dataUriToBase64Data(String dataUri) {
+        final int commaIndex = dataUri.indexOf(","); //$NON-NLS-1$
+        return dataUri.substring(commaIndex + 1);
+    }
+}
diff --git 
a/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/IconWithDefaultValidation.java
 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/IconWithDefaultValidation.java
new file mode 100644
index 0000000..c0e7adca
--- /dev/null
+++ 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/IconWithDefaultValidation.java
@@ -0,0 +1,23 @@
+package org.ovirt.engine.ui.uicommonweb.validation;
+
+import org.ovirt.engine.ui.uicommonweb.models.vms.IconWithDefault;
+
+/**
+ * TODO jakub add javadoc
+ */
+public class IconWithDefaultValidation implements IValidation {
+
+    @Override
+    public ValidationResult validate(Object value) {
+        if (value instanceof IconWithDefault) {
+            final IconWithDefault iconWithDefault = (IconWithDefault) value;
+            return validate(iconWithDefault);
+        }
+        throw new IllegalArgumentException(
+                "Illegal argument type: " + value == null ? "null" : 
value.getClass().toString()); //$NON-NLS-1$ //$NON-NLS-2$
+    }
+
+    private ValidationResult validate(IconWithDefault iconWithDefault) {
+        return new IconValidation().validate(iconWithDefault.getIcon());
+    }
+}
diff --git 
a/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/ValidationResult.java
 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/ValidationResult.java
index 99ded31..fef57e7 100644
--- 
a/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/ValidationResult.java
+++ 
b/frontend/webadmin/modules/uicommonweb/src/main/java/org/ovirt/engine/ui/uicommonweb/validation/ValidationResult.java
@@ -33,7 +33,7 @@
 
     public ValidationResult()
     {
-        this(true, new ArrayList<String>());
+        this(true, new ArrayList<String>(0));
     }
 
     public ValidationResult(boolean success, List<String> reasons) {
@@ -48,4 +48,12 @@
     public static ValidationResult fail(String... reasons) {
         return new ValidationResult(false, Arrays.asList(reasons));
     }
+
+    @Override
+    public String toString() {
+        return "ValidationResult{" + //$NON-NLS-1$
+                "success=" + privateSuccess + //$NON-NLS-1$
+                ", reasons=" + privateReasons + //$NON-NLS-1$
+                "}"; //$NON-NLS-1$
+    }
 }
diff --git 
a/frontend/webadmin/modules/uicompat/src/main/java/org/ovirt/engine/ui/uicompat/UIConstants.java
 
b/frontend/webadmin/modules/uicompat/src/main/java/org/ovirt/engine/ui/uicompat/UIConstants.java
index aa836f0..c0cd5d7 100644
--- 
a/frontend/webadmin/modules/uicompat/src/main/java/org/ovirt/engine/ui/uicompat/UIConstants.java
+++ 
b/frontend/webadmin/modules/uicompat/src/main/java/org/ovirt/engine/ui/uicompat/UIConstants.java
@@ -944,7 +944,9 @@
     String diskExistsOnAllActiveStorageDomainsMsg();
 
     @DefaultStringValue("The Template that the VM is based on does not exist 
on any active Storage Domain")
-    String noActiveStorageDomainWithTemplateMsg();;
+    String noActiveStorageDomainWithTemplateMsg();
+
+    ;
 
     @DefaultStringValue("Field value should follow: 
<parameter=value;parameter=value;...>")
     String fieldValueShouldFollowMsg();
@@ -2500,4 +2502,4 @@
 
     @DefaultStringValue("This field is not a valid Guid (use 0-9,A-F format: 
00000000-0000-0000-0000-000000000000)")
     String invalidGuidMsg();
-}
+}
\ No newline at end of file
diff --git 
a/frontend/webadmin/modules/uicompat/src/main/java/org/ovirt/engine/ui/uicompat/UIMessages.java
 
b/frontend/webadmin/modules/uicompat/src/main/java/org/ovirt/engine/ui/uicompat/UIMessages.java
index 64d019b..3eb59e4 100644
--- 
a/frontend/webadmin/modules/uicompat/src/main/java/org/ovirt/engine/ui/uicompat/UIMessages.java
+++ 
b/frontend/webadmin/modules/uicompat/src/main/java/org/ovirt/engine/ui/uicompat/UIMessages.java
@@ -1,5 +1,7 @@
 package org.ovirt.engine.ui.uicompat;
 
+import com.google.gwt.i18n.client.Messages;
+
 import java.util.Date;
 import java.util.List;
 
@@ -415,4 +417,13 @@
 
     @DefaultMessage("Do you approve trusting self signed certificate subject 
{0}, SHA-1 fingerprint {1}?")
     String approveRootCertificateTrust(String subject, String sha1Fingerprint);
+
+    @DefaultMessage("Icon dimensions are too large: {0}x{1}, maximum allowed: 
{2}x{3}")
+    String iconDimensionsTooLarge(int width, int height, int maxWidht, int 
maxHeight);
+
+    @DefaultMessage("Icon file is too large. Maximum size is {0}kB.")
+    String iconFileTooLarge(int maxSize);
+
+    @DefaultMessage("Unknown file format. Supported formats are {0}.")
+    String invalidIconFormat(String s);
 }


-- 
To view, visit http://gerrit.ovirt.org/37985
To unsubscribe, visit http://gerrit.ovirt.org/settings

Gerrit-MessageType: newchange
Gerrit-Change-Id: Ibd5e8c120282e35d5940a66286d600b128c57753
Gerrit-PatchSet: 1
Gerrit-Project: ovirt-engine
Gerrit-Branch: master
Gerrit-Owner: Jakub Niedermertl <jnied...@redhat.com>
_______________________________________________
Engine-patches mailing list
Engine-patches@ovirt.org
http://lists.ovirt.org/mailman/listinfo/engine-patches

Reply via email to