Compatibilidad multilingüe con los mensajes predeterminados que se muestran durante la validación de entrada

Fecha de creación de la página :

medio ambiente

núcleo ASP.NET
  • 5.0 MVC

Al principio

Es posible que la compatibilidad multilingüe de los mensajes de validación de entrada predeterminados con estos consejos no esté completa. Por favor, trate con los que faltan.

Además, cambiar el idioma según el estado de la sesión o el estado de la memoria caché puede seguir apareciendo en el texto del idioma anterior.

Hay varias formas de cambiar el mensaje de validación de entrada predeterminado, cada una de las cuales se puede combinar. No es uno de ellos, por lo que si desea cambiarlo de manera confiable, debe corresponder con todos los patrones.

premisa

Estos consejos están escritos para comprender los siguientes consejos:

Además, si está creando un nuevo proyecto, debe haber agregado los siguientes archivos y código según los consejos anteriores.

  • Cree un archivo SharedResource.resx (+en, es). (Dado que solo se traduce el mensaje de estos Consejos, el contenido puede estar vacío).
  • Crear un archivo SharedResource.cs
  • Agregar código de localización a Startup.ConfigureServices
  • Agregar código de localización a Startup.Configure
  • Se agregó UserViewModel (esta vez, no tejamos explícitamente los mensajes predeterminados en varios idiomas)
  • Se agregaron acciones y vistas de pantalla creadas por el usuario (Create.cshtml)

Código de inicio

Sobre la base de los supuestos anteriores, cada código será el siguiente:

Inicio.cs
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.AspNetCore.Localization;
using Microsoft.AspNetCore.Mvc.Razor;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using System.Globalization;

namespace LocalizationDefaultValidation
{
  public class Startup
  {
    // 省略

    public void ConfigureServices(IServiceCollection services)
    {
      services.AddControllersWithViews();

      services.AddMvc()
        // ローカライズに必要。Resx ファイルのフォルダパスを指定
        .AddViewLocalization(LanguageViewLocationExpanderFormat.Suffix, opts => { opts.ResourcesPath = "Resources"; })
        // DataAnnotations のローカライズに必要
        .AddDataAnnotationsLocalization(options =>
        {
          // DataAnnotation を使ったときのメッセージは SharedResource に集約する
          options.DataAnnotationLocalizerProvider = (type, factory) => factory.Create(typeof(SharedResource));
        });
    }

    public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
    {
      if (env.IsDevelopment())
      {
        app.UseDeveloperExceptionPage();
      }
      else
      {
        app.UseExceptionHandler("/Home/Error");
        // The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
        app.UseHsts();
      }

      // 標準の機能で切り替えたい言語を定義します。
      var supportedCultures = new[]
      {
        new CultureInfo("ja"),
        new CultureInfo("en"),
        new CultureInfo("es"),
      };

      // 標準の言語切り替え機能を有効にします。対応しているのは「クエリ文字列」「Cookie」「Accept-Language HTTP ヘッダー」です。
      app.UseRequestLocalization(new RequestLocalizationOptions
      {
        DefaultRequestCulture = new RequestCulture("ja"),
        SupportedCultures = supportedCultures,
        SupportedUICultures = supportedCultures
      });

      app.UseHttpsRedirection();
      app.UseStaticFiles();

      // 省略
    }
  }
}
HomeController.cs
using LocalizationDefaultValidation.Models;
using Microsoft.AspNetCore.Mvc;
using Microsoft.AspNetCore.Mvc.Rendering;
using Microsoft.Extensions.Logging;
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Threading.Tasks;

namespace LocalizationDefaultValidation.Controllers
{
  public class HomeController : Controller
  {
    // 省略

    public IActionResult Create()
    {
      SetDayOfWeeksViewData();
      return View();
    }

    [HttpPost]
    public IActionResult Create(UserViewModel model)
    {
      SetDayOfWeeksViewData();

      // エラーなら差し戻し
      if (ModelState.IsValid == false) return View(model);

      // 正常に登録できた場合は Index に戻る
      return RedirectToAction(nameof(Index));
    }

    /// <summary>曜日の一覧を ViewData に設定します。</summary>
    private void SetDayOfWeeksViewData()
      => ViewData["DayOfWeeks"] = ((DayOfWeek[])Enum.GetValues(typeof(DayOfWeek))).Select(x => new SelectListItem(x.ToString(), ((int)x).ToString()));
  }
}
Index.cshtml
@{
  ViewData["Title"] = "Home Page";
}

