/* Copyright (c) Citrix Systems Inc. * All rights reserved. * * Redistribution and use in source and binary forms, * with or without modification, are permitted provided * that the following conditions are met: * * * Redistributions of source code must retain the above * copyright notice, this list of conditions and the * following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the * following disclaimer in the documentation and/or other * materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ using System; using System.Collections.Generic; using System.ComponentModel; using System.Linq; using System.Windows.Forms; using XenAdmin.Commands; using XenAdmin.Controls; using XenAdmin.Core; using XenAdmin.Mappings; using XenAPI; namespace XenAdmin.Wizards.GenericPages { /// /// Class representing the page of the ImportAppliance wizard where the user specifies /// the targets where the VMs of the appliance will be imported /// internal abstract partial class SelectMultipleVMDestinationPage : XenTabPage { private Dictionary m_vmMappings; private IXenObject m_selectedObject; private bool updatingDestinationCombobox; private bool m_buttonNextEnabled; /// /// Combobox item that can executes a command but also be an IEnableableComboBoxItem /// private class AddHostExecutingComboBoxItem : IEnableableComboBoxItem { public override string ToString() { return Messages.ADD_POOL_OR_SERVER; } public void ExecuteCommand(Control parent) { new AddHostCommand(Program.MainWindow.CommandInterface, parent).Execute(); } public bool Enabled { get { return true; } } } protected SelectMultipleVMDestinationPage() { InitializeComponent(); InitializeText(); ConnectionsManager.XenConnections.CollectionChanged += CollectionChanged; ShowWarning(null); } public override void PageCancelled() { Program.Invoke(Program.MainWindow, ClearComboBox); Program.Invoke(Program.MainWindow, ClearDataGridView); ChosenItem = null; } public override void PageLeave(PageLoadedDirection direction, ref bool cancel) { SetDefaultTarget(ChosenItem); Program.Invoke(Program.MainWindow, ClearComboBox); base.PageLeave(direction, ref cancel); } private void InitializeText() { m_labelIntro.Text = InstructionText; label1.Text = HomeServerText; label2.Text = HomeServerSelectionIntroText; } private IXenObject _chosenItem; public IXenObject ChosenItem { get { return _chosenItem; } private set { _chosenItem = value; OnChosenItemChanged(); } } /// /// Text containing instructions for use - at the top of the page /// protected abstract string InstructionText { get; } /// /// Text demarking what the label fot the home server drop down should be /// protected abstract string HomeServerText { get; } /// /// Text above the table containing a list of VMs and concomitant home server /// protected abstract string HomeServerSelectionIntroText { get; } protected virtual void OnChosenItemChanged() {} protected void ShowWarning(string warningText) { if (string.IsNullOrEmpty(warningText)) tableLayoutPanelWarning.Visible = false; else { labelWarning.Text = warningText; tableLayoutPanelWarning.Visible = true; } } #region Base class (XenTabPage) overrides public override void PageLoaded(PageLoadedDirection direction) { base.PageLoaded(direction); ChosenItem = null; restoreGridHomeServerSelection = (direction == PageLoadedDirection.Back); PopulateComboBox(); } public override bool EnableNext() { return m_buttonNextEnabled; } #endregion #region Accessors public Dictionary VmMappings { get { foreach (DataGridViewRow row in m_dataGridView.Rows) { string sysId = (string)row.Cells[0].Tag; if (m_vmMappings.ContainsKey(sysId)) { var mapping = m_vmMappings[sysId]; DataGridViewEnableableComboBoxCell cbCell = row.Cells[m_colTarget.Index] as DataGridViewEnableableComboBoxCell; System.Diagnostics.Debug.Assert(cbCell != null, "ComboBox cell was not found"); IEnableableXenObjectComboBoxItem selectedItem = cbCell.Value as IEnableableXenObjectComboBoxItem; System.Diagnostics.Debug.Assert(selectedItem != null, "Vm has no target mapped"); var type = selectedItem.Item.GetType(); if (type == typeof(Pool)) mapping.XenRef = new XenRef(selectedItem.Item.opaque_ref); else if (type == typeof(Host)) mapping.XenRef = new XenRef(selectedItem.Item.opaque_ref); mapping.TargetName = selectedItem.Item.Name; } } return m_vmMappings; } set { m_vmMappings = value; } } #endregion public void SetDefaultTarget(IXenObject xenObject) { m_selectedObject = xenObject; } public abstract DelayLoadingOptionComboBoxItem CreateDelayLoadingOptionComboBoxItem(IXenObject xenItem); #region Private methods protected void SetButtonNextEnabled(bool enabled) { m_buttonNextEnabled = enabled; OnPageUpdated(); } private void ClearComboBox() { Program.AssertOnEventThread(); foreach (var item in m_comboBoxConnection.Items) { DelayLoadingOptionComboBoxItem tempItem = item as DelayLoadingOptionComboBoxItem; if (tempItem != null) tempItem.ReasonUpdated -= DelayLoadedComboBoxItem_ReasonChanged; } m_comboBoxConnection.Items.Clear(); } private void ClearDataGridView() { //Clear up comboboxes foreach (DataGridViewRow row in m_dataGridView.Rows) { row.Cells[m_colTarget.Index].Dispose(); } m_dataGridView.Rows.Clear(); m_dataGridView.Refresh(); } private void PopulateComboBox() { Program.AssertOnEventThread(); Program.Invoke(Program.MainWindow, ClearDataGridView); updatingDestinationCombobox = true; Program.Invoke(Program.MainWindow, ClearComboBox); foreach (var xenConnection in ConnectionsManager.XenConnectionsCopy.Where(con => con.IsConnected)) { DelayLoadingOptionComboBoxItem item = null; Pool pool = Helpers.GetPool(xenConnection); if (pool == null) { Host host = Helpers.GetMaster(xenConnection); if (host != null) { item = CreateDelayLoadingOptionComboBoxItem(host); m_comboBoxConnection.Items.Add(item); item.ReasonUpdated += DelayLoadedComboBoxItem_ReasonChanged; item.Load(); host.PropertyChanged -= PropertyChanged; host.PropertyChanged += PropertyChanged; } } else { item = CreateDelayLoadingOptionComboBoxItem(pool); m_comboBoxConnection.Items.Add(item); item.ReasonUpdated += DelayLoadedComboBoxItem_ReasonChanged; item.Load(); pool.PropertyChanged -= PropertyChanged; pool.PropertyChanged += PropertyChanged; } if (item != null && m_selectedObject != null && item.Item.Connection == m_selectedObject.Connection) item.PreferAsSelectedItem = true; xenConnection.ConnectionStateChanged -= xenConnection_ConnectionStateChanged; xenConnection.ConnectionStateChanged += xenConnection_ConnectionStateChanged; xenConnection.CachePopulated -= xenConnection_CachePopulated; xenConnection.CachePopulated += xenConnection_CachePopulated; xenConnection.Cache.RegisterCollectionChanged(Program.ProgramInvokeHandler(CollectionChanged)); } m_comboBoxConnection.Items.Add(new AddHostExecutingComboBoxItem()); updatingDestinationCombobox = false; } protected bool DestinationHasBeenSelected() { return m_comboBoxConnection.SelectedItem != null; } private bool MatchingWithXenRefObject(IEnableableXenObjectComboBoxItem item, object xenRef) { XenRef hostRef = xenRef as XenRef; if (hostRef != null) return hostRef.opaque_ref == item.Item.opaque_ref; XenRef poolRef = xenRef as XenRef; if (poolRef != null) return poolRef.opaque_ref == item.Item.opaque_ref; return false; } private bool restoreGridHomeServerSelection; private void RestoreGridHomeServerSelectionFromMapping() { foreach (DataGridViewRow row in m_dataGridView.Rows) { string sysId = (string)row.Cells[0].Tag; if (m_vmMappings.ContainsKey(sysId)) { var mapping = m_vmMappings[sysId]; DataGridViewEnableableComboBoxCell cbCell = row.Cells[m_colTarget.Index] as DataGridViewEnableableComboBoxCell; if (cbCell == null) return; List list = cbCell.Items.OfType().ToList(); IEnableableXenObjectComboBoxItem item = list.FirstOrDefault(cbi => MatchingWithXenRefObject(cbi, mapping.XenRef)); if (item != null) cbCell.Value = item; } } } private bool updatingHomeServerList; private void PopulateDataGridView(List homeserverFilters) { Program.AssertOnEventThread(); updatingHomeServerList = true; try { Connection = null; var target = m_comboBoxConnection.SelectedItem as DelayLoadingOptionComboBoxItem; if (target != null) Connection = target.Item.Connection; ClearDataGridView(); SetButtonNextEnabled(true); var hasPoolSharedStorage = HasPoolSharedStorage(); foreach (var kvp in m_vmMappings) { var tb = new DataGridViewTextBoxCell {Value = kvp.Value.VmNameLabel, Tag = kvp.Key}; var cb = new DataGridViewEnableableComboBoxCell{FlatStyle = FlatStyle.Flat}; if (Connection != null) { if (hasPoolSharedStorage) { foreach (var pool in Connection.Cache.Pools) { var item = new DoNotAssignHomeServerPoolItem(pool); cb.Items.Add(item); if ((m_selectedObject != null && m_selectedObject.opaque_ref == pool.opaque_ref) || (target != null && target.Item.opaque_ref == pool.opaque_ref)) cb.Value = item; break; //there exists one pool per connection } } foreach (var host in Connection.Cache.Hosts) { HomeServerItem item = new HomeServerItem(host, homeserverFilters); cb.Items.Add(item); if ((m_selectedObject != null && m_selectedObject.opaque_ref == host.opaque_ref) || (target != null && target.Item.opaque_ref == host.opaque_ref)) cb.Value = item; } } SetComboBoxPreSelection(cb); var row = new DataGridViewRow(); row.Cells.AddRange(tb, cb); m_dataGridView.Rows.Add(row); } HelpersGUI.ResizeLastGridViewColumn(m_colTarget); //set properly the width of the last column if (restoreGridHomeServerSelection) { RestoreGridHomeServerSelectionFromMapping(); restoreGridHomeServerSelection = false; } } finally { updatingHomeServerList = false; } } private void SetComboBoxPreSelection(DataGridViewEnableableComboBoxCell cb) { if (cb.Value == null) { if (cb.Items.Count > 0) { if (cb.Items.OfType().All(i => !i.Enabled)) { cb.Value = null; SetButtonNextEnabled(false); } else cb.Value = cb.Items.OfType().First(i => i.Enabled); } else SetButtonNextEnabled(false); //do not allow to leave the page if a vm has no target } } private bool HasPoolSharedStorage() { if (Connection == null) return false; foreach (var pbd in Connection.Cache.PBDs.Where(thePbd => thePbd.SR != null)) { var sr = Connection.Resolve(pbd.SR); if (sr != null && sr.SupportsVdiCreate() && sr.shared) return true; } return false; } #endregion #region Event Handlers private void DelayLoadedComboBoxItem_ReasonChanged(object sender, EventArgs e) { DelayLoadingOptionComboBoxItem item = sender as DelayLoadingOptionComboBoxItem; if (item == null) throw new NullReferenceException("Trying to update delay loaded reason but failed to extract reason"); int index = m_comboBoxConnection.Items.IndexOf(item); if (index > -1) { Program.Invoke( Program.MainWindow, delegate() { int selectedIndex = m_comboBoxConnection.SelectedIndex; if (index > m_comboBoxConnection.Items.Count) return; if(updatingDestinationCombobox || updatingHomeServerList) return; DelayLoadingOptionComboBoxItem tempItem = m_comboBoxConnection.Items[index] as DelayLoadingOptionComboBoxItem; if(tempItem == null) throw new NullReferenceException("Trying to update delay loaded reason but failed to extract reason"); tempItem.CopyFrom(item); m_comboBoxConnection.BeginUpdate(); m_comboBoxConnection.Items.RemoveAt(index); if (updatingDestinationCombobox || updatingHomeServerList) { m_comboBoxConnection.EndUpdate(); return; } m_comboBoxConnection.Items.Insert(index, tempItem); m_comboBoxConnection.SelectedIndex = selectedIndex; m_comboBoxConnection.EndUpdate(); if (tempItem.PreferAsSelectedItem) m_comboBoxConnection.SelectedItem = tempItem; item.ReasonUpdated -= DelayLoadedComboBoxItem_ReasonChanged; }); } } private void PropertyChanged(object sender, PropertyChangedEventArgs e) { if (e.PropertyName == "name_label" || e.PropertyName == "metrics" || e.PropertyName == "enabled" || e.PropertyName == "live" || e.PropertyName == "patches") Program.Invoke(this, PopulateComboBox); } private void CollectionChanged(object sender, CollectionChangeEventArgs e) { Program.BeginInvoke(this, PopulateComboBox); } private void xenConnection_CachePopulated(object sender, EventArgs e) { Program.Invoke(this, PopulateComboBox); } private void xenConnection_ConnectionStateChanged(object sender, EventArgs e) { Program.Invoke(this, PopulateComboBox); } #endregion #region Control event handlers private void m_comboBoxConnection_SelectedIndexChanged(object sender, EventArgs e) { if (updatingHomeServerList) return; //If the item is delay loading and them item is disabled, null the selection made //and clear the table containing server data IEnableableXenObjectComboBoxItem item = m_comboBoxConnection.SelectedItem as IEnableableXenObjectComboBoxItem; if(item != null && !item.Enabled) { m_comboBoxConnection.SelectedIndex = -1; m_dataGridView.Rows.Clear(); ChosenItem = null; return; } AddHostExecutingComboBoxItem exeItem = m_comboBoxConnection.SelectedItem as AddHostExecutingComboBoxItem; if (exeItem != null && !updatingDestinationCombobox) exeItem.ExecuteCommand(this); else if(!updatingDestinationCombobox) { try { Cursor.Current = Cursors.WaitCursor; ChosenItem = item == null ? null : item.Item; Program.Invoke(Program.MainWindow, ()=> PopulateDataGridView(CreateHomeServerFilterList(item))); } finally { Cursor.Current = Cursors.Default; } } IsDirty = true; } /// /// Create a set of filters for the homeserver combo box selection /// /// selected item from the host combobox /// protected virtual List CreateHomeServerFilterList(IEnableableXenObjectComboBoxItem item) { return new List(); } private void m_dataGridView_CellClick(object sender, DataGridViewCellEventArgs e) { if (e.ColumnIndex != m_colTarget.Index || e.RowIndex < 0 || e.RowIndex >= m_dataGridView.RowCount) return; m_dataGridView.BeginEdit(false); if (m_dataGridView.EditingControl != null && m_dataGridView.EditingControl is ComboBox) (m_dataGridView.EditingControl as ComboBox).DroppedDown = true; } private void m_dataGridView_CurrentCellDirtyStateChanged(object sender, EventArgs e) { m_dataGridView.CommitEdit(DataGridViewDataErrorContexts.Commit); IsDirty = true; } #endregion } }