Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Route localization in ASP.NET Core 2.2

I am developing application using ASP.NET Core 2.2 and I am struggling with how to implement route localization, ex. depending on request I need to redirect to route /en/products, if language is not specified in the route. If language is not specified then get locale from accept-language header.

like image 953
Anton M. Avatar asked Dec 22 '22 23:12

Anton M.


1 Answers

Below demo is applied to use twoLetterLanguageName.Refer to this tutorial

1.Create a RouteDataRequestCultureProvider class:

public class RouteDataRequestCultureProvider : RequestCultureProvider
{
    public int IndexOfCulture;
    public int IndexofUICulture;

    public override Task<ProviderCultureResult> DetermineProviderCultureResult(HttpContext httpContext)
    {
        if (httpContext == null)
            throw new ArgumentNullException(nameof(httpContext));

        string culture = null;
        string uiCulture = null;

        var twoLetterCultureName = httpContext.Request.Path.Value.Split('/')[IndexOfCulture]?.ToString();
        var twoLetterUICultureName = httpContext.Request.Path.Value.Split('/')[IndexofUICulture]?.ToString();

        if (twoLetterCultureName == "de")
            culture = "de-DE";
        else if (twoLetterCultureName == "en")
            culture = uiCulture = "en-US";

        if (twoLetterUICultureName == "de")
            culture = "de-DE";
        else if (twoLetterUICultureName == "en")
            culture = uiCulture = "en-US";

        if (culture == null && uiCulture == null)
            return NullProviderCultureResult;

        if (culture != null && uiCulture == null)
            uiCulture = culture;

        if (culture == null && uiCulture != null)
            culture = uiCulture;

        var providerResultCulture = new ProviderCultureResult(culture, uiCulture);

        return Task.FromResult(providerResultCulture);
    }
}

2.And a LanguageRouteConstraintclass

public class LanguageRouteConstraint : IRouteConstraint
{
    public bool Match(HttpContext httpContext, IRouter route, string routeKey, RouteValueDictionary values, RouteDirection routeDirection)
    {

        if (!values.ContainsKey("culture"))
            return false;

        var culture = values["culture"].ToString();
        return culture == "en" || culture == "de";
    }
}

3.startup.cs ConfigureServices:

services.AddLocalization(options => options.ResourcesPath = "Resources");

services.AddMvc()
        .AddViewLocalization(LanguageViewLocationExpanderFormat.Suffix)
        .AddDataAnnotationsLocalization();

services.Configure<RequestLocalizationOptions>(options =>
{
    var supportedCultures = new List<CultureInfo>
    {
        new CultureInfo("en"),
        new CultureInfo("de"),
    };

    options.DefaultRequestCulture = new RequestCulture(culture: "en", uiCulture: "en-US");
    options.SupportedCultures = supportedCultures;
    options.SupportedUICultures = supportedCultures;

    options.RequestCultureProviders = new[]{ new RouteDataRequestCultureProvider{
        IndexOfCulture=1,
        IndexofUICulture=1
    }};

});

services.Configure<RouteOptions>(options =>
{
    options.ConstraintMap.Add("culture", typeof(LanguageRouteConstraint));
});

4.startup.cs Configure

public void Configure(IApplicationBuilder app, IHostingEnvironment env)
    {
        var options = app.ApplicationServices.GetService<IOptions<RequestLocalizationOptions>>();
        app.UseRequestLocalization(options.Value);

        //other middlewares

        app.UseMvc(routes =>
        {
            routes.MapRoute(
                   name: "LocalizedDefault",
                   template: "{culture:culture}/{controller}/{action}/{id?}",
                   defaults: new {controller = "Home", action = "Index" });
            //constraints: new { culture = new CultureConstraint(defaultCulture: "en", pattern: "[a-z]{2}") });

            routes.MapRoute(
                 name: "default",
                 template: "{controller}/{action}/{id?}",
                 defaults: new { controller = "Home", action = "Index" });

        });
    }

Then you could change culture in browser url directly using /en/Home/Privacy.

If language is not specified then get locale from accept-language header

You could use Url Rewriting Middleware to check the route value and redirect it to a new route with default culture.

1.Create a Redirect rule:

public class RewriteRules
{
    public static void RedirectRequests(RewriteContext context)
    {
        //Your logic

        var request = context.HttpContext.Request;
        var path = request.Path.Value;

        var userLangs = request.Headers["Accept-Language"].ToString();
        var firstLang = userLangs.Split(',').FirstOrDefault();
        var defultCulture = string.IsNullOrEmpty(firstLang) ? "en" : firstLang.Substring(0,2);

        //Add your conditions of redirecting
        if ((path.Split("/")[1] != "en") && (path.Split("/")[1] != "de"))// If the url does not contain culture
        {
            context.HttpContext.Response.Redirect($"/{defultCulture}{ request.Path.Value }");
        }

    }
}

2..Use the middleware in Startup Configure method:

app.UseAuthentication();//before the Rewriter middleware

app.UseRewriter(new RewriteOptions()
        .Add(RewriteRules.RedirectRequests)
        );
//MVC middleware

Then if your input /Home/Privacy in browser it will redirect to url like /en/Home/Privacy.

like image 144
Ryan Avatar answered Jan 11 '23 11:01

Ryan