461 lines
18 KiB
Plaintext
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);
|
|
}
|
|
}
|
|
|
|
}
|