Skip to content

Commit

Permalink
Merge pull request #829 from sbwalker/master
Browse files Browse the repository at this point in the history
introduce Resource Declaration and Location properties to offer more resource management options for developers
  • Loading branch information
sbwalker authored Oct 19, 2020
2 parents 34122a3 + ecacb68 commit 2abc2cd
Show file tree
Hide file tree
Showing 9 changed files with 134 additions and 28 deletions.
13 changes: 8 additions & 5 deletions Oqtane.Client/Modules/ModuleBase.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
using Microsoft.AspNetCore.Components;
using Microsoft.AspNetCore.Components;
using Oqtane.Shared;
using Oqtane.Models;
using System.Threading.Tasks;
Expand Down Expand Up @@ -53,12 +53,15 @@ protected override async Task OnAfterRenderAsync(bool firstRender)
if (Resources != null && Resources.Exists(item => item.ResourceType == ResourceType.Script))
{
var scripts = new List<object>();
foreach (Resource resource in Resources.Where(item => item.ResourceType == ResourceType.Script))
foreach (Resource resource in Resources.Where(item => item.ResourceType == ResourceType.Script && item.Declaration != ResourceDeclaration.Global))
{
scripts.Add(new { href = resource.Url, bundle = resource.Bundle ?? "", integrity = resource.Integrity ?? "", crossorigin = resource.CrossOrigin ?? "" });
}
var interop = new Interop(JSRuntime);
await interop.IncludeScripts(scripts.ToArray());
if (scripts.Any())
{
var interop = new Interop(JSRuntime);
await interop.IncludeScripts(scripts.ToArray());
}
}
}
}
Expand Down Expand Up @@ -292,4 +295,4 @@ public async Task LogCritical(Exception exception, string message, params object
}
}
}
}
}
9 changes: 6 additions & 3 deletions Oqtane.Client/Themes/ThemeBase.cs
Original file line number Diff line number Diff line change
Expand Up @@ -32,12 +32,15 @@ protected override async Task OnAfterRenderAsync(bool firstRender)
if (Resources != null && Resources.Exists(item => item.ResourceType == ResourceType.Script))
{
var scripts = new List<object>();
foreach (Resource resource in Resources.Where(item => item.ResourceType == ResourceType.Script))
foreach (Resource resource in Resources.Where(item => item.ResourceType == ResourceType.Script && item.Declaration != ResourceDeclaration.Global))
{
scripts.Add(new { href = resource.Url, bundle = resource.Bundle ?? "", integrity = resource.Integrity ?? "", crossorigin = resource.CrossOrigin ?? "" });
}
var interop = new Interop(JSRuntime);
await interop.IncludeScripts(scripts.ToArray());
if (scripts.Any())
{
var interop = new Interop(JSRuntime);
await interop.IncludeScripts(scripts.ToArray());
}
}
}
}
Expand Down
9 changes: 6 additions & 3 deletions Oqtane.Client/UI/ThemeBuilder.razor
Original file line number Diff line number Diff line change
Expand Up @@ -33,11 +33,14 @@
// manage stylesheets for this page
string batch = DateTime.Now.ToString("yyyyMMddHHmmssfff");
var links = new List<object>();
foreach (Resource resource in PageState.Page.Resources.Where(item => item.ResourceType == ResourceType.Stylesheet))
foreach (Resource resource in PageState.Page.Resources.Where(item => item.ResourceType == ResourceType.Stylesheet && item.Declaration != ResourceDeclaration.Global))
{
links.Add(new { id = "app-stylesheet-" + batch + "-" + (links.Count + 1).ToString("00"), rel = "stylesheet", href = resource.Url, type = "text/css", integrity = resource.Integrity ?? "", crossorigin = resource.CrossOrigin ?? "", key = "" });
}
await interop.IncludeLinks(links.ToArray());
if (links.Any())
{
await interop.IncludeLinks(links.ToArray());
}
await interop.RemoveElementsById("app-stylesheet", "", "app-stylesheet-" + batch + "-00");

// add favicon
Expand All @@ -46,7 +49,7 @@
await interop.IncludeLink("app-favicon", "shortcut icon", Utilities.ContentUrl(PageState.Alias, PageState.Site.FaviconFileId.Value), "image/x-icon", "", "", "id");
}
// add PWA support
if (PageState.Site.PwaIsEnabled)
if (PageState.Site.PwaIsEnabled && PageState.Site.PwaAppIconFileId != null && PageState.Site.PwaSplashIconFileId != null)
{
await InitializePwa(interop);
}
Expand Down
4 changes: 3 additions & 1 deletion Oqtane.Server/Infrastructure/Interfaces/IHostResources.cs
Original file line number Diff line number Diff line change
@@ -1,8 +1,10 @@
using Oqtane.Models;
using Oqtane.Models;
using System.Collections.Generic;

