Многоязычные URL-адреса asp.net mvc/routing
Это вопрос из двух частей, относящийся к мультиязычным URL-адресам asp.net mvc и поисковым методам и преимуществам SEO...
Вопрос Часть 1)
Мне предложили создать новый веб-сайт ASP.NET MVC, который будет поддерживать минимум (сначала) двух языков (английский и французский), возможно, в будущем, 3 языка...
Что касается локализации приложения (метки, ошибки jQuery и т.д.), все должно быть в порядке с использованием файлов ресурсов, и Ive нашел много примеров по этому вопросу... но мое беспокойство/вопрос больше о URL-адресах.
Что касается SEO, каков рекомендуемый подход между этими двумя модами?
Fashion 1 (no culture folder)
www.mydomain.com/create-account
www.mydomain.com/creer-un-compte
Fashion 2 (with built in culture folder)
www.mydomain.com/create-account
www.mydomain.com/fr/creer-un-compte <--notice the "fr" folder
Есть ли известная проблема/штраф при использовании одного над другим?
Или он настолько мал, что становится неуместным!
Часть вопроса 2)
Чтобы достичь моды 2, Ive уже нашел статью здесь:
ASP.NET MVC - маршрут локализации
Но Id будет любопытно найти, как достичь моды 1.
Есть ли у кого-нибудь ссылки?
Кроме того, насколько я знаю, URL-реферирование не, что я ищу, поскольку я не хочу "перенаправлять" пользователей... Я просто хочу, чтобы URL-адреса отображались на соответствующем языке без необходимости показывать культуру в URL-адресе
Заранее благодарим за помощь!
Ответы
Ответ 1
вы можете создать базовый контроллер с логикой локализации, как показано ниже:
public abstract class LocalizedController : Controller
{
protected override void ExecuteCore()
{
HttpCookie cookie;
string lang = GetCurrentCulture();
Thread.CurrentThread.CurrentUICulture = new CultureInfo(lang, false);
// set the lang value into route data
RouteData.Values["lang"] = lang;
// save the location into cookie
cookie = new HttpCookie("DPClick.CurrentUICulture",
Thread.CurrentThread.CurrentUICulture.TwoLetterISOLanguageName)
{
Expires = DateTime.Now.AddYears(1)
};
HttpContext.Response.SetCookie(cookie);
base.ExecuteCore();
}
private string GetCurrentCulture()
{
string lang;
// set the culture from the route data (url)
if (RouteData.Values["lang"] != null &&
!string.IsNullOrWhiteSpace(RouteData.Values["lang"].ToString()))
{
lang = RouteData.Values["lang"].ToString();
if (Localization.Locales.TryGetValue(lang, out lang))
{
return lang;
}
}
// load the culture info from the cookie
HttpCookie cookie = HttpContext.Request.Cookies["DPClick.CurrentUICulture"];
if (cookie != null)
{
// set the culture by the cookie content
lang = cookie.Value;
if (Localization.Locales.TryGetValue(lang, out lang))
{
return lang;
}
}
// set the culture by the location if not speicified
lang = HttpContext.Request.UserLanguages[0];
if (Localization.Locales.TryGetValue(lang, out lang))
{
return lang;
}
//English is default
return Localization.Locales.FirstOrDefault().Value;
}
}
Вышеуказанный контроллер удовлетворяет моде 2 вашего вопроса, если вы хотите игнорировать папку с культурой, просто не назначайте lang в RouteDate. для достижения моды 2 вам необходимо добавить маршрутизацию для культуры, как показано ниже:
routes.MapRoute(
"Localization", // Route name
"{lang}/{controller}/{action}/{id}", // URL with parameters
new {controller = "Default", action = "Index", id = UrlParameter.Optional}, // Parameter defaults
new {lang = @"\w{2,3}(-\w{4})?(-\w{2,3})?"}
);
Ответ 2
AttributeRouting может решить Fashion 1:
AttributeRouting - Локализация
Ответ 3
Чтобы достичь желаемого, вам в основном нужно реализовать три вещи:
Маршрут с поддержкой нескольких языков для обработки входящих URL-адресов:
routes.MapRoute(
name: "DefaultLocalized",
url: "{lang}/{controller}/{action}/{id}",
constraints: new { lang = @"(\w{2})|(\w{2}-\w{2})" }, // en or en-US
defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional }
);
ЛокализацияAttribute для обработки этих многоязычных запросов:
public class LocalizationAttribute : ActionFilterAttribute
{
private string _DefaultLanguage = "en";
public LocalizationAttribute(string defaultLanguage)
{
_DefaultLanguage = defaultLanguage;
}
public override void OnActionExecuting(ActionExecutingContext filterContext)
{
string lang = (string)filterContext.RouteData.Values["lang"] ?? _DefaultLanguage;
if (lang != _DefaultLanguage)
{
try
{
Thread.CurrentThread.CurrentCulture =
Thread.CurrentThread.CurrentUICulture = new CultureInfo(lang);
}
catch (Exception e)
{
throw new NotSupportedException(String.Format("ERROR: Invalid language code '{0}'.", lang));
}
}
}
}
Вспомогательный метод для создания этих URL в вашем приложении: это можно сделать несколькими способами, в зависимости от вашей логики приложения. Например, если вам нужно сделать это в своих Razor Views, самое лучшее, что вы можете сделать, это написать пару методов расширения, чтобы ваши Html.ActionLink
и Url.Action
принять объект CultureInfo
в качестве параметра (и/или используйте CultureInfo.CurrentCulture
как значение по умолчанию), например следующее:
(оба написаны на С#)
Вы также можете избежать шаблона метода расширения и вместо этого записать их как MultiLanguageActionLink
/MultiLanguageAction
.
Для получения дополнительной информации и дополнительных примеров по этой теме вы также можете прочитать этот пост.