mirror of https://github.com/abpframework/abp
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
353 lines
14 KiB
353 lines
14 KiB
|
|
# ASP.NET Core MVC Bundling & Minification
|
|
|
|
There are many ways of bundling & minification of client side resources (JavaScript and CSS files). Most common ways are:
|
|
|
|
* Using the [Bundler & Minifier](https://marketplace.visualstudio.com/items?itemName=MadsKristensen.BundlerMinifier) Visual Studio extension or the [NuGet package](https://www.nuget.org/packages/BuildBundlerMinifier/).
|
|
* Using [Gulp](https://gulpjs.com/)/[Grunt](https://gruntjs.com/) task managers and their plugins.
|
|
|
|
ABP offers a simple, dynamic, powerful, modular and built-in way.
|
|
|
|
## Volo.Abp.AspNetCore.Mvc.UI.Bundling Package
|
|
|
|
> This package is already installed by default with the startup templates. So, most of the time, you don't need to install it manually.
|
|
|
|
Install the `Volo.Abp.AspNetCore.Mvc.UI.Bundling` nuget package to your project:
|
|
|
|
````
|
|
install-package Volo.Abp.AspNetCore.Mvc.UI.Bundling
|
|
````
|
|
|
|
Then you can add the `AbpAspNetCoreMvcUiBundlingModule` dependency to your module:
|
|
|
|
````C#
|
|
using Volo.Abp.Modularity;
|
|
using Volo.Abp.AspNetCore.Mvc.UI.Bundling;
|
|
|
|
namespace MyCompany.MyProject
|
|
{
|
|
[DependsOn(typeof(AbpAspNetCoreMvcUiBundlingModule))]
|
|
public class MyWebModule : AbpModule
|
|
{
|
|
//...
|
|
}
|
|
}
|
|
````
|
|
|
|
## Razor Bundling Tag Helpers
|
|
|
|
The simplest way of creating a bundle is to use `abp-script-bundle` or `abp-style-bundle` tag helpers. Example:
|
|
|
|
````html
|
|
<abp-style-bundle name="MyGlobalBundle">
|
|
<abp-style src="/libs/bootstrap/css/bootstrap.css" />
|
|
<abp-style src="/libs/font-awesome/css/font-awesome.css" />
|
|
<abp-style src="/libs/toastr/toastr.css" />
|
|
<abp-style src="/styles/my-global-style.css" />
|
|
</abp-style-bundle>
|
|
````
|
|
|
|
This bundle defines a style bundle with a **unique name**: `MyGlobalBundle`. It's very easy to understand how to use it. Let's see how it *works*:
|
|
|
|
* ABP creates the bundle as **lazy** from the provided files when it's **first requested**. For the subsequent calls, it's returned from the **cache**. That means if you conditionally add the files to the bundle, it's executed only once and any changes of the condition will not effect the bundle for the next requests.
|
|
* ABP adds bundle files **individually** to the page for the `development` environment. It automatically bundles & minifies for other environments (`staging`, `production`...).
|
|
* The bundle files may be **physical** files or [**virtual/embedded** files](../Virtual-File-System.md).
|
|
* ABP automatically adds **version query string** to the bundle file URL to prevent browsers from caching when the bundle is being updated. (like ?_v=67872834243042 - generated from last change date of the related files). The versioning works even if the bundle files are individually added to the page (on the development environment).
|
|
|
|
### Importing The Bundling Tag Helpers
|
|
|
|
> This is already imported by default with the startup templates. So, most of the time, you don't need to add it manually.
|
|
|
|
In order to use bundle tag helpers, you need to add it into your `_ViewImports.cshtml` file or into your page:
|
|
|
|
````
|
|
@addTagHelper *, Volo.Abp.AspNetCore.Mvc.UI.Bundling
|
|
````
|
|
|
|
### Unnamed Bundles
|
|
|
|
The `name` is **optional** for the razor bundle tag helpers. If you don't define a name, it's automatically **calculated** based on the used bundle file names (they are **concatenated** and **hashed**). Example:
|
|
|
|
````html
|
|
<abp-style-bundle>
|
|
<abp-style src="/libs/bootstrap/css/bootstrap.css" />
|
|
<abp-style src="/libs/font-awesome/css/font-awesome.css" />
|
|
<abp-style src="/libs/toastr/toastr.css" />
|
|
@if (ViewBag.IncludeCustomStyles != false)
|
|
{
|
|
<abp-style src="/styles/my-global-style.css" />
|
|
}
|
|
</abp-style-bundle>
|
|
````
|
|
|
|
This will potentially create **two different bundles** (one incudes the `my-global-style.css` and other does not).
|
|
|
|
Advantages of **unnamed** bundles:
|
|
|
|
* Can **conditionally add items** to the bundle. But this may lead to multiple variations of the bundle based on the conditions.
|
|
|
|
Advantages of **named** bundles:
|
|
|
|
* Other **modules can contribute** to the bundle by its name (see the sections below).
|
|
|
|
### Single File
|
|
|
|
If you need to just add a single file to the page, you can use the `abp-script` or `abp-style` tag without a wrapping in the `abp-script-bundle` or `abp-style-bundle` tag. Example:
|
|
|
|
````xml
|
|
<abp-script src="/scripts/my-script.js" />
|
|
````
|
|
|
|
The bundle name will be *scripts.my-scripts* for the example above ("/" is replaced by "."). All bundling features are work as expected for single file bundles too.
|
|
|
|
## Bundling Options
|
|
|
|
If you need to use same bundle in **multiple pages** or want to use some more **powerful features**, you can configure bundles **by code** in your [module](../Module-Development-Basics.md) class.
|
|
|
|
### Creating A New Bundle
|
|
|
|
Example usage:
|
|
|
|
````C#
|
|
[DependsOn(typeof(AbpAspNetCoreMvcUiBundlingModule))]
|
|
public class MyWebModule : AbpModule
|
|
{
|
|
public override void ConfigureServices(ServiceConfigurationContext context)
|
|
{
|
|
Configure<AbpBundlingOptions>(options =>
|
|
{
|
|
options
|
|
.ScriptBundles
|
|
.Add("MyGlobalBundle", bundle => {
|
|
bundle.AddFiles(
|
|
"/libs/jquery/jquery.js",
|
|
"/libs/bootstrap/js/bootstrap.js",
|
|
"/libs/toastr/toastr.min.js",
|
|
"/scripts/my-global-scripts.js"
|
|
);
|
|
});
|
|
});
|
|
}
|
|
}
|
|
````
|
|
|
|
> You can use the same name (*MyGlobalBundle* here) for a script & style bundle since they are added to different collections (`ScriptBundles` and `StyleBundles`).
|
|
|
|
After defining such a bundle, it can be included into a page using the same tag helpers defined above. Example:
|
|
|
|
````html
|
|
<abp-script-bundle name="MyGlobalBundle" />
|
|
````
|
|
|
|
This time, no file defined in the tag helper definition because the bundle files are defined by the code.
|
|
|
|
### Configuring An Existing Bundle
|
|
|
|
ABP supports [modularity](../Module-Development-Basics.md) for bundling as well. A module can modify an existing bundle that is created by a dependant module. Example:
|
|
|
|
````C#
|
|
[DependsOn(typeof(MyWebModule))]
|
|
public class MyWebExtensionModule : AbpModule
|
|
{
|
|
public override void ConfigureServices(ServiceConfigurationContext context)
|
|
{
|
|
Configure<AbpBundlingOptions>(options =>
|
|
{
|
|
options
|
|
.ScriptBundles
|
|
.Configure("MyGlobalBundle", bundle => {
|
|
bundle.AddFiles(
|
|
"/scripts/my-extension-script.js"
|
|
);
|
|
});
|
|
});
|
|
}
|
|
}
|
|
````
|
|
|
|
> It's not possible to configure unnamed bundle tag helpers by code, because their name are not known at the development time. It's suggested to always use a name for a bundle tag helper.
|
|
|
|
## Bundle Contributors
|
|
|
|
Adding files to an existing bundle seems useful. What if you need to **replace** a file in the bundle or you want to **conditionally** add files? Defining a bundle contributor provides extra power for such cases.
|
|
|
|
An example bundle contributor that replaces bootstrap.css with a customized version:
|
|
|
|
````C#
|
|
public class MyExtensionGlobalStyleContributor : BundleContributor
|
|
{
|
|
public override void ConfigureBundle(BundleConfigurationContext context)
|
|
{
|
|
context.Files.ReplaceOne(
|
|
"/libs/bootstrap/css/bootstrap.css",
|
|
"/styles/extensions/bootstrap-customized.css"
|
|
);
|
|
}
|
|
}
|
|
````
|
|
|
|
Then you can use this contributor as like below:
|
|
|
|
````C#
|
|
services.Configure<AbpBundlingOptions>(options =>
|
|
{
|
|
options
|
|
.ScriptBundles
|
|
.Configure("MyGlobalBundle", bundle => {
|
|
bundle.AddContributors(typeof(MyExtensionGlobalStyleContributor));
|
|
});
|
|
});
|
|
````
|
|
|
|
> You can also add contributors while creating a new bundle.
|
|
|
|
Contributors can also be used in the bundle tag helpers. Example:
|
|
|
|
````xml
|
|
<abp-style-bundle>
|
|
<abp-style type="@typeof(BootstrapStyleContributor)" />
|
|
<abp-style src="/libs/font-awesome/css/font-awesome.css" />
|
|
<abp-style src="/libs/toastr/toastr.css" />
|
|
</abp-style-bundle>
|
|
````
|
|
|
|
`abp-style` and `abp-script` tags can get `type` attributes (instead of `src` attributes) as shown in this sample. When you add a bundle contributor, its dependencies are also automatically added to the bundle.
|
|
|
|
### Contributor Dependencies
|
|
|
|
A bundle contributor can have one or more dependencies to other contributors.
|
|
Example:
|
|
|
|
````C#
|
|
[DependsOn(typeof(MyDependedBundleContributor))] //Define the dependency
|
|
public class MyExtensionStyleBundleContributor : BundleContributor
|
|
{
|
|
//...
|
|
}
|
|
````
|
|
|
|
When a bundle contributor is added, its dependencies are **automatically and recursively** added. Dependencies added by the **dependency order** by preventing **duplicates**. Duplicates are prevented even if they are in separated bundles. ABP organizes all bundles in a page and eliminates duplications.
|
|
|
|
Creating contributors and defining dependencies is a way of organizing bundle creation across different modules.
|
|
|
|
### Contributor Extensions
|
|
|
|
In some advanced scenarios, you may want to do some additional configuration whenever a bundle contributor is used. Contributor extensions works seamlessly when the extended contributor is used.
|
|
|
|
The example below adds some styles for prism.js library:
|
|
|
|
````csharp
|
|
public class MyPrismjsStyleExtension : BundleContributor
|
|
{
|
|
public override void ConfigureBundle(BundleConfigurationContext context)
|
|
{
|
|
context.Files.AddIfNotContains("/libs/prismjs/plugins/toolbar/prism-toolbar.css");
|
|
}
|
|
}
|
|
````
|
|
|
|
Then you can configure `BundleContributorOptions` to extend existing `PrismjsStyleBundleContributor`.
|
|
|
|
````csharp
|
|
Configure<BundleContributorOptions>(options =>
|
|
{
|
|
options
|
|
.Extensions<PrismjsStyleBundleContributor>()
|
|
.Add<MyPrismjsStyleExtension>();
|
|
});
|
|
````
|
|
|
|
Whenever `PrismjsStyleBundleContributor` is added into a bundle, `MyPrismjsStyleExtension` will also be automatically added.
|
|
|
|
### Accessing to the IServiceProvider
|
|
|
|
While it is rarely needed, `BundleConfigurationContext` has a `ServiceProvider` property that you can resolve service dependencies inside the `ConfigureBundle` method.
|
|
|
|
### Standard Package Contributors
|
|
|
|
Adding a specific NPM package resource (js, css files) into a bundle is pretty straight forward for that package. For example you always add the `bootstrap.css` file for the bootstrap NPM package.
|
|
|
|
There are built-in contributors for all [standard NPM packages](Client-Side-Package-Management.md). For example, if your contributor depends on the bootstrap, you can just declare it, instead of adding the bootstrap.css yourself.
|
|
|
|
````C#
|
|
[DependsOn(typeof(BootstrapStyleContributor))] //Define the bootstrap style dependency
|
|
public class MyExtensionStyleBundleContributor : BundleContributor
|
|
{
|
|
//...
|
|
}
|
|
````
|
|
|
|
Using the built-in contributors for standard packages;
|
|
|
|
* Prevents you typing **the invalid resource paths**.
|
|
* Prevents changing your contributor if the resource **path changes** (the dependant contributor will handle it).
|
|
* Prevents multiple modules adding the **duplicate files**.
|
|
* Manages **dependencies recursively** (adds dependencies of dependencies, if necessary).
|
|
|
|
#### Volo.Abp.AspNetCore.Mvc.UI.Packages Package
|
|
|
|
> This package is already installed by default in the startup templates. So, most of the time, you don't need to install it manually.
|
|
|
|
Standard package contributors are defined in the `Volo.Abp.AspNetCore.Mvc.UI.Packages` NuGet package.
|
|
To install it to your project:
|
|
|
|
````
|
|
install-package Volo.Abp.AspNetCore.Mvc.UI.Packages
|
|
````
|
|
|
|
Then add the `AbpAspNetCoreMvcUiPackagesModule` module dependency to your own module;
|
|
|
|
````C#
|
|
using Volo.Abp.Modularity;
|
|
using Volo.Abp.AspNetCore.Mvc.UI.Bundling;
|
|
|
|
namespace MyCompany.MyProject
|
|
{
|
|
[DependsOn(typeof(AbpAspNetCoreMvcUiPackagesModule))]
|
|
public class MyWebModule : AbpModule
|
|
{
|
|
//...
|
|
}
|
|
}
|
|
````
|
|
|
|
### Bundle Inheritance
|
|
|
|
In some specific cases, it may be needed to create a **new** bundle **inherited** from other bundle(s). Inheriting from a bundle (recursively) inherits all files/contributors of that bundle. Then the derived bundle can add or modify files/contributors **without modifying** the original bundle.
|
|
Example:
|
|
|
|
````c#
|
|
services.Configure<AbpBundlingOptions>(options =>
|
|
{
|
|
options
|
|
.StyleBundles
|
|
.Add("MyTheme.MyGlobalBundle", bundle => {
|
|
bundle
|
|
.AddBaseBundles("MyGlobalBundle") //Can add multiple
|
|
.AddFiles(
|
|
"/styles/mytheme-global-styles.css"
|
|
);
|
|
});
|
|
});
|
|
````
|
|
|
|
## Themes
|
|
|
|
Themes uses the standard package contributors to add library resources to page layouts. Themes may also define some standard/global bundles, so any module can contribute to these standard/global bundles. See the [theming documentation](Theming.md) for more.
|
|
|
|
## Best Practices & Suggestions
|
|
|
|
It's suggested to define multiple bundles for an application, each one is used for different purposes.
|
|
|
|
* **Global bundle**: Global style/script bundles are included to every page in the application. Themes already defines global style & script bundles. Your module can contribute to them.
|
|
* **Layout bundles**: This is a specific bundle to an individual layout. Only contains resources shared among all the pages use the layout. Use the bundling tag helpers to create the bundle as a good practice.
|
|
* **Module bundles**: For shared resources among the pages of an individual module.
|
|
* **Page bundles**: Specific bundles created for each page. Use the bundling tag helpers to create the bundle as a best practice.
|
|
|
|
Establish a balance between performance, network bandwidth usage and count of many bundles.
|
|
|
|
## See Also
|
|
|
|
* [Client Side Package Management](Client-Side-Package-Management.md)
|
|
* [Theming](Theming.md)
|