namespace Oqtane.Infrastructure
{
// Obsolete - use the Resources collection as part of IModuleControl or IThemeCOntrol and use the ResourceDeclaration.Global property
// note - not using the [Obsolete] attribute in this case as we want to avoid compilation warnings in the core framework
public interface IHostResources
{
List<Resource> Resources { get; } // identifies global resources for an application
Expand Down
5 changes: 3 additions & 2 deletions Oqtane.Server/Pages/_Host.cshtml
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
@page "/"
@page "/"
@namespace Oqtane.Pages
@addTagHelper *, Microsoft.AspNetCore.Mvc.TagHelpers
@using System.Globalization
Expand All @@ -24,7 +24,7 @@
<link id="app-manifest" rel="manifest" />
<link rel="stylesheet" href="css/app.css" />
<script src="js/loadjs.min.js"></script>
@Html.Raw(@Model.Resources)
@Html.Raw(@Model.HeadResources)
</head>
<body>
@(Html.AntiForgeryToken())
Expand Down Expand Up @@ -53,5 +53,6 @@
{
<script src="_framework/blazor.server.js"></script>
}
@Html.Raw(@Model.BodyResources)
</body>
</html>
102 changes: 89 additions & 13 deletions Oqtane.Server/Pages/_Host.cshtml.cs
Original file line number Diff line number Diff line change
@@ -1,6 +1,9 @@
using Microsoft.AspNetCore.Mvc.RazorPages;
using Microsoft.AspNetCore.Mvc.RazorPages;
using Oqtane.Infrastructure;
using Oqtane.Shared;
using Oqtane.Modules;
using Oqtane.Models;
using Oqtane.Themes;
using System;
using System.Linq;
using System.Reflection;
Expand All @@ -9,33 +12,106 @@ namespace Oqtane.Pages
{
public class HostModel : PageModel
{
public string Resources = "";
public string HeadResources = "";
public string BodyResources = "";

public void OnGet()
{
var assemblies = AppDomain.CurrentDomain.GetOqtaneAssemblies();
foreach (Assembly assembly in assemblies)
{
var types = assembly.GetTypes().Where(item => item.GetInterfaces().Contains(typeof(IHostResources)));
foreach (var type in types)
ProcessHostResources(assembly);
ProcessModuleControls(assembly);
ProcessThemeControls(assembly);
}
}

private void ProcessHostResources(Assembly assembly)
{
var types = assembly.GetTypes().Where(item => item.GetInterfaces().Contains(typeof(IHostResources)));
foreach (var type in types)
{
var obj = Activator.CreateInstance(type) as IHostResources;
foreach (var resource in obj.Resources)
{
ProcessResource(resource);
}
}
}

private void ProcessModuleControls(Assembly assembly)
{
var types = assembly.GetTypes().Where(item => item.GetInterfaces().Contains(typeof(IModuleControl)));
foreach (var type in types)
{
// Check if type should be ignored
if (type.IsOqtaneIgnore()) continue;

var obj = Activator.CreateInstance(type) as IModuleControl;
if (obj.Resources != null)
{
var obj = Activator.CreateInstance(type) as IHostResources;
foreach (var resource in obj.Resources)
{
switch (resource.ResourceType)
if (resource.Declaration == ResourceDeclaration.Global)
{
case ResourceType.Stylesheet:
Resources += "<link rel=\"stylesheet\" href=\"" + resource.Url + "\"" + CrossOrigin(resource.CrossOrigin) + Integrity(resource.Integrity) + " />" + Environment.NewLine;
break;
case ResourceType.Script:
Resources += "<script src=\"" + resource.Url + "\"" + CrossOrigin(resource.CrossOrigin) + Integrity(resource.Integrity) + "></script>" + Environment.NewLine;
break;
ProcessResource(resource);
}
}
}
}
}

private void ProcessThemeControls(Assembly assembly)
{
var types = assembly.GetTypes().Where(item => item.GetInterfaces().Contains(typeof(IThemeControl)));
foreach (var type in types)
{
// Check if type should be ignored
if (type.IsOqtaneIgnore()) continue;

var obj = Activator.CreateInstance(type) as IThemeControl;
if (obj.Resources != null)
{
foreach (var resource in obj.Resources)
{
if (resource.Declaration == ResourceDeclaration.Global)
{
ProcessResource(resource);
}
}
}
}
}

private void ProcessResource(Resource resource)
{
switch (resource.ResourceType)
{
case ResourceType.Stylesheet:
if (!HeadResources.Contains(resource.Url, StringComparison.OrdinalIgnoreCase))
{
HeadResources += "<link rel=\"stylesheet\" href=\"" + resource.Url + "\"" + CrossOrigin(resource.CrossOrigin) + Integrity(resource.Integrity) + " />" + Environment.NewLine;
}
break;
case ResourceType.Script:
if (resource.Location == Shared.ResourceLocation.Body)
{
if (!BodyResources.Contains(resource.Url, StringComparison.OrdinalIgnoreCase))
{
BodyResources += "<script src=\"" + resource.Url + "\"" + CrossOrigin(resource.CrossOrigin) + Integrity(resource.Integrity) + "></script>" + Environment.NewLine;
}
}
else
{
if (!HeadResources.Contains(resource.Url, StringComparison.OrdinalIgnoreCase))
{
HeadResources += "<script src=\"" + resource.Url + "\"" + CrossOrigin(resource.CrossOrigin) + Integrity(resource.Integrity) + "></script>" + Environment.NewLine;
}
}
break;
}
}

private string CrossOrigin(string crossorigin)
{
if (!string.IsNullOrEmpty(crossorigin))
Expand All @@ -59,4 +135,4 @@ private string Integrity(string integrity)
}
}
}
}
}
8 changes: 8 additions & 0 deletions Oqtane.Shared/Enums/ResourceDeclaration.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
namespace Oqtane.Shared
{
public enum ResourceDeclaration
{
Local,
Global
}
}
8 changes: 8 additions & 0 deletions Oqtane.Shared/Enums/ResourceLocation.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
namespace Oqtane.Shared
{
public enum ResourceLocation
{
Head,
Body
}
}
4 changes: 3 additions & 1 deletion Oqtane.Shared/Models/Resource.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
using Oqtane.Shared;
using Oqtane.Shared;

namespace Oqtane.Models
{
Expand All @@ -9,5 +9,7 @@ public class Resource
public string Integrity { get; set; }
public string CrossOrigin { get; set; }
public string Bundle { get; set; }
public ResourceDeclaration Declaration { get; set; }
public ResourceLocation Location { get; set; }
}
}

0 comments on commit 2abc2cd

Please sign in to comment.