2017-01-26 7 views
42

У меня возникла проблема с написанием плагина Webpack для службы перевода.Как написать плагин Webpack, который добавляет модули в пакет на лету на основе других модулей?

Цель состоит в том, чтобы:

  1. Получить имена (и исходный код) всех необходимых модулей во время компиляции. Мне нужно иметь возможность сканировать прилагаемый исходный код для специального использования функции t(), но я хочу сканировать только те модули, которые будут включены в комплект (который, в зависимости от конфигурации сборки, может быть подмножеством всех модулей проекта).
  2. На основе собранных модулей я хочу создать дополнительные модули (с переводами) на лету и добавить их в комплект. Эти модули должны иметь возможность импортировать свои собственные зависимости.

Дополнительным требованием является функция разделения кода на WebPack должен работать с модулями, созданных на лету (я хочу, чтобы извлечь их в отдельные файлы - например bundle.[lang].js). Кроме того, это может быть вне сферы действия этого вопроса, я должен сделать эти куски с необязательными переводами (так что вам не нужно загружать все языки, но только один).

Более подробную информацию можно найти в https://github.com/ckeditor/ckeditor5/issues/387.

Я пытался использовать несколько решений, но документация Webpack 2 не очень полезна. Я могу получить все модули, прослушивая крючки разрешения модуля (before-resolve), но я не знаю, когда разрешены все зависимости, и я не знаю, смогу ли после этого добавлять дополнительные модули (и как это сделать - это addEntry ok и когда я могу его использовать?).

Я также думал о подключении плагина Webpack и загрузчика Webpack (потому что мне нужна функция очень похожа на Webpack's style-loader), но из уровня плагина я могу добавить путь к загрузчику, а не самому загрузчика, поэтому я могу 't передать объект конфигурации в качестве параметра - я ошибаюсь?

PS. Я использую Webpack 2. Если вам кажутся вам странные требования, см. https://github.com/ckeditor/ckeditor5/issues/387 :).

+0

Эта проблема похожа на http://stackoverflow.com/questions/31698254/webpack-dynamically-create-a-module –

+0

И это также связано: http://stackoverflow.com/questions/35092183/webpack- plugin-how-can-i-modify-and-re-parse-a-module-after-compilation –

+0

Если вы говорите «в зависимости от конфигурации сборки», вы подразумеваете, что вы устанавливаете в одном из своих файлов webpack.config? Или он основан исключительно на коде, который он кормил? Кроме того, вы можете создать свой язык. пакет «требует» динамически, я думаю, используя что-то вроде https://www.npmjs.com/package/string-replace-loader – user5328504

ответ

2

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

const CommonJsRequireDependency = require("webpack/lib/dependencies/CommonJsRequireDependency") 

class MyPlugin { 
    apply(compiler) { 
    compiler.plugin("compilation", compilation => { 
     compilation.plugin("succeed-module", module => { 
     // this will be called for every successfully built module, but before it's parsed and 
     // its dependencies are built. The built source is available as module._source.source() 
     // and you can add additional dependencies like so: 
     module.dependencies.push(new CommonJsRequireDependency("my-dependency", null)) 
     } 
    } 
    } 
} 

Это всего лишь одна его часть. Вам также, вероятно, потребуется написать собственный загрузчик, чтобы на самом деле генерировать переводы (вы можете заменить my-dependency выше на my-loader!path/to/module, чтобы вызвать его немедленно) и некоторый шаг после создания кусков, чтобы, возможно, извлечь их в новый актив и загрузить их с тех пор, как они на самом деле не require d в любом месте.