Files
Fellowship-Reference/Fellowship/Web/Pages/Keyboard.razor
T

461 lines
18 KiB
Plaintext

@page "/keyboard"
<PageTitle>Keyboard</PageTitle>
<div class="character-selector">
<label>Character:</label>
<select @bind="selectedCharacter" @bind:after="RebuildKeyboardData">
@foreach (var c in Characters)
{
<option value="@c">@c</option>
}
</select>
</div>
<div class="keyboard-container" @onkeydown="HandleKeyDown" tabindex="0" @ref="containerRef">
<div class="keyboard-wrapper">
<div class="kb-row">
@foreach (var k in Keys.Take(6))
{
<div class="kb-key @(k.OnCooldown ? "on-cd" : "")" @onclick="() => ActivateKey(k)" title="@k.Tooltip">
<span class="key-label">@k.Label</span>
@if (k.SkillName != null)
{
<span class="skill-name">@k.SkillName</span>
}
@if (k.OnCooldown)
{
<div class="cd-overlay">
<svg viewBox="0 0 100 100" class="cd-svg">
<circle cx="50" cy="50" r="45" class="cd-bg"/>
<circle cx="50" cy="50" r="45" class="cd-progress"
stroke-dasharray="282.74"
stroke-dashoffset="@(282.74 * (1 - k.CooldownFraction))"/>
</svg>
<span class="cd-text">@k.RemainingSeconds</span>
</div>
}
</div>
}
</div>
<div class="kb-row kb-offset-1">
@foreach (var k in Keys.Skip(6).Take(5))
{
<div class="kb-key @(k.OnCooldown ? "on-cd" : "")" @onclick="() => ActivateKey(k)" title="@k.Tooltip">
<span class="key-label">@k.Label</span>
@if (k.SkillName != null)
{
<span class="skill-name">@k.SkillName</span>
}
@if (k.OnCooldown)
{
<div class="cd-overlay">
<svg viewBox="0 0 100 100" class="cd-svg">
<circle cx="50" cy="50" r="45" class="cd-bg"/>
<circle cx="50" cy="50" r="45" class="cd-progress"
stroke-dasharray="282.74"
stroke-dashoffset="@(282.74 * (1 - k.CooldownFraction))"/>
</svg>
<span class="cd-text">@k.RemainingSeconds</span>
</div>
}
</div>
}
</div>
<div class="kb-row kb-offset-2">
@foreach (var k in Keys.Skip(11).Take(5))
{
<div class="kb-key @(k.OnCooldown ? "on-cd" : "")" @onclick="() => ActivateKey(k)" title="@k.Tooltip">
<span class="key-label">@k.Label</span>
@if (k.SkillName != null)
{
<span class="skill-name">@k.SkillName</span>
}
@if (k.OnCooldown)
{
<div class="cd-overlay">
<svg viewBox="0 0 100 100" class="cd-svg">
<circle cx="50" cy="50" r="45" class="cd-bg"/>
<circle cx="50" cy="50" r="45" class="cd-progress"
stroke-dasharray="282.74"
stroke-dashoffset="@(282.74 * (1 - k.CooldownFraction))"/>
</svg>
<span class="cd-text">@k.RemainingSeconds</span>
</div>
}
</div>
}
</div>
<div class="kb-row kb-offset-1">
@foreach (var k in Keys.Skip(16).Take(5))
{
<div class="kb-key @(k.OnCooldown ? "on-cd" : "")" @onclick="() => ActivateKey(k)" title="@k.Tooltip">
<span class="key-label">@k.Label</span>
@if (k.SkillName != null)
{
<span class="skill-name">@k.SkillName</span>
}
@if (k.OnCooldown)
{
<div class="cd-overlay">
<svg viewBox="0 0 100 100" class="cd-svg">
<circle cx="50" cy="50" r="45" class="cd-bg"/>
<circle cx="50" cy="50" r="45" class="cd-progress"
stroke-dasharray="282.74"
stroke-dashoffset="@(282.74 * (1 - k.CooldownFraction))"/>
</svg>
<span class="cd-text">@k.RemainingSeconds</span>
</div>
}
</div>
}
</div>
<div class="kb-row kb-space-row">
<div class="kb-key kb-space @(spaceKey.OnCooldown ? "on-cd" : "")" @onclick="() => ActivateKey(spaceKey)"
title="@spaceKey.Tooltip">
<span class="key-label">@spaceKey.Label</span>
@if (spaceKey.SkillName != null)
{
<span class="skill-name">@spaceKey.SkillName</span>
}
@if (spaceKey.OnCooldown)
{
<div class="cd-overlay">
<svg viewBox="0 0 100 100" class="cd-svg">
<circle cx="50" cy="50" r="45" class="cd-bg"/>
<circle cx="50" cy="50" r="45" class="cd-progress"
stroke-dasharray="282.74"
stroke-dashoffset="@(282.74 * (1 - spaceKey.CooldownFraction))"/>
</svg>
<span class="cd-text">@spaceKey.RemainingSeconds</span>
</div>
}
</div>
</div>
<div class="kb-divider">Shift + Key</div>
<div class="kb-row">
@foreach (var k in ShiftKeys.Take(6))
{
<div class="kb-key kb-shift @(k.OnCooldown ? "on-cd" : "")" @onclick="() => ActivateKey(k)"
title="@k.Tooltip">
<span class="key-label">@k.Label</span>
@if (k.SkillName != null)
{
<span class="skill-name">@k.SkillName</span>
}
@if (k.OnCooldown)
{
<div class="cd-overlay">
<svg viewBox="0 0 100 100" class="cd-svg">
<circle cx="50" cy="50" r="45" class="cd-bg"/>
<circle cx="50" cy="50" r="45" class="cd-progress"
stroke-dasharray="282.74"
stroke-dashoffset="@(282.74 * (1 - k.CooldownFraction))"/>
</svg>
<span class="cd-text">@k.RemainingSeconds</span>
</div>
}
</div>
}
</div>
<div class="kb-row kb-offset-1">
@foreach (var k in ShiftKeys.Skip(6).Take(5))
{
<div class="kb-key kb-shift @(k.OnCooldown ? "on-cd" : "")" @onclick="() => ActivateKey(k)"
title="@k.Tooltip">
<span class="key-label">@k.Label</span>
@if (k.SkillName != null)
{
<span class="skill-name">@k.SkillName</span>
}
@if (k.OnCooldown)
{
<div class="cd-overlay">
<svg viewBox="0 0 100 100" class="cd-svg">
<circle cx="50" cy="50" r="45" class="cd-bg"/>
<circle cx="50" cy="50" r="45" class="cd-progress"
stroke-dasharray="282.74"
stroke-dashoffset="@(282.74 * (1 - k.CooldownFraction))"/>
</svg>
<span class="cd-text">@k.RemainingSeconds</span>
</div>
}
</div>
}
</div>
<div class="kb-row kb-offset-2">
@foreach (var k in ShiftKeys.Skip(11).Take(5))
{
<div class="kb-key kb-shift @(k.OnCooldown ? "on-cd" : "")" @onclick="() => ActivateKey(k)"
title="@k.Tooltip">
<span class="key-label">@k.Label</span>
@if (k.SkillName != null)
{
<span class="skill-name">@k.SkillName</span>
}
@if (k.OnCooldown)
{
<div class="cd-overlay">
<svg viewBox="0 0 100 100" class="cd-svg">
<circle cx="50" cy="50" r="45" class="cd-bg"/>
<circle cx="50" cy="50" r="45" class="cd-progress"
stroke-dasharray="282.74"
stroke-dashoffset="@(282.74 * (1 - k.CooldownFraction))"/>
</svg>
<span class="cd-text">@k.RemainingSeconds</span>
</div>
}
</div>
}
</div>
<div class="kb-row kb-offset-1">
@foreach (var k in ShiftKeys.Skip(16).Take(5))
{
<div class="kb-key kb-shift @(k.OnCooldown ? "on-cd" : "")" @onclick="() => ActivateKey(k)"
title="@k.Tooltip">
<span class="key-label">@k.Label</span>
@if (k.SkillName != null)
{
<span class="skill-name">@k.SkillName</span>
}
@if (k.OnCooldown)
{
<div class="cd-overlay">
<svg viewBox="0 0 100 100" class="cd-svg">
<circle cx="50" cy="50" r="45" class="cd-bg"/>
<circle cx="50" cy="50" r="45" class="cd-progress"
stroke-dasharray="282.74"
stroke-dashoffset="@(282.74 * (1 - k.CooldownFraction))"/>
</svg>
<span class="cd-text">@k.RemainingSeconds</span>
</div>
}
</div>
}
</div>
<div class="kb-row kb-space-row">
<div class="kb-key kb-space kb-shift @(shiftSpaceKey.OnCooldown ? "on-cd" : "")"
@onclick="() => ActivateKey(shiftSpaceKey)" title="@shiftSpaceKey.Tooltip">
<span class="key-label">@shiftSpaceKey.Label</span>
@if (shiftSpaceKey.SkillName != null)
{
<span class="skill-name">@shiftSpaceKey.SkillName</span>
}
@if (shiftSpaceKey.OnCooldown)
{
<div class="cd-overlay">
<svg viewBox="0 0 100 100" class="cd-svg">
<circle cx="50" cy="50" r="45" class="cd-bg"/>
<circle cx="50" cy="50" r="45" class="cd-progress"
stroke-dasharray="282.74"
stroke-dashoffset="@(282.74 * (1 - shiftSpaceKey.CooldownFraction))"/>
</svg>
<span class="cd-text">@shiftSpaceKey.RemainingSeconds</span>
</div>
}
</div>
</div>
</div>
</div>
@code {
private ElementReference containerRef;
private List<KeyData> Keys { get; set; } = new();
private List<KeyData> ShiftKeys { get; set; } = new();
private KeyData spaceKey = null!;
private KeyData shiftSpaceKey = null!;
private string selectedCharacter = "Xavian";
private List<string> Characters = [];
public class KeyData
{
public string Id { get; set; } = "";
public string Label { get; set; } = "";
public string? SkillName { get; set; }
public string? Character { get; set; }
public double CooldownDuration { get; set; }
public double Remaining { get; set; }
public bool OnCooldown => Remaining > 0;
public double RemainingSeconds => Math.Ceiling(Remaining);
public double CooldownFraction => CooldownDuration > 0 ? Math.Min(1, Remaining / CooldownDuration) : 0;
public void Tick(double seconds)
{
if (Remaining > 0)
Remaining = Math.Max(0, Remaining - seconds);
}
public string Tooltip
{
get
{
var parts = new List<string>();
if (Character != null) parts.Add(Character);
if (SkillName != null) parts.Add(SkillName);
if (CooldownDuration > 0) parts.Add($"CD: {CooldownDuration}s");
return parts.Count > 0 ? string.Join(" - ", parts) : Label;
}
}
}
protected override void OnInitialized()
{
Characters = DocsData.All
.OfType<SkillDoc>()
.Where(s => s.Character != null)
.Select(s => s.Character!)
.Distinct()
.Order()
.ToList();
if (Characters.Count == 0) Characters.Add("Xavian");
RebuildKeyboardData();
}
private void RebuildKeyboardData()
{
var skillByKey = DocsData.All
.OfType<SkillDoc>()
.Where(s => !string.IsNullOrEmpty(s.Key) && s.Character == selectedCharacter)
.GroupBy(s => s.Key!)
.ToDictionary(g => g.Key, g =>
g.OrderBy(s => string.IsNullOrEmpty(s.CostSwiftReprieval) ? 0 : 1).First());
var actionByKey = DocsData.All
.OfType<KeyDoc>()
.Where(k => !string.IsNullOrEmpty(k.Action))
.Select(k => new
{
Key = Path.GetFileNameWithoutExtension(k.FileName),
k.Action
})
.GroupBy(x => x.Key)
.ToDictionary(g => g.Key, g => g.First().Action!, StringComparer.OrdinalIgnoreCase);
var keyLabels = new[] { "1", "2", "3", "4", "5", "6", "Q", "W", "E", "R", "T", "A", "S", "D", "F", "G", "Z", "X", "C", "V", "B" };
Keys = keyLabels.Select(label =>
{
var skill = skillByKey.GetValueOrDefault(label);
var action = actionByKey.GetValueOrDefault(label);
var skillName = skill != null ? Path.GetFileNameWithoutExtension(skill.FileName) : action;
return new KeyData
{
Id = label.ToLower(),
Label = label,
SkillName = skillName,
Character = skill?.Character,
CooldownDuration = ParseCooldown(skill?.Cooldown)
};
}).ToList();
var spaceSkill = skillByKey.GetValueOrDefault(" ");
var spaceAction = actionByKey.GetValueOrDefault("Space");
spaceKey = new KeyData
{
Id = "space",
Label = "Space",
SkillName = spaceSkill != null ? Path.GetFileNameWithoutExtension(spaceSkill.FileName) : spaceAction,
Character = spaceSkill?.Character,
CooldownDuration = ParseCooldown(spaceSkill?.Cooldown)
};
ShiftKeys = keyLabels.Select(label =>
{
var shiftKeyStr = "Shift + " + label;
var skill = skillByKey.GetValueOrDefault(shiftKeyStr);
var action = actionByKey.GetValueOrDefault(shiftKeyStr);
var skillName = skill != null ? Path.GetFileNameWithoutExtension(skill.FileName) : action;
return new KeyData
{
Id = "shift-" + label.ToLower(),
Label = "Shift+" + label,
SkillName = skillName,
Character = skill?.Character,
CooldownDuration = ParseCooldown(skill?.Cooldown)
};
}).ToList();
var shiftSpaceSkill = skillByKey.GetValueOrDefault("Shift + Space");
var shiftSpaceAction = actionByKey.GetValueOrDefault("Shift + Space");
shiftSpaceKey = new KeyData
{
Id = "shift-space",
Label = "Shift+Space",
SkillName = shiftSpaceSkill != null ? Path.GetFileNameWithoutExtension(shiftSpaceSkill.FileName) : shiftSpaceAction,
Character = shiftSpaceSkill?.Character,
CooldownDuration = ParseCooldown(shiftSpaceSkill?.Cooldown)
};
}
private static double ParseCooldown(string? cooldown)
{
if (string.IsNullOrEmpty(cooldown)) return 0;
if (double.TryParse(cooldown, out var result)) return result;
return 0;
}
protected override async Task OnAfterRenderAsync(bool firstRender)
{
if (firstRender)
{
await containerRef.FocusAsync();
}
}
private void HandleKeyDown(KeyboardEventArgs e)
{
var isShift = e.ShiftKey;
var key = e.Key.ToLower();
var allKeys = new List<KeyData>();
allKeys.AddRange(Keys);
allKeys.Add(spaceKey);
allKeys.AddRange(ShiftKeys);
allKeys.Add(shiftSpaceKey);
KeyData? target;
if (isShift)
{
if (key == "shift") return;
var lookup = "shift-" + (key == " " ? "space" : key);
target = allKeys.FirstOrDefault(k => k.Id == lookup);
}
else
{
var lookup = key == " " ? "space" : key;
target = allKeys.FirstOrDefault(k => k.Id == lookup);
}
if (target != null)
{
ActivateKey(target);
}
}
private void ActivateKey(KeyData key)
{
if (key.OnCooldown) return;
if (key.Character == null) return;
var duration = key.CooldownDuration > 0 ? key.CooldownDuration : 0.5;
key.Remaining = duration;
var allKeys = new List<KeyData>();
allKeys.AddRange(Keys);
allKeys.Add(spaceKey);
allKeys.AddRange(ShiftKeys);
allKeys.Add(shiftSpaceKey);
foreach (var k in allKeys)
{
k.Tick(1.5);
}
}
}