Skip to content

Commit c0686d8

Browse files
committed
feat: 完善版本与插件项目模板
1 parent a6e1661 commit c0686d8

File tree

12 files changed

+334
-35
lines changed

12 files changed

+334
-35
lines changed

src/CurvaLauncher.Common/CurvaLauncher.Common.csproj

+14
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
<Project Sdk="Microsoft.NET.Sdk">
22

3+
<Import Project="../CurvaLauncher.props" />
4+
35
<PropertyGroup>
46
<TargetFramework>net8.0-windows</TargetFramework>
57
<ImplicitUsings>enable</ImplicitUsings>
@@ -11,6 +13,18 @@
1113
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
1214
<RootNamespace>CurvaLauncher</RootNamespace>
1315
<EnableWindowsTargeting>true</EnableWindowsTargeting>
16+
<Description>Common library of CurvaLauncher</Description>
17+
<RepositoryType>git</RepositoryType>
18+
<RepositoryUrl>https://github.com/OrgEleCho/CurvaLauncher</RepositoryUrl>
19+
<PackageIcon>Icon128.png</PackageIcon>
20+
<PackageLicenseExpression>MIT</PackageLicenseExpression>
1421
</PropertyGroup>
1522

23+
<ItemGroup>
24+
<None Include="..\..\assets\Icon128.png">
25+
<Pack>True</Pack>
26+
<PackagePath>\</PackagePath>
27+
</None>
28+
</ItemGroup>
29+
1630
</Project>

src/CurvaLauncher.Plugins/CurvaLauncher.Plugins.csproj

+18-2
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,30 @@
1-
<Project Sdk="Microsoft.NET.Sdk">
1+
<Project Sdk="Microsoft.NET.Sdk">
2+
3+
<Import Project="../CurvaLauncher.props" />
24

35
<PropertyGroup>
46
<TargetFramework>net8.0-windows</TargetFramework>
57
<ImplicitUsings>enable</ImplicitUsings>
68
<Nullable>enable</Nullable>
7-
<UseWPF>true</UseWPF>
89
<RootNamespace>CurvaLauncher</RootNamespace>
10+
11+
<UseWPF>true</UseWPF>
912
<EnableWindowsTargeting>true</EnableWindowsTargeting>
13+
14+
<Description>Plugin library of CurvaLauncher</Description>
15+
<RepositoryType>git</RepositoryType>
16+
<RepositoryUrl>https://github.com/OrgEleCho/CurvaLauncher</RepositoryUrl>
17+
<PackageIcon>Icon128.png</PackageIcon>
18+
<PackageLicenseExpression>MIT</PackageLicenseExpression>
1019
</PropertyGroup>
1120

21+
<ItemGroup>
22+
<None Include="..\..\assets\Icon128.png">
23+
<Pack>True</Pack>
24+
<PackagePath>\</PackagePath>
25+
</None>
26+
</ItemGroup>
27+
1228
<ItemGroup>
1329
<ProjectReference Include="..\CurvaLauncher.Common\CurvaLauncher.Common.csproj" />
1430
</ItemGroup>

src/CurvaLauncher.props

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
<Project>
2+
<PropertyGroup>
3+
<Version>0.7.1-beta</Version>
4+
</PropertyGroup>
5+
</Project>

src/CurvaLauncher.sln

