mirror of
https://github.com/xcp-ng/xenadmin.git
synced 2025-01-20 15:29:26 +01:00
5b79b1dc94
hidden and in all other cases it was set by the implementing control, i.e. the label was not reusable, hence no reason to have it in the control and risk layout issues. This means that the SrPicker does not need to be a control encapsulating a CustomTreeVIew, but simply derive from the latter. Signed-off-by: Konstantina Chremmou <konstantina.chremmou@citrix.com>
382 lines
12 KiB
C#
382 lines
12 KiB
C#
/* 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.ComponentModel;
|
|
using System.Windows.Forms;
|
|
using XenAdmin.Actions;
|
|
using XenAdmin.Network;
|
|
using XenAPI;
|
|
using XenAdmin.Core;
|
|
|
|
|
|
namespace XenAdmin.Controls
|
|
{
|
|
public partial class SrPicker : CustomTreeView
|
|
{
|
|
// Migrate is the live VDI move operation
|
|
public enum SRPickerType { VM, InstallFromTemplate, MoveOrCopy, Migrate, LunPerVDI };
|
|
private SRPickerType usage = SRPickerType.VM;
|
|
|
|
//Used in the MovingVDI usage
|
|
private VDI[] existingVDIs;
|
|
public void SetExistingVDIs(VDI[] vdis)
|
|
{
|
|
existingVDIs = vdis;
|
|
}
|
|
|
|
private IXenConnection connection;
|
|
private Host affinity;
|
|
private SrPickerItem LastSelectedItem;
|
|
|
|
public event Action<object> SrSelectionChanged;
|
|
public long DiskSize = 0;
|
|
|
|
private readonly CollectionChangeEventHandler SR_CollectionChangedWithInvoke;
|
|
|
|
public SrPicker()
|
|
{
|
|
SR_CollectionChangedWithInvoke = Program.ProgramInvokeHandler(SR_CollectionChanged);
|
|
}
|
|
|
|
public override bool ShowCheckboxes
|
|
{
|
|
get { return false; }
|
|
}
|
|
|
|
public override bool ShowDescription
|
|
{
|
|
get { return true; }
|
|
}
|
|
|
|
public override bool ShowImages
|
|
{
|
|
get { return true; }
|
|
}
|
|
|
|
public override int NodeIndent
|
|
{
|
|
get { return 3; }
|
|
}
|
|
|
|
public SRPickerType Usage
|
|
{
|
|
set { usage = value; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// For new disk dialog only
|
|
/// </summary>
|
|
public IXenConnection Connection
|
|
{
|
|
get
|
|
{
|
|
return connection;
|
|
}
|
|
set
|
|
{
|
|
if (value == null)
|
|
return;
|
|
connection = value;
|
|
|
|
Pool pool = Helpers.GetPoolOfOne(connection);
|
|
if (pool != null)
|
|
{
|
|
pool.PropertyChanged -= Server_PropertyChanged;
|
|
pool.PropertyChanged += Server_PropertyChanged;
|
|
}
|
|
|
|
connection.Cache.RegisterCollectionChanged<SR>(SR_CollectionChangedWithInvoke);
|
|
|
|
refresh();
|
|
foreach (SrPickerItem srItem in Items)
|
|
{
|
|
SrRefreshAction a = new SrRefreshAction(srItem.TheSR, true);
|
|
a.RunAsync();
|
|
}
|
|
}
|
|
}
|
|
|
|
protected override void OnSelectedIndexChanged(EventArgs e)
|
|
{
|
|
base.OnSelectedIndexChanged(e);
|
|
onItemSelect();
|
|
}
|
|
|
|
private void onItemSelect()
|
|
{
|
|
SrPickerItem item = SelectedItem as SrPickerItem;
|
|
if (item == null || !item.Enabled)
|
|
{
|
|
if (SrSelectionChanged != null)
|
|
SrSelectionChanged(null);
|
|
return;
|
|
}
|
|
|
|
if (SrSelectionChanged != null)
|
|
SrSelectionChanged(item);
|
|
|
|
if (!item.Enabled && LastSelectedItem != null && LastSelectedItem.TheSR.opaque_ref != item.TheSR.opaque_ref)
|
|
SelectedItem = LastSelectedItem;
|
|
else if (!item.Enabled && LastSelectedItem != null && LastSelectedItem.TheSR.opaque_ref == item.TheSR.opaque_ref)
|
|
{
|
|
SrPickerItem first = Items[0] as SrPickerItem;
|
|
if (first != null && first.Enabled)
|
|
SelectedItem = first;
|
|
else
|
|
SelectedItem = null;
|
|
}
|
|
else
|
|
LastSelectedItem = item;
|
|
}
|
|
|
|
public SR SR
|
|
{
|
|
get
|
|
{
|
|
return SelectedItem is SrPickerItem && (SelectedItem as SrPickerItem).Enabled ? (SelectedItem as SrPickerItem).TheSR : null;
|
|
}
|
|
}
|
|
|
|
public SR DefaultSR = null;
|
|
|
|
public void SetAffinity(Host host)
|
|
{
|
|
affinity = host;
|
|
refresh();
|
|
}
|
|
|
|
private readonly SrPickerItemFactory itemFactory = new SrPickerItemFactory();
|
|
|
|
public void refresh()
|
|
{
|
|
Program.AssertOnEventThread();
|
|
|
|
SR selectedSr = SR;
|
|
bool selected = false;
|
|
BeginUpdate();
|
|
try
|
|
{
|
|
ClearAllNodes();
|
|
|
|
foreach (SR sr in connection.Cache.SRs)
|
|
{
|
|
SrPickerItem item = itemFactory.PickerItem(sr, usage, affinity, DiskSize, existingVDIs);
|
|
if (item.Show)
|
|
AddNode(item);
|
|
foreach (PBD pbd in sr.Connection.ResolveAll(sr.PBDs))
|
|
{
|
|
if (pbd != null)
|
|
{
|
|
pbd.PropertyChanged -= Server_PropertyChanged;
|
|
pbd.PropertyChanged += Server_PropertyChanged;
|
|
}
|
|
}
|
|
sr.PropertyChanged -= Server_PropertyChanged;
|
|
sr.PropertyChanged += Server_PropertyChanged;
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
EndUpdate();
|
|
}
|
|
|
|
if (selectedSr != null)
|
|
{
|
|
foreach (SrPickerItem node in Items)
|
|
{
|
|
if (node.TheSR != null && node.TheSR.uuid == selectedSr.uuid)
|
|
{
|
|
SelectedItem = node;
|
|
onItemSelect();
|
|
selected = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!selected && Items.Count > 0)
|
|
{// If no selection made, select default SR
|
|
if (!selectDefaultSR())
|
|
{
|
|
// If no default SR, select first entry in list
|
|
SelectedIndex = 0;
|
|
onItemSelect();
|
|
}
|
|
}
|
|
}
|
|
|
|
public void UpdateDiskSize()
|
|
{
|
|
Program.AssertOnEventThread();
|
|
try
|
|
{
|
|
foreach (SrPickerItem node in Items)
|
|
{
|
|
node.UpdateDiskSize(DiskSize);
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
Refresh();
|
|
onItemSelect();
|
|
}
|
|
}
|
|
|
|
private void Server_PropertyChanged(object sender, PropertyChangedEventArgs e)
|
|
{
|
|
if (e.PropertyName == "name_label" || e.PropertyName == "PBDs" || e.PropertyName == "physical_utilisation" || e.PropertyName == "currently_attached" || e.PropertyName == "default_SR")
|
|
Program.Invoke(this, refresh);
|
|
}
|
|
|
|
void SR_CollectionChanged(object sender, CollectionChangeEventArgs e)
|
|
{
|
|
Program.Invoke(this, refresh);
|
|
}
|
|
|
|
private void UnregisterHandlers()
|
|
{
|
|
if (connection == null)
|
|
return;
|
|
|
|
var pool = Helpers.GetPoolOfOne(connection);
|
|
if (pool != null)
|
|
pool.PropertyChanged -= Server_PropertyChanged;
|
|
|
|
foreach (var sr in connection.Cache.SRs)
|
|
{
|
|
foreach (var pbd in sr.Connection.ResolveAll(sr.PBDs))
|
|
{
|
|
if (pbd != null)
|
|
pbd.PropertyChanged -= Server_PropertyChanged;
|
|
}
|
|
sr.PropertyChanged -= Server_PropertyChanged;
|
|
}
|
|
|
|
connection.Cache.DeregisterCollectionChanged<SR>(SR_CollectionChangedWithInvoke);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Selects the default SR, if it exists.
|
|
/// </summary>
|
|
/// <returns>true if the default SR was selected, otherwise false.</returns>
|
|
public bool selectDefaultSR()
|
|
{
|
|
if (DefaultSR == null)
|
|
return false;
|
|
|
|
foreach (SrPickerItem node in Items)
|
|
{
|
|
if (node.TheSR != null && node.TheSR == DefaultSR && node.Enabled)
|
|
{
|
|
SelectedItem = node;
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
internal void selectSRorNone(SR TheSR)
|
|
{
|
|
foreach (SrPickerItem node in Items)
|
|
{
|
|
if (node.TheSR != null && node.TheSR.opaque_ref == TheSR.opaque_ref)
|
|
{
|
|
SelectedItem = node;
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (SrSelectionChanged != null)
|
|
SrSelectionChanged(null);
|
|
}
|
|
|
|
internal void selectDefaultSROrAny()
|
|
{
|
|
if (selectDefaultSR())
|
|
return;
|
|
foreach (SrPickerItem item in Items)
|
|
{
|
|
if (item == null)
|
|
continue;
|
|
if (item.Enabled)
|
|
{
|
|
selectSRorNone(item.TheSR);
|
|
return;
|
|
}
|
|
}
|
|
if (SrSelectionChanged != null)
|
|
SrSelectionChanged(null);
|
|
}
|
|
|
|
public void selectSRorDefaultorAny(SR sr)
|
|
{
|
|
if (sr != null)
|
|
{
|
|
foreach (SrPickerItem node in Items)
|
|
{
|
|
if (node.TheSR != null && node.TheSR.opaque_ref == sr.opaque_ref)
|
|
{
|
|
SelectedItem = node;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
selectDefaultSROrAny();
|
|
}
|
|
|
|
public bool ValidSelectionExists
|
|
{
|
|
get
|
|
{
|
|
foreach (SrPickerItem item in Items)
|
|
{
|
|
if (item == null)
|
|
continue;
|
|
if (item.Enabled)
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
protected override void Dispose(bool disposing)
|
|
{
|
|
if (disposing)
|
|
UnregisterHandlers();
|
|
|
|
base.Dispose(disposing);
|
|
}
|
|
}
|
|
}
|