2018-06-29 19:55:15 +02:00
using System ;
using System.Collections.Generic ;
using System.Linq ;
using System.Net ;
using System.Net.Http.Formatting ;
using System.Web.Http ;
using Umbraco.Core ;
using Umbraco.Core.Models ;
using Umbraco.Core.Models.Entities ;
using Umbraco.Core.Services ;
2018-10-29 17:27:33 +11:00
using Umbraco.Web.Actions ;
2018-06-29 19:55:15 +02:00
using Umbraco.Web.Models.Trees ;
using Umbraco.Web.Mvc ;
using Umbraco.Web.WebApi.Filters ;
using Umbraco.Web.Models.ContentEditing ;
using Umbraco.Web.Search ;
2019-10-15 00:07:44 +11:00
using Umbraco.Core.Cache ;
using Umbraco.Core.Configuration ;
using Umbraco.Core.Logging ;
using Umbraco.Core.Persistence ;
2019-11-05 13:45:42 +01:00
using Constants = Umbraco . Core . Constants ;
2018-06-29 19:55:15 +02:00
namespace Umbraco.Web.Trees
{
//We will not allow the tree to render unless the user has access to any of the sections that the tree gets rendered
// this is not ideal but until we change permissions to be tree based (not section) there's not much else we can do here.
[ UmbracoApplicationAuthorize (
2019-11-05 13:45:42 +01:00
Constants . Applications . Content ,
Constants . Applications . Media ,
Constants . Applications . Users ,
Constants . Applications . Settings ,
Constants . Applications . Packages ,
Constants . Applications . Members ) ]
[Tree(Constants.Applications.Content, Constants.Trees.Content)]
2018-06-29 19:55:15 +02:00
[PluginController("UmbracoTrees")]
[CoreTree]
2019-05-31 20:29:00 +02:00
[SearchableTree("searchResultFormatter", "configureContentResult", 10)]
2018-06-29 19:55:15 +02:00
public class ContentTreeController : ContentTreeControllerBase , ISearchableTree
{
2018-12-03 22:10:56 +11:00
private readonly UmbracoTreeSearcher _treeSearcher ;
2018-12-11 15:42:32 +11:00
private readonly ActionCollection _actions ;
2018-12-03 22:10:56 +11:00
2019-11-05 13:45:42 +01:00
protected override int RecycleBinId = > Constants . System . RecycleBinContent ;
2018-06-29 19:55:15 +02:00
protected override bool RecycleBinSmells = > Services . ContentService . RecycleBinSmells ( ) ;
private int [ ] _userStartNodes ;
2019-10-15 00:07:44 +11:00
2018-06-29 19:55:15 +02:00
protected override int [ ] UserStartNodes
= > _userStartNodes ? ? ( _userStartNodes = Security . CurrentUser . CalculateContentStartNodeIds ( Services . EntityService ) ) ;
2019-10-15 00:07:44 +11:00
public ContentTreeController ( UmbracoTreeSearcher treeSearcher , ActionCollection actions , IGlobalSettings globalSettings , IUmbracoContextAccessor umbracoContextAccessor , ISqlContext sqlContext , ServiceContext services , AppCaches appCaches , IProfilingLogger logger , IRuntimeState runtimeState , UmbracoHelper umbracoHelper ) : base ( globalSettings , umbracoContextAccessor , sqlContext , services , appCaches , logger , runtimeState , umbracoHelper )
{
_treeSearcher = treeSearcher ;
_actions = actions ;
}
2018-06-29 19:55:15 +02:00
/// <inheritdoc />
protected override TreeNode GetSingleTreeNode ( IEntitySlim entity , string parentId , FormDataCollection queryStrings )
{
2018-09-21 15:49:37 +10:00
var culture = queryStrings ? [ "culture" ] ;
2018-07-09 18:26:15 +02:00
2018-06-29 19:55:15 +02:00
var allowedUserOptions = GetAllowedUserMenuItemsForNode ( entity ) ;
2018-09-21 15:49:37 +10:00
if ( CanUserAccessNode ( entity , allowedUserOptions , culture ) )
2018-06-29 19:55:15 +02:00
{
2019-01-26 10:52:19 -05:00
//Special check to see if it is a container, if so then we'll hide children.
2018-06-29 19:55:15 +02:00
var isContainer = entity . IsContainer ; // && (queryStrings.Get("isDialog") != "true");
var node = CreateTreeNode (
entity ,
2019-11-05 13:45:42 +01:00
Constants . ObjectTypes . Document ,
2018-06-29 19:55:15 +02:00
parentId ,
queryStrings ,
2018-12-19 18:01:36 +01:00
entity . HasChildren ) ;
2018-06-29 19:55:15 +02:00
2018-11-15 15:24:09 +11:00
// set container style if it is one
2018-06-29 19:55:15 +02:00
if ( isContainer )
{
node . AdditionalData . Add ( "isContainer" , true ) ;
node . SetContainerStyle ( ) ;
}
2018-11-15 15:24:09 +11:00
var documentEntity = ( IDocumentEntitySlim ) entity ;
if ( ! documentEntity . Variations . VariesByCulture ( ) )
{
if ( ! documentEntity . Published )
node . SetNotPublishedStyle ( ) ;
else if ( documentEntity . Edited )
node . SetHasPendingVersionStyle ( ) ;
}
2018-06-29 19:55:15 +02:00
else
{
2018-11-15 15:24:09 +11:00
if ( ! culture . IsNullOrWhiteSpace ( ) )
2018-09-21 15:49:37 +10:00
{
2018-11-15 15:24:09 +11:00
if ( ! documentEntity . Published | | ! documentEntity . PublishedCultures . Contains ( culture ) )
2018-09-21 15:49:37 +10:00
node . SetNotPublishedStyle ( ) ;
2018-11-15 15:24:09 +11:00
else if ( documentEntity . EditedCultures . Contains ( culture ) )
2018-09-27 19:30:31 +10:00
node . SetHasPendingVersionStyle ( ) ;
2018-09-21 15:49:37 +10:00
}
2018-06-29 19:55:15 +02:00
}
2018-11-15 15:24:09 +11:00
node . AdditionalData . Add ( "variesByCulture" , documentEntity . Variations . VariesByCulture ( ) ) ;
node . AdditionalData . Add ( "contentType" , documentEntity . ContentTypeAlias ) ;
2018-06-29 19:55:15 +02:00
if ( Services . PublicAccessService . IsProtected ( entity . Path ) )
node . SetProtectedStyle ( ) ;
return node ;
}
return null ;
}
protected override MenuItemCollection PerformGetMenuForNode ( string id , FormDataCollection queryStrings )
{
2019-11-05 13:45:42 +01:00
if ( id = = Constants . System . RootString )
2018-06-29 19:55:15 +02:00
{
var menu = new MenuItemCollection ( ) ;
// if the user's start node is not the root then the only menu item to display is refresh
2019-11-05 13:45:42 +01:00
if ( UserStartNodes . Contains ( Constants . System . Root ) = = false )
2018-06-29 19:55:15 +02:00
{
2018-10-29 17:27:33 +11:00
menu . Items . Add ( new RefreshNode ( Services . TextService , true ) ) ;
2018-06-29 19:55:15 +02:00
return menu ;
}
//set the default to create
2018-10-29 17:27:33 +11:00
menu . DefaultMenuAlias = ActionNew . ActionAlias ;
2018-06-29 19:55:15 +02:00
// we need to get the default permissions as you can't set permissions on the very root node
2019-11-05 13:45:42 +01:00
var permission = Services . UserService . GetPermissions ( Security . CurrentUser , Constants . System . Root ) . First ( ) ;
2018-12-11 15:42:32 +11:00
var nodeActions = _actions . FromEntityPermission ( permission )
2018-06-29 19:55:15 +02:00
. Select ( x = > new MenuItem ( x ) ) ;
//these two are the standard items
2018-10-30 00:34:43 +11:00
menu . Items . Add < ActionNew > ( Services . TextService , opensDialog : true ) ;
menu . Items . Add < ActionSort > ( Services . TextService , true ) ;
2018-06-29 19:55:15 +02:00
//filter the standard items
FilterUserAllowedMenuItems ( menu , nodeActions ) ;
if ( menu . Items . Any ( ) )
{
2019-01-14 09:32:32 -05:00
menu . Items . Last ( ) . SeparatorBefore = true ;
2018-06-29 19:55:15 +02:00
}
// add default actions for *all* users
2018-10-29 17:27:33 +11:00
menu . Items . Add ( new RefreshNode ( Services . TextService , true ) ) ;
2018-06-29 19:55:15 +02:00
return menu ;
}
//return a normal node menu:
int iid ;
if ( int . TryParse ( id , out iid ) = = false )
{
throw new HttpResponseException ( HttpStatusCode . NotFound ) ;
}
var item = Services . EntityService . Get ( iid , UmbracoObjectTypes . Document ) ;
if ( item = = null )
{
throw new HttpResponseException ( HttpStatusCode . NotFound ) ;
}
//if the user has no path access for this node, all they can do is refresh
2018-11-15 15:24:09 +11:00
if ( ! Security . CurrentUser . HasContentPathAccess ( item , Services . EntityService ) )
2018-06-29 19:55:15 +02:00
{
var menu = new MenuItemCollection ( ) ;
2018-10-29 17:27:33 +11:00
menu . Items . Add ( new RefreshNode ( Services . TextService , true ) ) ;
2018-06-29 19:55:15 +02:00
return menu ;
}
2019-11-05 12:54:22 +01:00
var nodeMenu = GetAllNodeMenuItems ( item ) ;
2018-07-11 15:58:48 +10:00
//if the content node is in the recycle bin, don't have a default menu, just show the regular menu
2018-06-29 19:55:15 +02:00
if ( item . Path . Split ( new [ ] { ',' } , StringSplitOptions . RemoveEmptyEntries ) . Contains ( RecycleBinId . ToInvariantString ( ) ) )
{
nodeMenu . DefaultMenuAlias = null ;
2018-07-11 15:58:48 +10:00
nodeMenu = GetNodeMenuItemsForDeletedContent ( item ) ;
2018-06-29 19:55:15 +02:00
}
else
{
//set the default to create
2018-10-29 17:27:33 +11:00
nodeMenu . DefaultMenuAlias = ActionNew . ActionAlias ;
2018-06-29 19:55:15 +02:00
}
2018-07-11 15:58:48 +10:00
var allowedMenuItems = GetAllowedUserMenuItemsForNode ( item ) ;
FilterUserAllowedMenuItems ( nodeMenu , allowedMenuItems ) ;
2018-06-29 19:55:15 +02:00
return nodeMenu ;
}
2018-08-17 00:50:05 +10:00
protected override UmbracoObjectTypes UmbracoObjectType = > UmbracoObjectTypes . Document ;
2018-06-29 19:55:15 +02:00
/// <summary>
/// Returns true or false if the current user has access to the node based on the user's allowed start node (path) access
/// </summary>
/// <param name="id"></param>
/// <param name="queryStrings"></param>
/// <returns></returns>
protected override bool HasPathAccess ( string id , FormDataCollection queryStrings )
{
var entity = GetEntityFromId ( id ) ;
return HasPathAccess ( entity , queryStrings ) ;
2018-05-10 18:20:33 +10:00
}
protected override IEnumerable < IEntitySlim > GetChildEntities ( string id , FormDataCollection queryStrings )
{
var result = base . GetChildEntities ( id , queryStrings ) ;
var culture = queryStrings [ "culture" ] . TryConvertTo < string > ( ) ;
2018-06-29 19:55:15 +02:00
//if this is null we'll set it to the default.
var cultureVal = ( culture . Success ? culture . Result : null ) ? ? Services . LocalizationService . GetDefaultLanguageIsoCode ( ) ;
// set names according to variations
2018-05-10 18:20:33 +10:00
foreach ( var entity in result )
2019-01-30 22:01:57 +11:00
{
2018-05-10 18:20:33 +10:00
EnsureName ( entity , cultureVal ) ;
2019-01-30 22:01:57 +11:00
}
2018-05-10 18:20:33 +10:00
2018-07-09 18:26:15 +02:00
return result ;
2018-06-29 19:55:15 +02:00
}
/// <summary>
/// Returns a collection of all menu items that can be on a content node
/// </summary>
/// <param name="item"></param>
/// <returns></returns>
protected MenuItemCollection GetAllNodeMenuItems ( IUmbracoEntity item )
{
var menu = new MenuItemCollection ( ) ;
2018-10-30 00:34:43 +11:00
AddActionNode < ActionNew > ( item , menu , opensDialog : true ) ;
AddActionNode < ActionDelete > ( item , menu , opensDialog : true ) ;
AddActionNode < ActionCreateBlueprintFromContent > ( item , menu , opensDialog : true ) ;
AddActionNode < ActionMove > ( item , menu , true , opensDialog : true ) ;
AddActionNode < ActionCopy > ( item , menu , opensDialog : true ) ;
2018-08-02 13:52:13 +01:00
AddActionNode < ActionSort > ( item , menu , true ) ;
2018-10-30 00:34:43 +11:00
AddActionNode < ActionAssignDomain > ( item , menu , opensDialog : true ) ;
AddActionNode < ActionRights > ( item , menu , opensDialog : true ) ;
2019-01-30 22:01:57 +11:00
AddActionNode < ActionProtect > ( item , menu , true , opensDialog : true ) ;
2019-01-17 12:05:56 +01:00
if ( EmailSender . CanSendRequiredEmail )
2018-10-29 23:23:21 +11:00
{
2019-01-17 12:05:56 +01:00
menu . Items . Add ( new MenuItem ( "notify" , Services . TextService )
{
Icon = "megaphone" ,
2019-01-14 09:32:32 -05:00
SeparatorBefore = true ,
2019-01-17 12:05:56 +01:00
OpensDialog = true
} ) ;
}
2019-05-26 19:06:24 +02:00
if ( ( item is DocumentEntitySlim documentEntity & & documentEntity . IsContainer ) = = false )
{
menu . Items . Add ( new RefreshNode ( Services . TextService , true ) ) ;
}
2018-06-29 19:55:15 +02:00
return menu ;
}
2018-07-11 15:58:48 +10:00
/// <summary>
/// Returns a collection of all menu items that can be on a deleted (in recycle bin) content node
/// </summary>
/// <param name="item"></param>
/// <returns></returns>
protected MenuItemCollection GetNodeMenuItemsForDeletedContent ( IUmbracoEntity item )
{
var menu = new MenuItemCollection ( ) ;
2018-10-30 00:34:43 +11:00
menu . Items . Add < ActionRestore > ( Services . TextService , opensDialog : true ) ;
2018-12-20 16:58:01 +11:00
menu . Items . Add < ActionMove > ( Services . TextService , opensDialog : true ) ;
2018-10-30 00:34:43 +11:00
menu . Items . Add < ActionDelete > ( Services . TextService , opensDialog : true ) ;
2018-07-11 15:58:48 +10:00
2018-10-29 17:27:33 +11:00
menu . Items . Add ( new RefreshNode ( Services . TextService , true ) ) ;
2018-07-11 15:58:48 +10:00
return menu ;
}
2019-11-05 12:54:22 +01:00
2018-06-29 19:55:15 +02:00
/// <summary>
/// set name according to variations
/// </summary>
/// <param name="entity"></param>
/// <param name="culture"></param>
private void EnsureName ( IEntitySlim entity , string culture )
{
if ( culture = = null )
{
if ( string . IsNullOrWhiteSpace ( entity . Name ) )
2019-01-30 22:01:57 +11:00
{
2018-06-29 19:55:15 +02:00
entity . Name = "[[" + entity . Id + "]]" ;
2019-01-30 22:01:57 +11:00
}
2018-06-29 19:55:15 +02:00
return ;
}
2018-05-10 18:20:33 +10:00
2018-05-10 19:16:46 +10:00
if ( ! ( entity is IDocumentEntitySlim docEntity ) )
2019-01-30 22:01:57 +11:00
{
2018-05-10 19:16:46 +10:00
throw new InvalidOperationException ( $"Cannot render a tree node for a culture when the entity isn't {typeof(IDocumentEntitySlim)}, instead it is {entity.GetType()}" ) ;
2019-01-30 22:01:57 +11:00
}
2018-05-10 18:20:33 +10:00
2018-05-10 19:16:46 +10:00
// we are getting the tree for a given culture,
// for those items that DO support cultures, we need to get the proper name, IF it exists
2018-08-17 00:50:05 +10:00
// otherwise, invariant is fine (with brackets)
2018-05-10 18:20:33 +10:00
2018-08-17 00:50:05 +10:00
if ( docEntity . Variations . VariesByCulture ( ) )
2018-05-10 19:16:46 +10:00
{
2018-08-17 00:50:05 +10:00
if ( docEntity . CultureNames . TryGetValue ( culture , out var name ) & &
! string . IsNullOrWhiteSpace ( name ) )
{
entity . Name = name ;
}
else
{
entity . Name = "(" + entity . Name + ")" ;
}
2018-05-10 18:20:33 +10:00
}
2018-05-10 19:16:46 +10:00
if ( string . IsNullOrWhiteSpace ( entity . Name ) )
2019-01-30 22:01:57 +11:00
{
2018-06-29 19:55:15 +02:00
entity . Name = "[[" + entity . Id + "]]" ;
2019-01-30 22:01:57 +11:00
}
2018-06-29 19:55:15 +02:00
}
2019-01-30 22:01:57 +11:00
private void AddActionNode < TAction > ( IUmbracoEntity item , MenuItemCollection menu , bool hasSeparator = false , bool opensDialog = false )
2018-06-29 19:55:15 +02:00
where TAction : IAction
{
2019-01-30 22:01:57 +11:00
var menuItem = menu . Items . Add < TAction > ( Services . TextService . Localize ( "actions" , _actions . GetAction < TAction > ( ) . Alias ) , hasSeparator , opensDialog ) ;
2018-06-29 19:55:15 +02:00
}
2018-12-04 14:25:37 +11:00
public IEnumerable < SearchResultEntity > Search ( string query , int pageSize , long pageIndex , out long totalFound , string searchFrom = null )
2018-06-29 19:55:15 +02:00
{
2019-06-13 23:47:48 +10:00
return _treeSearcher . ExamineSearch ( query , UmbracoEntityTypes . Document , pageSize , pageIndex , out totalFound , searchFrom ) ;
2018-05-10 18:20:33 +10:00
}
2018-06-29 19:55:15 +02:00
}
}