bazowa wersja aplikacji
This commit is contained in:
commit
879c29cc74
3
.gitignore
vendored
Normal file
3
.gitignore
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
bin/
|
||||||
|
obj/
|
||||||
|
/packages/
|
3
.idea/.gitignore
vendored
Normal file
3
.idea/.gitignore
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
|
||||||
|
# Default ignored files
|
||||||
|
/.idea.Life/.idea/workspace.xml
|
48
.idea/.idea.Life/.idea/contentModel.xml
Normal file
48
.idea/.idea.Life/.idea/contentModel.xml
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="ContentModelStore">
|
||||||
|
<e p="C:\Users\Kacper\.Rider2019.2\system\extResources" t="IncludeRecursive" />
|
||||||
|
<e p="C:\Users\Kacper\.Rider2019.2\system\resharper-host\local\Transient\ReSharperHost\v192\SolutionCaches\_Life.1627287830.00" t="ExcludeRecursive" />
|
||||||
|
<e p="C:\Users\Kacper\Studies\dotnet\Life" t="IncludeFlat">
|
||||||
|
<e p="Life" t="IncludeRecursive">
|
||||||
|
<e p="Annotations.cs" t="Include" />
|
||||||
|
<e p="App.xaml" t="Include" />
|
||||||
|
<e p="App.xaml.cs" t="Include" />
|
||||||
|
<e p="Automaton" t="Include">
|
||||||
|
<e p="BasicField.cs" t="Include" />
|
||||||
|
<e p="Cell.cs" t="Include" />
|
||||||
|
<e p="GameOfLife.cs" t="Include" />
|
||||||
|
<e p="IAutomaton.cs" t="Include" />
|
||||||
|
<e p="IField.cs" t="Include" />
|
||||||
|
<e p="MooreField.cs" t="Include" />
|
||||||
|
</e>
|
||||||
|
<e p="AutomatonField.xaml" t="Include" />
|
||||||
|
<e p="AutomatonField.xaml.cs" t="Include" />
|
||||||
|
<e p="bin" t="ExcludeRecursive" />
|
||||||
|
<e p="Life.csproj" t="IncludeRecursive" />
|
||||||
|
<e p="MainWindow.xaml" t="Include" />
|
||||||
|
<e p="MainWindow.xaml.cs" t="Include" />
|
||||||
|
<e p="Misc" t="Include">
|
||||||
|
<e p="DelegateSubscriber.cs" t="Include" />
|
||||||
|
<e p="DelegateUnsubscriber.cs" t="Include" />
|
||||||
|
<e p="Position.cs" t="Include" />
|
||||||
|
<e p="Size.cs" t="Include" />
|
||||||
|
</e>
|
||||||
|
<e p="obj" t="ExcludeRecursive">
|
||||||
|
<e p="Debug" t="Include">
|
||||||
|
<e p="netcoreapp3.0" t="Include">
|
||||||
|
<e p="Life.AssemblyInfo.cs" t="Include" />
|
||||||
|
</e>
|
||||||
|
</e>
|
||||||
|
</e>
|
||||||
|
<e p="ViewModel" t="Include">
|
||||||
|
<e p="BaseViewModel.cs" t="Include" />
|
||||||
|
<e p="CellViewModel.cs" t="Include" />
|
||||||
|
<e p="FieldViewModel.cs" t="Include" />
|
||||||
|
</e>
|
||||||
|
</e>
|
||||||
|
<e p="Life.sln" t="IncludeFlat" />
|
||||||
|
<e p="packages" t="ExcludeRecursive" />
|
||||||
|
</e>
|
||||||
|
</component>
|
||||||
|
</project>
|
4
.idea/.idea.Life/.idea/encodings.xml
Normal file
4
.idea/.idea.Life/.idea/encodings.xml
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="Encoding" addBOMForNewFiles="with BOM under Windows, with no BOM otherwise" />
|
||||||
|
</project>
|
8
.idea/.idea.Life/.idea/indexLayout.xml
Normal file
8
.idea/.idea.Life/.idea/indexLayout.xml
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="ContentModelUserStore">
|
||||||
|
<attachedFolders />
|
||||||
|
<explicitIncludes />
|
||||||
|
<explicitExcludes />
|
||||||
|
</component>
|
||||||
|
</project>
|
8
.idea/.idea.Life/.idea/modules.xml
Normal file
8
.idea/.idea.Life/.idea/modules.xml
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="ProjectModuleManager">
|
||||||
|
<modules>
|
||||||
|
<module fileurl="file://$PROJECT_DIR$/.idea/.idea.Life/riderModule.iml" filepath="$PROJECT_DIR$/.idea/.idea.Life/riderModule.iml" />
|
||||||
|
</modules>
|
||||||
|
</component>
|
||||||
|
</project>
|
6
.idea/.idea.Life/.idea/projectSettingsUpdater.xml
Normal file
6
.idea/.idea.Life/.idea/projectSettingsUpdater.xml
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="RiderProjectSettingsUpdater">
|
||||||
|
<option name="vcsConfiguration" value="1" />
|
||||||
|
</component>
|
||||||
|
</project>
|
6
.idea/.idea.Life/.idea/vcs.xml
Normal file
6
.idea/.idea.Life/.idea/vcs.xml
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="VcsDirectoryMappings">
|
||||||
|
<mapping directory="$PROJECT_DIR$" vcs="Git" />
|
||||||
|
</component>
|
||||||
|
</project>
|
7
.idea/.idea.Life/riderModule.iml
Normal file
7
.idea/.idea.Life/riderModule.iml
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<module type="RIDER_MODULE" version="4">
|
||||||
|
<component name="NewModuleRootManager">
|
||||||
|
<content url="file://$MODULE_DIR$/../.." />
|
||||||
|
<orderEntry type="sourceFolder" forTests="false" />
|
||||||
|
</component>
|
||||||
|
</module>
|
16
Life.sln
Normal file
16
Life.sln
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
|
||||||
|
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Life", "Life\Life.csproj", "{359EE343-AEB5-48C7-BAED-A6BD10072733}"
|
||||||
|
EndProject
|
||||||
|
Global
|
||||||
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
|
Debug|Any CPU = Debug|Any CPU
|
||||||
|
Release|Any CPU = Release|Any CPU
|
||||||
|
EndGlobalSection
|
||||||
|
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||||
|
{359EE343-AEB5-48C7-BAED-A6BD10072733}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{359EE343-AEB5-48C7-BAED-A6BD10072733}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{359EE343-AEB5-48C7-BAED-A6BD10072733}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{359EE343-AEB5-48C7-BAED-A6BD10072733}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
EndGlobalSection
|
||||||
|
EndGlobal
|
2
Life.sln.DotSettings
Normal file
2
Life.sln.DotSettings
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
<wpf:ResourceDictionary xml:space="preserve" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:s="clr-namespace:System;assembly=mscorlib" xmlns:ss="urn:shemas-jetbrains-com:settings-storage-xaml" xmlns:wpf="http://schemas.microsoft.com/winfx/2006/xaml/presentation">
|
||||||
|
<s:Boolean x:Key="/Default/CodeInspection/CodeAnnotations/NamespacesWithAnnotations/=Life_002EAnnotations/@EntryIndexedValue">True</s:Boolean></wpf:ResourceDictionary>
|
2
Life.sln.DotSettings.user
Normal file
2
Life.sln.DotSettings.user
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
<wpf:ResourceDictionary xml:space="preserve" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:s="clr-namespace:System;assembly=mscorlib" xmlns:ss="urn:shemas-jetbrains-com:settings-storage-xaml" xmlns:wpf="http://schemas.microsoft.com/winfx/2006/xaml/presentation">
|
||||||
|
<s:Boolean x:Key="/Default/UserDictionary/Words/=Unsubscriber/@EntryIndexedValue">True</s:Boolean></wpf:ResourceDictionary>
|
1162
Life/Annotations.cs
Normal file
1162
Life/Annotations.cs
Normal file
File diff suppressed because it is too large
Load Diff
9
Life/App.xaml
Normal file
9
Life/App.xaml
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<Application x:Class="Life.App"
|
||||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
||||||
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||||
|
xmlns:local="clr-namespace:Life"
|
||||||
|
StartupUri="MainWindow.xaml">
|
||||||
|
<Application.Resources>
|
||||||
|
|
||||||
|
</Application.Resources>
|
||||||
|
</Application>
|
17
Life/App.xaml.cs
Normal file
17
Life/App.xaml.cs
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Configuration;
|
||||||
|
using System.Data;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using System.Windows;
|
||||||
|
|
||||||
|
namespace Life
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Interaction logic for App.xaml
|
||||||
|
/// </summary>
|
||||||
|
public partial class App : Application
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
64
Life/Automaton/BasicField.cs
Normal file
64
Life/Automaton/BasicField.cs
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.ComponentModel;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using Life.Annotations;
|
||||||
|
using Life.Misc;
|
||||||
|
|
||||||
|
namespace Life.Automaton
|
||||||
|
{
|
||||||
|
public abstract class BasicField : IField, INotifyPropertyChanged
|
||||||
|
{
|
||||||
|
List<IObserver<IField>> _observers = new List<IObserver<IField>>();
|
||||||
|
|
||||||
|
protected BasicField(int width, int height)
|
||||||
|
{
|
||||||
|
Size = new Size(width, height);
|
||||||
|
_cells = new Cell[width, height];
|
||||||
|
|
||||||
|
for (int x = 0; x < width; x++)
|
||||||
|
for (int y = 0; y < height; y++)
|
||||||
|
_cells[x, y] = new Cell();
|
||||||
|
}
|
||||||
|
|
||||||
|
private Cell[,] _cells;
|
||||||
|
|
||||||
|
public Size Size { get; }
|
||||||
|
|
||||||
|
public Cell this[int x, int y]
|
||||||
|
{
|
||||||
|
get => _cells[x, y];
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_cells[x, y] = value;
|
||||||
|
_observers.ForEach(observer => observer.OnNext(this));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public IEnumerable<Cell> Cells => _cells.Cast<Cell>();
|
||||||
|
|
||||||
|
public abstract IEnumerable<Cell> Neighbours(int x, int y);
|
||||||
|
|
||||||
|
public void Transform(Rule transform)
|
||||||
|
{
|
||||||
|
_cells = transform(this);
|
||||||
|
_observers.ForEach(observer => observer.OnNext(this));
|
||||||
|
}
|
||||||
|
|
||||||
|
public event PropertyChangedEventHandler PropertyChanged;
|
||||||
|
|
||||||
|
[NotifyPropertyChangedInvocator]
|
||||||
|
protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
|
||||||
|
{
|
||||||
|
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
|
||||||
|
}
|
||||||
|
|
||||||
|
public IDisposable Subscribe(IObserver<IField> observer)
|
||||||
|
{
|
||||||
|
_observers.Add(observer);
|
||||||
|
|
||||||
|
return new DelegateUnsubscriber(() => _observers.Remove(observer));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
9
Life/Automaton/Cell.cs
Normal file
9
Life/Automaton/Cell.cs
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
using Life.Misc;
|
||||||
|
|
||||||
|
namespace Life.Automaton
|
||||||
|
{
|
||||||
|
public class Cell
|
||||||
|
{
|
||||||
|
public bool IsAlive { get; set; }
|
||||||
|
}
|
||||||
|
}
|
33
Life/Automaton/GameOfLife.cs
Normal file
33
Life/Automaton/GameOfLife.cs
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
using System.Linq;
|
||||||
|
|
||||||
|
namespace Life.Automaton
|
||||||
|
{
|
||||||
|
public static class GameOfLife
|
||||||
|
{
|
||||||
|
public static Rule Rule(string born, string survive)
|
||||||
|
{
|
||||||
|
bool WillSurvive(int neighbours) => survive.Contains(neighbours.ToString());
|
||||||
|
bool WillBorn(int neighbours) => born.Contains(neighbours.ToString());
|
||||||
|
|
||||||
|
return field =>
|
||||||
|
{
|
||||||
|
var @new = new Cell[field.Size.Width, field.Size.Height];
|
||||||
|
|
||||||
|
for (int i = 0; i < field.Size.Width; i++)
|
||||||
|
{
|
||||||
|
for (int j = 0; j < field.Size.Height; j++)
|
||||||
|
{
|
||||||
|
var neighbours = field.Neighbours(i, j).Count(c => c.IsAlive);
|
||||||
|
|
||||||
|
@new[i,j] = new Cell
|
||||||
|
{
|
||||||
|
IsAlive = field[i,j].IsAlive ? WillSurvive(neighbours) : WillBorn(neighbours)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return @new;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
11
Life/Automaton/IAutomaton.cs
Normal file
11
Life/Automaton/IAutomaton.cs
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
namespace Life.Automaton
|
||||||
|
{
|
||||||
|
public interface IAutomaton
|
||||||
|
{
|
||||||
|
int Iteration { get; }
|
||||||
|
IField Field { get; }
|
||||||
|
|
||||||
|
void Initialize(IField field);
|
||||||
|
void Advance();
|
||||||
|
}
|
||||||
|
}
|
21
Life/Automaton/IField.cs
Normal file
21
Life/Automaton/IField.cs
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using Life.Misc;
|
||||||
|
|
||||||
|
namespace Life.Automaton
|
||||||
|
{
|
||||||
|
public delegate Cell[,] Rule(IField old);
|
||||||
|
|
||||||
|
public interface IField : IObservable<IField>
|
||||||
|
{
|
||||||
|
Size Size { get; }
|
||||||
|
|
||||||
|
Cell this[int x, int y] { get; set; }
|
||||||
|
|
||||||
|
IEnumerable<Cell> Cells { get; }
|
||||||
|
|
||||||
|
IEnumerable<Cell> Neighbours(int x, int y);
|
||||||
|
|
||||||
|
void Transform(Rule rule);
|
||||||
|
}
|
||||||
|
}
|
31
Life/Automaton/MooreField.cs
Normal file
31
Life/Automaton/MooreField.cs
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
|
||||||
|
namespace Life.Automaton
|
||||||
|
{
|
||||||
|
public class MooreField : BasicField
|
||||||
|
{
|
||||||
|
public MooreField(int width, int height) : base(width, height)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public override IEnumerable<Cell> Neighbours(int x, int y)
|
||||||
|
{
|
||||||
|
var potential = new []
|
||||||
|
{
|
||||||
|
(x - 1, y + 1), (x, y + 1), (x + 1, y + 1),
|
||||||
|
(x - 1, y) /*, (x, y) */, (x + 1, y),
|
||||||
|
(x - 1, y - 1), (x, y - 1), (x + 1, y - 1),
|
||||||
|
};
|
||||||
|
|
||||||
|
return potential
|
||||||
|
.Where(c => c.Item1 >= 0 && c.Item1 < Size.Width && c.Item2 >= 0 && c.Item2 < Size.Height)
|
||||||
|
.Select(c => this[c.Item1, c.Item2]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
return $"MooreField {Size.Width} x {Size.Height}";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
89
Life/AutomatonField.xaml
Normal file
89
Life/AutomatonField.xaml
Normal file
@ -0,0 +1,89 @@
|
|||||||
|
<UserControl x:Class="Life.AutomatonField"
|
||||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
||||||
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||||
|
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
|
||||||
|
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
|
||||||
|
xmlns:local="clr-namespace:Life"
|
||||||
|
mc:Ignorable="d"
|
||||||
|
d:DesignHeight="300" d:DesignWidth="300">
|
||||||
|
<UserControl.Resources>
|
||||||
|
<ResourceDictionary>
|
||||||
|
<Style TargetType="Button" x:Key="Cell">
|
||||||
|
<Style.Triggers>
|
||||||
|
<DataTrigger Binding="{Binding IsAlive}" Value="True">
|
||||||
|
<DataTrigger.EnterActions>
|
||||||
|
<BeginStoryboard>
|
||||||
|
<Storyboard>
|
||||||
|
<ColorAnimation Duration="0:0:0.15" To="Black" Storyboard.TargetProperty="(Button.Background).(SolidColorBrush.Color)"/>
|
||||||
|
</Storyboard>
|
||||||
|
</BeginStoryboard>
|
||||||
|
</DataTrigger.EnterActions>
|
||||||
|
<DataTrigger.ExitActions>
|
||||||
|
<BeginStoryboard>
|
||||||
|
<Storyboard>
|
||||||
|
<ColorAnimation Duration="0:0:0.15" From="Black" Storyboard.TargetProperty="(Button.Background).(SolidColorBrush.Color)"/>
|
||||||
|
</Storyboard>
|
||||||
|
</BeginStoryboard>
|
||||||
|
</DataTrigger.ExitActions>
|
||||||
|
</DataTrigger>
|
||||||
|
<EventTrigger RoutedEvent="MouseEnter">
|
||||||
|
<BeginStoryboard>
|
||||||
|
<Storyboard>
|
||||||
|
<ColorAnimation Duration="0:0:0.05" To="Aqua" Storyboard.TargetProperty="(Button.BorderBrush).(SolidColorBrush.Color)"/>
|
||||||
|
</Storyboard>
|
||||||
|
</BeginStoryboard>
|
||||||
|
</EventTrigger>
|
||||||
|
<EventTrigger RoutedEvent="MouseLeave">
|
||||||
|
<BeginStoryboard>
|
||||||
|
<Storyboard>
|
||||||
|
<ColorAnimation Duration="0:0:0.05" From="Aqua" Storyboard.TargetProperty="(Button.BorderBrush).(SolidColorBrush.Color)"/>
|
||||||
|
</Storyboard>
|
||||||
|
</BeginStoryboard>
|
||||||
|
</EventTrigger>
|
||||||
|
</Style.Triggers>
|
||||||
|
|
||||||
|
<Setter Property="Background" Value="LightGray"/>
|
||||||
|
<Setter Property="Template">
|
||||||
|
<Setter.Value>
|
||||||
|
<ControlTemplate TargetType="Button">
|
||||||
|
<Border x:Name="Border"
|
||||||
|
BorderThickness="1"
|
||||||
|
Background="{TemplateBinding Background}"
|
||||||
|
BorderBrush="{TemplateBinding BorderBrush}">
|
||||||
|
<ContentPresenter Margin="2"
|
||||||
|
HorizontalAlignment="Center"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
RecognizesAccessKey="True"/>
|
||||||
|
</Border>
|
||||||
|
</ControlTemplate>
|
||||||
|
</Setter.Value>
|
||||||
|
</Setter>
|
||||||
|
</Style>
|
||||||
|
</ResourceDictionary>
|
||||||
|
</UserControl.Resources>
|
||||||
|
<Border Background="LightGray" BorderThickness="{Binding Separation}"
|
||||||
|
BorderBrush="LightGray" DataContext="{Binding ViewModel, RelativeSource={RelativeSource AncestorType={x:Type local:AutomatonField}}}">
|
||||||
|
<ItemsControl ItemsSource="{Binding Cells}">
|
||||||
|
<ItemsControl.ItemsPanel>
|
||||||
|
<ItemsPanelTemplate>
|
||||||
|
<Canvas Width="{Binding Width}" Height="{Binding Height}" />
|
||||||
|
</ItemsPanelTemplate>
|
||||||
|
</ItemsControl.ItemsPanel>
|
||||||
|
<ItemsControl.ItemTemplate>
|
||||||
|
<DataTemplate>
|
||||||
|
<Button Width="{Binding Size, RelativeSource={RelativeSource AncestorType={x:Type local:AutomatonField}}}"
|
||||||
|
Height="{Binding Size, RelativeSource={RelativeSource AncestorType={x:Type local:AutomatonField}}}"
|
||||||
|
Click="Cell_Click"
|
||||||
|
Style="{DynamicResource Cell}"
|
||||||
|
/>
|
||||||
|
</DataTemplate>
|
||||||
|
</ItemsControl.ItemTemplate>
|
||||||
|
<ItemsControl.ItemContainerStyle>
|
||||||
|
<Style>
|
||||||
|
<Setter Property="Canvas.Left" Value="{Binding Left}"/>
|
||||||
|
<Setter Property="Canvas.Top" Value="{Binding Top}"/>
|
||||||
|
</Style>
|
||||||
|
</ItemsControl.ItemContainerStyle>
|
||||||
|
</ItemsControl>
|
||||||
|
</Border>
|
||||||
|
</UserControl>
|
92
Life/AutomatonField.xaml.cs
Normal file
92
Life/AutomatonField.xaml.cs
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.ObjectModel;
|
||||||
|
using System.Windows;
|
||||||
|
using System.Windows.Controls;
|
||||||
|
using System.Windows.Media;
|
||||||
|
using System.Windows.Shapes;
|
||||||
|
using Life.Automaton;
|
||||||
|
using Life.ViewModel;
|
||||||
|
|
||||||
|
namespace Life
|
||||||
|
{
|
||||||
|
public partial class AutomatonField : UserControl
|
||||||
|
{
|
||||||
|
#region Dependency Property Wrappers
|
||||||
|
public int Size
|
||||||
|
{
|
||||||
|
get { return (int) GetValue(SizeProperty); }
|
||||||
|
set { SetValue(SizeProperty, value); }
|
||||||
|
}
|
||||||
|
|
||||||
|
public int Separation
|
||||||
|
{
|
||||||
|
get { return (int) GetValue(SeparationProperty); }
|
||||||
|
set { SetValue(SeparationProperty, value); }
|
||||||
|
}
|
||||||
|
|
||||||
|
public IField Field
|
||||||
|
{
|
||||||
|
get { return (IField) GetValue(FieldProperty); }
|
||||||
|
set { SetValue(FieldProperty, value); }
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
public FieldViewModel ViewModel { get; set; } = new FieldViewModel();
|
||||||
|
|
||||||
|
public AutomatonField()
|
||||||
|
{
|
||||||
|
InitializeComponent();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void OnFieldChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
|
||||||
|
{
|
||||||
|
var field = d as AutomatonField;
|
||||||
|
|
||||||
|
field.ViewModel.Field = e.NewValue as IField;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void OnSizeChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
|
||||||
|
{
|
||||||
|
var field = d as AutomatonField;
|
||||||
|
|
||||||
|
field.ViewModel.Size = (int)e.NewValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void OnSeparationChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
|
||||||
|
{
|
||||||
|
var field = d as AutomatonField;
|
||||||
|
|
||||||
|
field.ViewModel.Separation = (int)e.NewValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
#region Dependency Properties
|
||||||
|
public static readonly DependencyProperty SizeProperty = DependencyProperty.Register(
|
||||||
|
"Size",
|
||||||
|
typeof(int), typeof(AutomatonField),
|
||||||
|
new PropertyMetadata(OnSizeChanged)
|
||||||
|
);
|
||||||
|
|
||||||
|
public static readonly DependencyProperty SeparationProperty = DependencyProperty.Register(
|
||||||
|
"Separation",
|
||||||
|
typeof(int), typeof(AutomatonField),
|
||||||
|
new PropertyMetadata(OnSeparationChanged)
|
||||||
|
);
|
||||||
|
|
||||||
|
public static readonly DependencyProperty FieldProperty = DependencyProperty.Register(
|
||||||
|
nameof(Field),
|
||||||
|
typeof(IField), typeof(AutomatonField),
|
||||||
|
new PropertyMetadata(OnFieldChanged)
|
||||||
|
);
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
private void Cell_Click(object sender, RoutedEventArgs e)
|
||||||
|
{
|
||||||
|
var vm = (sender as Button).DataContext as CellViewModel;
|
||||||
|
var cell = Field[vm.Position.X, vm.Position.Y];
|
||||||
|
|
||||||
|
cell.IsAlive = !cell.IsAlive;
|
||||||
|
|
||||||
|
ViewModel.Sync();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
9
Life/Life.csproj
Normal file
9
Life/Life.csproj
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk.WindowsDesktop">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<OutputType>WinExe</OutputType>
|
||||||
|
<TargetFramework>netcoreapp3.0</TargetFramework>
|
||||||
|
<UseWPF>true</UseWPF>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
</Project>
|
39
Life/MainWindow.xaml
Normal file
39
Life/MainWindow.xaml
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
<Window x:Class="Life.MainWindow"
|
||||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
||||||
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||||
|
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
|
||||||
|
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
|
||||||
|
xmlns:local="clr-namespace:Life"
|
||||||
|
mc:Ignorable="d"
|
||||||
|
Title="MainWindow" Height="450" Width="800" d:DataContext="{d:DesignInstance local:MainWindow}">
|
||||||
|
<Grid>
|
||||||
|
<Grid.ColumnDefinitions>
|
||||||
|
<ColumnDefinition Width="2*" MinWidth="400"/>
|
||||||
|
<ColumnDefinition Width="3"/>
|
||||||
|
<ColumnDefinition Width="1*" MinWidth="250"/>
|
||||||
|
</Grid.ColumnDefinitions>
|
||||||
|
|
||||||
|
<ScrollViewer HorizontalScrollBarVisibility="Auto" VerticalScrollBarVisibility="Auto">
|
||||||
|
<local:AutomatonField Field="{Binding Field}" Size="{Binding Size}" Separation="3"/>
|
||||||
|
</ScrollViewer>
|
||||||
|
|
||||||
|
<GridSplitter Grid.Column="1" Width="3" ResizeDirection="Columns" ResizeBehavior="PreviousAndNext"/>
|
||||||
|
<Grid Margin="8" Grid.Column="2">
|
||||||
|
<Grid.RowDefinitions>
|
||||||
|
<RowDefinition Height="3*"/>
|
||||||
|
<RowDefinition Height="1*"/>
|
||||||
|
</Grid.RowDefinitions>
|
||||||
|
|
||||||
|
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center">
|
||||||
|
<TextBlock Text="{Binding Field.Size.Width}"/>
|
||||||
|
<TextBlock Text=" x "/>
|
||||||
|
<TextBlock Text="{Binding Field.Size.Height}"/>
|
||||||
|
</StackPanel>
|
||||||
|
|
||||||
|
<StackPanel Orientation="Vertical" Grid.Row="1" VerticalAlignment="Bottom">
|
||||||
|
<Button VerticalAlignment="Stretch" Margin="0 0 0 8">Step</Button>
|
||||||
|
<Button VerticalAlignment="Stretch" Click="Step_OnClick">Step</Button>
|
||||||
|
</StackPanel>
|
||||||
|
</Grid>
|
||||||
|
</Grid>
|
||||||
|
</Window>
|
63
Life/MainWindow.xaml.cs
Normal file
63
Life/MainWindow.xaml.cs
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.ComponentModel;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using System.Windows;
|
||||||
|
using Life.Annotations;
|
||||||
|
using Life.Automaton;
|
||||||
|
|
||||||
|
namespace Life
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Interaction logic for MainWindow.xaml
|
||||||
|
/// </summary>
|
||||||
|
public partial class MainWindow : Window, INotifyPropertyChanged
|
||||||
|
{
|
||||||
|
private IField _field;
|
||||||
|
private int _size;
|
||||||
|
private Rule _rule = GameOfLife.Rule("3", "23");
|
||||||
|
|
||||||
|
public IField Field
|
||||||
|
{
|
||||||
|
get => _field;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_field = value;
|
||||||
|
OnPropertyChanged(nameof(Field));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public int Size
|
||||||
|
{
|
||||||
|
get => _size;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_size = value;
|
||||||
|
OnPropertyChanged(nameof(Size));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public MainWindow()
|
||||||
|
{
|
||||||
|
Size = 20;
|
||||||
|
Field = new MooreField(20, 20);
|
||||||
|
Field[5, 5].IsAlive = true;
|
||||||
|
DataContext = this;
|
||||||
|
|
||||||
|
InitializeComponent();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void Step_OnClick(object sender, RoutedEventArgs e)
|
||||||
|
{
|
||||||
|
Field.Transform(_rule);
|
||||||
|
}
|
||||||
|
|
||||||
|
public event PropertyChangedEventHandler PropertyChanged;
|
||||||
|
|
||||||
|
[NotifyPropertyChangedInvocator]
|
||||||
|
protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
|
||||||
|
{
|
||||||
|
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
26
Life/Misc/DelegateSubscriber.cs
Normal file
26
Life/Misc/DelegateSubscriber.cs
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Life.Misc
|
||||||
|
{
|
||||||
|
public class DelegateSubscriber<T> : IObserver<T>
|
||||||
|
{
|
||||||
|
public Action<T> Action;
|
||||||
|
public Action<Exception> Error;
|
||||||
|
public Action Completed;
|
||||||
|
|
||||||
|
public void OnCompleted()
|
||||||
|
{
|
||||||
|
Completed?.Invoke();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnError(Exception error)
|
||||||
|
{
|
||||||
|
Error?.Invoke(error);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnNext(T value)
|
||||||
|
{
|
||||||
|
Action?.Invoke(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
20
Life/Misc/DelegateUnsubscriber.cs
Normal file
20
Life/Misc/DelegateUnsubscriber.cs
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
using System;
|
||||||
|
using System.Diagnostics.CodeAnalysis;
|
||||||
|
|
||||||
|
namespace Life.Misc
|
||||||
|
{
|
||||||
|
public class DelegateUnsubscriber : IDisposable
|
||||||
|
{
|
||||||
|
private readonly Action _action;
|
||||||
|
|
||||||
|
public DelegateUnsubscriber([NotNull] Action action)
|
||||||
|
{
|
||||||
|
_action = action;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
_action.Invoke();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
19
Life/Misc/Position.cs
Normal file
19
Life/Misc/Position.cs
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
namespace Life.Misc
|
||||||
|
{
|
||||||
|
public struct Position
|
||||||
|
{
|
||||||
|
public Position(int x, int y)
|
||||||
|
{
|
||||||
|
X = x;
|
||||||
|
Y = y;
|
||||||
|
}
|
||||||
|
|
||||||
|
public readonly int X;
|
||||||
|
public readonly int Y;
|
||||||
|
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
return $"({X}, {Y})";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
14
Life/Misc/Size.cs
Normal file
14
Life/Misc/Size.cs
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
namespace Life.Misc
|
||||||
|
{
|
||||||
|
public struct Size
|
||||||
|
{
|
||||||
|
public int Width { get; }
|
||||||
|
public int Height { get; }
|
||||||
|
|
||||||
|
public Size(int width, int height)
|
||||||
|
{
|
||||||
|
Width = width;
|
||||||
|
Height = height;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
17
Life/ViewModel/BaseViewModel.cs
Normal file
17
Life/ViewModel/BaseViewModel.cs
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
using System.ComponentModel;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using Life.Annotations;
|
||||||
|
|
||||||
|
namespace Life.ViewModel
|
||||||
|
{
|
||||||
|
public abstract class BaseViewModel : INotifyPropertyChanged
|
||||||
|
{
|
||||||
|
public event PropertyChangedEventHandler PropertyChanged;
|
||||||
|
|
||||||
|
[NotifyPropertyChangedInvocator]
|
||||||
|
protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
|
||||||
|
{
|
||||||
|
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
48
Life/ViewModel/CellViewModel.cs
Normal file
48
Life/ViewModel/CellViewModel.cs
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
using Life.Misc;
|
||||||
|
|
||||||
|
namespace Life.ViewModel
|
||||||
|
{
|
||||||
|
public class CellViewModel : BaseViewModel
|
||||||
|
{
|
||||||
|
private int _left;
|
||||||
|
private int _top;
|
||||||
|
private bool _isAlive;
|
||||||
|
|
||||||
|
public CellViewModel(int x, int y)
|
||||||
|
{
|
||||||
|
Position = new Position(x, y);
|
||||||
|
}
|
||||||
|
|
||||||
|
public int Left
|
||||||
|
{
|
||||||
|
get => _left;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_left = value;
|
||||||
|
OnPropertyChanged(nameof(Left));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public int Top
|
||||||
|
{
|
||||||
|
get => _top;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_top = value;
|
||||||
|
OnPropertyChanged(nameof(Top));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool IsAlive
|
||||||
|
{
|
||||||
|
get => _isAlive;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_isAlive = value;
|
||||||
|
OnPropertyChanged(nameof(IsAlive));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public Position Position { get; }
|
||||||
|
}
|
||||||
|
}
|
116
Life/ViewModel/FieldViewModel.cs
Normal file
116
Life/ViewModel/FieldViewModel.cs
Normal file
@ -0,0 +1,116 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.ObjectModel;
|
||||||
|
using Life.Automaton;
|
||||||
|
using Life.Misc;
|
||||||
|
|
||||||
|
namespace Life.ViewModel
|
||||||
|
{
|
||||||
|
public class FieldViewModel : BaseViewModel
|
||||||
|
{
|
||||||
|
private int _width;
|
||||||
|
private int _height;
|
||||||
|
private IField _field;
|
||||||
|
private int _size;
|
||||||
|
private int _separation;
|
||||||
|
private IDisposable _fieldObserverDisposal;
|
||||||
|
|
||||||
|
public ObservableCollection<CellViewModel> Cells { get; set; } = new ObservableCollection<CellViewModel>();
|
||||||
|
|
||||||
|
public int Width
|
||||||
|
{
|
||||||
|
get => _width;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_width = value;
|
||||||
|
OnPropertyChanged(nameof(Width));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public int Height
|
||||||
|
{
|
||||||
|
get => _height;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_height = value;
|
||||||
|
OnPropertyChanged(nameof(Height));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public IField Field
|
||||||
|
{
|
||||||
|
get => _field;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_fieldObserverDisposal?.Dispose();
|
||||||
|
_fieldObserverDisposal = value.Subscribe(new DelegateSubscriber<IField>
|
||||||
|
{
|
||||||
|
Action = field => Sync()
|
||||||
|
});
|
||||||
|
|
||||||
|
_field = value;
|
||||||
|
ResetFields();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public int Size
|
||||||
|
{
|
||||||
|
get => _size;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_size = value;
|
||||||
|
OnPropertyChanged(nameof(Size));
|
||||||
|
RecalculateSize();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public int Separation
|
||||||
|
{
|
||||||
|
get => _separation;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_separation = value;
|
||||||
|
OnPropertyChanged(nameof(Separation));
|
||||||
|
RecalculateSize();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ResetFields()
|
||||||
|
{
|
||||||
|
Cells.Clear();
|
||||||
|
|
||||||
|
for (int i = 0; i < Field.Size.Width; i++)
|
||||||
|
for (int j = 0; j < Field.Size.Height; j++)
|
||||||
|
Cells.Add(CreateCellViewModel(i, j));
|
||||||
|
|
||||||
|
Sync();
|
||||||
|
RecalculateSize();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void RecalculateSize()
|
||||||
|
{
|
||||||
|
if (Field == null) return;
|
||||||
|
|
||||||
|
foreach (var cell in Cells)
|
||||||
|
{
|
||||||
|
cell.Left = cell.Position.X * (Separation + Size);
|
||||||
|
cell.Top = cell.Position.Y * (Separation + Size);
|
||||||
|
}
|
||||||
|
|
||||||
|
Width = Field.Size.Width * Size + (Field.Size.Width - 1) * Separation;
|
||||||
|
Height = Field.Size.Height * Size + (Field.Size.Height - 1) * Separation;
|
||||||
|
}
|
||||||
|
|
||||||
|
private CellViewModel CreateCellViewModel(int x, int y)
|
||||||
|
{
|
||||||
|
return new CellViewModel(x, y);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Sync()
|
||||||
|
{
|
||||||
|
foreach (var cell in Cells)
|
||||||
|
{
|
||||||
|
cell.IsAlive = Field[cell.Position.X, cell.Position.Y].IsAlive;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user