Files
Aaru/Aaru.Gui/Controls/BlockMap.cs

115 lines
3.8 KiB
C#
Raw Normal View History

using System;
using System.Collections.ObjectModel;
using System.Collections.Specialized;
using System.Linq;
2020-02-27 00:33:26 +00:00
namespace Aaru.Gui.Controls
{
public class BlockMap : ColoredGrid
{
ulong _sectors;
public uint blocksToRead;
public uint sectorsToRead;
public BlockMap()
{
ColoredSectors = new ObservableCollection<ColoredBlock>();
ColoredSectors.CollectionChanged += OnColoredSectorsChanged;
}
public ulong Sectors
{
get => _sectors;
set
{
_sectors = value;
CalculateBoundaries();
Invalidate();
}
}
public ulong SectorsPerBlock { get; private set; }
public uint SectorsToRead
{
get => sectorsToRead;
set
{
sectorsToRead = value;
blocksToRead = (uint)(sectorsToRead / SectorsPerBlock);
if(sectorsToRead % SectorsPerBlock > 0) blocksToRead++;
}
}
public ObservableCollection<ColoredBlock> ColoredSectors { get; }
void OnColoredSectorsChanged(object sender, NotifyCollectionChangedEventArgs args)
{
switch(args.Action)
{
case NotifyCollectionChangedAction.Add:
foreach(object item in args.NewItems)
{
if(!(item is ColoredBlock block)) continue;
for(ulong i = 0; i < blocksToRead; i++)
ColoredBlocks.Add(new ColoredBlock(block.Block / SectorsPerBlock + i, block.Color));
}
break;
case NotifyCollectionChangedAction.Move: break;
case NotifyCollectionChangedAction.Remove:
foreach(object item in args.OldItems)
{
if(!(item is ColoredBlock block)) continue;
for(ulong i = 0; i < blocksToRead; i++)
ColoredBlocks.Remove(ColoredBlocks.FirstOrDefault(t => t.Block == block.Block /
SectorsPerBlock + i));
}
break;
case NotifyCollectionChangedAction.Replace:
foreach(object item in args.OldItems)
{
if(!(item is ColoredBlock block)) continue;
for(ulong i = 0; i < blocksToRead; i++)
ColoredBlocks.Remove(ColoredBlocks.FirstOrDefault(t => t.Block == block.Block /
SectorsPerBlock + i));
}
foreach(object item in args.NewItems)
{
if(!(item is ColoredBlock block)) continue;
for(ulong i = 0; i < blocksToRead; i++)
ColoredBlocks.Add(new ColoredBlock(block.Block / SectorsPerBlock + i, block.Color));
}
break;
case NotifyCollectionChangedAction.Reset:
ColoredBlocks.Clear();
break;
default: throw new ArgumentOutOfRangeException();
}
}
void CalculateBoundaries()
{
SectorsPerBlock = Blocks == 0 ? 0 : _sectors / Blocks;
ColoredBlocks.Clear();
if(SectorsPerBlock > 0) return;
SectorsPerBlock = 1;
for(ulong i = Sectors; i < Blocks; i++) ColoredBlocks.Add(new ColoredBlock(i, GridColor));
}
public void Clear()
{
ColoredSectors.Clear();
}
}
}