Xtensive.Orm.Localization
7.1.3
See the version list below for details.
dotnet add package Xtensive.Orm.Localization --version 7.1.3
NuGet\Install-Package Xtensive.Orm.Localization -Version 7.1.3
<PackageReference Include="Xtensive.Orm.Localization" Version="7.1.3" />
paket add Xtensive.Orm.Localization --version 7.1.3
#r "nuget: Xtensive.Orm.Localization, 7.1.3"
// Install Xtensive.Orm.Localization as a Cake Addin #addin nuget:?package=Xtensive.Orm.Localization&version=7.1.3 // Install Xtensive.Orm.Localization as a Cake Tool #tool nuget:?package=Xtensive.Orm.Localization&version=7.1.3
Xtensive.Orm.Localization
Summary
The extension transparently solves a task of application or service localization. This implies that localizable resources are a part of domain model so they are stored in database.
Prerequisites
DataObjects.Net 7.1.x or later (http://dataobjects.net)
Implementation
Implement ILocalizable<TLocalization> on your localizable entities, e.g.:
[HierarchyRoot]
public class Page : Entity, ILocalizable<PageLocalization>
{
[Field, Key]
public int Id { get; private set; }
// Localizable field. Note that it is non-persistent
public string Title
{
get { return Localizations.Current.Title; }
set { Localizations.Current.Title = value; }
}
[Field] // This is a storage of all localizations for Page class
public LocalizationSet<PageLocalization> Localizations { get; private set; }
public Page(Session session) : base(session) {}
}
Define corresponding localizations, e.g.:
[HierarchyRoot]
public class PageLocalization : Localization<Page>
{
[Field(Length = 100)]
public string Title { get; set; }
public PageLocalization(Session session, CultureInfo culture, Page target)
: base(session, culture, target) {}
}
Examples of usage
Example #1. Access localizable properties as regular ones, e.g.:
page.Title = "Welcome";
string title = page.Title;
Example #2. Mass editing of localizable properties:
var en = new CultureInfo("en-US");
var sp = new CultureInfo("es-ES");
var page = new Page(session);
page.Localizations[en].Title = "Welcome";
page.Localizations[sp].Title = "Bienvenido";
Example #3. Value of localizable properties reflects culture of the current Thread, e.g.:
Thread.CurrentThread.CurrentCulture = new CultureInfo("en-US");
string title = page.Title; // title is "Welcome"
Thread.CurrentThread.CurrentCulture = new CultureInfo("es-ES");
string title = page.Title; // title is "Bienvenido"
Example #4. Instead of altering CurrentThread, instance of LocalizationScope can be used, e.g.:
using (new LocalizationScope(new CultureInfo("en-US"))) {
string title = page.Title; // title is "Welcome"
}
using (new LocalizationScope(new CultureInfo("es-ES"))) {
string title = page.Title; // title is "Bienvenido"
}
Example #5. LINQ queries that include localizable properties are transparently translated
Thread.CurrentThread.CurrentCulture = new CultureInfo("en-US");
var query = from p in session.Query.All<Page>()
where p.Title=="Welcome"
select p;
Assert.AreEqual(1, query.Count());
Thread.CurrentThread.CurrentCulture = new CultureInfo("es-ES");
var query = from p in session.Query.All<Page>()
where p.Title=="Bienvenido"
select p;
Assert.AreEqual(1, query.Count());
Examples of how to configure extension
Following examples show different ways to configure extension in configuration files of various types.
Example #1 Confugure default culture in App.config/Web.config
<configuration>
<configSections>
<section name="Xtensive.Orm" type="Xtensive.Orm.Configuration.Elements.ConfigurationSection, Xtensive.Orm"/>
<section name="Xtensive.Orm.Localization" type="Xtensive.Orm.Localization.Configuration.ConfigurationSection, Xtensive.Orm.Localization"/>
</configSections>
<Xtensive.Orm>
</Xtensive.Orm>
<Xtensive.Orm.Localization>
<defaultCulture name="es-ES"/>
</Xtensive.Orm.Localization>
</configuration>
Such configuration is usually read with System.Configuration.ConfigurationManager
.
If project still supports such configurations then Localization configuration will be read automatically when it needs to be read.
Sometimes a work-around is needed to read such configuration, for more read Example #2 and Example #3
Example #2 Reading old-style configuration of an assembly in NET 5 and newer.
Due to new architecture without AppDomain (which among the other things was in charge of gathering configuration files of loaded assemblies
as it would be one configuration file) System.Configuration.ConfigurationManager
now reads only configuration file of actual executable, loaded
assemblies' configuration files stay unreachable by default, though there is need to read some data from them.
A great example is test projects which are usually get loaded by test runner executable, and the only configuration accessible in this case
is test runner one.
Extra step is required to read configuration files in such cases. Thankfully, ConfigurationManager
has methods to get access to assemblies' configuration files.
To get access to an assembly configuration file it should be opened explicitly by
var configuration = ConfigurationManager.OpenExeConfiguration(typeof(SomeTypeInConfigOwnerAssembly).Assembly.Location);
The instance returned from OpenExeConfiguration
provides access to sections of the assembly configuration. DataObjects.Net configurations
(DomainConfiguration
, LocalizationConfiguration
, etc.) have Load()
methods that can recieve this instance.
LocalizationConfiguration
can be read like so
var configuration = ConfigurationManager.OpenExeConfiguration(typeof(SomeTypeInConfigOwnerAssembly).Assembly.Location);
var localizationConfig = LocalizationConfiguration.Load(configuration);
// loaded configuration should be manually placed to
domainConfiguration.ExtensionConfigurations.Set(localizationConfig);
The domainConfiguration.ExtensionConfigurations
is a new unified place from which the extension will try to get its configuration
instead of calling default parameterless Load()
method, which has not a lot of sense now, though the method is kept as a second source
for backwards compatibility.
For more convenience, DomainConfiguration
extensions are provided, which make code neater.
For instance,
var configuration = ConfigurationManager.OpenExeConfiguration(typeof(SomeTypeInConfigOwnerAssembly).Assembly.Location);
// the extension hides getting configuration with LocalizationConfiguration.Load(configuration)
// and also putting it to ExtensionConfigurations collection.
domainConfiguration.ConfigureLocalizationExtension(configuration);
Custom section names are also supported if for some reason default section name is not used.
Example #3 Reading old-style configuration of an assembly in a project that uses appsettings.json file.
If for some reason there is need to keep the old-style configuration then there is a work-around as well.
Static configuration manager provides method OpenMappedExeConfiguration()
which allows to get
any *.config file as System.Configuration.Configuration
instance. For example,
ExeConfigurationFileMap configFileMap = new ExeConfigurationFileMap();
configFileMap.ExeConfigFilename = "Orm.config"; //or other file name, the file should exist bin folder
var configuration = System.Configuration.ConfigurationManager.OpenMappedExeConfiguration(configFileMap, ConfigurationUserLevel.None);
After that, as in previous example, the instance can be passed to Load
method of LocalizationConfiguration
to read extension configuration
and later put it to DomainConfiguration.ExtensionConfigurations
var localizationConfiguration = LocalizationConfiguration.Load(configuration);
domainConfiguration.ExtensionConfigurations.Set(localizationConfiguration);
Extension usage will look like
domainConfiguration.ConfigureLocalizationExtension(configuration);
Example #4 Configure using Microsoft.Extensions.Configuration API.
This API allows to have configurations in various forms including JSON and XML formats. Loading of such files may differ depending on .NET version, check Microsoft manuals for instructions.
Allowed JSON and XML configuration definition look like below
<configuration>
<Xtensive.Orm.Localization>
<DefaultCulture>es-ES</DefaultCulture>
</Xtensive.Orm.Localization>
</configuration>
{
"Xtensive.Orm.Localization": {
"DefaultCulture": "es-ES"
}
}
The API has certain issues with XML elements with attributes so it is recommended to use more up-to-date attributeless nodes. For JSON it is pretty clear, almost averyone knows its format.
LocalizationConfiguration.Load
method can accept different types of abstractions from the API, including
Microsoft.Extensions.Configuration.IConfiguration
;Microsoft.Extensions.Configuration.IConfigurationRoot
;Microsoft.Extensions.Configuration.IConfigurationSection
.
Loading of configuration may look like
var app = builder.Build();
//...
// tries to load from default section "Xtensive.Orm.Localization"
var localizationConfig = LocalizationConfiguration.Load(app.Configuration);
domainConfiguration.ExtensionConfigurations.Set(localizationConfig);
or, with use of extension, like
var app = builder.Build();
//...
// tries to load from default section "Xtensive.Orm.Localization"
// and additionally adds Xtensive.Orm.Localization assembly to domain types.
domainConfiguration.ConfigureLocalizationExtension(app.Configuration);
Example #5 Configure using Microsoft.Extensions.Configuration API from section with non-default name.
For configurations like
<configuration>
<Orm.Localization>
<DefaultCulture>es-ES</DefaultCulture>
</Orm.Localization>
</configuration>
{
"Orm.Localization": {
"DefaultCulture": "es-ES"
}
}
Loading of configuration may look like
var app = builder.Build();
//...
var localizationConfig = LocalizationConfiguration.Load(app.Configuration, "Orm.Localization");
domainConfiguration.ExtensionConfigurations.Set(localizationConfig);
or, with use of extension, like
var app = builder.Build();
domainConfiguration.ConfigureLocalizationExtension(app.Configuration, "Orm.Localization");
Example #6 Configure using Microsoft.Extensions.Configuration API from sub-section deeper in section tree.
If for some reason extension configuration should be moved deeper in section tree like something below
<configuration>
<Orm.Extensions>
<Xtensive.Orm.Localization>
<DefaultCulture>es-ES</DefaultCulture>
</Xtensive.Orm.Localization>
</Orm.Extensions>
</configuration>
or in JSON
{
"Orm.Extensions": {
"Xtensive.Orm.Localization": {
"DefaultCulture": "es-ES"
}
}
}
Then section must be provided manually, code may look like
var app = builder.Build();
//...
var configurationRoot = app.Configuration;
var extensionsGroupSection = configurationRoot.GetSection("Orm.Extensions");
var localizationSection = extensionsGroupSection.GetSection("Xtensive.Orm.Localization");
var localizationConfig = LocalizationConfiguration.Load(localizationSection);
domainConfiguration.ExtensionConfigurations.Set(localizationConfig);
or, with use of extension method, like
var app = builder.Build();
//...
var configurationRoot = app.Configuration;
var extensionsGroupSection = configurationRoot.GetSection("Orm.Extensions");
var localizationSection = extensionsGroupSection.GetSection("Xtensive.Orm.Localization");
domainConfiguration.ConfigureLocalizationExtension(localizationSection);
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 is compatible. net5.0-windows was computed. net6.0 is compatible. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
-
net5.0
- Microsoft.Extensions.Configuration.Abstractions (>= 6.0.0)
- Microsoft.Extensions.Configuration.Binder (>= 6.0.0)
- System.Configuration.ConfigurationManager (>= 5.0.0)
- Xtensive.Orm (>= 7.1.3)
-
net6.0
- Microsoft.Extensions.Configuration.Abstractions (>= 6.0.0)
- Microsoft.Extensions.Configuration.Binder (>= 6.0.0)
- System.Configuration.ConfigurationManager (>= 6.0.1)
- Xtensive.Orm (>= 7.1.3)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
7.2.0-Beta-1 | 106 | 12/28/2023 |
7.1.3 | 101 | 12/24/2024 |
7.1.2 | 126 | 10/18/2024 |
7.1.1 | 207 | 11/14/2023 |
7.1.0 | 337 | 4/12/2023 |
7.1.0-RC | 145 | 3/9/2023 |
7.1.0-Beta-2 | 215 | 12/19/2022 |
7.1.0-Beta-1 | 163 | 7/4/2022 |
7.0.6 | 86 | 12/19/2024 |
7.0.5 | 124 | 6/3/2024 |
7.0.4 | 101 | 11/12/2023 |
7.0.3 | 522 | 3/21/2022 |
7.0.2 | 491 | 2/8/2022 |
7.0.1 | 365 | 10/29/2021 |
7.0.0 | 373 | 6/2/2021 |
6.0.14 | 85 | 12/17/2024 |
6.0.13 | 135 | 4/4/2024 |
6.0.12 | 114 | 11/10/2023 |
6.0.11 | 312 | 1/12/2023 |
6.0.10 | 459 | 4/29/2022 |
6.0.9 | 508 | 2/2/2022 |
6.0.8 | 426 | 10/28/2021 |
6.0.7 | 382 | 8/27/2021 |
6.0.6 | 428 | 5/24/2021 |
6.0.5 | 450 | 3/9/2021 |
6.0.4 | 496 | 12/22/2020 |
6.0.3 | 509 | 9/29/2020 |
6.0.0 | 671 | 1/28/2020 |
5.1.0-Beta-1 | 1,044 | 1/30/2015 |
5.0.24 | 415 | 4/27/2021 |
5.0.23 | 424 | 2/4/2021 |
5.0.22 | 542 | 11/18/2020 |
5.0.21 | 527 | 11/6/2020 |
5.0.20 | 649 | 12/25/2019 |
5.0.19 | 659 | 5/30/2019 |
5.0.19-Beta-2 | 444 | 4/16/2019 |
5.0.19-Beta-1 | 529 | 12/29/2018 |
5.0.18 | 875 | 9/28/2018 |
5.0.18-Beta-3 | 774 | 7/2/2018 |
5.0.18-Beta-2 | 694 | 6/6/2018 |
5.0.18-Beta-1 | 770 | 4/24/2018 |
5.0.17 | 958 | 2/27/2018 |
5.0.17-Beta-3 | 691 | 2/12/2018 |
5.0.17-Beta-2 | 770 | 1/12/2018 |
5.0.17-Beta-1 | 783 | 12/28/2017 |
5.0.16 | 1,043 | 12/1/2017 |
5.0.16-Beta-1 | 727 | 9/27/2017 |
5.0.15 | 985 | 8/1/2017 |
5.0.14 | 1,032 | 6/19/2017 |
5.0.13 | 1,024 | 3/22/2017 |
5.0.12 | 1,021 | 2/14/2017 |
5.0.11 | 991 | 1/25/2017 |
5.0.11-RC2 | 813 | 12/16/2016 |
5.0.11-RC | 1,119 | 9/20/2016 |
5.0.10 | 1,077 | 8/5/2016 |
5.0.10-RC | 812 | 6/30/2016 |
5.0.9 | 1,156 | 3/3/2016 |
5.0.8 | 1,072 | 2/15/2016 |
5.0.7 | 1,073 | 1/27/2016 |
5.0.7-RC2 | 812 | 12/8/2015 |
5.0.7-RC | 873 | 9/10/2015 |
5.0.6 | 1,171 | 7/3/2015 |
5.0.5 | 1,243 | 4/23/2015 |
5.0.4 | 1,128 | 3/19/2015 |
5.0.4-RC | 1,291 | 2/25/2015 |
5.0.3 | 1,825 | 10/31/2014 |
5.0.2 | 1,127 | 9/11/2014 |
5.0.0 | 1,226 | 8/15/2014 |
5.0.0-RC2 | 936 | 8/1/2014 |
5.0.0-RC | 907 | 7/21/2014 |
5.0.0-Beta-3 | 885 | 5/28/2014 |
5.0.0-Beta-2 | 949 | 2/28/2014 |
5.0.0-Beta-1 | 972 | 11/14/2013 |
4.6.9 | 1,109 | 7/3/2015 |
4.6.8 | 1,156 | 8/1/2014 |
4.6.7 | 1,238 | 6/23/2014 |
4.6.6 | 1,377 | 4/9/2014 |
4.6.5 | 1,196 | 1/7/2014 |
4.6.4 | 1,345 | 9/30/2013 |
4.6.3 | 1,322 | 2/4/2013 |
4.6.2 | 1,469 | 11/28/2012 |
4.6.0 | 1,442 | 10/11/2012 |
4.6.0-RC | 1,106 | 10/4/2012 |
4.5.8 | 1,194 | 9/30/2013 |
4.5.7 | 1,313 | 2/4/2013 |
4.5.6 | 1,361 | 11/28/2012 |
4.5.5 | 1,395 | 10/11/2012 |
4.5.5-RC | 1,092 | 10/4/2012 |
4.5.3 | 1,392 | 8/6/2012 |
4.5.2 | 1,505 | 5/10/2012 |
4.5.0 | 1,441 | 3/13/2012 |