+18
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,15 @@ Project("{6EC3EE1D-3C4E-46DD-8F32-0CC8E7565705}") = "CurvaLauncher.Plugins.SHLoa
3737
EndProject
3838
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CurvaLauncher.Plugins.Everything", "Plugins\CurvaLauncher.Plugins.Everything\CurvaLauncher.Plugins.Everything.csproj", "{C7AC6C75-89C6-42E5-8D2A-D06994949F05}"
3939
EndProject
40+
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{F36EDE5B-9C67-44F4-87A9-792390FECF85}"
41+
ProjectSection(SolutionItems) = preProject
42+
CurvaLauncher.props = CurvaLauncher.props
43+
EndProjectSection
44+
EndProject
45+
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Templates", "Templates", "{45D0660D-436E-4419-AEB9-B6ED5BC3E0ED}"
46+
EndProject
47+
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "CurvaLauncher.PluginTemplate", "Templates\CurvaLauncher.PluginTemplate\CurvaLauncher.PluginTemplate.csproj", "{702E5FF3-99D0-4FD3-86E2-04C0A2E82560}"
48+
EndProject
4049
Global
4150
GlobalSection(SolutionConfigurationPlatforms) = preSolution
4251
Debug|Any CPU = Debug|Any CPU
@@ -165,6 +174,14 @@ Global
165174
{C7AC6C75-89C6-42E5-8D2A-D06994949F05}.Release|Any CPU.Build.0 = Release|Any CPU
166175
{C7AC6C75-89C6-42E5-8D2A-D06994949F05}.Release|x64.ActiveCfg = Release|x64
167176
{C7AC6C75-89C6-42E5-8D2A-D06994949F05}.Release|x64.Build.0 = Release|x64
177+
{702E5FF3-99D0-4FD3-86E2-04C0A2E82560}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
178+
{702E5FF3-99D0-4FD3-86E2-04C0A2E82560}.Debug|Any CPU.Build.0 = Debug|Any CPU
179+
{702E5FF3-99D0-4FD3-86E2-04C0A2E82560}.Debug|x64.ActiveCfg = Debug|Any CPU
180+
{702E5FF3-99D0-4FD3-86E2-04C0A2E82560}.Debug|x64.Build.0 = Debug|Any CPU
181+
{702E5FF3-99D0-4FD3-86E2-04C0A2E82560}.Release|Any CPU.ActiveCfg = Release|Any CPU
182+
{702E5FF3-99D0-4FD3-86E2-04C0A2E82560}.Release|Any CPU.Build.0 = Release|Any CPU
183+
{702E5FF3-99D0-4FD3-86E2-04C0A2E82560}.Release|x64.ActiveCfg = Release|Any CPU
184+
{702E5FF3-99D0-4FD3-86E2-04C0A2E82560}.Release|x64.Build.0 = Release|Any CPU
168185
EndGlobalSection
169186
GlobalSection(SolutionProperties) = preSolution
170187
HideSolutionNode = FALSE
@@ -181,6 +198,7 @@ Global
181198
{F3F6F783-4636-457F-80E1-CC489F524B43} = {BAACD50D-2F94-4A65-8B13-49031D617CAC}
182199
{8CFC1C29-51AA-45ED-A91F-01F513182002} = {4A86F98E-B276-4F75-9847-8D0E4280D887}
183200
{C7AC6C75-89C6-42E5-8D2A-D06994949F05} = {BAACD50D-2F94-4A65-8B13-49031D617CAC}
201+
{702E5FF3-99D0-4FD3-86E2-04C0A2E82560} = {45D0660D-436E-4419-AEB9-B6ED5BC3E0ED}
184202
EndGlobalSection
185203
GlobalSection(ExtensibilityGlobals) = postSolution
186204
SolutionGuid = {3FC4E11A-3D67-43DE-84D8-DCA1841F0D71}

src/CurvaLauncher/CurvaLauncher.csproj

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
<Project Sdk="Microsoft.NET.Sdk">
22

3+
<Import Project="../CurvaLauncher.props"/>
4+
35
<PropertyGroup>
46
<OutputType>WinExe</OutputType>
57
<TargetFramework>net8.0-windows</TargetFramework>
@@ -12,8 +14,6 @@
1214
<ApplicationManifest>app.manifest</ApplicationManifest>
1315
<ApplicationIcon>Assets\Icon.ico</ApplicationIcon>
1416

15-
<Version>0.7.1-beta</Version>
16-
1717
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
1818
<EnableWindowsTargeting>true</EnableWindowsTargeting>
1919
</PropertyGroup>

src/CurvaLauncher/Services/PluginService.cs

+91-31
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,8 @@
1414
using CurvaLauncher.Plugins;
1515
using CurvaLauncher.PluginInteraction;
1616
using System.Diagnostics;
17+
using System.IO.Compression;
18+
using System.Runtime.Loader;
1719

1820
namespace CurvaLauncher.Services;
1921

@@ -50,65 +52,123 @@ private void CoreLoadPlugins(out List<PluginInstance> plugins)
5052
plugins = new List<PluginInstance>();
5153

5254
var dir = EnsurePluginDirectory();
53-
var dllFiles = dir.GetFiles("*.dll");
5455