<div class="text-center">
  <h1 class="display-4">Welcome</h1>
  <p>Learn about <a href="https://docs.microsoft.com/aspnet/core">building Web apps with ASP.NET Core</a>.</p>
</div>

<p>ユーザー作成画面に遷移します。リンクごとに選択した言語で表示できます。</p>
<ul>
  <li><a asp-action="Create">Create</a></li>
  <li><a asp-action="Create" asp-route-culture="ja">Create (ja)</a></li>
  <li><a asp-action="Create" asp-route-culture="en">Create (en)</a></li>
  <li><a asp-action="Create" asp-route-culture="es">Create (es)</a></li>
</ul>
Crear.cshtml
@model LocalizationDefaultValidation.Models.UserViewModel
@using System.Globalization;
@{
  ViewData["Title"] = "Create";
}

<h1>Create</h1>

<h4>UserViewModel</h4>
<hr />
<div class="row">
  <div class="col-md-4">
    <form asp-action="Create" enctype="multipart/form-data" >
      <div asp-validation-summary="ModelOnly" class="text-danger"></div>
      <div class="form-group">
        <label asp-for="ID" class="control-label"></label>
        <input asp-for="ID" class="form-control" />
        <span asp-validation-for="ID" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Name" class="control-label"></label>
        <input asp-for="Name" class="form-control" />
        <span asp-validation-for="Name" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Password" class="control-label"></label>
        <input asp-for="Password" class="form-control" />
        <span asp-validation-for="Password" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="ConfirmPassword" class="control-label"></label>
        <input asp-for="ConfirmPassword" class="form-control" />
        <span asp-validation-for="ConfirmPassword" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Email" class="control-label"></label>
        <input asp-for="Email" class="form-control" />
        <span asp-validation-for="Email" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Age" class="control-label"></label>
        <input asp-for="Age" class="form-control" />
        <span asp-validation-for="Age" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Gender"></label>
        <div>
          <label><input type="radio" asp-for="Gender" value="@(GenderType.None)" />@(GenderType.None)</label>
          <label><input type="radio" asp-for="Gender" value="@(GenderType.Man)" />@(GenderType.Man)</label>
          <label><input type="radio" asp-for="Gender" value="@(GenderType.Woman)" />@(GenderType.Woman)</label>
        </div>
        <span asp-validation-for="Gender" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Birthday" class="control-label"></label>
        <input asp-for="Birthday" class="form-control" />
        <span asp-validation-for="Birthday" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Phone" class="control-label"></label>
        <input asp-for="Phone" class="form-control" />
        <span asp-validation-for="Phone" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="PostalCode" class="control-label"></label>
        <input asp-for="PostalCode" class="form-control" />
        <span asp-validation-for="PostalCode" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="CreditCard" class="control-label"></label>
        <input asp-for="CreditCard" class="form-control" />
        <span asp-validation-for="CreditCard" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Money" class="control-label"></label>
        <input asp-for="Money" class="form-control" />
        <span asp-validation-for="Money" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="StartDateTime" class="control-label"></label>
        <input asp-for="StartDateTime" class="form-control" />
        <span asp-validation-for="StartDateTime" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="WakeUpTime" class="control-label"></label>
        <input asp-for="WakeUpTime" class="form-control" />
        <span asp-validation-for="WakeUpTime" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Homepage" class="control-label"></label>
        <input asp-for="Homepage" class="form-control" />
        <span asp-validation-for="Homepage" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="MyImage" class="control-label"></label>
        <input asp-for="MyImage" class="form-control" />
        <span asp-validation-for="MyImage" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="MyColor" class="control-label"></label>
        <input asp-for="MyColor" class="form-control" type="color" />
        <span asp-validation-for="MyColor" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="WorkingDays" class="control-label"></label>
        <select asp-for="WorkingDays" class="form-control" asp-items="@((IEnumerable<SelectListItem>)ViewData["DayOfWeeks"])" multiple></select>
        <span asp-validation-for="WorkingDays" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="VacationDay" class="control-label"></label>
        <select asp-for="VacationDay" class="form-control" asp-items="@((IEnumerable<SelectListItem>)ViewData["DayOfWeeks"])" multiple></select>
        <span asp-validation-for="VacationDay" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Comment" class="control-label"></label>
        <textarea asp-for="Comment" class="form-control"></textarea>
        <span asp-validation-for="Comment" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="FileName" class="control-label"></label>
        <input asp-for="FileName" class="form-control" />
        <span asp-validation-for="FileName" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="UploadFile" class="control-label"></label>
        <input asp-for="UploadFile" type="file" multiple />
        <span asp-validation-for="UploadFile" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Month" class="control-label"></label>
        <input asp-for="Month" class="form-control" type="month" />
        <span asp-validation-for="Month" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Search" class="control-label"></label>
        <input asp-for="Search" class="form-control" type="search" />
        <span asp-validation-for="Search" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Range" class="control-label"></label>
        <input asp-for="Range" class="form-control" type="range" min="10" max="100" />
        <span asp-validation-for="Range" class="text-danger"></span>
      </div>
      <div class="form-group">
        <label asp-for="Week" class="control-label"></label>
        <input asp-for="Week" class="form-control" type="week" />
        <span asp-validation-for="Week" class="text-danger"></span>
      </div>
      <div class="form-group form-check">
        <label class="form-check-label">
          <input class="form-check-input" asp-for="IsAccepted" /> @Html.DisplayNameFor(model => model.IsAccepted)
        </label>
      </div>
      <div class="form-group">
        <input type="submit" value="Create" class="btn btn-primary" asp-route-culture="@CultureInfo.CurrentCulture.Name" />
      </div>
    </form>
  </div>
