Подтвердить что ты не робот

Динамически импортировать изображения из каталога с помощью webpack

Итак, вот мой текущий рабочий процесс для импорта изображений и значков в webpack через ES6:

import cat from './images/cat1.jpg'
import cat2 from './images/cat2.svg'
import doggy from './images/doggy.png'
import turtle from './images/turtle.png'

<img src={doggy} />

Это быстро становится беспорядочным. Вот что я хочу:

import * from './images'

<img src={doggy} />
<img src={turtle} />

Я чувствую, что должен быть какой-то способ динамически импортировать все файлы из определенного каталога в качестве имени без расширения, а затем использовать эти файлы по мере необходимости.

Кто-нибудь видел это, или есть какие-то мысли о том, как лучше всего это сделать?


UPDATE:

Используя выбранный ответ, я смог сделать это:

function importAll(r) {
  let images = {};
  r.keys().map((item, index) => { images[item.replace('./', '')] = r(item); });
  return images;
}

const images = importAll(require.context('./images', false, /\.(png|jpe?g|svg)$/));

<img src={images['doggy.png']} />
4b9b3361

Ответ 1

Я чувствую, что должен быть какой-то способ динамически импортировать все файлы из определенного каталога в качестве имени без расширения, а затем использовать эти файлы по мере необходимости.

Не в ES6. Вся точка import и export заключается в том, что зависимости могут быть определены статически, то есть без выполнения кода.

Но так как вы используете webpack, посмотрите require.context. Вы должны быть в состоянии сделать следующее:

function importAll(r) {
  return r.keys().map(r);
}

const images = importAll(require.context('./', false, /\.(png|jpe?g|svg)$/));

Ответ 2

Это легко. Вы можете использовать require (статический метод, импорт - только для динамических файлов) внутри render. Как показано ниже:

render() {
    const {
      someProp,
    } = this.props

    const graphImage = require('./graph-' + anyVariable + '.png')
    const tableImage = require('./table-' + anyVariable2 + '.png')

    return (
    <img src={graphImage}/>
    )
}

Ответ 3

UPDATE Похоже, я не совсем понял вопрос. @Феликс понял это правильно, поэтому проверьте его ответ. Следующий код будет работать только в среде Nodejs.

Добавьте index.js файл в папку images

const testFolder = './';
const fs = require('fs');
const path = require('path')

const allowedExts = [
  '.png' // add any extensions you need
]

const modules = {};

const files = fs.readdirSync(testFolder);

if (files && files.length) {
  files
    .filter(file => allowedExts.indexOf(path.extname(file)) > -1)
    .forEach(file => exports[path.basename(file, path.extname(file))] = require(`./${file}`));
}

module.exports = modules;

Это позволит вам импортировать все из другого файла, а Wepback будет анализировать его и загружать необходимые файлы.

Ответ 4

У меня есть каталог флагов страны png, названный как au.png, nl.png и т.д. Итак, у меня есть:

-svg-country-flags
 --png100px
   ---au.png
   ---au.png
 --index.js
 --CountryFlagByCode.js

index.js

const context = require.context('./png100px', true, /.png$/);

const obj = {};
context.keys().forEach((key) => {
  const countryCode = key.split('./').pop() // remove the first 2 characters
    .substring(0, key.length - 6); // remove the file extension
  obj[countryCode] = context(key);
});

export default obj;

Я прочитал такой файл:

CountryFlagByCode.js

import React from 'react';
import countryFlags from './index';

const CountryFlagByCode = (countryCode) => {
    return (
        <div>
          <img src={countryFlags[countryCode.toLowerCase()]} alt="country_flag" />
        </div>
      );
    };

export default CountryFlagByCode;

Ответ 5

Функциональный подход к решению этой проблемы:

const importAll = require =>
  require.keys().reduce((acc, next) => {
    acc[next.replace("./", "")] = require(next);
    return acc;
  }, {});

const images = importAll(
  require.context("./image", false, /\.(png|jpe?g|svg)$/)
);