stylecop has joined the server
This commit is contained in:
parent
514f70ddb2
commit
31894129c7
8 changed files with 262 additions and 97 deletions
|
@ -1,4 +1,4 @@
|
|||
[*]
|
||||
[*]
|
||||
guidelines = 80
|
||||
end_of_line = lf
|
||||
insert_final_newline = true
|
||||
|
@ -6,3 +6,9 @@ charset = utf-8-bom
|
|||
|
||||
[*.cs]
|
||||
guidelines = 80, 120
|
||||
|
||||
# IDE0021: Use block body for constructors
|
||||
csharp_style_expression_bodied_constructors = when_on_single_line
|
||||
|
||||
# IDE0024: Use block body for operators
|
||||
csharp_style_expression_bodied_operators = when_on_single_line
|
||||
|
|
25
Directory.build.props
Normal file
25
Directory.build.props
Normal file
|
@ -0,0 +1,25 @@
|
|||
<Project>
|
||||
<PropertyGroup>
|
||||
<LangVersion>10.0</LangVersion>
|
||||
<Nullable>enable</Nullable>
|
||||
</PropertyGroup>
|
||||
<!-- StyleCop Analyzers configuration -->
|
||||
<PropertyGroup>
|
||||
<CodeAnalysisRuleSet>$(SolutionDir)StyleCop.ruleset</CodeAnalysisRuleSet>
|
||||
</PropertyGroup>
|
||||
<ItemGroup>
|
||||
<PackageReference Include="StyleCop.Analyzers" Version="1.1.118">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
</PackageReference>
|
||||
<PackageReference Include="StyleCop.Analyzers.Error" Version="1.0.2">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
</PackageReference>
|
||||
<PackageReference Include="StyleCop.CSharp.Async.Rules" Version="6.1.41">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
</PackageReference>
|
||||
<AdditionalFiles Include="$(SolutionDir)stylecop.json" Link="stylecop.json" />
|
||||
</ItemGroup>
|
||||
</Project>
|
26
GlobalSuppressions.cs
Normal file
26
GlobalSuppressions.cs
Normal file
|
@ -0,0 +1,26 @@
|
|||
// <copyright file="GlobalSuppressions.cs" company="alterNERDtive">
|
||||
// Copyright 2020–2022 alterNERDtive.
|
||||
//
|
||||
// This file is part of bindED VoiceAttack plugin.
|
||||
//
|
||||
// bindED VoiceAttack plugin is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// bindED VoiceAttack plugin is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with bindED VoiceAttack plugin. If not, see <https://www.gnu.org/licenses/>.
|
||||
// </copyright>
|
||||
|
||||
// This file is used by Code Analysis to maintain SuppressMessage
|
||||
// attributes that are applied to this project.
|
||||
// Project-level suppressions either have no target or are given
|
||||
// a specific target and scoped to a namespace, type, member, etc.
|
||||
using System.Diagnostics.CodeAnalysis;
|
||||
|
||||
[assembly: SuppressMessage("StyleCop.CSharp.NamingRules", "SA1300:Element should begin with upper-case letter", Justification = "just cause", Scope = "namespace", Target = "~N:bindEDplugin")]
|
0
StyleCop.ruleset
Normal file
0
StyleCop.ruleset
Normal file
267
bindED.cs
267
bindED.cs
|
@ -1,4 +1,23 @@
|
|||
#nullable enable
|
||||
// <copyright file="bindED.cs" company="alterNERDtive">
|
||||
// Copyright 2020–2022 alterNERDtive.
|
||||
//
|
||||
// This file is part of bindED VoiceAttack plugin.
|
||||
//
|
||||
// bindED VoiceAttack plugin is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// bindED VoiceAttack plugin is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with bindED VoiceAttack plugin. If not, see <https://www.gnu.org/licenses/>.
|
||||
// </copyright>
|
||||
|
||||
#nullable enable
|
||||
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
|
@ -10,102 +29,132 @@ using System.Xml.Linq;
|
|||
|
||||
namespace bindEDplugin
|
||||
{
|
||||
/// <summary>
|
||||
/// This VoiceAttack plugin reads Elite Dangerous .binds files for keyboard
|
||||
/// bindings and makes them available in VoiceAttack variables.
|
||||
/// </summary>
|
||||
[System.Diagnostics.CodeAnalysis.SuppressMessage("StyleCop.CSharp.NamingRules", "SA1300:Element should begin with upper-case letter", Justification = "historic, grandfathered in")]
|
||||
[System.Diagnostics.CodeAnalysis.SuppressMessage("StyleCop.CSharp.DocumentationRules", "SA1649:File name should match first type name", Justification = "historic, grandfathered in")]
|
||||
[System.Diagnostics.CodeAnalysis.SuppressMessage("Style", "IDE1006:Naming Styles", Justification = "historic, grandfathered in")]
|
||||
public class bindEDPlugin
|
||||
{
|
||||
private static dynamic? _VA;
|
||||
private static string? _pluginPath;
|
||||
private static readonly string _bindingsDir = Path.Combine(Environment.GetFolderPath(
|
||||
private static readonly string BindingsDir = Path.Combine(
|
||||
Environment.GetFolderPath(
|
||||
Environment.SpecialFolder.LocalApplicationData),
|
||||
@"Frontier Developments\Elite Dangerous\Options\Bindings"
|
||||
);
|
||||
private static readonly Dictionary<string, int> _fileEventCount = new Dictionary<string, int>();
|
||||
@"Frontier Developments\Elite Dangerous\Options\Bindings");
|
||||
|
||||
private static readonly Dictionary<string, int> FileEventCount = new ();
|
||||
|
||||
[System.Diagnostics.CodeAnalysis.SuppressMessage("StyleCop.CSharp.NamingRules", "SA1306:Field names should begin with lower-case letter", Justification = "just cause")]
|
||||
private static dynamic? VA;
|
||||
private static string? pluginPath;
|
||||
private static FileSystemWatcher? bindsWatcher;
|
||||
private static FileSystemWatcher? mapWatcher;
|
||||
private static string? layout;
|
||||
private static Dictionary<string, int>? keyMap;
|
||||
private static string? preset;
|
||||
private static Dictionary<string, List<string>>? binds;
|
||||
|
||||
private static FileSystemWatcher BindsWatcher
|
||||
{
|
||||
get
|
||||
{
|
||||
if (_bindsWatcher == null)
|
||||
if (bindsWatcher == null)
|
||||
{
|
||||
_bindsWatcher = new FileSystemWatcher(_bindingsDir);
|
||||
_bindsWatcher.NotifyFilter = NotifyFilters.LastWrite | NotifyFilters.FileName;
|
||||
_bindsWatcher.Changed += (source, EventArgs) => { FileChangedHandler(EventArgs.Name); };
|
||||
_bindsWatcher.Created += (source, EventArgs) => { FileChangedHandler(EventArgs.Name); };
|
||||
_bindsWatcher.Renamed += (source, EventArgs) => { FileChangedHandler(EventArgs.Name); };
|
||||
bindsWatcher = new FileSystemWatcher(BindingsDir);
|
||||
bindsWatcher.NotifyFilter = NotifyFilters.LastWrite | NotifyFilters.FileName;
|
||||
bindsWatcher.Changed += (source, eventArgs) => { FileChangedHandler(eventArgs.Name); };
|
||||
bindsWatcher.Created += (source, eventArgs) => { FileChangedHandler(eventArgs.Name); };
|
||||
bindsWatcher.Renamed += (source, eventArgs) => { FileChangedHandler(eventArgs.Name); };
|
||||
}
|
||||
return _bindsWatcher!;
|
||||
|
||||
return bindsWatcher!;
|
||||
}
|
||||
}
|
||||
private static FileSystemWatcher? _bindsWatcher;
|
||||
|
||||
private static FileSystemWatcher MapWatcher
|
||||
{
|
||||
get
|
||||
{
|
||||
if (_mapWatcher == null)
|
||||
if (mapWatcher == null)
|
||||
{
|
||||
_mapWatcher = new FileSystemWatcher(_pluginPath);
|
||||
_mapWatcher.NotifyFilter = NotifyFilters.LastWrite | NotifyFilters.FileName;
|
||||
_mapWatcher.Changed += (source, EventArgs) => { FileChangedHandler(EventArgs.Name); };
|
||||
_mapWatcher.Created += (source, EventArgs) => { FileChangedHandler(EventArgs.Name); };
|
||||
_mapWatcher.Renamed += (source, EventArgs) => { FileChangedHandler(EventArgs.Name); };
|
||||
mapWatcher = new FileSystemWatcher(pluginPath);
|
||||
mapWatcher.NotifyFilter = NotifyFilters.LastWrite | NotifyFilters.FileName;
|
||||
mapWatcher.Changed += (source, eventArgs) => { FileChangedHandler(eventArgs.Name); };
|
||||
mapWatcher.Created += (source, eventArgs) => { FileChangedHandler(eventArgs.Name); };
|
||||
mapWatcher.Renamed += (source, eventArgs) => { FileChangedHandler(eventArgs.Name); };
|
||||
}
|
||||
return _mapWatcher!;
|
||||
|
||||
return mapWatcher!;
|
||||
}
|
||||
}
|
||||
private static FileSystemWatcher? _mapWatcher;
|
||||
|
||||
private static string? Layout
|
||||
{
|
||||
get => _layout ??= _VA?.GetText("bindED.layout#") ?? "en-us";
|
||||
get => layout ??= VA?.GetText("bindED.layout#") ?? "en-us";
|
||||
set
|
||||
{
|
||||
_layout = value;
|
||||
layout = value;
|
||||
KeyMap = null;
|
||||
}
|
||||
}
|
||||
private static string? _layout;
|
||||
|
||||
private static Dictionary<string, int>? KeyMap
|
||||
{
|
||||
get => _keyMap ??= LoadKeyMap(Layout!);
|
||||
set => _keyMap = value;
|
||||
get => keyMap ??= LoadKeyMap(Layout!);
|
||||
set => keyMap = value;
|
||||
}
|
||||
private static Dictionary<string, int>? _keyMap;
|
||||
|
||||
private static string? Preset
|
||||
{
|
||||
get => _preset ??= DetectPreset();
|
||||
get => preset ??= DetectPreset();
|
||||
set
|
||||
{
|
||||
_preset = value;
|
||||
preset = value;
|
||||
Binds = null;
|
||||
}
|
||||
}
|
||||
private static string? _preset;
|
||||
|
||||
private static Dictionary<string, List<string>>? Binds
|
||||
{
|
||||
get => _binds ??= ReadBinds(DetectBindsFile(Preset!));
|
||||
set => _binds = value;
|
||||
get => binds ??= ReadBinds(DetectBindsFile(Preset!));
|
||||
set => binds = value;
|
||||
}
|
||||
private static Dictionary<string, List<string>>? _binds;
|
||||
|
||||
public static string VERSION = "4.2.1";
|
||||
[System.Diagnostics.CodeAnalysis.SuppressMessage("StyleCop.CSharp.OrderingRules", "SA1201:Elements should appear in the correct order", Justification = "nicer grouping")]
|
||||
private static readonly Version VERSION = new ("4.2.1");
|
||||
|
||||
/// <summary>
|
||||
/// The plugin’s display name, as required by the VoiceAttack plugin API.
|
||||
/// </summary>
|
||||
/// <returns>The display name.</returns>
|
||||
public static string VA_DisplayName() => $"bindED Plugin v{VERSION}-alterNERDtive";
|
||||
|
||||
/// <summary>
|
||||
/// The plugin’s description, as required by the VoiceAttack plugin API.
|
||||
/// </summary>
|
||||
/// <returns>The description.</returns>
|
||||
public static string VA_DisplayInfo() => "bindED Plugin\r\n\r\n2016 VoiceAttack.com\r\n2020–2021 alterNERDtive";
|
||||
|
||||
public static Guid VA_Id() => new Guid("{524B4B9A-3965-4045-A39A-A239BF6E2838}");
|
||||
/// <summary>
|
||||
/// The plugin’s GUID, as required by the VoiceAttack plugin API.
|
||||
/// </summary>
|
||||
/// <returns>The GUID.</returns>
|
||||
public static Guid VA_Id() => new ("{524B4B9A-3965-4045-A39A-A239BF6E2838}");
|
||||
|
||||
/// <summary>
|
||||
/// The Init method, as required by the VoiceAttack plugin API.
|
||||
/// Runs when the plugin is initially loaded.
|
||||
/// </summary>
|
||||
/// <param name="vaProxy">The VoiceAttack proxy object.</param>
|
||||
public static void VA_Init1(dynamic vaProxy)
|
||||
{
|
||||
_VA = vaProxy;
|
||||
_VA.TextVariableChanged += new Action<string, string, string, Guid?>(TextVariableChanged);
|
||||
_pluginPath = Path.GetDirectoryName(_VA.PluginPath());
|
||||
VA = vaProxy;
|
||||
VA.TextVariableChanged += new Action<string, string, string, Guid?>(TextVariableChanged);
|
||||
pluginPath = Path.GetDirectoryName(VA.PluginPath());
|
||||
|
||||
_VA.SetText("bindED.version", VERSION);
|
||||
_VA.SetText("bindED.fork", "alterNERDtive");
|
||||
VA.SetText("bindED.version", VERSION);
|
||||
VA.SetText("bindED.fork", "alterNERDtive");
|
||||
|
||||
try
|
||||
{
|
||||
|
@ -122,32 +171,38 @@ namespace bindEDplugin
|
|||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// The Invoke method, as required by the VoiceAttack plugin API.
|
||||
/// Runs whenever a plugin context is invoked.
|
||||
/// </summary>
|
||||
/// <param name="vaProxy">The VoiceAttack proxy object.</param>
|
||||
public static void VA_Invoke1(dynamic vaProxy)
|
||||
{
|
||||
_VA = vaProxy;
|
||||
VA = vaProxy;
|
||||
try
|
||||
{
|
||||
string context = _VA.Context.ToLower();
|
||||
string context = VA.Context.ToLower();
|
||||
if (context == "diagnostics")
|
||||
{
|
||||
LogInfo($"current keybord layout: {Layout}");
|
||||
LogInfo($"current preset: {Preset}");
|
||||
LogInfo($"detected binds file: {(new FileInfo(DetectBindsFile(Preset!))).Name}");
|
||||
LogInfo($"detected binds file: {new FileInfo(DetectBindsFile(Preset!)).Name}");
|
||||
LogInfo($"detected binds file (full path): {DetectBindsFile(Preset!)}");
|
||||
}
|
||||
else if (context == "listbinds")
|
||||
{
|
||||
ListBinds(Binds, _VA.GetText("bindED.separator") ?? "\r\n");
|
||||
ListBinds(Binds, VA.GetText("bindED.separator") ?? "\r\n");
|
||||
}
|
||||
else if (context == "loadbinds")
|
||||
{
|
||||
// force reset everything
|
||||
Layout = null;
|
||||
Preset = null;
|
||||
if (!String.IsNullOrWhiteSpace(_VA.GetText("~bindsFile")))
|
||||
if (!string.IsNullOrWhiteSpace(VA.GetText("~bindsFile")))
|
||||
{
|
||||
Binds = ReadBinds(Path.Combine(_bindingsDir, _VA.GetText("~bindsFile")));
|
||||
Binds = ReadBinds(Path.Combine(BindingsDir, VA.GetText("~bindsFile")));
|
||||
}
|
||||
|
||||
LoadBinds(Binds);
|
||||
}
|
||||
else if (context == "missingbinds")
|
||||
|
@ -156,7 +211,7 @@ namespace bindEDplugin
|
|||
}
|
||||
else
|
||||
{
|
||||
if (String.IsNullOrWhiteSpace(context))
|
||||
if (string.IsNullOrWhiteSpace(context))
|
||||
{
|
||||
LogError("Empty plugin context.");
|
||||
}
|
||||
|
@ -164,6 +219,7 @@ namespace bindEDplugin
|
|||
{
|
||||
LogError($"Invalid plugin context '{context}'.");
|
||||
}
|
||||
|
||||
LogError("You generally do not need to invoke the plugin manually.");
|
||||
}
|
||||
}
|
||||
|
@ -173,11 +229,26 @@ namespace bindEDplugin
|
|||
}
|
||||
}
|
||||
|
||||
public static void VA_StopCommand() { }
|
||||
/// <summary>
|
||||
/// The Exit method, as required by the VoiceAttack plugin API.
|
||||
/// Runs when VoiceAttack is shut down.
|
||||
/// </summary>
|
||||
/// <param name="vaProxy">The VoiceAttack proxy object.</param>
|
||||
[System.Diagnostics.CodeAnalysis.SuppressMessage("Style", "IDE0060:Remove unused parameter", Justification = "required by VoiceAttack plugin API")]
|
||||
public static void VA_Exit1(dynamic vaProxy)
|
||||
{
|
||||
}
|
||||
|
||||
public static void VA_Exit1(dynamic vaProxy) { }
|
||||
/// <summary>
|
||||
/// The StopCommand method, as required by the VoiceAttack plugin API.
|
||||
/// Runs whenever all commands are stopped using the “Stop All Commands”
|
||||
/// button or action.
|
||||
/// </summary>
|
||||
public static void VA_StopCommand()
|
||||
{
|
||||
}
|
||||
|
||||
public static void TextVariableChanged(string name, string from, string to, Guid? internalID)
|
||||
private static void TextVariableChanged(string name, string from, string to, Guid? internalID = null)
|
||||
{
|
||||
if (name == "bindED.layout#")
|
||||
{
|
||||
|
@ -196,22 +267,22 @@ namespace bindEDplugin
|
|||
|
||||
private static void LogError(string message)
|
||||
{
|
||||
_VA!.WriteToLog($"ERROR | bindED: {message}", "red");
|
||||
VA!.WriteToLog($"ERROR | bindED: {message}", "red");
|
||||
}
|
||||
|
||||
private static void LogInfo(string message)
|
||||
{
|
||||
_VA!.WriteToLog($"INFO | bindED: {message}", "blue");
|
||||
VA!.WriteToLog($"INFO | bindED: {message}", "blue");
|
||||
}
|
||||
|
||||
private static void LogWarn(string message)
|
||||
{
|
||||
_VA!.WriteToLog($"WARN | bindED: {message}", "yellow");
|
||||
VA!.WriteToLog($"WARN | bindED: {message}", "yellow");
|
||||
}
|
||||
|
||||
public static void ListBinds(Dictionary<string, List<string>>? binds, string separator)
|
||||
private static void ListBinds(Dictionary<string, List<string>>? binds, string separator)
|
||||
{
|
||||
_VA!.SetText("~bindED.bindsList", string.Join(separator, binds!.Keys));
|
||||
VA!.SetText("~bindED.bindsList", string.Join(separator, binds!.Keys));
|
||||
LogInfo("List of Elite binds saved to TXT variable '~bindED.bindsList'.");
|
||||
}
|
||||
|
||||
|
@ -223,7 +294,7 @@ namespace bindEDplugin
|
|||
bool valid = true;
|
||||
if (bind.Value.Count == 0)
|
||||
{
|
||||
//LogInfo($"No keyboard bind for '{bind.Key}' found, skipping …");
|
||||
// LogInfo($"No keyboard bind for '{bind.Key}' found, skipping …");
|
||||
}
|
||||
else
|
||||
{
|
||||
|
@ -239,31 +310,32 @@ namespace bindEDplugin
|
|||
LogError($"No valid key code for '{key}' found, skipping bind for '{bind.Key}' …");
|
||||
}
|
||||
}
|
||||
|
||||
if (valid)
|
||||
{
|
||||
_VA!.SetText(bind.Key, value);
|
||||
VA!.SetText(bind.Key, value);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
LogInfo($"Elite binds '{(String.IsNullOrWhiteSpace(_VA!.GetText("~bindsFile")) ? Preset : _VA!.GetText("~bindsFile"))}' for layout '{Layout}' loaded successfully.");
|
||||
LogInfo($"Elite binds '{(string.IsNullOrWhiteSpace(VA!.GetText("~bindsFile")) ? Preset : VA!.GetText("~bindsFile"))}' for layout '{Layout}' loaded successfully.");
|
||||
}
|
||||
|
||||
private static void MissingBinds(Dictionary<string, List<string>>? binds)
|
||||
{
|
||||
List<string> missing = new List<string>(256);
|
||||
List<string> missing = new (256);
|
||||
foreach (KeyValuePair<string, List<string>> bind in binds!)
|
||||
{
|
||||
|
||||
if (bind.Value.Count == 0)
|
||||
{
|
||||
missing.Add(bind.Key);
|
||||
}
|
||||
}
|
||||
|
||||
if (missing.Count > 0)
|
||||
{
|
||||
_VA!.SetText("~bindED.missingBinds", string.Join("\r\n", missing));
|
||||
_VA!.SetBoolean("~bindED.missingBinds", true);
|
||||
VA!.SetText("~bindED.missingBinds", string.Join("\r\n", missing));
|
||||
VA!.SetBoolean("~bindED.missingBinds", true);
|
||||
LogInfo("List of missing Elite binds saved to TXT variable '~bindED.missingBinds'.");
|
||||
}
|
||||
else
|
||||
|
@ -272,40 +344,45 @@ namespace bindEDplugin
|
|||
}
|
||||
}
|
||||
|
||||
private static Dictionary<String, int> LoadKeyMap(string layout)
|
||||
private static Dictionary<string, int> LoadKeyMap(string layout)
|
||||
{
|
||||
string mapFile = Path.Combine(_pluginPath, $"EDMap-{layout.ToLower()}.txt");
|
||||
string mapFile = Path.Combine(pluginPath, $"EDMap-{layout.ToLower()}.txt");
|
||||
if (!File.Exists(mapFile))
|
||||
{
|
||||
throw new FileNotFoundException($"No map file for layout '{layout}' found.");
|
||||
}
|
||||
Dictionary<string, int> map = new Dictionary<string, int>(256);
|
||||
foreach (String line in File.ReadAllLines(mapFile, System.Text.Encoding.UTF8))
|
||||
|
||||
Dictionary<string, int> map = new (256);
|
||||
foreach (string line in File.ReadAllLines(mapFile, System.Text.Encoding.UTF8))
|
||||
{
|
||||
String[] arItem = line.Split(";".ToCharArray(), 2, StringSplitOptions.RemoveEmptyEntries);
|
||||
if ((arItem.Count() == 2) && (!String.IsNullOrWhiteSpace(arItem[0])) && (!map.ContainsKey(arItem[0])))
|
||||
string[] arItem = line.Split(";".ToCharArray(), 2, StringSplitOptions.RemoveEmptyEntries);
|
||||
if ((arItem.Count() == 2) && (!string.IsNullOrWhiteSpace(arItem[0])) && (!map.ContainsKey(arItem[0])))
|
||||
{
|
||||
ushort iKey;
|
||||
if (ushort.TryParse(arItem[1], out iKey))
|
||||
{
|
||||
if (iKey > 0 && iKey < 256)
|
||||
{
|
||||
map.Add(arItem[0].Trim(), iKey);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (map.Count == 0)
|
||||
{
|
||||
throw new Exception($"Map file for {layout} does not contain any elements.");
|
||||
}
|
||||
|
||||
return map;
|
||||
}
|
||||
|
||||
private static string DetectPreset()
|
||||
{
|
||||
string startFile = Path.Combine(_bindingsDir, "StartPreset.4.start");
|
||||
string startFile = Path.Combine(BindingsDir, "StartPreset.4.start");
|
||||
if (!File.Exists(startFile))
|
||||
{
|
||||
startFile = Path.Combine(_bindingsDir, "StartPreset.start");
|
||||
startFile = Path.Combine(BindingsDir, "StartPreset.start");
|
||||
if (!File.Exists(startFile))
|
||||
{
|
||||
throw new FileNotFoundException("No 'StartPreset.start' file found. Please run Elite: Dangerous at least once, then restart VoiceAttack.");
|
||||
|
@ -315,8 +392,8 @@ namespace bindEDplugin
|
|||
IEnumerable<string> presets = File.ReadAllLines(startFile).Distinct();
|
||||
if (presets.Count() > 1)
|
||||
{
|
||||
LogError($"You have selected multiple control presets ('{ String.Join("', '", presets) }'). "
|
||||
+ $"Only binds from '{ presets.First() }' will be used. Please refer to the documentation for more information.");
|
||||
LogError($"You have selected multiple control presets ('{string.Join("', '", presets)}'). "
|
||||
+ $"Only binds from '{presets.First()}' will be used. Please refer to the documentation for more information.");
|
||||
}
|
||||
|
||||
return presets.First();
|
||||
|
@ -324,7 +401,7 @@ namespace bindEDplugin
|
|||
|
||||
private static string DetectBindsFile(string preset)
|
||||
{
|
||||
DirectoryInfo dirInfo = new DirectoryInfo(_bindingsDir);
|
||||
DirectoryInfo dirInfo = new (BindingsDir);
|
||||
FileInfo[] bindFiles = dirInfo.GetFiles()
|
||||
.Where(i => Regex.Match(i.Name, $@"^{Regex.Escape(preset)}\.[34]\.0\.binds$").Success)
|
||||
.OrderByDescending(p => p.Name).ToArray();
|
||||
|
@ -347,61 +424,70 @@ namespace bindEDplugin
|
|||
|
||||
rootElement = XElement.Load(file);
|
||||
|
||||
Dictionary<string, List<string>> binds = new Dictionary<string, List<string>>(512);
|
||||
Dictionary<string, List<string>> binds = new (512);
|
||||
if (rootElement != null)
|
||||
{
|
||||
foreach (XElement c in rootElement.Elements().Where(i => i.Elements().Count() > 0))
|
||||
{
|
||||
List<string> keys = new List<string>();
|
||||
List<string> keys = new ();
|
||||
foreach (var element in c.Elements().Where(i => i.HasAttributes))
|
||||
{
|
||||
if (element.Name == "Primary")
|
||||
{
|
||||
if (element.Attribute("Device").Value == "Keyboard" && !String.IsNullOrWhiteSpace(element.Attribute("Key").Value) && element.Attribute("Key").Value.StartsWith("Key_"))
|
||||
if (element.Attribute("Device").Value == "Keyboard"
|
||||
&& !string.IsNullOrWhiteSpace(element.Attribute("Key").Value) && element.Attribute("Key").Value.StartsWith("Key_"))
|
||||
{
|
||||
foreach (var modifier in element.Elements().Where(i => i.Name.LocalName == "Modifier"))
|
||||
{
|
||||
keys.Add(modifier.Attribute("Key").Value);
|
||||
}
|
||||
|
||||
keys.Add(element.Attribute("Key").Value);
|
||||
}
|
||||
}
|
||||
if (keys.Count == 0 && element.Name == "Secondary") //nothing found in primary... look in secondary
|
||||
{
|
||||
if (element.Attribute("Device").Value == "Keyboard" && !String.IsNullOrWhiteSpace(element.Attribute("Key").Value) && element.Attribute("Key").Value.StartsWith("Key_"))
|
||||
|
||||
if (keys.Count == 0 && element.Name == "Secondary")
|
||||
{ // nothing found in primary... look in secondary
|
||||
if (element.Attribute("Device").Value == "Keyboard"
|
||||
&& !string.IsNullOrWhiteSpace(element.Attribute("Key").Value) && element.Attribute("Key").Value.StartsWith("Key_"))
|
||||
{
|
||||
foreach (var modifier in element.Elements().Where(i => i.Name.LocalName == "Modifier"))
|
||||
{
|
||||
keys.Add(modifier.Attribute("Key").Value);
|
||||
}
|
||||
|
||||
keys.Add(element.Attribute("Key").Value);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
binds.Add($"ed{c.Name.LocalName}", keys);
|
||||
}
|
||||
}
|
||||
|
||||
return binds;
|
||||
}
|
||||
|
||||
private static void FileChangedHandler(string name)
|
||||
{
|
||||
// so apparently these events all fire twice … let’s make sure we only handle it once.
|
||||
if (_fileEventCount.ContainsKey(name))
|
||||
if (FileEventCount.ContainsKey(name))
|
||||
{
|
||||
_fileEventCount[name] += 1;
|
||||
FileEventCount[name] += 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
_fileEventCount.Add(name, 1);
|
||||
FileEventCount.Add(name, 1);
|
||||
}
|
||||
if (_fileEventCount[name] % 2 == 0)
|
||||
|
||||
if (FileEventCount[name] % 2 == 0)
|
||||
{
|
||||
try
|
||||
{
|
||||
// let’s make semi-sure that the file isn’t locked …
|
||||
// FIXXME: solve this properly
|
||||
Thread.Sleep(500);
|
||||
|
||||
// Going by name only is a bit naïve given we’re watching 2
|
||||
// separate directories, but hey … worst case if something
|
||||
// is doing unintended things is unnecessarily reloading the
|
||||
|
@ -423,15 +509,14 @@ namespace bindEDplugin
|
|||
LogInfo($"Bindings file '{name}' has changed, reloading …");
|
||||
Binds = null;
|
||||
LoadBinds(Binds);
|
||||
|
||||
|
||||
// copy Odyssey -> Horizons
|
||||
if (name == $"{Preset}.4.0.binds" && !_VA!.GetBoolean("bindED.disableHorizonsSync#"))
|
||||
if (name == $"{Preset}.4.0.binds" && !VA!.GetBoolean("bindED.disableHorizonsSync#"))
|
||||
{
|
||||
File.WriteAllText(
|
||||
Path.Combine(_bindingsDir, $"{Preset}.3.0.binds"),
|
||||
File.ReadAllText(Path.Combine(_bindingsDir, name))
|
||||
.Replace("MajorVersion=\"4\" MinorVersion=\"0\">", "MajorVersion=\"3\" MinorVersion=\"0\">")
|
||||
);
|
||||
Path.Combine(BindingsDir, $"{Preset}.3.0.binds"),
|
||||
File.ReadAllText(Path.Combine(BindingsDir, name))
|
||||
.Replace("MajorVersion=\"4\" MinorVersion=\"0\">", "MajorVersion=\"3\" MinorVersion=\"0\">"));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -21,7 +21,7 @@
|
|||
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
||||
<ErrorReport>prompt</ErrorReport>
|
||||
<WarningLevel>4</WarningLevel>
|
||||
<LangVersion>8.0</LangVersion>
|
||||
<CodeAnalysisRuleSet />
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
|
||||
<DebugType>pdbonly</DebugType>
|
||||
|
@ -30,7 +30,7 @@
|
|||
<DefineConstants>TRACE</DefineConstants>
|
||||
<ErrorReport>prompt</ErrorReport>
|
||||
<WarningLevel>4</WarningLevel>
|
||||
<LangVersion>8.0</LangVersion>
|
||||
<CodeAnalysisRuleSet />
|
||||
</PropertyGroup>
|
||||
<ItemGroup>
|
||||
<Reference Include="Microsoft.CSharp" />
|
||||
|
@ -40,6 +40,7 @@
|
|||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<Compile Include="bindED.cs" />
|
||||
<Compile Include="GlobalSuppressions.cs" />
|
||||
<Compile Include="Properties\AssemblyInfo.cs" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
|
@ -99,4 +100,4 @@
|
|||
<Target Name="AfterBuild">
|
||||
</Target>
|
||||
-->
|
||||
</Project>
|
||||
</Project>
|
||||
|
|
|
@ -1,11 +1,17 @@
|
|||
|
||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||
# Visual Studio Version 16
|
||||
VisualStudioVersion = 16.0.30517.126
|
||||
# Visual Studio Version 17
|
||||
VisualStudioVersion = 17.3.32519.111
|
||||
MinimumVisualStudioVersion = 10.0.40219.1
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "bindEDplugin", "bindEDplugin.csproj", "{C8AC9134-639D-45D2-B5EF-138E0550E0C9}"
|
||||
EndProject
|
||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{FC89314B-E504-4D5D-BB48-F1E3E4980A13}"
|
||||
ProjectSection(SolutionItems) = preProject
|
||||
.editorconfig = .editorconfig
|
||||
Directory.build.props = Directory.build.props
|
||||
GlobalSuppressions.cs = GlobalSuppressions.cs
|
||||
stylecop.json = stylecop.json
|
||||
EndProjectSection
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
|
|
16
stylecop.json
Normal file
16
stylecop.json
Normal file
|
@ -0,0 +1,16 @@
|
|||
{
|
||||
"$schema": "https://raw.githubusercontent.com/DotNetAnalyzers/StyleCopAnalyzers/master/StyleCop.Analyzers/StyleCop.Analyzers/Settings/stylecop.schema.json",
|
||||
"settings": {
|
||||
"orderingRules": {
|
||||
"usingDirectivesPlacement": "outsideNamespace"
|
||||
},
|
||||
"documentationRules": {
|
||||
"companyName": "alterNERDtive",
|
||||
"copyrightText": "Copyright {year} {companyName}.\n\nThis file is part of {application}.\n\n{application} is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\n{application} is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with {application}. If not, see <https://www.gnu.org/licenses/>.",
|
||||
"variables": {
|
||||
"application": "bindED VoiceAttack plugin",
|
||||
"year": "2020–2022"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue