Невозможно импортировать модули CSS/SCSS. TypeScript говорит: "Не удается найти модуль"
Я пытаюсь импортировать тему из модуля CSS, но TypeScript дает мне ошибку "Не удается найти модуль", и тема не применяется во время выполнения. Я думаю, что что-то не так с моей конфигурацией Webpack, но я не уверен, где проблема.
Я использую следующие инструменты:
"typescript": "^2.0.3"
"webpack": "2.1.0-beta.25"
"webpack-dev-server": "^2.1.0-beta.9"
"react": "^15.4.0-rc.4"
"react-toolbox": "^1.2.3"
"node-sass": "^3.10.1"
"style-loader": "^0.13.1"
"css-loader": "^0.25.0"
"sass-loader": "^4.0.2"
"sass-lint": "^1.9.1"
"sasslint-webpack-plugin": "^1.0.4"
Вот мой webpack.config.js
var path = require('path');
var webpack = require('webpack');
var sassLintPlugin = require('sasslint-webpack-plugin');
module.exports = {
entry: [
'webpack-dev-server/client?http://localhost:8080',
'webpack/hot/dev-server',
'./src/index.tsx',
],
output: {
path: path.resolve(__dirname, 'dist'),
publicPath: 'http://localhost:8080/',
filename: 'dist/bundle.js',
},
devtool: 'source-map',
resolve: {
extensions: ['.webpack.js', '.web.js', '.ts', '.tsx', '.js'],
},
module: {
rules: [{
test: /\.js$/,
loader: 'source-map-loader',
exclude: /node_modules/,
enforce: 'pre',
}, {
test: /\.tsx?$/,
loader: 'tslint-loader',
exclude: /node_modules/,
enforce: 'pre',
}, {
test: /\.tsx?$/,
loaders: [
'react-hot-loader/webpack',
'awesome-typescript-loader',
],
exclude: /node_modules/,
}, {
test: /\.scss$/,
loaders: ['style', 'css', 'sass']
}, {
test: /\.css$/,
loaders: ['style', 'css']
}],
},
externals: {
'react': 'React',
'react-dom': 'ReactDOM'
},
plugins: [
new sassLintPlugin({
glob: 'src/**/*.s?(a|c)ss',
ignoreFiles: ['src/normalize.scss'],
failOnWarning: false, // Do it.
}),
new webpack.HotModuleReplacementPlugin(),
],
devServer: {
contentBase: './'
},
};
и мой App.tsx
, где я пытаюсь импортировать:
import * as React from 'react';
import { AppBar } from 'react-toolbox';
import appBarTheme from 'react-toolbox/components/app_bar/theme.scss'
// local ./theme.scss stylesheets aren't found either
interface IAppStateProps {
// No props yet
}
interface IAppDispatchProps {
// No state yet
}
class App extends React.Component<IAppStateProps & IAppDispatchProps, any> {
constructor(props: IAppStateProps & IAppDispatchProps) {
super(props);
}
public render() {
return (
<div className='wrapper'>
<AppBar title='My App Bar' theme={appBarTheme}>
</AppBar>
</div>
);
}
}
export default App;
Что еще необходимо для включения импорта модулей стилей стилей?
Ответы
Ответ 1
TypeScript не знает, что существуют файлы, отличные от .ts
или .tsx
поэтому он выдаст ошибку, если импорт имеет неизвестный суффикс файла.
Если у вас есть конфигурация webpack, которая позволяет вам импортировать другие типы файлов, вы должны сообщить компилятору TypeScript, что эти файлы существуют. Для этого добавьте файл объявлений, в котором вы объявляете модули с подходящими именами.
Содержимое модуля для объявления зависит от загрузчика webpack, используемого для типа файла. В конфигурации веб-пакета, которая *.scss
файлы *.scss
через sass-loader → css-loader → style-loader, в импортированном модуле не будет содержимого, и правильное объявление модуля будет выглядеть так:
// declaration.d.ts
declare module '*.scss';
Если загрузчики настроены для css-модулей, просто расширьте объявление следующим образом:
// declaration.d.ts
declare module '*.scss' {
const content: {[className: string]: string};
export default content;
}
Ответ 2
Вот полная конфигурация, которая работает для меня (я просто потратил час мучительных проб и ошибок на это - на случай, если кто-то столкнется с такими же проблемами):
TypeScript + WebPack + Sass
webpack.config.js
module.exports = {
//mode: "production",
mode: "development", devtool: "inline-source-map",
entry: [ "./src/app.tsx"/*main*/ ],
output: {
filename: "./bundle.js" // in /dist
},
resolve: {
// Add '.ts' and '.tsx' as a resolvable extension.
extensions: [".ts", ".tsx", ".js", ".css", ".scss"]
},
module: {
rules: [
{ test: /\.tsx?$/, loader: "ts-loader" },
{ test: /\.scss$/, use: [
{ loader: "style-loader" }, // to inject the result into the DOM as a style block
{ loader: "css-modules-typescript-loader"}, // to generate a .d.ts module next to the .scss file (also requires a declaration.d.ts with "declare modules '*.scss';" in it to tell TypeScript that "import styles from './styles.scss';" means to load the module "./styles.scss.d.td")
{ loader: "css-loader", options: { modules: true } }, // to convert the resulting CSS to Javascript to be bundled (modules:true to rename CSS classes in output to cryptic identifiers, except if wrapped in a :global(...) pseudo class)
{ loader: "sass-loader" }, // to convert SASS to CSS
// NOTE: The first build after adding/removing/renaming CSS classes fails, since the newly generated .d.ts typescript module is picked up only later
] },
]
}
};
Также поместите файл declarations.d.ts
в ваш проект:
// We need to tell TypeScript that when we write "import styles from './styles.scss' we mean to load a module (to look for a './styles.scss.d.ts').
declare module '*.scss';
И вам все это понадобится в ваших package.json
-зависимостях package.json
:
"devDependencies": {
"@types/node-sass": "^4.11.0",
"node-sass": "^4.12.0",
"css-loader": "^1.0.0",
"css-modules-typescript-loader": "^2.0.1",
"sass-loader": "^7.1.0",
"style-loader": "^0.23.1",
"ts-loader": "^5.3.3",
"typescript": "^3.4.4",
"webpack": "^4.30.0",
"webpack-cli": "^3.3.0"
}
Затем вы должны получить mystyle.d.ts
рядом с вашим mystyle.scss
содержащим определенные вами CSS-классы, которые вы можете импортировать как модуль Typescript и использовать так:
import * as styles from './mystyles.scss';
const foo = <div className={styles.myClass}>FOO</div>;
CSS будет автоматически загружен (введен как элемент style
в DOM) и будет содержать загадочные идентификаторы вместо ваших CSS-классов в .scss, чтобы изолировать ваши стили на странице (если вы не используете :global(.a-global-class) {... }
).
Обратите внимание, что первая компиляция завершится неудачно всякий раз, когда вы добавляете CSS-классы или удаляете их или переименовываете их, поскольку импортированный mystyles.d.ts является старой версией, а не новой версией, сгенерированной только во время компиляции. Просто скомпилируйте снова.
Наслаждаться.