</div>

<div>
  <a asp-action="Index">Back to List</a>
</div>

@section Scripts {
  @{await Html.RenderPartialAsync("_ValidationScriptsPartial");}
}
UserViewModel.cs
using Microsoft.AspNetCore.Http;
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;

namespace LocalizationDefaultValidation.Models
{
  public class UserViewModel
  {
    [Required]
    [Display(Name = "ID (半角英数字)")]
    [StringLength(20)]
    [RegularExpression(@"^[0-9a-zA-Z]*$")]
    public string ID { get; set; }

    [StringLength(50)]
    public string Name { get; set; }

    [StringLength(50, MinimumLength = 8)]
    [DataType(DataType.Password)]
    [RegularExpression(@"^[0-9a-zA-Z]*$")]
    public string Password { get; set; }

    [StringLength(50, MinimumLength = 8)]
    [DataType(DataType.Password)]
    [Compare(nameof(Password))]
    public string ConfirmPassword { get; set; }

    [EmailAddress]
    [DataType(DataType.EmailAddress)] // スキャフォールディングするときはコメントアウトする
    public string Email { get; set; }

    //[Int]
    [Range(0, 150)]
    public int Age { get; set; }

    [Required]
    [EnumDataType(typeof(GenderType))]
    public GenderType Gender { get; set; }

    [DataType(DataType.Date)]
    public DateTime Birthday { get; set; }

    [Phone()]
    [DataType(DataType.PhoneNumber)] // スキャフォールディングするときはコメントアウトする
    public string Phone { get; set; }

    [DataType(DataType.PostalCode)]
    public string PostalCode { get; set; }

    [CreditCard()]
    [DataType(DataType.CreditCard)] // スキャフォールディングするときはコメントアウトする
    public string CreditCard { get; set; }

    [DataType(DataType.Currency)]
    public decimal Money { get; set; }

    [DataType(DataType.DateTime)]
    public DateTime StartDateTime { get; set; }

    [DataType(DataType.Time)]
    public TimeSpan WakeUpTime { get; set; }

    [Url]
    [DataType(DataType.Url)] // スキャフォールディングするときはコメントアウトする
    public string Homepage { get; set; }

    [Url]
    [DataType(DataType.ImageUrl)] // スキャフォールディングするときはコメントアウトする
    public string MyImage { get; set; }

    public string MyColor { get; set; }

    [MaxLength(5)]
    public DayOfWeek[] WorkingDays { get; set; }

    [MinLength(3)]
    public DayOfWeek[] VacationDay { get; set; }

    [StringLength(200)]
    [DataType(DataType.MultilineText)]
    public string Comment { get; set; }

    [Display(Name = "ファイル名 (.png)")]
    [FileExtensions(Extensions = "png")]
    public string FileName { get; set; }

    [DataType(DataType.Upload)]
    public List<IFormFile> UploadFile { get; set; }

    public DateTime Month { get; set; }

    public string Search { get; set; }

    [Range(10, 100)]
    public int Range { get; set; }

