350 lines
14 KiB
C#
350 lines
14 KiB
C#
using System;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.ComponentModel;
|
|
using System.Linq;
|
|
using System.Reflection;
|
|
using System.Runtime.Serialization;
|
|
|
|
namespace Umbraco.Core.Models
|
|
{
|
|
/// <summary>
|
|
/// Represents a Content object
|
|
/// </summary>
|
|
[Serializable]
|
|
[DataContract(IsReference = true)]
|
|
public class Content : ContentBase, IContent
|
|
{
|
|
private IContentType _contentType;
|
|
private ITemplate _template;
|
|
private bool _published;
|
|
private string _language;
|
|
private DateTime? _releaseDate;
|
|
private DateTime? _expireDate;
|
|
private int _writer;
|
|
private string _nodeName;//NOTE Once localization is introduced this will be the non-localized Node Name.
|
|
private bool _permissionsChanged;
|
|
/// <summary>
|
|
/// Constructor for creating a Content object
|
|
/// </summary>
|
|
/// <param name="name">Name of the content</param>
|
|
/// <param name="parent">Parent <see cref="IContent"/> object</param>
|
|
/// <param name="contentType">ContentType for the current Content object</param>
|
|
public Content(string name, IContent parent, IContentType contentType)
|
|
: this(name, parent, contentType, new PropertyCollection())
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Constructor for creating a Content object
|
|
/// </summary>
|
|
/// <param name="name">Name of the content</param>
|
|
/// <param name="parent">Parent <see cref="IContent"/> object</param>
|
|
/// <param name="contentType">ContentType for the current Content object</param>
|
|
/// <param name="properties">Collection of properties</param>
|
|
public Content(string name, IContent parent, IContentType contentType, PropertyCollection properties)
|
|
: base(name, parent, contentType, properties)
|
|
{
|
|
Mandate.ParameterNotNull(contentType, "contentType");
|
|
|
|
_contentType = contentType;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Constructor for creating a Content object
|
|
/// </summary>
|
|
/// <param name="name">Name of the content</param>
|
|
/// <param name="parentId">Id of the Parent content</param>
|
|
/// <param name="contentType">ContentType for the current Content object</param>
|
|
public Content(string name, int parentId, IContentType contentType)
|
|
: this(name, parentId, contentType, new PropertyCollection())
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Constructor for creating a Content object
|
|
/// </summary>
|
|
/// <param name="name">Name of the content</param>
|
|
/// <param name="parentId">Id of the Parent content</param>
|
|
/// <param name="contentType">ContentType for the current Content object</param>
|
|
/// <param name="properties">Collection of properties</param>
|
|
public Content(string name, int parentId, IContentType contentType, PropertyCollection properties)
|
|
: base(name, parentId, contentType, properties)
|
|
{
|
|
Mandate.ParameterNotNull(contentType, "contentType");
|
|
|
|
_contentType = contentType;
|
|
}
|
|
|
|
private static readonly Lazy<PropertySelectors> Ps = new Lazy<PropertySelectors>();
|
|
|
|
private class PropertySelectors
|
|
{
|
|
public readonly PropertyInfo TemplateSelector = ExpressionHelper.GetPropertyInfo<Content, ITemplate>(x => x.Template);
|
|
public readonly PropertyInfo PublishedSelector = ExpressionHelper.GetPropertyInfo<Content, bool>(x => x.Published);
|
|
public readonly PropertyInfo LanguageSelector = ExpressionHelper.GetPropertyInfo<Content, string>(x => x.Language);
|
|
public readonly PropertyInfo ReleaseDateSelector = ExpressionHelper.GetPropertyInfo<Content, DateTime?>(x => x.ReleaseDate);
|
|
public readonly PropertyInfo ExpireDateSelector = ExpressionHelper.GetPropertyInfo<Content, DateTime?>(x => x.ExpireDate);
|
|
public readonly PropertyInfo WriterSelector = ExpressionHelper.GetPropertyInfo<Content, int>(x => x.WriterId);
|
|
public readonly PropertyInfo NodeNameSelector = ExpressionHelper.GetPropertyInfo<Content, string>(x => x.NodeName);
|
|
public readonly PropertyInfo PermissionsChangedSelector = ExpressionHelper.GetPropertyInfo<Content, bool>(x => x.PermissionsChanged);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the template used by the Content.
|
|
/// This is used to override the default one from the ContentType.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// If no template is explicitly set on the Content object,
|
|
/// the Default template from the ContentType will be returned.
|
|
/// </remarks>
|
|
[DataMember]
|
|
public virtual ITemplate Template
|
|
{
|
|
get { return _template; }
|
|
set { SetPropertyValueAndDetectChanges(value, ref _template, Ps.Value.TemplateSelector); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the current status of the Content
|
|
/// </summary>
|
|
[IgnoreDataMember]
|
|
public ContentStatus Status
|
|
{
|
|
get
|
|
{
|
|
if(Trashed)
|
|
return ContentStatus.Trashed;
|
|
|
|
if(ExpireDate.HasValue && ExpireDate.Value > DateTime.MinValue && DateTime.Now > ExpireDate.Value)
|
|
return ContentStatus.Expired;
|
|
|
|
if(ReleaseDate.HasValue && ReleaseDate.Value > DateTime.MinValue && ReleaseDate.Value > DateTime.Now)
|
|
return ContentStatus.AwaitingRelease;
|
|
|
|
if(Published)
|
|
return ContentStatus.Published;
|
|
|
|
return ContentStatus.Unpublished;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Boolean indicating whether this Content is Published or not
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Setting Published to true/false should be private or internal and should ONLY be used for wiring up the value
|
|
/// from the db or modifying it based on changing the published state.
|
|
/// </remarks>
|
|
[DataMember]
|
|
public bool Published
|
|
{
|
|
get { return _published; }
|
|
internal set { SetPropertyValueAndDetectChanges(value, ref _published, Ps.Value.PublishedSelector); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Language of the data contained within this Content object.
|
|
/// </summary>
|
|
[Obsolete("This is not used and will be removed from the codebase in future versions")]
|
|
[EditorBrowsable(EditorBrowsableState.Never)]
|
|
public string Language
|
|
{
|
|
get { return _language; }
|
|
set { SetPropertyValueAndDetectChanges(value, ref _language, Ps.Value.LanguageSelector); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// The date this Content should be released and thus be published
|
|
/// </summary>
|
|
[DataMember]
|
|
public DateTime? ReleaseDate
|
|
{
|
|
get { return _releaseDate; }
|
|
set { SetPropertyValueAndDetectChanges(value, ref _releaseDate, Ps.Value.ReleaseDateSelector); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// The date this Content should expire and thus be unpublished
|
|
/// </summary>
|
|
[DataMember]
|
|
public DateTime? ExpireDate
|
|
{
|
|
get { return _expireDate; }
|
|
set { SetPropertyValueAndDetectChanges(value, ref _expireDate, Ps.Value.ExpireDateSelector); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Id of the user who wrote/updated this Content
|
|
/// </summary>
|
|
[DataMember]
|
|
public virtual int WriterId
|
|
{
|
|
get { return _writer; }
|
|
set { SetPropertyValueAndDetectChanges(value, ref _writer, Ps.Value.WriterSelector); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Name of the Node (non-localized).
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// This Property is kept internal until localization is introduced.
|
|
/// </remarks>
|
|
[DataMember]
|
|
internal string NodeName
|
|
{
|
|
get { return _nodeName; }
|
|
set { SetPropertyValueAndDetectChanges(value, ref _nodeName, Ps.Value.NodeNameSelector); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Used internally to track if permissions have been changed during the saving process for this entity
|
|
/// </summary>
|
|
[IgnoreDataMember]
|
|
internal bool PermissionsChanged
|
|
{
|
|
get { return _permissionsChanged; }
|
|
set { SetPropertyValueAndDetectChanges(value, ref _permissionsChanged, Ps.Value.PermissionsChangedSelector); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the ContentType used by this content object
|
|
/// </summary>
|
|
[IgnoreDataMember]
|
|
public IContentType ContentType
|
|
{
|
|
get { return _contentType; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Changes the <see cref="ContentType"/> for the current content object
|
|
/// </summary>
|
|
/// <param name="contentType">New ContentType for this content</param>
|
|
/// <remarks>Leaves PropertyTypes intact after change</remarks>
|
|
public void ChangeContentType(IContentType contentType)
|
|
{
|
|
ContentTypeId = contentType.Id;
|
|
_contentType = contentType;
|
|
ContentTypeBase = contentType;
|
|
Properties.EnsurePropertyTypes(PropertyTypes);
|
|
Properties.CollectionChanged += PropertiesChanged;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Changes the <see cref="ContentType"/> for the current content object and removes PropertyTypes,
|
|
/// which are not part of the new ContentType.
|
|
/// </summary>
|
|
/// <param name="contentType">New ContentType for this content</param>
|
|
/// <param name="clearProperties">Boolean indicating whether to clear PropertyTypes upon change</param>
|
|
public void ChangeContentType(IContentType contentType, bool clearProperties)
|
|
{
|
|
if(clearProperties)
|
|
{
|
|
ContentTypeId = contentType.Id;
|
|
_contentType = contentType;
|
|
ContentTypeBase = contentType;
|
|
Properties.EnsureCleanPropertyTypes(PropertyTypes);
|
|
Properties.CollectionChanged += PropertiesChanged;
|
|
return;
|
|
}
|
|
|
|
ChangeContentType(contentType);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Changes the Published state of the content object
|
|
/// </summary>
|
|
public void ChangePublishedState(PublishedState state)
|
|
{
|
|
Published = state == PublishedState.Published;
|
|
PublishedState = state;
|
|
}
|
|
|
|
[DataMember]
|
|
internal PublishedState PublishedState { get; set; }
|
|
|
|
/// <summary>
|
|
/// Gets or sets the unique identifier of the published version, if any.
|
|
/// </summary>
|
|
[IgnoreDataMember]
|
|
public Guid PublishedVersionGuid { get; internal set; }
|
|
|
|
/// <summary>
|
|
/// Gets a value indicating whether the content has a published version.
|
|
/// </summary>
|
|
public bool HasPublishedVersion { get { return PublishedVersionGuid != default(Guid); } }
|
|
|
|
/// <summary>
|
|
/// Changes the Trashed state of the content object
|
|
/// </summary>
|
|
/// <param name="isTrashed">Boolean indicating whether content is trashed (true) or not trashed (false)</param>
|
|
/// <param name="parentId"> </param>
|
|
public override void ChangeTrashedState(bool isTrashed, int parentId = -20)
|
|
{
|
|
Trashed = isTrashed;
|
|
ParentId = parentId;
|
|
|
|
//If the content is trashed and is published it should be marked as unpublished
|
|
if (isTrashed && Published)
|
|
{
|
|
ChangePublishedState(PublishedState.Unpublished);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Method to call when Entity is being updated
|
|
/// </summary>
|
|
/// <remarks>Modified Date is set and a new Version guid is set</remarks>
|
|
internal override void UpdatingEntity()
|
|
{
|
|
base.UpdatingEntity();
|
|
Version = Guid.NewGuid();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a deep clone of the current entity with its identity and it's property identities reset
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
[Obsolete("Use DeepCloneWithResetIdentities instead")]
|
|
public IContent Clone()
|
|
{
|
|
return DeepCloneWithResetIdentities();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a deep clone of the current entity with its identity and it's property identities reset
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
public IContent DeepCloneWithResetIdentities()
|
|
{
|
|
var clone = (Content)DeepClone();
|
|
clone.Key = Guid.Empty;
|
|
clone.Version = Guid.NewGuid();
|
|
clone.ResetIdentity();
|
|
|
|
foreach (var property in clone.Properties)
|
|
{
|
|
property.ResetIdentity();
|
|
property.Version = clone.Version;
|
|
}
|
|
|
|
return clone;
|
|
}
|
|
|
|
public override object DeepClone()
|
|
{
|
|
var clone = (Content)base.DeepClone();
|
|
//turn off change tracking
|
|
clone.DisableChangeTracking();
|
|
//need to manually clone this since it's not settable
|
|
clone._contentType = (IContentType)ContentType.DeepClone();
|
|
//this shouldn't really be needed since we're not tracking
|
|
clone.ResetDirtyProperties(false);
|
|
//re-enable tracking
|
|
clone.EnableChangeTracking();
|
|
|
|
return clone;
|
|
|
|
}
|
|
}
|
|
} |