5556
AppConfig config = _configService.Config;
5657

57-
foreach (FileInfo dllFile in dllFiles)
58-
if (CoreLoadPlugin(config, dllFile.FullName, out PluginInstance? plugin))
58+
foreach (var dllFile in dir.GetFiles("*.dll"))
59+
{
60+
if (CoreLoadDllPlugin(config, dllFile.FullName, out PluginInstance? plugin))
61+
{
62+
plugins.Add(plugin);
63+
}
64+
}
65+
66+
foreach (var zipFile in dir.GetFiles("*.zip"))
67+
{
68+
if (CoreLoadZipPlugin(config, zipFile.FullName, out PluginInstance? plugin))
5969
{
6070
plugins.Add(plugin);
6171
}
72+
}
6273
}
6374

64-
private bool CoreLoadPlugin(AppConfig config, string dllFilePath, [NotNullWhen(true)] out PluginInstance? pluginInstance)
75+
private bool CoreLoadPluginFromAssembly(AppConfig config, Assembly assembly, [NotNullWhen(true)] out PluginInstance? pluginInstance)
6576
{
6677
pluginInstance = null;
6778

68-
try
69-
{
70-
var assembly = Assembly.LoadFile(dllFilePath);
71-
72-
Type? pluginType = assembly.ExportedTypes
79+
Type? pluginType = assembly.ExportedTypes
7380
.Where(type => type.IsAssignableTo(typeof(ISyncPlugin)) || type.IsAssignableTo(typeof(IAsyncPlugin)))
7481
.FirstOrDefault();
7582

76-
if (pluginType == null)
77-
return false;
83+
if (pluginType == null)
84+
return false;
7885

79-
if (!PluginInstance.TryCreate(pluginType, out pluginInstance))
80-
return false;
86+
if (!PluginInstance.TryCreate(pluginType, out pluginInstance))
87+
return false;
8188

82-
var typeName = pluginType.FullName!;
89+
var typeName = pluginType.FullName!;
8390

84-
if (config.Plugins.TryGetValue(typeName, out var pluginConfig))
85-
{
86-
var props = pluginInstance.Plugin.GetType().GetProperties()
91+
if (config.Plugins.TryGetValue(typeName, out var pluginConfig))
92+
{
93+
var props = pluginInstance.Plugin.GetType().GetProperties()
8794
.Where(p => p.GetCustomAttribute<PluginOptionAttribute>() is not null
8895
|| p.GetCustomAttribute<PluginI18nOptionAttribute>() is not null);
8996

90-
if (pluginConfig.Options != null)
97+
if (pluginConfig.Options != null)
98+
{
99+
foreach (var property in props)
91100
{
92-
foreach (var property in props)
101+
if (pluginConfig.Options.TryGetPropertyValue(property.Name, out var value))
93102
{
94-
if (pluginConfig.Options.TryGetPropertyValue(property.Name, out var value))
95-
{
96-
var type = property.PropertyType;
97-
var val = JsonSerializer.Deserialize(value, type);
98-
property.SetValue(pluginInstance.Plugin, val);
99-
}
103+
var type = property.PropertyType;
104+
var val = JsonSerializer.Deserialize(value, type);
105+
property.SetValue(pluginInstance.Plugin, val);
100106
}
101107
}
102-
103-
pluginInstance.IsEnabled = pluginConfig.IsEnabled;
104-
pluginInstance.Weight = pluginConfig.Weight;
105108
}
106-
else
109+
110+
pluginInstance.IsEnabled = pluginConfig.IsEnabled;
111+
pluginInstance.Weight = pluginConfig.Weight;
112+
}
113+
else
114+
{
115+
pluginInstance.IsEnabled = true;
116+
}
117+
118+
return true;
119+
}
120+
121+
private bool CoreLoadDllPlugin(AppConfig config, string dllFilePath, [NotNullWhen(true)] out PluginInstance? pluginInstance)
122+
{
123+
pluginInstance = null;
124+
125+
try
126+
{
127+
var assembly = Assembly.LoadFile(dllFilePath);
128+
return CoreLoadPluginFromAssembly(config, assembly, out pluginInstance);
129+
}
130+
catch (Exception ex)
131+
{
132+
Debug.WriteLine($"Plugin load failed, {ex}");
133+
return false;
134+
}
135+
}
136+
137+
private bool CoreLoadZipPlugin(AppConfig config, string zipFilePath, [NotNullWhen(true)] out PluginInstance? pluginInstance)
138+
{
139+
pluginInstance = null;
140+
141+
try
142+
{
143+
using var zipFile = File.OpenRead(zipFilePath);
144+
using var zipArchive = new ZipArchive(zipFile, ZipArchiveMode.Read);
145+
var assemblyLoadContext = new AssemblyLoadContext(null, false);
146+
147+
foreach (var entry in zipArchive.Entries)
107148
{
108-
pluginInstance.IsEnabled = true;
149+
if (!entry.FullName.EndsWith(".dll", StringComparison.OrdinalIgnoreCase))
150+
{
151+
continue;
152+
}
153+
154+
using var entryStream = entry.Open();
155+
156+
try
157+
{
158+
var assembly = assemblyLoadContext.LoadFromStream(entryStream);
159+
160+
if (pluginInstance is null)
161+
{
162+
CoreLoadPluginFromAssembly(config, assembly, out pluginInstance);
163+
}
164+
}
165+
catch (Exception ex)
166+
{
167+
Debug.WriteLine($"DLL load failed, {ex}");
168+
}
109169
}
110170

111-
return true;
171+
return pluginInstance is not null;
112172
}
113173
catch (Exception ex)
114174
{
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
{
2+
"$schema": "http://json.schemastore.org/template",
3+
"author": "EleCho",
4+
"identity": "CurvaLauncher.PluginTemplate",
5+
"classifications": [
6+
"CurvaLauncher",
7+
"Library"
8+
],
9+
"name": "CurvaLauncher Plugin",
10+
"description": "Creates a CurvaLauncher plugin",
11+
"tags": {
12+
"language": "C#",
13+
"type": "project"
14+
},
15+
"shortName": "clp",
16+
"sourceName": "CurvaLauncher.PluginTemplate",
17+
"symbols": {
18+
"EnableI18n": {
19+
"type": "parameter",
20+
"datatype": "bool",
21+
"defaultValue": "true"
22+
}
23+
},
24+
"sources": [
25+
{
26+
"modifiers": [
27+
{
28+
"condition": "(!EnableI18n)",
29+
"exclude": [
30+
"I18n/EnUs.xaml",
31+
"I18n/ZhHans.xaml"
32+
]
33+
}
34+
]
35+
}
36+
]
37+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
<Project Sdk="Microsoft.NET.Sdk">
2+
3+
<PropertyGroup>
4+
<TargetFramework>net8.0-windows</TargetFramework>
5+
<Nullable>enable</Nullable>
6+
</PropertyGroup>
7+
8+
<ItemGroup>
9+
<PackageReference Include="CurvaLauncher.Plugins" Version="0.7.1-beta" />
10+
</ItemGroup>
11+
12+
<ItemGroup>
13+
<Page Include="I18n\EnUs.xaml">
14+
<Generator>MSBuild:Compile</Generator>
15+
</Page>
16+
<Page Include="I18n\ZhHans.xaml">
17+
<Generator>MSBuild:Compile</Generator>
18+
</Page>
19+
</ItemGroup>
20+
21+
</Project>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
2+
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
3+
xmlns:sys="clr-namespace:System;assembly=mscorlib">
4+
<sys:String x:Key="StrPluginName">Sample Plugin</sys:String>
5+
<sys:String x:Key="StrPluginDescription">This is a plugin with i18n support</sys:String>
6+
7+
<sys:String x:Key="StrTestResultTitle">Test Plugin</sys:String>
8+
<sys:String x:Key="StrTestResultDescription">Your plugin is running well</sys:String>
9+
</ResourceDictionary>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
2+
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
3+
xmlns:sys="clr-namespace:System;assembly=mscorlib">
4+
<sys:String x:Key="StrPluginName">示例插件</sys:String>
5+
<sys:String x:Key="StrPluginDescription">这是一个支持本地化的插件</sys:String>
6+
7+
<sys:String x:Key="StrTestResultTitle">测试插件</sys:String>
8+
<sys:String x:Key="StrTestResultDescription">你的插件运行良好</sys:String>
9+
</ResourceDictionary>

0 commit comments

Comments
 (0)