    public string Week { get; set; }

    [Required]
    public bool IsAccepted { get; set; }
  }

  public enum GenderType
  {
    None,
    Man,
    Woman,
    Other,
  }
}
SharedResource.cs
namespace LocalizationDefaultValidation
{
  // クラス名は作成した .resx のファイル名と同じにする必要がある
  public class SharedResource { }
}

Modelar mensajes de enlace

Si establece el tipo de propiedad de modelo int en o y lo enlaza a la DateTime vista, e intenta registrarlo sin escribir, verá un mensaje como "El valor '' no es válido". Estos mensajes aparecen cuando no se puede enlazar una cadena int vacía, por ejemplo, en el modelo. Debido a que es un tiempo que no se puede enlazar, solo se produce antes de la validación de otros valores y el procesamiento del lado del servidor.

Estos mensajes DefaultModelBindingMessageProvider se definen como .

Startup.cs Agregue a los argumentos del método que se han definido desde el principio services.AddControllersWithViews en , y luego agregue Action . Puede ser multilingüe estableciendo el pasado en el options ModelBindingMessageProvider argumento.

Hay 11 tipos de mensajes que puede establecer, por lo que primero debe definir el mensaje de la siguiente manera: SharedResource.resx El nombre de la clave es opcional. El mensaje predeterminado en inglés en la columna de comentarios es (no es posible incluir comentarios). No tendrá que traducir cada idioma, por lo que debe hacer su propia traducción (el código de ejemplo también incluye el SharedResource.resx traducido).

Comentario de valor de nombre
ModelBinding_AttemptedValueIsInvalid '{0}' es un valor no válido en el {1}. El valor '{0}' no es válido para {1}.
ModelBinding_MissingBindRequiredValue No se especifica el valor de la {0}. No se proporcionó un valor para el parámetro o la propiedad '{0}'.
ModelBinding_MissingKeyOrValue Obligatorio. Se requiere un valor.
ModelBinding_MissingRequestBodyRequiredValue La solicitud debe tener un cuerpo. Se requiere un cuerpo de solicitud no vacío.
ModelBinding_NonPropertyAttemptedValueIsInvalid "{0}" no es válido. El valor '{0}' no es válido.
ModelBinding_NonPropertyUnknownValueIsInvalid El valor no es válido. El valor proporcionado no es válido.
ModelBinding_NonPropertyValueMustBeANumber Se debe especificar el número. El campo debe ser un número.
ModelBinding_UnknownValueIsInvalid El valor del {0} no es válido. El valor proporcionado no es válido para {0}.
ModelBinding_ValueIsInvalid "{0}" no es válido. El valor '{0}' no es válido.
ModelBinding_ValueMustBeANumber {0} debe ser un número. El campo {0} debe ser un número.
ModelBinding_ValueMustNotBeNull Entrada requerida. El valor '{0}' no es válido.

Corrija el inicio.cs de la siguiente manera:

using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.AspNetCore.Localization;
using Microsoft.AspNetCore.Mvc.Razor;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Localization;
using System;
using System.Globalization;
using System.Reflection;

namespace LocalizationDefaultValidation
{
  public class Startup
  {
    // 省略 (初期コード)

    /// <summary>
    /// 検証メッセージローカライズで使用。
    /// </summary>
    private IServiceProvider ServiceProvider { get; set; }

    private IStringLocalizer _localizer = null;
    /// <summary>
    /// 検証メッセージローカライズで使用。
    /// </summary>
    private IStringLocalizer Localizer
      => _localizer ?? (_localizer = ServiceProvider.GetService<IStringLocalizerFactory>()
               .Create(nameof(SharedResource), new AssemblyName(typeof(SharedResource).Assembly.FullName).Name));

