None of the answers on this thread will work on newer Next.js versions (v11.1.3-canary.2
and above), because in PR#28529, the Next.js team has switched to a customized version of css-loader
that has no defaultGetLocalIdent
, nor does it checks if getLocalIndent
is null
or undefined
.
Any solution that ultimately removes getLocalIdent
from the loader configuration will lead to TypeError: getLocalIdent is not a function
. In other words, it is now mandatory to provide a function as getLocalIdent
. Here is an example:
const path = require('path');
const loaderUtils = require('loader-utils');
// based on https://github.com/vercel/next.js/blob/992c46e63bef20d7ab7e40131667ed3debaf67de/packages/next/build/webpack/config/blocks/css/loaders/getCssModuleLocalIdent.ts
const hashOnlyIdent = (context, _, exportName) =>
loaderUtils
.getHashDigest(
Buffer.from(
`filePath:${path
.relative(context.rootContext, context.resourcePath)
.replace(/\\+/g, "https://stackoverflow.com/")}#className:${exportName}`,
),
'md4',
'base64',
6,
)
.replace(/[^a-zA-Z0-9-_]/g, '_')
.replace(/^(-?\d|--)/, '_$1');
module.exports = {
webpack(config, { dev }) {
const rules = config.module.rules
.find((rule) => typeof rule.oneOf === 'object')
.oneOf.filter((rule) => Array.isArray(rule.use));
if (!dev)
rules.forEach((rule) => {
rule.use.forEach((moduleLoader) => {
if (
moduleLoader.loader?.includes('css-loader') &&
!moduleLoader.loader?.includes('postcss-loader')
)
moduleLoader.options.modules.getLocalIdent = hashOnlyIdent;
// earlier below statements were sufficient:
// delete moduleLoader.options.modules.getLocalIdent;
// moduleLoader.options.modules.localIdentName="[hash:base64:6]";
});
});
return config;
},
};
This works on Next v12 and v13 (without TurboPack) too.
Demo:
While development –
On production –