mirror of
https://github.com/xcp-ng/xenadmin.git
synced 2025-01-12 13:32:27 +01:00
7c0bc50b4a
Inc. Signed-off-by: Gabor Apati-Nagy<gabor.apati-nagy@citrix.com>
406 lines
15 KiB
C#
406 lines
15 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.Collections.Generic;
|
|
using System.ComponentModel;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Windows.Forms;
|
|
using XenAdmin.Core;
|
|
using XenAdmin.Network;
|
|
using XenAPI;
|
|
|
|
namespace XenAdmin.Controls
|
|
{
|
|
class FilterLocationToolStripDropDownButton : ToolStripDropDownButton
|
|
{
|
|
[Browsable(true)]
|
|
public event Action FilterChanged;
|
|
|
|
/// <summary>
|
|
/// Maintain a list of all the objects we currently have events on for clearing out on rebuild
|
|
/// </summary>
|
|
private List<IXenConnection> connectionsWithEvents = new List<IXenConnection>();
|
|
private List<Pool> poolsWithEvents = new List<Pool>();
|
|
private List<Host> hostsWithEvents = new List<Host>();
|
|
/// <summary>
|
|
/// Store only host check states because the pools can be in an
|
|
/// indeterminate state.
|
|
/// </summary>
|
|
private Dictionary<string, bool> HostCheckStates = new Dictionary<string, bool>();
|
|
private readonly CollectionChangeEventHandler m_hostCollectionChangedWithInvoke;
|
|
private bool inFilterListUpdate;
|
|
private bool retryFilterListUpdate;
|
|
|
|
private ToolStripMenuItem toolStripMenuItemAll;
|
|
|
|
public FilterLocationToolStripDropDownButton()
|
|
{
|
|
ConnectionsManager.XenConnections.CollectionChanged += XenConnections_CollectionChanged;
|
|
m_hostCollectionChangedWithInvoke = Program.ProgramInvokeHandler(Host_CollectionChanged);
|
|
}
|
|
|
|
protected override void Dispose(bool disposing)
|
|
{
|
|
base.Dispose(disposing);
|
|
DeregisterEvents();
|
|
ConnectionsManager.XenConnections.CollectionChanged -= XenConnections_CollectionChanged;
|
|
}
|
|
|
|
private void OnFilterChanged()
|
|
{
|
|
if (FilterChanged != null)
|
|
FilterChanged();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Build the list of hosts to filter by for the first time and set all
|
|
/// of them to be checked
|
|
/// </summary>
|
|
public void InitializeHostList()
|
|
{
|
|
foreach (IXenConnection c in ConnectionsManager.XenConnectionsCopy)
|
|
{
|
|
foreach (Host h in c.Cache.Hosts)
|
|
HostCheckStates[h.uuid] = true;
|
|
}
|
|
}
|
|
|
|
public void RefreshLists()
|
|
{
|
|
BuildFilterList();
|
|
OnFilterChanged();
|
|
}
|
|
|
|
public void BuildFilterList()
|
|
{
|
|
Program.AssertOnEventThread();
|
|
|
|
if (inFilterListUpdate)
|
|
{
|
|
// queue up an update after the current one has finished,
|
|
// in case the update has missed the relevant change
|
|
retryFilterListUpdate = true;
|
|
return;
|
|
}
|
|
|
|
inFilterListUpdate = true;
|
|
|
|
try
|
|
{
|
|
DropDownItems.Clear();
|
|
DeregisterEvents();
|
|
RegisterEvents();
|
|
|
|
foreach (IXenConnection c in ConnectionsManager.XenConnectionsCopy)
|
|
{
|
|
Pool p = Helpers.GetPool(c);
|
|
|
|
if (p == null)// Stand alone host
|
|
{
|
|
foreach (Host h in c.Cache.Hosts)
|
|
{
|
|
var item = GenerateFilterItem(h, h.uuid);
|
|
item.Checked = HostCheckStates.ContainsKey(h.uuid);
|
|
DropDownItems.Add(item);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
DropDownItems.Add(GeneratePoolFilterItem(p));
|
|
}
|
|
|
|
if (DropDownItems.Count > 0)
|
|
{
|
|
toolStripMenuItemAll = new ToolStripMenuItem
|
|
{
|
|
Text = Messages.FILTER_SHOW_ALL,
|
|
Enabled = FilterIsOn
|
|
};
|
|
|
|
DropDownItems.AddRange(new ToolStripItem[]
|
|
{
|
|
new ToolStripSeparator(),
|
|
toolStripMenuItemAll
|
|
});
|
|
}
|
|
|
|
Enabled = DropDownItems.Count > 0;
|
|
}
|
|
finally
|
|
{
|
|
inFilterListUpdate = false;
|
|
if (retryFilterListUpdate)
|
|
{
|
|
// there was a request to update while we were building,
|
|
// rebuild in case we missed something
|
|
retryFilterListUpdate = false;
|
|
BuildFilterList();
|
|
}
|
|
}
|
|
}
|
|
|
|
public bool HideByLocation(string hostUuid)
|
|
{
|
|
if (hostUuid == null)
|
|
return false;
|
|
|
|
if (HostCheckStates.ContainsKey(hostUuid) && !HostCheckStates[hostUuid])
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
public bool HideByLocation(List<string> hostUuids)
|
|
{
|
|
if (hostUuids.Count == 0)
|
|
return false;
|
|
|
|
return hostUuids.TrueForAll(uuid => HostCheckStates.ContainsKey(uuid) && !HostCheckStates[uuid]);
|
|
}
|
|
|
|
public bool FilterIsOn
|
|
{
|
|
get { return HostCheckStates.ContainsValue(false); }
|
|
}
|
|
|
|
private ToolStripMenuItem GeneratePoolFilterItem(Pool p)
|
|
{
|
|
List<ToolStripMenuItem> subItems = new List<ToolStripMenuItem>();
|
|
|
|
foreach (Host h in p.Connection.Cache.Hosts)
|
|
{
|
|
var hostItem = GenerateFilterItem(h, h.uuid);
|
|
hostItem.Checked = HostCheckStates.ContainsKey(h.uuid);
|
|
subItems.Add(hostItem);
|
|
}
|
|
|
|
var poolItem = GenerateFilterItem(p, p.uuid);
|
|
poolItem.DropDownItems.AddRange(subItems.ToArray());
|
|
poolItem.CheckState = subItems.TrueForAll(item => item.Checked)
|
|
? CheckState.Checked
|
|
: subItems.TrueForAll(item => !item.Checked)
|
|
? CheckState.Unchecked
|
|
: CheckState.Indeterminate;
|
|
poolItem.DropDownItemClicked += poolItem_DropDownItemClicked;
|
|
return poolItem;
|
|
}
|
|
|
|
private ToolStripMenuItem GenerateFilterItem(IXenObject xenObject, string xenObjectUuid)
|
|
{
|
|
var item = new ToolStripMenuItem
|
|
{
|
|
Text = Helpers.GetName(xenObject),
|
|
Tag = xenObjectUuid
|
|
};
|
|
return item;
|
|
}
|
|
|
|
private void RegisterEvents()
|
|
{
|
|
foreach (IXenConnection c in ConnectionsManager.XenConnectionsCopy)
|
|
{
|
|
c.ConnectionStateChanged += connection_ConnectionStateChanged;
|
|
c.Cache.RegisterCollectionChanged<Host>(m_hostCollectionChangedWithInvoke);
|
|
c.CachePopulated += connection_CachePopulated;
|
|
connectionsWithEvents.Add(c);
|
|
|
|
foreach (var pool in c.Cache.Pools)
|
|
{
|
|
pool.PropertyChanged += pool_PropertyChanged;
|
|
poolsWithEvents.Add(pool);
|
|
}
|
|
|
|
foreach (Host host in c.Cache.Hosts)
|
|
{
|
|
RegisterHostEvents(host);
|
|
hostsWithEvents.Add(host);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void DeregisterEvents()
|
|
{
|
|
foreach (IXenConnection c in connectionsWithEvents)
|
|
{
|
|
c.ConnectionStateChanged -= connection_ConnectionStateChanged;
|
|
c.Cache.DeregisterCollectionChanged<Host>(m_hostCollectionChangedWithInvoke);
|
|
c.CachePopulated -= connection_CachePopulated;
|
|
}
|
|
|
|
foreach (var pool in poolsWithEvents)
|
|
pool.PropertyChanged -= pool_PropertyChanged;
|
|
|
|
foreach (Host h in hostsWithEvents)
|
|
DeregisterHostEvents(h);
|
|
|
|
connectionsWithEvents.Clear();
|
|
poolsWithEvents.Clear();
|
|
hostsWithEvents.Clear();
|
|
}
|
|
|
|
private void RegisterHostEvents(Host host)
|
|
{
|
|
Host_metrics metrics = host.Connection.Resolve(host.metrics);
|
|
if (metrics != null)
|
|
metrics.PropertyChanged += hostMetrics_PropertyChanged;
|
|
host.PropertyChanged += host_PropertyChanged;
|
|
}
|
|
|
|
private void DeregisterHostEvents(Host host)
|
|
{
|
|
Host_metrics metrics = host.Connection.Resolve(host.metrics);
|
|
if (metrics != null)
|
|
metrics.PropertyChanged -= hostMetrics_PropertyChanged;
|
|
host.PropertyChanged -= host_PropertyChanged;
|
|
}
|
|
|
|
private void connection_ConnectionStateChanged(object sender, EventArgs e)
|
|
{
|
|
Program.Invoke(Program.MainWindow, RefreshLists);
|
|
}
|
|
|
|
private void connection_CachePopulated(object sender, EventArgs e)
|
|
{
|
|
Program.Invoke(Program.MainWindow, RefreshLists);
|
|
}
|
|
|
|
private void XenConnections_CollectionChanged(object sender, CollectionChangeEventArgs e)
|
|
{
|
|
if (e.Action == CollectionChangeAction.Add)
|
|
{
|
|
IXenConnection connection = e.Element as IXenConnection;
|
|
|
|
foreach (Host host in connection.Cache.Hosts)
|
|
HostCheckStates[host.uuid] = true;
|
|
}
|
|
}
|
|
|
|
private void Host_CollectionChanged(object sender, CollectionChangeEventArgs e)
|
|
{
|
|
if (e.Action == CollectionChangeAction.Add)
|
|
HostCheckStates[((Host)e.Element).uuid] = true;
|
|
|
|
Program.Invoke(Parent, RefreshLists);
|
|
}
|
|
|
|
private void pool_PropertyChanged(object obj, PropertyChangedEventArgs e)
|
|
{
|
|
if (e.PropertyName == "other_config" || e.PropertyName == "name_label")
|
|
Program.Invoke(Parent, RefreshLists);
|
|
}
|
|
|
|
private void hostMetrics_PropertyChanged(object sender, PropertyChangedEventArgs e)
|
|
{
|
|
if (e.PropertyName == "live")
|
|
Program.Invoke(Parent, RefreshLists);
|
|
}
|
|
|
|
private void host_PropertyChanged(object sender, PropertyChangedEventArgs e)
|
|
{
|
|
if (e.PropertyName == "name_label" || e.PropertyName == "metrics")
|
|
Program.Invoke(Parent, RefreshLists);
|
|
}
|
|
|
|
protected override void OnDropDownItemClicked(ToolStripItemClickedEventArgs e)
|
|
{
|
|
//this method pertains to pool or stand alone host items
|
|
base.OnDropDownItemClicked(e);
|
|
HandleItemClicked(e.ClickedItem as ToolStripMenuItem);
|
|
}
|
|
|
|
private void poolItem_DropDownItemClicked(object sender, ToolStripItemClickedEventArgs e)
|
|
{
|
|
HandleItemClicked(e.ClickedItem as ToolStripMenuItem);
|
|
}
|
|
|
|
private void HandleItemClicked(ToolStripMenuItem item)
|
|
{
|
|
if (item == null)
|
|
return;
|
|
|
|
string uuid = (string)item.Tag;
|
|
|
|
if (item.HasDropDownItems)
|
|
{
|
|
//this is a pool node
|
|
|
|
item.CheckState = item.CheckState == CheckState.Checked
|
|
? CheckState.Unchecked
|
|
: CheckState.Checked;
|
|
|
|
foreach (ToolStripMenuItem child in item.DropDownItems)
|
|
{
|
|
child.Checked = item.Checked;
|
|
string hostUuid = (string)child.Tag;
|
|
HostCheckStates[hostUuid] = child.Checked;
|
|
}
|
|
|
|
toolStripMenuItemAll.Enabled = FilterIsOn;
|
|
}
|
|
else if (item == toolStripMenuItemAll)
|
|
{
|
|
toolStripMenuItemAll.Enabled = false;
|
|
InitializeHostList();
|
|
BuildFilterList();
|
|
}
|
|
else
|
|
{
|
|
//this is a host node
|
|
|
|
item.Checked = !item.Checked;
|
|
HostCheckStates[uuid] = item.Checked;
|
|
|
|
ToolStripMenuItem poolItem = item.OwnerItem as ToolStripMenuItem;
|
|
|
|
if (poolItem != null)
|
|
{
|
|
//this is not a standalone host; change the parent pool's check state
|
|
|
|
var itemArray = new ToolStripMenuItem[poolItem.DropDownItems.Count];
|
|
poolItem.DropDownItems.CopyTo(itemArray, 0);
|
|
|
|
poolItem.CheckState = Array.TrueForAll(itemArray, i => i.Checked)
|
|
? CheckState.Checked
|
|
: Array.TrueForAll(itemArray, i => !i.Checked)
|
|
? CheckState.Unchecked
|
|
: CheckState.Indeterminate;
|
|
}
|
|
|
|
toolStripMenuItemAll.Enabled = FilterIsOn;
|
|
}
|
|
|
|
OnFilterChanged();
|
|
}
|
|
}
|
|
}
|