using System; using System.Collections.Generic; using System.Linq; using System.Reflection; using System.Web.Mvc; using System.Web.Routing; using Orchard.ContentManagement; using Orchard.ContentManagement.Aspects; using Orchard.ContentManagement.MetaData; using Orchard.ContentManagement.MetaData.Models; using Orchard.Core.Common.Models; using Orchard.Core.Contents.Settings; using Orchard.Core.Contents.ViewModels; using Orchard.Data; using Orchard.DisplayManagement; using Orchard.Localization; using Orchard.Logging; using Orchard.Mvc.Results; using Orchard.UI.Notify; namespace Orchard.Core.Contents.Controllers { [ValidateInput(false)] public class AdminController : Controller, IUpdateModel { private readonly IContentManager _contentManager; private readonly IContentDefinitionManager _contentDefinitionManager; private readonly ITransactionManager _transactionManager; public AdminController( IOrchardServices orchardServices, IContentManager contentManager, IContentDefinitionManager contentDefinitionManager, ITransactionManager transactionManager, IShapeHelperFactory shapeHelperFactory) { Services = orchardServices; _contentManager = contentManager; _contentDefinitionManager = contentDefinitionManager; _transactionManager = transactionManager; T = NullLocalizer.Instance; Logger = NullLogger.Instance; Shape = shapeHelperFactory.CreateHelper(); } dynamic Shape { get; set; } public IOrchardServices Services { get; private set; } public Localizer T { get; set; } public ILogger Logger { get; set; } public ActionResult List(ListContentsViewModel model) { if (model.ContainerId != null && _contentManager.GetLatest((int)model.ContainerId) == null) return new NotFoundResult(); const int pageSize = 20; var skip = (Math.Max(model.Page ?? 0, 1) - 1) * pageSize; var query = _contentManager.Query(VersionOptions.Latest, GetCreatableTypes().Select(ctd => ctd.Name).ToArray()); if (!string.IsNullOrEmpty(model.TypeName)) { var contentTypeDefinition = _contentDefinitionManager.GetTypeDefinition(model.TypeName); if (contentTypeDefinition == null) return new NotFoundResult(); model.TypeDisplayName = !string.IsNullOrWhiteSpace(contentTypeDefinition.DisplayName) ? contentTypeDefinition.DisplayName : contentTypeDefinition.Name; query = query.ForType(model.TypeName); } if (model.ContainerId != null) query = query.Join().Where(cr => cr.Container.Id == model.ContainerId); // Ordering //-- want something like //switch (model.Options.OrderBy) { // case ContentsOrder.Modified: // query = query.OrderByDescending(cr => cr.ModifiedUtc); // break; // case ContentsOrder.Published: // query = query.OrderByDescending(cr => cr.PublishedUtc); // break; // case ContentsOrder.Created: // query = query.OrderByDescending(cr => cr.CreatedUtc); // break; //} //-- but resorting to var contentItems = query.List(); switch (model.Options.OrderBy) { case ContentsOrder.Modified: contentItems = contentItems.OrderByDescending(ci => ci.VersionRecord.Id); break; //case ContentsOrder.Published: // would be lying w/out a published date instead of a bool but that only comes with the common aspect // contentItems = contentItems.OrderByDescending(ci => ci.VersionRecord.Published/*Date*/); // break; case ContentsOrder.Created: contentItems = contentItems.OrderByDescending(ci => ci.Id); break; } //-- for the moment //-- because I'd rather do this //var contentItems = query.Slice(skip, pageSize); //-- instead of this (having the ordering and skip/take after the query) contentItems = contentItems.Skip(skip).Take(pageSize).ToList(); model.Options.SelectedFilter = model.TypeName; model.Options.FilterOptions = GetCreatableTypes() .Select(ctd => new KeyValuePair(ctd.Name, ctd.DisplayName)) .ToList().OrderBy(kvp => kvp.Key); var list = Shape.List(); list.AddRange(contentItems.Select(ci => _contentManager.BuildDisplayModel(ci, "SummaryAdmin"))); var viewModel = Shape.ViewModel() .ContentItems(list) .Options(model.Options) .TypeDisplayName(model.TypeDisplayName ?? ""); return View(viewModel); } private IEnumerable GetCreatableTypes() { return _contentDefinitionManager.ListTypeDefinitions().Where(ctd => ctd.Settings.GetModel().Creatable); } [HttpPost, ActionName("List")] [FormValueRequired("submit.Filter")] public ActionResult ListFilterPOST(ContentOptions options) { var routeValues = ControllerContext.RouteData.Values; if (options != null) { routeValues["Options.OrderBy"] = options.OrderBy; //todo: don't hard-code the key if (GetCreatableTypes().Any(ctd => string.Equals(ctd.Name, options.SelectedFilter, StringComparison.OrdinalIgnoreCase))) { routeValues["id"] = options.SelectedFilter; } else { routeValues.Remove("id"); } } return RedirectToAction("List", routeValues); } [HttpPost, ActionName("List")] [FormValueRequired("submit.BulkEdit")] public ActionResult ListPOST(ContentOptions options, IEnumerable itemIds, string returnUrl) { if (itemIds != null) { var accessChecked = false; switch (options.BulkAction) { case ContentsBulkAction.None: break; case ContentsBulkAction.PublishNow: foreach (var item in itemIds.Select(itemId => _contentManager.GetLatest(itemId))) { if (!accessChecked && !Services.Authorizer.Authorize(Permissions.PublishContent, item, T("Couldn't publish selected content."))) return new HttpUnauthorizedResult(); accessChecked = true; _contentManager.Publish(item); Services.ContentManager.Flush(); } Services.Notifier.Information(T("Content successfully published.")); break; case ContentsBulkAction.Unpublish: foreach (var item in itemIds.Select(itemId => _contentManager.GetLatest(itemId))) { if (!accessChecked && !Services.Authorizer.Authorize(Permissions.PublishContent, item, T("Couldn't unpublish selected content."))) return new HttpUnauthorizedResult(); accessChecked = true; _contentManager.Unpublish(item); Services.ContentManager.Flush(); } Services.Notifier.Information(T("Content successfully unpublished.")); break; case ContentsBulkAction.Remove: foreach (var item in itemIds.Select(itemId => _contentManager.GetLatest(itemId))) { if (!accessChecked && !Services.Authorizer.Authorize(Permissions.DeleteContent, item, T("Couldn't remove selected content."))) return new HttpUnauthorizedResult(); accessChecked = true; _contentManager.Remove(item); Services.ContentManager.Flush(); } Services.Notifier.Information(T("Content successfully removed.")); break; default: throw new ArgumentOutOfRangeException(); } } if (!String.IsNullOrEmpty(returnUrl)) return Redirect(returnUrl); return RedirectToAction("List"); } ActionResult CreatableTypeList() { var list = Shape.List(); list.AddRange(GetCreatableTypes()); var viewModel = Shape.ViewModel() .ContentTypes(list); return View("CreatableTypeList", viewModel); } public ActionResult Create(string id) { if (string.IsNullOrEmpty(id)) return CreatableTypeList(); var contentItem = _contentManager.New(id); if (!Services.Authorizer.Authorize(Permissions.PublishContent, contentItem, T("Cannot create content"))) return new HttpUnauthorizedResult(); var model = _contentManager.BuildEditorModel(contentItem); return View(model); } [HttpPost, ActionName("Create")] public ActionResult CreatePOST(string id) { var contentItem = _contentManager.New(id); if (!Services.Authorizer.Authorize(Permissions.PublishContent, contentItem, T("Couldn't create content"))) return new HttpUnauthorizedResult(); _contentManager.Create(contentItem, VersionOptions.Draft); var model = _contentManager.UpdateEditorModel(contentItem, this); if (!ModelState.IsValid) { _transactionManager.Cancel(); return View(model); } if (!contentItem.Has()) _contentManager.Publish(contentItem); Services.Notifier.Information(string.IsNullOrWhiteSpace(contentItem.TypeDefinition.DisplayName) ? T("Your content has been created.") : T("Your {0} has been created.", contentItem.TypeDefinition.DisplayName)); return RedirectToAction("Edit", new RouteValueDictionary { { "Id", contentItem.Id } }); } public ActionResult Edit(int id) { var contentItem = _contentManager.Get(id, VersionOptions.Latest); if (contentItem == null) return new NotFoundResult(); if (!Services.Authorizer.Authorize(Permissions.EditOthersContent, contentItem, T("Cannot edit content"))) return new HttpUnauthorizedResult(); var model = _contentManager.BuildEditorModel(contentItem); return View(model); } [HttpPost, ActionName("Edit")] public ActionResult EditPOST(int id) { var contentItem = _contentManager.Get(id, VersionOptions.DraftRequired); if (contentItem == null) return new NotFoundResult(); if (!Services.Authorizer.Authorize(Permissions.EditOthersContent, contentItem, T("Couldn't edit content"))) return new HttpUnauthorizedResult(); var model = _contentManager.UpdateEditorModel(contentItem, this); if (!ModelState.IsValid) { _transactionManager.Cancel(); return View("Edit", model); } //need to go about this differently - to know when to publish (IPlublishableAspect ?) if (!contentItem.Has()) _contentManager.Publish(contentItem); Services.Notifier.Information(string.IsNullOrWhiteSpace(contentItem.TypeDefinition.DisplayName) ? T("Your content has been saved.") : T("Your {0} has been saved.", contentItem.TypeDefinition.DisplayName)); return RedirectToAction("Edit", new RouteValueDictionary { { "Id", contentItem.Id } }); } [HttpPost, ActionName("Remove")] public ActionResult RemovePOST(int id, string returnUrl) { var contentItem = _contentManager.Get(id, VersionOptions.Latest); if (!Services.Authorizer.Authorize(Permissions.DeleteOthersContent, contentItem, T("Couldn't remove content"))) return new HttpUnauthorizedResult(); if (contentItem != null) { _contentManager.Remove(contentItem); Services.Notifier.Information(string.IsNullOrWhiteSpace(contentItem.TypeDefinition.DisplayName) ? T("That content has been removed.") : T("That {0} has been removed.", contentItem.TypeDefinition.DisplayName)); } if (!String.IsNullOrEmpty(returnUrl)) return Redirect(returnUrl); return RedirectToAction("List"); } [HttpPost] public ActionResult Publish(int id, string returnUrl) { var contentItem = _contentManager.GetLatest(id); if (contentItem == null) return new NotFoundResult(); if (!Services.Authorizer.Authorize(Permissions.PublishContent, contentItem, T("Couldn't publish content"))) return new HttpUnauthorizedResult(); _contentManager.Publish(contentItem); Services.ContentManager.Flush(); Services.Notifier.Information(string.IsNullOrWhiteSpace(contentItem.TypeDefinition.DisplayName) ? T("That content has been published.") : T("That {0} has been published.", contentItem.TypeDefinition.DisplayName)); if (!String.IsNullOrEmpty(returnUrl)) return Redirect(returnUrl); return RedirectToAction("List"); } [HttpPost] public ActionResult Unpublish(int id, string returnUrl) { var contentItem = _contentManager.GetLatest(id); if (contentItem == null) return new NotFoundResult(); if (!Services.Authorizer.Authorize(Permissions.PublishContent, contentItem, T("Couldn't unpublish content"))) return new HttpUnauthorizedResult(); _contentManager.Unpublish(contentItem); Services.ContentManager.Flush(); Services.Notifier.Information(string.IsNullOrWhiteSpace(contentItem.TypeDefinition.DisplayName) ? T("That content has been unpublished.") : T("That {0} has been unpublished.", contentItem.TypeDefinition.DisplayName)); if (!String.IsNullOrEmpty(returnUrl)) return Redirect(returnUrl); return RedirectToAction("List"); } bool IUpdateModel.TryUpdateModel(TModel model, string prefix, string[] includeProperties, string[] excludeProperties) { return TryUpdateModel(model, prefix, includeProperties, excludeProperties); } void IUpdateModel.AddModelError(string key, LocalizedString errorMessage) { ModelState.AddModelError(key, errorMessage.ToString()); } } public class FormValueRequiredAttribute : ActionMethodSelectorAttribute { private readonly string _submitButtonName; public FormValueRequiredAttribute(string submitButtonName) { _submitButtonName = submitButtonName; } public override bool IsValidForRequest(ControllerContext controllerContext, MethodInfo methodInfo) { var value = controllerContext.HttpContext.Request.Form[_submitButtonName]; return !string.IsNullOrEmpty(value); } } }