402 lines
16 KiB
C#
402 lines
16 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using Umbraco.Core.Auditing;
|
|
using Umbraco.Core.Configuration;
|
|
using Umbraco.Core.Events;
|
|
using Umbraco.Core.Logging;
|
|
using Umbraco.Core.Models;
|
|
using Umbraco.Core.Persistence;
|
|
using Umbraco.Core.Persistence.Querying;
|
|
using Umbraco.Core.Persistence.SqlSyntax;
|
|
using Umbraco.Core.Persistence.UnitOfWork;
|
|
|
|
namespace Umbraco.Core.Services
|
|
{
|
|
/// <summary>
|
|
/// Represents the Localization Service, which is an easy access to operations involving <see cref="Language"/> and <see cref="DictionaryItem"/>
|
|
/// </summary>
|
|
public class LocalizationService : RepositoryService, ILocalizationService
|
|
{
|
|
|
|
|
|
public LocalizationService(IDatabaseUnitOfWorkProvider provider, RepositoryFactory repositoryFactory, ILogger logger, IEventMessagesFactory eventMessagesFactory)
|
|
: base(provider, repositoryFactory, logger, eventMessagesFactory)
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Adds or updates a translation for a dictionary item and language
|
|
/// </summary>
|
|
/// <param name="item"></param>
|
|
/// <param name="language"></param>
|
|
/// <param name="value"></param>
|
|
/// <returns></returns>
|
|
/// <remarks>
|
|
/// This does not save the item, that needs to be done explicitly
|
|
/// </remarks>
|
|
public void AddOrUpdateDictionaryValue(IDictionaryItem item, ILanguage language, string value)
|
|
{
|
|
if (item == null) throw new ArgumentNullException("item");
|
|
if (language == null) throw new ArgumentNullException("language");
|
|
|
|
var existing = item.Translations.FirstOrDefault(x => x.Language.Id == language.Id);
|
|
if (existing != null)
|
|
{
|
|
existing.Value = value;
|
|
}
|
|
else
|
|
{
|
|
item.Translations = new List<IDictionaryTranslation>(item.Translations)
|
|
{
|
|
new DictionaryTranslation(language, value)
|
|
};
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates and saves a new dictionary item and assigns a value to all languages if defaultValue is specified.
|
|
/// </summary>
|
|
/// <param name="key"></param>
|
|
/// <param name="parentId"></param>
|
|
/// <param name="defaultValue"></param>
|
|
/// <returns></returns>
|
|
public IDictionaryItem CreateDictionaryItemWithIdentity(string key, Guid? parentId, string defaultValue = null)
|
|
{
|
|
var uow = UowProvider.GetUnitOfWork();
|
|
using (var repository = RepositoryFactory.CreateDictionaryRepository(uow))
|
|
{
|
|
//validate the parent
|
|
if (parentId.HasValue && parentId.Value != Guid.Empty)
|
|
{
|
|
var parent = GetDictionaryItemById(parentId.Value);
|
|
if (parent == null)
|
|
{
|
|
throw new ArgumentException("No parent dictionary item was found with id " + parentId.Value);
|
|
}
|
|
}
|
|
|
|
var item = new DictionaryItem(parentId, key);
|
|
|
|
if (defaultValue.IsNullOrWhiteSpace() == false)
|
|
{
|
|
var langs = GetAllLanguages();
|
|
var translations = langs.Select(language => new DictionaryTranslation(language, defaultValue))
|
|
.Cast<IDictionaryTranslation>()
|
|
.ToList();
|
|
|
|
item.Translations = translations;
|
|
}
|
|
|
|
if (SavingDictionaryItem.IsRaisedEventCancelled(new SaveEventArgs<IDictionaryItem>(item), this))
|
|
return item;
|
|
|
|
repository.AddOrUpdate(item);
|
|
uow.Commit();
|
|
|
|
SavedDictionaryItem.RaiseEvent(new SaveEventArgs<IDictionaryItem>(item), this);
|
|
|
|
return item;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a <see cref="IDictionaryItem"/> by its <see cref="Int32"/> id
|
|
/// </summary>
|
|
/// <param name="id">Id of the <see cref="IDictionaryItem"/></param>
|
|
/// <returns><see cref="IDictionaryItem"/></returns>
|
|
public IDictionaryItem GetDictionaryItemById(int id)
|
|
{
|
|
using (var repository = RepositoryFactory.CreateDictionaryRepository(UowProvider.GetUnitOfWork()))
|
|
{
|
|
return repository.Get(id);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a <see cref="IDictionaryItem"/> by its <see cref="Guid"/> id
|
|
/// </summary>
|
|
/// <param name="id">Id of the <see cref="IDictionaryItem"/></param>
|
|
/// <returns><see cref="DictionaryItem"/></returns>
|
|
public IDictionaryItem GetDictionaryItemById(Guid id)
|
|
{
|
|
using (var repository = RepositoryFactory.CreateDictionaryRepository(UowProvider.GetUnitOfWork()))
|
|
{
|
|
return repository.Get(id);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a <see cref="IDictionaryItem"/> by its key
|
|
/// </summary>
|
|
/// <param name="key">Key of the <see cref="IDictionaryItem"/></param>
|
|
/// <returns><see cref="IDictionaryItem"/></returns>
|
|
public IDictionaryItem GetDictionaryItemByKey(string key)
|
|
{
|
|
using (var repository = RepositoryFactory.CreateDictionaryRepository(UowProvider.GetUnitOfWork()))
|
|
{
|
|
return repository.Get(key);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a list of children for a <see cref="IDictionaryItem"/>
|
|
/// </summary>
|
|
/// <param name="parentId">Id of the parent</param>
|
|
/// <returns>An enumerable list of <see cref="IDictionaryItem"/> objects</returns>
|
|
public IEnumerable<IDictionaryItem> GetDictionaryItemChildren(Guid parentId)
|
|
{
|
|
using (var repository = RepositoryFactory.CreateDictionaryRepository(UowProvider.GetUnitOfWork()))
|
|
{
|
|
var query = QueryFactory.Create<IDictionaryItem>().Where(x => x.ParentId == parentId);
|
|
var items = repository.GetByQuery(query);
|
|
|
|
return items;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a list of descendants for a <see cref="IDictionaryItem"/>
|
|
/// </summary>
|
|
/// <param name="parentId">Id of the parent, null will return all dictionary items</param>
|
|
/// <returns>An enumerable list of <see cref="IDictionaryItem"/> objects</returns>
|
|
public IEnumerable<IDictionaryItem> GetDictionaryItemDescendants(Guid? parentId)
|
|
{
|
|
using (var repository = RepositoryFactory.CreateDictionaryRepository(UowProvider.GetUnitOfWork()))
|
|
{
|
|
return repository.GetDictionaryItemDescendants(parentId);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the root/top <see cref="IDictionaryItem"/> objects
|
|
/// </summary>
|
|
/// <returns>An enumerable list of <see cref="IDictionaryItem"/> objects</returns>
|
|
public IEnumerable<IDictionaryItem> GetRootDictionaryItems()
|
|
{
|
|
using (var repository = RepositoryFactory.CreateDictionaryRepository(UowProvider.GetUnitOfWork()))
|
|
{
|
|
var query = repository.Query.Create<IDictionaryItem>().Where(x => x.ParentId == null);
|
|
var items = repository.GetByQuery(query);
|
|
|
|
return items;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks if a <see cref="IDictionaryItem"/> with given key exists
|
|
/// </summary>
|
|
/// <param name="key">Key of the <see cref="IDictionaryItem"/></param>
|
|
/// <returns>True if a <see cref="IDictionaryItem"/> exists, otherwise false</returns>
|
|
public bool DictionaryItemExists(string key)
|
|
{
|
|
using (var repository = RepositoryFactory.CreateDictionaryRepository(UowProvider.GetUnitOfWork()))
|
|
{
|
|
return repository.Get(key) != null;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Saves a <see cref="IDictionaryItem"/> object
|
|
/// </summary>
|
|
/// <param name="dictionaryItem"><see cref="IDictionaryItem"/> to save</param>
|
|
/// <param name="userId">Optional id of the user saving the dictionary item</param>
|
|
public void Save(IDictionaryItem dictionaryItem, int userId = 0)
|
|
{
|
|
if (SavingDictionaryItem.IsRaisedEventCancelled(new SaveEventArgs<IDictionaryItem>(dictionaryItem), this))
|
|
return;
|
|
|
|
var uow = UowProvider.GetUnitOfWork();
|
|
using (var repository = RepositoryFactory.CreateDictionaryRepository(uow))
|
|
{
|
|
repository.AddOrUpdate(dictionaryItem);
|
|
uow.Commit();
|
|
}
|
|
|
|
SavedDictionaryItem.RaiseEvent(new SaveEventArgs<IDictionaryItem>(dictionaryItem, false), this);
|
|
|
|
Audit(AuditType.Save, "Save DictionaryItem performed by user", userId, dictionaryItem.Id);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Deletes a <see cref="IDictionaryItem"/> object and its related translations
|
|
/// as well as its children.
|
|
/// </summary>
|
|
/// <param name="dictionaryItem"><see cref="IDictionaryItem"/> to delete</param>
|
|
/// <param name="userId">Optional id of the user deleting the dictionary item</param>
|
|
public void Delete(IDictionaryItem dictionaryItem, int userId = 0)
|
|
{
|
|
if (DeletingDictionaryItem.IsRaisedEventCancelled(new DeleteEventArgs<IDictionaryItem>(dictionaryItem), this))
|
|
return;
|
|
|
|
var uow = UowProvider.GetUnitOfWork();
|
|
using (var repository = RepositoryFactory.CreateDictionaryRepository(uow))
|
|
{
|
|
//NOTE: The recursive delete is done in the repository
|
|
repository.Delete(dictionaryItem);
|
|
uow.Commit();
|
|
}
|
|
|
|
DeletedDictionaryItem.RaiseEvent(new DeleteEventArgs<IDictionaryItem>(dictionaryItem, false), this);
|
|
|
|
Audit(AuditType.Delete, "Delete DictionaryItem performed by user", userId, dictionaryItem.Id);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a <see cref="Language"/> by its id
|
|
/// </summary>
|
|
/// <param name="id">Id of the <see cref="Language"/></param>
|
|
/// <returns><see cref="Language"/></returns>
|
|
public ILanguage GetLanguageById(int id)
|
|
{
|
|
using (var repository = RepositoryFactory.CreateLanguageRepository(UowProvider.GetUnitOfWork()))
|
|
{
|
|
return repository.Get(id);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a <see cref="Language"/> by its culture code
|
|
/// </summary>
|
|
/// <param name="cultureName">Culture Name - also refered to as the Friendly name</param>
|
|
/// <returns><see cref="Language"/></returns>
|
|
public ILanguage GetLanguageByCultureCode(string cultureName)
|
|
{
|
|
using (var repository = RepositoryFactory.CreateLanguageRepository(UowProvider.GetUnitOfWork()))
|
|
{
|
|
return repository.GetByCultureName(cultureName);
|
|
//var query = Query<ILanguage>.Builder.Where(x => x.CultureName == cultureName);
|
|
//var items = repository.GetByQuery(query);
|
|
|
|
//return items.FirstOrDefault();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a <see cref="Language"/> by its iso code
|
|
/// </summary>
|
|
/// <param name="isoCode">Iso Code of the language (ie. en-US)</param>
|
|
/// <returns><see cref="Language"/></returns>
|
|
public ILanguage GetLanguageByIsoCode(string isoCode)
|
|
{
|
|
using (var repository = RepositoryFactory.CreateLanguageRepository(UowProvider.GetUnitOfWork()))
|
|
{
|
|
return repository.GetByIsoCode(isoCode);
|
|
//var query = Query<ILanguage>.Builder.Where(x => x.IsoCode == isoCode);
|
|
//var items = repository.GetByQuery(query);
|
|
|
|
//return items.FirstOrDefault();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets all available languages
|
|
/// </summary>
|
|
/// <returns>An enumerable list of <see cref="ILanguage"/> objects</returns>
|
|
public IEnumerable<ILanguage> GetAllLanguages()
|
|
{
|
|
using (var repository = RepositoryFactory.CreateLanguageRepository(UowProvider.GetUnitOfWork()))
|
|
{
|
|
var languages = repository.GetAll();
|
|
return languages;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Saves a <see cref="ILanguage"/> object
|
|
/// </summary>
|
|
/// <param name="language"><see cref="ILanguage"/> to save</param>
|
|
/// <param name="userId">Optional id of the user saving the language</param>
|
|
public void Save(ILanguage language, int userId = 0)
|
|
{
|
|
if (SavingLanguage.IsRaisedEventCancelled(new SaveEventArgs<ILanguage>(language), this))
|
|
return;
|
|
|
|
var uow = UowProvider.GetUnitOfWork();
|
|
using (var repository = RepositoryFactory.CreateLanguageRepository(uow))
|
|
{
|
|
repository.AddOrUpdate(language);
|
|
uow.Commit();
|
|
}
|
|
|
|
SavedLanguage.RaiseEvent(new SaveEventArgs<ILanguage>(language, false), this);
|
|
|
|
Audit(AuditType.Save, "Save Language performed by user", userId, language.Id);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Deletes a <see cref="ILanguage"/> by removing it (but not its usages) from the db
|
|
/// </summary>
|
|
/// <param name="language"><see cref="ILanguage"/> to delete</param>
|
|
/// <param name="userId">Optional id of the user deleting the language</param>
|
|
public void Delete(ILanguage language, int userId = 0)
|
|
{
|
|
if (DeletingLanguage.IsRaisedEventCancelled(new DeleteEventArgs<ILanguage>(language), this))
|
|
return;
|
|
|
|
var uow = UowProvider.GetUnitOfWork();
|
|
using (var repository = RepositoryFactory.CreateLanguageRepository(uow))
|
|
{
|
|
//NOTE: There isn't any constraints in the db, so possible references aren't deleted
|
|
repository.Delete(language);
|
|
uow.Commit();
|
|
}
|
|
|
|
DeletedLanguage.RaiseEvent(new DeleteEventArgs<ILanguage>(language, false), this);
|
|
|
|
Audit(AuditType.Delete, "Delete Language performed by user", userId, language.Id);
|
|
}
|
|
|
|
private void Audit(AuditType type, string message, int userId, int objectId)
|
|
{
|
|
var uow = UowProvider.GetUnitOfWork();
|
|
using (var auditRepo = RepositoryFactory.CreateAuditRepository(uow))
|
|
{
|
|
auditRepo.AddOrUpdate(new AuditItem(objectId, message, type, userId));
|
|
uow.Commit();
|
|
}
|
|
}
|
|
|
|
#region Event Handlers
|
|
/// <summary>
|
|
/// Occurs before Delete
|
|
/// </summary>
|
|
public static event TypedEventHandler<ILocalizationService, DeleteEventArgs<ILanguage>> DeletingLanguage;
|
|
|
|
/// <summary>
|
|
/// Occurs after Delete
|
|
/// </summary>
|
|
public static event TypedEventHandler<ILocalizationService, DeleteEventArgs<ILanguage>> DeletedLanguage;
|
|
|
|
/// <summary>
|
|
/// Occurs before Delete
|
|
/// </summary>
|
|
public static event TypedEventHandler<ILocalizationService, DeleteEventArgs<IDictionaryItem>> DeletingDictionaryItem;
|
|
|
|
/// <summary>
|
|
/// Occurs after Delete
|
|
/// </summary>
|
|
public static event TypedEventHandler<ILocalizationService, DeleteEventArgs<IDictionaryItem>> DeletedDictionaryItem;
|
|
|
|
/// <summary>
|
|
/// Occurs before Save
|
|
/// </summary>
|
|
public static event TypedEventHandler<ILocalizationService, SaveEventArgs<IDictionaryItem>> SavingDictionaryItem;
|
|
|
|
/// <summary>
|
|
/// Occurs after Save
|
|
/// </summary>
|
|
public static event TypedEventHandler<ILocalizationService, SaveEventArgs<IDictionaryItem>> SavedDictionaryItem;
|
|
|
|
/// <summary>
|
|
/// Occurs before Save
|
|
/// </summary>
|
|
public static event TypedEventHandler<ILocalizationService, SaveEventArgs<ILanguage>> SavingLanguage;
|
|
|
|
/// <summary>
|
|
/// Occurs after Save
|
|
/// </summary>
|
|
public static event TypedEventHandler<ILocalizationService, SaveEventArgs<ILanguage>> SavedLanguage;
|
|
#endregion
|
|
}
|
|
} |