    // このメソッドはランタイムによって呼び出されます。 このメソッドを使用して、コンテナーにサービスを追加します。
    public void ConfigureServices(IServiceCollection services)
    {
      services.AddControllersWithViews(options =>
      {
        // 検証メッセージのローカライズで使用
        // モデルバインディング失敗時のエラーメッセージをカスタマイズ
        // サーバ側でモデルに値を格納する際に発生する可能性がある
        
        // メッセージの {0} や {1} を置換するための関数を定義
        static string f1(string f, string a1) => string.Format(f, a1);
        static string f2(string f, string a1, string a2) => string.Format(f, a1, a2);

        // 各メソッドを読んでメッセージを置き換えます
        var mp = options.ModelBindingMessageProvider;
        mp.SetAttemptedValueIsInvalidAccessor((x, y) => f2(Localizer["ModelBinding_AttemptedValueIsInvalid"], x, y));
        mp.SetMissingBindRequiredValueAccessor((x) => f1(Localizer["ModelBinding_MissingBindRequiredValue"], x));
        mp.SetMissingKeyOrValueAccessor(() => Localizer["ModelBinding_MissingKeyOrValue"]);
        mp.SetMissingRequestBodyRequiredValueAccessor(() => Localizer["ModelBinding_MissingRequestBodyRequiredValue"]);
        mp.SetNonPropertyAttemptedValueIsInvalidAccessor((x) => f1(Localizer["ModelBinding_NonPropertyAttemptedValueIsInvalid"], x));
        mp.SetNonPropertyUnknownValueIsInvalidAccessor(() => Localizer["ModelBinding_NonPropertyUnknownValueIsInvalid"]);
        mp.SetNonPropertyValueMustBeANumberAccessor(() => Localizer["ModelBinding_NonPropertyValueMustBeANumber"]);
        mp.SetUnknownValueIsInvalidAccessor((x) => f1(Localizer["ModelBinding_UnknownValueIsInvalid"], x));
        mp.SetValueIsInvalidAccessor((x) => f1(Localizer["ModelBinding_ValueIsInvalid"], x));
        mp.SetValueMustBeANumberAccessor((x) => f1(Localizer["ModelBinding_ValueMustBeANumber"], x));
        mp.SetValueMustNotBeNullAccessor((x) => f1(Localizer["ModelBinding_ValueMustNotBeNull"], x));
      });

      services.AddMvc()
        // ローカライズに必要。Resx ファイルのフォルダパスを指定
        .AddViewLocalization(LanguageViewLocationExpanderFormat.Suffix, opts => { opts.ResourcesPath = "Resources"; })
        // DataAnnotations のローカライズに必要
        .AddDataAnnotationsLocalization(options =>
        {
          // DataAnnotation を使ったときのメッセージは SharedResource に集約する
          options.DataAnnotationLocalizerProvider = (type, factory) => factory.Create(typeof(SharedResource));
        });
    }

    // このメソッドはランタイムによって呼び出されます。 このメソッドを使用して、HTTP要求パイプラインを構成します。
    public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
    {
      // ローカライズで使用するため IServiceProvider をプロパティに保持しておきます。
      ServiceProvider = app.ApplicationServices;

      // 標準の機能で切り替えたい言語を定義します。
      var supportedCultures = new[]
      {
        new CultureInfo("ja"),
        new CultureInfo("en"),
        new CultureInfo("es"),
      };

      // 標準の言語切り替え機能を有効にします。対応しているのは「クエリ文字列」「Cookie」「Accept-Language HTTP ヘッダー」です。
      app.UseRequestLocalization(new RequestLocalizationOptions
      {
        DefaultRequestCulture = new RequestCulture("ja"),
        SupportedCultures = supportedCultures,
        SupportedUICultures = supportedCultures
      });

      // 省略 (初期コード)
    }
  }
}

El punto clave es services.AddControllersWithViews agregar a al método que recibe el option Action , options.ModelBindingMessageProvider Estoy configurando texto localizado para cada método Set en .

El texto traducido IStringLocalizer se recupera de . No hay forma de recibir IStringLocalizer código directo, que es un poco un código redondo. SharedResource.resx Si está generando código a partir de él, obtendrá valores localizados directamente de él.

IStringLocalizer La clave que especifique en SharedResource.resx especifica la clave que agregó a . Configéelo para que coincida con el contenido de cada método Set.

Además, debido a que cada mensaje tiene un argumento de cadena de formato, como , {0}{1} uso Simple con para que pueda reemplazar los valores recibidos. string.Format Func

Localizar mensajes de validación predeterminados

El mensaje de error predeterminado cuando los atributos de las Required propiedades de un modelo se establecen en o así está determinado por el marco y está StringLength básicamente en inglés.

Estos se pueden IValidationAttributeAdapterProvider localizar definiendo una clase derivada de la interfaz.

En primer SharedResource.resx lugar, registre el texto que es multilingüe. El nombre de la clave es arbitrario, pero para simplificar el programa, regístrelo en forma de "Validator_< nombre del atributo de validación>".

