From 6c521a59a2756f0ef1105245779d1c7146c99797 Mon Sep 17 00:00:00 2001 From: Argo Zhang Date: Sat, 9 Dec 2023 14:44:21 +0800 Subject: [PATCH] feat(AzureTranslator): add AzureTranslator service (#2519) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * feat: 增加 Translator 服务 * chore: 增加翻译服务 * doc: 增加翻译源码映射 * doc: 增加翻译服务示例 * chore: 更新内置服务菜单 * doc: 增加本地化资源 --- BootstrapBlazor.sln | 7 ++ .../BootstrapBlazor.Server.csproj | 1 + .../Components/Samples/Translators.razor | 30 +++++ .../Components/Samples/Translators.razor.cs | 65 +++++++++++ .../Extensions/MenusLocalizerExtensions.cs | 21 ++++ .../Extensions/ServicesExtensions.cs | 3 + src/BootstrapBlazor.Server/Locales/en.json | 11 +- src/BootstrapBlazor.Server/Locales/zh.json | 11 +- src/BootstrapBlazor.Server/docs.json | 1 + .../AzureTranslatorOption.cs | 28 +++++ .../BootstrapBlazor.AzureTranslator.csproj | 18 +++ .../Extensions/ServiceCollectionExtensions.cs | 35 ++++++ .../Services/AzureTranslatorService.cs | 103 ++++++++++++++++++ .../Services/IAzureTranslatorService.cs | 43 ++++++++ .../BootstrapBlazor.AzureTranslator/logo.png | Bin 0 -> 6414 bytes 15 files changed, 375 insertions(+), 2 deletions(-) create mode 100644 src/BootstrapBlazor.Server/Components/Samples/Translators.razor create mode 100644 src/BootstrapBlazor.Server/Components/Samples/Translators.razor.cs create mode 100644 src/Extensions/Components/BootstrapBlazor.AzureTranslator/AzureTranslatorOption.cs create mode 100644 src/Extensions/Components/BootstrapBlazor.AzureTranslator/BootstrapBlazor.AzureTranslator.csproj create mode 100644 src/Extensions/Components/BootstrapBlazor.AzureTranslator/Extensions/ServiceCollectionExtensions.cs create mode 100644 src/Extensions/Components/BootstrapBlazor.AzureTranslator/Services/AzureTranslatorService.cs create mode 100644 src/Extensions/Components/BootstrapBlazor.AzureTranslator/Services/IAzureTranslatorService.cs create mode 100644 src/Extensions/Components/BootstrapBlazor.AzureTranslator/logo.png diff --git a/BootstrapBlazor.sln b/BootstrapBlazor.sln index 3e737d0e1e8..d73a457fcae 100644 --- a/BootstrapBlazor.sln +++ b/BootstrapBlazor.sln @@ -120,6 +120,8 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "configuration", "configurat exclusion.dic = exclusion.dic EndProjectSection EndProject +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "BootstrapBlazor.AzureTranslator", "src\Extensions\Components\BootstrapBlazor.AzureTranslator\BootstrapBlazor.AzureTranslator.csproj", "{D63E50CB-EF78-47BF-9809-003D4574ABF4}" +EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|Any CPU = Debug|Any CPU @@ -258,6 +260,10 @@ Global {49D9E679-6824-43DF-9DAA-70E24ACCAA62}.Debug|Any CPU.Build.0 = Debug|Any CPU {49D9E679-6824-43DF-9DAA-70E24ACCAA62}.Release|Any CPU.ActiveCfg = Release|Any CPU {49D9E679-6824-43DF-9DAA-70E24ACCAA62}.Release|Any CPU.Build.0 = Release|Any CPU + {D63E50CB-EF78-47BF-9809-003D4574ABF4}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {D63E50CB-EF78-47BF-9809-003D4574ABF4}.Debug|Any CPU.Build.0 = Debug|Any CPU + {D63E50CB-EF78-47BF-9809-003D4574ABF4}.Release|Any CPU.ActiveCfg = Release|Any CPU + {D63E50CB-EF78-47BF-9809-003D4574ABF4}.Release|Any CPU.Build.0 = Release|Any CPU EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE @@ -303,6 +309,7 @@ Global {568237B9-209E-49B1-9BCF-E853F2D61442} = {CD062AB6-244D-402A-8F33-C37DAC5856CC} {25B67FF0-3D70-47D8-AC70-1934442A8881} = {CD062AB6-244D-402A-8F33-C37DAC5856CC} {49D9E679-6824-43DF-9DAA-70E24ACCAA62} = {CD062AB6-244D-402A-8F33-C37DAC5856CC} + {D63E50CB-EF78-47BF-9809-003D4574ABF4} = {CD062AB6-244D-402A-8F33-C37DAC5856CC} EndGlobalSection GlobalSection(ExtensibilityGlobals) = postSolution SolutionGuid = {0DCB0756-34FA-4FD0-AE1D-D3F08B5B3A6B} diff --git a/src/BootstrapBlazor.Server/BootstrapBlazor.Server.csproj b/src/BootstrapBlazor.Server/BootstrapBlazor.Server.csproj index 94c1cd890ea..fd4ff853589 100644 --- a/src/BootstrapBlazor.Server/BootstrapBlazor.Server.csproj +++ b/src/BootstrapBlazor.Server/BootstrapBlazor.Server.csproj @@ -30,6 +30,7 @@ + diff --git a/src/BootstrapBlazor.Server/Components/Samples/Translators.razor b/src/BootstrapBlazor.Server/Components/Samples/Translators.razor new file mode 100644 index 00000000000..02e770cf65e --- /dev/null +++ b/src/BootstrapBlazor.Server/Components/Samples/Translators.razor @@ -0,0 +1,30 @@ +@page "/translator" +@inject IStringLocalizer Localizer + +

@Localizer["TranslatorsTitle"]

+

@Localizer["TranslatorsDescription"]

+ + + +
@Localizer["TranslatorsInjectService"]
+
services.AddBootstrapBlazorAzureTranslator();
+ + + + + + + + + @if (_results.Any()) + { + @foreach (var translation in _results.First().Translations) + { +
@FormatResult(translation)
+ } + } +
+ + diff --git a/src/BootstrapBlazor.Server/Components/Samples/Translators.razor.cs b/src/BootstrapBlazor.Server/Components/Samples/Translators.razor.cs new file mode 100644 index 00000000000..ddc31a57616 --- /dev/null +++ b/src/BootstrapBlazor.Server/Components/Samples/Translators.razor.cs @@ -0,0 +1,65 @@ +// Copyright (c) Argo Zhang (argo@163.com). All rights reserved. +// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. +// Website: https://www.blazor.zone or https://argozhang.github.io/ + +using Azure.AI.Translation.Text; +using System.Globalization; + +namespace BootstrapBlazor.Server.Components.Samples; + +/// +/// 翻译示例 +/// +public partial class Translators +{ + private readonly List _languages = []; + + private List _selectedLanguages = []; + + private static readonly string[] sourceArray = ["zh-CN", "en-US", "ru-RU"]; + + private string _input = ""; + + private IReadOnlyList _results = new List(); + + [Inject] + [NotNull] + private IAzureTranslatorService? TranslatorService { get; set; } + + /// + /// + /// + protected override void OnInitialized() + { + base.OnInitialized(); + + _languages.AddRange(sourceArray.Select(i => new SelectedItem(i, new CultureInfo(i).NativeName))); + _selectedLanguages.AddRange(sourceArray); + } + + private async Task OnClickTranslate() + { + _results = await TranslatorService.TranslateAsync(_selectedLanguages, [_input], "en-US"); + } + + private static string FormatResult(Translation translation) + { + var culture = new CultureInfo(translation.To); + return $"{culture.NativeName}: {translation.Text}"; + } + + /// + /// 获得属性方法 + /// + /// + protected IEnumerable GetMethods() => new MethodItem[] + { + new() + { + Name = nameof(IAzureTranslatorService.TranslateAsync), + Description = Localizer[nameof(IAzureTranslatorService.TranslateAsync)], + Parameters = " - ", + ReturnValue = "IReadOnlyList" + }, + }; +} diff --git a/src/BootstrapBlazor.Server/Extensions/MenusLocalizerExtensions.cs b/src/BootstrapBlazor.Server/Extensions/MenusLocalizerExtensions.cs index b0f15b6b053..4fa4435d006 100644 --- a/src/BootstrapBlazor.Server/Extensions/MenusLocalizerExtensions.cs +++ b/src/BootstrapBlazor.Server/Extensions/MenusLocalizerExtensions.cs @@ -83,6 +83,13 @@ public static List GenerateMenus(this IStringLocalizer Locali }; AddSpeech(item); + item = new DemoMenuItem() + { + Text = Localizer["Services"], + Icon = "fa-fw fa-solid fa-screwdriver-wrench", + }; + AddServices(item); + item = new DemoMenuItem() { Text = Localizer["OtherComponents"], @@ -1291,6 +1298,20 @@ void AddLayout(DemoMenuItem item) AddBadge(item); } + void AddServices(DemoMenuItem item) + { + item.Items = new List + { + new() + { + IsNew = true, + Text = Localizer["AzureTranslator"], + Url = "translator" + } + }; + AddBadge(item); + } + void AddSummary(DemoMenuItem item) { // 计算组件总数 diff --git a/src/BootstrapBlazor.Server/Extensions/ServicesExtensions.cs b/src/BootstrapBlazor.Server/Extensions/ServicesExtensions.cs index 3c8ba9a2113..18c3dd6bb95 100644 --- a/src/BootstrapBlazor.Server/Extensions/ServicesExtensions.cs +++ b/src/BootstrapBlazor.Server/Extensions/ServicesExtensions.cs @@ -61,6 +61,9 @@ void Invoke(BootstrapBlazorOptions option) // 增加 AzureOpenAI 服务 services.AddBootstrapBlazorAzureOpenAIService(); + // 增加 AzureTranslator 服务 + services.AddBootstrapBlazorAzureTranslator(); + // 增加 Pdf 导出服务 services.AddBootstrapBlazorHtml2PdfService(); diff --git a/src/BootstrapBlazor.Server/Locales/en.json b/src/BootstrapBlazor.Server/Locales/en.json index fb5cde3dd79..6bc140c0ff6 100644 --- a/src/BootstrapBlazor.Server/Locales/en.json +++ b/src/BootstrapBlazor.Server/Locales/en.json @@ -4528,7 +4528,9 @@ "Clipboard": "Clipboard Service", "CodeEditor": "CodeEditor", "Gantt": "Gantt", - "ImageCropper": "ImageCropper" + "ImageCropper": "ImageCropper", + "Services": "Services", + "AzureTranslator": "AzureTranslator" }, "BootstrapBlazor.Server.Components.Samples.Table.TablesHeader": { "TablesHeaderTitle": "Header grouping function", @@ -6039,5 +6041,12 @@ "AttributesImageCropperOnBase64Result": "Crop result base64 callback method", "AttributesImageCropperCrop": "Crop, return base64, and execute OnResult/OnBase64Result callback", "AttributesImageCropperCropToStream": "Crop, return Stream" + }, + "BootstrapBlazor.Server.Components.Samples.Translators": { + "TranslatorsTitle": "Azure Translator", + "TranslatorsDescription": "Converts characters or letters of a source language to the corresponding characters or letters of a target language.", + "TranslatorsNormalTitle": "Basic usage", + "TranslatorsNormalIntro": "Call TranslateAsync method for convert target language", + "TranslatorsInjectService": "Inject Service" } } diff --git a/src/BootstrapBlazor.Server/Locales/zh.json b/src/BootstrapBlazor.Server/Locales/zh.json index 899c8f3c039..8728608c132 100644 --- a/src/BootstrapBlazor.Server/Locales/zh.json +++ b/src/BootstrapBlazor.Server/Locales/zh.json @@ -4528,7 +4528,9 @@ "Clipboard": "剪切板服务", "CodeEditor": "代码编辑器 CodeEditor", "Gantt": "甘特图 Gantt", - "ImageCropper": "图像裁剪 ImageCropper" + "ImageCropper": "图像裁剪 ImageCropper", + "Services": "内置服务", + "AzureTranslator": "翻译服务 AzureTranslator" }, "BootstrapBlazor.Server.Components.Samples.Table.TablesHeader": { "TablesHeaderTitle": "表头分组功能", @@ -6039,5 +6041,12 @@ "AttributesImageCropperOnBase64Result": "剪裁结果 base64 回调方法", "AttributesImageCropperCrop": "剪裁,返回 base64, 并执行 OnResult/OnBase64Result 回调", "AttributesImageCropperCropToStream": "剪裁,返回 Stream" + }, + "BootstrapBlazor.Server.Components.Samples.Translators": { + "TranslatorsTitle": "AzureTranslator 翻译服务", + "TranslatorsDescription": "将源语言的字符或字母转换为目标语言的对应字符或字母", + "TranslatorsNormalTitle": "基础用法", + "TranslatorsNormalIntro": "通过调用 TranslateAsync 进行文本翻译", + "TranslatorsInjectService": "注入服务" } } diff --git a/src/BootstrapBlazor.Server/docs.json b/src/BootstrapBlazor.Server/docs.json index 5f6ffd02dbc..c5f368bc2ff 100644 --- a/src/BootstrapBlazor.Server/docs.json +++ b/src/BootstrapBlazor.Server/docs.json @@ -191,6 +191,7 @@ "table/tree": "Table\\TablesTree", "table/virtualization": "Table\\TablesVirtualization", "table/wrap": "Table\\TablesWrap", + "translator": "Translators", "js-extensions": "JSRuntimeExtensions", "clipboard-service": "Clipboards", "image-cropper": "ImageCroppers" diff --git a/src/Extensions/Components/BootstrapBlazor.AzureTranslator/AzureTranslatorOption.cs b/src/Extensions/Components/BootstrapBlazor.AzureTranslator/AzureTranslatorOption.cs new file mode 100644 index 00000000000..58f26d1a84e --- /dev/null +++ b/src/Extensions/Components/BootstrapBlazor.AzureTranslator/AzureTranslatorOption.cs @@ -0,0 +1,28 @@ +// Copyright (c) Argo Zhang (argo@163.com). All rights reserved. +// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. +// Website: https://www.blazor.zone or https://argozhang.github.io/ + +namespace BootstrapBlazor.Components; + +/// +/// AzureTranslatorOption 配置类 +/// +public class AzureTranslatorOption +{ + /// + /// 获得/设置 订阅 Key 由 Azure 提供 + /// + [NotNull] + public string? Key { get; set; } + + /// + /// 获得/设置 Location/Region 描述 由 Azure 提供 + /// + [NotNull] + public string? Region { get; set; } + + /// + /// 获得/设置 超时时长 默认 0 未设置 + /// + public int Timeout { get; set; } +} diff --git a/src/Extensions/Components/BootstrapBlazor.AzureTranslator/BootstrapBlazor.AzureTranslator.csproj b/src/Extensions/Components/BootstrapBlazor.AzureTranslator/BootstrapBlazor.AzureTranslator.csproj new file mode 100644 index 00000000000..2050693171a --- /dev/null +++ b/src/Extensions/Components/BootstrapBlazor.AzureTranslator/BootstrapBlazor.AzureTranslator.csproj @@ -0,0 +1,18 @@ + + + + 8.0.0 + + + + Bootstrap Blazor Azure WebAssembly wasm Translator Components + Bootstrap UI components extensions of Azure Translator + BootstrapBlazor.Components + + + + + + + + diff --git a/src/Extensions/Components/BootstrapBlazor.AzureTranslator/Extensions/ServiceCollectionExtensions.cs b/src/Extensions/Components/BootstrapBlazor.AzureTranslator/Extensions/ServiceCollectionExtensions.cs new file mode 100644 index 00000000000..ba39c3a3436 --- /dev/null +++ b/src/Extensions/Components/BootstrapBlazor.AzureTranslator/Extensions/ServiceCollectionExtensions.cs @@ -0,0 +1,35 @@ +// Copyright (c) Argo Zhang (argo@163.com). All rights reserved. +// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. +// Website: https://www.blazor.zone or https://argozhang.github.io/ + +using BootstrapBlazor.Components; +using Microsoft.Extensions.DependencyInjection.Extensions; + +namespace Microsoft.Extensions.DependencyInjection; + +/// +/// BootstrapBlazor 服务扩展类 +/// +public static class ServiceCollectionExtensions +{ + /// + /// 增加 语音识别服务 + /// + /// + /// + /// + public static IServiceCollection AddBootstrapBlazorAzureTranslator(this IServiceCollection services, Action< + AzureTranslatorOption>? configOptions = null) + { + services.AddHttpClient(); + + services.AddSingleton(); + services.AddOptionsMonitor(); + + services.Configure(option => + { + configOptions?.Invoke(option); + }); + return services; + } +} diff --git a/src/Extensions/Components/BootstrapBlazor.AzureTranslator/Services/AzureTranslatorService.cs b/src/Extensions/Components/BootstrapBlazor.AzureTranslator/Services/AzureTranslatorService.cs new file mode 100644 index 00000000000..843bd0196fd --- /dev/null +++ b/src/Extensions/Components/BootstrapBlazor.AzureTranslator/Services/AzureTranslatorService.cs @@ -0,0 +1,103 @@ +// Copyright (c) Argo Zhang (argo@163.com). All rights reserved. +// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. +// Website: https://www.blazor.zone or https://argozhang.github.io/ + +using Azure; +using Azure.AI.Translation.Text; +using Microsoft.Extensions.Logging; +using Microsoft.Extensions.Options; + +namespace BootstrapBlazor.Components; + +/// +/// AzureTranslatorService 服务实现类 +/// +/// +/// 构造函数 +/// +/// +/// +class AzureTranslatorService(IOptionsMonitor option, ILogger logger) : IAzureTranslatorService +{ + private readonly IOptionsMonitor _option = option; + + private readonly ILogger _logger = logger; + + /// + /// 文本翻译方法 + /// + /// 目标语言 + /// 输入文本 + /// 源语言 + /// + /// + public async Task> TranslateAsync(string targetLanguage, string inputText, string? sourceLanguage = null, CancellationToken cancellationToken = default) + { + var key = _option.CurrentValue.Key; + var region = _option.CurrentValue.Region; + var client = new TextTranslationClient(new AzureKeyCredential(key), region); + IReadOnlyList? ret = null; + try + { + var response = await client.TranslateAsync(targetLanguage, inputText, sourceLanguage, cancellationToken); + ret = response.Value; + } + catch (RequestFailedException ex) + { + _logger.LogError(ex, "TranslateAsync method failed target language: {targetLanguage}, input text: {inputText}", targetLanguage, inputText); + } + return ret ?? new List(); + } + + /// + /// 文本翻译方法 + /// + /// 目标语言 + /// 输入文本集合 + /// 源语言 + /// + /// + public async Task> TranslateAsync(string targetLanguage, IEnumerable content, string? sourceLanguage = null, CancellationToken cancellationToken = default) + { + var key = _option.CurrentValue.Key; + var region = _option.CurrentValue.Region; + var client = new TextTranslationClient(new AzureKeyCredential(key), region); + IReadOnlyList? ret = null; + try + { + var response = await client.TranslateAsync(targetLanguage, content, sourceLanguage, cancellationToken); + ret = response.Value; + } + catch (RequestFailedException ex) + { + _logger.LogError(ex, "TranslateAsync method failed target language: {targetLanguage}, content: {inputText}", targetLanguage, content); + } + return ret ?? new List(); + } + + /// + /// 文本翻译方法 + /// + /// 目标语言集合 + /// 输入文本集合 + /// 源语言 + /// + /// + public async Task> TranslateAsync(IEnumerable targetLanguages, IEnumerable content, string? sourceLanguage = null, CancellationToken cancellationToken = default) + { + var key = _option.CurrentValue.Key; + var region = _option.CurrentValue.Region; + var client = new TextTranslationClient(new AzureKeyCredential(key), region); + IReadOnlyList? ret = null; + try + { + var response = await client.TranslateAsync(targetLanguages, content, sourceLanguage: sourceLanguage, cancellationToken: cancellationToken); + ret = response.Value; + } + catch (RequestFailedException ex) + { + _logger.LogError(ex, "TranslateAsync method failed target language: {targetLanguage}, content: {inputText}", targetLanguages, content); + } + return ret ?? new List(); + } +} diff --git a/src/Extensions/Components/BootstrapBlazor.AzureTranslator/Services/IAzureTranslatorService.cs b/src/Extensions/Components/BootstrapBlazor.AzureTranslator/Services/IAzureTranslatorService.cs new file mode 100644 index 00000000000..c5a8f86dde8 --- /dev/null +++ b/src/Extensions/Components/BootstrapBlazor.AzureTranslator/Services/IAzureTranslatorService.cs @@ -0,0 +1,43 @@ +// Copyright (c) Argo Zhang (argo@163.com). All rights reserved. +// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. +// Website: https://www.blazor.zone or https://argozhang.github.io/ + +using Azure.AI.Translation.Text; + +namespace BootstrapBlazor.Components; + +/// +/// IAzureTranslatorService 文本翻译服务 +/// +public interface IAzureTranslatorService +{ + /// + /// 翻译方法 + /// + /// + /// + /// + /// + /// + Task> TranslateAsync(string targetLanguage, string inputText, string? sourceLanguage = null, CancellationToken cancellationToken = default); + + /// + /// 翻译方法 + /// + /// + /// + /// + /// + /// + Task> TranslateAsync(string targetLanguage, IEnumerable content, string? sourceLanguage = null, CancellationToken cancellationToken = default); + + /// + /// 翻译方法 + /// + /// + /// + /// + /// + /// + Task> TranslateAsync(IEnumerable targetLanguages, IEnumerable content, string? sourceLanguage = null, CancellationToken cancellationToken = default); +} diff --git a/src/Extensions/Components/BootstrapBlazor.AzureTranslator/logo.png b/src/Extensions/Components/BootstrapBlazor.AzureTranslator/logo.png new file mode 100644 index 0000000000000000000000000000000000000000..a30290780f1d7fbb242279e1ed64a94be6341d16 GIT binary patch literal 6414 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