269 lines
9.1 KiB
C#
269 lines
9.1 KiB
C#
using Microsoft.AspNetCore.Authorization;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.Extensions.DependencyInjection;
|
|
using Umbraco.Cms.Core;
|
|
using Umbraco.Cms.Core.DependencyInjection;
|
|
using Umbraco.Cms.Core.IO;
|
|
using Umbraco.Cms.Core.Mapping;
|
|
using Umbraco.Cms.Core.Models;
|
|
using Umbraco.Cms.Core.Models.ContentEditing;
|
|
using Umbraco.Cms.Core.Services;
|
|
using Umbraco.Cms.Core.Strings;
|
|
using Umbraco.Cms.Web.Common.Attributes;
|
|
using Umbraco.Cms.Web.Common.Authorization;
|
|
using Umbraco.Extensions;
|
|
|
|
namespace Umbraco.Cms.Web.BackOffice.Controllers;
|
|
|
|
[PluginController(Constants.Web.Mvc.BackOfficeApiArea)]
|
|
[Authorize(Policy = AuthorizationPolicies.TreeAccessTemplates)]
|
|
[ParameterSwapControllerActionSelector(nameof(GetById), "id", typeof(int), typeof(Guid), typeof(Udi))]
|
|
public class TemplateController : BackOfficeNotificationsController
|
|
{
|
|
private readonly IDefaultViewContentProvider _defaultViewContentProvider;
|
|
private readonly IFileService _fileService;
|
|
private readonly IShortStringHelper _shortStringHelper;
|
|
private readonly IUmbracoMapper _umbracoMapper;
|
|
|
|
[ActivatorUtilitiesConstructor]
|
|
public TemplateController(
|
|
IFileService fileService,
|
|
IUmbracoMapper umbracoMapper,
|
|
IShortStringHelper shortStringHelper,
|
|
IDefaultViewContentProvider defaultViewContentProvider)
|
|
{
|
|
_fileService = fileService ?? throw new ArgumentNullException(nameof(fileService));
|
|
_umbracoMapper = umbracoMapper ?? throw new ArgumentNullException(nameof(umbracoMapper));
|
|
_shortStringHelper = shortStringHelper ?? throw new ArgumentNullException(nameof(shortStringHelper));
|
|
_defaultViewContentProvider = defaultViewContentProvider ??
|
|
throw new ArgumentNullException(nameof(defaultViewContentProvider));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets data type by alias
|
|
/// </summary>
|
|
/// <param name="alias"></param>
|
|
/// <returns></returns>
|
|
public TemplateDisplay? GetByAlias(string alias)
|
|
{
|
|
ITemplate? template = _fileService.GetTemplate(alias);
|
|
return template == null ? null : _umbracoMapper.Map<ITemplate, TemplateDisplay>(template);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get all templates
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
public IEnumerable<EntityBasic>? GetAll() => _fileService.GetTemplates()
|
|
?.Select(_umbracoMapper.Map<ITemplate, EntityBasic>).WhereNotNull();
|
|
|
|
/// <summary>
|
|
/// Gets the template json for the template id
|
|
/// </summary>
|
|
/// <param name="id"></param>
|
|
/// <returns></returns>
|
|
public ActionResult<TemplateDisplay?> GetById(int id)
|
|
{
|
|
ITemplate? template = _fileService.GetTemplate(id);
|
|
if (template == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
return _umbracoMapper.Map<ITemplate, TemplateDisplay>(template);
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Gets the template json for the template guid
|
|
/// </summary>
|
|
/// <param name="id"></param>
|
|
/// <returns></returns>
|
|
public ActionResult<TemplateDisplay?> GetById(Guid id)
|
|
{
|
|
ITemplate? template = _fileService.GetTemplate(id);
|
|
if (template == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
return _umbracoMapper.Map<ITemplate, TemplateDisplay>(template);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the template json for the template udi
|
|
/// </summary>
|
|
/// <param name="id"></param>
|
|
/// <returns></returns>
|
|
public ActionResult<TemplateDisplay?> GetById(Udi id)
|
|
{
|
|
var guidUdi = id as GuidUdi;
|
|
if (guidUdi == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
ITemplate? template = _fileService.GetTemplate(guidUdi.Guid);
|
|
if (template == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
return _umbracoMapper.Map<ITemplate, TemplateDisplay>(template);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Deletes a template with a given ID
|
|
/// </summary>
|
|
/// <param name="id"></param>
|
|
/// <returns></returns>
|
|
[HttpDelete]
|
|
[HttpPost]
|
|
public IActionResult DeleteById(int id)
|
|
{
|
|
ITemplate? template = _fileService.GetTemplate(id);
|
|
if (template == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
_fileService.DeleteTemplate(template.Alias);
|
|
return Ok();
|
|
}
|
|
|
|
public TemplateDisplay? GetScaffold(int id)
|
|
{
|
|
//empty default
|
|
var dt = new Template(_shortStringHelper, string.Empty, string.Empty)
|
|
{
|
|
Path = "-1"
|
|
};
|
|
|
|
if (id > 0)
|
|
{
|
|
ITemplate? master = _fileService.GetTemplate(id);
|
|
if (master != null)
|
|
{
|
|
dt.SetMasterTemplate(master);
|
|
}
|
|
}
|
|
|
|
var content = _defaultViewContentProvider.GetDefaultFileContent(dt.MasterTemplateAlias);
|
|
TemplateDisplay? scaffold = _umbracoMapper.Map<ITemplate, TemplateDisplay>(dt);
|
|
|
|
if (scaffold is not null)
|
|
{
|
|
scaffold.Content = content;
|
|
}
|
|
|
|
return scaffold;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Saves the data type
|
|
/// </summary>
|
|
/// <param name="display"></param>
|
|
/// <returns></returns>
|
|
public ActionResult<TemplateDisplay> PostSave(TemplateDisplay display)
|
|
{
|
|
//Checking the submitted is valid with the Required attributes decorated on the ViewModel
|
|
if (ModelState.IsValid == false)
|
|
{
|
|
return ValidationProblem(ModelState);
|
|
}
|
|
|
|
if (display.Id > 0)
|
|
{
|
|
// update
|
|
ITemplate? template = _fileService.GetTemplate(display.Id);
|
|
if (template == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
var changeMaster = template.MasterTemplateAlias != display.MasterTemplateAlias;
|
|
var changeAlias = template.Alias != display.Alias;
|
|
|
|
_umbracoMapper.Map(display, template);
|
|
|
|
if (changeMaster)
|
|
{
|
|
if (string.IsNullOrEmpty(display.MasterTemplateAlias) == false)
|
|
{
|
|
ITemplate? master = _fileService.GetTemplate(display.MasterTemplateAlias);
|
|
if (master == null || master.Id == display.Id)
|
|
{
|
|
template.SetMasterTemplate(null);
|
|
}
|
|
else
|
|
{
|
|
template.SetMasterTemplate(master);
|
|
|
|
//After updating the master - ensure we update the path property if it has any children already assigned
|
|
IEnumerable<ITemplate> templateHasChildren = _fileService.GetTemplateDescendants(display.Id);
|
|
|
|
foreach (ITemplate childTemplate in templateHasChildren)
|
|
{
|
|
//template ID to find
|
|
var templateIdInPath = "," + display.Id + ",";
|
|
|
|
if (string.IsNullOrEmpty(childTemplate.Path))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
//Find position in current comma separate string path (so we get the correct children path)
|
|
var positionInPath = childTemplate.Path.IndexOf(templateIdInPath) + templateIdInPath.Length;
|
|
|
|
//Get the substring of the child & any children (descendants it may have too)
|
|
var childTemplatePath = childTemplate.Path.Substring(positionInPath);
|
|
|
|
//As we are updating the template to be a child of a master
|
|
//Set the path to the master's path + its current template id + the current child path substring
|
|
childTemplate.Path = master.Path + "," + display.Id + "," + childTemplatePath;
|
|
|
|
//Save the children with the updated path
|
|
_fileService.SaveTemplate(childTemplate);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//remove the master
|
|
template.SetMasterTemplate(null);
|
|
}
|
|
}
|
|
|
|
_fileService.SaveTemplate(template);
|
|
|
|
if (changeAlias)
|
|
{
|
|
template = _fileService.GetTemplate(template.Id);
|
|
}
|
|
|
|
_umbracoMapper.Map(template, display);
|
|
}
|
|
else
|
|
{
|
|
//create
|
|
ITemplate? master = null;
|
|
if (string.IsNullOrEmpty(display.MasterTemplateAlias) == false)
|
|
{
|
|
master = _fileService.GetTemplate(display.MasterTemplateAlias);
|
|
if (master == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
}
|
|
|
|
// we need to pass the template name as alias to keep the template file casing consistent with templates created with content
|
|
// - see comment in FileService.CreateTemplateForContentType for additional details
|
|
ITemplate template =
|
|
_fileService.CreateTemplateWithIdentity(display.Name, display.Name, display.Content, master);
|
|
_umbracoMapper.Map(template, display);
|
|
}
|
|
|
|
return display;
|
|
}
|
|
}
|