diff --git a/.editorconfig b/.editorconfig
index 6e48422..93b895d 100644
--- a/.editorconfig
+++ b/.editorconfig
@@ -9,7 +9,7 @@ insert_final_newline = true
 indent_style = tab
 trim_trailing_whitespace = true
 
-# YAML files require spaces
-[*.{yaml,yml}]
+# YAML files require spaces. JSON doesn't, but our file uses them.
+[*.{yaml,yml,json}]
 indent_style = space
 indent_size = 2
diff --git a/.eslintignore b/.eslintignore
index 537506f..5c7cc88 100644
--- a/.eslintignore
+++ b/.eslintignore
@@ -10,5 +10,4 @@ docs/
 src/vendor
 
 # The test folder uses ES modules
-test/src
 test/build
diff --git a/.eslintrc.js b/.eslintrc.js
index 5837717..560d2f6 100644
--- a/.eslintrc.js
+++ b/.eslintrc.js
@@ -105,7 +105,7 @@ module.exports = {
 		'new-parens': 'error',
 		'newline-after-var': 'off',
 		'newline-before-return': 'off',
-		'newline-per-chained-call': 'error',
+		'newline-per-chained-call': 'off',
 		'no-alert': 'error',
 		'no-array-constructor': 'error',
 		'no-async-promise-executor': 'error',
@@ -127,7 +127,7 @@ module.exports = {
 		'no-extra-label': 'error',
 		'no-extra-parens': 'off',
 		'no-floating-decimal': 'error',
-		'no-implicit-coercion': 'error',
+		'no-implicit-coercion': 'off',
 		'no-implicit-globals': 'error',
 		'no-implied-eval': 'error',
 		'no-inline-comments': 'error',
@@ -251,7 +251,10 @@ module.exports = {
 			'error',
 			'last'
 		],
-		'sort-imports': 'error',
+		'sort-imports': [
+			'error',
+			{ memberSyntaxSortOrder: [ 'all', 'multiple', 'single', 'none' ] },
+		],
 		'sort-keys': 'off',
 		'sort-vars': 'error',
 		'space-before-blocks': 'error',
diff --git a/.gitignore b/.gitignore
index 6a28462..77e8e70 100644
--- a/.gitignore
+++ b/.gitignore
@@ -2,3 +2,5 @@ node_modules
 *.log
 package-lock.json
 test/build
+
+.DS_Store
diff --git a/.travis.yml b/.travis.yml
index 88bcb31..175810c 100644
--- a/.travis.yml
+++ b/.travis.yml
@@ -1,11 +1,12 @@
 language: node_js
 script: "npm run lint && npm run test && npm run test-build"
 node_js:
-  - node
   - lts/*
+  - 16
+  - 14
   - 12
-  - 10
 branches:
   only:
     - main
     - beta
+    - v1
diff --git a/babel-preset.js b/babel-preset.js
index ae44199..6028904 100644
--- a/babel-preset.js
+++ b/babel-preset.js
@@ -2,12 +2,7 @@ module.exports = ( api ) => {
 	api.cache.forever();
 
 	return {
-		presets: [ '@wordpress/default' ],
-		plugins: [
-			'@babel/plugin-proposal-class-properties',
-			[ 'transform-react-jsx', {
-				pragma: 'wp.element.createElement',
-			} ],
-		],
+		presets: [ '@wordpress/babel-preset-default' ],
+		plugins: [ '@babel/plugin-proposal-class-properties' ],
 	};
 };
diff --git a/docs/changelog.md b/docs/changelog.md
index 5b4df9b..c1687ab 100644
--- a/docs/changelog.md
+++ b/docs/changelog.md
@@ -6,8 +6,25 @@ nav_order: 10
 
 # Changelog
 
-## Next
-
+## v1.0
+
+- **Breaking**: End support for Node v10. Node v12.13 or later is now required.
+- **Breaking**: Switch to Webpack 5 and Webpack DevServer 4
+- **Breaking**: Include a contenthash string in default bundle file names. Set `filename: '[name].js'` in your output configuration to restore the old behavior.
+- **Breaking**: Default DevServer manifest name is now `development-asset-manifest.json`, not `asset-manifest.json`.
+- **Breaking**: Replace `filterLoaders` system with [individual hooks accesible via the new `addFilter` and `removeFilter` helpers](https://humanmade.github.io/webpack-helpers/modules/presets.html#customizing-presets).
+- **Breaking**: Remove deprecated `eslint-loader` and add `eslint-webpack-plugin` to presets as `plugins.eslint()`.
+- **Potentially Breaking**: Remove `loaders.url()` and `loaders.file()` in favor of Webpack 5 [`asset` modules](https://webpack.js.org/guides/asset-modules/), now usable by including `loaders.asset()` (for assets which can be inlined) and `loaders.resource()` (as a catch-all for other types) in your module rules list. Asset modules are handled automatically in both presets, so this is only breaking if `loaders.url()` or `loaders.file()` was used directly.
+- A `webpack-manifest-plugin` instance is now automatically injected in development mode even if `publicPath` is not specified.
+- A `webpack-bundle-analyzer` plugin is now automatically added to production builds when Webpack is invoked with the `--analyze` flag.
+- Add the [`simple-build-report-webpack-plugin`](https://github.com/kadamwhite/simple-build-report-webpack-plugin) as `plugins.simpleBuildReport()`
+- Include a `plugins.simpleBuildReport()` instance in production preset builds to improve legibility of Webpack console output.
+- `plugins.fixStyleOnlyEntries()` now uses [`webpack-remove-empty-scripts`](https://github.com/webdiscus/webpack-remove-empty-scripts#webpack-remove-empty-scripts) instead of `webpack-fix-style-only-entries` due to Webpack 5 compatiblity issues with the original plugin.
+- Allow `null` to be returned from an `addFilter` callback to skip a loader when using a configuration preset.
+- Do not deep merge options passed to `plugins.terserPlugin()`: options object can now fully overwrite `terserOptions` property if needed.
+- Permit filtering the Terser default configuration using `addFilter( 'plugins/terser/defaults', cb )`.
+- Remove `OptimizeCssAssetsPlugin` (`plugins.optimizeCssAssets()`) in favor of Webpack 5-compatible [CssMinimizerPlugin](https://github.com/webpack-contrib/css-minimizer-webpack-plugin) (`plugins.cssMinimizer()`)
+- Remove `plugins.hotModuleReplacement()`, which is now handled automatically by the DevServer in `hot` mode.
 - Include the `contenthash` in generated CSS filenames. [#204](https://github.com/humanmade/webpack-helpers/pull/204)
 
 ## v0.11.1
diff --git a/docs/guides/getting-started.md b/docs/guides/getting-started.md
index 506d6c5..3aca816 100644
--- a/docs/guides/getting-started.md
+++ b/docs/guides/getting-started.md
@@ -264,7 +264,7 @@ To make use of this utility, we wrap our development config in a `choosePort` pr
 +] );
 ```
 
-While these bundles are served from memory, now that we've specified a `publicPath`, a new file `asset-manifest.json` will be output into each project's `build/` folder. `themes/myproject/build/asset-manifest.json` will look like this:
+While these bundles are served from memory, now that we've specified a `publicPath`, a new file `development-asset-manifest.json` will be output into each project's `build/` folder. `themes/myproject/build/development-asset-manifest.json` will look like this:
 ```json
 {
   "frontend.js": "http://localhost:8080/myproject-theme/frontend.js",
@@ -286,8 +286,8 @@ We use the `cleanOnExit` helper to delete these files when the server shuts down
 +
 +// Clean up manifests on exit.
 +cleanOnExit( [
-+	filePath( 'mu-plugins/myproject-blocks/build/asset-manifest.json' ),
-+	filePath( 'themes/myproject/build/asset-manifest.json' ),
++	filePath( 'mu-plugins/myproject-blocks/build/development-asset-manifest.json' ),
++	filePath( 'themes/myproject/build/development-asset-manifest.json' ),
 +] );
 ```
 
diff --git a/docs/guides/upgrading-to-v1.md b/docs/guides/upgrading-to-v1.md
new file mode 100644
index 0000000..60bdc3f
--- /dev/null
+++ b/docs/guides/upgrading-to-v1.md
@@ -0,0 +1,70 @@
+---
+title: Upgrading to v1
+parent: Guides
+nav_order: 1
+---
+
+# Upgrading to v1
+
+## Installation
+
+Install `@humanmade/webpack-helpers@latest` with [npm](http://npmjs.com), along with more updated versions of `webpack` and its attendant dependencies:
+
+```bash
+npm install --save-dev @humanmade/webpack-helpers@latest webpack@5 webpack-cli@4 webpack-dev-server@4
+```
+
+## Breaking Changes
+
+### Handling multi-config arrays in DevServer
+
+In a large project you may have a Webpack configuration that [exports multiple separate Webpack configuration objects](https://webpack.js.org/configuration/configuration-types/#exporting-multiple-configurations) in an array. The `presets.development()` helper adds a `devServer` property to each generated configuration, and with Webpack 4 and the pre-1.0 version of this helper library this was not a problem: if each entry in an exported array of configurations had its own `devServer` property, DevServer used the configuration in the first item in the exported array and ignored the rest.
+
+In Webpack 5 and the latest DevServer, however, it is an error to define `devServer` properties on any configuration but the first item in the exported array. This means that you may need to map over the exported configurations and unset the `devServer` property on subsequent configurations.
+
+```js
+const configs = [
+	presets.development( { /* ... */ } ),
+	presets.development( { /* ... */ } ),
+	presets.development( { /* ... */ } ),
+];
+module.exports = configs.map( ( config, index ) => {
+	if ( index > 0 ) {
+		Reflect.deleteProperty( config, 'devServer' );
+	}
+	return config;
+} );
+```
+
+### Goodbye `filterLoaders`, welcome `addFilter`
+
+Previously, a second argument could be passed to a preset to define a function used to filter loaders used while generating that preset configuration. In v1.0, a [hooks system](https://humanmade.github.io/webpack-helpers/reference/hooks.html) has been introduced which provides an `addFilter` method. `addFilter` can be used to register a callback function that may adjust the value of a loader globally, or make changes specific to a given preset call, in a manner similar to the WordPress PHP hooks system. Read the [hooks reference](https://humanmade.github.io/webpack-helpers/reference/hooks.html), [loaders](https://humanmade.github.io/webpack-helpers/modules/loaders.html), and [preset](https://humanmade.github.io/webpack-helpers/modules/presets.html) documentation pages for more information.
+
+### ESLint v6 and earlier not supported
+
+The ESLint integration for Webpack 5 requires ESLint v7 or later. If your project uses ESLint 6 or before, you will need to upgrade ESLint before using Webpack Helpers 1.0.
+
+```bash
+npm install --save-dev eslint@7
+```
+
+On most projects ESLint should continue to work after upgrade with no further adjustments.
+
+### Importing named properties from JSON
+
+Due to ongoing changes to how JSON files are handled by Webpack, this code
+
+```js
+import { name } from './block.json';
+```
+
+may now trigger the error
+
+```
+Should not import the named export 'name' (imported as 'name') from default-exporting module (only default export is available soon)
+```
+This error means that your code above needs to be changed to this:
+```js
+import blockData from './block.json';
+const { name } = blockData;
+```
diff --git a/docs/modules/helpers.md b/docs/modules/helpers.md
index b0b0afc..f52aea3 100644
--- a/docs/modules/helpers.md
+++ b/docs/modules/helpers.md
@@ -33,15 +33,15 @@ module.exports = presets.production( {
 ## `cleanOnExit`
 
 
-When using the `presets.development()` generator, an `asset-manifest.json` will automatically be generated so long as a `publicPath` URI can be determined. When working with an `asset-manifest.json` file, the `manifest` module provides a `cleanOnExit` method to easily remove manifests once the `webpack-dev-server` shuts down.
+When using the `presets.development()` generator, a `development-asset-manifest.json` will automatically be generated so long as a `publicPath` URI can be determined. When working with a `development-asset-manifest.json` file, the `manifest` module provides a `cleanOnExit` method to easily remove manifests once the `webpack-dev-server` shuts down.
 
 ```js
 const { join } = require( 'path' );
 const { helpers } = require( '@humanmade/webpack-helpers' );
 
 helpers.cleanOnExit( [
-	join( process.cwd(), 'content/mu-plugins/custom-blocks/build/asset-manifest.json' ),
-	join( process.cwd(), 'content/themes/my-theme/build/asset-manifest.json' ),
+	join( process.cwd(), 'content/mu-plugins/custom-blocks/build/development-asset-manifest.json' ),
+	join( process.cwd(), 'content/themes/my-theme/build/development-asset-manifest.json' ),
 ] );
 ```
 
@@ -98,3 +98,69 @@ module.exports = helpers.choosePort( 9090 ).then( port => [
 	} ),
 ] );
 ```