Comentario de valor de nombre
Validator_CompareAttribute La {0} y la {1} no coinciden. '{0}' y '{1}' no coinciden.
Validator_CreditCardAttribute {0} no es un número de tarjeta válido. El campo {0} no es un número de tarjeta de crédito válido.
Validator_DataTypeAttribute_Date Introduzca una fecha válida. Introduzca una fecha válida.
Validator_EmailAddressAttribute {0} no es una dirección de correo electrónico válida. El campo {0} no es una dirección de correo electrónico válida.
Validator_FileExtensionsAttribute {0} solo acepta archivos con las siguientes extensiones: : {1} El campo {0} sólo acepta archivos con las siguientes extensiones: {1}
Validator_MaxLengthAttribute {0} debe ser un tipo de cadena o matriz con una longitud máxima de '{1}'. El campo {0} debe ser un tipo de cadena o matriz con una longitud máxima de '{1}'.
Validator_MinLengthAttribute {0} debe ser un tipo de cadena o matriz con una longitud mínima de '{1}'. El campo {0} debe ser un tipo de cadena o matriz con una longitud mínima de '{1}'.
Validator_PhoneAttribute {0} no es un número de teléfono válido. El campo {0} no es un número de teléfono válido.
Validator_RangeAttribute {0} deben variar de {1} a {2}. El campo {0} debe estar entre {1} y {2}.
Validator_RegularExpressionAttribute {0} deben coincidir con la expresión regular «{1}». El campo {0} debe coincidir con la expresión regular «{1}».
Validator_RequiredAttribute {0} requiere. El campo {0} es obligatorio.
Validator_StringLengthAttribute {0} debe estar dentro de {1} dígitos. El campo {0} debe ser una cadena con una longitud máxima de {1}.
Validator_UrlAttribute {0} no es una URL válida. El campo {0} no es una URL http, https o ftp válida y completa.
Validator_StringLengthAttributeWithMin {0} debe tener al menos {2} {1} dígitos. El campo {0} debe ser una cadena con una longitud máxima de {1} y una longitud mínima de {2}.

A continuación, cree la siguiente clase AdapterProvider:

El nombre de la clase es arbitrario, pero esta vez CustomValidationAttributeAdapterProvider escribiré una clase llamada y escribiré el código de la siguiente manera: La ubicación del archivo de código es arbitraria, pero el ejemplo se coloca en una carpeta denominada Adaptador.

using Microsoft.AspNetCore.Mvc.DataAnnotations;
using Microsoft.Extensions.Localization;
using System;
using System.ComponentModel.DataAnnotations;

namespace LocalizationDefaultValidation
{
  public class CustomValidationAttributeAdapterProvider : IValidationAttributeAdapterProvider
  {
    /// <summary>コード簡略化のためのリソースキー命名規則プリフィックス</summary>
    private const string RESOURCE_KEY_PREFIX = "Validator_";

    private readonly IValidationAttributeAdapterProvider _fallback = new ValidationAttributeAdapterProvider();

    /// <summary>
    /// 指定された ValidationAttribute の IAttributeAdapter を返します。
    /// </summary>
    /// <param name="attribute">IAttributeAdapter を作成するための ValidationAttribute。</param>
    /// <param name="stringLocalizer">メッセージの作成に使用される IStringLocalizer。</param>
    /// <returns>指定された属性の IAttributeAdapter。</returns>
    IAttributeAdapter IValidationAttributeAdapterProvider.GetAttributeAdapter(ValidationAttribute attribute, IStringLocalizer stringLocalizer)
    {
      // すでにエラーメッセージが設定されている場合はそれを使用するのでここでは何も設定しない
      if (attribute.ErrorMessageResourceName != null) return _fallback.GetAttributeAdapter(attribute, stringLocalizer);

      // attribute には「Required」や「StringLength」などが設定されています
      Type attrType = attribute.GetType();
      
      // プリフィックスと属性のクラス名からローカライズ用のキーを生成します
      var key = RESOURCE_KEY_PREFIX + attrType.Name;

      // IStringLocalizer から指定したキーでローカライズされたテキストを取得します
      // ない場合は getString にそのまま key の値が入ります
      var getString = stringLocalizer[key];

      // 正しくローカライズされたテキストが取得できた場合は ErrorMessage に値をセットします。
      if (key != getString && attribute.ErrorMessage != getString)
      {
        attribute.ErrorMessage = getString;
      }

      // 設定した attribute を渡します
      return _fallback.GetAttributeAdapter(attribute, stringLocalizer);
    }
  }
}

