Skip to content

Commit 60178ac

Browse files
andreypoppgaearon
authored andcommitted
Load favicon through html-loader. (facebook#428)
* Load favicon through html-loader. Fixes facebook#291. * Add test for *.ico in e2e test suite * Configure html-loader to process <link href="..."> * Address feedback on html-loader inclusion. * Place favicon.ico at the root of the build dir * Make comment style consistent between prod and dev webpack configs * Fix html-loader config in dev mode
1 parent 25a0b66 commit 60178ac

File tree

7 files changed

+49
-11
lines changed

7 files changed

+49
-11
lines changed

config/paths.js

-3
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,6 @@ if (isInCreateReactAppSource) {
3636
module.exports = {
3737
appBuild: resolveOwn('../build'),
3838
appHtml: resolveOwn('../template/index.html'),
39-
appFavicon: resolveOwn('../template/favicon.ico'),
4039
appPackageJson: resolveOwn('../package.json'),
4140
appSrc: resolveOwn('../template/src'),
4241
appNodeModules: resolveOwn('../node_modules'),
@@ -47,7 +46,6 @@ if (isInCreateReactAppSource) {
4746
module.exports = {
4847
appBuild: resolveApp('build'),
4948
appHtml: resolveApp('index.html'),
50-
appFavicon: resolveApp('favicon.ico'),
5149
appPackageJson: resolveApp('package.json'),
5250
appSrc: resolveApp('src'),
5351
appNodeModules: resolveApp('node_modules'),
@@ -59,7 +57,6 @@ if (isInCreateReactAppSource) {
5957
module.exports = {
6058
appBuild: resolveApp('build'),
6159
appHtml: resolveApp('index.html'),
62-
appFavicon: resolveApp('favicon.ico'),
6360
appPackageJson: resolveApp('package.json'),
6461
appSrc: resolveApp('src'),
6562
appNodeModules: resolveApp('node_modules'),

config/webpack.config.dev.js

+20-2
Original file line numberDiff line numberDiff line change
@@ -126,13 +126,23 @@ module.exports = {
126126
// When you `import` an asset, you get its (virtual) filename.
127127
// In production, they would get copied to the `build` folder.
128128
{
129-
test: /\.(jpg|png|gif|eot|svg|ttf|woff|woff2)(\?.*)?$/,
129+
test: /\.(ico|jpg|png|gif|eot|svg|ttf|woff|woff2)(\?.*)?$/,
130130
include: [paths.appSrc, paths.appNodeModules],
131+
exclude: /\/favicon.ico$/,
131132
loader: 'file',
132133
query: {
133134
name: 'static/media/[name].[ext]'
134135
}
135136
},
137+
// A special case for favicon.ico to place it into build root directory.
138+
{
139+
test: /\/favicon.ico$/,
140+
include: [paths.appSrc],
141+
loader: 'file',
142+
query: {
143+
name: 'favicon.ico?[hash:8]'
144+
}
145+
},
136146
// "url" loader works just like "file" loader but it also embeds
137147
// assets smaller than specified size as data URLs to avoid requests.
138148
{
@@ -143,6 +153,15 @@ module.exports = {
143153
limit: 10000,
144154
name: 'static/media/[name].[ext]'
145155
}
156+
},
157+
// "html" loader is used to process template page (index.html) to resolve
158+
// resources linked with <link href="./relative/path"> HTML tags.
159+
{
160+
test: /\.html$/,
161+
loader: 'html',
162+
query: {
163+
attrs: ['link:href'],
164+
}
146165
}
147166
]
148167
},
@@ -169,7 +188,6 @@ module.exports = {
169188
new HtmlWebpackPlugin({
170189
inject: true,
171190
template: paths.appHtml,
172-
favicon: paths.appFavicon,
173191
}),
174192
// Makes some environment variables available to the JS code, for example:
175193
// if (process.env.NODE_ENV === 'development') { ... }. See `env.js`.

config/webpack.config.prod.js

+24-6
Original file line numberDiff line numberDiff line change
@@ -126,23 +126,33 @@ module.exports = {
126126
loader: ExtractTextPlugin.extract('style', 'css?-autoprefixer!postcss')
127127
// Note: this won't work without `new ExtractTextPlugin()` in `plugins`.
128128
},
129+
// JSON is not enabled by default in Webpack but both Node and Browserify
130+
// allow it implicitly so we also enable it.
129131
{
130-
// JSON is not enabled by default in Webpack but both Node and Browserify
131-
// allow it implicitly so we also enable it.
132132
test: /\.json$/,
133133
include: [paths.appSrc, paths.appNodeModules],
134134
loader: 'json'
135135
},
136+
// "file" loader makes sure those assets end up in the `build` folder.
137+
// When you `import` an asset, you get its filename.
136138
{
137-
// "file" loader makes sure those assets end up in the `build` folder.
138-
// When you `import` an asset, you get its filename.
139-
test: /\.(jpg|png|gif|eot|svg|ttf|woff|woff2)(\?.*)?$/,
139+
test: /\.(ico|jpg|png|gif|eot|svg|ttf|woff|woff2)(\?.*)?$/,
140+
exclude: /\/favicon.ico$/,
140141
include: [paths.appSrc, paths.appNodeModules],
141142
loader: 'file',
142143
query: {
143144
name: 'static/media/[name].[hash:8].[ext]'
144145
}
145146
},
147+
// A special case for favicon.ico to place it into build root directory.
148+
{
149+
test: /\/favicon.ico$/,
150+
include: [paths.appSrc],
151+
loader: 'file',
152+
query: {
153+
name: 'favicon.ico?[hash:8]'
154+
}
155+
},
146156
// "url" loader works just like "file" loader but it also embeds
147157
// assets smaller than specified size as data URLs to avoid requests.
148158
{
@@ -153,6 +163,15 @@ module.exports = {
153163
limit: 10000,
154164
name: 'static/media/[name].[hash:8].[ext]'
155165
}
166+
},
167+
// "html" loader is used to process template page (index.html) to resolve
168+
// resources linked with <link href="./relative/path"> HTML tags.
169+
{
170+
test: /\.html$/,
171+
loader: 'html',
172+
query: {
173+
attrs: ['link:href'],
174+
}
156175
}
157176
]
158177
},
@@ -181,7 +200,6 @@ module.exports = {
181200
new HtmlWebpackPlugin({
182201
inject: true,
183202
template: paths.appHtml,
184-
favicon: paths.appFavicon,
185203
minify: {
186204
removeComments: true,
187205
collapseWhitespace: true,

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,7 @@
6161
"filesize": "3.3.0",
6262
"fs-extra": "0.30.0",
6363
"gzip-size": "3.0.0",
64+
"html-loader": "0.4.3",
6465
"html-webpack-plugin": "2.22.0",
6566
"http-proxy-middleware": "0.17.0",
6667
"jest": "14.1.0",

tasks/e2e.sh

+3
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,7 @@ test -e build/*.html
6767
test -e build/static/js/*.js
6868
test -e build/static/css/*.css
6969
test -e build/static/media/*.svg
70+
test -e build/favicon.ico
7071

7172
# Run tests
7273
npm run test
@@ -96,6 +97,7 @@ test -e build/*.html
9697
test -e build/static/js/*.js
9798
test -e build/static/css/*.css
9899
test -e build/static/media/*.svg
100+
test -e build/favicon.ico
99101

100102
# Run tests
101103
npm run test
@@ -113,6 +115,7 @@ test -e build/*.html
113115
test -e build/static/js/*.js
114116
test -e build/static/css/*.css
115117
test -e build/static/media/*.svg
118+
test -e build/favicon.ico
116119

117120
# Run tests
118121
npm run test

template/index.html

+1
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
<head>
44
<meta charset="utf-8">
55
<meta name="viewport" content="width=device-width, initial-scale=1">
6+
<link rel="shortcut icon" href="./src/favicon.ico">
67
<title>React App</title>
78
</head>
89
<body>
File renamed without changes.

0 commit comments

Comments
 (0)