+
+## Filters
+
+Webpack Helpers v1.0 introduced a new system for altering the behavior of bundled loaders, plugins, and presets, also covered under [Customizing Presets](./presets.html#customizing-presets). This system uses a filtering approach similar to [the way PHP filters work within WordPress](https://developer.wordpress.org/reference/functions/add_filter/), with `addFilter` and `removeFilter` helpers you can use to modify various types of internal data at runtime.
+
+```js
+// webpack.config.js
+const { helpers } = require( '@humanmade/webpack-helpers' );
+
+const { addFilter, removeFilter } = helpers;
+
+// Hook onto a specific filter by name, and provide a callback function to
+// alter the value returned from that filter.
+addFilter( 'filter/name', function( value ) {
+	return 'modified value';
+} );
+
+// To filter only some values, you can remove a filter by passing the same
+// function reference to removeFilter.
+const filterFunction = ( value ) => 'modified value';
+
+addFilter( 'filter/name', filterFunction );
+
+// Do something where the filter gets applied.
+
+removeFilter( 'filter/name', filterFunction );
+
+// Now if you do that same thing, the filter will no longer apply.
+```
+
+### Filter List
+
+Each [loader](./loaders.html) exposes at minimum two filters, `loaders/{name}` and `loaders/{name}/defaults`. For example, the defaults for `loaders.ts()` can be filtered using
+
+```js
+addFilter( 'loaders/ts/defaults', ( loaderDefaultsObject ) => {
+	// return a filtered value
+} );
+```
+or the computed final loader object can be modified after the fact with
+
+```js
+addFilter( 'loaders/ts', ( loaderObject ) => {
+	// return a filtered value
+} );
+```
+
+If you return `null` from the `loaders/{name}` filter, it will remove that loader from the preset entirely.
+
+Additional filters provided by Webpack Helpers:
+
+**`presets/stylesheet-loaders`**
+
+Filter the [stylesheet loader chain](https://webpack.js.org/concepts/loaders/#configuration) used in the `presets.production()` and `presets.development()` helpers. Functions added to this filter receive the stylesheet chain as their first argument, and the name of the compilation mode (`production` or `development`) as the second argument.
+
+**`plugins/terser/defaults`**
+
+Filter the [TerserPlugin default options object](https://webpack.js.org/plugins/terser-webpack-plugin/#options) if you wish to alter the minification settings used by the production preset.
+
+**`loaders/postcss/plugins`**
+
+Filter the list of [PostCSS plugins](https://github.com/postcss/postcss/blob/main/docs/plugins.md) used by the `loaders.postcss()` loader.
+
+**`loaders/postcss/preset-env`**
+
+Filter the [`postcss-preset-env` options](https://github.com/csstools/postcss-preset-env#options) used by the `loaders.postcss()` loader.
diff --git a/docs/modules/loaders.md b/docs/modules/loaders.md
index f036aa2..3dd58de 100644
--- a/docs/modules/loaders.md
+++ b/docs/modules/loaders.md
@@ -6,19 +6,23 @@ nav_order: 3
 
 # Loaders Module
 
-`const { loaders } = require( '@humanmade/webpack-helpers' );`
+```js
+const { loaders } = require( '@humanmade/webpack-helpers' );
+```
 
 This module provides functions that generate configurations for commonly-needed Webpack loaders. Use them within the `.module.rules` array, or use `presets.development()`/`presets.production()` to opt-in to some opinionated defaults.
 
-- `loaders.eslint()`: Return a configured Webpack module loader rule for `eslint-loader`.
-- `loaders.js()`: Return a configured Webpack module loader rule for `js-loader`.
-- `loaders.ts()`: Return a configured Webpack module loader rule for `ts-loader`.
-- `loaders.url()`: Return a configured Webpack module loader rule for `url-loader`.
-- `loaders.style()`: Return a configured Webpack module loader rule for `style-loader`.
+- `loaders.asset()`: Return a configured Webpack module loader rule for [`asset` modules](https://webpack.js.org/guides/asset-modules/#inlining-assets) which will be inlined when small enough.
 - `loaders.css()`: Return a configured Webpack module loader rule for `css-loader`.
+- `loaders.js()`: Return a configured Webpack module loader rule for `js-loader`.
 - `loaders.postcss()`: Return a configured Webpack module loader rule for `postcss-loader`.
+- `loaders.resource()`: Return a configured Webpack module loader rule for [`asset/resource` modules](https://webpack.js.org/guides/asset-modules/#resource-assets).
 - `loaders.sass()`: Return a configured Webpack module loader rule for `sass-loader`.
-- `loaders.file()`: Return a configured Webpack module loader rule for `file-loader`.
+- `loaders.sourcemap()`: Return a configured Webpack module loader rule for `source-map-loader`.
+- `loaders.style()`: Return a configured Webpack module loader rule for `style-loader`.
+- `loaders.ts()`: Return a configured Webpack module loader rule for `ts-loader`.
+
+The output from these loaders can optionally be [filtered](https://humanmade.github.io/webpack-helpers/reference/hooks.html).
 
 ## Customizing Loaders
 
@@ -41,16 +45,25 @@ module.exports = {
 };
 ```
 
-To alter the configuration for a loader prior to use within a preset, you may mutate the `.defaults` property on the loader method.
+To alter the configuration for a loader prior to use within a preset, you may either [filter](https://humanmade.github.io/webpack-helpers/reference/hooks.html) the default loader options, or the final merged loader configuration.
 
 ```js
 const { helpers, loaders, presets } = require( '@humanmade/webpack-helpers' );
+const { addFilter } = helpers;
+
+// Adjust the loader defaults.
+addFilter( 'loaders/js/defaults', ( defaults ) => ( {
+	...defaults,
+	include: helpers.filePath( 'themes/my-theme/src' ),
+} ) );
 
-// Mutate the loader defaults.
-loaders.js.defaults.include = helpers.filePath( 'themes/my-theme/src' );
-loaders.css.defaults.options.url = false;
+addFilter( 'loaders/css/defaults', ( defaults ) => {
+	defaults.options.url = false;
+	return defaults;
+} );
 
+// The above customizations will now apply to all calls to loader or preset factories.
 module.exports = presets.development( { /* ... */ } );
 ```
 
-These loaders are also used by the [presets](https://humanmade.github.io/webpack-helpers/modules/presets.html) methods described above. To adjust the behavior of a loader for a specific configuration generated using a preset, you may pass a second argument to the preset defining a filter function which can modify loader options as they are computed. See ["Customizing Presets"](https://humanmade.github.io/webpack-helpers/modules/presets.html#customizing-presets) for more information.
+These loaders are also used by the [presets](https://humanmade.github.io/webpack-helpers/modules/presets.html) methods described above. To adjust the behavior of a loader for a specific configuration generated using a preset, see ["Customizing Presets"](https://humanmade.github.io/webpack-helpers/modules/presets.html#customizing-presets).
diff --git a/docs/modules/plugins.md b/docs/modules/plugins.md
index ce31a83..7ef80f0 100644
--- a/docs/modules/plugins.md
+++ b/docs/modules/plugins.md
@@ -12,16 +12,15 @@ This module provides methods which create new instances of commonly-needed Webpa
 
   | Plugin | Description
 ------ | ------ | ------------
-  | `plugins.bundleAnalyzer()` | Create and return a new [`webpack-bundle-analyzer`](https://github.com/webpack-contrib/webpack-bundle-analyzer) instance. When included this plugin is disabled by default unless the `--analyze` flag is passed on the command line.
+  | `plugins.bundleAnalyzer()` | Create and return a new [`webpack-bundle-analyzer`](https://github.com/webpack-contrib/webpack-bundle-analyzer) instance. This plugin is included in production preset builds automatically when the `--analyze` flag is passed on the command line.
   | `plugins.clean()` | Create and return a new [`clean-webpack-plugin`](https://github.com/johnagan/clean-webpack-plugin) instance.
   | `plugins.copy()` | Create and return a new [`copy-webpack-plugin`](https://github.com/webpack-contrib/copy-webpack-plugin) instance.
   | `plugins.errorBell()` | Create and return a new [`bell-on-bundle-error-plugin`](https://www.npmjs.com/package/bell-on-bundler-error-plugin) instance.
-  | `plugins.fixStyleOnlyEntries()` | Create and return a [`webpack-fix-style-only-entries`](https://github.com/fqborges/webpack-fix-style-only-entries) instance to remove empty JS bundles for style-only entrypoints.
-**D**  | `plugins.hotModuleReplacement()` | Create and return a new [`webpack.HotModuleReplacementPlugin`](https://webpack.js.org/plugins/hot-module-replacement-plugin/) instance.
+  | `plugins.eslint()` | Create and return a [`eslint-webpack-plugin`](https://webpack.js.org/plugins/eslint-webpack-plugin/) instance.
+  | `plugins.fixStyleOnlyEntries()` | Create and return a [`webpack-remove-empty-scripts`](https://github.com/webdiscus/webpack-remove-empty-scripts) instance (forked from the non-Webpack 5-compatible `webpack-fix-style-only-entries`) to remove empty JS bundles for style-only entrypoints.
   | `plugins.manifest()` | : Create and return a new [`webpack-manifest-plugin`](https://github.com/danethurber/webpack-manifest-plugin) instance, preconfigured to write the manifest file while running from a dev server.
 **P**  | `plugins.miniCssExtract()` | Create and return a new [`mini-css-extract-plugin`](https://github.com/webpack-contrib/mini-css-extract-plugin) instance.
 **P**  | `plugins.terser()` | Create and return a new [`terser-webpack-plugin`](https://github.com/webpack-contrib/terser-webpack-plugin) instance, preconfigured with defaults based on `create-react-app`.
+**P**  | `plugins.cssMinimizer()` | Create and return a new [`css-minimizer-webpack-plugin`](https://webpack.js.org/plugins/css-minimizer-webpack-plugin/) instance.
 
 **P**: Included in `presets.production()`
-
-**D**: Included in `presets.development()`
diff --git a/docs/modules/presets.md b/docs/modules/presets.md
index 205d671..c7fb68f 100644
--- a/docs/modules/presets.md
+++ b/docs/modules/presets.md
@@ -91,7 +91,6 @@ module.exports = {
 
 [Read up on the `...` "spread operator"](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Spread_syntax#Spread_in_object_literals) if this syntax is unfamiliar to you; it's quite useful!
 
-
 Customize a generated production configuration by object mutation:
 
 ```js
@@ -109,27 +108,57 @@ module.exports.optimization.minimizer = [
 ];
 ```
 
-Note that array values are _merged_, not overwritten. This allows you to easily add plugins, but it can make it hard to _remove_ values from array properties like the module loader rules. To change loader configuration options without completely removing a loader, we recommend the approach described below in "Customizing Loaders". However, if you do need to completely change or remove the loaders from a default, you may overwrite the `.module.rules` array using one of the above methods.
-
-### Modify loaders within a generated configuration
+### Filtering values
 
-Adjusting loaders within a generated configuration tree is difficult because loader arrays are not keyed and module rules may be nested. Instead, each `preset` generator accepts a second argument in which you can pass a callback function that will be run on the output of each computed loader definition.
+Array values are _merged_ when processing a preset, not overwritten. This allows you to easily add plugins, but it can make it hard to _remove_ values from array properties like the module loader rules or plugins list. To change loader configuration options without completely removing a loader, or adjust loaders deep within a generated configuration tree, this library provides a `addFilter` helper function. `addFilter()` lets you register a callback which can transform the value of each computed loader, as well as some other useful configuration values.
 
 ```js
-// Alter the publicPath value of the files-loader and url-loader.
-const config = production.preset(
-	{ /* ...configuration options described above ... */ },
-	{
-		filterLoaders: ( loader, loaderType ) => {
-			if ( loaderType === 'file' || loaderType === 'url' ) {
-				loader.options.publicPath = '../../';
-			}
-			return loader;
-		}
-	}
+const { helpers, presets } = require( '@humanmade/webpack-helpers' );
+const { addFilter } = helpers;
+
+// Intercept the "sass" loader and replace it with a Stylus loader.
+addFilter( 'loaders/sass', () => {
+	return {
+		loader: require.resolve( 'stylus-loader' ),
+	};
+} );
+// Alter presets to use a different targeting regular expression.
+addFilter( 'presets/stylesheet-loaders', ( loader ) => {
+	loader.test = /.\styl$/;
+	return loader;
+} );
+
+// Now, use the preset as normal and it will pick up Stylus files instead!
+const config = presets.production(
+	{ /* ...normal configuration options as described above ... */ }
 );
 ```
 
-The `loaderType` option will be [one of the keys in the `loaders` object](./loaders.html), or else the special key `stylesheet` which will be passed alongside the entire computed stylesheet loader chain. Filter on `stylesheet` to, for example, completely replace the Sass stylesheet support with a different preprocessor like Stylus.
+If you are using a multi-configuration Webpack setup where the config file exports an array of individual configurations, you may wish to filter a loader, preset chain, or other value for only one of those configuration objects. To permit this, each filter callback will be passed the preset's configuration object as a final argument when the filter or loader is called in the context of a preset. For example,
+
+```js
+const { helpers, externals, presets } = require( '@humanmade/webpack-helpers' );
+const { addFilter } = helpers;
+
+// Do not use the JS loader in the frontend build.
+addFilter( 'presets/js', ( loader, config ) => {
+	if ( config.name === 'frontend' ) {
+		return null;
+	}
+	return frontend;
+} );
+
+module.exports = [
+	presets.production( {
+		name: 'frontend',
+		// ...
+	} ),
+	presets.production( {
+		name: 'editor',
+		externals,
+		// ...
+	} ),
+];
+```
 
-The values of the `loaderType` argument in this callback correspond to the names of the available [loader factory functions](https://humanmade.github.io/webpack-helpers/modules/loaders.html).
+For more information, see [the full hooks reference.](https://humanmade.github.io/webpack-helpers/reference/hooks.html).
diff --git a/docs/reference.md b/docs/reference.md
new file mode 100644
index 0000000..deb8d0e
--- /dev/null
+++ b/docs/reference.md
@@ -0,0 +1,10 @@
+---
+title: Reference
+has_children: true
+permalink: /reference
+nav_order: 3
+---
+
+# Reference Documentation
+
+Code-level documentation of specific components of `@humanmade/webpack-helpers`.
diff --git a/docs/reference/hooks.md b/docs/reference/hooks.md
new file mode 100644
index 0000000..71cbf33
--- /dev/null
+++ b/docs/reference/hooks.md
@@ -0,0 +1,139 @@
+---
+title: Filter Hooks
+parent: Reference
+nav_order: 1
+---
+
+# Filter Hooks
+
+These hooks are provided by Webpack Helpers to customize the behavior of a specific loader or preset within your project.
+
+`{loader name}` in the below filters may be any slug for an available [loader](https://humanmade.github.io/webpack-helpers/modules/loaders.html): `asset`, `css`, `js`, `postcss`, `resource`, `sass`, `sourcemap`, `style`, and `ts`.
+
+Remember to always return a value from a filter callback.
+
+## `loaders/{loader name}`
+
+Filter the full loader configuration object after merging any user-provided options with the (filtered) defaults.
+
+When called in the context of a preset, the callback will receive the preset factory's configuration object as a second argument.
+
+**Arguments:**
+
+ name | type | description
+----- | ---- | ------
+`loader` | `Object` | Loader configuration object.
+`config` | `Object` or `null` | Preset configuration object if a preset is being rendered, else `null`.
+
+Return `null` from the callback to remove this loader from preset-generated configurations.
+
+```js
+addFilter( 'loaders/{loader name}', ( loader, config = null ) => {
+	// Filter the loader definition globally or based on the specific
+	// configuration options provided to a preset factory.
+	return loader;
+} );
+```
+
+## `loaders/{loader name}/defaults`
+
+Adjust the default values used to define a given loader configuration.
+
+When called in the context of a preset, the callback will receive the preset factory's configuration object as a second argument.
+
+**Arguments:**
+
+ name | type | description
+----- | ---- | ------
+`defaults` | `Object` | Defaults for this specific loader.
+`config` | `Object` or `null` | Preset configuration object if a preset is being rendered, else `null`.
+
+```js
+addFilter( 'loaders/{loader name}/defaults', ( defaults, config = null ) => {
+	// Filter the loader's defaults globally or based on the specific
+	// configuration options provided to a preset factory.
+	return defaults;
+} );
+```
+
+## `presets/stylesheet-loaders`
+
+Filter the loaders used to process stylesheet imports when building your project.
+
+The callback will receive the environment type of the preset being rendered as its second argument, and the preset's configuration object argument as a third argument.
+
+**Arguments:**
+
+ name | type | description
+----- | ---- | ------
+`loader` | `Object` | Combined stylesheet loader rule.
+`environment` | `string` | `'development'` or `'production'`.
+`config` | `Object` | Preset configuration object.
+
+```js
+addFilter( 'presets/stylesheet-loaders', ( loader, environment, config ) => {
+	// Filter the configured stylesheet loaders based on environment or the
+	// specific configuration options provided to the preset factory.
+	return loader;
+} );
+```
+
+## `presets/manifest-options`
+
+Filter the [`webpack-manifest-plugin` options](https://github.com/shellscape/webpack-manifest-plugin#options) used when auto-injecting asset manifest plugins.
+
+The callback will receive the environment type of the preset being rendered as its second argument, and the preset's configuration object argument as a third argument.
+
+**Arguments:**
+
+ name | type | description
+----- | ---- | ------
+`options` | `Object` | Manifest plugin options object.
+`environment` | `string` | `'development'` or `'production'`.
+`config` | `Object` | Preset configuration object.
+
+```js
+addFilter( 'presets/manifest-options', ( loader, environment, config ) => {
+	// Filter the configured stylesheet loaders based on environment or the
+	// specific configuration options provided to the preset factory.
+	return loader;
+} );
+```
+
+## `loaders/postcss/plugins`
+
+Filter the default list of PostCSS plugins used by the PostCSS loader.
+
+Unlike the hooks above, callbacks added to this filter do not receive the preset configuration object.
+
+**Arguments:**
+
+ name | type | description
+----- | ---- | ------
+`pluginsArray` | `Array` | Array of [PostCSS plugin](https://github.com/postcss/postcss#plugins) definitions.
+
+```js
+addFilter( 'loaders/postcss/plugins', ( pluginsArray ) => {
+	// Filter the plugins loaded by PostCSS.
+	return pluginsArray;
+} );
+```
+
+## `loaders/postcss/preset-env`
+
+Filter the [`postcss-preset-env` plugin's](https://github.com/csstools/postcss-plugins/tree/main/plugin-packs/postcss-preset-env#readme) configuration object.
+
+Unlike the hooks above, callbacks added to this filter do not receive the preset configuration object.
+
+**Arguments:**
+
+ name | type | description
+----- | ---- | ------
+`pluginOptions` | `Object` | `postcss-preset-env` plugin options object.
+
+```js
+addFilter( 'loaders/postcss/preset-env', ( pluginOptions ) => {
+	// Filter the PostCSS Preset Env plugin options.
+	return pluginOptions;
+} );
+```
diff --git a/index.js b/index.js
index f8bf15c..429f333 100644
--- a/index.js
+++ b/index.js
@@ -1,3 +1,5 @@
+const filters = require( './src/helpers/filters' );
+
 /**
  * Expose public package API.
  */
@@ -6,6 +8,8 @@ module.exports = {
 	config: require( './src/config' ),
 	externals: require( './src/externals' ),
 	helpers: {
+		addFilter: filters.addFilter,
+		removeFilter: filters.removeFilter,
 		choosePort: require( './src/helpers/choose-port' ),
 		cleanOnExit: require( './src/helpers/clean-on-exit' ),
 		filePath: require( './src/helpers/file-path' ),
diff --git a/package.json b/package.json
index 0ee394b..3ff1d3a 100644
--- a/package.json
+++ b/package.json
@@ -1,7 +1,7 @@
 {
   "name": "@humanmade/webpack-helpers",
   "public": true,
-  "version": "0.11.1",
+  "version": "1.0.0-beta.17",
   "description": "Reusable Webpack configuration components & related helper utilities.",
   "main": "index.js",
   "author": "Human Made",
@@ -18,7 +18,7 @@
     "webpack"
   ],
   "scripts": {
-    "lint": "eslint .",
+    "lint": "eslint . --ignore-pattern linting-test.js",
     "test": "jest",
     "test-build": "rm -rf test/build && webpack --config=test/test-config.js",
     "test-dev-server": "webpack-dev-server --config=test/test-config.js"
@@ -29,53 +29,53 @@
     ]
   },
   "devDependencies": {
-    "eslint": "^7.17.0",
+    "eslint": "^7.32.0",
     "jest": "^26.6.3",
-    "sass": "^1.32.4",
-    "typescript": "^4.0.2",
-    "webpack": "^4.46.0",
-    "webpack-cli": "^3.3.12",
-    "webpack-dev-server": "^3.11.2"
+    "sass": "^1.51.0",
+    "typescript": "^4.6.3",
+    "webpack": "^5.72.0",
+    "webpack-cli": "^4.9.2",
+    "webpack-dev-server": "^4.8.1"
   },
   "dependencies": {
-    "@babel/core": "^7.10.3",
-    "@babel/plugin-proposal-class-properties": "^7.10.1",
-    "@wordpress/babel-preset-default": "^4.16.0",
-    "babel-loader": "^8.0.6",
-    "babel-plugin-transform-react-jsx": "^6.24.1",
+    "@babel/core": "^7.17.9",
+    "@babel/plugin-proposal-class-properties": "^7.16.7",
+    "@wordpress/babel-preset-default": "^6.9.0",
+    "babel-loader": "^8.2.5",
     "bell-on-bundler-error-plugin": "^2.0.0",
-    "block-editor-hmr": "^0.6.1",
-    "chalk": "^4.1.0",
-    "clean-webpack-plugin": "^3.0.0",
-    "copy-webpack-plugin": "^6.0.2",
-    "css-loader": "^5.0.1",
+    "block-editor-hmr": "^0.7.0",
+    "chalk": "^4.1.2",
+    "clean-webpack-plugin": "^4.0.0",
+    "copy-webpack-plugin": "^10.2.4",
+    "css-loader": "^6.7.1",
+    "css-minimizer-webpack-plugin": "^3.4.1",
     "detect-port-alt": "^1.1.6",
-    "eslint-loader": "^4.0.2",
-    "file-loader": "^6.0.0",
-    "inquirer": "^7.2.0",
+    "eslint-webpack-plugin": "^3.1.1",
     "is-root": "^2.1.0",
-    "mini-css-extract-plugin": "^1.3.4",
-    "optimize-css-assets-webpack-plugin": "^5.0.1",
-    "postcss": "^8.2.4",
+    "lodash.clonedeep": "^4.5.0",
+    "mini-css-extract-plugin": "^2.6.0",
+    "postcss": "^8.4.12",
     "postcss-flexbugs-fixes": "^5.0.2",
-    "postcss-loader": "^4.1.0",
-    "postcss-preset-env": "^6.7.0",
-    "run-parallel": "1.1.9",
-    "sass-loader": "^10.0.2",
-    "signal-exit": "^3.0.2",
-    "style-loader": "^2.0.0",
-    "terser-webpack-plugin": "^4.2.0",
-    "ts-loader": "^8.0.1",
-    "url-loader": "^4.1.0",
-    "webpack-bundle-analyzer": "^4.3.0",
-    "webpack-fix-style-only-entries": "^0.6.0",
-    "webpack-manifest-plugin": "^3.0.0"
+    "postcss-loader": "^6.2.1",
+    "postcss-preset-env": "^7.4.4",
+    "prompts": "^2.4.2",
+    "run-parallel": "^1.2.0",
+    "sass-loader": "^12.6.0",
+    "signal-exit": "^3.0.7",
+    "simple-build-report-webpack-plugin": "^1.0.2",
+    "source-map-loader": "^3.0.1",
+    "style-loader": "^3.3.1",
+    "terser-webpack-plugin": "^5.3.1",
+    "ts-loader": "^9.2.9",
+    "webpack-bundle-analyzer": "^4.5.0",
+    "webpack-manifest-plugin": "^5.0.0",
+    "webpack-remove-empty-scripts": "^0.8.0"
   },
   "peerDependencies": {
     "sass": "*",
     "typescript": "*",
-    "webpack": "^4.0.0",
-    "webpack-cli": "^3.0.0",
-    "webpack-dev-server": "^3.0.0"
+    "webpack": "^5.0.0",
+    "webpack-cli": "^4.0.0",
+    "webpack-dev-server": "^4.0.0"
   }
 }
diff --git a/src/config.js b/src/config.js
index 144df73..f117d72 100644
--- a/src/config.js
+++ b/src/config.js
@@ -7,13 +7,21 @@ module.exports = {
 	 * @type {Object}
 	 */
 	devServer: {
-		disableHostCheck: true,
+		allowedHosts: 'all',
 		headers: {
 			'Access-Control-Allow-Origin': '*',
+			'Access-Control-Allow-Methods': '*',
+			'Access-Control-Allow-Headers': '*',
 		},
-		hotOnly: true,
-		watchOptions: {
-			aggregateTimeout: 300,
+		// Enable gzip compression of generated files.
+		compress: true,
+		hot: 'only',
+		client: {
+			// Do not show disruptive overlay for warnings.
+			overlay: {
+				errors: true,
+				warnings: false,
+			},
 		},
 	},
 
@@ -22,12 +30,10 @@ module.exports = {
 	 * @type {Object}
 	 */
 	stats: {
-		all: false,
+		preset: 'summary',
 		assets: true,
 		colors: true,
 		errors: true,
-		performance: true,
-		timings: true,
 		warnings: true,
 	},
 };
diff --git a/src/helpers/camel-case-dash.js b/src/helpers/camel-case-dash.js
index 11b3ffd..87a6b41 100644
--- a/src/helpers/camel-case-dash.js
+++ b/src/helpers/camel-case-dash.js
@@ -2,7 +2,7 @@
  * Given a kebab-case string, returns a new camelCase string.
  *
  * @param {string} string Input kebab-case string.
- * @return {string} Camel-cased string.
+ * @returns {string} Camel-cased string.
  */
 module.exports = string => string.replace(
 	/-([a-z])/g,
diff --git a/src/helpers/choose-port.js b/src/helpers/choose-port.js
index d4c02de..ce1b826 100644
--- a/src/helpers/choose-port.js
+++ b/src/helpers/choose-port.js
@@ -8,7 +8,7 @@
  *
  */
 const chalk = require( 'chalk' );
-const inquirer = require( 'inquirer' );
+const prompts = require( 'prompts' );
 const detect = require( 'detect-port-alt' );
 const isRoot = require( 'is-root' );
 
@@ -17,6 +17,8 @@ const getProcessForPort = require( '../vendor/get-process-for-port' );
 const DEFAULT_PORT = process.env.PORT || 8080;
 const HOST = process.env.HOST || '0.0.0.0';
 
+const isInteractive = process.stdout.isTTY;
+
 /**
  * choosePort method adapted from create-react-app's `react-dev-utils`.
  *
@@ -33,25 +35,29 @@ const choosePort = ( host, defaultPort ) => detect( defaultPort, host ).then(
 		const message = process.platform !== 'win32' && defaultPort < 1024 && ! isRoot() ?
 			'Admin permissions are required to run a server on a port below 1024.' :
 			`Something is already running on port ${ defaultPort }.`;
-		// clearConsole();
-		const existingProcess = getProcessForPort( defaultPort );
-		const question = {
-			type: 'confirm',
-			name: 'shouldChangePort',
-			message: `${
-				chalk.yellow( `${ message }${
-					existingProcess ? ` Probably:\n  ${ existingProcess }` : ''
-				}` )
-			}\n\nWould you like to run the app on another port instead?`,
-			default: true,
-		};
-		inquirer.prompt( question ).then( answer => {
-			if ( answer.shouldChangePort ) {
-				resolve( port );
-			} else {
-				resolve( null );
-			}
-		} );
+		if ( isInteractive ) {
+			const existingProcess = getProcessForPort( defaultPort );
+			const question = {
+				type: 'confirm',
+				name: 'shouldChangePort',
+				message: `${
+					chalk.yellow( `${ message }${
+						existingProcess ? ` Probably:\n  ${ existingProcess }` : ''
+					}` )
+				}\n\nWould you like to run the app on another port instead?`,
+				initial: true,
+			};
+			prompts( question ).then( answer => {
+				if ( answer.shouldChangePort ) {
+					resolve( port );
+				} else {
+					resolve( null );
+				}
+			} );
+		} else {
+			console.log( chalk.red( message ) );
+			resolve( null );
+		}
 	} ),
 	err => {
 		throw new Error( `${
diff --git a/src/helpers/deep-merge.js b/src/helpers/deep-merge.js
index cf0f3a5..586bcbd 100644
--- a/src/helpers/deep-merge.js
+++ b/src/helpers/deep-merge.js
@@ -12,7 +12,7 @@ const isArr = val => Array.isArray( val );
  * @param {*} val A value to test for object-ness.
  * @returns {Boolean} Whether the provided argument is an object.
  */
-const isObj = val => ( typeof val === 'object' );
+const isObj = val => ( typeof val === 'object' && val !== null );
 
 /**
  * Given two objects, merge array and object properties, then allow scalar
diff --git a/src/helpers/deep-merge.test.js b/src/helpers/deep-merge.test.js
index 1ec25dc..2867239 100644
--- a/src/helpers/deep-merge.test.js
+++ b/src/helpers/deep-merge.test.js
@@ -35,4 +35,19 @@ describe( 'helpers/deep-merge', () => {
 			},
 		} );
 	} );
+
+	it( 'can remove a property by specifying `undefined` in the second merged object', () => {
+		const obj1 = {
+			test: '.jsx?$',
+			devServer: {
+				port: 9191,
+			},
+		};
+		const obj2 = {
+			devServer: undefined,
+		};
+		expect( deepMerge( obj1, obj2 ) ).toEqual( {
+			test: '.jsx?$',
+		} );
+	} );
 } );
diff --git a/src/helpers/filters.js b/src/helpers/filters.js
new file mode 100644
index 0000000..b43a220
--- /dev/null
+++ b/src/helpers/filters.js
@@ -0,0 +1,102 @@
+const registry = {};
+
+/**
+ * Internal helper to return an ordered array of callbacks for a given priority.
+ *
+ * @private
+ * @param {string} hookName A hook name for which to return filters.
+ * @returns {Function[]} Array of filter functions.
+ */
+const getCallbacks = ( hookName ) => {
+	const priorities = Object.values( registry[ hookName ] || {} )
+		// eslint-disable-next-line id-length
+		.sort( ( a, b ) => ( +a < +b ? -1 : 1 ) );
+	return priorities.reduce(
+		( callbacks, priority ) => {
+			return callbacks.concat( priority || [] );
+		},
+		[]
+	);
+};
+
+/**
+ * Internal helper to set the registry to a specific state for testing.
+ *
+ * @private
+ * @param {Object} values New registry shape.
+ */
+const setupRegistry = ( values = {} ) => {
+	Object.keys( registry ).forEach( ( existingKey ) => {
+		Reflect.deleteProperty( registry, existingKey );
+	} );
+
+	Object.keys( values ).forEach( ( newKey ) => {
+		registry[ newKey ] = values[ newKey ];
+	} );
+};
+
+/**
+ * Register a filter function for a given filter hook name.
+ *
+ * @param {string}   hookName   Name of hook to add a filter on.
+ * @param {Function} callback   Callback to run on this hook.
+ * @param {Number}   [priority] Optional numeric priority, default 10.
+ */
+const addFilter = ( hookName, callback, priority = 10 ) => {
+	if ( ! registry[ hookName ] ) {
+		registry[ hookName ] = {};
+	}
+	if ( ! Array.isArray( registry[ hookName ][ priority ] ) ) {
+		registry[ hookName ][ priority ] = [];
+	}
+	registry[ hookName ][ priority ].push( callback );
+};
+
+/**
+ * Remove a previously-added filter function for a given hook name.
+ *
+ * To remove a hook, the callback and priority arguments must match when the hook was added.
+ *
+ * @param {string}   hookName   Name of hook to add a filter on.
+ * @param {Function} callback   Callback to run on this hook.
+ * @param {Number}   [priority] Priority from which to remove the hook.
+ */
+const removeFilter = ( hookName, callback, priority = 10 ) => {
+	if ( ! registry[ hookName ] || ! registry[ hookName ][ priority ] ) {
+		// Not added, no action needed.
+		return;
+	}
+	registry[ hookName ][ priority ] = registry[ hookName ][ priority ]
+		.filter( ( hook ) => hook !== callback );
+};
+
+/**
+ * Run input arguments through all registered callbacks for a specified filter,
+ * allowing each filter to transform the first argument.
+ *
+ * @param {string} hookName Name of hook to filter on.
+ * @param {...any} args     Filter input arguments.
+ * @returns {any} Output of filter chain.
+ */
+const applyFilters = ( hookName, ...args ) => {
+	const callbacks = getCallbacks( hookName );
+
+	const [ firstArg, ...otherArgs ] = args;
+
+	return callbacks.reduce(
+		( val, callback ) => callback( val, ...otherArgs ),
+		firstArg,
+	);
+};
+
+module.exports = {
+	addFilter,
+	removeFilter,
+	applyFilters,
+};
+
+if ( process.env.JEST_WORKER_ID ) {
+	// Exposed only for testing purposes.
+	module.exports.getCallbacks = getCallbacks;
+	module.exports.setupRegistry = setupRegistry;
+}
diff --git a/src/helpers/filters.test.js b/src/helpers/filters.test.js
new file mode 100644
index 0000000..09fce2b
--- /dev/null
+++ b/src/helpers/filters.test.js
@@ -0,0 +1,142 @@
+const {
+	getCallbacks,
+	setupRegistry,
+	addFilter,
+	removeFilter,
+	applyFilters,
+} = require( './filters' );
+
+// Functions to use when testing filtering.
+const toKebabCase = ( str ) => str.split( /[_ ]+/g ).join( '-' );
+const toLowerCase = ( str ) => str.toLowerCase();
+const toUpperCase = ( str ) => str.toUpperCase();
+const reverse = ( str ) => str.split( '' ).reverse().join( '' );
+const reverseWords = ( str ) => str.split( ' ' ).reverse().join( ' ' );
+
+describe( 'filters', () => {
+	describe( '.getCallbacks() [internal]', () => {
+
+		it( 'is a function', () => {
+			expect( getCallbacks ).toBeInstanceOf( Function );
+		} );
+
+		it( 'returns callbacks for the requested hook', () => {
+			setupRegistry( {
+				'loaders/js': {
+					10: [
+						toKebabCase,
+						toLowerCase,
+					],
+				},
+				'loaders/css': {
+					10: [
+						toUpperCase,
+					],
+				},
+			} );
+			expect( getCallbacks( 'loaders/js' ) ).toEqual( [
+				toKebabCase,
+				toLowerCase,
+			] );
+			expect( getCallbacks( 'loaders/css' ) ).toEqual( [
+				toUpperCase,
+			] );
+		} );
+
+		it( 'returns callbacks in priority order', () => {
+			setupRegistry( {
+				'loaders/js': {
+					10: [ 'a', 'b' ],
+					3: [ 'c' ],
+					22: [ 'd' ],
+					7: [ 'e', 'f' ],
+				},
+			} );
+			const hooks = getCallbacks( 'loaders/js' );
+			expect( hooks ).toEqual( [ 'c', 'e', 'f', 'a', 'b', 'd' ] );
+		} );
+
+		it( 'returns empty array for uninitialized hook', () => {
+			setupRegistry();
+			const hooks = getCallbacks( 'unused/hook' );
+			expect( hooks ).toEqual( [] );
+		} );
+	} );
+
+	describe( 'addFilter', () => {
+		beforeEach( () => {
+			setupRegistry();
+		} );
+
+		it( 'adds a callback for the specified filter', () => {
+			const hook = () => 'Woo';
+			addFilter( 'loaders/js', hook );
+			expect( getCallbacks( 'loaders/js' ) ).toEqual( [ hook ] );
+		} );
+
+		it( 'adds callbacks with the specified priority', () => {
+			const hook = () => 'Woo';
+			const hook2 = () => 'Woo-er';
+			const hook3 = () => 'Woo-est';
+			addFilter( 'loaders/js', hook2, 9 );
+			addFilter( 'loaders/js', hook3, 11 );
+			addFilter( 'loaders/js', hook );
+			expect( getCallbacks( 'loaders/js' ) ).toEqual( [ hook2, hook, hook3 ] );
+		} );
+	} );
+
+	describe( 'removeFilter', () => {
+		beforeEach( () => {
+			setupRegistry();
+		} );
+
+		it( 'removes a previously-added callback for the specified filter', () => {
+			const hook = () => 'Woo';
+			addFilter( 'loaders/js', hook );
+			removeFilter( 'loaders/js', hook );
+			expect( getCallbacks( 'loaders/js' ) ).toEqual( [] );
+		} );
+
+		it( 'only removes the requested callback', () => {
+			const hook = () => 'Woo';
+			const hook2 = () => 'Woo-er';
+			addFilter( 'loaders/js', hook );
+			addFilter( 'loaders/js', hook2 );
+			removeFilter( 'loaders/js', hook );
+			expect( getCallbacks( 'loaders/js' ) ).toEqual( [ hook2 ] );
+		} );
+
+		it( 'does not remove a callback if the priority does not match', () => {
+			const hook = () => 'Woo';
+			addFilter( 'loaders/js', hook, 9 );
+			removeFilter( 'loaders/js', hook, 10 );
+			expect( getCallbacks( 'loaders/js' ) ).toEqual( [ hook ] );
+		} );
+	} );
+
+	describe( 'applyFilters', () => {
+		const input = 'Pasta Carbonara';
+
+		beforeEach( () => {
+			setupRegistry();
+		} );
+
+		it( 'returns input unchanged if no filters', () => {
+			expect( applyFilters( 'do/stuff', input ) ).toBe( 'Pasta Carbonara' );
+		} );
+
+		it( 'processes input with the registered filters', () => {
+			addFilter( 'words', toKebabCase );
+			addFilter( 'words', toUpperCase );
+			expect( applyFilters( 'words', input ) ).toBe( 'PASTA-CARBONARA' );
+		} );
+
+		it( 'processes input through callbacks in the specified order', () => {
+			addFilter( 'words', toLowerCase, 9 );
+			addFilter( 'words', toKebabCase, 11 );
+			addFilter( 'words', reverseWords, 10 );
+			addFilter( 'words', reverse, 10 );
+			expect( applyFilters( 'words', input ) ).toBe( 'atsap-aranobrac' );
+		} );
+	} );
+} );
diff --git a/src/helpers/infer-public-path.js b/src/helpers/infer-public-path.js
index e60f024..e2a9b71 100644
--- a/src/helpers/infer-public-path.js
+++ b/src/helpers/infer-public-path.js
@@ -1,16 +1,35 @@
 const filePath = require( './file-path' );
 const findInObject = require( './find-in-object' );
 
+/**
+ * Dictionary of generated publicPath strings by file system path.
+ *
+ * @type {Object.<string, string>}
+ */
+const publicPaths = {};
+
+/**
+ * Return a consistent publicPath per output directory path.
+ *
+ * Only used when automatically generating public paths.
+ *
+ * @param {String} path Output directory path
+ * @returns {string} Shared publicPath string.
+ */
+const getPublicPathForDirectory = ( path ) => {
+	return publicPaths[ path ];
+};
+
 /**
  * Infer the public path based on the defined output path.
  *
  * @param {webpack.Configuration} config     Webpack configuration object.
  * @param {Number}                port       Port to use for webpack-dev-server.
  * @param {Object}                [defaults] Optional config default values.
- * @return {String} Public path.
+ * @returns {String} Public path.
  */
 const inferPublicPath = ( config, port, defaults = {} ) => {
-	const protocol = findInObject( config, 'devServer.https' ) ? 'https' : 'http';
+	const protocol = ( findInObject( config, 'devServer.https' ) || findInObject( config, 'devServer.server' ) === 'https' ) ? 'https' : 'http';
 
 	const outputPath = findInObject( config, 'output.path' ) || findInObject( defaults, 'output.path' );
 
@@ -21,7 +40,22 @@ const inferPublicPath = ( config, port, defaults = {} ) => {
 		.replace( /^\/*/, '' )
 		.replace( /\/*$/, '/' );
 
-	return `${ protocol }://localhost:${ port }/${ relPath }`;
+	const publicPath = `${ protocol }://localhost:${ port }/${ relPath }`;
+
+	publicPaths[ outputPath ] = publicPath;
+	return publicPath;
+};
+
+module.exports = {
+	inferPublicPath,
+	getPublicPathForDirectory,
 };
 
-module.exports = inferPublicPath;
+if ( process.env.JEST_WORKER_ID ) {
+	// Exposed only for testing purposes.
+	module.exports.resetPublicPathsCache = () => {
+		Object.keys( publicPaths ).forEach( ( key ) => {
+			Reflect.deleteProperty( publicPaths, key );
+		} );
+	};
+}
diff --git a/src/helpers/with-dynamic-port.js b/src/helpers/with-dynamic-port.js
index a22567f..b09df26 100644
--- a/src/helpers/with-dynamic-port.js
+++ b/src/helpers/with-dynamic-port.js
@@ -6,7 +6,7 @@
 const choosePort = require( './choose-port' );
 const filePath = require( './file-path' );
 const findInObject = require( './find-in-object' );
-const inferPublicPath = require( './infer-public-path' );
+const { inferPublicPath } = require( './infer-public-path' );
 
 const DEFAULT_PORT = 9090;
 
diff --git a/src/helpers/with-dynamic-port.test.js b/src/helpers/with-dynamic-port.test.js
index 7955c63..5b2b7ff 100644
--- a/src/helpers/with-dynamic-port.test.js
+++ b/src/helpers/with-dynamic-port.test.js
@@ -1,6 +1,7 @@
 const withDynamicPort = require( './with-dynamic-port' );
 
 const choosePort = require( './choose-port' );
+const { resetPublicPathsCache } = require( './infer-public-path' );
 
 jest.mock( './choose-port', () => jest.fn() );
 
@@ -9,6 +10,7 @@ describe( 'withDynamicPort', () => {
 
 	beforeEach( () => {
 		oldArgv = process.argv;
+		resetPublicPathsCache();
 	} );
 
 	afterEach( () => {
@@ -151,7 +153,7 @@ describe( 'withDynamicPort', () => {
 		it( 'does not overwrite existing config properties', async () => {
 			const config = {
 				devServer: {
-					https: true,
+					server: 'https',
 				},
 				entry: {
 					'name': './bundle.js',
@@ -167,7 +169,7 @@ describe( 'withDynamicPort', () => {
 			choosePort.mockImplementationOnce( () => Promise.resolve( 8082 ) );
 			expect( await withDynamicPort( 9090, config ) ).toEqual( {
 				devServer: {
-					https: true,
+					server: 'https',
 					port: 8082,
 				},
 				entry: {
diff --git a/src/loaders.js b/src/loaders.js
index 0a23115..58345b9 100644
--- a/src/loaders.js
+++ b/src/loaders.js
@@ -1,43 +1,73 @@
 /**
  * Export generator functions for common Webpack loader configurations.
  */
+const cloneDeep = require( 'lodash.clonedeep' );
 const postcssFlexbugsFixes = require( 'postcss-flexbugs-fixes' );
 const postcssPresetEnv = require( 'postcss-preset-env' );
 
 const deepMerge = require( './helpers/deep-merge' );
+const { applyFilters } = require( './helpers/filters' );
 
 /**
  * Export an object of named methods that generate corresponding loader config
- * objects. To customize the default values of the loader, mutate the .defaults
- * property exposed on each method (or pass a filterLoaders option to a preset).
+ * objects. To customize the default values of the loader, use addFilter() on
+ * either the hook `loader/loadername/defaults` (to adjust the default loader
+ * configuration) or `loader/loadername` (to alter the final, computed loader).
  */
 const loaders = {};
 
+/**
+ * Create a loader factory function for a given loader slug.
+ *
+ * @private
+ * @param {string} loaderKey Loader slug
+ * @returns {Function} Factory function to generate and filter a loader with the specified slug.
+ */
 const createLoaderFactory = loaderKey => {
-	const getFilteredLoader = ( options ) => {
-		// Handle missing options object.
-		if ( typeof options === 'function' ) {
-			return getFilteredLoader( {}, options );
-		}
-
-		// Generate the requested loader definition.
-		return deepMerge( loaders[ loaderKey ].defaults, options );
+	return ( options = {}, config = null ) => {
+		/**
+		 * Generate the requested loader definition.
+		 *
+		 * Allows customization of the final rendered loader via the loaders/{loaderslug}
+		 * filter, which also receives the configuration passed to a preset factory if
+		 * the loader is invoked in the context of a preset.
+		 *
+		 * @hook loaders/{$loader_slug}
+		 * @param {Object}      loader   Complete loader definition object, after merging user-provided values with filtered defaults.
+		 * @param {Object|null} [config] Configuration object for the preset being rendered, if loader is called while generating a preset.
+		 */
+		return applyFilters(
+			`loaders/${ loaderKey }`,
+			deepMerge(
+				/**
+				 * Filter the loader's default configuration.
+				 *
+				 * @hook loaders/{$loader_slug}/defaults
+				 * @param {Object}      options  Loader default options object.
+				 * @param {Object|null} [config] Configuration object for the preset being rendered, if loader is called while generating a preset.
+				 */
+				applyFilters( `loaders/${ loaderKey }/defaults`, cloneDeep( loaders[ loaderKey ].defaults ), config ),
+				options
+			),
+			config
+		);
 	};
-
-	return getFilteredLoader;
 };
 
 // Define all supported loader factories within the loaders object.
-[ 'eslint', 'js', 'ts', 'url', 'style', 'css', 'postcss', 'sass', 'file' ].forEach( loaderKey => {
+[ 'asset', 'js', 'ts', 'style', 'css', 'postcss', 'sass', 'sourcemap', 'resource' ].forEach( loaderKey => {
 	loaders[ loaderKey ] = createLoaderFactory( loaderKey );
 } );
 
-loaders.eslint.defaults = {
-	test: /\.jsx?$/,
-	exclude: /(node_modules|bower_components)/,
-	enforce: 'pre',
-	loader: require.resolve( 'eslint-loader' ),
-	options: {},
+loaders.asset.defaults = {
+	test: /\.(png|jpg|jpeg|gif|avif|webp|svg|woff|woff2|eot|ttf)$/,
+	type: 'asset',
+	parser: {
+		dataUrlCondition: {
+			// Inline if less than 10kb.
+			maxSize: 10 * 1024,
+		},
+	},
 };
 
 loaders.js.defaults = {
@@ -56,14 +86,6 @@ loaders.ts.defaults = {
 	loader: require.resolve( 'ts-loader' ),
 };
 
-loaders.url.defaults = {
-	test: /\.(png|jpg|jpeg|gif|svg|woff|woff2|eot|ttf)$/,
-	loader: require.resolve( 'url-loader' ),
-	options: {
-		limit: 10000,
-	},
-};
-
 loaders.style.defaults = {
 	loader: require.resolve( 'style-loader' ),
 	options: {},
@@ -80,15 +102,28 @@ loaders.postcss.defaults = {
 	loader: require.resolve( 'postcss-loader' ),
 	options: {
 		postcssOptions: {
-			plugins: [
+			ident: 'postcss',
+			/**
+			 * Filter the default PostCSS Plugins array.
+			 *
+			 * @hook loaders/postcss/plugins
+			 * @param {Array} plugins Array of PostCSS plugins.
+			 */
+			plugins: applyFilters( 'loaders/postcss/plugins', [
 				postcssFlexbugsFixes,
-				postcssPresetEnv( {
+				/**
+				 * Filter the default PostCSS Preset Env configuration.
+				 *
+				 * @hook loaders/postcss/preset-env
+				 * @param {Object} presetEnvConfig PostCSS Preset Env plugin configuration.
+				 */
+				postcssPresetEnv( applyFilters( 'loaders/postcss/preset-env', {
 					autoprefixer: {
 						flexbox: 'no-2009',
 					},
 					stage: 3,
-				} ),
-			],
+				} ) ),
+			] ),
 		},
 	},
 };
@@ -102,11 +137,21 @@ loaders.sass.defaults = {
 	},
 };
 
-loaders.file.defaults = {
-	// Exclude `js`, `html` and `json`, but match anything else.
-	exclude: /\.(js|html|json)$/,
-	loader: require.resolve( 'file-loader' ),
+loaders.sourcemap.defaults = {
+	test: /\.(js|mjs|jsx|ts|tsx|css)$/,
+	exclude: /@babel(?:\/|\\{1,2})runtime/,
+	enforce: 'pre',
+	loader: require.resolve( 'source-map-loader' ),
 	options: {},
 };
 
+loaders.resource.defaults = {
+	// Exclude `js` files to keep "css" loader working as it injects
+	// its runtime that would otherwise be processed through "file" loader.
+	// Also exclude `html` and `json` extensions so they get processed
+	// by webpacks internal loaders.
+	exclude: [ /^$/, /\.(js|mjs|jsx|ts|tsx)$/, /\.html?$/, /\.json$/ ],
+	type: 'asset/resource',
+};
+
 module.exports = loaders;
diff --git a/src/loaders.test.js b/src/loaders.test.js
index deb445d..586731a 100644
--- a/src/loaders.test.js
+++ b/src/loaders.test.js
@@ -1,3 +1,4 @@
+const { addFilter, setupRegistry } = require( './helpers/filters' );
 const loaders = require( './loaders' );
 
 describe( 'loaders', () => {
@@ -22,23 +23,15 @@ describe( 'loaders', () => {
 		expect( result.value ).toBe( 42 );
 	} );
 
-	describe( '.eslint()', () => {
-		it( 'tests for .js or .jsx files', () => {
-			expect( 'file.js'.match( loaders.eslint().test ) ).not.toBeNull();
-			expect( 'file.jsx'.match( loaders.eslint().test ) ).not.toBeNull();
-			expect( 'file.scss'.match( loaders.eslint().test ) ).toBeNull();
-		} );
-	} );
-
 	describe( '.js()', () => {
 		it( 'tests for .js or .jsx files', () => {
-			expect( 'file.js'.match( loaders.eslint().test ) ).not.toBeNull();
-			expect( 'file.jsx'.match( loaders.eslint().test ) ).not.toBeNull();
-			expect( 'file.scss'.match( loaders.eslint().test ) ).toBeNull();
+			expect( 'file.js'.match( loaders.js().test ) ).not.toBeNull();
+			expect( 'file.jsx'.match( loaders.js().test ) ).not.toBeNull();
+			expect( 'file.scss'.match( loaders.js().test ) ).toBeNull();
 		} );
 	} );
 
-	describe( '.url()', () => {
+	describe( '.asset()', () => {
 		it( 'tests for static assets', () => {
 			[
 				'file.png',
@@ -51,15 +44,33 @@ describe( 'loaders', () => {
 				'file.eot',
 				'file.ttf',
 			].forEach( acceptedFileType => {
-				expect( acceptedFileType.match( loaders.url().test ) ).not.toBeNull();
+				expect( acceptedFileType.match( loaders.asset().test ) ).not.toBeNull();
 			} );
 			[
 				'file.js',
 				'file.css',
 				'file.html',
 			].forEach( unacceptedFileType => {
-				expect( unacceptedFileType.match( loaders.url().test ) ).toBeNull();
+				expect( unacceptedFileType.match( loaders.asset().test ) ).toBeNull();
+			} );
+		} );
+	} );
+
+	describe( 'filtering', () => {
+		beforeEach( () => {
+			setupRegistry();
+		} );
+
+		it( 'does not mutate defaults', () => {
+			addFilter( 'loaders/js/defaults', ( defaults ) => {
+				defaults.test = /\.js$/;
+				return defaults;
 			} );
+
+			const loader = loaders.js();
+
+			expect( loader.test ).toEqual( /\.js$/ );
+			expect( loaders.js.defaults.test ).toEqual( /\.jsx?$/ );
 		} );
 	} );
 
diff --git a/src/plugins.js b/src/plugins.js
index 3de1ad3..094a0da 100644
--- a/src/plugins.js
+++ b/src/plugins.js
@@ -1,15 +1,18 @@
 const { BundleAnalyzerPlugin } = require( 'webpack-bundle-analyzer' );
 const { CleanWebpackPlugin } = require( 'clean-webpack-plugin' );
-const { HotModuleReplacementPlugin } = require( 'webpack' );
 const BellOnBundleErrorPlugin = require( 'bell-on-bundler-error-plugin' );
 const CopyPlugin = require( 'copy-webpack-plugin' );
-const FixStyleOnlyEntriesPlugin = require( 'webpack-fix-style-only-entries' );
+const ESLintPlugin = require( 'eslint-webpack-plugin' );
 const { WebpackManifestPlugin: ManifestPlugin } = require( 'webpack-manifest-plugin' );
 const MiniCssExtractPlugin = require( 'mini-css-extract-plugin' );
+const RemoveEmptyScriptsPlugin = require( 'webpack-remove-empty-scripts' );
+const SimpleBuildReportPlugin = require( 'simple-build-report-webpack-plugin' );
 const TerserPlugin = require( 'terser-webpack-plugin' );
-const OptimizeCssAssetsPlugin = require( 'optimize-css-assets-webpack-plugin' );
+const CssMinimizerPlugin = require( 'css-minimizer-webpack-plugin' );
 
-const deepMerge = require( './helpers/deep-merge' );
+const { applyFilters } = require( './helpers/filters' );
+
+const isProfileMode = process.argv.includes( '--profile' );
 
 module.exports = {
 	/**
@@ -22,11 +25,12 @@ module.exports = {
 		BundleAnalyzerPlugin,
 		CleanWebpackPlugin,
 		CopyPlugin,
-		FixStyleOnlyEntriesPlugin,
-		HotModuleReplacementPlugin,
+		ESLintPlugin,
 		ManifestPlugin,
 		MiniCssExtractPlugin,
-		OptimizeCssAssetsPlugin,
+		CssMinimizerPlugin,
+		RemoveEmptyScriptsPlugin,
+		SimpleBuildReportPlugin,
 		TerserPlugin,
 	},
 
@@ -47,14 +51,14 @@ module.exports = {
 	},
 
 	/**
-	 * Create a new BundleAnalyzerPlugin instance. The analyzer is enabled by default
-	 * only if `--analyze` is passed on the command line.
+	 * Create a new BundleAnalyzerPlugin instance.
 	 *
 	 * @param {Object} [options] Optional plugin options object.
 	 * @returns {BundleAnalyzerPlugin} A configured BundleAnalyzerPlugin instance.
 	 */
 	bundleAnalyzer: ( options = {} ) => new BundleAnalyzerPlugin( {
-		analyzerMode: process.argv.indexOf( '--analyze' ) >= 0 ? 'static' : 'disabled',
+		analyzerMode: 'static',
+		generateStatsFile: true,
 		openAnalyzer: false,
 		reportFilename: 'bundle-analyzer-report.html',
 		...options,
@@ -99,22 +103,26 @@ module.exports = {
 	errorBell: () => new BellOnBundleErrorPlugin(),
 
 	/**
-	 * Create a new FixStyleOnlyEntriesPlugin instance to remove unnecessary JS
-	 * files generated for style-only bundle entries.
+	 * Create a new ESLintPlugin instance to check your build for syntax or style issues.
 	 *
-	 * @param {Object} [options]         Optional plugin options object.
-	 * @param {RegExp} [options.exclude] Regular expression to filter what gets cleaned.
-	 * @returns {FixStyleOnlyEntriesPlugin} A configured FixStyleOnlyEntriesPlugin instance.
+	 * @param {Object} options Optional plugin options object.
+	 * @returns {ESLintPlugin} A configured ESLintPlugin instance.
 	 */
-	fixStyleOnlyEntries: ( options ) => new FixStyleOnlyEntriesPlugin( options ),
+	eslint: ( options ) => new ESLintPlugin( {
+		extensions: [ 'js', 'jsx', 'ts', 'tsx' ],
+		exclude: [ 'node_modules', 'vendor', '*.min.js', '*.min.jsx' ],
+		...options,
+	} ),
 
 	/**
-	 * Create a webpack.HotModuleReplacementPlugin instance.
+	 * Create a new RemoveEmptyScriptsPlugin instance to remove unnecessary JS
+	 * files generated for style-only bundle entries.
 	 *
-	 * @param {Object} [options] Optional plugin options object.
-	 * @returns {HotModuleReplacementPlugin} A configured HMR Plugin instance.
+	 * @param {Object} [options]         Optional plugin options object.
+	 * @param {RegExp} [options.exclude] Regular expression to filter what gets cleaned.
+	 * @returns {RemoveEmptyScriptsPlugin} A configured RemoveEmptyScriptsPlugin instance.
 	 */
-	hotModuleReplacement: ( options = {} ) => new HotModuleReplacementPlugin( options ),
+	fixStyleOnlyEntries: ( options ) => new RemoveEmptyScriptsPlugin( options ),
 
 	/**
 	 * Create a new ManifestPlugin instance to output an asset-manifest.json
@@ -154,12 +162,20 @@ module.exports = {
 	} ),
 
 	/**
-	 * Create a new OptimizeCssAssetsPlugin instance.
+	 * Create a new CssMinimizerPlugin instance.
 	 *
 	 * @param {Object} [options] Optional plugin configuration options.
-	 * @returns {OptimizeCssAssetsPlugin} A configured OptimizeCssAssetsPlugin instance.
+	 * @returns {CssMinimizerPlugin} A configured CssMinimizerPlugin instance.
 	 */
-	optimizeCssAssets: ( options = {} ) => new OptimizeCssAssetsPlugin( options ),
+	cssMinimizer: ( options = {} ) => new CssMinimizerPlugin( options ),
+
+	/**
+	 * Instantiate a SimpleBuildReportPlugin to render build output in a human-
+	 * oriented fashion.
+	 *
+	 * @returns {SimpleBuildReportPlugin} Output formatter plugin instance.
+	 */
+	simpleBuildReport: () => new SimpleBuildReportPlugin(),
 
 	/**
 	 * Create a new TerserPlugin instance, defaulting to a set of options
@@ -170,41 +186,46 @@ module.exports = {
 	 * @param {Object} [options.terserOptions] Terser compressor options object.
 	 * @returns {TerserPlugin} A configured TerserPlugin instance.
 	 */
-	terser: ( options = {} ) => new TerserPlugin( deepMerge( {
-		terserOptions: {
-			parse: {
-				// we want terser to parse ecma 8 code. However, we don't want it
-				// to apply any minfication steps that turns valid ecma 5 code
-				// into invalid ecma 5 code. This is why the 'compress' and 'output'
-				// sections only apply transformations that are ecma 5 safe
-				// https://github.com/facebook/create-react-app/pull/4234
-				ecma: 8,
-			},
-			compress: {
-				ecma: 5,
-				warnings: false,
-				// Disabled because of an issue with Uglify breaking seemingly valid code:
-				// https://github.com/facebook/create-react-app/issues/2376
-				// Pending further investigation:
-				// https://github.com/mishoo/UglifyJS2/issues/2011
-				comparisons: false,
-				// Disabled because of an issue with Terser breaking valid code:
-				// https://github.com/facebook/create-react-app/issues/5250
-				// Pending futher investigation:
-				// https://github.com/terser-js/terser/issues/120
-				inline: 2,
+	terser: ( options = {} ) => new TerserPlugin( {
+		...applyFilters( 'plugins/terser/defaults', {
+			terserOptions: {
+				parse: {
+					// We want terser to parse ecma 8 code. However, we don't want it
+					// to apply any minification steps that turns valid ecma 5 code
+					// into invalid ecma 5 code. This is why the 'compress' and 'output'
+					// sections only apply transformations that are ecma 5 safe
+					// https://github.com/facebook/create-react-app/pull/4234
+					ecma: 8,
+				},
+				compress: {
+					ecma: 5,
+					warnings: false,
+					// Disabled because of an issue with Uglify breaking seemingly valid code:
+					// https://github.com/facebook/create-react-app/issues/2376
+					// Pending further investigation:
+					// https://github.com/mishoo/UglifyJS2/issues/2011
+					comparisons: false,
+					// Disabled because of an issue with Terser breaking valid code:
+					// https://github.com/facebook/create-react-app/issues/5250
+					// Pending further investigation:
+					// https://github.com/terser-js/terser/issues/120
+					inline: 2,
+				},
+				mangle: {
+					safari10: true,
+				},
+				// Added for profiling in devtools
+				keep_classnames: isProfileMode,
+				keep_fnames: isProfileMode,
+				output: {
+					ecma: 5,
+					comments: false,
+					// Turned on because emoji and regex is not minified properly using default
+					// https://github.com/facebook/create-react-app/issues/2488
+					ascii_only: true,
+				},
 			},
-			mangle: {
-				safari10: true,
-			},
-			output: {
-				ecma: 5,
-				comments: false,
-				// Turned on because emoji and regex is not minified properly using default
-				// https://github.com/facebook/create-react-app/issues/2488
-				ascii_only: true,
-			},
-		},
-		extractComments: false,
-	}, options ) ),
+		} ),
+		...options,
+	} ),
 };
diff --git a/src/presets.js b/src/presets.js
index 9f3a28a..75386f9 100644
--- a/src/presets.js
+++ b/src/presets.js
@@ -2,12 +2,15 @@ const { devServer, stats } = require( './config' );
 const deepMerge = require( './helpers/deep-merge' );
 const filePath = require( './helpers/file-path' );
 const findInObject = require( './helpers/find-in-object' );
-const inferPublicPath = require( './helpers/infer-public-path' );
+const { inferPublicPath, getPublicPathForDirectory } = require( './helpers/infer-public-path' );
 const isInstalled = require( './helpers/is-installed' );
+const { applyFilters } = require( './helpers/filters' );
 const loaders = require( './loaders' );
 const plugins = require( './plugins' );
 const { ManifestPlugin, MiniCssExtractPlugin } = plugins.constructors;
 
+const isAnalyzeMode = process.argv.includes( '--analyze' );
+
 /**
  * Dictionary of shared seed objects by path.
  *
@@ -30,11 +33,11 @@ const getSeedByDirectory = ( path ) => {
 
 /**
  * Helper to detect whether a given package is installed, and return a spreadable
- * array containing an appropriate loader if so.
+ * array containing an appropriate loader or plugin if so.
  *
  * @example
- *     rules: [
- *         ...ifInstalled( 'eslint', loaders.eslint() ),
+ *     plugins: [
+ *         ...ifInstalled( 'eslint', plugins.eslint() ),
  *     ],
  *
  * @param {String} packageName The string name of the dependency for which to test.
@@ -50,38 +53,36 @@ const ifInstalled = ( packageName, loader ) => {
 };
 
 /**
- * Given a reference to a (possibly-undefined) filtering method, return a pair
- * of helper functions to filter a loader definition object, or to get a loader
- * definition object by its loaders dictionary key and then filter it.
+ * Remove null entries from Webpack loaders array, in case a user returned null
+ * from a filter to opt out of a given loader in the preset.
+ *
+ * Detects and removes null entries from nested .oneOf or .use arrays.
+ *
+ * @param {Object[]} moduleRules Array of Webpack loader rules.
  *
- * @param {Function} [filterLoaders] An optional filterLoaders function. Defaults
- *                                   to the identity function.
- * @returns {Object} Object with `filterLoaders` and `getFilteredLoader` methods.
+ * @returns {Object[]} Filtered array with null items removed.
  */
-const createFilteringHelpers = ( filterLoaders = ( input ) => input ) => ( {
-	/**
-	 * Given a loader object and its key string, pass that object through the
-	 * filterLoaders method (if one was provided) and return the filtered output.
-	 *
-	 * @param {Object} loader    Webpack loader configuration object.
-	 * @param {String} loaderKey String identifying which loader is being filtered.
-	 * @returns {Object} Filtered loader object.
-	 */
-	filterLoaders: ( loader, loaderKey ) => {
-		return filterLoaders( loader, loaderKey );
-	},
-
-	/**
-	 * Helper method to reduce duplication when accessing and invoking loader factories.
-	 *
-	 * @param {String} loaderKey String key of a loader factory in the loaders object.
-	 * @param {Object} [options] Options for this loader (optional).
-	 * @returns {Object} Configured and filtered loader definition.
-	 */
-	getFilteredLoader: ( loaderKey, options ) => {
-		return filterLoaders( loaders[ loaderKey ]( options ), loaderKey );
-	},
-} );
+const removeNullLoaders = ( moduleRules ) => moduleRules
+	.map( ( rule ) => {
+		if ( rule && Array.isArray( rule.oneOf ) ) {
+			const loaders = removeNullLoaders( rule.oneOf );
+			if ( ! Array.isArray( loaders ) || ! loaders.length ) {
+				return null;
+			}
+			return {
+				...rule,
+				oneOf: loaders,
+			};
+		}
+		if ( rule && Array.isArray( rule.use ) ) {
+			return {
+				...rule,
+				use: removeNullLoaders( rule.use ),
+			};
+		}
+		return rule;
+	} )
+	.filter( Boolean );
 
 /**
  * Promote a partial Webpack config into a full development-oriented configuration.
@@ -95,17 +96,10 @@ const createFilteringHelpers = ( filterLoaders = ( input ) => input ) => ( {
  * - an `.output.publicPath` string (unless a devServer.port is specified,
  *   in which case publicPath defaults to `http://localhost:${ port }`)
  *
- * @param {webpack.Configuration} config                  Configuration options to deeply merge into the defaults.
- * @param {Object}                [options]               Optional options to modify configuration generation.
- * @param {Function}              [options.filterLoaders] An optional filter function that receives each
- *                                                        computed loader definition and the name of that
- *                                                        loader as it is generated, to permit per-config
- *                                                        customization of loader options.
+ * @param {webpack.Configuration} config Configuration options to deeply merge into the defaults.
  * @returns {webpack.Configuration} A merged Webpack configuration object.
  */
-const development = ( config = {}, options = {} ) => {
-	const { filterLoaders, getFilteredLoader } = createFilteringHelpers( options.filterLoaders );
-
+const development = ( config = {} ) => {
 	/**
 	 * Default development environment-oriented Webpack options. This object is
 	 * defined at the time of function execution so that any changes to the
@@ -128,75 +122,88 @@ const development = ( config = {}, options = {} ) => {
 			// Add /* filename */ comments to generated require()s in the output.
 			pathinfo: true,
 			// Provide a default output name.
-			filename: '[name].js',
+			filename: '[name].[fullhash].js',
 			// Provide chunk filename. Requires content hash for cache busting.
-			chunkFilename: '[name].[contenthash].chunk.js',
+			chunkFilename: '[name].[fullhash].chunk.js',
 			// `publicPath` will be inferred as a localhost URL based on output.path
 			// when a devServer.port value is available.
 		},
 
 		module: {
 			strictExportPresence: true,
-			rules: [
-				// Run all JS files through ESLint, if installed.
-				...ifInstalled( 'eslint', getFilteredLoader( 'eslint', {
-					options: {
-						emitWarning: true,
-					},
-				} ) ),
+			rules: removeNullLoaders( [
+				// Handle node_modules packages that contain sourcemaps.
+				loaders.sourcemap( {}, config ),
 				{
 					// "oneOf" will traverse all following loaders until one will
 					// match the requirements. If no loader matches, it will fall
-					// back to the "file" loader at the end of the loader list.
+					// back to the resource loader at the end of the loader list.
 					oneOf: [
 						// Enable processing TypeScript, if installed.
-						...ifInstalled( 'typescript', getFilteredLoader( 'ts' ) ),
+						...ifInstalled( 'typescript', loaders.ts( {}, config ) ),
 						// Process JS with Babel.
-						getFilteredLoader( 'js' ),
-						// Convert small files to data URIs.
-						getFilteredLoader( 'url' ),
-						// Parse styles using SASS, then PostCSS.
-						filterLoaders( {
-							test: /\.s?css$/,
-							use: [
-								getFilteredLoader( 'style' ),
-								getFilteredLoader( 'css', {
-									options: {
-										sourceMap: true,
-									},
-								} ),
-								getFilteredLoader( 'postcss', {
-									options: {
-										sourceMap: true,
-									},
-								} ),
-								getFilteredLoader( 'sass', {
-									options: {
-										sourceMap: true,
-									},
-								} ),
-							],
-						}, 'stylesheet' ),
-						// "file" loader makes sure any non-matching assets still get served.
-						// When you `import` an asset you get its filename.
-						getFilteredLoader( 'file' ),
+						loaders.js( {}, config ),
+						// Handle static asset files.
+						loaders.asset( {}, config ),
+						/**
+						 * Filter the full stylesheet loader definition for this preset.
+						 *
+						 * By default parses styles using Sass and then PostCSS.
+						 *
+						 * @hook presets/stylesheet-loaders
+						 * @param {Object} loader      Stylesheet loader rule.
+						 * @param {string} environment "development" or "production".
+						 * @param {Object} config      Preset configuration object.
+						 */
+						applyFilters(
+							'presets/stylesheet-loaders',
+							{
+								test: /\.s?css$/,
+								use: [
+									loaders.style( {}, config ),
+									loaders.css( {
+										options: {
+											sourceMap: true,
+										},
+									}, config ),
+									loaders.postcss( {
+										options: {
+											sourceMap: true,
+										},
+									}, config ),
+									loaders.sass( {
+										options: {
+											sourceMap: true,
+										},
+									}, config ),
+								],
+							},
+							'development',
+							config
+						),
+						// Resource loader makes sure any non-matching assets still get served.
+						// When you `import` an asset, you get its (virtual) filename.
+						loaders.resource( {}, config ),
 					],
 				},
-			],
+			] ),
 		},
 
 		optimization: {
-			nodeEnv: 'development',
+			nodeEnv: 'development'
 		},
 
-		devServer: {
-			...devServer,
-			stats,
-		},
+		devServer,
+
+		stats,
 
 		plugins: [
-			plugins.hotModuleReplacement(),
-		],
+			// Run all JS files through ESLint, if installed.
+			...ifInstalled( 'eslint', plugins.eslint( {
+				// But don't let errors block the build.
+				failOnError: false,
+			} ) ),
+		].filter( Boolean ),
 	};
 
 	// If no entry was provided, inject a default entry value.
@@ -214,23 +221,52 @@ const development = ( config = {}, options = {} ) => {
 		publicPath = inferPublicPath( config, port, devDefaults );
 	}
 
-	// If we had enough value to guess a publicPath, set that path as a default
-	// wherever appropriate and inject a ManifestPlugin instance to expose that
-	// public path to consuming applications. Any inferred values will still be
-	// overridden with their relevant values from `config`, when provided.
+	const outputPath = ( config.output && config.output.path ) || devDefaults.output.path;
+	// If we used a publicPath for this outputPath before, re-use it.
+	if ( ! publicPath && outputPath ) {
+		publicPath = getPublicPathForDirectory( outputPath );
+	}
+
+	// If we had enough config values to guess a publicPath, set that path in
+	// the default config so it can be used in generated manifests.
 	if ( publicPath ) {
+		// If publicPath is a URL, default the devServer host to match.
+		if ( typeof publicPath === 'string' && publicPath.includes( 'http' ) ) {
+			devDefaults.devServer.host = new URL( publicPath ).hostname;
+		}
 		devDefaults.output.publicPath = publicPath;
+	}
 
-		// Check for an existing ManifestPlugin instance in config.plugins.
-		const hasManifestPlugin = plugins.findExistingInstance( config.plugins, ManifestPlugin );
-		// Add a manifest with the inferred publicPath if none was present.
-		if ( ! hasManifestPlugin ) {
-			const outputPath = ( config.output && config.output.path ) || devDefaults.output.path;
-			devDefaults.plugins.push( plugins.manifest( {
-				fileName: 'asset-manifest.json',
-				seed: getSeedByDirectory( outputPath ),
-			} ) );
-		}
+	// Check for an existing ManifestPlugin instance in config.plugins.
+	// Inject a ManifestPlugin instance if none is present to ensure generated
+	// files can be located by consuming aplications. Any inferred values will
+	// still be overridden with their relevant values from `config`, if provided.
+	const hasManifestPlugin = plugins.findExistingInstance( config.plugins, ManifestPlugin );
+	// Add a manifest if none was present.
+	if ( ! hasManifestPlugin ) {
+		/* eslint-disable function-paren-newline */
+		devDefaults.plugins.push( plugins.manifest(
+			/**
+			 * Filter the full stylesheet loader definition for this preset.
+			 *
+			 * By default parses styles using Sass and then PostCSS.
+			 *
+			 * @hook presets/manifest-options
+			 * @param {Object} options     Manifest plugin options object.
+			 * @param {string} environment "development" or "production".
+			 * @param {Object} config      Preset configuration object.
+			 */
+			applyFilters(
+				'presets/manifest-options',
+				{
+					fileName: 'development-asset-manifest.json',
+					seed: getSeedByDirectory( outputPath ),
+				},
+				'development',
+				config
+			)
+		) );
+		/* eslint-enable */
 	}
 
 	return deepMerge( devDefaults, config );
@@ -243,17 +279,10 @@ const development = ( config = {}, options = {} ) => {
  * merges specified options into an opinionated default production configuration
  * template.
  *
- * @param {webpack.Configuration} config                  Configuration options to deeply merge into the defaults.
- * @param {Object}                [options]               Optional options to modify configuration generation.
- * @param {Function}              [options.filterLoaders] An optional filter function that receives each
- *                                                        computed loader definition and the name of that
- *                                                        loader as it is generated, to permit per-config
- *                                                        customization of loader options.
+ * @param {webpack.Configuration} config Configuration options to deeply merge into the defaults.
  * @returns {webpack.Configuration} A merged Webpack configuration object.
  */
-const production = ( config = {}, options = {} ) => {
-	const { filterLoaders, getFilteredLoader } = createFilteringHelpers( options.filterLoaders );
-
+const production = ( config = {} ) => {
 	// Determine whether source maps have been requested, and prepare an options
 	// object to be passed to all CSS loaders to honor that request.
 	const cssOptions = config.devtool ?
@@ -281,74 +310,91 @@ const production = ( config = {}, options = {} ) => {
 		// Inject a default entry point later on if none was specified.
 
 		output: {
+			// Webpack 5 defaults "publicPath" to "auto," which we are not set up to handle.
+			publicPath: '',
 			// Provide a default output path.
 			path: filePath( 'build' ),
 			pathinfo: false,
 			// Provide a default output name.
-			filename: '[name].js',
+			filename: '[name].[contenthash].js',
 			// Provide chunk filename. Requires content hash for cache busting.
 			chunkFilename: '[name].[contenthash].chunk.js',
 		},
 
 		module: {
 			strictExportPresence: true,
-			rules: [
-				// Run all JS files through ESLint, if installed.
-				...ifInstalled( 'eslint', getFilteredLoader( 'eslint' ) ),
+			rules: removeNullLoaders( [
 				{
 					// "oneOf" will traverse all following loaders until one will
 					// match the requirements. If no loader matches, it will fall
-					// back to the "file" loader at the end of the loader list.
+					// back to the resource loader at the end of the loader list.
 					oneOf: [
 						// Enable processing TypeScript, if installed.
-						...ifInstalled( 'typescript', getFilteredLoader( 'ts' ) ),
+						...ifInstalled( 'typescript', loaders.ts( {}, config ) ),
 						// Process JS with Babel.
-						getFilteredLoader( 'js' ),
-						// Convert small files to data URIs.
-						getFilteredLoader( 'url' ),
-						// Parse styles using SASS, then PostCSS.
-						filterLoaders( {
-							test: /\.s?css$/,
-							use: [
-								// Extract CSS to its own file.
-								MiniCssExtractPlugin.loader,
-								// Process SASS into CSS.
-								getFilteredLoader( 'css', cssOptions ),
-								getFilteredLoader( 'postcss', cssOptions ),
-								getFilteredLoader( 'sass', cssOptions ),
-							],
-						}, 'stylesheet' ),
-						// "file" loader makes sure any non-matching assets still get served.
-						// When you `import` an asset you get its filename.
-						getFilteredLoader( 'file' ),
+						loaders.js( {}, config ),
+						// Handle static asset files.
+						loaders.asset( {}, config ),
+						/**
+						 * Filter the full stylesheet loader definition for this preset.
+						 *
+						 * By default parses styles using Sass and then PostCSS.
+						 *
+						 * @hook presets/stylesheet-loaders
+						 * @param {Object} loader      Stylesheet loader rule.
+						 * @param {string} environment "development" or "production".
+						 * @param {Object} config      Preset configuration object.
+						 */
+						applyFilters(
+							'presets/stylesheet-loaders',
+							{
+								test: /\.s?css$/,
+								use: [
+									// Extract CSS to its own file.
+									MiniCssExtractPlugin.loader,
+									// Process SASS into CSS.
+									loaders.css( cssOptions, config ),
+									loaders.postcss( cssOptions, config ),
+									loaders.sass( cssOptions, config ),
+								],
+							},
+							'production',
+							config
+						),
+						// Resource loader makes sure any non-matching assets still get served.
+						// When you `import` an asset, you get its (virtual) filename.
+						loaders.resource( {}, config ),
 					],
 				},
-			],
+			] ),
 		},
 
 		optimization: {
+			minimize: true,
 			minimizer: [
 				plugins.terser(),
-				plugins.optimizeCssAssets( (
-					// Set option to output source maps if devtool is set.
-					config.devtool && ! ( /inline-/ ).test( config.devtool ) ?
-						{
-							cssProcessorOptions: {
-								map: {
-									inline: false,
-								},
-							},
-						} :
-						undefined
-				) ),
+				plugins.cssMinimizer(),
 			],
-			nodeEnv: 'production',
-			noEmitOnErrors: true,
+			emitOnErrors: false,
 		},
 
 		stats,
 
-		plugins: [],
+		plugins: [
+			// Run all JS files through ESLint, if installed.
+			...ifInstalled( 'eslint', plugins.eslint() ),
+			// Use the simple build report plugin to clean up Webpack's terminal output.
+			plugins.simpleBuildReport(),
+			// If webpack was invoked with the --analyze flag, include a bundleAnalyzer
+			// in production builds. Use the configuration name when present to separate
+			// output files for each webpack build in multi-configuration setups.
+			isAnalyzeMode ?
+				plugins.bundleAnalyzer( {
+					reportFilename: config.name ? `${ config.name }-analyzer-report.html` : 'bundle-analyzer-report.html',
+					statsFilename: config.name ? `${ config.name }-stats.json` : 'stats.json',
+				} ) :
+				null,
+		].filter( Boolean ),
 	};
 
 	// If no entry was provided, inject a default entry value.
@@ -369,10 +415,29 @@ const production = ( config = {}, options = {} ) => {
 	// Add a manifest with the inferred publicPath if none was present.
 	if ( ! hasManifestPlugin ) {
 		const outputPath = ( config.output && config.output.path ) || prodDefaults.output.path;
-		prodDefaults.plugins.push( plugins.manifest( {
-			fileName: 'production-asset-manifest.json',
-			seed: getSeedByDirectory( outputPath ),
-		} ) );
+		/* eslint-disable function-paren-newline */
+		prodDefaults.plugins.push( plugins.manifest(
+			/**
+			 * Filter the full stylesheet loader definition for this preset.
+			 *
+			 * By default parses styles using Sass and then PostCSS.
+			 *
+			 * @hook presets/manifest-options
+			 * @param {Object} options     Manifest plugin options object.
+			 * @param {string} environment "development" or "production".
+			 * @param {Object} config      Preset configuration object.
+			 */
+			applyFilters(
+				'presets/manifest-options',
+				{
+					fileName: 'production-asset-manifest.json',
+					seed: getSeedByDirectory( outputPath ),
+				},
+				'production',
+				config
+			)
+		) );
+		/* eslint-enable */
 	}
 
 	return deepMerge( prodDefaults, config );
diff --git a/src/presets.test.js b/src/presets.test.js
index b623b2a..e1821b4 100644
--- a/src/presets.test.js
+++ b/src/presets.test.js
@@ -3,8 +3,20 @@ const {
 	production,
 } = require( './presets' );
 const plugins = require( './plugins' );
+const { addFilter, setupRegistry } = require( './helpers/filters' );
+const { resetPublicPathsCache } = require( './helpers/infer-public-path' );
 
-jest.mock( 'process', () => ( { cwd: () => 'cwd' } ) );
+jest.mock( 'process', () => ( {
+	cwd: () => 'cwd',
+	versions: {},
+} ) );
+
+/**
+ * Helper function to return null, used when testing filters.
+ *
+ * @returns {null} Null.
+ */
+const returnNull = () => null;
 
 /**
  * Filter an array of plugins to only contain plugins of the provided type.
@@ -26,6 +38,10 @@ const getLoaderByName = ( rules, loaderType ) => {
 		if ( rule.loader && rule.loader.indexOf( loaderType ) > -1 ) {
 			return rule;
 		}
+		// Permit selection of asset module loaders with no .loader property.
+		if ( rule.type && rule.type === loaderType ) {
+			return rule;
+		}
 		if ( rule.oneOf ) {
 			const nestedMatch = getLoaderByName( rule.oneOf, loaderType );
 			if ( nestedMatch ) {
@@ -61,6 +77,11 @@ const getLoaderByTest = ( rules, loaderTest ) => {
 };
 
 describe( 'presets', () => {
+	beforeEach( () => {
+		setupRegistry();
+		resetPublicPathsCache();
+	} );
+
 	describe( 'development()', () => {
 		it( 'is a function', () => {
 			expect( development ).toBeInstanceOf( Function );
@@ -76,8 +97,8 @@ describe( 'presets', () => {
 			expect( config.entry ).toEqual( 'some-file.js' );
 			expect( config.output ).toEqual( {
 				pathinfo: true,
-				filename: '[name].js',
-				chunkFilename: '[name].[contenthash].chunk.js',
+				filename: '[name].[fullhash].js',
+				chunkFilename: '[name].[fullhash].chunk.js',
 				path: 'build/',
 			} );
 		} );
@@ -109,7 +130,18 @@ describe( 'presets', () => {
 			expect( config.output.publicPath ).toBe( 'http://localhost:9090/build/' );
 		} );
 
-		it( 'accounts for the value of devServer.https when inferring publicPath URI', () => {
+		it( 'accounts for the value of devServer.server when inferring publicPath URI', () => {
+			const config = development( {
+				devServer: {
+					server: 'https',
+					port: 9090,
+				},
+				entry: 'some-file.js',
+			} );
+			expect( config.output.publicPath ).toBe( 'https://localhost:9090/build/' );
+		} );
+
+		it( 'accounts for the value of devServer.https (deprecated in favor of .server) when inferring publicPath URI', () => {
 			const config = development( {
 				devServer: {
 					https: true,
@@ -150,7 +182,25 @@ describe( 'presets', () => {
 			expect( config.output.publicPath ).toBe( 'https://my-custom-domain.local/' );
 		} );
 
-		it( 'injects a ManifestPlugin if publicPath can be inferred and no manifest plugin is already present', () => {
+		it( 're-uses previously inferred publicPath URIs in subsequent builds to the same directory', () => {
+			const config1 = development( {
+				devServer: {
+					port: 9090,
+				},
+				output: {
+					path: 'some/folder',
+				}
+			} );
+			const config2 = development( {
+				output: {
+					path: 'some/folder',
+				},
+			} );
+			expect( config1.output.publicPath ).toBe( 'http://localhost:9090/some/folder/' );
+			expect( config1.output.publicPath ).toBe( config2.output.publicPath );
+		} );
+
+		it( 'injects a ManifestPlugin if no manifest plugin is already present', () => {
 			const { ManifestPlugin } = plugins.constructors;
 			const config = development( {
 				devServer: {
@@ -165,19 +215,7 @@ describe( 'presets', () => {
 			] ) );
 			const manifestPlugins = config.plugins.filter( filterPlugins( ManifestPlugin ) );
 			expect( manifestPlugins.length ).toBe( 1 );
-			expect( manifestPlugins[ 0 ].options.fileName ).toEqual( 'asset-manifest.json' );
-		} );
-
-		it( 'does not inject a ManifestPlugin if publicPath cannot be inferred', () => {
-			const config = development( {
-				entry: {
-					main: 'some-file.css',
-				},
-			} );
-			expect( config.output ).not.toHaveProperty( 'publicPath' );
-			expect( config.plugins ).toEqual( expect.not.arrayContaining( [
-				expect.any( plugins.constructors.ManifestPlugin ),
-			] ) );
+			expect( manifestPlugins[ 0 ].options.fileName ).toEqual( 'development-asset-manifest.json' );
 		} );
 
 		it( 'does not override or duplicate existing ManifestPlugin instances', () => {
@@ -252,57 +290,56 @@ describe( 'presets', () => {
 		} );
 
 		it( 'permits filtering the computed output of individual loaders', () => {
+			addFilter( 'loaders/asset', ( loader ) => {
+				loader.test = /\.(png|jpg|jpeg|gif|svg)$/;
+				return loader;
+			} );
+			addFilter( 'loaders/resource', ( loader ) => {
+				loader.options = {
+					publicPath: '../../',
+				};
+				return loader;
+			} );
 			const config = development( {
 				entry: {
 					main: 'some-file.js',
 				},
-			}, {
-				filterLoaders: ( loader, loaderType ) => {
-					if ( loaderType === 'file' ) {
-						loader.options.publicPath = '../../';
-					}
-					if ( loaderType === 'url' ) {
-						loader.test = /\.(png|jpg|jpeg|gif|svg)$/;
-					}
-					return loader;
-				},
 			} );
-			const fileLoader = getLoaderByName( config.module.rules, 'file-loader' );
-			const urlLoader = getLoaderByName( config.module.rules, 'url-loader' );
+			const resourceLoader = getLoaderByName( config.module.rules, 'asset/resource' );
+			const assetLoader = getLoaderByName( config.module.rules, 'asset' );
 			const jsLoader = getLoaderByName( config.module.rules, 'babel-loader' );
-			expect( fileLoader ).toEqual( expect.objectContaining( {
-				exclude: /\.(js|html|json)$/,
+			expect( resourceLoader ).toEqual( expect.objectContaining( {
+				exclude: [ /^$/, /\.(js|mjs|jsx|ts|tsx)$/, /\.html?$/, /\.json$/ ],
+				type: 'asset/resource',
 				options: {
 					publicPath: '../../',
 				},
 			} ) );
-			expect( urlLoader ).toEqual( expect.objectContaining( {
+			expect( assetLoader ).toEqual( expect.objectContaining( {
 				test: /\.(png|jpg|jpeg|gif|svg)$/,
-				options: {
-					limit: 10000,
+				type: 'asset',
+				parser: {
+					dataUrlCondition: {
+						maxSize: 10 * 1024,
+					},
 				},
 			} ) );
 			expect( jsLoader ).not.toBeNull();
 		} );
 
 		it( 'permits filtering the entire stylesheet loader chain', () => {
+			addFilter( 'presets/stylesheet-loaders', ( loader ) => {
+				loader.test = /\.styl$/;
+				return loader;
+			} );
+			addFilter( 'loaders/sass', () => ( {
+				loader: 'stylus',
+				mode: 'development',
+			} ) );
 			const config = development( {
 				entry: {
 					main: 'some-file.js',
 				},
-			}, {
-				filterLoaders: ( loader, loaderType ) => {
-					if ( loaderType === 'stylesheet' ) {
-						loader.test = /\.styl$/;
-					}
-					if ( loaderType === 'sass' ) {
-						return {
-							loader: 'stylus',
-							mode: 'development',
-						};
-					}
-					return loader;
-				},
 			} );
 			const styleChain = getLoaderByTest( config.module.rules, /\.styl$/ );
 			expect( styleChain ).toEqual( {
@@ -317,6 +354,110 @@ describe( 'presets', () => {
 			const sassLoader = getLoaderByTest( config.module.rules, /\.s?css$/ );
 			expect( sassLoader ).toBeNull();
 		} );
+
+		it( 'passes the preset configuration object to the stylesheet chain filter', () => {
+			const config = {
+				entry: {
+					main: 'some-file.js',
+				},
+			};
+			addFilter( 'presets/stylesheet-loaders', ( loader, preset, presetConfig ) => {
+				expect( presetConfig ).toBe( config );
+				expect( preset ).toBe( 'development' );
+				return loader;
+			} );
+			development( config );
+		} );
+
+		it( 'permits skipping a specific stylesheet loader by filtering it to null', () => {
+			addFilter( 'loaders/postcss', returnNull );
+			addFilter( 'loaders/sass', returnNull );
+			const config = development( {
+				entry: {
+					main: 'some-file.js',
+				},
+			} );
+			const styleChain = getLoaderByTest( config.module.rules, /\.s?css$/ );
+			expect( styleChain ).toEqual( {
+				test: /\.s?css$/,
+				use: [
+					{
+						loader: require.resolve( 'style-loader' ),
+						options: {},
+					},
+					{
+						loader: require.resolve( 'css-loader' ),
+						options: {
+							importLoaders: 1,
+							sourceMap: true,
+						},
+					},
+				],
+			} );
+		} );
+
+		it( 'does not include null loader entries if a loader was disabled with a filter', () => {
+			addFilter( 'presets/stylesheet-loaders', returnNull );
+			addFilter( 'loaders/ts', returnNull );
+			const config = development( {
+				entry: {
+					main: 'some-file.js',
+				},
+			} );
+			// Should have stripped out the null entries.
+			expect( config.module.rules[ 1 ].oneOf.length ).toBe( 3 );
+			expect( config.module.rules[ 1 ].oneOf[ 0 ] )
+				.toEqual( expect.objectContaining( { test: /\.jsx?$/ } ) );
+			expect( config.module.rules[ 1 ].oneOf[ 1 ] )
+				.toEqual( expect.objectContaining( { type: 'asset' } ) );
+			expect( config.module.rules[ 1 ].oneOf[ 2 ] )
+				.toEqual( expect.objectContaining( { type: 'asset/resource' } ) );
+		} );
+
+		it( 'passes the preset config as argument 2 to loader filters', () => {
+			addFilter( 'loaders/js', ( options, config ) => {
+				expect( config ).not.toBeNull();
+				expect( config.name ).toBe( 'dev-build' );
+				return options;
+			} );
+			development( {
+				name: 'dev-build',
+			} );
+		} );
+
+		it( 'permits filtering only a specific invocation of a preset', () => {
+			addFilter( 'presets/stylesheet-loaders', returnNull );
+			addFilter( 'loaders/ts', returnNull );
+
+			const filterToNullInBuild1 = ( options, config ) => {
+				if ( config.name === 'build1' ) {
+					return null;
+				}
+				// Simplified version to make test easier.
+				return options.test ?
+					{ test: options.test } :
+					{ type: options.type };
+			};
+			addFilter( 'loaders/js', filterToNullInBuild1 );
+			addFilter( 'loaders/asset', filterToNullInBuild1 );
+			addFilter( 'loaders/resource', filterToNullInBuild1 );
+			addFilter( 'loaders/sourcemap', filterToNullInBuild1 );
+
+			const config1 = development( { name: 'build1' } );
+			const config2 = development( { name: 'build2' } );
+
+			expect( config1.module.rules ).toEqual( [] );
+			expect( config2.module.rules ).toEqual( [
+				{ test: /\.(js|mjs|jsx|ts|tsx|css)$/ },
+				{
+					oneOf: [
+						{ test: /\.jsx?$/ },
+						{ test: /\.(png|jpg|jpeg|gif|avif|webp|svg|woff|woff2|eot|ttf)$/ },
+						{ type: 'asset/resource' },
+					],
+				},
+			] );
+		} );
 	} );
 
 	describe( 'production()', () => {
@@ -334,9 +475,10 @@ describe( 'presets', () => {
 			expect( config.entry ).toEqual( 'some-file.js' );
 			expect( config.output ).toEqual( {
 				pathinfo: false,
-				filename: '[name].js',
+				filename: '[name].[contenthash].js',
 				chunkFilename: '[name].[contenthash].chunk.js',
 				path: 'build/',
+				publicPath: '',
 			} );
 		} );
 
@@ -461,56 +603,55 @@ describe( 'presets', () => {
 		} );
 
 		it( 'permits filtering the computed output of individual loaders', () => {
+			addFilter( 'loaders/asset', ( loader ) => {
+				loader.test = /\.(png|jpg|jpeg|gif|svg)$/;
+				return loader;
+			} );
+			addFilter( 'loaders/resource', ( loader ) => {
+				loader.options = {
+					publicPath: '../../',
+				};
+				return loader;
+			} );
 			const config = production( {
 				entry: {
 					main: 'some-file.js',
 				},
-			}, {
-				filterLoaders: ( loader, loaderType ) => {
-					if ( loaderType === 'file' ) {
-						loader.options.publicPath = '../../';
-					}
-					if ( loaderType === 'url' ) {
-						loader.test = /\.(png|jpg|jpeg|gif|svg)$/;
-					}
-					return loader;
-				},
 			} );
-			const fileLoader = getLoaderByName( config.module.rules, 'file-loader' );
-			const urlLoader = getLoaderByName( config.module.rules, 'url-loader' );
+			const resourceLoader = getLoaderByName( config.module.rules, 'asset/resource' );
+			const assetLoader = getLoaderByName( config.module.rules, 'asset' );
 			const jsLoader = getLoaderByName( config.module.rules, 'babel-loader' );
-			expect( fileLoader ).toEqual( expect.objectContaining( {
-				exclude: /\.(js|html|json)$/,
+			expect( resourceLoader ).toEqual( expect.objectContaining( {
+				exclude: [ /^$/, /\.(js|mjs|jsx|ts|tsx)$/, /\.html?$/, /\.json$/ ],
+				type: 'asset/resource',
 				options: {
 					publicPath: '../../',
 				},
 			} ) );
-			expect( urlLoader ).toEqual( expect.objectContaining( {
+			expect( assetLoader ).toEqual( expect.objectContaining( {
 				test: /\.(png|jpg|jpeg|gif|svg)$/,
-				options: {
-					limit: 10000,
+				type: 'asset',
+				parser: {
+					dataUrlCondition: {
+						maxSize: 10 * 1024,
+					},
 				},
 			} ) );
 			expect( jsLoader ).not.toBeNull();
 		} );
 
 		it( 'permits filtering the entire stylesheet loader chain', () => {
+			addFilter( 'presets/stylesheet-loaders', ( loader ) => {
+				loader.test = /\.styl$/;
+				return loader;
+			} );
+			addFilter( 'loaders/sass', () => ( {
+				loader: 'stylus',
+			} ) );
 			const config = production( {
 				entry: {
 					main: 'some-file.js',
 				},
-			}, {
-				filterLoaders: ( loader, loaderType ) => {
-					if ( loaderType === 'stylesheet' ) {
-						loader.test = /\.styl$/;
-					}
-					if ( loaderType === 'sass' ) {
-						return {
-							loader: 'stylus',
-						};
-					}
-					return loader;
-				},
 			} );
 			const styleChain = getLoaderByTest( config.module.rules, /\.styl$/ );
 			expect( styleChain ).toEqual( {
@@ -524,5 +665,103 @@ describe( 'presets', () => {
 			const sassLoader = getLoaderByTest( config.module.rules, /\.s?css$/ );
 			expect( sassLoader ).toBeNull();
 		} );
+
+		it( 'passes the preset configuration object to the stylesheet chain filter', () => {
+			const config = {
+				entry: {
+					main: 'some-file.js',
+				},
+			};
+			addFilter( 'presets/stylesheet-loaders', ( loader, preset, presetConfig ) => {
+				expect( presetConfig ).toBe( config );
+				expect( preset ).toBe( 'production' );
+				return loader;
+			} );
+			production( config );
+		} );
+
+		it( 'permits skipping a specific stylesheet loader by filtering it to null', () => {
+			addFilter( 'loaders/postcss', returnNull );
+			addFilter( 'loaders/sass', returnNull );
+			const config = production( {
+				entry: {
+					main: 'some-file.js',
+				},
+			} );
+			const styleChain = getLoaderByTest( config.module.rules, /\.s?css$/ );
+			expect( styleChain ).toEqual( {
+				test: /\.s?css$/,
+				use: [
+					plugins.constructors.MiniCssExtractPlugin.loader,
+					{
+						loader: require.resolve( 'css-loader' ),
+						options: {
+							importLoaders: 1,
+						},
+					},
+				],
+			} );
+		} );
+
+		it( 'does not include null loader entries if a loader was disabled with a filter', () => {
+			addFilter( 'presets/stylesheet-loaders', returnNull );
+			addFilter( 'loaders/ts', returnNull );
+			const config = production( {
+				entry: {
+					main: 'some-file.js',
+				},
+			} );
+			// Should have stripped out the null entries.
+			expect( config.module.rules[ 0 ].oneOf.length ).toBe( 3 );
+			expect( config.module.rules[ 0 ].oneOf[ 0 ] )
+				.toEqual( expect.objectContaining( { test: /\.jsx?$/ } ) );
+			expect( config.module.rules[ 0 ].oneOf[ 1 ] )
+				.toEqual( expect.objectContaining( { type: 'asset' } ) );
+			expect( config.module.rules[ 0 ].oneOf[ 2 ] )
+				.toEqual( expect.objectContaining( { type: 'asset/resource' } ) );
+		} );
+	} );
+
+	it( 'passes the preset config as argument 2 to loader filters', () => {
+		addFilter( 'loaders/js', ( options, config ) => {
+			expect( config ).not.toBeNull();
+			expect( config.name ).toBe( 'dev-build' );
+			return options;
+		} );
+		production( {
+			name: 'dev-build',
+		} );
+	} );
+
+	it( 'permits filtering only a specific invocation of a preset', () => {
+		addFilter( 'presets/stylesheet-loaders', returnNull );
+		addFilter( 'loaders/ts', returnNull );
+
+		const filterToNullInBuild1 = ( options, config ) => {
+			if ( config.name === 'build1' ) {
+				return null;
+			}
+			// Simplified version to make test easier.
+			return options.test ?
+				{ test: options.test } :
+				{ type: options.type };
+		};
+		addFilter( 'loaders/js', filterToNullInBuild1 );
+		addFilter( 'loaders/asset', filterToNullInBuild1 );
+		addFilter( 'loaders/resource', filterToNullInBuild1 );
+
+		const config1 = production( { name: 'build1' } );
+		const config2 = production( { name: 'build2' } );
+
+		expect( config1.module.rules ).toEqual( [] );
+		expect( config2.module.rules ).toEqual( [
+			{
+				oneOf: [
+					{ test: /\.jsx?$/ },
+					{ test: /\.(png|jpg|jpeg|gif|avif|webp|svg|woff|woff2|eot|ttf)$/ },
+					{ type: 'asset/resource' },
+				],
+			},
+		] );
 	} );
 } );
diff --git a/test/src/.eslintrc.js b/test/src/.eslintrc.js
new file mode 100644
index 0000000..de7ac3d
--- /dev/null
+++ b/test/src/.eslintrc.js
@@ -0,0 +1,5 @@
+module.exports = {
+	parserOptions: {
+		sourceType: 'module',
+	},
+};
diff --git a/test/src/helpers.js b/test/src/helpers.js
index 1c0daa0..393fd0b 100644
--- a/test/src/helpers.js
+++ b/test/src/helpers.js
@@ -4,3 +4,7 @@ export const getResults = async () => {
 	await wait( 500 );
 	return 'Results';
 };
+
+export const lintingErrorsInProdButWarnsInDev = () => {
+	console.log( 'Very contrived example to hopefuly make test-build output clearer' );
+};
diff --git a/test/src/linting-test.js b/test/src/linting-test.js
new file mode 100644
index 0000000..6495acc
--- /dev/null
+++ b/test/src/linting-test.js
@@ -0,0 +1 @@
+import { lintingErrorsInProdButWarnsInDev } from './helpers';
diff --git a/test/test-config.js b/test/test-config.js
index f196d43..7abf56d 100644
--- a/test/test-config.js
+++ b/test/test-config.js
@@ -7,6 +7,9 @@ module.exports = [
 		name: 'production-test',
 		entry: {
 			prod: filePath( 'test/src/index.js' ),
+			// Expect this not to build. Currently cannot do this because our CI script
+			// anticipates that all commands will return a 0 status.
+			// 'linting-test': filePath( 'test/src/linting-test.js' ),
 		},
 		output: {
 			path: filePath( 'test/build/prod' ),
@@ -42,6 +45,8 @@ module.exports = [
 		name: 'dev-test',
 		entry: {
 			prod: filePath( 'test/src/index.js' ),
+			// Expect this to warn.
+			'linting-test': filePath( 'test/src/linting-test.js' ),
 		},
 		output: {
 			path: filePath( 'test/build/dev' ),