Si establece las propiedades del modelo en o , se llama al Required método para cada StringLength IValidationAttributeAdapterProvider.GetAttributeAdapter validación.

ErrorMessageResourceName Si la propiedad contiene un valor, el código del lado del modelo ya tiene un mensaje o una clave de localización, por lo que devuelve tal cual.

Si está vacío, haga coincidir los nombres de clase del prefijo y el atributo de validación para convertirlo en la clave de localización y, a continuación, recupere el texto localizado de la clave y estacórelo ErrorMessage en . Esto permite localizar la mayoría de los mensajes predeterminados.

A continuación, registre esta clase en la .cs de inicio. Básicamente, puede agregarlo de la siguiente manera.

// 省略

using Microsoft.AspNetCore.Mvc.DataAnnotations;

namespace LocalizationDefaultValidation
{
  public class Startup
  {
    // 省略

    // このメソッドはランタイムによって呼び出されます。 このメソッドを使用して、コンテナーにサービスを追加します。
    public void ConfigureServices(IServiceCollection services)
    {
      // 省略

      // 作成した CustomValidationAttributeAdapterProvider をシングルトンとして登録します
      services.AddSingleton<IValidationAttributeAdapterProvider, CustomValidationAttributeAdapterProvider>();
    }
    
    // 省略
  }
}

Ejecutarlo para asegurarse de que funciona correctamente.

Cambiar un mensaje existente por parámetros

Por ejemplo, StringLength el mensaje de atributo está localizado, como "{0} debe estar dentro de {1} dígitos". MinimumLength Si la propiedad está establecida, es posible que desee cambiar el {0} como "Especificar {2} dígitos o más {1} dentro de un número de dígitos".

En ese CustomValidationAttributeAdapterProvider caso, amplíe la clase de la siguiente manera:

// 省略

namespace LocalizationDefaultValidation
{
  public class CustomValidationAttributeAdapterProvider : IValidationAttributeAdapterProvider
  {
    // 省略

    IAttributeAdapter IValidationAttributeAdapterProvider.GetAttributeAdapter(ValidationAttribute attribute, IStringLocalizer stringLocalizer)
    {
      if (attribute is StringLengthAttribute slAttribute)
      {
        // attribute が StringLengthAttribute で MinimumLength が設定されている場合はメッセージを変える
        if (slAttribute.MinimumLength >= 1)
        {
          attribute.ErrorMessage = stringLocalizer["Validator_StringLengthAttributeWithMin"];
          return _fallback.GetAttributeAdapter(slAttribute, stringLocalizer);
        }
      }

      // 省略
    }
  }
}

Dado que se llama al método para cada GetAttributeAdapter validación, si la variable pasada attribute fue StringLengthAttribute un MinimumLength atributo, compruebe la propiedad, Si está configurado, obtenga un mensaje que también tenga en cuenta el número mínimo de dígitos y ErrorMessage reemplace .

Si lo ejecuta, comprueba y el mensaje cambia, está bien.

Otros mensajes genéricos

Algunos mensajes no se localizan después de hacerlo hasta ahora. Por ejemplo, mensajes que se determinan solo en Javascript.

El destino es jquery.validate.js el mensaje que aparece en el archivo.

Debido a que este archivo no debe editarse directamente, para reemplazar estos mensajes con texto localizado, input debe agregar un atributo como a la etiqueta y establecer texto localizado data-val-XXXX allí. La parte XXXX contiene la clave de la figura jquery.validate.js anterior. (por ejemplo, data-val-number etc.)

data-val-XXXX Para establecer texto localizado en atributos en

Vamos a localizar el mensaje cuando se incluyen caracteres no numéricos en el campo de entrada Edad como ejemplo. UserViewModel.Age tiene Range un atributo y el HTML de salida data-val-range anexa el atributo de entrada.

<div class="form-group">
  <label class="control-label" for="Age">Age</label>
  <input class="form-control" type="number" data-val="true" data-val-range="Ageは0から150の範囲で指定してください。" data-val-range-max="150" data-val-range-min="0" data-val-required="Ageは必須です。" id="Age" name="Age" value="" />
  <span class="text-danger field-validation-valid" data-valmsg-for="Age" data-valmsg-replace="true"></span>
</div>

