previndexinfonext

code guessing, round #72 (completed)

started at ; stage 2 at ; ended at

specification

the bee says you are to apply sound changes. submissions may be written in any language.

a sound change is a change in the pronunciation of a language over time. sometimes, linguistics people get bored, make languages, and simulate their evolution. it's like playing with toys for them, except instead of toys they are languages and the way they are playing is by evolving them. when they are slightly less bored, they get a program to do this for them. today you will make that program.

let me show you with some notation1. the easiest sound changes to apply just shift a sound to another:
​ ​ f > h
but sound changes can also be conditioned by an environment (the sounds around it), so
​ ​ ə > ∅ / __#
could mean the elision of word-final ə (# is a common symbol for a word boundary), or even
​ ​ {p,t,k} > {b,d,g} / V__V
which seems to turn unvoiced plosives into their voiced counterparts when they are between vowels.

there is no fixed syntax for the sound changes; you may parse rules out of an input file, or even hardcode some pleasing ones into your program.

your challenge, given a lexicon, is to evolve the words in it according to a list of sound changes. as every language is allowed, there is no fixed API.

  1. haha I did the bad for accessibility thing you're supposed to not do that they like on TVTropes where you make each word be a different link

results

  1. 👑 oleander +2 -1 = 1
    1. chirk
    2. moshikoi
  2. moshikoi +2 -1 = 1
    1. chirk
    2. oleander
  3. chirk +0 -2 = -2
    1. moshikoi (was oleander)
    2. oleander (was moshikoi)

entries

you can download all the entries

entry #1

written by chirk
submitted at
0 likes

guesses
comments 0

post a comment


ba.sh ASCII text
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
soundChanges=()
while read inputLine; do
    test "$inputLine" = ';' && break || soundChanges+=("$inputLine")
done
soundChangeSedExpressions=()
for soundChangeIdx in "${!soundChanges[@]}"; do
    soundChange="${soundChanges[$soundChangeIdx]}"
    soundToChange="${soundChange%% >*}"
    soundToChangeTo="${soundChange##$soundToChange > }"
    soundToChangeTo="${soundToChangeTo%% / *}"
    if [[ "$soundChange" == *" / "* ]]; then
        conditionalChangeRule="${soundChange##$soundToChange > $soundToChangeTo / }"
    else
        conditionalChangeRule="__"
    fi
    soundConditional="${conditionalChangeRule//__/$soundToChange}"
    soundReplacement="${conditionalChangeRule//__/$soundToChangeTo}"
    soundChangeSedExpression="s/$soundConditional/$soundReplacement/g"
    soundChangeSedExpressions+=("$soundChangeSedExpression")
done
inputToRunReplacementsOn="$(cat)"
for soundChangeSedExpressionIdx in "${!soundChangeSedExpressions[@]}"; do
    soundChangeSedExpression="${soundChangeSedExpressions[$soundChangeSedExpressionIdx]}"
    inputToRunReplacementsOn=$(printf "%s" "$inputToRunReplacementsOn" | sed -e "$soundChangeSedExpression")
done
echo "$inputToRunReplacementsOn"

entry #2

written by oleander
submitted at
0 likes

guesses
comments 0

post a comment


72.sb3 data

entry #3

written by moshikoi
submitted at
0 likes

guesses
comments 0

post a comment


dir Baum-Archive
dir Baum.AvaloniaApp
dir Assets
Baum.png PNG image data, 320 x 320, 8-bit/color RGBA, non-interlaced
ipa-consonants.csv CSV Unicode text, UTF-8 text
  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
Symbol,Place,Method,Voicing
m̥,bilabial,nasal,
m,bilabial,nasal,voiced
ɱ,labiodental,nasal,voiced
n̼,linguolabial,nasal,voiced
n̥,alveolar,nasal,
n,alveolar,nasal,voiced
ɳ̊,retroflex,nasal,
ɳ,retroflex,nasal,voiced
ɲ̊,palatal,nasal,
ɲ,palatal,nasal,voiced
ŋ̊,velar,nasal,
ŋ,velar,nasal,voiced
ɴ,uvular,nasal,voiced
p,bilabial,plosive,
b,bilabial,plosive,voiced
p̪,labiodental,plosive,
b̪,labiodental,plosive,voiced
t̼,linguolabial,plosive,
d̼,linguolabial,plosive,voiced
t,alveolar,plosive,
d,alveolar,plosive,voiced
ʈ,retroflex,plosive,
ɖ,retroflex,plosive,voiced
c,palatal,plosive,
ɟ,palatal,plosive,voiced
k,velar,plosive,
ɡ,velar,plosive,voiced
q,uvular,plosive,
ɢ,uvular,plosive,voiced
ʡ,epiglottal,plosive,
ʔ,glottal,stop,
s,alveolar,fricative,
z,alveolar,fricative,voiced
ʃ,postalveolar,fricative,
ʒ,postalveolar,fricative,voiced
ʂ,retroflex,fricative,
ʐ,retroflex,fricative,voiced
ɕ,alveolo-palatal,fricative,
ʑ,alveolo-palatal,fricative,voiced
ɸ,bilabial,fricative,
β,bilabial,fricative,voiced
f,labiodental,fricative,
v,labiodental,fricative,voiced
θ̼,linguolabial,fricative,
ð̼,linguolabial,fricative,voiced
θ,dental,fricative,
ð,dental,fricative,voiced
θ̠,alveolar,non-sibilant fricative,
ð̠,alveolar,non-sibilant fricative,voiced
ɹ̠̊˔,postalveolar,non-sibilant fricative,
ɹ̠˔,postalveolar,non-sibilant fricative,voiced
ɻ̊˔,retroflex,non-sibilant fricative,
ɻ˔,retroflex,non-sibilant fricative,voiced
ç,palatal,fricative,
ʝ,palatal,fricative,voiced
x,velar,fricative,
ɣ,velar,fricative,voiced
χ,uvular,fricative,
ʁ,uvular,fricative,voiced
ħ,pharyngeal,fricative,
ʕ,pharyngeal,fricative,voiced
h,glottal,fricative,
ɦ,glottal,fricative,voiced
ʋ,labiodental,approximant,voiced
ɹ,alveolar,approximant,voiced
ɻ,retroflex,approximant,voiced
j,palatal,approximant,voiced
ɰ,velar,approximant,voiced
ʔ̞,creaky-glottal,approximant,
ⱱ̟,bilabial,flap,voiced
ⱱ,labiodental,flap,voiced
ɾ̼,linguolabial,tap,voiced
ɾ̥,alveolar,tap,
ɾ,dental,and alveolar taps and flaps,voiced
ɽ̊,retroflex,flap,
ɽ,retroflex,flap,voiced
ɢ̆,uvular,tap and flap,voiced
ʡ̆,epiglottal,tap,voiced
ʙ̥,bilabial,trill,
ʙ,bilabial,trill,voiced
r̥,alveolar,trill,
r,alveolar,trill,voiced
ɽ̊r̥,retroflex,trill,
ɽr,retroflex,trill,voiced
ʀ̥,uvular,trill,
ʀ,uvular,trill,voiced
ʜ,epiglottal,trill,
ʢ,epiglottal,trill,voiced
ɬ,alveolar,lateral fricative,
ɮ,alveolar,lateral fricative,voiced
ꞎ,retroflex,lateral fricative,
ɭ˔,retroflex,lateral fricative,voiced
𝼆,palatal,lateral fricative,
ʎ̝,palatal,lateral fricative,voiced
𝼄,velar,lateral fricative,
ʟ̝,velar,lateral fricative,voiced
l,alveolar,lateral approximant,voiced
ɭ,retroflex,lateral approximant,voiced
ʎ,palatal,lateral approximant,voiced
ʟ,velar,lateral approximant,voiced
ʟ̠,uvular,lateral approximant,voiced
ɺ̥,alveolar,lateral flap,
ɺ,alveolar,lateral flap,voiced
𝼈̥,retroflex,lateral flap,
𝼈,retroflex,lateral flap,voiced
ʎ̆,palatal,lateral flap,voiced
ʟ̆,velar,lateral tap,voiced
ipa-vowels.csv CSV Unicode text, UTF-8 text
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
Symbol,Height,Backness,Roundedness
i,close,front,
y,close,front,rounded
ɨ,close,central,
ʉ,close,central,rounded
ɯ,close,back,
u,close,back,rounded
ɪ,near-close,near-front,
ʏ,near-close,near-front,rounded
ʊ,near-close,near-back,rounded
e,close-mid,front,
ø,close-mid,front,rounded
ɘ,close-mid,central,
ɵ,close-mid,central,rounded
ɤ,close-mid,back,
o,close-mid,back,rounded
e̞,mid,front,
ø̞,mid,front,rounded
ə,mid,central,
ɤ̞,mid,back,
o̞,mid,back,rounded
ɛ,open-mid,front,
œ,open-mid,front,rounded
ɜ,open-mid,central,
ɞ,open-mid,central,rounded
ʌ,open-mid,back,
ɔ,open-mid,back,rounded
æ,near-open,front,
ɐ,near-open,central,
a,open,front,
ɶ,open,front,rounded
ä,open,central,
ɑ,open,back,
ɒ,open,back,rounded
dir Models
LanguageModel.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
using System.Diagnostics.CodeAnalysis;
using ReactiveUI;

namespace Baum.AvaloniaApp.Models;

public class LanguageModel : ReactiveObject
{
    public LanguageModel(string? name, int? parentId = null, string soundChange = "")
        => (_name, _parentId, _soundChange) = (name, parentId, soundChange);

    public int Id { get; set; }

    string? _name;
    public string? Name { get => _name; set => this.RaiseAndSetIfChanged(ref _name, value); }

    int? _parentId;
    public int? ParentId { get => _parentId; set => this.RaiseAndSetIfChanged(ref _parentId, value); }

    string _soundChange;
    public string SoundChange { get => _soundChange; set => this.RaiseAndSetIfChanged(ref _soundChange, value); }
}
WordModel.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
using ReactiveUI;

namespace Baum.AvaloniaApp.Models;

public class WordModel : ReactiveObject
{
    public WordModel(string name, string ipa)
        => (_name, _ipa) = (name, ipa);

    public required bool Transient { get; set; }
    public int Id { get; set; }
    public int? AncestorId { get; set; }
    public required int LanguageId { get; set; }

    string _name;
    public string Name { get => _name; set => this.RaiseAndSetIfChanged(ref _name, value); }

    string _ipa;
    public string IPA { get => _ipa; set => this.RaiseAndSetIfChanged(ref _ipa, value); }
}
dir Services
IProjectDatabase.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
using System.Collections.Generic;
using System.IO;
using System.Threading.Tasks;

using Baum.Phonology;
using Baum.AvaloniaApp.Models;

namespace Baum.AvaloniaApp.Services;

public interface IProjectDatabase
{
    bool HasMigrations();
    Task MigrateAsync();

    void SaveToFile(FileInfo fileInfo);

    Task AddAsync(LanguageModel language);
    Task UpdateAsync(LanguageModel language);
    Task<IEnumerable<LanguageModel>> GetLanguagesAsync();
    Task<IEnumerable<LanguageModel>> GetChildrenAsync(int languageId);

    Task<WordModel> AddAsync(WordModel word);
    Task UpdateAsync(WordModel word);
    Task<IEnumerable<WordModel>> GetWordsAsync(int languageId, PhonologyData data);
    Task<IEnumerable<WordModel>> GetAncestryAsync(WordModel word, PhonologyData data);
}
IProjectDatabaseFactory.cs ASCII text, with CRLF line terminators
1
2
3
4
5
6
7
8
using System.IO;

namespace Baum.AvaloniaApp.Services;

public interface IProjectDatabaseFactory
{
    IProjectDatabase Create(FileInfo fileInfo);
}
ProjectDatabase.cs ASCII text, with CRLF line terminators
  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.EntityFrameworkCore;

using Baum.Phonology;
using Baum.Data;
using Baum.AvaloniaApp.Models;

namespace Baum.AvaloniaApp.Services;

class ProjectDatabase : IProjectDatabase
{
    FileInfo File { get; }

    public ProjectDatabase(FileInfo fileInfo) => File = fileInfo;

    public async Task AddAsync(LanguageModel languageModel)
    {
        using var context = new ProjectContext(File);

        await context.Languages.AddAsync(new Language
        {
            Name = languageModel.Name,
            ParentId = languageModel.ParentId,
            SoundChange = languageModel.SoundChange,
        });

        await context.SaveChangesAsync();
    }

    public async Task UpdateAsync(LanguageModel languageModel)
    {
        using var context = new ProjectContext(File);

        var language = await context.Languages.FindAsync(languageModel.Id);
        if (language == null) throw new InvalidOperationException("No language found in database");

        language.Name = languageModel.Name;
        language.SoundChange = languageModel.SoundChange;

        await context.SaveChangesAsync();
    }

    public async Task<IEnumerable<LanguageModel>> GetChildrenAsync(int languageId)
    {
        using var context = new ProjectContext(File);

        var entity = await context.Languages.FindAsync(languageId);
        if (entity == null) throw new InvalidOperationException();

        return await (
            from language in context.Languages
            where language.ParentId == languageId
            select new LanguageModel(
                language.Name,
                language.ParentId,
                language.SoundChange) { Id = language.Id }).ToArrayAsync();
    }

    public async Task<IEnumerable<LanguageModel>> GetLanguagesAsync()
    {
        using var context = new ProjectContext(File);

        return await context.Languages
            .Select(l => new LanguageModel(l.Name,l.ParentId,l.SoundChange) { Id = l.Id })
            .ToArrayAsync();
    }

    async Task<WordModel> GetWordAsync(int wordId)
    {
        using var context = new ProjectContext(File);

        var word = await context.Words.FindAsync(wordId);
        if (word == null) throw new InvalidOperationException("Word doesn't exist");

        return new WordModel(word.Name, word.IPA)
        {
            Transient = false,
            Id = word.Id,
            AncestorId = word.AncestorId,
            LanguageId = word.LanguageId,
        };
    }


    public async Task<IEnumerable<WordModel>> GetWordsAsync(int languageId, PhonologyData data)
    {
        using var context = new ProjectContext(File);

        var language = await context.Languages.FindAsync(languageId);
        if (language == null) throw new InvalidOperationException("No language found in database");

        List<WordModel> words = new();

        await foreach (var word in context.Entry(language).Collection(l => l.Words).Query().AsAsyncEnumerable())
        {
            words.Add(new WordModel(word.Name, word.IPA)
            {
                Transient = false,
                Id = word.Id,
                AncestorId = word.AncestorId,
                LanguageId = word.LanguageId,
            });
        }

        if (language.ParentId != null)
        {
            var parentWords = await GetWordsAsync((int)language.ParentId, data);
            foreach (var parentWord in parentWords)
            {
                SoundChange.TryApply(parentWord.IPA, language.SoundChange, data, out var IPA);

                words.Add(new WordModel(parentWord.Name, IPA)
                {
                    Transient = true,
                    LanguageId = languageId,
                    AncestorId = parentWord.Transient ? parentWord.AncestorId : parentWord.Id
                });
            }
        }

        return words;
    }

    public async Task<IEnumerable<WordModel>> GetAncestryAsync(WordModel word, PhonologyData data)
    {
        using var context = new ProjectContext(File);

        if (word.AncestorId == null)
            return Enumerable.Empty<WordModel>();

        var ancester = await context.Words.FindAsync(word.AncestorId);

        if (ancester == null)
            throw new InvalidOperationException("Ancestor does not exist");


        var wordLanguage = await context.Languages.FindAsync(word.LanguageId);
        if (wordLanguage == null)
            throw new InvalidOperationException("Language does not exist");

        await context.Entry(wordLanguage)
            .Reference(l => l.Parent)
            .LoadAsync();

        List<Language> languageChain = new() { };

        while (wordLanguage.Id != ancester.LanguageId)
        {
            if (!string.IsNullOrEmpty(wordLanguage.SoundChange))
                languageChain.Add(wordLanguage);

            await context.Entry(wordLanguage)
                .Reference(l => l.Parent)
                .LoadAsync();

            wordLanguage = wordLanguage.Parent ?? throw new InvalidOperationException("Ancestor is not an ancestor");
        }

        List<WordModel> wordChain = new();
        wordChain.Add(new(ancester.Name, ancester.IPA)
        {
            Transient = false,
            AncestorId = ancester.AncestorId,
            LanguageId = ancester.LanguageId,
        });

        foreach (Language intermediate in Enumerable.Reverse(languageChain))
        {
            var last = wordChain.Last();

            SoundChange.TryApply(last.IPA, intermediate.SoundChange, data, out var next);

            // TODO? Possibly do some error handling or notification here instead of just skipping
            if (last.IPA == next) continue;
            wordChain.Add(new WordModel(last.Name, next)
            {
                Transient = true,
                AncestorId = ancester.Id,
                LanguageId = intermediate.Id
            });
        }

        return wordChain;
    }

    public async Task<WordModel> AddAsync(WordModel word)
    {
        using var context = new ProjectContext(File);

        var language = await context.Languages.FindAsync(word.LanguageId);
        if (language == null) throw new InvalidOperationException("Language doesn't exist");

        var entry = await context.Words.AddAsync(new Word
        {
            Language = language,
            Name = word.Name,
            IPA = word.IPA,
            AncestorId = word.AncestorId,
        });

        await context.SaveChangesAsync();

        word.Id = entry.Entity.Id;
        word.Transient = false;

        return word;
    }

    public async Task UpdateAsync(WordModel wordModel)
    {
        using var context = new ProjectContext(File);

        var word = await context.Words.FindAsync(wordModel.Id);
        if (word == null) throw new InvalidOperationException("Word doesn't exist");

        word.Name = wordModel.Name;
        word.IPA = wordModel.IPA;
        word.AncestorId = wordModel.AncestorId;

        await context.SaveChangesAsync();
    }

    public bool HasMigrations()
    {
        using var context = new ProjectContext(File);
        return context.Database.GetMigrations().Any();
    }

    public async Task MigrateAsync()
    {
        using var context = new ProjectContext(File);
        await context.Database.MigrateAsync();
    }

    public void SaveToFile(FileInfo fileInfo)
    {
        if (fileInfo != File)
        {
            File.CopyTo(fileInfo.FullName, true); // TODO: Prompt user to confirm overwrite
        }
    }
}
ProjectDatabaseFactory.cs ASCII text, with CRLF line terminators
1
2
3
4
5
6
7
8
using System.IO;

namespace Baum.AvaloniaApp.Services;

class ProjectDatabaseFactory : IProjectDatabaseFactory
{
    public IProjectDatabase Create(FileInfo fileInfo) => new ProjectDatabase(fileInfo);
}
dir ViewModels
HomeViewModel.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
using System.Reactive;
using ReactiveUI;

namespace Baum.AvaloniaApp.ViewModels;

public class HomeViewModel : ViewModelBase
{
    public string OpenButtonText { get; } = "Open File";

    public ReactiveCommand<Unit, Unit> OpenCommand { get; }
    public ReactiveCommand<Unit, Unit> NewCommand { get; }

    public HomeViewModel(ReactiveCommand<Unit, Unit> openCommand, ReactiveCommand<Unit, Unit> newCommand)
    {
        OpenCommand = openCommand;
        NewCommand = newCommand;
    }
}
LanguageForestViewModel.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
using System.Collections.ObjectModel;
using System.Linq;
using System.Reactive;
using System.Threading.Tasks;
using ReactiveUI;

using Baum.AvaloniaApp.Models;
using Baum.AvaloniaApp.Services;

namespace Baum.AvaloniaApp.ViewModels;

public class LanguageForestViewModel : ViewModelBase
{
    IProjectDatabase Database { get; }

    ObservableCollection<LanguageTreeViewModel> LanguageTrees { get; set; }

    ReactiveCommand<LanguageModel, Unit> OpenLanguageCommand { get; }
    ReactiveCommand<Unit, Unit> AddLanguageCommand { get; }

    public LanguageForestViewModel(ReactiveCommand<LanguageModel, Unit> openLanguageCommand, IProjectDatabase database)
    {
        Database = database;

        OpenLanguageCommand = openLanguageCommand;
        LanguageTrees = new();
        AddLanguageCommand = ReactiveCommand.CreateFromTask(async () =>
        {
            await database.AddAsync(new LanguageModel("Unnamed Language"));
            await LoadAsync();
        });
    }

    public async Task LoadAsync()
    {
        var trees =
            from tree in await Database.GetLanguagesAsync()
            where tree.ParentId == null
            select tree;

        LanguageTrees.Clear();
        foreach (var tree in trees)
        {
            LanguageTrees.Add(new(tree, OpenLanguageCommand, Database));
        }
    }
}
LanguageTreeViewModel.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
using System.Collections.ObjectModel;
using System.Reactive;
using System.Threading.Tasks;
using ReactiveUI;

using Baum.AvaloniaApp.Models;
using Baum.AvaloniaApp.Services;

namespace Baum.AvaloniaApp.ViewModels;

public class LanguageTreeViewModel : ViewModelBase
{
    LanguageModel _language;
    LanguageModel Language { get => _language; set => this.RaiseAndSetIfChanged(ref _language, value); }

    ObservableCollection<LanguageTreeViewModel> Children { get; set; }

    string DecendedFromMessage { get; } = "Decended From:";

    ReactiveCommand<LanguageModel, Unit> OpenLanguageCommand { get; }
    ReactiveCommand<Unit, Unit> AddChildCommand { get; }

    IProjectDatabase Database { get; }

    public LanguageTreeViewModel(
        LanguageModel language,
        ReactiveCommand<LanguageModel, Unit> openLanguageCommand,
        IProjectDatabase database)
    {
        _language = language;
        Children = new();

        Database = database;

        OpenLanguageCommand = openLanguageCommand;
        AddChildCommand = ReactiveCommand.CreateFromTask(async () =>
        {
            await Database.AddAsync(new LanguageModel("Unnamed Language", _language.Id) { Id = Language.Id});
            await LoadAsync();
        });
    }

    public async Task LoadAsync()
    {
        Children.Clear();
        foreach (var child in await Database.GetChildrenAsync(Language.Id))
        {
            Children.Add(new(child, OpenLanguageCommand, Database));
        }
    }
}
LanguageViewModel.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
using System.Collections.ObjectModel;
using System.Reactive;
using System.Reactive.Linq;
using System.Threading.Tasks;
using ReactiveUI;

using Baum.Phonology;

using Baum.AvaloniaApp.Models;
using Baum.AvaloniaApp.Services;

namespace Baum.AvaloniaApp.ViewModels;

public class LanguageViewModel : ViewModelBase
{
    PhonologyData Data { get; set; }
    IProjectDatabase Database { get; set; }

    LanguageModel _languageModel;
    public LanguageModel Language { get => _languageModel; set => this.RaiseAndSetIfChanged(ref _languageModel, value); }

    public ObservableCollection<WordViewModel> Words { get; }

    WordViewModel? _currentWord;
    public WordViewModel? CurrentWord { get => _currentWord; set => this.RaiseAndSetIfChanged(ref _currentWord, value); }

    public ReactiveCommand<Unit, Unit> AddWordCommand { get; }
    public ReactiveCommand<LanguageModel, Unit> SaveCommand { get; }

    public LanguageViewModel(LanguageModel language, IProjectDatabase database, PhonologyData data)
    {
        _languageModel = language;
        Words = new();
        Database = database;
        Data = data;
        AddWordCommand = ReactiveCommand.CreateFromTask(async () =>
        {
            await Database.AddAsync(new WordModel("New Word", "")
            {
                Transient = false,
                LanguageId = Language.Id
            });
            await LoadAsync();
        });

        SaveCommand = ReactiveCommand.CreateFromTask(async (LanguageModel language) =>
        {
            await Database.UpdateAsync(language);
            await LoadAsync();
        });

        this.WhenAnyValue(
            _ => _.Language,
            _ => _.Language.Name,
            _ => _.Language.SoundChange,
            (l, _, _) => l)
            .InvokeCommand(SaveCommand);

        this.WhenAnyValue(_ => _.CurrentWord)
            .IgnoreElements()
            .InvokeCommand(ReactiveCommand.CreateFromTask(_ => LoadAsync()));
    }

    public async Task LoadAsync()
    {
        Words.Clear();
        foreach (var word in await Database.GetWordsAsync(Language.Id, Data))
        {
            Words.Add(new(word, Database, Data));
        }
    }
}
MainWindowViewModel.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
using System;
using System.IO;
using System.Collections.Generic;
using System.Threading.Tasks;
using System.Reactive;
using System.Reactive.Linq;
using Avalonia;
using Avalonia.Platform;
using ReactiveUI;

using Baum.AvaloniaApp.Services;

namespace Baum.AvaloniaApp.ViewModels;

public class MainWindowViewModel : ViewModelBase
{
    ViewModelBase _content;
    public ViewModelBase Content { get => _content; set => this.RaiseAndSetIfChanged(ref _content, value); }

    public Interaction<Unit, bool> ConfirmMigrationInteraction { get; }
    public Interaction<Unit, FileInfo?> RequestFileInteraction { get; }
    public Interaction<Unit, FileInfo?> RequestSaveFileInteraction { get; }
    public Interaction<Unit, FileInfo> RequestTemporaryFileInteraction { get; }

    IProjectDatabaseFactory DatabaseFactory { get; }

    public MainWindowViewModel(IProjectDatabaseFactory databaseFactory)
    {
        _content = new HomeViewModel(
            ReactiveCommand.CreateFromTask(OpenFileAsync),
            ReactiveCommand.CreateFromTask(NewFileAsync));

        ConfirmMigrationInteraction = new();
        RequestFileInteraction = new();
        RequestSaveFileInteraction = new();
        RequestTemporaryFileInteraction = new();
        DatabaseFactory = databaseFactory;
    }

    async Task OpenFileAsync()
    {
        var file = await RequestFileInteraction.Handle(Unit.Default);

        if (file != null)
        {
            // TODO: Implement dirty tracking and stuff
            // var tempFile = await RequestTemporaryFileInteraction.Handle(Unit.Default);
            // file.CopyTo(tempFile.FullName, true);
            var tempFile = file;
            var database = await GetDatabase(tempFile);
            if (database != null)
                await OpenAsync(database, file);
        }
    }

    async Task NewFileAsync()
    {
        var file = await RequestTemporaryFileInteraction.Handle(Unit.Default);
        var database = await GetDatabase(file);
        if (database != null)
            await OpenAsync(database, null);
    }

    async Task OpenAsync(IProjectDatabase database, FileInfo? file)
    {
        var assets = AvaloniaLocator.Current.GetService<IAssetLoader>();
        List<Phonology.Sound> sounds = new();
        using (var stream = assets.Open(new Uri("avares://Baum.AvaloniaApp/Assets/ipa-consonants.csv")))
        {
            using var reader = new StreamReader(stream);
            sounds.AddRange(await Phonology.Utils.CsvLoader.LoadAsync(reader));
        }

        using (var stream = assets.Open(new Uri("avares://Baum.AvaloniaApp/Assets/ipa-vowels.csv")))
        {
            using var reader = new StreamReader(stream);
            sounds.AddRange(await Phonology.Utils.CsvLoader.LoadAsync(reader));
        }

        var vm = new ProjectViewModel(
            database,
            file,
            new(sounds));

        vm.RequestSaveFileInteraction.RegisterHandler(
            async i => i.SetOutput(await RequestSaveFileInteraction.Handle(Unit.Default)));

        Content = vm;
    }

    async Task<IProjectDatabase?> GetDatabase(FileInfo file)
    {
        var database = DatabaseFactory.Create(file);
        if (database.HasMigrations())
        {
            if (!await ConfirmMigrationInteraction.Handle(Unit.Default))
                return null;

            await database.MigrateAsync();
        }
        return database;
    }
}
MigrationConfirmationWindowViewModel.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
using System.Reactive;
using ReactiveUI;

namespace Baum.AvaloniaApp.ViewModels;

public class MigrationConfirmationWindowViewModel : ViewModelBase
{
    public ReactiveCommand<Unit, bool> ConfirmCommand { get; }
    public ReactiveCommand<Unit, bool> RejectCommand { get; }

    public MigrationConfirmationWindowViewModel()
    {
        ConfirmCommand = ReactiveCommand.Create(() => true);
        RejectCommand = ReactiveCommand.Create(() => false);
    }
}
ProjectViewModel.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
using System.IO;
using System.Reactive;
using System.Reactive.Linq;
using System.Threading.Tasks;
using ReactiveUI;

using Baum.Phonology;

using Baum.AvaloniaApp.Models;
using Baum.AvaloniaApp.Services;

namespace Baum.AvaloniaApp.ViewModels;

public class ProjectViewModel : ViewModelBase
{
    PhonologyData Data { get; set; }
    IProjectDatabase Database { get; }
    FileInfo? SaveFileInfo { get; set; }

    ViewModelBase? _content;
    public ViewModelBase? Content { get => _content; set => this.RaiseAndSetIfChanged(ref _content, value); }

    public ReactiveCommand<Unit, Unit> OpenLanguageForestCommand { get; }
    public ReactiveCommand<LanguageModel, Unit> OpenLanguageCommand { get; }
    public ReactiveCommand<Unit, Unit> SaveCommand { get; }

    public Interaction<Unit, FileInfo> RequestSaveFileInteraction { get; }

    public ProjectViewModel(IProjectDatabase database, FileInfo? file, PhonologyData data)
    {
        Data = data;
        Database = database;
        SaveFileInfo = file;
        OpenLanguageCommand = ReactiveCommand.Create<LanguageModel>(OpenLanguage);
        OpenLanguageForestCommand = ReactiveCommand.Create(() =>
        {
            Content = new LanguageForestViewModel(OpenLanguageCommand, Database);
        });
        SaveCommand = ReactiveCommand.CreateFromTask(SaveToFile);
        RequestSaveFileInteraction = new();
        Content = new LanguageForestViewModel(OpenLanguageCommand, Database);
    }

    async Task SaveToFile()
    {
        if (SaveFileInfo == null)
        {
            SaveFileInfo = await RequestSaveFileInteraction.Handle(Unit.Default);
        }
        if (SaveFileInfo != null)
        {
            Database.SaveToFile(SaveFileInfo);
        }
    }

    void OpenLanguage(LanguageModel language) => Content = new LanguageViewModel(language, Database, Data);
}
ViewModelBase.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
1
2
3
4
5
6
7
using ReactiveUI;

namespace Baum.AvaloniaApp.ViewModels;

public class ViewModelBase : ReactiveObject
{
}
WordViewModel.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
using System.Collections.ObjectModel;
using System.Reactive;
using System.Reactive.Linq;
using System.Linq;
using ReactiveUI;

using Baum.Phonology;

using Baum.AvaloniaApp.Models;
using Baum.AvaloniaApp.Services;

namespace Baum.AvaloniaApp.ViewModels;

public class WordViewModel : ViewModelBase
{
    IProjectDatabase Database { get; }

    WordModel _wordModel;
    public WordModel Word { get => _wordModel; set => this.RaiseAndSetIfChanged(ref _wordModel, value); }

    public ObservableCollection<WordModel> Ancestry { get; set; }

    ReactiveCommand<WordModel, Unit> SaveCommand { get; }

    public WordViewModel(WordModel word, IProjectDatabase database, PhonologyData data)
    {
        _wordModel = word;
        Ancestry = new();
        Database = database;

        SaveCommand = ReactiveCommand.CreateFromTask(async (WordModel word) =>
        {
            if (word.Transient)
                Word = await Database.AddAsync(word);
            else
                await Database.UpdateAsync(word);
        });

        this.WhenAnyValue(
            _ => _.Word,
            _ => _.Word.Name,
            _ => _.Word.IPA,
            (w, _, _) => w)
            .Skip(1)
            .InvokeCommand(SaveCommand);

        this.WhenAnyValue(_ => _.Word)
            .InvokeCommand(ReactiveCommand.CreateFromTask(async (WordModel word) =>
            {
                Ancestry.Clear();
                foreach (var ancestor in await Database.GetAncestryAsync(word, data))
                {
                    Ancestry.Add(ancestor);
                }
            }));
    }
}
dir Views
dir LanguageForestView
LanguageForestView.axaml ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
<UserControl
   xmlns="https://github.com/avaloniaui"
   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"
   mc:Ignorable="d"
   d:DesignWidth="800"
   d:DesignHeight="450"
   x:Class="Baum.AvaloniaApp.Views.LanguageForestView">
  <StackPanel>
    <Button
       HorizontalAlignment="Center"
       Command="{Binding AddLanguageCommand}"
       Content="Add Root Language"/>
    <ItemsControl
       Items="{Binding LanguageTrees}"
       Margin="20">
      <ItemsControl.ItemTemplate>
        <DataTemplate>
          <ContentControl
             Content="{Binding}"
             Margin="20"
             HorizontalAlignment="Center"/>
        </DataTemplate>
      </ItemsControl.ItemTemplate>
    </ItemsControl>
  </StackPanel>
</UserControl>
LanguageForestView.axaml.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
using Avalonia.ReactiveUI;
using ReactiveUI;

using Baum.AvaloniaApp.ViewModels;

namespace Baum.AvaloniaApp.Views;

public partial class LanguageForestView : ReactiveUserControl<LanguageForestViewModel>
{
    public LanguageForestView()
    {
        InitializeComponent();

        this.WhenActivated(async d => await ViewModel!.LoadAsync());
    }
}
LanguageTreeView.axaml ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
<UserControl
   xmlns="https://github.com/avaloniaui"
   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"
   mc:Ignorable="d"
   d:DesignWidth="800"
   d:DesignHeight="450"
   x:Class="Baum.AvaloniaApp.Views.LanguageTreeView">
  <StackPanel
     Orientation="Vertical"
     HorizontalAlignment="Center">
    <Button
       HorizontalAlignment="Center"
       Content="{Binding Language.Name}">
      <Button.Flyout>
        <Flyout
           Placement="Bottom">
          <StackPanel>
            <TextBlock
               Text="{Binding Language.Name}"/>
            <Button
               Command="{Binding OpenLanguageCommand}"
               CommandParameter="{Binding Language}"
               Content="Open"/>
          </StackPanel>
        </Flyout>
      </Button.Flyout>
    </Button>
    <Button
       HorizontalAlignment="Center"
       Command="{Binding AddChildCommand}"
       Content="Add Child"/>
    <ItemsControl
       HorizontalAlignment="Center"
       Margin="10"
       Items="{Binding Children}">
      <ItemsControl.ItemsPanel>
        <ItemsPanelTemplate>
          <StackPanel
             Orientation="Horizontal"
             HorizontalAlignment="Center"/>
        </ItemsPanelTemplate>
      </ItemsControl.ItemsPanel>
      <ItemsControl.ItemTemplate>
        <DataTemplate>
          <ContentControl
             Content="{Binding}"
             HorizontalAlignment="Center"/>
        </DataTemplate>
      </ItemsControl.ItemTemplate>
    </ItemsControl>
  </StackPanel>
</UserControl>
LanguageTreeView.axaml.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
using Avalonia.ReactiveUI;
using ReactiveUI;

using Baum.AvaloniaApp.ViewModels;

namespace Baum.AvaloniaApp.Views;

public partial class LanguageTreeView : ReactiveUserControl<LanguageTreeViewModel>
{
    public LanguageTreeView()
    {
        InitializeComponent();

        this.WhenActivated(async d => await ViewModel!.LoadAsync());
    }
}
dir LanguageView
LanguageView.axaml Unicode text, UTF-8 text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
<UserControl
   xmlns="https://github.com/avaloniaui"
   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:views="using:Baum.AvaloniaApp.Views"
   mc:Ignorable="d"
   d:DesignWidth="800"
   d:DesignHeight="450"
   x:Class="Baum.AvaloniaApp.Views.LanguageView">
  <StackPanel
     Orientation="Vertical">
    <Grid
       RowDefinitions="Auto,Auto,Auto"
       ColumnDefinitions="Auto,Auto">
      <TextBox
         Grid.Row="0"
         Grid.Column="0"
         Grid.ColumnSpan="2"
         Margin="10"
         VerticalAlignment="Center"
         Text="{Binding Language.Name}"/>
      <TextBlock
         Grid.Row="1"
         Grid.Column="0"
         Margin="4"
         VerticalAlignment="Center"
         Text="Decended From: "/>
      <TextBlock
         Grid.Row="1"
         Grid.Column="1"
         Margin="4"
         VerticalAlignment="Center"
         Text="{Binding Language.Parent.Name}"/>
      <TextBlock
         Grid.Row="2"
         Grid.Column="0"
         Margin="4"
         VerticalAlignment="Center"
         Text="Sound Change: "/>
      <TextBox
         Grid.Row="2"
         Grid.Column="1"
         Margin="4"
         VerticalAlignment="Center"
         Text="{Binding Language.SoundChange}"/>
    </Grid>
    <Grid
       ColumnDefinitions="*, 4, *">
      <StackPanel>
        <Button
           Command="{Binding AddWordCommand}"
           Content="Add"/>
        <ListBox
           Grid.Column="0"
           Items="{Binding Words}"
           SelectedItem="{Binding CurrentWord}">
          <ListBox.ItemTemplate>
            <DataTemplate>
              <StackPanel
                 Orientation="Horizontal">
                <TextBlock
                   Text="{Binding Word.Name}"/>
                <TextBlock
                   Text="🌿"
                   IsVisible="{Binding Word.Transient}"/>
              </StackPanel>
            </DataTemplate>
          </ListBox.ItemTemplate>
        </ListBox>
      </StackPanel>
      <GridSplitter
         Grid.Column="1"
         Background="Black"
         ResizeDirection="Columns"/>
      <ContentControl
         Grid.Column="2"
         Content="{Binding CurrentWord}"/>
    </Grid>
  </StackPanel>
</UserControl>
LanguageView.axaml.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
using Avalonia.ReactiveUI;
using ReactiveUI;

using Baum.AvaloniaApp.ViewModels;

namespace Baum.AvaloniaApp.Views;

public partial class LanguageView : ReactiveUserControl<LanguageViewModel>
{
    public LanguageView()
    {
        InitializeComponent();

        this.WhenActivated(async d => await ViewModel!.LoadAsync());
    }
}
WordEntryView.axaml ASCII text, with CRLF line terminators
1
2
3
4
5
6
7
8
<UserControl xmlns="https://github.com/avaloniaui"
             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"
             mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
             x:Class="Baum.AvaloniaApp.Views.WordEntryView">
  Welcome to Avalonia!
</UserControl>
WordEntryView.axaml.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
using Avalonia.Controls;

namespace Baum.AvaloniaApp.Views;

public partial class WordEntryView : UserControl
{
    public WordEntryView()
    {
        InitializeComponent();
    }
}
WordView.axaml ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
<UserControl
   xmlns="https://github.com/avaloniaui"
   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"
   mc:Ignorable="d"
   d:DesignWidth="800"
   d:DesignHeight="450"
   x:Class="Baum.AvaloniaApp.Views.WordView">
  <Grid
     RowDefinitions="Auto,Auto,Auto"
     ColumnDefinitions="Auto,Auto">
    <ItemsControl
       Grid.Row="0"
       Grid.Column="0"
       Grid.ColumnSpan="2"
       Items="{Binding Ancestry}">
      <ItemsControl.ItemsPanel>
        <ItemsPanelTemplate>
          <StackPanel
             Orientation="Horizontal"/>
        </ItemsPanelTemplate>
      </ItemsControl.ItemsPanel>
      <ItemsControl.ItemTemplate>
        <DataTemplate>
          <TextBlock
             Margin="4"
             Text="{Binding IPA}"/>
        </DataTemplate>
      </ItemsControl.ItemTemplate>
    </ItemsControl>
    <TextBlock
       Grid.Row="1"
       Grid.Column="0"
       Margin="10"
       VerticalAlignment="Center"
       Text="Word"/>
    <TextBox
       Grid.Row="1"
       Grid.Column="1"
       VerticalAlignment="Center"
       Text="{Binding Word.Name}"/>
    <TextBlock
       Grid.Row="2"
       Grid.Column="0"
       Margin="10"
       VerticalAlignment="Center"
       Text="IPA"/>
    <TextBox
       Grid.Row="2"
       Grid.Column="1"
       VerticalAlignment="Center"
       Text="{Binding Word.IPA}"/>
  </Grid>
</UserControl>
WordView.axaml.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
using Avalonia.Controls;

namespace Baum.AvaloniaApp.Views;

public partial class WordView : UserControl
{
    public WordView()
    {
        InitializeComponent();
    }
}
HomeView.axaml ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
<UserControl
   xmlns="https://github.com/avaloniaui"
   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:vm="using:Baum.AvaloniaApp.ViewModels"
   mc:Ignorable="d"
   d:DesignWidth="800"
   d:DesignHeight="450"
   x:Class="Baum.AvaloniaApp.Views.HomeView"
   x:DataType="vm:HomeViewModel">
  <StackPanel
     Orientation="Horizontal"
     HorizontalAlignment="Center">
    <Button
       Command="{Binding OpenCommand}"
       Content="{Binding OpenButtonText}"/>
    <Button
       Command="{Binding NewCommand}"
       Content="New"/>
  </StackPanel>
</UserControl>
HomeView.axaml.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
using Avalonia.ReactiveUI;

using Baum.AvaloniaApp.ViewModels;

namespace Baum.AvaloniaApp.Views;

public partial class HomeView : ReactiveUserControl<HomeViewModel>
{
    public HomeView()
    {
        InitializeComponent();
    }
}
MainWindow.axaml ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
<Window xmlns="https://github.com/avaloniaui"
        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:vm="using:Baum.AvaloniaApp.ViewModels"
        mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
        x:Class="Baum.AvaloniaApp.Views.MainWindow"
        x:DataType="vm:MainWindowViewModel"
        Title="Baum.AvaloniaApp"
        Icon="/Assets/Baum.png"
        Content="{Binding Content}">
</Window>
MainWindow.axaml.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
using System.IO;
using System.Reactive;
using System.Threading.Tasks;
using Avalonia.Controls;
using Avalonia.ReactiveUI;
using ReactiveUI;

using Baum.AvaloniaApp.ViewModels;

namespace Baum.AvaloniaApp.Views;

public partial class MainWindow : ReactiveWindow<MainWindowViewModel>
{
    public MainWindow()
    {
        InitializeComponent();

        this.WhenActivated(d =>
        {
            d(ViewModel!.RequestFileInteraction.RegisterHandler(ShowOpenFileDialog));
            d(ViewModel.RequestSaveFileInteraction.RegisterHandler(ShowSaveFileDialog));
            d(ViewModel.RequestTemporaryFileInteraction.RegisterHandler(GetTemporaryFile));
            d(ViewModel.ConfirmMigrationInteraction.RegisterHandler(ShowMigrationConfirmationDialog));
        });
    }

    async Task ShowMigrationConfirmationDialog(InteractionContext<Unit, bool> interaction)
    {
        var dialog = new MigrationConfirmationWindow();
        dialog.DataContext = new MigrationConfirmationWindowViewModel();

        var result = await dialog.ShowDialog<bool?>(this);
        interaction.SetOutput(result ?? false);
    }

    public void GetTemporaryFile(InteractionContext<Unit, FileInfo> interaction)
    {
        // TODO: I should probably clean up temp files after usage
        // https://stackoverflow.com/questions/400140/how-do-i-automatically-delete-temp-files-in-c
        interaction.SetOutput(new FileInfo(Path.GetTempFileName()));
    }

    public async Task ShowOpenFileDialog(InteractionContext<Unit, FileInfo?> interaction)
    {
        var dialog = new OpenFileDialog
        {
            AllowMultiple = false,
            Filters = new() {
                new FileDialogFilter {
                    Extensions = new() { "db", "baum" }
                }
            }
        };
        var selections = await dialog.ShowAsync(this);

        if (selections is [var filePath])
        {
            interaction.SetOutput(new FileInfo(filePath));
        }
        else
        {
            interaction.SetOutput(null);
        }
    }

    public async Task ShowSaveFileDialog(InteractionContext<Unit, FileInfo?> interaction)
    {
        var dialog = new SaveFileDialog
        {
            Filters = new() {
                new FileDialogFilter {
                    Extensions = new() { "baum" }
                }
            }
        };

        var saveFile = await dialog.ShowAsync(this);

        if (saveFile != null)
        {
            interaction.SetOutput(new FileInfo(saveFile));
        }
        else
        {
            interaction.SetOutput(null);
        }
    }
}
MigrationConfirmationWindow.axaml ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
<Window
   xmlns="https://github.com/avaloniaui"
   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"
   mc:Ignorable="d"
   d:DesignWidth="800"
   d:DesignHeight="450"
   x:Class="Baum.AvaloniaApp.Views.MigrationConfirmationWindow"
   Title="MigrationConfirmationWindow"
   SizeToContent="WidthAndHeight">
  <Grid
     RowDefinitions="Auto,Auto"
     ColumnDefinitions="Auto,Auto">
    <TextBlock
       Margin="10"
       Grid.Row="0"
       Grid.Column="0"
       Grid.ColumnSpan="2"
       Text="Apply migrations?"/>
    <Button
       Margin="10"
       Grid.Row="1"
       Grid.Column="0"
       Command="{Binding ConfirmCommand}">Yes</Button>
    <Button
       Margin="10"
       Grid.Row="1"
       Grid.Column="1"
       Command="{Binding RejectCommand}">No</Button>
  </Grid>
</Window>
MigrationConfirmationWindow.axaml.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
using System;
using Avalonia;
using Avalonia.Controls;
using Avalonia.Markup.Xaml;
using Avalonia.ReactiveUI;
using ReactiveUI;

using Baum.AvaloniaApp.ViewModels;

namespace Baum.AvaloniaApp.Views;

public partial class MigrationConfirmationWindow : ReactiveWindow<MigrationConfirmationWindowViewModel>
{
    public MigrationConfirmationWindow()
    {
        InitializeComponent();
        this.WhenActivated(d => {
            d(ViewModel!.ConfirmCommand.Subscribe(b => Close(b)));
            d(ViewModel.RejectCommand.Subscribe(b => Close(b)));
        });
    }
}
ProjectView.axaml ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
<UserControl
   xmlns="https://github.com/avaloniaui"
   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:vm="using:Baum.AvaloniaApp.ViewModels"
   mc:Ignorable="d"
   d:DesignWidth="800"
   d:DesignHeight="450"
   x:Class="Baum.AvaloniaApp.Views.ProjectView"
   x:DataType="vm:ProjectViewModel">
  <DockPanel>
    <Menu
       DockPanel.Dock="Top">
      <MenuItem
         Header="_File">
        <MenuItem
           Header="_Save"
           Command="{Binding SaveCommand}"/>
        <Separator/>
        <MenuItem
           Header="_Exit"/>
      </MenuItem>
      <MenuItem
         Header="_Goto">
        <MenuItem
           Header="_Overview"
           Command="{Binding OpenLanguageForestCommand}"/>
      </MenuItem>
    </Menu>
    <ContentControl
       Content="{Binding Content}"/>
  </DockPanel>
</UserControl>
ProjectView.axaml.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
using Avalonia.ReactiveUI;

using Baum.AvaloniaApp.ViewModels;

namespace Baum.AvaloniaApp.Views;

public partial class ProjectView : ReactiveUserControl<ProjectViewModel>
{
    public ProjectView()
    {
        InitializeComponent();
    }
}
App.axaml ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
<Application xmlns="https://github.com/avaloniaui"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:local="using:Baum.AvaloniaApp"
             x:Class="Baum.AvaloniaApp.App">

    <Application.DataTemplates>
        <local:ViewLocator/>
    </Application.DataTemplates>

    <Application.Styles>
        <FluentTheme Mode="Light"/>
    </Application.Styles>
</Application>
App.axaml.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
using Avalonia;
using Avalonia.Controls.ApplicationLifetimes;
using Avalonia.Markup.Xaml;
using Baum.AvaloniaApp.Services;
using Baum.AvaloniaApp.ViewModels;
using Baum.AvaloniaApp.Views;

namespace Baum.AvaloniaApp;

public partial class App : Application
{
    public override void Initialize()
    {
        AvaloniaXamlLoader.Load(this);
    }

    public override void OnFrameworkInitializationCompleted()
    {
        if (ApplicationLifetime is IClassicDesktopStyleApplicationLifetime desktop)
        {
            // Dependency stuff

            ProjectDatabaseFactory databaseFactory = new();

            desktop.MainWindow = new MainWindow
            {
                DataContext = new MainWindowViewModel(databaseFactory),
            };
        }

        base.OnFrameworkInitializationCompleted();
    }
}
Baum.AvaloniaApp.csproj Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
<Project Sdk="Microsoft.NET.Sdk">
  <PropertyGroup>
    <OutputType>WinExe</OutputType>
    <TargetFrameworks>net6.0;net7.0</TargetFrameworks>
    <LangVersion>11.0</LangVersion>
    <Nullable>enable</Nullable>
    <BuiltInComInteropSupport>true</BuiltInComInteropSupport>
    <ApplicationManifest>app.manifest</ApplicationManifest>
    <PublishSingleFile>true</PublishSingleFile>
  </PropertyGroup>

  <ItemGroup>
    <Folder Include="Models\" />
    <AvaloniaResource Include="Assets\**" />
  </ItemGroup>

  <ItemGroup>
    <TrimmerRootAssembly Include="Avalonia.Themes.Fluent" />
  </ItemGroup>

  <ItemGroup>
    <PackageReference Include="Avalonia" Version="0.10.19" />
    <PackageReference Include="Avalonia.Desktop" Version="0.10.19" />
    <!--Condition below is needed to remove Avalonia.Diagnostics package from build output in Release configuration.-->
    <PackageReference Condition="'$(Configuration)' == 'Debug'" Include="Avalonia.Diagnostics" Version="0.10.19" />
    <PackageReference Include="Avalonia.ReactiveUI" Version="0.10.19" />
    <PackageReference Include="PolySharp" Version="1.12.1">
      <IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
      <PrivateAssets>all</PrivateAssets>
    </PackageReference>
    <PackageReference Include="XamlNameReferenceGenerator" Version="1.6.1" />
  </ItemGroup>

  <ItemGroup>
    <ProjectReference Include="..\Baum.Phonology\Baum.Phonology.csproj" />
    <ProjectReference Include="..\Baum.Data\Baum.Data.csproj" />
  </ItemGroup>
</Project>
Program.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
using Avalonia;
using Avalonia.ReactiveUI;
using System;

namespace Baum.AvaloniaApp;

class Program
{
    // Initialization code. Don't use any Avalonia, third-party APIs or any
    // SynchronizationContext-reliant code before AppMain is called: things aren't initialized
    // yet and stuff might break.
    [STAThread]
    public static void Main(string[] args) => BuildAvaloniaApp()
        .StartWithClassicDesktopLifetime(args);

    // Avalonia configuration, don't remove; also used by visual designer.
    public static AppBuilder BuildAvaloniaApp()
        => AppBuilder.Configure<App>()
            .UsePlatformDetect()
            .LogToTrace()
            .UseReactiveUI();
}
ViewLocator.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
using System;
using Avalonia.Controls;
using Avalonia.Controls.Templates;
using Baum.AvaloniaApp.ViewModels;

namespace Baum.AvaloniaApp;

public class ViewLocator : IDataTemplate
{
    public IControl Build(object data)
    {
        var name = data.GetType().FullName!.Replace("ViewModel", "View");
        var type = Type.GetType(name);

        if (type != null)
        {
            return (Control)Activator.CreateInstance(type)!;
        }
        
        return new TextBlock { Text = "Not Found: " + name };
    }

    public bool Match(object data)
    {
        return data is ViewModelBase;
    }
}
app.manifest Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
<?xml version="1.0" encoding="utf-8"?>
<assembly manifestVersion="1.0" xmlns="urn:schemas-microsoft-com:asm.v1">
  <!-- This manifest is used on Windows only.
       Don't remove it as it might cause problems with window transparency and embeded controls.
       For more details visit https://learn.microsoft.com/en-us/windows/win32/sbscs/application-manifests -->
  <assemblyIdentity version="1.0.0.0" name="AvaloniaTest.Desktop"/>

  <compatibility xmlns="urn:schemas-microsoft-com:compatibility.v1">
    <application>
      <!-- A list of the Windows versions that this application has been tested on
           and is designed to work with. Uncomment the appropriate elements
           and Windows will automatically select the most compatible environment. -->

      <!-- Windows 10 -->
      <supportedOS Id="{8e0f7a12-bfb3-4fe8-b9a5-48fd50a15a9a}" />
    </application>
  </compatibility>
</assembly>
dir Baum.Data
dir Migrations
20230327054716_InitialCreate.Designer.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
// <auto-generated />
using System;
using Baum.Data;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Migrations;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;

#nullable disable

namespace Baum.AvaloniaApp.Migrations
{
    [DbContext(typeof(ProjectContext))]
    [Migration("20230327054716_InitialCreate")]
    partial class InitialCreate
    {
        /// <inheritdoc />
        protected override void BuildTargetModel(ModelBuilder modelBuilder)
        {
#pragma warning disable 612, 618
            modelBuilder.HasAnnotation("ProductVersion", "7.0.4");

            modelBuilder.Entity("Baum.AvaloniaApp.Models.Language", b =>
                {
                    b.Property<int>("Id")
                        .ValueGeneratedOnAdd()
                        .HasColumnType("INTEGER");

                    b.Property<string>("Name")
                        .HasColumnType("TEXT");

                    b.Property<int?>("ParentId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("SoundChange")
                        .HasColumnType("TEXT");

                    b.HasKey("Id");

                    b.HasIndex("ParentId");

                    b.ToTable("Languages");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Models.Word", b =>
                {
                    b.Property<int>("Id")
                        .ValueGeneratedOnAdd()
                        .HasColumnType("INTEGER");

                    b.Property<string>("IPA")
                        .IsRequired()
                        .HasColumnType("TEXT");

                    b.Property<int>("LanguageId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("Name")
                        .IsRequired()
                        .HasColumnType("TEXT");

                    b.HasKey("Id");

                    b.HasIndex("LanguageId");

                    b.ToTable("Words");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Models.Language", b =>
                {
                    b.HasOne("Baum.AvaloniaApp.Models.Language", "Parent")
                        .WithMany("Children")
                        .HasForeignKey("ParentId");

                    b.Navigation("Parent");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Models.Word", b =>
                {
                    b.HasOne("Baum.AvaloniaApp.Models.Language", "Language")
                        .WithMany("Words")
                        .HasForeignKey("LanguageId")
                        .OnDelete(DeleteBehavior.Cascade)
                        .IsRequired();

                    b.Navigation("Language");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Models.Language", b =>
                {
                    b.Navigation("Children");

                    b.Navigation("Words");
                });
#pragma warning restore 612, 618
        }
    }
}
20230327054716_InitialCreate.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
using Microsoft.EntityFrameworkCore.Migrations;

#nullable disable

namespace Baum.AvaloniaApp.Migrations
{
    /// <inheritdoc />
    public partial class InitialCreate : Migration
    {
        /// <inheritdoc />
        protected override void Up(MigrationBuilder migrationBuilder)
        {
            migrationBuilder.CreateTable(
                name: "Languages",
                columns: table => new
                {
                    Id = table.Column<int>(type: "INTEGER", nullable: false)
                        .Annotation("Sqlite:Autoincrement", true),
                    Name = table.Column<string>(type: "TEXT", nullable: true),
                    ParentId = table.Column<int>(type: "INTEGER", nullable: true),
                    SoundChange = table.Column<string>(type: "TEXT", nullable: true)
                },
                constraints: table =>
                {
                    table.PrimaryKey("PK_Languages", x => x.Id);
                    table.ForeignKey(
                        name: "FK_Languages_Languages_ParentId",
                        column: x => x.ParentId,
                        principalTable: "Languages",
                        principalColumn: "Id");
                });

            migrationBuilder.CreateTable(
                name: "Words",
                columns: table => new
                {
                    Id = table.Column<int>(type: "INTEGER", nullable: false)
                        .Annotation("Sqlite:Autoincrement", true),
                    Name = table.Column<string>(type: "TEXT", nullable: false),
                    IPA = table.Column<string>(type: "TEXT", nullable: false),
                    LanguageId = table.Column<int>(type: "INTEGER", nullable: false)
                },
                constraints: table =>
                {
                    table.PrimaryKey("PK_Words", x => x.Id);
                    table.ForeignKey(
                        name: "FK_Words_Languages_LanguageId",
                        column: x => x.LanguageId,
                        principalTable: "Languages",
                        principalColumn: "Id",
                        onDelete: ReferentialAction.Cascade);
                });

            migrationBuilder.CreateIndex(
                name: "IX_Languages_ParentId",
                table: "Languages",
                column: "ParentId");

            migrationBuilder.CreateIndex(
                name: "IX_Words_LanguageId",
                table: "Words",
                column: "LanguageId");
        }

        /// <inheritdoc />
        protected override void Down(MigrationBuilder migrationBuilder)
        {
            migrationBuilder.DropTable(
                name: "Words");

            migrationBuilder.DropTable(
                name: "Languages");
        }
    }
}
20230327055105_AddParentToWord.Designer.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
// <auto-generated />
using System;
using Baum.Data;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Migrations;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;

#nullable disable

namespace Baum.AvaloniaApp.Migrations
{
    [DbContext(typeof(ProjectContext))]
    [Migration("20230327055105_AddParentToWord")]
    partial class AddParentToWord
    {
        /// <inheritdoc />
        protected override void BuildTargetModel(ModelBuilder modelBuilder)
        {
#pragma warning disable 612, 618
            modelBuilder.HasAnnotation("ProductVersion", "7.0.4");

            modelBuilder.Entity("Baum.AvaloniaApp.Models.Language", b =>
                {
                    b.Property<int>("Id")
                        .ValueGeneratedOnAdd()
                        .HasColumnType("INTEGER");

                    b.Property<string>("Name")
                        .HasColumnType("TEXT");

                    b.Property<int?>("ParentId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("SoundChange")
                        .HasColumnType("TEXT");

                    b.HasKey("Id");

                    b.HasIndex("ParentId");

                    b.ToTable("Languages");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Models.Word", b =>
                {
                    b.Property<int>("Id")
                        .ValueGeneratedOnAdd()
                        .HasColumnType("INTEGER");

                    b.Property<string>("IPA")
                        .IsRequired()
                        .HasColumnType("TEXT");

                    b.Property<int>("LanguageId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("Name")
                        .IsRequired()
                        .HasColumnType("TEXT");

                    b.Property<int?>("ParentId")
                        .HasColumnType("INTEGER");

                    b.HasKey("Id");

                    b.HasIndex("LanguageId");

                    b.HasIndex("ParentId");

                    b.ToTable("Words");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Models.Language", b =>
                {
                    b.HasOne("Baum.AvaloniaApp.Models.Language", "Parent")
                        .WithMany("Children")
                        .HasForeignKey("ParentId");

                    b.Navigation("Parent");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Models.Word", b =>
                {
                    b.HasOne("Baum.AvaloniaApp.Models.Language", "Language")
                        .WithMany("Words")
                        .HasForeignKey("LanguageId")
                        .OnDelete(DeleteBehavior.Cascade)
                        .IsRequired();

                    b.HasOne("Baum.AvaloniaApp.Models.Word", "Parent")
                        .WithMany()
                        .HasForeignKey("ParentId");

                    b.Navigation("Language");

                    b.Navigation("Parent");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Models.Language", b =>
                {
                    b.Navigation("Children");

                    b.Navigation("Words");
                });
#pragma warning restore 612, 618
        }
    }
}
20230327055105_AddParentToWord.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
using Microsoft.EntityFrameworkCore.Migrations;

#nullable disable

namespace Baum.AvaloniaApp.Migrations
{
    /// <inheritdoc />
    public partial class AddParentToWord : Migration
    {
        /// <inheritdoc />
        protected override void Up(MigrationBuilder migrationBuilder)
        {
            migrationBuilder.AddColumn<int>(
                name: "ParentId",
                table: "Words",
                type: "INTEGER",
                nullable: true);

            migrationBuilder.CreateIndex(
                name: "IX_Words_ParentId",
                table: "Words",
                column: "ParentId");

            migrationBuilder.AddForeignKey(
                name: "FK_Words_Words_ParentId",
                table: "Words",
                column: "ParentId",
                principalTable: "Words",
                principalColumn: "Id");
        }

        /// <inheritdoc />
        protected override void Down(MigrationBuilder migrationBuilder)
        {
            migrationBuilder.DropForeignKey(
                name: "FK_Words_Words_ParentId",
                table: "Words");

            migrationBuilder.DropIndex(
                name: "IX_Words_ParentId",
                table: "Words");

            migrationBuilder.DropColumn(
                name: "ParentId",
                table: "Words");
        }
    }
}
20230328211555_RenameParentToAncestor.Designer.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
// <auto-generated />
using System;
using Baum.Data;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Migrations;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;

#nullable disable

namespace Baum.AvaloniaApp.Migrations
{
    [DbContext(typeof(ProjectContext))]
    [Migration("20230328211555_RenameParentToAncestor")]
    partial class RenameParentToAncestor
    {
        /// <inheritdoc />
        protected override void BuildTargetModel(ModelBuilder modelBuilder)
        {
#pragma warning disable 612, 618
            modelBuilder.HasAnnotation("ProductVersion", "7.0.4");

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Language", b =>
                {
                    b.Property<int>("Id")
                        .ValueGeneratedOnAdd()
                        .HasColumnType("INTEGER");

                    b.Property<string>("Name")
                        .HasColumnType("TEXT");

                    b.Property<int?>("ParentId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("SoundChange")
                        .HasColumnType("TEXT");

                    b.HasKey("Id");

                    b.HasIndex("ParentId");

                    b.ToTable("Languages");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Word", b =>
                {
                    b.Property<int>("Id")
                        .ValueGeneratedOnAdd()
                        .HasColumnType("INTEGER");

                    b.Property<int?>("AncestorId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("IPA")
                        .IsRequired()
                        .HasColumnType("TEXT");

                    b.Property<int>("LanguageId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("Name")
                        .IsRequired()
                        .HasColumnType("TEXT");

                    b.HasKey("Id");

                    b.HasIndex("AncestorId");

                    b.HasIndex("LanguageId");

                    b.ToTable("Words");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Language", b =>
                {
                    b.HasOne("Baum.AvaloniaApp.Services.Database.Language", "Parent")
                        .WithMany("Children")
                        .HasForeignKey("ParentId");

                    b.Navigation("Parent");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Word", b =>
                {
                    b.HasOne("Baum.AvaloniaApp.Services.Database.Word", "Ancestor")
                        .WithMany()
                        .HasForeignKey("AncestorId");

                    b.HasOne("Baum.AvaloniaApp.Services.Database.Language", "Language")
                        .WithMany("Words")
                        .HasForeignKey("LanguageId")
                        .OnDelete(DeleteBehavior.Cascade)
                        .IsRequired();

                    b.Navigation("Ancestor");

                    b.Navigation("Language");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Language", b =>
                {
                    b.Navigation("Children");

                    b.Navigation("Words");
                });
#pragma warning restore 612, 618
        }
    }
}
20230328211555_RenameParentToAncestor.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
using Microsoft.EntityFrameworkCore.Migrations;

#nullable disable

namespace Baum.AvaloniaApp.Migrations
{
    /// <inheritdoc />
    public partial class RenameParentToAncestor : Migration
    {
        /// <inheritdoc />
        protected override void Up(MigrationBuilder migrationBuilder)
        {
            migrationBuilder.DropForeignKey(
                name: "FK_Words_Words_ParentId",
                table: "Words");

            migrationBuilder.RenameColumn(
                name: "ParentId",
                table: "Words",
                newName: "AncestorId");

            migrationBuilder.RenameIndex(
                name: "IX_Words_ParentId",
                table: "Words",
                newName: "IX_Words_AncestorId");

            migrationBuilder.AddForeignKey(
                name: "FK_Words_Words_AncestorId",
                table: "Words",
                column: "AncestorId",
                principalTable: "Words",
                principalColumn: "Id");
        }

        /// <inheritdoc />
        protected override void Down(MigrationBuilder migrationBuilder)
        {
            migrationBuilder.DropForeignKey(
                name: "FK_Words_Words_AncestorId",
                table: "Words");

            migrationBuilder.RenameColumn(
                name: "AncestorId",
                table: "Words",
                newName: "ParentId");

            migrationBuilder.RenameIndex(
                name: "IX_Words_AncestorId",
                table: "Words",
                newName: "IX_Words_ParentId");

            migrationBuilder.AddForeignKey(
                name: "FK_Words_Words_ParentId",
                table: "Words",
                column: "ParentId",
                principalTable: "Words",
                principalColumn: "Id");
        }
    }
}
20230330020149_MakeSoundChangeNonNullable.Designer.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
// <auto-generated />
using System;
using Baum.Data;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Migrations;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;

#nullable disable

namespace Baum.AvaloniaApp.Migrations
{
    [DbContext(typeof(ProjectContext))]
    [Migration("20230330020149_MakeSoundChangeNonNullable")]
    partial class MakeSoundChangeNonNullable
    {
        /// <inheritdoc />
        protected override void BuildTargetModel(ModelBuilder modelBuilder)
        {
#pragma warning disable 612, 618
            modelBuilder.HasAnnotation("ProductVersion", "7.0.4");

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Language", b =>
                {
                    b.Property<int>("Id")
                        .ValueGeneratedOnAdd()
                        .HasColumnType("INTEGER");

                    b.Property<string>("Name")
                        .HasColumnType("TEXT");

                    b.Property<int?>("ParentId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("SoundChange")
                        .IsRequired()
                        .HasColumnType("TEXT");

                    b.HasKey("Id");

                    b.HasIndex("ParentId");

                    b.ToTable("Languages");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Word", b =>
                {
                    b.Property<int>("Id")
                        .ValueGeneratedOnAdd()
                        .HasColumnType("INTEGER");

                    b.Property<int?>("AncestorId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("IPA")
                        .IsRequired()
                        .HasColumnType("TEXT");

                    b.Property<int>("LanguageId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("Name")
                        .IsRequired()
                        .HasColumnType("TEXT");

                    b.HasKey("Id");

                    b.HasIndex("AncestorId");

                    b.HasIndex("LanguageId");

                    b.ToTable("Words");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Language", b =>
                {
                    b.HasOne("Baum.AvaloniaApp.Services.Database.Language", "Parent")
                        .WithMany("Children")
                        .HasForeignKey("ParentId");

                    b.Navigation("Parent");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Word", b =>
                {
                    b.HasOne("Baum.AvaloniaApp.Services.Database.Word", "Ancestor")
                        .WithMany()
                        .HasForeignKey("AncestorId");

                    b.HasOne("Baum.AvaloniaApp.Services.Database.Language", "Language")
                        .WithMany("Words")
                        .HasForeignKey("LanguageId")
                        .OnDelete(DeleteBehavior.Cascade)
                        .IsRequired();

                    b.Navigation("Ancestor");

                    b.Navigation("Language");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Language", b =>
                {
                    b.Navigation("Children");

                    b.Navigation("Words");
                });
#pragma warning restore 612, 618
        }
    }
}
20230330020149_MakeSoundChangeNonNullable.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
using Microsoft.EntityFrameworkCore.Migrations;

#nullable disable

namespace Baum.AvaloniaApp.Migrations
{
    /// <inheritdoc />
    public partial class MakeSoundChangeNonNullable : Migration
    {
        /// <inheritdoc />
        protected override void Up(MigrationBuilder migrationBuilder)
        {
            migrationBuilder.AlterColumn<string>(
                name: "SoundChange",
                table: "Languages",
                type: "TEXT",
                nullable: false,
                defaultValue: "",
                oldClrType: typeof(string),
                oldType: "TEXT",
                oldNullable: true);
        }

        /// <inheritdoc />
        protected override void Down(MigrationBuilder migrationBuilder)
        {
            migrationBuilder.AlterColumn<string>(
                name: "SoundChange",
                table: "Languages",
                type: "TEXT",
                nullable: true,
                oldClrType: typeof(string),
                oldType: "TEXT");
        }
    }
}
ProjectContextModelSnapshot.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
// <auto-generated />
using System;
using Baum.Data;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;

#nullable disable

namespace Baum.AvaloniaApp.Migrations
{
    [DbContext(typeof(ProjectContext))]
    partial class ProjectContextModelSnapshot : ModelSnapshot
    {
        protected override void BuildModel(ModelBuilder modelBuilder)
        {
#pragma warning disable 612, 618
            modelBuilder.HasAnnotation("ProductVersion", "7.0.4");

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Language", b =>
                {
                    b.Property<int>("Id")
                        .ValueGeneratedOnAdd()
                        .HasColumnType("INTEGER");

                    b.Property<string>("Name")
                        .HasColumnType("TEXT");

                    b.Property<int?>("ParentId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("SoundChange")
                        .IsRequired()
                        .HasColumnType("TEXT");

                    b.HasKey("Id");

                    b.HasIndex("ParentId");

                    b.ToTable("Languages");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Word", b =>
                {
                    b.Property<int>("Id")
                        .ValueGeneratedOnAdd()
                        .HasColumnType("INTEGER");

                    b.Property<int?>("AncestorId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("IPA")
                        .IsRequired()
                        .HasColumnType("TEXT");

                    b.Property<int>("LanguageId")
                        .HasColumnType("INTEGER");

                    b.Property<string>("Name")
                        .IsRequired()
                        .HasColumnType("TEXT");

                    b.HasKey("Id");

                    b.HasIndex("AncestorId");

                    b.HasIndex("LanguageId");

                    b.ToTable("Words");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Language", b =>
                {
                    b.HasOne("Baum.AvaloniaApp.Services.Database.Language", "Parent")
                        .WithMany("Children")
                        .HasForeignKey("ParentId");

                    b.Navigation("Parent");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Word", b =>
                {
                    b.HasOne("Baum.AvaloniaApp.Services.Database.Word", "Ancestor")
                        .WithMany()
                        .HasForeignKey("AncestorId");

                    b.HasOne("Baum.AvaloniaApp.Services.Database.Language", "Language")
                        .WithMany("Words")
                        .HasForeignKey("LanguageId")
                        .OnDelete(DeleteBehavior.Cascade)
                        .IsRequired();

                    b.Navigation("Ancestor");

                    b.Navigation("Language");
                });

            modelBuilder.Entity("Baum.AvaloniaApp.Services.Database.Language", b =>
                {
                    b.Navigation("Children");

                    b.Navigation("Words");
                });
#pragma warning restore 612, 618
        }
    }
}
Baum.Data.csproj ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
    <TargetFrameworks>net6.0;net7.0</TargetFrameworks>
    <LangVersion>11.0</LangVersion>
    <ImplicitUsings>enable</ImplicitUsings>
    <Nullable>enable</Nullable>
  </PropertyGroup>

  <ItemGroup>
    <PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="7.0.4">
      <IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
      <PrivateAssets>all</PrivateAssets>
    </PackageReference>
    <PackageReference Include="Microsoft.EntityFrameworkCore.Sqlite" Version="7.0.4" />
    <PackageReference Include="PolySharp" Version="1.12.1">
      <IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
      <PrivateAssets>all</PrivateAssets>
    </PackageReference>
  </ItemGroup>

</Project>
Language.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations.Schema;

namespace Baum.Data;

public class Language
{
    public int Id { get; set; }

    public string? Name { get; set; }
    public List<Word> Words { get; set; } = new();
    public string SoundChange { get; set; } = "";

    public int? ParentId { get; set; }
    public Language? Parent { get; set; }


    [InverseProperty(nameof(Parent))]
    public List<Language> Children { get; set; } = new();
}
ProjectContext.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
using System.IO;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Design;
using Microsoft.Data.Sqlite;

namespace Baum.Data;

public class ProjectContext : DbContext
{
    string ConnectionString { get; init; }

    public DbSet<Language> Languages { get; set; } = default!;
    public DbSet<Word> Words { get; set; } = default!;

    public ProjectContext(string connectionString)
    {
        ConnectionString = connectionString;
    }
    public ProjectContext(FileInfo fileInfo)
    {
        ConnectionString = new SqliteConnectionStringBuilder
        {
            DataSource = fileInfo.FullName,
            Mode = SqliteOpenMode.ReadWriteCreate,
        }.ConnectionString;
    }

    protected override void OnConfiguring(DbContextOptionsBuilder options)
        => options.UseSqlite(ConnectionString);
}

class ProjectContextFactory : IDesignTimeDbContextFactory<ProjectContext>
{
    public ProjectContext CreateDbContext(string[] opts)
    {
        return new ProjectContext("dummy");
    }
}
Word.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
using System.ComponentModel.DataAnnotations.Schema;

namespace Baum.Data;

public class Word
{
    public int Id { get; set; }

    public int? AncestorId { get; set; }

    [ForeignKey(nameof(AncestorId))]
    public Word? Ancestor { get; set; }

    public required string Name { get; set; }

    public required string IPA { get; set; }

    public int LanguageId { get; set; }
    public required Language Language { get; set; }
}
dir Baum.Phonology
dir Notation
MatchNode.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
namespace Baum.Phonology.Notation;

public abstract record MatchNode
{
    public abstract T Accept<T>(IMatchNodeVisitor<T> visitor);
}

public record FeatureSetMatchNode(IReadOnlySet<Feature> Included, IReadOnlySet<Feature> Excluded) : MatchNode
{
    public override T Accept<T>(IMatchNodeVisitor<T> visitor) => visitor.Visit(this);
}

public record SoundMatchNode(IReadOnlySet<Feature> Features) : MatchNode
{
    public override T Accept<T>(IMatchNodeVisitor<T> visitor) => visitor.Visit(this);
}

// List is used for enumeration ordering guarantees
public record MatchListNode(List<MatchNode> Nodes) : MatchNode
{
    public override T Accept<T>(IMatchNodeVisitor<T> visitor) => visitor.Visit(this);
}

public record EmptyNode : MatchNode
{
    public override T Accept<T>(IMatchNodeVisitor<T> visitor) => visitor.Visit(this);
}

public record EndMatchNode : MatchNode
{
    public override T Accept<T>(IMatchNodeVisitor<T> visitor) => visitor.Visit(this);
}
MatchNodeVisitor.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
namespace Baum.Phonology.Notation;

public interface IMatchNodeVisitor<T>
{
    T Visit(FeatureSetMatchNode node);
    T Visit(SoundMatchNode node);
    T Visit(MatchListNode node);
    T Visit(EmptyNode node);
    T Visit(EndMatchNode node);
}
NotationParser.cs ASCII text, with CRLF line terminators
  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
using Baum.Rewrite;

namespace Baum.Phonology.Notation;

public ref struct NotationParser
{
    public static IRewriter<IReadOnlySet<Feature>> Parse(string source, PhonologyData data)
        => new NotationParser(source, data).Parse();

    IEnumerator<Token> _tokens;
    bool _isValid;
    PhonologyData _data;

    Token? CurrentToken => _isValid ? _tokens.Current : null;

    public NotationParser(string source, PhonologyData data)
    {
        _data = data;
        _tokens = new Tokenization(source, data).GetEnumerator();
        Advance();
    }

    public IRewriter<IReadOnlySet<Feature>> Parse()
    {
        var match = NextMatchNode();
        Consume<DerivationSymbol>();
        var replace = NextMatchNode();

        var rewriter = replace.Accept(match.Accept(new SoundChangeRewriteParser()));

        if (_isValid)
        {
            Consume<Slash>();
            rewriter = ParseCondition(rewriter);
        }

        if (_isValid)
            throw new Exception("Incomplete parse: There are still tokens left, but it couln't be parsed");

        return rewriter;

    }

    List<MatchNode> NextMatchSequence()
    {
        List<MatchNode> matchNodes = new();
        while (_isValid && _tokens.Current is SoundToken or OpenBracket or OpenBrace or EndToken)
        {
            matchNodes.Add(NextMatchNode());
        }
        return matchNodes;
    }

    // MatchNode : IPASymbol
    //           | FeatureSet
    //           | List
    //           | End
    MatchNode NextMatchNode()
    {
        switch (CurrentToken)
        {
            case SoundToken { Features: var features }:
                Advance();
                return new SoundMatchNode(features);
            case EndToken:
                Advance();
                return new EndMatchNode();
            case OpenBracket:
                return NextFeatureSet();
            case OpenBrace:
                return NextMatchList();
            default:
                throw new NotImplementedException();
        }
    }

    FeatureSetMatchNode NextFeatureSet()
    {
        Advance();
        HashSet<Feature> included = new(), excluded = new();
        while (true)
        {
            if (CurrentToken is PositiveFeature { Feature: var positive })
            {
                Advance();
                included.Add(positive);
            }
            else if (CurrentToken is NegativeFeature { Feature: var negative })
            {
                Advance();
                excluded.Add(negative);
            }
            else
            {
                break;
            }
            if (CurrentToken is Comma)
                Advance();
        }
        Consume<CloseBracket>();
        return new FeatureSetMatchNode(included, excluded);
    }

    MatchNode NextMatchList()
    {
        Consume<OpenBrace>();
        List<MatchNode> nodes = new();
        while (true)
        {
            if (CurrentToken is SoundToken or OpenBracket or OpenBrace or EndToken)
            {
                nodes.Add(NextMatchNode());
            }
            else
            {
                break;
            }
            if (CurrentToken is Comma)
                Advance();
        }
        Consume<CloseBrace>();
        if (nodes.Any())
            return new MatchListNode(nodes);
        else
            return new EmptyNode();
    }

    IRewriter<IReadOnlySet<Feature>> ParseCondition(IRewriter<IReadOnlySet<Feature>> changeRewriter)
    {
        var rewriter = new SequenceRewriter<IReadOnlySet<Feature>>();

        foreach (var match in NextMatchSequence())
            rewriter.Add(match.Accept(new SoundMatchRewriteMatchParser()));

        Consume<Underscore>();
        rewriter.Add(changeRewriter);

        foreach (var match in NextMatchSequence())
            rewriter.Add(match.Accept(new SoundMatchRewriteMatchParser()));

        return rewriter;
    }

    void Consume<T>() where T : Token
    {
        if (_tokens.Current is T)
        {
            Advance();
        }
        else
        {
            throw new InvalidOperationException();
        }
    }

    void Advance() => _isValid = _tokens.MoveNext();
}
RewriteParser.cs ASCII text, with CRLF line terminators
  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
using Baum.Rewrite;
using Baum.Phonology.Notation;

namespace Baum.Phonology.Notation;

class SoundChangeRewriteParser : IMatchNodeVisitor<IMatchNodeVisitor<IRewriter<IReadOnlySet<Feature>>>>
{
    public IMatchNodeVisitor<IRewriter<IReadOnlySet<Feature>>> Visit(FeatureSetMatchNode matchNode)
        => new SoundMatchRewriteParser(featureSet
            => featureSet.IsSupersetOf(matchNode.Included)
            && !featureSet.Intersect(matchNode.Excluded).Any());

    public IMatchNodeVisitor<IRewriter<IReadOnlySet<Feature>>> Visit(SoundMatchNode matchNode)
        => new SoundMatchRewriteParser(featureSet => featureSet.SetEquals(matchNode.Features));

    public IMatchNodeVisitor<IRewriter<IReadOnlySet<Feature>>> Visit(EmptyNode matchNode)
        => new EmptyMatchRewriteParser();

    public IMatchNodeVisitor<IRewriter<IReadOnlySet<Feature>>> Visit(MatchListNode matchNode)
        => new MatchListRewriteParser(matchNode.Nodes);

    public IMatchNodeVisitor<IRewriter<IReadOnlySet<Feature>>> Visit(EndMatchNode node)
        => new EndMatchRewriteParser();
}

class SoundMatchRewriteParser : IMatchNodeVisitor<IRewriter<IReadOnlySet<Feature>>>
{
    Predicate<IReadOnlySet<Feature>> Match;
    public SoundMatchRewriteParser(Predicate<IReadOnlySet<Feature>> match) => Match = match;

    public IRewriter<IReadOnlySet<Feature>> Visit(FeatureSetMatchNode replaceNode)
        => new MatchRewriter<IReadOnlySet<Feature>>(
            Match,
            match => new[] { new HashSet<Feature>(match.Except(replaceNode.Excluded).Union(replaceNode.Included)) });

    public IRewriter<IReadOnlySet<Feature>> Visit(SoundMatchNode replaceNode)
        => new MatchRewriter<IReadOnlySet<Feature>>(Match, new[] { replaceNode.Features });

    public IRewriter<IReadOnlySet<Feature>> Visit(EmptyNode node)
        => new MatchRewriter<IReadOnlySet<Feature>>(Match, Enumerable.Empty<IReadOnlySet<Feature>>());

    public IRewriter<IReadOnlySet<Feature>> Visit(MatchListNode replaceNode)
        // a > {b,c} makes no sense
        => throw new Exception("Cannot decide between replacements in list");

    public IRewriter<IReadOnlySet<Feature>> Visit(EndMatchNode node)
        => throw new Exception("Word terminator is not a valid replacement");
}

class MatchListRewriteParser : IMatchNodeVisitor<IRewriter<IReadOnlySet<Feature>>>
{
    List<MatchNode> MatchNodes { get; set; }

    public MatchListRewriteParser(List<MatchNode> matchNodes) => MatchNodes = matchNodes;

    public IRewriter<IReadOnlySet<Feature>> Visit(FeatureSetMatchNode replaceNode)
        => new AlternativeRewriter<IReadOnlySet<Feature>>(
            MatchNodes.Select(matchNode => matchNode.Accept(new SoundChangeRewriteParser()).Visit(replaceNode)));

    public IRewriter<IReadOnlySet<Feature>> Visit(SoundMatchNode replaceNode)
        => new AlternativeRewriter<IReadOnlySet<Feature>>(
            MatchNodes.Select(matchNode => matchNode.Accept(new SoundChangeRewriteParser()).Visit(replaceNode)));

    public IRewriter<IReadOnlySet<Feature>> Visit(MatchListNode replaceNode)
        => new AlternativeRewriter<IReadOnlySet<Feature>>(
            Enumerable.Zip(MatchNodes, replaceNode.Nodes)
                .Select(pair => pair.Second.Accept(pair.First.Accept(new SoundChangeRewriteParser()))));

    public IRewriter<IReadOnlySet<Feature>> Visit(EmptyNode replaceNode)
        => new AlternativeRewriter<IReadOnlySet<Feature>>(
            MatchNodes.Select(matchNode => matchNode.Accept(new SoundChangeRewriteParser()).Visit(replaceNode)));

    public IRewriter<IReadOnlySet<Feature>> Visit(EndMatchNode node)
        => throw new Exception("Word terminator is not a valid replacement");
}

class EndMatchRewriteParser : IMatchNodeVisitor<IRewriter<IReadOnlySet<Feature>>>
{
    public IRewriter<IReadOnlySet<Feature>> Visit(FeatureSetMatchNode replaceNode)
        // # > [+voiced] makes little sense
        => throw new Exception("Cannot add or subtract features from the word terminator");

    public IRewriter<IReadOnlySet<Feature>> Visit(SoundMatchNode replaceNode)
        => new EndRewriter<IReadOnlySet<Feature>>(new[] { replaceNode.Features });

    public IRewriter<IReadOnlySet<Feature>> Visit(EmptyNode replaceNode)
        // # > {} also makes no sense
        => throw new Exception("Cannot replace nothing with nothing");

    public IRewriter<IReadOnlySet<Feature>> Visit(MatchListNode node)
        => throw new Exception("Cannot decide between replacements in list");

    public IRewriter<IReadOnlySet<Feature>> Visit(EndMatchNode node)
        => throw new Exception("Word terminator is not a valid replacement");
}

class EmptyMatchRewriteParser : IMatchNodeVisitor<IRewriter<IReadOnlySet<Feature>>>
{
    public IRewriter<IReadOnlySet<Feature>> Visit(FeatureSetMatchNode replaceNode)
        // {} > [+voiced] makes little sense
        => throw new Exception("Cannot add or subtract features from the empty symbol");

    public IRewriter<IReadOnlySet<Feature>> Visit(SoundMatchNode replaceNode)
        => new EmptyRewriter<IReadOnlySet<Feature>>(new[] { replaceNode.Features });

    public IRewriter<IReadOnlySet<Feature>> Visit(EmptyNode replaceNode)
        // {} > {} also makes no sense
        => throw new Exception("Cannot replace nothing with nothing");

    public IRewriter<IReadOnlySet<Feature>> Visit(MatchListNode node)
        => throw new Exception("Cannot decide between replacements in list");

    public IRewriter<IReadOnlySet<Feature>> Visit(EndMatchNode node)
        => throw new Exception("Word terminator is not a valid replacement");
}
SoundChangeNode.cs ASCII text, with CRLF line terminators
1
2
3
4
5
6
7
namespace Baum.Phonology.Notation;

public record SoundChangeNode(
    MatchNode Match,
    MatchNode Replace,
    MatchNode? Precondition,
    MatchNode? PostCondition);
SoundMatchRewriteParser.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
using Baum.Rewrite;
using Baum.Phonology.Notation;

namespace Baum.Phonology.Notation;

// Only does matching, not replacing
class SoundMatchRewriteMatchParser : IMatchNodeVisitor<IRewriter<IReadOnlySet<Feature>>>
{
    public IRewriter<IReadOnlySet<Feature>> Visit(FeatureSetMatchNode matchNode)
        => new MatchRewriter<IReadOnlySet<Feature>>(featureSet
            => featureSet.IsSupersetOf(matchNode.Included)
            && !featureSet.Intersect(matchNode.Excluded).Any());

    public IRewriter<IReadOnlySet<Feature>> Visit(SoundMatchNode matchNode)
        => new MatchRewriter<IReadOnlySet<Feature>>(featureSet => featureSet.SetEquals(matchNode.Features));

    public IRewriter<IReadOnlySet<Feature>> Visit(EmptyNode node)
        => new EmptyRewriter<IReadOnlySet<Feature>>(Enumerable.Empty<IReadOnlySet<Feature>>());

    public IRewriter<IReadOnlySet<Feature>> Visit(MatchListNode node)
        => new AlternativeRewriter<IReadOnlySet<Feature>>(node.Nodes.Select(node => node.Accept(this)));

    public IRewriter<IReadOnlySet<Feature>> Visit(EndMatchNode node)
        => new EndRewriter<IReadOnlySet<Feature>>(Enumerable.Empty<IReadOnlySet<Feature>>());
}
Tokenizer.cs ASCII text, with CRLF line terminators
  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
using System.Collections;

namespace Baum.Phonology.Notation;

record Token;

record DerivationSymbol : Token
{
    private DerivationSymbol() { }
    public static readonly DerivationSymbol Default = new();
}

record Underscore : Token
{
    private Underscore() { }
    public static readonly Underscore Default = new();
}

record Slash : Token
{
    private Slash() { }
    public static readonly Slash Default = new();
}

record OpenBracket : Token
{
    private OpenBracket() { }
    public static readonly OpenBracket Default = new();
}

record CloseBracket : Token
{
    private CloseBracket() { }
    public static readonly CloseBracket Default = new();
}

record OpenBrace : Token
{
    private OpenBrace() { }
    public static readonly OpenBrace Default = new();
}

record CloseBrace : Token
{
    private CloseBrace() { }
    public static readonly CloseBrace Default = new();
}

record Comma : Token
{
    private Comma() { }
    public static readonly Comma Default = new();
}

record EndToken : Token
{
    private EndToken() { }
    public static readonly EndToken Default = new();
} 

record PositiveFeature(Feature Feature) : Token;
record NegativeFeature(Feature Feature) : Token;

record SoundToken(IReadOnlySet<Feature> Features) : Token;

class Tokenizer
{
    PhonologyData _data;
    public Tokenizer(PhonologyData data) => _data = data;
    public Tokenization Tokenize(string source) => new Tokenization(source, _data);
}

class Tokenization : IEnumerable<Token>
{
    string _source;
    PhonologyData _data;

    public Tokenization(string source, PhonologyData data)
        => (_source, _data) = (source, data);

    public IEnumerator<Token> GetEnumerator() => new SoundEnumerator(_source, _data);
    IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
}

class SoundEnumerator : IEnumerator<Token>
{
    PhonologyData _data;
    string _source;
    int _pos;
    Token? _current;

    public SoundEnumerator(string source, PhonologyData data)
    {
        _data = data;
        _source = source;
        _pos = 0;
    }

    public Token Current => _current ?? throw new InvalidOperationException();

    object IEnumerator.Current => Current;

    public bool MoveNext()
    {
        while (_pos < _source.Length)
        {
            switch (_source[_pos])
            {
                case char c when char.IsWhiteSpace(c): ++_pos; continue;
                case '>': ++_pos; _current = DerivationSymbol.Default; return true;
                case '_': ++_pos; _current = Underscore.Default; return true;
                case '/': ++_pos; _current = Slash.Default; return true;
                case '[': ++_pos; _current = OpenBracket.Default; return true;
                case ']': ++_pos; _current = CloseBracket.Default; return true;
                case '{': ++_pos; _current = OpenBrace.Default; return true;
                case '}': ++_pos; _current = CloseBrace.Default; return true;
                case ',': ++_pos; _current = Comma.Default; return true;
                case '#': ++_pos; _current = EndToken.Default; return true;
                case '+':
                    ++_pos;
                    _current = new PositiveFeature(NextFeature());
                    return true;
                case '-':
                    ++_pos;
                    _current = new NegativeFeature(NextFeature());
                    return true;
                default:
                    var sound = _data.GetStartSound(_source.Substring(_pos));
                    _pos += sound.Symbol.Length;
                    _current = new SoundToken(sound.Features);
                    return true;
            }
        }
        return false;
    }

    Feature NextFeature()
    {
        var start = _pos;

        while (char.IsLetter(_source, _pos))
            ++_pos;

        return new Feature(_source[start.._pos].ToString());
    }

    public void Reset() => _pos = 0;

    public void Dispose() { }
}
dir Utils
CSVLoader.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
namespace Baum.Phonology.Utils;

public static class CsvLoader
{
    public static async Task<IEnumerable<Sound>> LoadAsync(TextReader stream)
    {
        // TODO: Use header row for feature categories
        var headerLine = await stream.ReadLineAsync();

        if (headerLine == null)
            throw new InvalidDataException("No header row found");

        var header = headerLine.Split(',').Select(s => s.Trim());
        List<Sound> sounds = new();
        while (stream.ReadLine() is string line)
        {
            var fields = line.Split(',').Select(s => s.Trim());
            var sound = new Sound(
                fields.First(),
                new HashSet<Feature>(fields.Skip(1).Select(field => new Feature(field))));

            sounds.Add(sound);
        }

        return sounds;
    }
}
Baum.Phonology.csproj ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
    <TargetFrameworks>net6.0;net7.0</TargetFrameworks>
    <LangVersion>11.0</LangVersion>
    <ImplicitUsings>enable</ImplicitUsings>
    <Nullable>enable</Nullable>
  </PropertyGroup>

  <ItemGroup>
    <PackageReference Include="PolySharp" Version="1.12.1">
      <IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
      <PrivateAssets>all</PrivateAssets>
    </PackageReference>
  </ItemGroup>

  <ItemGroup>
    <ProjectReference Include="..\Baum.Rewrite\Baum.Rewrite.csproj" />
  </ItemGroup>

</Project>
PhonologyData.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
namespace Baum.Phonology;

public class PhonologyData
{
    IEnumerable<Sound> _sounds;
    public PhonologyData(IEnumerable<Sound> sounds) => _sounds = sounds;

    // TODO? Should this just return null?
    public Sound GetStartSound(string symbol)
        => _sounds.Where(sound => symbol.StartsWith(sound.Symbol))
            .MaxBy(sound => sound.Symbol.Length) ?? throw new Exception($"There is no sound for {symbol}");

    public Sound GetSound(IEnumerable<Feature> features)
        => Enumerable.Single(_sounds.Where(sound => sound.Features.SetEquals(features)));

    public IReadOnlySet<Sound> GetSounds(IReadOnlySet<Feature> includedFeatures, IReadOnlySet<Feature> excludedFeatures)
        => new HashSet<Sound>(_sounds.Where(sound =>
            includedFeatures.IsSubsetOf(sound.Features) &&
            !excludedFeatures.Intersect(sound.Features).Any()));
}
Sound.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
namespace Baum.Phonology;

public sealed record Feature(string Name);
public sealed record Sound(string Symbol, IReadOnlySet<Feature> Features) : IEquatable<Sound>
{
    public override int GetHashCode()
        => Symbol.GetHashCode() ^ Features.Select(f => f.GetHashCode())
            .Aggregate((a, b) => a ^ b);

    public bool Equals(Sound? other)
        => other is not null
        && Symbol == other.Symbol
        && Features.SetEquals(other.Features);
}
SoundChange.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
using Baum.Rewrite;
using Baum.Phonology.Notation;

namespace Baum.Phonology;

public class SoundChange
{
    public static bool TryApply(string initial, string rule, PhonologyData data, out string after)
    {
        try
        {
            var rewriter = NotationParser.Parse(rule, data);
            var change = new SoundChange
            {
                PhonologyData = data,
                Rewriter = rewriter
            };

            after = change.Apply(initial);
            return true;
        }
        catch (Exception) // TODO: Specialize exception
        {
            after = initial;
            return false;
        }
    }

    public required IRewriter<IReadOnlySet<Feature>> Rewriter { get; set; }
    public required PhonologyData PhonologyData { get; set; }

    string Apply(string str)
    {
        var featureString = new Tokenization(str, PhonologyData)
            .Select(sound => ((SoundToken)sound).Features);

        // TODO: Not quite sure if this algorithm is actually the best way to do this
        // Replaces every match in the string
        int maxLength = featureString.Count() * 3;
        for (int pos = 0; pos < featureString.Count(); ++pos)
        {
            // Prevents infinite insertions like {} > p / p_ where the Count keeps growing
            if (pos > maxLength)
                throw new Exception("Word tripled in length, triggering infinite loop protection");
            var rewrites = Rewriter.Rewrite(featureString, pos);
            if (rewrites.Any())
            {
                var replacement = rewrites.MaxBy(pair => pair.RewritePosition);
                featureString = featureString
                    .Take(pos)
                    .Concat(replacement.Rewrite)
                    .Concat(featureString.Skip(replacement.RewritePosition));
            }
        }

        return string.Concat(featureString.Select(f => PhonologyData.GetSound(f).Symbol));
    }
}
dir Baum.Phonology.Tests
Baum.Phonology.Tests.csproj ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
    <TargetFrameworks>net6.0;net7.0</TargetFrameworks>
    <LangVersion>11.0</LangVersion>
    <ImplicitUsings>enable</ImplicitUsings>
    <Nullable>enable</Nullable>

    <IsPackable>false</IsPackable>
    <IsTestProject>true</IsTestProject>
  </PropertyGroup>

  <ItemGroup>
    <PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.5.0" />
    <PackageReference Include="PolySharp" Version="1.12.1">
      <IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
      <PrivateAssets>all</PrivateAssets>
    </PackageReference>
    <PackageReference Include="xunit" Version="2.4.2" />
    <PackageReference Include="xunit.runner.visualstudio" Version="2.4.5">
      <IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
      <PrivateAssets>all</PrivateAssets>
    </PackageReference>
    <PackageReference Include="coverlet.collector" Version="3.2.0">
      <IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
      <PrivateAssets>all</PrivateAssets>
    </PackageReference>
  </ItemGroup>

  <ItemGroup>
    <ProjectReference Include="..\Baum.Phonology\Baum.Phonology.csproj" />
  </ItemGroup>

</Project>
CsvLoaderTest.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
using Baum.Phonology.Utils;

namespace Baum.Phonology.Tests;

public class CsvLoaderTest
{
    [Fact]
    async Task SimpleCsvPasses()
    {
        var inputText =
            "Symbol,Feature1,Feature2\n" +
            "x,a,b\n" +
            "y,c,d";

        var expected = new Sound[]
        {
            new("x", new HashSet<Feature> { new("a"), new("b") }),
            new("y", new HashSet<Feature> { new("c"), new("d") })
        };

        using var reader = new StringReader(inputText);

        var sounds = await CsvLoader.LoadAsync(reader);

        Assert.Equal(expected, sounds);
    }

    [Fact]
    async Task EmptyCsvThrows()
    {
        var inputText = "";
        using var reader = new StringReader(inputText);

        await Assert.ThrowsAnyAsync<Exception>(async () =>
        {
            await CsvLoader.LoadAsync(reader);
        });
    }
}
MatchNodeTest.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
namespace Baum.Phonology.Tests;

public class FeatureSetTest
{
    PhonologyData stubData = new(new[]
    {
        new Sound("a", new HashSet<Feature>() { new("vowel"), new("open")}),
        new Sound("e", new HashSet<Feature>() { new("vowel"), new("close-mid") }),
        new Sound("m", new HashSet<Feature>() { new("consonant"), new("nasal") } ),
        new Sound("p", new HashSet<Feature>() { new("consonant"), new("plosive") } ),
        new Sound("b", new HashSet<Feature>() { new("consonant"), new("plosive"), new("voiced") } ),
    });

    // p > [+voice]
    // pat > bat
    // p > [+voice] / _a
    // pat > bat, put > put


    // Or conditions
    // Syllable/word boundary conditions

}
NotationTest.cs Unicode text, UTF-8 text, with CRLF line terminators
  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
namespace Baum.Phonology.Tests;

public class NotationTest
{
    PhonologyData stubData = new(new[]
    {
        new Sound("a", new HashSet<Feature>() { new("vowel"), new("open")}),
        new Sound("e", new HashSet<Feature>() { new("vowel"), new("close-mid") }),
        new Sound("m̥", new HashSet<Feature>() { new("consonant"), new("nasal") }),
        new Sound("m", new HashSet<Feature>() { new("consonant"), new("nasal"), new("voiced") }),
        new Sound("p", new HashSet<Feature>() { new("consonant"), new("plosive") } ),
        new Sound("b", new HashSet<Feature>() { new("consonant"), new("plosive"), new("voiced") } ),
    });

    [Theory]
    [InlineData("p>b", "pam", "bam")]
    [InlineData("p>b", "papam", "babam")]
    [InlineData("a>e", "pam", "pem")]
    public void SimpleNotationPasses(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }

    [Theory]
    [InlineData("p > b / a_", "pape", "pabe")]
    [InlineData("p > b / a_", "pamap", "pamab")]
    [InlineData("a > e / [-voiced]_", "pap", "pep")]
    [InlineData("a > e / [-voiced]_", "map", "map")]
    public void PreconditionNotationPasses(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }

    [Theory]
    [InlineData("p > b / _a", "pape", "bape")]
    [InlineData("p > b / _a", "pamap", "bamap")]
    [InlineData("a > e / _[-voiced]", "pap", "pep")]
    [InlineData("a > e / _[-voiced]", "pam", "pam")]
    public void PostconditionNotationPasses(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }

    [Theory]
    [InlineData("p > {}", "pamep", "ame")]
    public void UnconditionalDeletionPasses(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }

    [Theory]
    [InlineData("{} > p / e_", "eme", "epmep")]
    public void PreconditionInsertionPasses(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }

    [Theory]
    [InlineData("{} > p / p_", "pep")]
    public void InfiniteSoundChangeFails(string rule, string initial)
    {
        Assert.False(SoundChange.TryApply(initial, rule, stubData, out var result));
    }

    [Theory]
    [InlineData("p > [+voiced]", "pem", "bem")]
    public void AddingFeaturesPasses(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }

    [Theory]
    [InlineData("b > [-voiced]", "bem", "pem")]
    public void SubtractingFeaturesPasses(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }

    [Theory]
    [InlineData("[+voiced] > m", "peb", "pem")]
    [InlineData("[+plosive -voiced] > {}", "peb", "eb")]
    public void MatchingFeaturesPasses(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }

    [Theory]
    [InlineData("{a,e,{p,b}} > {e,a,{b,p}}", "pabe", "bepa")]
    public void MappingBracedList(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }

    [Theory]
    [InlineData("{p,b} > m", "pabe", "mame")]
    public void BracedListToSound(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }

    [Theory]
    [InlineData("{b,m} > [-voiced]", "bame", "pam̥e")]
    [InlineData("{p,m̥} > [+voiced]", "pam̥e", "bame")]
    public void BracedListToFeatureChange(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }
    
    [Theory]
    [InlineData("p > b / _{a,e}", "papepm", "babepm")]
    [InlineData("p > b / _{e,#}", "pep", "beb")]
    public void BracedListCondition(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }

    [Theory]
    [InlineData("e > {} / _#", "peme", "pem")]
    public void WordEndPostCondition(string rule, string initial, string expected)
    {
        Assert.True(SoundChange.TryApply(initial, rule, stubData, out var result));
        Assert.Equal(expected, result);
    }
}
SoundChangeTest.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
namespace Baum.Phonology.Tests;

public class SoundChangeTest
{
    PhonologyData stubData = new(new[]
    {
        new Sound("a", new HashSet<Feature>() { new("vowel"), new("open")}),
        new Sound("e", new HashSet<Feature>() { new("vowel"), new("close-mid") }),
        new Sound("m", new HashSet<Feature>() { new("consonant"), new("nasal") } ),
        new Sound("p", new HashSet<Feature>() { new("consonant"), new("plosive") } ),
        new Sound("b", new HashSet<Feature>() { new("consonant"), new("plosive"), new("voiced") } ),
    });

    // [Fact]
    // public void VoicingAddition()
    // {
    //     var change = new SoundChange
    //     {
    //         PhonologyData = stubData,
    //         MatchNode = new Notation.SoundMatchNode(new HashSet<Feature> { new("consonant"), new("plosive") }),
    //         Replacement = new Notation.FeatureSetMatchNode(
    //             new HashSet<Feature> { new("voiced") },
    //             new HashSet<Feature>())
    //     };

    //     var result = change.Apply("pam");
    //     Assert.Equal("bam", result);
    // }

    // [Fact]
    // public void Devoicing()
    // {
    //     var change = new SoundChange
    //     {
    //         PhonologyData = stubData,
    //         MatchNode = new Notation.SoundMatchNode(
    //             new HashSet<Feature> { new("consonant"), new("plosive"), new("voiced") }),
    //         Replacement = new Notation.FeatureSetMatchNode(
    //             new HashSet<Feature>(),
    //             new HashSet<Feature> { new("voiced") })
    //     };

    //     var result = change.Apply("bam");
    //     Assert.Equal("pam", result);
    // }

    // [Fact]
    // public void Insertion()
    // {
    //     var change = new SoundChange
    //     {
    //         PhonologyData = stubData,
    //         Regex = new("(?<=p)(?=t)"),
    //         Actions = new() {
    //             new SoundChange.InsertAction(new Feature[] { new("vowel"), new("open") })
    //         }
    //     };

    //     var result = change.Apply("pt");
    //     Assert.Equal("pat", result);
    // }

    // [Fact]
    // public void Deletion()
    // {
    //     var change = new SoundChange
    //     {
    //         PhonologyData = stubData,
    //         Regex = new("(?<=p)a(?=t)"),
    //         Actions = new() { new SoundChange.DeleteAction() }
    //     };

    //     var result = change.Apply("pat");
    //     Assert.Equal("pt", result);
    // }
}
SoundTest.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
namespace Baum.Phonology.Tests;

public class SoundTest
{
    [Fact]
    public void OrderOfFeaturesDoesNotMatterForEquality()
    {
        Sound firstSound = new("x", new HashSet<Feature> { new("a"), new("b"), new("c") });
        Sound secondSound = new("x", new HashSet<Feature> { new("b"), new("c"), new("a") });

        Assert.True(firstSound == secondSound);
        Assert.True(secondSound == firstSound);

        Assert.True(firstSound.Equals(secondSound));
        Assert.True(secondSound.Equals(firstSound));

        Assert.Equal(firstSound, secondSound);
        Assert.Equal(firstSound.GetHashCode(), secondSound.GetHashCode());
    }
}
Usings.cs ASCII text, with no line terminators
1
global using Xunit;
dir Baum.Rewrite
AlternativeRewriter.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
using System.Collections;

namespace Baum.Rewrite;

public class AlternativeRewriter<T> : IRewriter<T>, IEnumerable<IRewriter<T>>
{
    List<IRewriter<T>> Rewriters { get; set; }

    public AlternativeRewriter() : this(new List<IRewriter<T>>()) { }
    public AlternativeRewriter(IEnumerable<IRewriter<T>> rewriters) => Rewriters = rewriters.ToList();

    public IEnumerable<RewritePair<T>> Rewrite(IEnumerable<T> sequence, int startPosition)
        => Rewriters.SelectMany(rewriter => rewriter.Rewrite(sequence, startPosition));

    #region IEnumerable<IRewriter<T>>

    public IEnumerator<IRewriter<T>> GetEnumerator()
    {
        throw new NotImplementedException();
    }

    IEnumerator IEnumerable.GetEnumerator()
    {
        throw new NotImplementedException();
    }

    #endregion

    public void Add(IRewriter<T> rewriter) => Rewriters.Add(rewriter);
}
Baum.Rewrite.csproj ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
    <TargetFrameworks>net6.0;net7.0</TargetFrameworks>
    <LangVersion>11.0</LangVersion>
    <ImplicitUsings>enable</ImplicitUsings>
    <Nullable>enable</Nullable>
  </PropertyGroup>

  <ItemGroup>
    <PackageReference Include="PolySharp" Version="1.12.1">
      <IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
      <PrivateAssets>all</PrivateAssets>
    </PackageReference>
  </ItemGroup>

</Project>
EmptyRewriter.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
namespace Baum.Rewrite;

public class EmptyRewriter<T> : IRewriter<T>
{
    IEnumerable<T> Insertion;

    public EmptyRewriter(IEnumerable<T> insertion) => Insertion = insertion;

    public IEnumerable<RewritePair<T>> Rewrite(IEnumerable<T> sequence, int startPosition)
    {
        return new RewritePair<T>[]
        {
            new RewritePair<T>
            {
                Rewrite = Insertion,
                RewritePosition = startPosition
            }
        };
    }
}
EndRewriter.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
namespace Baum.Rewrite;

public class EndRewriter<T> : IRewriter<T>
{
    IEnumerable<T> Insertion;

    public EndRewriter(IEnumerable<T> insertion) => Insertion = insertion;

    public IEnumerable<RewritePair<T>> Rewrite(IEnumerable<T> sequence, int startPosition)
    {
        if (startPosition == sequence.Count())
        {
            return new RewritePair<T>[]
            {
                new RewritePair<T>
                {
                    Rewrite = Insertion,
                    RewritePosition = startPosition
                }
            };
        }
        else {
            return Enumerable.Empty<RewritePair<T>>();
        }
    }
}
IRewriter.cs Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
using System.Diagnostics.CodeAnalysis;

namespace Baum.Rewrite;

public struct RewritePair<T>
{
    public required IEnumerable<T> Rewrite;
    public required int RewritePosition;
}

public interface IRewriter<T>
{
    IEnumerable<RewritePair<T>> Rewrite(IEnumerable<T> sequence, int startPosition);
}
MatchRewriter.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
using System.Diagnostics.CodeAnalysis;

namespace Baum.Rewrite;

public class MatchRewriter<T> : IRewriter<T>
{
    Predicate<T> Match;
    Func<T, IEnumerable<T>> Replace;

    public MatchRewriter(T match)
        : this(x => object.Equals(x, match)) { }

    public MatchRewriter(Predicate<T> match)
        : this(match, x => new[] { x }) { }

    public MatchRewriter(T match, IEnumerable<T> replacement)
        : this(x => object.Equals(x, match), x => replacement) { }

    public MatchRewriter(Predicate<T> match, IEnumerable<T> replacement)
        : this(match, x => replacement) { }

    public MatchRewriter(Predicate<T> match, Func<T, IEnumerable<T>> replace)
        => (Match, Replace) = (match, replace);


    public IEnumerable<RewritePair<T>> Rewrite(IEnumerable<T> sequence, int startPosition)
    {
        var element = sequence.ElementAtOrDefault(startPosition);
        if (element is not null && Match(element))
        {
            return new RewritePair<T>[] {
                new RewritePair<T> {
                    Rewrite = Replace(element),
                    RewritePosition = startPosition + 1
                }
            };
        }
        else
        {
            return Enumerable.Empty<RewritePair<T>>();
        }
    }
}
SequenceRewriter.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
using System.Collections;
using System.Diagnostics.CodeAnalysis;
using System.Linq;

namespace Baum.Rewrite;


public class SequenceRewriter<T> : IRewriter<T>, IEnumerable<IRewriter<T>>
{
    List<IRewriter<T>> Rewriters = new();

    #region IRewriter<T>

    // TODO: Consider: If Rewriters is empty, should it throw, return an empty result, or return an unchanged result
    public IEnumerable<RewritePair<T>> Rewrite(IEnumerable<T> sequence, int startPosition)
    {
        var rewritePairs = Rewriters.First().Rewrite(sequence, startPosition);

        foreach (var rewriter in Rewriters.Skip(1))
        {
            rewritePairs = rewritePairs.SelectMany(pair
                => rewriter.Rewrite(sequence, pair.RewritePosition)
                    .Select(nextPair => new RewritePair<T>
                    {
                        Rewrite = pair.Rewrite.Concat(nextPair.Rewrite),
                        RewritePosition = nextPair.RewritePosition
                    }));
        }

        return rewritePairs;
    }

    #endregion

    #region IEnumerable<IRewriter<T>>

    public IEnumerator<IRewriter<T>> GetEnumerator()
    {
        throw new NotImplementedException();
    }

    IEnumerator IEnumerable.GetEnumerator()
    {
        throw new NotImplementedException();
    }

    #endregion

    public void Add(IRewriter<T> rewriter) => Rewriters.Add(rewriter);
}
dir Baum.Rewrite.Tests
AlternativeRewriterTest.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
namespace Baum.Rewrite.Tests;

public class AlternativeRewriterTest
{
    [Fact]
    void Test1()
    {
        AlternativeRewriter<char> rewriter = new() {
            new MatchRewriter<char>('a', "b"),
            new MatchRewriter<char>('b', "a")
        };

        var result1 = string.Concat(rewriter.Rewrite("a", 0).First().Rewrite);
        var result2 = string.Concat(rewriter.Rewrite("b", 0).First().Rewrite);

        Assert.Equal("b", result1);
        Assert.Equal("a", result2);
    }
}
Baum.Rewrite.Tests.csproj ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
    <TargetFrameworks>net6.0;net7.0</TargetFrameworks>
    <ImplicitUsings>enable</ImplicitUsings>
    <Nullable>enable</Nullable>

    <IsPackable>false</IsPackable>
    <IsTestProject>true</IsTestProject>
  </PropertyGroup>

  <ItemGroup>
    <PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.5.0" />
    <PackageReference Include="xunit" Version="2.4.2" />
    <PackageReference Include="xunit.runner.visualstudio" Version="2.4.5">
      <IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
      <PrivateAssets>all</PrivateAssets>
    </PackageReference>
    <PackageReference Include="coverlet.collector" Version="3.2.0">
      <IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
      <PrivateAssets>all</PrivateAssets>
    </PackageReference>
  </ItemGroup>

  <ItemGroup>
    <ProjectReference Include="..\Baum.Rewrite\Baum.Rewrite.csproj" />
  </ItemGroup>

</Project>
SequenceRewriterTest.cs ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
namespace Baum.Rewrite.Tests;

public class SequenceRewriterTest
{
    [Fact]
    public void Test1()
    {
        SequenceRewriter<char> rewriter = new() {
            new MatchRewriter<char>('a'),
            new MatchRewriter<char>('b', "XY"),
            new MatchRewriter<char>('c')
        };

        var result = string.Concat(rewriter.Rewrite("abc", 0).First().Rewrite);

        Assert.Equal("aXYc", result);
    }

    [Fact]
    public void InsertionAtEndWorks()
    {
        SequenceRewriter<char> rewriter = new() {
            new MatchRewriter<char>('a'),
            new MatchRewriter<char>('b'),
            new EndRewriter<char>("c")
        };

        var result = string.Concat(rewriter.Rewrite("ab", 0).First().Rewrite);

        Assert.Equal("abc", result);
    }

    [Fact]
    public void EndDoesntMatchNotAtEnd()
    {
        SequenceRewriter<char> rewriter = new() {
            new MatchRewriter<char>('a'),
            new EndRewriter<char>("")
        };
        Assert.Empty(rewriter.Rewrite("ab", 0));
    }

    [Fact]
    public void InsertionInBetweenMatchesWorks()
    {
        SequenceRewriter<char> rewriter = new() {
            new MatchRewriter<char>('a'),
            new EmptyRewriter<char>("b"),
            new MatchRewriter<char>('c')
        };

        var result = string.Concat(rewriter.Rewrite("ac", 0).First().Rewrite);

        Assert.Equal("abc", result);
    }
}
Usings.cs ASCII text, with no line terminators
1
global using Xunit;
.gitignore ASCII text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
*.swp
*.*~
project.lock.json
.DS_Store
*.pyc
nupkg/

# Visual Studio Code
.vscode

# Rider
.idea

# User-specific files
*.suo
*.user
*.userosscache
*.sln.docstates

# Build results
[Dd]ebug/
[Dd]ebugPublic/
[Rr]elease/
[Rr]eleases/
x64/
x86/
build/
bld/
[Bb]in/
[Oo]bj/
[Oo]ut/
msbuild.log
msbuild.err
msbuild.wrn

# Visual Studio 2015
.vs/
Baum.sln Unicode text, UTF-8 (with BOM) text, with CRLF line terminators
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 17
VisualStudioVersion = 17.0.31903.59
MinimumVisualStudioVersion = 10.0.40219.1
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Baum.Phonology", "Baum.Phonology\Baum.Phonology.csproj", "{23CE9414-F739-470C-9ED6-8581A3ACA230}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Baum.Phonology.Tests", "Baum.Phonology.Tests\Baum.Phonology.Tests.csproj", "{D9D77B81-0BC2-46D6-A5BC-D3452590347D}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Baum.AvaloniaApp", "Baum.AvaloniaApp\Baum.AvaloniaApp.csproj", "{BA32A7A9-89B4-4700-A50F-0057A1E74AB8}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Baum.Rewrite", "Baum.Rewrite\Baum.Rewrite.csproj", "{9D05D107-156B-4949-86FE-B3C8AA3CB0B4}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Baum.Rewrite.Tests", "Baum.Rewrite.Tests\Baum.Rewrite.Tests.csproj", "{51936B52-11DD-4AF1-9793-BFF1C54DF025}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Baum.Data", "Baum.Data\Baum.Data.csproj", "{E5FABE17-5DF8-40B4-A2F6-3BE1E1E0B8E0}"
EndProject
Global
	GlobalSection(SolutionConfigurationPlatforms) = preSolution
		Debug|Any CPU = Debug|Any CPU
		Release|Any CPU = Release|Any CPU
	EndGlobalSection
	GlobalSection(SolutionProperties) = preSolution
		HideSolutionNode = FALSE
	EndGlobalSection
	GlobalSection(ProjectConfigurationPlatforms) = postSolution
		{23CE9414-F739-470C-9ED6-8581A3ACA230}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
		{23CE9414-F739-470C-9ED6-8581A3ACA230}.Debug|Any CPU.Build.0 = Debug|Any CPU
		{23CE9414-F739-470C-9ED6-8581A3ACA230}.Release|Any CPU.ActiveCfg = Release|Any CPU
		{23CE9414-F739-470C-9ED6-8581A3ACA230}.Release|Any CPU.Build.0 = Release|Any CPU
		{D9D77B81-0BC2-46D6-A5BC-D3452590347D}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
		{D9D77B81-0BC2-46D6-A5BC-D3452590347D}.Debug|Any CPU.Build.0 = Debug|Any CPU
		{D9D77B81-0BC2-46D6-A5BC-D3452590347D}.Release|Any CPU.ActiveCfg = Release|Any CPU
		{D9D77B81-0BC2-46D6-A5BC-D3452590347D}.Release|Any CPU.Build.0 = Release|Any CPU
		{F3643DAD-E987-4A4C-B5DE-99ADD5A050BE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
		{F3643DAD-E987-4A4C-B5DE-99ADD5A050BE}.Debug|Any CPU.Build.0 = Debug|Any CPU
		{F3643DAD-E987-4A4C-B5DE-99ADD5A050BE}.Release|Any CPU.ActiveCfg = Release|Any CPU
		{F3643DAD-E987-4A4C-B5DE-99ADD5A050BE}.Release|Any CPU.Build.0 = Release|Any CPU
		{BA32A7A9-89B4-4700-A50F-0057A1E74AB8}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
		{BA32A7A9-89B4-4700-A50F-0057A1E74AB8}.Debug|Any CPU.Build.0 = Debug|Any CPU
		{BA32A7A9-89B4-4700-A50F-0057A1E74AB8}.Release|Any CPU.ActiveCfg = Release|Any CPU
		{BA32A7A9-89B4-4700-A50F-0057A1E74AB8}.Release|Any CPU.Build.0 = Release|Any CPU
		{9D05D107-156B-4949-86FE-B3C8AA3CB0B4}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
		{9D05D107-156B-4949-86FE-B3C8AA3CB0B4}.Debug|Any CPU.Build.0 = Debug|Any CPU
		{9D05D107-156B-4949-86FE-B3C8AA3CB0B4}.Release|Any CPU.ActiveCfg = Release|Any CPU
		{9D05D107-156B-4949-86FE-B3C8AA3CB0B4}.Release|Any CPU.Build.0 = Release|Any CPU
		{51936B52-11DD-4AF1-9793-BFF1C54DF025}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
		{51936B52-11DD-4AF1-9793-BFF1C54DF025}.Debug|Any CPU.Build.0 = Debug|Any CPU
		{51936B52-11DD-4AF1-9793-BFF1C54DF025}.Release|Any CPU.ActiveCfg = Release|Any CPU
		{51936B52-11DD-4AF1-9793-BFF1C54DF025}.Release|Any CPU.Build.0 = Release|Any CPU
		{E5FABE17-5DF8-40B4-A2F6-3BE1E1E0B8E0}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
		{E5FABE17-5DF8-40B4-A2F6-3BE1E1E0B8E0}.Debug|Any CPU.Build.0 = Debug|Any CPU
		{E5FABE17-5DF8-40B4-A2F6-3BE1E1E0B8E0}.Release|Any CPU.ActiveCfg = Release|Any CPU
		{E5FABE17-5DF8-40B4-A2F6-3BE1E1E0B8E0}.Release|Any CPU.Build.0 = Release|Any CPU
	EndGlobalSection
EndGlobal