Merge pull request #6281 from abpframework/refactor-abp-bundle

Bundle command refactoring
pull/6290/head
Halil İbrahim Kalkan 5 years ago committed by GitHub
commit e8ed831aa0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -4,9 +4,9 @@ Some modules may require additional styles or scripts that need to be referenced
To update script & style references without worrying about dependencies, ordering, etc in a project, you can use the [bundle command](../../CLI.md#bundle).
You can also add custom styles and scripts and let ABP manage them for you. In your Blazor project, you can create a class implementing `IBundleContributer` interface.
You can also add custom styles and scripts and let ABP manage them for you. In your Blazor project, you can create a class implementing `IBundleContributor` interface.
`IBundleContributer` interface contains two methods.
`IBundleContributor` interface contains two methods.
* `AddScripts(...)`
* `AddStyles(...)`
@ -17,7 +17,7 @@ Both methods get `BundleContext` as a parameter. You can add scripts and styles
```csharp
namespace MyProject.Blazor
{
public class MyProjectBundleContributer : IBundleContributer
public class MyProjectBundleContributor : IBundleContributor
{
public void AddScripts(BundleContext context)
{
@ -33,6 +33,6 @@ namespace MyProject.Blazor
}
```
> There is a BundleContributer class implementing `IBundleContributer` interface coming by default with the startup templates. So, most of the time, you don't need to add it manually.
> There is a BundleContributor class implementing `IBundleContributor` interface coming by default with the startup templates. So, most of the time, you don't need to add it manually.
> Bundle command adds style and script references individually. Bundling and minification support will be added to incoming releases.

@ -2,7 +2,7 @@
namespace Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme
{
public class BasicThemeBundleContributer : IBundleContributer
public class BasicThemeBundleContributor : IBundleContributor
{
public void AddScripts(BundleContext context)
{

@ -2,7 +2,7 @@
namespace Volo.Abp.AspNetCore.Components.WebAssembly.Theming
{
public class ThemingBundleContributer : IBundleContributer
public class ThemingBundleContributor : IBundleContributor
{
public void AddScripts(BundleContext context)
{

@ -2,7 +2,7 @@
namespace Volo.Abp.AspNetCore.Components.WebAssembly
{
public class ComponentsWebAssemblyBundleContributer : IBundleContributer
public class ComponentsWebAssemblyBundleContributor : IBundleContributor
{
public void AddScripts(BundleContext context)
{

@ -2,7 +2,7 @@
namespace Volo.Abp.BlazoriseUI
{
public class BlazoriseUIBundleContributer : IBundleContributer
public class BlazoriseUIBundleContributor : IBundleContributor
{
public void AddScripts(BundleContext context)
{

@ -5,6 +5,7 @@ namespace Volo.Abp.Cli.Bundling
internal class BundleTypeDefinition
{
public int Level { get; set; }
public Type BundleContributerType { get; set; }
}
}

@ -1,10 +1,9 @@
using System;
namespace Volo.Abp.Cli.Bundling
namespace Volo.Abp.Cli.Bundling
{
public class BundlingException : Exception
public class BundlingException : AbpException
{
public BundlingException(string message) : base(message)
public BundlingException(string message)
: base(message)
{
}

@ -1,5 +1,4 @@
using Microsoft.CodeAnalysis.CSharp;
using System;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
@ -29,7 +28,7 @@ namespace Volo.Abp.Cli.Bundling
var projectFiles = Directory.GetFiles(directory, "*.csproj");
if (!projectFiles.Any())
{
throw new BundlingException("No project file found in the directory");
throw new BundlingException("No project file found in the directory. The working directory must have a Blazor project file.");
}
var projectFilePath = projectFiles[0];
@ -40,7 +39,7 @@ namespace Volo.Abp.Cli.Bundling
{
new DotNetProjectInfo(string.Empty, projectFilePath, true)
};
DotNetProjectBuilder.BuildProjects(projects, string.Empty);
}
@ -63,11 +62,13 @@ namespace Volo.Abp.Cli.Bundling
private BundleContext GetScriptContext(List<BundleTypeDefinition> bundleDefinitions)
{
var scriptContext = new BundleContext();
foreach (var bundleDefinition in bundleDefinitions)
{
var contributer = CreateContributerInstance(bundleDefinition.BundleContributerType);
contributer.AddScripts(scriptContext);
}
scriptContext.Add("_framework/blazor.webassembly.js");
return scriptContext;
}
@ -75,11 +76,13 @@ namespace Volo.Abp.Cli.Bundling
private BundleContext GetStyleContext(List<BundleTypeDefinition> bundleDefinitions)
{
var styleContext = new BundleContext();
foreach (var bundleDefinition in bundleDefinitions)
{
var contributer = CreateContributerInstance(bundleDefinition.BundleContributerType);
contributer.AddStyles(styleContext);
};
}
return styleContext;
}
@ -102,9 +105,11 @@ namespace Volo.Abp.Cli.Bundling
content = UpdatePlaceholders(content, StylePlaceholderStart, StylePlaceholderEnd, styleDefinitions);
content = UpdatePlaceholders(content, ScriptPlaceholderStart, ScriptPlaceholderEnd, scriptDefinitions);
using var writer = new StreamWriter(htmlFilePath, false, fileEncoding);
await writer.WriteAsync(content);
await writer.FlushAsync();
using (var writer = new StreamWriter(htmlFilePath, false, fileEncoding))
{
await writer.WriteAsync(content);
await writer.FlushAsync();
}
}
private string UpdatePlaceholders(string content, string placeholderStart, string placeholderEnd, string definitions)
@ -118,24 +123,22 @@ namespace Volo.Abp.Cli.Bundling
private string GenerateStyleDefinitions(BundleContext context)
{
var builder = new StringBuilder();
builder.AppendLine($"{StylePlaceholderStart}");
foreach (var style in context.BundleDefinitions)
{
if (style.AdditionalProperties != null && style.AdditionalProperties.Any())
{
builder.Append($"\t<link href=\"{style.Source}\" rel=\"stylesheet\" ");
foreach (var additionalProperty in style.AdditionalProperties)
{
builder.Append($"{additionalProperty.Key}={additionalProperty.Value} ");
}
builder.AppendLine("/>");
}
else
builder.Append($" <link href=\"{style.Source}\" rel=\"stylesheet\"");
foreach (var additionalProperty in style.AdditionalProperties)
{
builder.AppendLine($"\t<link href=\"{style.Source}\" rel=\"stylesheet\" />");
builder.Append($"{additionalProperty.Key}={additionalProperty.Value} ");
}
builder.AppendLine("/>");
}
builder.Append($"\t{StylePlaceholderEnd}");
builder.Append($" {StylePlaceholderEnd}");
return builder.ToString();
}
@ -146,50 +149,41 @@ namespace Volo.Abp.Cli.Bundling
builder.AppendLine($"{ScriptPlaceholderStart}");
foreach (var script in context.BundleDefinitions)
{
if (script.AdditionalProperties != null && script.AdditionalProperties.Any())
{
builder.Append($"\t<script src=\"{script.Source}\" ");
foreach (var additionalProperty in script.AdditionalProperties)
{
builder.Append($"{additionalProperty.Key}={additionalProperty.Value} ");
}
builder.AppendLine("></script>");
}
else
builder.Append($" <script src=\"{script.Source}\"");
foreach (var additionalProperty in script.AdditionalProperties)
{
builder.AppendLine($"\t<script src=\"{script.Source}\"></script>");
builder.Append($"{additionalProperty.Key}={additionalProperty.Value} ");
}
builder.AppendLine("></script>");
}
builder.Append($"\t{ScriptPlaceholderEnd}");
builder.Append($" {ScriptPlaceholderEnd}");
return builder.ToString();
}
private IBundleContributer CreateContributerInstance(Type bundleContributerType)
private IBundleContributor CreateContributerInstance(Type bundleContributerType)
{
var instance = Activator.CreateInstance(bundleContributerType);
return instance.As<IBundleContributer>();
return (IBundleContributor)Activator.CreateInstance(bundleContributerType);
}
private void ExecuteBundleContributers(List<BundleTypeDefinition> bundleDefinitions, BundleContext styleContext, BundleContext scriptContext)
private void FindBundleContributersRecursively(
Type module,
int level,
List<BundleTypeDefinition> bundleDefinitions)
{
foreach (var bundleDefinition in bundleDefinitions)
var bundleContributers = module.Assembly
.GetTypes()
.Where(t => t.IsAssignableTo<IBundleContributor>())
.ToList();
if (bundleContributers.Count > 1)
{
var contributer = CreateContributerInstance(bundleDefinition.BundleContributerType);
contributer.AddStyles(styleContext);
contributer.AddScripts(scriptContext);
throw new BundlingException($"Each project must contain only one class implementing {nameof(IBundleContributor)}");
}
}
private void FindBundleContributersRecursively(Type module, int level, List<BundleTypeDefinition> bundleDefinitions)
{
var dependencyDescriptors = module
.GetCustomAttributes()
.OfType<IDependedTypesProvider>();
var bundleContributer = module.Assembly.GetTypes().SingleOrDefault(t => t.IsAssignableTo<IBundleContributer>());
if (bundleContributer != null)
if (bundleContributers.Any())
{
var bundleContributer = bundleContributers[0];
var definition = bundleDefinitions.SingleOrDefault(t => t.BundleContributerType == bundleContributer);
if (definition != null)
{
@ -208,6 +202,10 @@ namespace Volo.Abp.Cli.Bundling
}
}
var dependencyDescriptors = module
.GetCustomAttributes()
.OfType<IDependedTypesProvider>();
foreach (var descriptor in dependencyDescriptors)
{
foreach (var dependedModuleType in descriptor.GetDependedTypes())
@ -219,19 +217,10 @@ namespace Volo.Abp.Cli.Bundling
private Type GetStartupModule(string assemblyPath)
{
var assembly = Assembly.LoadFrom(assemblyPath);
return assembly.GetTypes().SingleOrDefault(IsAbpModule);
static bool IsAbpModule(Type type)
{
var typeInfo = type.GetTypeInfo();
return
typeInfo.IsClass &&
!typeInfo.IsAbstract &&
!typeInfo.IsGenericType &&
typeof(IAbpModule).GetTypeInfo().IsAssignableFrom(type);
}
return Assembly
.LoadFrom(assemblyPath)
.GetTypes()
.SingleOrDefault(AbpModule.IsAbpModule);
}
private string GetFrameworkFolderPath(string projectDirectory, string frameworkVersion)
@ -244,15 +233,12 @@ namespace Volo.Abp.Cli.Bundling
var document = new XmlDocument();
document.Load(projectFilePath);
var sdk = document.DocumentElement.GetAttribute("Sdk");
if (sdk == SupportedWebAssemblyProjectType)
{
var frameworkVersion = document.SelectSingleNode("//TargetFramework").InnerText;
return frameworkVersion;
}
else
if (sdk != SupportedWebAssemblyProjectType)
{
throw new BundlingException($"Unsupported project type. Project type must be {SupportedWebAssemblyProjectType}.");
}
return document.SelectSingleNode("//TargetFramework").InnerText;
}
private string GetAssemblyFilePath(string directory, string frameworkVersion, string projectFileName)

@ -1,5 +1,4 @@
using System;
using System.Collections.Generic;
using System.Collections.Generic;
namespace Volo.Abp.Bundling
{
@ -17,9 +16,13 @@ namespace Volo.Abp.Bundling
var bundleDefinition = new BundleDefinition
{
Source = source,
AdditionalProperties = additionalProperties
};
if (additionalProperties != null)
{
bundleDefinition.AdditionalProperties = additionalProperties;
}
BundleDefinitions.AddIfNotContains((item) => item.Source == bundleDefinition.Source, () => bundleDefinition);
}
}

@ -5,6 +5,12 @@ namespace Volo.Abp.Bundling
public class BundleDefinition
{
public string Source { get; set; }
public Dictionary<string, string> AdditionalProperties { get; set; }
public BundleDefinition()
{
AdditionalProperties = new Dictionary<string, string>();
}
}
}

@ -1,6 +1,6 @@
namespace Volo.Abp.Bundling
{
public interface IBundleContributer
public interface IBundleContributor
{
void AddScripts(BundleContext context);
void AddStyles(BundleContext context);

@ -2,7 +2,7 @@
namespace Volo.Abp.Http.Client.IdentityModel.WebAssembly
{
public class IdentityModelWebAssemblyBundleContributer : IBundleContributer
public class IdentityModelWebAssemblyBundleContributor : IBundleContributor
{
public void AddScripts(BundleContext context)
{

@ -2,7 +2,7 @@
namespace MyCompanyName.MyProjectName.Blazor
{
public class MyProjectNameBundleContributer : IBundleContributer
public class MyProjectNameBundleContributor : IBundleContributor
{
public void AddScripts(BundleContext context)
{

@ -8,14 +8,14 @@
<base href="/" />
<!--ABP:Styles-->
<link href="_content/Volo.Abp.AspNetCore.Components.WebAssembly.Theming/libs/bootstrap/css/bootstrap.min.css" rel="stylesheet" />
<link href="_content/Volo.Abp.AspNetCore.Components.WebAssembly.Theming/libs/fontawesome/css/all.css" rel="stylesheet" />
<link href="_content/Blazorise/blazorise.css" rel="stylesheet" />
<link href="_content/Blazorise.Bootstrap/blazorise.bootstrap.css" rel="stylesheet" />
<link href="_content/Blazorise.Snackbar/blazorise.snackbar.css" rel="stylesheet" />
<link href="_content/Volo.Abp.AspNetCore.Components.WebAssembly.Theming/libs/flag-icon/css/flag-icon.css" rel="stylesheet" />
<link href="_content/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/libs/abp/css/theme.css" rel="stylesheet" />
<link href="main.css" rel="stylesheet" />
<link href="_content/Volo.Abp.AspNetCore.Components.WebAssembly.Theming/libs/bootstrap/css/bootstrap.min.css" rel="stylesheet"/>
<link href="_content/Volo.Abp.AspNetCore.Components.WebAssembly.Theming/libs/fontawesome/css/all.css" rel="stylesheet"/>
<link href="_content/Blazorise/blazorise.css" rel="stylesheet"/>
<link href="_content/Blazorise.Bootstrap/blazorise.bootstrap.css" rel="stylesheet"/>
<link href="_content/Blazorise.Snackbar/blazorise.snackbar.css" rel="stylesheet"/>
<link href="_content/Volo.Abp.AspNetCore.Components.WebAssembly.Theming/libs/flag-icon/css/flag-icon.css" rel="stylesheet"/>
<link href="_content/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/libs/abp/css/theme.css" rel="stylesheet"/>
<link href="main.css" rel="stylesheet"/>
<!--/ABP:Styles-->
</head>

Loading…
Cancel
Save