Sin embargo, data-val-number el mensaje de comprobación numérica no está localizado porque no hay ningún atributo. También agregará otro atributo en el lado del modelo para imprimir el atributo agregado de data-val-number mensaje localizado.

El primer paso es crear una clase de atributo que compruebe si es IntAttribute un número: Se limita a aquí, pero usted es int libre de cambiarlo dependiendo de la situación. El archivo de código puede estar en cualquier lugar, pero esta vez está en la carpeta Adaptador.

using System.ComponentModel.DataAnnotations;

namespace LocalizationDefaultValidation
{
  public class IntAttribute : ValidationAttribute
  {
    public override bool IsValid(object value)
    {
      // 渡された値が int であれば有効な値とする
      return value is int;
    }
  }
}

Si el valor establecido es int un tipo, es un atributo que es solo un veredicto normal, pero en la práctica es 100% normal si se establece en una propiedad de tipo int, por lo que el procesamiento de esta clase en sí no tiene ningún sentido. El propósito de este tiempo es mostrar mensajes de error localizados en el cliente.

A continuación, cree la siguiente clase de IntAttributeAdapter adaptador: Esto también se encuentra en la carpeta Adaptador.

using Microsoft.AspNetCore.Mvc.DataAnnotations;
using Microsoft.AspNetCore.Mvc.ModelBinding.Validation;
using Microsoft.Extensions.Localization;

namespace LocalizationDefaultValidation
{
  /// <summary>Int 属性のアダプター。</summary>
  public class IntAttributeAdapter : AttributeAdapterBase<IntAttribute>
  {
    /// <summary>受け取った IStringLocalizer を保持しておく</summary>
    IStringLocalizer _stringLocalizer;

    public IntAttributeAdapter(IntAttribute attribute, IStringLocalizer stringLocalizer)
        : base(attribute, stringLocalizer)
    {
      _stringLocalizer = stringLocalizer;
    }

    /// <summary>バリデーションの追加処理として呼ばれる。</summary>
    public override void AddValidation(ClientModelValidationContext context)
    {
      // 新たに data-val-number 属性をマージして追加します。値はローカライズしたテキストをセットします。
      MergeAttribute(context.Attributes, "data-val-number", _stringLocalizer["ModelBinding_NonPropertyValueMustBeANumber"]);
    }

    /// <summary>サーバーのエラーメッセージはそのまま返します。</summary>
    public override string GetErrorMessage(ModelValidationContextBase validationContext) => Attribute.ErrorMessage;
  }
}

El punto aquí es AddValidation el método descrito en el MergeAttribute método.

Combinar atributos como atributos de la etiqueta de entrada para volver data-val-number al cliente. El valor que se establece en el atributo establece un mensaje de error localizado.

Devuelva este adaptador en la clase que creó CustomValidationAttributeAdapterProvider anteriormente.

// 省略

namespace LocalizationDefaultValidation
{
  public class CustomValidationAttributeAdapterProvider : IValidationAttributeAdapterProvider
  {
    // 省略
    
    IAttributeAdapter IValidationAttributeAdapterProvider.GetAttributeAdapter(ValidationAttribute attribute, IStringLocalizer stringLocalizer)
    {
      // IntAttribute の場合は IntAttributeAdapter 経由で返す
      if (attribute is IntAttribute intAttribute)
      {
        return new IntAttributeAdapter(intAttribute, stringLocalizer);
      }

      // 省略 (前に追加したコード)
    }
  }
}

Si el valor que se va a validar IntAttribute es , devuelva el que creó IntAttributeAdapter anteriormente. La propiedad con el atributo Int ahora se agrega cuando se muestra en el lado del data-val-number cliente.

Finalmente, UserViewModel.Age IntAttribute agregue a .

Una vez que hayas arreglado tu código, pruébalo para ver si quieres comprobarlo.

<div class="form-group">
  <label class="control-label" for="Age">Age</label>
  <input class="form-control" type="number" data-val="true" data-val-number="数字を指定してください。" data-val-range="Ageは0から150の範囲で指定してください。" data-val-range-max="150" data-val-range-min="0" data-val-required="Ageは必須です。" id="Age" name="Age" value="" />
  <span class="text-danger field-validation-valid" data-valmsg-for="Age" data-valmsg-replace="true"></span>
</div>

Hay otros mensajes no localizados, pero puede localizarlos en los métodos descritos anteriormente. Agregue código cuando lo necesite.