diff --git a/.github/workflows/grumphp.yml b/.github/workflows/grumphp.yml
new file mode 100644
index 0000000..e67f897
--- /dev/null
+++ b/.github/workflows/grumphp.yml
@@ -0,0 +1,28 @@
+on: [push]
+
+jobs:
+ build:
+ name: Running GrumPHP
+ runs-on: ubuntu-latest
+ env:
+ extensions: intl, mbstring
+ steps:
+ - uses: actions/checkout@v2
+ - name: Setup PHP
+ uses: shivammathur/setup-php@v2
+ with:
+ php-version: '7.3'
+ extensions: ${{ env.extensions }}
+ tools: composer
+ - name: Get Composer Cache Directory
+ id: composercache
+ run: echo "::set-output name=dir::$(composer config cache-files-dir)"
+ - name: Cache dependencies
+ uses: actions/cache@v2
+ with:
+ path: ${{ steps.composercache.outputs.dir }}
+ key: ${{ runner.os }}-composer-${{ hashFiles('**/composer.json') }}
+ restore-keys: ${{ runner.os }}-composer-
+ - name: Install dependencies
+ run: composer config "http-basic.repo.magento.com" ${{ secrets.MAGE_USER }} ${{ secrets.MAGE_PASS }} && composer install --prefer-dist
+ - run: ./vendor/bin/grumphp run
\ No newline at end of file
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..eb0a8e7
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,3 @@
+/composer.lock
+/vendor
+/.phpunit.result.cache
diff --git a/.php_cs b/.php_cs
new file mode 100644
index 0000000..cecf418
--- /dev/null
+++ b/.php_cs
@@ -0,0 +1,46 @@
+in('./')
+ ->name(['*.php']);
+
+return Config::create()
+ ->setRules([
+ '@PSR2' => true,
+ 'no_useless_else' => true,
+ 'no_useless_return' => true,
+ 'ordered_class_elements' => true,
+ 'ordered_imports' => true,
+ 'phpdoc_order' => true,
+ 'phpdoc_summary' => false,
+ 'blank_line_after_opening_tag' => false,
+ 'concat_space' => ['spacing' => 'one'],
+ 'array_syntax' => ['syntax' => 'short'],
+ // Removes @param and @return tags that don't provide any useful information;
+ // Set to false to ensure custom Magento API implementations do not fail. Magento uses reflection based on its docblocks to process request/responses
+ 'no_superfluous_phpdoc_tags' => false,
+ // add declare strict type to every file
+ 'declare_strict_types' => true,
+ // use native phpunit methods
+ 'php_unit_construct' => true,
+ // Enforce camel case for PHPUnit test methods
+ 'php_unit_method_casing' => ['case' => 'camel_case'],
+ 'yoda_style' => ['equal' => true, 'identical' => true, 'less_and_greater' => true],
+ 'php_unit_test_case_static_method_calls' => true,
+ // comparisons should always be strict
+ 'strict_comparison' => true,
+ // functions should be used with $strict param set to true
+ 'strict_param' => true,
+ 'array_indentation' => true,
+ 'compact_nullable_typehint' => true,
+ 'fully_qualified_strict_types' => true,
+ ])
+ ->setFinder($finder)
+ ->setRiskyAllowed(true)
+ ->setUsingCache(false);
+
diff --git a/Api/Data/TranslationInterface.php b/Api/Data/TranslationInterface.php
new file mode 100644
index 0000000..172a9f4
--- /dev/null
+++ b/Api/Data/TranslationInterface.php
@@ -0,0 +1,83 @@
+ __('Back'),
+ 'on_click' => sprintf("location.href = '%s';", $this->getBackUrl()),
+ 'class' => 'back',
+ 'sort_order' => 10
+ ];
+ }
+
+ /**
+ * Get URL for back (reset) button
+ *
+ * @return string
+ */
+ public function getBackUrl()
+ {
+ return $this->getUrl('*/*/');
+ }
+}
diff --git a/Block/Adminhtml/Translation/Edit/DeleteButton.php b/Block/Adminhtml/Translation/Edit/DeleteButton.php
new file mode 100644
index 0000000..1267834
--- /dev/null
+++ b/Block/Adminhtml/Translation/Edit/DeleteButton.php
@@ -0,0 +1,39 @@
+getModelId()) {
+ $data = [
+ 'label' => __('Delete Translation'),
+ 'class' => 'delete',
+ 'on_click' => 'deleteConfirm(\'' . __(
+ 'Are you sure you want to do this?'
+ ) . '\', \'' . $this->getDeleteUrl() . '\')',
+ 'sort_order' => 20,
+ ];
+ }
+ return $data;
+ }
+
+ /**
+ * Get URL for delete button
+ *
+ * @return string
+ */
+ public function getDeleteUrl()
+ {
+ return $this->getUrl('*/*/delete', ['key_id' => $this->getModelId()]);
+ }
+}
diff --git a/Block/Adminhtml/Translation/Edit/GenericButton.php b/Block/Adminhtml/Translation/Edit/GenericButton.php
new file mode 100644
index 0000000..f209eae
--- /dev/null
+++ b/Block/Adminhtml/Translation/Edit/GenericButton.php
@@ -0,0 +1,41 @@
+context = $context;
+ }
+
+ /**
+ * Return model ID
+ *
+ * @return int|null
+ */
+ public function getModelId()
+ {
+ return $this->context->getRequest()->getParam('key_id');
+ }
+
+ /**
+ * Generate url by route and parameters
+ *
+ * @param string $route
+ * @param array $params
+ * @return string
+ */
+ public function getUrl($route = '', $params = [])
+ {
+ return $this->context->getUrlBuilder()->getUrl($route, $params);
+ }
+}
diff --git a/Block/Adminhtml/Translation/Edit/SaveAndContinueButton.php b/Block/Adminhtml/Translation/Edit/SaveAndContinueButton.php
new file mode 100644
index 0000000..528e71b
--- /dev/null
+++ b/Block/Adminhtml/Translation/Edit/SaveAndContinueButton.php
@@ -0,0 +1,27 @@
+ __('Save and Continue Edit'),
+ 'class' => 'save',
+ 'data_attribute' => [
+ 'mage-init' => [
+ 'button' => ['event' => 'saveAndContinueEdit'],
+ ],
+ ],
+ 'sort_order' => 80,
+ ];
+ }
+}
diff --git a/Block/Adminhtml/Translation/Edit/SaveButton.php b/Block/Adminhtml/Translation/Edit/SaveButton.php
new file mode 100644
index 0000000..63eeb5a
--- /dev/null
+++ b/Block/Adminhtml/Translation/Edit/SaveButton.php
@@ -0,0 +1,26 @@
+ __('Save Translation'),
+ 'class' => 'save primary',
+ 'data_attribute' => [
+ 'mage-init' => ['button' => ['event' => 'save']],
+ 'form-role' => 'save',
+ ],
+ 'sort_order' => 90,
+ ];
+ }
+}
diff --git a/Block/Adminhtml/Translation/GenerateJson/SaveButton.php b/Block/Adminhtml/Translation/GenerateJson/SaveButton.php
new file mode 100644
index 0000000..bb12336
--- /dev/null
+++ b/Block/Adminhtml/Translation/GenerateJson/SaveButton.php
@@ -0,0 +1,27 @@
+ __('Generate translations files'),
+ 'class' => 'save primary',
+ 'data_attribute' => [
+ 'mage-init' => ['button' => ['event' => 'save']],
+ 'form-role' => 'save',
+ ],
+ 'sort_order' => 90,
+ ];
+ }
+}
diff --git a/CHANGELOG.md b/CHANGELOG.md
new file mode 100644
index 0000000..4c5fa19
--- /dev/null
+++ b/CHANGELOG.md
@@ -0,0 +1,9 @@
+# Changelog
+## [1.0.0] - 2022-01-28
+### Added
+- Simple backend CRUD to manage translations
+- Single and multi-inline editing via the grid-overview
+- Import and export via CSV files via CLI
+- Import and export via default Magento import/export functionality
+- Prepare new translations via data patch scripts
+- (Re)generate frontend translations (JSON translation files) via CLI and backend
\ No newline at end of file
diff --git a/Console/Command/Export.php b/Console/Command/Export.php
new file mode 100644
index 0000000..00be805
--- /dev/null
+++ b/Console/Command/Export.php
@@ -0,0 +1,59 @@
+exporter = $exporter;
+ parent::__construct();
+ }
+
+ protected function configure(): void
+ {
+ $this->setName(self::COMMAND_NAME);
+ $this->setDescription('Export translations from database to CSV.');
+ $this->addArgument(self::ARGUMENT_LOCALES, InputArgument::IS_ARRAY, 'Locales (multiple locales with a space');
+
+ parent::configure();
+ }
+
+ protected function execute(InputInterface $input, OutputInterface $output): void
+ {
+ try {
+ $locales = $input->getArgument(self::ARGUMENT_LOCALES);
+
+ $output->writeln(
+ sprintf(
+ 'Exporting translations to CSV file'
+ )
+ );
+
+ $exportStats = $this->exporter->export($locales);
+
+ $output->writeln('Csv file: ' . $exportStats->getFileName() . '');
+ $output->writeln('Total written rows: ' . $exportStats->getTotalRows() . '');
+
+ $output->writeln('Done!');
+ } catch (\Exception $e) {
+ $output->writeln('' . $e->getMessage() . '');
+ }
+ }
+}
diff --git a/Console/Command/GenerateFrontendTranslations.php b/Console/Command/GenerateFrontendTranslations.php
new file mode 100644
index 0000000..465ee15
--- /dev/null
+++ b/Console/Command/GenerateFrontendTranslations.php
@@ -0,0 +1,66 @@
+inlineTranslationsGenerator = $inlineTranslations;
+ parent::__construct();
+ }
+
+ protected function configure(): void
+ {
+ $this->setName(self::COMMAND_NAME);
+ $this->setDescription('Re-generate the frontend translations (js-translations.json) for given store view.');
+ $this->addArgument('storeId', InputArgument::OPTIONAL, 'Leave empty for all store views', 0);
+ parent::configure();
+ }
+
+ protected function execute(InputInterface $input, OutputInterface $output): void
+ {
+ try {
+ $storeId = (int)$input->getArgument('storeId');
+ $statsCollection = $this->generatedTranslations($storeId);
+
+ $table = new Table($output);
+ $table->setHeaders(['# Translations', 'Store view information', 'Store id']);
+ foreach ($statsCollection as $stats) {
+ $table->addRow([
+ $stats->getAmountGenerated(),
+ $stats->getStoreInformation(),
+ $stats->getStoreId(),
+ ]);
+ }
+ $output->writeln('Inline frontend translations successfully re-generated for:');
+ $table->render();
+ $output->writeln('Please clean full_page and block_html caches manually');
+ } catch (\Exception $e) {
+ $output->writeln('' . $e->getMessage() . '');
+ }
+ }
+
+ private function generatedTranslations(int $storeId): InlineGenerateStatsCollection
+ {
+ if (0 === $storeId) {
+ return $this->inlineTranslationsGenerator->forAll();
+ }
+
+ return $this->inlineTranslationsGenerator->forStores([$storeId]);
+ }
+}
diff --git a/Console/Command/Import.php b/Console/Command/Import.php
new file mode 100644
index 0000000..9ce7f3d
--- /dev/null
+++ b/Console/Command/Import.php
@@ -0,0 +1,81 @@
+importer = $importer;
+ $this->cacheManager = $cacheManager;
+ parent::__construct();
+ }
+
+ protected function configure(): void
+ {
+ $this->setName(self::COMMAND_NAME);
+ $this->setDescription('Import Magento translation CSVs to the database.');
+ $this->addArgument(self::ARGUMENT_CSV_FILE, InputArgument::REQUIRED);
+ $this->addArgument(self::ARGUMENT_LOCALE, InputArgument::REQUIRED);
+ $this->addOption(self::OPTION_CLEAR_CACHE, null, InputOption::VALUE_NONE, 'Clear related caches');
+
+ parent::configure();
+ }
+
+ protected function execute(InputInterface $input, OutputInterface $output): void
+ {
+ try {
+ $csvFile = $input->getArgument(self::ARGUMENT_CSV_FILE);
+ $locale = $input->getArgument(self::ARGUMENT_LOCALE);
+
+ $output->writeln(
+ sprintf(
+ 'Importing CSV file %s for locale %s',
+ $csvFile,
+ $locale
+ )
+ );
+
+ $importStats = $this->importer->importMagentoCsv($csvFile, $locale);
+
+ $output->writeln('#created: ' . $importStats->getCreatedCount() . '');
+ $output->writeln('#skipped: ' . $importStats->getSkippedCount() . '');
+ $output->writeln('#failed: ' . $importStats->getFailedCount() . '');
+
+ if (false !== $input->hasOption(self::OPTION_CLEAR_CACHE)) {
+ $cacheTypes = ['full_page', 'block_html', 'translate'];
+ $this->cacheManager->clean(['full_page', 'block_html', 'translate']);
+ $output->writeln('Caches cleared: ' . implode(', ', $cacheTypes) . '');
+ }
+
+ $output->writeln('Done!');
+ } catch (\Exception $e) {
+ $output->writeln('' . $e->getMessage() . '');
+ }
+ }
+}
diff --git a/Console/Command/ImportFull.php b/Console/Command/ImportFull.php
new file mode 100644
index 0000000..8742587
--- /dev/null
+++ b/Console/Command/ImportFull.php
@@ -0,0 +1,77 @@
+importer = $importer;
+ $this->cacheManager = $cacheManager;
+ parent::__construct();
+ }
+
+ protected function configure(): void
+ {
+ $this->setName(self::COMMAND_NAME);
+ $this->setDescription('Import translation CSVs to the database based on exports.');
+ $this->addArgument(self::ARGUMENT_CSV_FILE, InputArgument::REQUIRED);
+ $this->addOption(self::OPTION_CLEAR_CACHE, null, InputOption::VALUE_NONE, 'Clear related caches');
+
+ parent::configure();
+ }
+
+ protected function execute(InputInterface $input, OutputInterface $output): void
+ {
+ try {
+ $csvFile = $input->getArgument(self::ARGUMENT_CSV_FILE);
+
+ $output->writeln(
+ sprintf(
+ 'Importing CSV file %s',
+ $csvFile
+ )
+ );
+
+ $importStats = $this->importer->importFull($csvFile);
+
+ $output->writeln('#created: ' . $importStats->getCreatedCount() . '');
+ $output->writeln('#skipped: ' . $importStats->getSkippedCount() . '');
+ $output->writeln('#failed: ' . $importStats->getFailedCount() . '');
+
+ if (false !== $input->hasOption(self::OPTION_CLEAR_CACHE)) {
+ $cacheTypes = ['full_page', 'block_html', 'translate'];
+ $this->cacheManager->clean(['full_page', 'block_html', 'translate']);
+ $output->writeln('Caches cleared: ' . implode(', ', $cacheTypes) . '');
+ }
+
+ $output->writeln('Done!');
+ } catch (\Exception $e) {
+ $output->writeln('' . $e->getMessage() . '');
+ }
+ }
+}
diff --git a/Console/Command/PrepareKeysCommand.php b/Console/Command/PrepareKeysCommand.php
new file mode 100644
index 0000000..e0bbbda
--- /dev/null
+++ b/Console/Command/PrepareKeysCommand.php
@@ -0,0 +1,89 @@
+optionResolverFactory = $optionResolverFactory;
+ $this->parser = $parser;
+
+ parent::__construct($name);
+ $this->translationDataManagement = $translationDataManagement;
+ }
+
+ /**
+ * @inheritDoc
+ */
+ protected function configure()
+ {
+ $this->setName('phpro:translations:prepare-keys');
+ $this->setDescription('Prepare translations keys');
+ parent::configure();
+ }
+
+ /**
+ * CLI command description
+ *
+ * @param InputInterface $input
+ * @param OutputInterface $output
+ *
+ * @return void
+ */
+ protected function execute(InputInterface $input, OutputInterface $output): void
+ {
+ $phraseCollector = new PhraseCollector(new Tokenizer());
+ $adapters = [
+ 'php' => new Php($phraseCollector),
+ 'html' => new Html(),
+ 'js' => new Js(),
+ 'xml' => new Xml(),
+ ];
+ $optionResolver = $this->optionResolverFactory->create(BP, false);
+ foreach ($adapters as $type => $adapter) {
+ $this->parser->addAdapter($type, $adapter);
+ }
+ $this->parser->parse($optionResolver->getOptions());
+ $phraseList = $this->parser->getPhrases();
+
+ foreach ($phraseList as $phrase) {
+ $this->translationDataManagement->prepare($phrase->getPhrase(), $phrase->getTranslation());
+ }
+
+ $output->writeln('Keys successfully created.');
+ }
+}
diff --git a/Controller/Adminhtml/Translation.php b/Controller/Adminhtml/Translation.php
new file mode 100644
index 0000000..d965701
--- /dev/null
+++ b/Controller/Adminhtml/Translation.php
@@ -0,0 +1,26 @@
+setActiveMenu(self::ADMIN_RESOURCE)
+ ->addBreadcrumb(__('Phpro'), __('Phpro'))
+ ->addBreadcrumb(__('Translations'), __('Translations'));
+ return $resultPage;
+ }
+}
diff --git a/Controller/Adminhtml/Translation/Delete.php b/Controller/Adminhtml/Translation/Delete.php
new file mode 100644
index 0000000..987bf15
--- /dev/null
+++ b/Controller/Adminhtml/Translation/Delete.php
@@ -0,0 +1,50 @@
+translationRepository = $translationRepository;
+ }
+
+ /**
+ * @inheridoc
+ */
+ public function execute()
+ {
+ /** @var \Magento\Backend\Model\View\Result\Redirect $resultRedirect */
+ $resultRedirect = $this->resultRedirectFactory->create();
+ $id = (int)$this->getRequest()->getParam('key_id');
+ if ($id) {
+ try {
+ $this->translationRepository->deleteById($id);
+ $this->messageManager->addSuccessMessage(__('You deleted the Translation.'));
+ return $resultRedirect->setPath('*/*/');
+ } catch (\Exception $e) {
+ $this->messageManager->addErrorMessage($e->getMessage());
+ return $resultRedirect->setPath('*/*/edit', ['key_id' => $id]);
+ }
+ }
+
+ $this->messageManager->addErrorMessage(__('We can\'t find a Translation to delete.'));
+
+ return $resultRedirect->setPath('*/*/');
+ }
+}
diff --git a/Controller/Adminhtml/Translation/DoGenerate.php b/Controller/Adminhtml/Translation/DoGenerate.php
new file mode 100644
index 0000000..e6ae713
--- /dev/null
+++ b/Controller/Adminhtml/Translation/DoGenerate.php
@@ -0,0 +1,77 @@
+inlineTranslationsGenerator = $inlineTranslationsGenerator;
+ $this->cacheTypeList = $cacheTypeList;
+ parent::__construct($context);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function execute()
+ {
+ /** @var \Magento\Backend\Model\View\Result\Redirect $resultRedirect */
+ $resultRedirect = $this->resultRedirectFactory->create();
+ $data = $this->getRequest()->getPostValue();
+
+ if (!isset($data['storeviews'])) {
+ return $resultRedirect->setPath('*/*/generatejson');
+ }
+
+ $statsCollection = $this->generatedTranslations(
+ array_map(function ($storeId) {
+ return (int)$storeId;
+ }, $data['storeviews'])
+ );
+
+ $this->messageManager->addComplexSuccessMessage(
+ 'inlineGenerateSuccessMessage',
+ [
+ 'statsItems' => $statsCollection->toArray()
+ ]
+ );
+
+ $this->cacheTypeList->invalidate(Type::TYPE_IDENTIFIER);
+ $this->cacheTypeList->invalidate(Block::TYPE_IDENTIFIER);
+ $this->cacheTypeList->invalidate(Translate::TYPE_IDENTIFIER);
+
+ return $resultRedirect->setPath('*/*/generatejson');
+ }
+
+ private function generatedTranslations(array $storeIds): InlineGenerateStatsCollection
+ {
+ if (isset($storeIds[0]) && (0 === $storeIds[0])) {
+ return $this->inlineTranslationsGenerator->forAll();
+ }
+
+ return $this->inlineTranslationsGenerator->forStores($storeIds);
+ }
+}
diff --git a/Controller/Adminhtml/Translation/Edit.php b/Controller/Adminhtml/Translation/Edit.php
new file mode 100644
index 0000000..ee489ad
--- /dev/null
+++ b/Controller/Adminhtml/Translation/Edit.php
@@ -0,0 +1,68 @@
+resultPageFactory = $resultPageFactory;
+ $this->translationRepository = $translationRepository;
+ $this->coreRegistry = $coreRegistry;
+ }
+
+ /**
+ * @inheridoc
+ */
+ public function execute()
+ {
+ try {
+ $translationId = $this->getRequest()->getParam('key_id');
+ $translationDataModel = $this->translationRepository->getById($translationId);
+ $this->coreRegistry->register('phpro_translations_translation', $translationDataModel);
+ $resultPage = $this->resultPageFactory->create();
+
+ $this->initPage($resultPage)->addBreadcrumb(__('Edit Translation'), __('Edit Translation'));
+ $resultPage->getConfig()->getTitle()->prepend(__('Translations'));
+ $resultPage->getConfig()->getTitle()->prepend(__('Edit Translation %1', $translationDataModel->getKeyId()));
+
+ return $resultPage;
+ } catch (LocalizedException $e) {
+ $this->messageManager->addErrorMessage($e->getMessage());
+ } catch (\Exception $e) {
+ $this->messageManager->addExceptionMessage($e, __('Something went wrong during loading translation.'));
+ }
+
+ return $this->resultRedirectFactory->create()->setPath('*/*/');
+ }
+}
diff --git a/Controller/Adminhtml/Translation/GenerateJson.php b/Controller/Adminhtml/Translation/GenerateJson.php
new file mode 100644
index 0000000..e1c7773
--- /dev/null
+++ b/Controller/Adminhtml/Translation/GenerateJson.php
@@ -0,0 +1,42 @@
+resultPageFactory = $resultPageFactory;
+ }
+
+ /**
+ * @inheridoc
+ */
+ public function execute()
+ {
+ /** @var Page $resultPage */
+ $resultPage = $this->resultPageFactory->create();
+ $this->initPage($resultPage)->addBreadcrumb(
+ __('Generate Frontend Translations'),
+ __('Generate Frontend Translations')
+ );
+
+ $resultPage->getConfig()->getTitle()->prepend(__('Translations'));
+ $resultPage->getConfig()->getTitle()->prepend(__('Generate Frontend Translations'));
+ return $resultPage;
+ }
+}
diff --git a/Controller/Adminhtml/Translation/Index.php b/Controller/Adminhtml/Translation/Index.php
new file mode 100644
index 0000000..5bf1f98
--- /dev/null
+++ b/Controller/Adminhtml/Translation/Index.php
@@ -0,0 +1,36 @@
+resultPageFactory = $resultPageFactory;
+ parent::__construct($context);
+ }
+
+ /**
+ * @inheridoc
+ */
+ public function execute()
+ {
+ $resultPage = $this->resultPageFactory->create();
+ $resultPage->getConfig()->getTitle()->prepend(__("Translations"));
+ return $resultPage;
+ }
+}
diff --git a/Controller/Adminhtml/Translation/InlineEdit.php b/Controller/Adminhtml/Translation/InlineEdit.php
new file mode 100644
index 0000000..1433bd9
--- /dev/null
+++ b/Controller/Adminhtml/Translation/InlineEdit.php
@@ -0,0 +1,67 @@
+jsonFactory = $jsonFactory;
+ $this->translationFactory = $translationFactory;
+ }
+
+ /**
+ * @inheridoc
+ */
+ public function execute()
+ {
+ /** @var \Magento\Framework\Controller\Result\Json $resultJson */
+ $resultJson = $this->jsonFactory->create();
+ $error = false;
+ $messages = [];
+
+ if ($this->getRequest()->getParam('isAjax')) {
+ $postItems = $this->getRequest()->getParam('items', []);
+ if (!count($postItems)) {
+ $messages[] = __('Please correct the data sent.');
+ $error = true;
+ } else {
+ foreach (array_keys($postItems) as $entityId) {
+ $model = $this->translationFactory->create()->load($entityId);
+ try {
+ // phpcs:ignore
+ $model->setData(array_merge($model->getData(), $postItems[$entityId]));
+ $model->save();
+ } catch (\Exception $e) {
+ $messages[] = "[Translation ID: {$entityId}] {$e->getMessage()}";
+ $error = true;
+ }
+ }
+ }
+ }
+
+ return $resultJson->setData([
+ 'messages' => $messages,
+ 'error' => $error
+ ]);
+ }
+}
diff --git a/Controller/Adminhtml/Translation/NewAction.php b/Controller/Adminhtml/Translation/NewAction.php
new file mode 100644
index 0000000..cefdfc7
--- /dev/null
+++ b/Controller/Adminhtml/Translation/NewAction.php
@@ -0,0 +1,40 @@
+resultPageFactory = $resultPageFactory;
+ }
+
+ /**
+ * @inheridoc
+ */
+ public function execute()
+ {
+ /** @var \Magento\Backend\Model\View\Result\Page $resultPage */
+ $resultPage = $this->resultPageFactory->create();
+ $this->initPage($resultPage)->addBreadcrumb(__('New Translation'), __('New Translation'));
+ $resultPage->getConfig()->getTitle()->prepend(__('Translations'));
+ $resultPage->getConfig()->getTitle()->prepend(__('New Translation'));
+
+ return $resultPage;
+ }
+}
diff --git a/Controller/Adminhtml/Translation/Save.php b/Controller/Adminhtml/Translation/Save.php
new file mode 100644
index 0000000..618051e
--- /dev/null
+++ b/Controller/Adminhtml/Translation/Save.php
@@ -0,0 +1,81 @@
+dataPersistor = $dataPersistor;
+ $this->translationRepository = $translationRepository;
+ parent::__construct($context);
+ }
+
+ /**
+ * @inheridoc
+ */
+ public function execute()
+ {
+ /** @var \Magento\Backend\Model\View\Result\Redirect $resultRedirect */
+ $resultRedirect = $this->resultRedirectFactory->create();
+ $data = $this->getRequest()->getPostValue();
+ $translationId = $this->getRequest()->getParam('key_id');
+ if (!$translationId || !$data) {
+ return $resultRedirect->setPath('*/*/');
+ }
+
+ try {
+ $translationDataModel = $this->translationRepository->getById($translationId);
+
+ // Fix issue: for some reason the "locale" of the backend-theme is changed if field was named "locale".
+ $translationDataModel->setLocale($this->getRequest()->getParam('locale_field'));
+ $translationDataModel->setString($this->getRequest()->getParam('string'));
+ $translationDataModel->setTranslate($this->getRequest()->getParam('translate'));
+ $translationDataModel->setFrontend($this->getRequest()->getParam('frontend'));
+
+ $model = $this->translationRepository->save($translationDataModel);
+
+ $this->messageManager->addSuccessMessage(__('The translation was successfully updated.'));
+ $this->dataPersistor->clear('phpro_translations_translation');
+
+ if ($this->getRequest()->getParam('back')) {
+ return $resultRedirect->setPath('*/*/edit', ['key_id' => $model->getKeyId()]);
+ }
+
+ return $resultRedirect->setPath('*/*/');
+ } catch (NoSuchEntityException $e) {
+ $this->messageManager->addErrorMessage($e->getMessage());
+ return $resultRedirect->setPath('*/*/');
+ } catch (LocalizedException $e) {
+ $this->messageManager->addErrorMessage($e->getMessage());
+ } catch (\Exception $e) {
+ $this->messageManager->addExceptionMessage($e, __('Something went wrong while saving the translation.'));
+ }
+
+ $this->dataPersistor->set('phpro_translations_translation', $data);
+ return $resultRedirect->setPath('*/*/edit', ['key_id' => $this->getRequest()->getParam('key_id')]);
+ }
+}
diff --git a/Controller/Adminhtml/Translation/SaveNew.php b/Controller/Adminhtml/Translation/SaveNew.php
new file mode 100644
index 0000000..089a8ef
--- /dev/null
+++ b/Controller/Adminhtml/Translation/SaveNew.php
@@ -0,0 +1,104 @@
+dataPersistor = $dataPersistor;
+ $this->translationManagement = $translationManagement;
+ }
+
+ /**
+ * @inheridoc
+ */
+ public function execute()
+ {
+ /** @var \Magento\Backend\Model\View\Result\Redirect $resultRedirect */
+ $resultRedirect = $this->resultRedirectFactory->create();
+ $data = $this->getRequest()->getPostValue();
+ if (!$data) {
+ return $resultRedirect->setPath('*/*/');
+ }
+
+ try {
+ $this->createTranslations($this->getRequest());
+ $this->dataPersistor->clear('phpro_translations_translation');
+ return $resultRedirect->setPath('*/*/');
+ } catch (LocalizedException $e) {
+ $this->messageManager->addErrorMessage($e->getMessage());
+ } catch (\Exception $e) {
+ $this->messageManager->addExceptionMessage($e, __('Something went wrong while saving the translation.'));
+ }
+
+ $this->dataPersistor->set('phpro_translations_translation', $data);
+ return $resultRedirect->setPath('*/*/new');
+ }
+
+ /**
+ * Create all translations for given locales.
+ *
+ * @param RequestInterface $request
+ */
+ private function createTranslations(RequestInterface $request): void
+ {
+ $locales = $request->getParam('locale_field');
+ $translationKey = $this->getRequest()->getParam('string');
+ $translationValue = $this->getRequest()->getParam('translate');
+ $frontend = $this->getRequest()->getParam('frontend');
+ $successSavedLocales = [];
+ foreach ($locales as $locale) {
+ try {
+ $this->translationManagement->addTranslation(
+ $translationKey,
+ $translationValue,
+ $locale,
+ $frontend
+ );
+ $successSavedLocales[] = $locale;
+ } catch (LocalizedException $e) {
+ $this->messageManager->addErrorMessage(
+ __('Translation save failed for locale %1: %2', $locale, $e->getMessage())
+ );
+ } catch (\Exception $e) {
+ $this->messageManager->addExceptionMessage(
+ $e,
+ __('Translation save failed for locale %1: unknown error.')
+ );
+ }
+ }
+
+ if (0 >= count($successSavedLocales)) {
+ return;
+ }
+
+ $this->messageManager->addSuccessMessage(
+ __('The translation was successfully created for locale(s) %1.', implode(', ', $successSavedLocales))
+ );
+ }
+}
diff --git a/Files/Sample/translations.csv b/Files/Sample/translations.csv
new file mode 100644
index 0000000..acd4390
--- /dev/null
+++ b/Files/Sample/translations.csv
@@ -0,0 +1,5 @@
+key,translation,locale
+"First translation key","First value","en_US"
+"First translation key","Eerste waarde","nl_NL"
+"Second translation key","Second value","en_US"
+"Third translation key","Derde waarde","nl_NL"
diff --git a/LICENSE b/LICENSE
new file mode 100644
index 0000000..0837d18
--- /dev/null
+++ b/LICENSE
@@ -0,0 +1,21 @@
+The MIT License (MIT)
+
+Copyright (c) 2022 Phpro
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in
+all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+THE SOFTWARE.
\ No newline at end of file
diff --git a/Model/Data/ExportStats.php b/Model/Data/ExportStats.php
new file mode 100644
index 0000000..679520b
--- /dev/null
+++ b/Model/Data/ExportStats.php
@@ -0,0 +1,38 @@
+fileName = $fileName;
+ $this->totalRows = $totalRows;
+ }
+
+ public static function fromRawData(string $fileName, int $totalRows): ExportStats
+ {
+ return new self($fileName, $totalRows);
+ }
+
+ public function getFileName(): string
+ {
+ return $this->fileName;
+ }
+
+ public function getTotalRows(): int
+ {
+ return $this->totalRows;
+ }
+}
diff --git a/Model/Data/ImportStats.php b/Model/Data/ImportStats.php
new file mode 100644
index 0000000..aed8823
--- /dev/null
+++ b/Model/Data/ImportStats.php
@@ -0,0 +1,52 @@
+created[$row] = $key;
+ }
+
+ public function getCreatedCount(): int
+ {
+ return count($this->created);
+ }
+
+ public function addFailed(int $row, string $reason)
+ {
+ $this->failed[$row] = $reason;
+ }
+
+ public function getFailedCount(): int
+ {
+ return count($this->failed);
+ }
+
+ public function addSkipped(int $row, string $reason)
+ {
+ $this->skipped[$row] = $reason;
+ }
+
+ public function getSkippedCount(): int
+ {
+ return count($this->skipped);
+ }
+}
diff --git a/Model/Data/InlineGenerateStats.php b/Model/Data/InlineGenerateStats.php
new file mode 100644
index 0000000..3dff6ce
--- /dev/null
+++ b/Model/Data/InlineGenerateStats.php
@@ -0,0 +1,42 @@
+storeInformation = $storeInformation;
+ $this->storeId = $storeId;
+ $this->amountGenerated = $amountGenerated;
+ }
+
+ public function getStoreInformation(): string
+ {
+ return $this->storeInformation;
+ }
+
+ public function getStoreId(): int
+ {
+ return $this->storeId;
+ }
+
+ public function getAmountGenerated(): int
+ {
+ return $this->amountGenerated;
+ }
+}
diff --git a/Model/Data/InlineGenerateStatsCollection.php b/Model/Data/InlineGenerateStatsCollection.php
new file mode 100644
index 0000000..79d2ea6
--- /dev/null
+++ b/Model/Data/InlineGenerateStatsCollection.php
@@ -0,0 +1,53 @@
+statsItems = $statsItems;
+ }
+
+ public function add(InlineGenerateStats $stats)
+ {
+ $this->statsItems[] = $stats;
+ }
+
+ /**
+ * @return ArrayIterator|InlineGenerateStats[]
+ */
+ public function getIterator()
+ {
+ return new ArrayIterator($this->statsItems);
+ }
+
+ public function count(): int
+ {
+ return \count($this->statsItems);
+ }
+
+ public function toArray(): array
+ {
+ $result = [];
+ foreach ($this->statsItems as $stats) {
+ $result[] = sprintf(
+ '%s translations for store %s',
+ $stats->getAmountGenerated(),
+ $stats->getStoreInformation()
+ );
+ }
+
+ return $result;
+ }
+}
diff --git a/Model/Data/Translation.php b/Model/Data/Translation.php
new file mode 100644
index 0000000..cb4fc1f
--- /dev/null
+++ b/Model/Data/Translation.php
@@ -0,0 +1,80 @@
+_get(self::KEY_ID);
+ }
+
+ public function setKeyId($keyId)
+ {
+ return $this->setData(self::KEY_ID, $keyId);
+ }
+
+ public function getString()
+ {
+ return $this->_get(self::STRING);
+ }
+
+ public function setString($string)
+ {
+ return $this->setData(self::STRING, $string);
+ }
+
+ public function getLocale()
+ {
+ return $this->_get(self::LOCALE);
+ }
+
+ public function setLocale($locale)
+ {
+ return $this->setData(self::LOCALE, $locale);
+ }
+
+ public function getTranslate()
+ {
+ return $this->_get(self::TRANSLATE);
+ }
+
+ public function setTranslate($translate)
+ {
+ return $this->setData(self::TRANSLATE, $translate);
+ }
+
+ public function getFrontend()
+ {
+ return (boolean) $this->_get(self::FRONTEND);
+ }
+
+ public function setFrontend($translate)
+ {
+ return $this->setData(self::FRONTEND, $translate);
+ }
+
+ /**
+ * Retrieve existing extension attributes object or create a new one.
+ * @return \Phpro\Translations\Api\Data\TranslationExtensionInterface|null
+ */
+ public function getExtensionAttributes()
+ {
+ return $this->_getExtensionAttributes();
+ }
+
+ /**
+ * Set an extension attributes object.
+ * @param \Phpro\Translations\Api\Data\TranslationExtensionInterface $extensionAttributes
+ * @return $this
+ */
+ public function setExtensionAttributes(
+ \Phpro\Translations\Api\Data\TranslationExtensionInterface $extensionAttributes
+ ) {
+ return $this->_setExtensionAttributes($extensionAttributes);
+ }
+}
diff --git a/Model/ExportManagement.php b/Model/ExportManagement.php
new file mode 100644
index 0000000..9a3bd3e
--- /dev/null
+++ b/Model/ExportManagement.php
@@ -0,0 +1,92 @@
+csv = $csv;
+ $this->searchCriteriaBuilder = $searchCriteriaBuilder;
+ $this->repository = $repository;
+ $this->directoryList = $directoryList;
+ }
+
+ public function export(?array $locales): ExportStats
+ {
+ $this->csv->setDelimiter(',');
+ $this->csv->setEnclosure('"');
+
+ if (!empty($locales)) {
+ $this->searchCriteriaBuilder->addFilter('locale', $locales, 'in');
+ }
+
+ $fileName = $this->getExportPath((empty($locales)) ? 'all' : implode('_', $locales));
+ $result = $this->repository->getList($this->searchCriteriaBuilder->create());
+ $this->csv->saveData($fileName, $this->formatResultsToCsvData($result));
+
+ return ExportStats::fromRawData($fileName, $result->getTotalCount());
+ }
+
+ private function formatResultsToCsvData(\Magento\Framework\Api\SearchResultsInterface $result): \Generator
+ {
+ /** @var \Phpro\Translations\Api\Data\TranslationInterface $item */
+ foreach ($result->getItems() as $item) {
+ yield [$item->getString(), $item->getTranslate(), $item->getLocale()];
+ }
+ }
+
+ private function getExportPath(string $suffix): string
+ {
+ $date = new \DateTime();
+ $path = sprintf(
+ '%s/translations',
+ $this->directoryList->getPath(DirectoryList::VAR_DIR)
+ );
+
+ // phpcs:disable
+ if (!file_exists($path)) {
+ mkdir($path);
+ }
+ // phpcs:enable
+
+ return sprintf(
+ '%s/%s_export_%s.csv',
+ $path,
+ $date->format('Ymd_His'),
+ $suffix
+ );
+ }
+}
diff --git a/Model/Import/Translations.php b/Model/Import/Translations.php
new file mode 100644
index 0000000..f37bb45
--- /dev/null
+++ b/Model/Import/Translations.php
@@ -0,0 +1,228 @@
+jsonHelper = $jsonHelper;
+ $this->_importExportData = $importExportData;
+ $this->_resourceHelper = $resourceHelper;
+ $this->_dataSourceModel = $importData;
+ $this->_connection = $resource->getConnection(ResourceConnection::DEFAULT_CONNECTION);
+ $this->errorAggregator = $errorAggregator;
+ $this->localeValidator = $localeValidator;
+ $this->initMessageTemplates();
+ }
+
+ /**
+ * Row validation
+ *
+ * @param array $rowData
+ * @param int $rowNum
+ *
+ * @return bool
+ */
+ public function validateRow(array $rowData, $rowNum): bool
+ {
+ if (isset($this->_validatedRows[$rowNum])) {
+ return !$this->getErrorAggregator()->isRowInvalid($rowNum);
+ }
+
+ $translationKey = $rowData[self::HEADER_KEY] ?? '';
+ $locale = $rowData[self::HEADER_LOCALE] ?? '';
+
+ if ('' === trim($translationKey)) {
+ $this->addRowError('KeyIsRequired', $rowNum);
+ }
+
+ try {
+ $this->localeValidator->validate($locale);
+ } catch (\Exception $e) {
+ $this->addRowError('LocaleNotSupported', $rowNum);
+ }
+
+ $this->_validatedRows[$rowNum] = true;
+
+ return !$this->getErrorAggregator()->isRowInvalid($rowNum);
+ }
+
+ /**
+ * @return string
+ */
+ public function getEntityTypeCode()
+ {
+ return self::ENTITY_CODE;
+ }
+
+ /**
+ * @return string
+ */
+ public function getCreatedItemsCount()
+ {
+ return 'n/a';
+ }
+
+ /**
+ * @return string
+ */
+ public function getDeletedItemsCount()
+ {
+ return 'n/a';
+ }
+
+ /**
+ * @return string
+ */
+ public function getUpdatedItemsCount()
+ {
+ return sprintf(
+ '%s created/updated, Total: %s',
+ $this->countItemsUpdated,
+ $this->countTotal
+ );
+ }
+
+ /**
+ * Import data
+ * @throws \Exception
+ * @return bool
+ */
+ protected function _importData(): bool
+ {
+ if (Import::BEHAVIOR_APPEND === $this->getBehavior()) {
+ $this->saveAndReplaceEntity();
+ return true;
+ }
+
+ throw new \Exception(
+ sprintf(
+ 'Behavior %s not supported. Only add/update is supported.',
+ $this->getBehavior()
+ )
+ );
+ }
+
+ /**
+ * Save and replace entities
+ * @return void
+ */
+ private function saveAndReplaceEntity(): void
+ {
+ $behavior = $this->getBehavior();
+ $this->countTotal = 0;
+ while ($bunch = $this->_dataSourceModel->getNextBunch()) {
+ $entityList = [];
+ foreach ($bunch as $rowNum => $row) {
+ if (!$this->validateRow($row, $rowNum)) {
+ continue;
+ }
+ if ($this->getErrorAggregator()->hasToBeTerminated()) {
+ $this->getErrorAggregator()->addRowToSkip($rowNum);
+ continue;
+ }
+ $entityList[] = [
+ self::COL_KEY => $row[self::HEADER_KEY],
+ self::COL_TRANSLATION => $row[self::HEADER_TRANSLATION],
+ self::COL_LOCALE => $row[self::HEADER_LOCALE],
+ ];
+ $this->countTotal++;
+ }
+
+ $this->countTotal++;
+ if (Import::BEHAVIOR_APPEND === $behavior) {
+ $this->countItemsUpdated += $this->insertOnDuplicate($entityList);
+ }
+ }
+ }
+
+ /**
+ * Save entities
+ * @param array $entityList
+ * @return int
+ */
+ private function insertOnDuplicate(array $entityList): int
+ {
+ if ($entityList) {
+ $tableName = $this->_connection->getTableName(static::TABLE);
+ return (int) $this->_connection->insertOnDuplicate(
+ $tableName,
+ $entityList,
+ [
+ self::COL_TRANSLATION,
+ self::COL_LOCALE,
+ ]
+ );
+ }
+
+ return 0;
+ }
+
+ /**
+ * Init error messages
+ */
+ private function initMessageTemplates(): void
+ {
+ $this->addMessageTemplate('KeyIsRequired', __('The key cannot be empty.'));
+ $this->addMessageTemplate('LocaleNotSupported', __('The locale is not supported.'));
+ }
+}
diff --git a/Model/ImportManagement.php b/Model/ImportManagement.php
new file mode 100644
index 0000000..61de9bc
--- /dev/null
+++ b/Model/ImportManagement.php
@@ -0,0 +1,132 @@
+csv = $csv;
+ $this->localeValidator = $localeValidator;
+ $this->translationFactory = $translationFactory;
+ $this->repository = $repository;
+ $this->importStatsFactory = $importStatsFactory;
+ }
+
+ public function importMagentoCsv(string $filePath, string $locale): ImportStats
+ {
+ $this->localeValidator->validate($locale);
+
+ $this->csv->setDelimiter(',');
+ $this->csv->setEnclosure('"');
+ $csvData = $this->csv->getData($filePath);
+
+ /** @var ImportStats $importStats */
+ $importStats = $this->importStatsFactory->create();
+
+ foreach ($csvData as $row => $data) {
+ $rowNumber = $row + 1;
+ try {
+ if (2 > count($data)) {
+ throw new \Exception('Invalid CSV row');
+ }
+ if (empty($data[0]) || empty($data[1])) {
+ throw new \Exception('Empty values in CSV row');
+ }
+ $this->createTranslations($data[0], $data[1], $locale);
+ $importStats->addCreated($rowNumber, $data[0]);
+ } catch (\Exception $e) {
+ $this->handleException($e, $importStats, $rowNumber);
+ }
+ }
+
+ return $importStats;
+ }
+
+ public function importFull(string $filePath): ImportStats
+ {
+ $this->csv->setDelimiter(',');
+ $this->csv->setEnclosure('"');
+ $csvData = $this->csv->getData($filePath);
+
+ /** @var ImportStats $importStats */
+ $importStats = $this->importStatsFactory->create();
+
+ foreach ($csvData as $row => $data) {
+ $rowNumber = $row + 1;
+ try {
+ if (3 > count($data)) {
+ throw new \Exception('Invalid CSV row');
+ }
+ if (empty($data[0]) || empty($data[1]) || empty($data[2])) {
+ throw new \Exception('Empty values in CSV row');
+ }
+ $this->localeValidator->validate($data[2]);
+ $this->createTranslations($data[0], $data[1], $data[2]);
+ $importStats->addCreated($rowNumber, $data[0]);
+ } catch (\Exception $e) {
+ $this->handleException($e, $importStats, $rowNumber);
+ }
+ }
+
+ return $importStats;
+ }
+
+ private function createTranslations(string $translationKey, string $translationValue, string $locale)
+ {
+ /** @var Translation $translationModel */
+ $translationModel = $this->translationFactory->create();
+
+ $translationModel->setLocale($locale);
+ $translationModel->setString($translationKey);
+ $translationModel->setTranslate($translationValue);
+
+ $this->repository->save($translationModel);
+ }
+
+ private function handleException(\Exception $e, ImportStats $importStats, int $rowNumber)
+ {
+ if ($e instanceof CouldNotSaveException && (false !== strpos(strtolower($e->getMessage()), 'unique'))) {
+ $importStats->addSkipped($rowNumber, 'Translation already exist.');
+ return;
+ }
+
+ $importStats->addFailed($rowNumber, $e->getMessage());
+ }
+}
diff --git a/Model/InlineTranslations/FileManager.php b/Model/InlineTranslations/FileManager.php
new file mode 100644
index 0000000..e9f1b40
--- /dev/null
+++ b/Model/InlineTranslations/FileManager.php
@@ -0,0 +1,181 @@
+serializer = $serializer;
+ $this->fileDriver = $driverFile;
+ $this->filesystem = $filesystem;
+ $this->directoryList = $directoryList;
+ $this->urlGenerator = $urlGenerator;
+ $this->date = $dateTime;
+ }
+
+ /**
+ * This method will return a public URL containing the replacement url for the original js-translation.json
+ * located in pub/static/frontend////... folders.
+ *
+ * E.g. https://www.webshop.com/media/phpro_translations/ThemeName/default/nl_BE/1642691554translation.json
+ *
+ * @param string $themePath
+ * @throws LocalizedException
+ * @throws \Magento\Framework\Exception\FileSystemException
+ * @return string
+ */
+ public function getJsTranslationUrl(string $themePath): string
+ {
+ $versionFilePath = $this->getRelativeVersionFilePath($themePath);
+ $reader = $this->filesystem->getDirectoryRead(DirectoryList::MEDIA);
+ if (!$reader->isExist($versionFilePath)) {
+ throw new LocalizedException(
+ __('No %1 file found in directory %2', self::JS_DEPLOYED_VERSION_FILENAME, $versionFilePath)
+ );
+ }
+
+ return \sprintf(
+ '%s/%s/%s/%s',
+ rtrim($this->urlGenerator->getBaseUrl(['_type' => UrlInterface::URL_TYPE_MEDIA]), '/'),
+ self::JS_TRANSLATION_DIRECTORY,
+ $themePath,
+ $reader->readFile($versionFilePath) . self::JS_TRANSLATION_FILENAME_SUFFIX
+ );
+ }
+
+ /**
+ * This method saves the translations as json string in the media directory for related theme and locale.
+ * Also old generated translation files will be cleaned up and a new version string is created.
+ *
+ * Example save path: pub/media/phpro_translations/ThemeName/default/nl_BE/1642691554translation.json
+ *
+ * @param array $translations
+ * @param string $themePath
+ * @throws \Magento\Framework\Exception\FileSystemException
+ */
+ public function writeJsTranslationFileContent(array $translations, string $themePath): void
+ {
+ $savePath = sprintf(
+ '%s/%s/%s',
+ $this->directoryList->getPath(DirectoryList::MEDIA),
+ self::JS_TRANSLATION_DIRECTORY,
+ $themePath
+ );
+
+ // Ensure directory exists:
+ $savePathExists = $this->fileDriver->isExists($savePath);
+ if (!$savePathExists) {
+ $this->fileDriver->createDirectory($savePath);
+ }
+
+ // Clean-up old translation json files:
+ if ($savePathExists) {
+ $this->cleanupOldTranslationFiles($themePath, $savePath);
+ }
+
+ $versionString = (string)$this->date->date()->getTimestamp();
+ // Write new translation json file:
+ $this->fileDriver->filePutContents(
+ sprintf(
+ '%s/%s',
+ $savePath,
+ $versionString . self::JS_TRANSLATION_FILENAME_SUFFIX
+ ),
+ $this->serializer->serialize($translations)
+ );
+
+ // Save version string
+ $this->fileDriver->filePutContents(
+ sprintf(
+ '%s/%s',
+ $savePath,
+ self::JS_DEPLOYED_VERSION_FILENAME
+ ),
+ $versionString
+ );
+ }
+
+ /**
+ * @param string $themePath
+ * @param string $savePath
+ * @throws \Magento\Framework\Exception\FileSystemException
+ */
+ private function cleanupOldTranslationFiles(string $themePath, string $savePath): void
+ {
+ $versionFilePath = $this->getRelativeVersionFilePath($themePath);
+ $reader = $this->filesystem->getDirectoryRead(DirectoryList::MEDIA);
+ if (!$reader->isExist($versionFilePath)) {
+ return;
+ }
+
+ $currentTranslationFile = $reader->readFile($versionFilePath) . self::JS_TRANSLATION_FILENAME_SUFFIX;
+ foreach ($this->fileDriver->readDirectory($savePath) as $file) {
+ // phpcs:disable
+ if ('json' !== pathinfo($file, PATHINFO_EXTENSION) ||
+ $currentTranslationFile === pathinfo($file, PATHINFO_BASENAME)) {
+ continue;
+ }
+ // phpcs:enable
+ $this->fileDriver->deleteFile($file);
+ }
+ }
+
+ private function getRelativeVersionFilePath(string $themeLocalePath): string
+ {
+ return sprintf(
+ '%s/%s/%s',
+ self::JS_TRANSLATION_DIRECTORY,
+ $themeLocalePath,
+ self::JS_DEPLOYED_VERSION_FILENAME
+ );
+ }
+}
diff --git a/Model/InlineTranslationsGenerator.php b/Model/InlineTranslationsGenerator.php
new file mode 100644
index 0000000..8f05439
--- /dev/null
+++ b/Model/InlineTranslationsGenerator.php
@@ -0,0 +1,111 @@
+dataProvider = $dataProvider;
+ $this->state = $state;
+ $this->translate = $translate;
+ $this->fileManager = $fileManager;
+ $this->emulation = $emulation;
+ $this->viewDesign = $viewDesign;
+ $this->systemStore = $systemStore;
+ }
+
+ public function forAll(): InlineGenerateStatsCollection
+ {
+ $storeIds = [];
+ foreach ($this->systemStore->getStoreCollection() as $store) {
+ $storeIds[] = (int)$store->getId();
+ }
+
+ return $this->forStores($storeIds);
+ }
+
+ public function forStores(array $storeIds): InlineGenerateStatsCollection
+ {
+ $statsCollection = new InlineGenerateStatsCollection();
+ foreach ($storeIds as $storeId) {
+ $statsCollection->add(
+ $this->generate((int)$storeId)
+ );
+ }
+
+ return $statsCollection;
+ }
+
+ private function generate(int $storeId): InlineGenerateStats
+ {
+ $translations = [];
+ $area = 'frontend';
+
+ $this->state->emulateAreaCode($area, function () use ($storeId, $area, &$translations) {
+ // We need the emulation start and stop for saving the translation json file to have the correct context
+ $this->emulation->startEnvironmentEmulation($storeId, $area, true);
+ $locale = $this->viewDesign->getLocale();
+ $themePath = $this->viewDesign->getDesignTheme()->getThemePath();
+ // set locale and load translations string:
+ $this->translate
+ ->setLocale($locale)
+ ->loadData($area, true);
+ // find all translations for the frontend files (js en html templates):
+ $translations = $this->dataProvider->getData($themePath);
+ $this->fileManager->writeJsTranslationFileContent($translations, $themePath . '/' . $locale);
+ $this->emulation->stopEnvironmentEmulation();
+ });
+
+ $storeInfo = $this->systemStore->getStoreNameWithWebsite($storeId);
+ return new InlineGenerateStats($storeInfo, $storeId, count($translations));
+ }
+}
diff --git a/Model/ResourceModel/Translation.php b/Model/ResourceModel/Translation.php
new file mode 100644
index 0000000..e4bb6ec
--- /dev/null
+++ b/Model/ResourceModel/Translation.php
@@ -0,0 +1,43 @@
+getConnection();
+ $connection = $this->transactionManager->start($conn);
+ $quotedLocales = array_map(function ($t) use ($conn) {
+ return $conn->quote($t);
+ }, $locales);
+
+ try {
+ $this->objectRelationProcessor->delete(
+ $this->transactionManager,
+ $connection,
+ $this->getMainTable(),
+ $conn->quoteInto(' string = ?', $translationKey)
+ . ' AND locale IN (' . implode(',', $quotedLocales) . ')',
+ []
+ );
+ $this->transactionManager->commit();
+ } catch (\Exception $e) {
+ $this->transactionManager->rollBack();
+ throw $e;
+ }
+ return $this;
+ }
+ /**
+ * Define resource model
+ *
+ * @return void
+ */
+ protected function _construct()
+ {
+ $this->_init('translation', 'key_id');
+ }
+}
diff --git a/Model/ResourceModel/Translation/Collection.php b/Model/ResourceModel/Translation/Collection.php
new file mode 100644
index 0000000..a42690d
--- /dev/null
+++ b/Model/ResourceModel/Translation/Collection.php
@@ -0,0 +1,22 @@
+_init(
+ \Phpro\Translations\Model\Translation::class,
+ \Phpro\Translations\Model\ResourceModel\Translation::class
+ );
+ }
+}
diff --git a/Model/Source/Import/Behavior/TranslationsBasic.php b/Model/Source/Import/Behavior/TranslationsBasic.php
new file mode 100644
index 0000000..53e171f
--- /dev/null
+++ b/Model/Source/Import/Behavior/TranslationsBasic.php
@@ -0,0 +1,44 @@
+ __('Add/Update'),
+ ];
+ }
+
+ /**
+ * @inheritdoc
+ */
+ public function getCode()
+ {
+ return 'translationsbasic';
+ }
+
+ /**
+ * @inheritdoc
+ */
+ public function getNotes($entityCode)
+ {
+ $messages = ['translations' => [
+ Import::BEHAVIOR_APPEND => __(
+ "Add or update translations. Updates are based on translation key (first column of CSV)."
+ ),
+ ]];
+ return isset($messages[$entityCode]) ? $messages[$entityCode] : [];
+ }
+}
diff --git a/Model/Translation.php b/Model/Translation.php
new file mode 100644
index 0000000..3d9e7eb
--- /dev/null
+++ b/Model/Translation.php
@@ -0,0 +1,56 @@
+translationDataFactory = $translationDataFactory;
+ $this->dataObjectHelper = $dataObjectHelper;
+ parent::__construct($context, $registry, $resource, $resourceCollection, $data);
+ }
+
+ /**
+ * @return TranslationInterface
+ */
+ public function getDataModel()
+ {
+ $data = $this->getData();
+
+ $translationDataObject = $this->translationDataFactory->create();
+ $this->dataObjectHelper->populateWithArray(
+ $translationDataObject,
+ $data,
+ TranslationInterface::class
+ );
+
+ return $translationDataObject;
+ }
+}
diff --git a/Model/Translation/DataProvider.php b/Model/Translation/DataProvider.php
new file mode 100644
index 0000000..853b284
--- /dev/null
+++ b/Model/Translation/DataProvider.php
@@ -0,0 +1,81 @@
+collection = $collectionFactory->create();
+ $this->dataPersistor = $dataPersistor;
+ parent::__construct($name, $primaryFieldName, $requestFieldName, $meta, $data);
+ }
+
+ /**
+ * Get data
+ *
+ * @return array
+ */
+ public function getData()
+ {
+ if (isset($this->loadedData)) {
+ return $this->loadedData;
+ }
+ $items = $this->collection->getItems();
+ foreach ($items as $model) {
+ $data = $model->getData();
+ if (isset($data['locale'])) {
+ $data['locale_field'] = $data['locale'];
+ unset($data['locale']);
+ }
+ $this->loadedData[$model->getId()] = $data;
+ }
+ $data = $this->dataPersistor->get('phpro_translations_translation');
+
+ if (!empty($data)) {
+ $model = $this->collection->getNewEmptyItem();
+ $model->setData($data);
+ $this->loadedData[$model->getId()] = $model->getData();
+ $this->dataPersistor->clear('phpro_translations_translation');
+ }
+
+ return $this->loadedData;
+ }
+}
diff --git a/Model/Translation/EmptyDataProvider.php b/Model/Translation/EmptyDataProvider.php
new file mode 100644
index 0000000..29c23cf
--- /dev/null
+++ b/Model/Translation/EmptyDataProvider.php
@@ -0,0 +1,26 @@
+resourceConnection = $resourceConnection;
+ }
+
+ /**
+ * @inheritdoc
+ */
+ public function toOptionArray()
+ {
+ $result = [];
+ $connection = $this->resourceConnection->getConnection();
+ $bind = [':config_path' => self::XML_PATH_LOCALE];
+ $select = $connection
+ ->select()
+ ->from($this->resourceConnection->getTableName('core_config_data'), 'value')
+ ->distinct(true)
+ ->where('path = :config_path');
+ $rowSet = $connection->fetchAll($select, $bind);
+
+ foreach ($rowSet as $row) {
+ $result[] = ['value' => $row['value'], 'label' => $row['value']];
+ }
+
+ return $result;
+ }
+}
diff --git a/Model/TranslationDataManagement.php b/Model/TranslationDataManagement.php
new file mode 100644
index 0000000..26b01c4
--- /dev/null
+++ b/Model/TranslationDataManagement.php
@@ -0,0 +1,149 @@
+logger = $logger;
+ $this->localeValidator = $localeValidator;
+ $this->localeSource = $localeSource;
+ $this->translationFactory = $translationFactory;
+ $this->repository = $repository;
+ $this->throwException = $throwException;
+ }
+
+ /**
+ * @inheritdoc
+ */
+ public function prepare(string $translationKey, ?string $defaultTranslation = null)
+ {
+ $locales = $this->getEnabledLocales();
+ $translationValue = (empty($defaultTranslation)) ? $translationKey : $defaultTranslation;
+
+ foreach ($locales as $locale) {
+ $this->insertTranslation($translationKey, $translationValue, $locale);
+ }
+ }
+
+ /**
+ * @inheritdoc
+ */
+ public function create(string $translationKey, string $translationValue, array $locales = [])
+ {
+ foreach ($locales as $locale) {
+ $this->insertTranslation($translationKey, $translationValue, $locale);
+ }
+ }
+
+ /**
+ * @inheritdoc
+ */
+ public function delete(string $translationKey, ?array $locales = [])
+ {
+ if (empty($locales)) {
+ $locales = $this->getEnabledLocales();
+ }
+
+ try {
+ $this->repository->deleteByTranslationKeyAndLocales($translationKey, $locales);
+ } catch (\Exception $e) {
+ $this->logger->error(
+ sprintf(
+ '[PHPRO TRANSLATION] Cannot delete translation "%s" for locales "%s": %s',
+ $translationKey,
+ implode(',', $locales),
+ $e->getMessage()
+ )
+ );
+ if ($this->throwException) {
+ throw $e;
+ }
+ }
+ }
+
+ private function insertTranslation(string $translationKey, string $translationValue, string $locale)
+ {
+ try {
+ $this->localeValidator->validate($locale);
+
+ /** @var Translation $translationModel */
+ $translationModel = $this->translationFactory->create();
+
+ $translationModel->setLocale($locale);
+ $translationModel->setString($translationKey);
+ $translationModel->setTranslate($translationValue);
+
+ $this->repository->save($translationModel);
+ } catch (\Exception $e) {
+ $this->logger->error(
+ sprintf(
+ '[PHPRO TRANSLATION] Cannot insert translation "%s" for locale "%s": %s',
+ $translationKey,
+ $locale,
+ $e->getMessage()
+ )
+ );
+ if ($this->throwException) {
+ throw $e;
+ }
+ }
+ }
+
+ private function getEnabledLocales(): array
+ {
+ if (!empty($this->enabledLocales)) {
+ return $this->enabledLocales;
+ }
+
+ foreach ($this->localeSource->toOptionArray() as $locale) {
+ $localeCode = $locale['value'];
+ $this->enabledLocales[$localeCode] = $localeCode;
+ }
+
+ return $this->enabledLocales;
+ }
+}
diff --git a/Model/TranslationManagement.php b/Model/TranslationManagement.php
new file mode 100644
index 0000000..4a4e544
--- /dev/null
+++ b/Model/TranslationManagement.php
@@ -0,0 +1,52 @@
+localeValidator = $localeValidator;
+ $this->translationFactory = $translationFactory;
+ $this->repository = $repository;
+ }
+
+ public function addTranslation(
+ string $translationKey,
+ string $translationValue,
+ string $locale,
+ string $frontend
+ ): void {
+ $this->localeValidator->validate($locale);
+
+ /** @var Translation $translationModel */
+ $translationModel = $this->translationFactory->create();
+
+ $translationModel->setLocale($locale);
+ $translationModel->setString($translationKey);
+ $translationModel->setTranslate($translationValue);
+ $translationModel->setFrontend($frontend);
+
+ $this->repository->save($translationModel);
+ }
+}
diff --git a/Model/TranslationRepository.php b/Model/TranslationRepository.php
new file mode 100644
index 0000000..bd61499
--- /dev/null
+++ b/Model/TranslationRepository.php
@@ -0,0 +1,196 @@
+resource = $resource;
+ $this->translationFactory = $translationFactory;
+ $this->dataTranslationFactory = $dataTranslationFactory;
+ $this->translationCollectionFactory = $translationCollectionFactory;
+ $this->searchResultsFactory = $searchResultsFactory;
+ $this->dataObjectHelper = $dataObjectHelper;
+ $this->dataObjectProcessor = $dataObjectProcessor;
+ $this->storeManager = $storeManager;
+ $this->collectionProcessor = $collectionProcessor;
+ $this->extensionAttributesJoinProcessor = $extensionAttributesJoinProcessor;
+ $this->extensibleDataObjectConverter = $extensibleDataObjectConverter;
+ }
+
+ /**
+ * {@inheritdoc
+ */
+ public function save(TranslationInterface $translation)
+ {
+ $translationData = $this->extensibleDataObjectConverter->toNestedArray(
+ $translation,
+ [],
+ TranslationInterface::class
+ );
+
+ $translationModel = $this->translationFactory->create()->setData($translationData);
+
+ try {
+ $this->resource->save($translationModel);
+ } catch (\Exception $exception) {
+ throw new CouldNotSaveException(__(
+ 'Could not save the translation: %1',
+ $exception->getMessage()
+ ));
+ }
+ return $translationModel->getDataModel();
+ }
+
+ /**
+ * @inheritdoc
+ */
+ public function getById($translationId)
+ {
+ $translation = $this->translationFactory->create();
+ $this->resource->load($translation, $translationId);
+ if (!$translation->getId()) {
+ throw new NoSuchEntityException(__('Translation with id "%1" does not exist.', $translationId));
+ }
+ return $translation->getDataModel();
+ }
+
+ /**
+ * @inheritdoc
+ */
+ public function getList(SearchCriteriaInterface $criteria)
+ {
+ $collection = $this->translationCollectionFactory->create();
+
+ $this->extensionAttributesJoinProcessor->process(
+ $collection,
+ TranslationInterface::class
+ );
+
+ $this->collectionProcessor->process($criteria, $collection);
+
+ $searchResults = $this->searchResultsFactory->create();
+ $searchResults->setSearchCriteria($criteria);
+
+ $items = [];
+ foreach ($collection as $model) {
+ $items[] = $model->getDataModel();
+ }
+
+ $searchResults->setItems($items);
+ $searchResults->setTotalCount($collection->getSize());
+ return $searchResults;
+ }
+
+ /**
+ * @inheritdoc
+ */
+ public function deleteByTranslationKeyAndLocales(string $translationKey, array $locales)
+ {
+ try {
+ $this->resource->deleteByTranslationKeyAndLocales($translationKey, $locales);
+ } catch (\Exception $exception) {
+ throw new CouldNotDeleteException(__(
+ 'Could not delete the Translations: %1',
+ $exception->getMessage()
+ ));
+ }
+ return true;
+ }
+
+ /**
+ * @inheritdoc
+ */
+ public function deleteById($translationId)
+ {
+ try {
+ $translation = $this->translationFactory->create();
+ $this->resource->load($translation, $translationId);
+ if (!$translation->getId()) {
+ throw new NoSuchEntityException(__('Translation with id "%1" does not exist.', $translationId));
+ }
+ $translation->delete();
+ } catch (\Exception $exception) {
+ throw new CouldNotDeleteException(__(
+ 'Could not delete the Translation: %1',
+ $exception->getMessage()
+ ));
+ }
+ return true;
+ }
+}
diff --git a/Model/Validator/LocaleValidator.php b/Model/Validator/LocaleValidator.php
new file mode 100644
index 0000000..6c742f4
--- /dev/null
+++ b/Model/Validator/LocaleValidator.php
@@ -0,0 +1,50 @@
+localeLists = $localeLists;
+ }
+
+ /**
+ * @param string $locale
+ * @throws \Exception
+ */
+ public function validate(string $locale)
+ {
+ if (!in_array($locale, $this->getSupportedLocales(), true)) {
+ throw new \Exception(sprintf('Locale %s not supported by Magento', $locale));
+ }
+ }
+
+ private function getSupportedLocales(): array
+ {
+ if (!empty($this->locales)) {
+ return $this->locales;
+ }
+
+ $supportedLocaleList = $this->localeLists->getOptionLocales();
+ foreach ($supportedLocaleList as $supportedLocale) {
+ $localeCode = $supportedLocale['value'];
+ $this->locales[$localeCode] = $localeCode;
+ }
+
+ return $this->locales;
+ }
+}
diff --git a/Plugin/UpdateMetadataFields.php b/Plugin/UpdateMetadataFields.php
new file mode 100644
index 0000000..a8a8830
--- /dev/null
+++ b/Plugin/UpdateMetadataFields.php
@@ -0,0 +1,44 @@
+getName()) {
+ return [
+ Translations::COL_KEY,
+ Translations::COL_TRANSLATION,
+ Translations::COL_LOCALE
+ ];
+ }
+
+ return $result;
+ }
+
+ public function afterGetHeaders(Subject $subject, $result, UiComponentInterface $component)
+ {
+ if (self::UI_COMPONENT_NAME === $component->getName()) {
+ return [
+ Translations::HEADER_KEY,
+ Translations::HEADER_TRANSLATION,
+ Translations::HEADER_LOCALE,
+ ];
+ }
+
+ return $result;
+ }
+}
diff --git a/README.md b/README.md
new file mode 100644
index 0000000..90fbeb9
--- /dev/null
+++ b/README.md
@@ -0,0 +1,235 @@
+![](https://github.com/phpro/phpro-mage2-module-translations/workflows/.github/workflows/grumphp.yml/badge.svg)
+
+# Translation module for Magento 2
+
+The `Phpro_Translations` module helps you to manage translations via the Magento backend.
+
+## Features
+* Simple backend CRUD to manage translations
+* Single and multi-inline editing via the grid-overview
+* Import and export via CSV files via CLI
+* Import and export via default Magento import/export functionality
+* Prepare new translations via data patch scripts
+* (Re)generate frontend translations (JSON translation files) via CLI and backend
+
+## Installation
+```
+composer require phpro/mage2-module-translations
+```
+
+## End user documentation
+[Download the end user documentation (PDF)](./resources/phpro-translation-module-EUD.pdf)
+
+## Usage (technical)
+
+### Locales
+All locales must be defined in an ISO format. Locale = ISO-639 (language) + "_" + ISO-3166 (country).
+
+Examples of locales: en_US, nl_BE, nl_NL, fr_BE, de_DE, ...
+
+### Import and export
+
+#### Import Magento translation CSVs
+
+CSV structure must be (key, value):
+```
+"Transkey 1","Transvalue 2"
+"Transkey 2","Transvalue 2"
+...
+```
+Use the `phpro:translations:import` command to import a CSV file for a given locale. Duplicate records are skipped and no updates are applied.
+```
+bin/magento phpro:translations:import /path/to/cs_CZ/cs_CZ.csv cs_CZ --clear-cache
+```
+Output:
+```
+Importing CSV file /path/to/cs_CZ/cs_CZ.csv for locale cs_CZ
+#created: 193
+#skipped: 4
+#failed: 0
+Caches cleared: full_page, block_html, translate
+Done!
+```
+
+#### Export to CSV
+
+Use the `phpro:translations:export` command to export database translations to a CSV file for one or more locales. Separate multiple locales with a space. The exported CSV file is written in the `var/translations` folder of Magento.
+```
+bin/magento phpro:translations:export nl_BE cs_CZ
+```
+Output:
+```
+Exporting translations to CSV file
+Csv file: /path/to/var/translations/20190531_085402_export_nl_BE_cs_CZ.csv
+Total written rows: 390
+Done!
+```
+CSV structure:
+```
+"New Account","Nieuwe account",nl_BE
+"My Wish List","Mijn verlanglijst",nl_BE
+"New Account","Nový účet",cs_CZ
+"My Wish List","Mé oblíbené",cs_CZ
+...
+```
+
+#### Import CSV (including locale)
+You can use the exported CSV file(s) to import on another environment. For example you can prepare new translations on a staging environment and import them later on a production environment.
+
+CSV structure must be (key, value, locale):
+```
+"Transkey 1","Transvalue 2",nl_BE
+"Transkey 2","Transvalue 2",nl_BE
+"Transkey 1","Transvalue 2",fr_BE
+"Transkey 2","Transvalue 2",fr_BE
+...
+```
+Use the `phpro:translations:import-full` command to import a CSV file. Duplicate records are skipped and no updates are applied.
+```
+bin/magento phpro:translations:import-full /path/to/full_import_nl_BE_cs_CZ.csv --clear-cache
+```
+Output:
+```
+Importing CSV file /path/to/full_import_nl_BE_cs_CZ.csv
+#created: 10
+#skipped: 380
+#failed: 0
+Caches cleared: full_page, block_html, translate
+Done!
+```
+
+#### Import via backend
+Go to System → (Data Transfer) → Import to create or update translations based on a CSV file. Please reach out our end user documentation.
+
+### Re-generate frontend translations
+Generating/re-generating frontend translation will generate JSON file(s) which includes all the frontend/JS translations.
+These files are stored in the `pub/media/phpro_translations` directory in their related theme/locale subdirectory.
+
+#### Via backend
+- Go to "Translations -> Generate frontend translations" in the admin. Select "all store views" or select specific ones. Click the button "Generate translations files".
+- Clear full page and block html caches via "System -> Cache Management".
+- Also check the end user documentation
+
+#### Via CLI
+- Use the `phpro:translations:generate-frontend-translations` command to re-generate new JSON file(s)
+- Make sure you clean full_page and block_html cache manually afterwards to apply and enable the newest translation JSON file for the storefront.
+
+**Re-generate for single store view**
+
+Specify the `storeId` argument to re-generate for specific store view (locale). Use the `bin/magento store:list` command to show the store IDs.
+```
+bin/magento phpro:translations:generate-frontend-translations 5
+```
+
+**Re-generate for all**
+
+Leave the `storeId` argument empty to re-generate for all store views.
+```
+bin/magento phpro:translations:generate-frontend-translations
+```
+
+#### During build process
+We recommend to re-generate all frontend translations during your build process with `phpro:translations:generate-frontend-translations` after the `setup:upgrade --keep-generated` step and just before deactivating maintenance.
+
+#### Browser cache optimizations
+The translations JSON files are stored in the directory `pub/media/phpro_translations` with a specific version string.
+You can choose to have a these files optimally cached by browsers by configuring the Cache-Control header. A ngnix example below:
+```
+location /media/phpro_translations/ {
+ add_header X-Frame-Options "SAMEORIGIN";
+ add_header Cache-Control "public";
+ expires +1y;
+}
+```
+
+### Collect translations from code base
+Use the `phpro:translations:prepare-keys` command to collect translations phrases from the code base and prepare them. This will create a translation for every available locale.
+
+### Add translations during development
+To prepare, create or delete translations you can inject `\Phpro\Translations\Api\TranslationDataManagementInterface` as dependency into your data patch script of your module.
+
+#### Prepare
+Add the translation key for all enabled locales of the Magento instance. If default translation is not set, the translation key will be used as default translation.
+
+```
+use Magento\Framework\Setup\Patch\DataPatchInterface;
+use Magento\Framework\Setup\Patch\NonTransactionableInterface;
+use Phpro\Translations\Model\TranslationDataManagement;
+
+class HelloWorldTranslations implements DataPatchInterface, NonTransactionableInterface
+{
+ private TranslationDataManagement $translationDataManagement;
+
+ public function __construct(
+ TranslationDataManagement $translationDataManagement
+ ) {
+ $this->translationDataManagement = $translationDataManagement;
+ }
+
+ public function apply()
+ {
+ $this->translationDataManagement->prepare('Hello world!');
+ $this->translationDataManagement->prepare('Welcome %1', 'Hello %1');
+ // other translation keys here...
+ }
+}
+```
+#### Create
+Add a translation for given locales.
+```
+use Magento\Framework\Setup\Patch\DataPatchInterface;
+use Magento\Framework\Setup\Patch\NonTransactionableInterface;
+use Phpro\Translations\Model\TranslationDataManagement;
+
+class HelloWorldTranslations implements DataPatchInterface, NonTransactionableInterface
+{
+ private TranslationDataManagement $translationDataManagement;
+
+ public function __construct(
+ TranslationDataManagement $translationDataManagement
+ ) {
+ $this->translationDataManagement = $translationDataManagement;
+ }
+
+ public function apply()
+ {
+ $this->translationDataManagement->create('Hello world!', 'Hallo wereld!!!', ['nl_NL', 'nl_BE']);
+ $this->translationDataManagement->create('Hello world!', 'Hello world!!!', ['en_US']);
+ // other translation keys here...
+ }
+}
+```
+#### Delete
+Delete a translation for given translation key and locale(s). In case no locales are given, all enabled locales will be used for deletion.
+```
+use Magento\Framework\Setup\Patch\DataPatchInterface;
+use Magento\Framework\Setup\Patch\NonTransactionableInterface;
+use Phpro\Translations\Model\TranslationDataManagement;
+
+class DeleteHelloWorldTranslations implements DataPatchInterface, NonTransactionableInterface
+{
+ private TranslationDataManagement $translationDataManagement;
+
+ public function __construct(
+ TranslationDataManagement $translationDataManagement
+ ) {
+ $this->translationDataManagement = $translationDataManagement;
+ }
+
+ public function apply()
+ {
+ $this->translationDataManagement->delete('Hello world!');
+ $this->translationDataManagement->delete('Welcome %1', ['nl_BE', 'nl_NL']);
+ // other translation keys here...
+ }
+}
+```
+## PWA
+The checkbox "frontend" could be used to mark translations that need to be exported to a PWA installation.
+With a rest API call, the translations can be fetched and stored in the PWA translations files.
+
+### API call that can be used in build scripts
+```
+curl -G -k -H "Authorization: Bearer " --data-urlencode "searchCriteria[filter_groups][0][filters][0][field]=frontend" --data-urlencode "searchCriteria[filter_groups][0][filters][0][value]=1" --data-urlencode "searchCriteria[filter_groups][1][filters][0][field]=locale" --data-urlencode "searchCriteria[filter_groups][1][filters][0][value]=" rest/V1/phpro-translations/translation/search
+```
+
diff --git a/Test/Unit/Model/InlineTranslations/FileManagerTest.php b/Test/Unit/Model/InlineTranslations/FileManagerTest.php
new file mode 100644
index 0000000..4656681
--- /dev/null
+++ b/Test/Unit/Model/InlineTranslations/FileManagerTest.php
@@ -0,0 +1,168 @@
+serializer = $this->createMock(Json::class);
+ $this->directoryList = $this->createMock(DirectoryList::class);
+ $this->directoryList->method('getPath')->willReturn(self::MEDIA_DIR_ABSOLUTE);
+ $this->filesystem = $this->createMock(Filesystem::class);
+ $this->fileDriver = $this->createMock(File::class);
+ $this->urlGenerator = $this->createMock(UrlInterface::class);
+ $this->date = $this->createMock(TimezoneInterface::class);
+
+ $this->fileManager = new FileManager(
+ $this->serializer,
+ $this->directoryList,
+ $this->filesystem,
+ $this->fileDriver,
+ $this->urlGenerator,
+ $this->date
+ );
+ }
+
+ public function testWriteJsTranslationFileContentWithoutBaseDirYet()
+ {
+ $themePath = 'Theme/default/nl_BE';
+ $now = new \DateTime('now');
+ $version = (string)$now->getTimestamp();
+ $savePath = self::MEDIA_DIR_ABSOLUTE . '/phpro_translations/' . $themePath;
+
+ $this->fileDriver
+ ->expects(static::once())
+ ->method('isExists')
+ ->with(self::MEDIA_DIR_ABSOLUTE . '/phpro_translations/' . $themePath)
+ ->willReturn(false);
+ $this->fileDriver
+ ->expects(static::once())
+ ->method('createDirectory')
+ ->with($savePath);
+ $this->date
+ ->method('date')
+ ->willReturn($now);
+ $this->serializer
+ ->method('serialize')
+ ->willReturn('{"keyValue":"transValue"}');
+ $this->fileDriver
+ ->method('filePutContents')
+ ->withConsecutive(
+ [$savePath . '/' . $version . 'translation.json', '{"keyValue":"transValue"}'],
+ [$savePath . '/deployed_version.txt', $version],
+ );
+
+ $this->fileManager->writeJsTranslationFileContent(
+ ['keyValue' => 'transValue'],
+ 'Theme/default/nl_BE'
+ );
+ }
+
+ public function testWriteJsTranslationFileContentWithBaseDirAndCleanup()
+ {
+ $themePath = 'Theme/default/nl_BE';
+ $now = new \DateTime('now');
+ $version = (string)$now->getTimestamp();
+ $savePath = self::MEDIA_DIR_ABSOLUTE . '/phpro_translations/' . $themePath;
+ $relativePath = 'phpro_translations/' . $themePath;
+ $oldVersion = '123456';
+
+ $this->fileDriver
+ ->expects(static::once())
+ ->method('isExists')
+ ->with(self::MEDIA_DIR_ABSOLUTE . '/phpro_translations/' . $themePath)
+ ->willReturn(true);
+
+ $reader = $this->createMock(Read::class);
+ $this->filesystem
+ ->method('getDirectoryRead')
+ ->willReturn($reader);
+ $reader
+ ->expects(static::once())
+ ->method('isExist')
+ ->with($relativePath . '/deployed_version.txt')
+ ->willReturn(true);
+ $reader
+ ->expects(static::once())
+ ->method('readFile')
+ ->with($relativePath . '/deployed_version.txt')
+ ->willReturn($oldVersion);
+ $this->fileDriver
+ ->expects(static::once())
+ ->method('readDirectory')
+ ->with($savePath)
+ ->willReturn([
+ $savePath . '/deployed_version.txt',
+ $savePath . '/other.file',
+ $savePath . '/123456translation.json',
+ $savePath . '/654321translation.json',
+ $savePath . '/001122translation.json',
+ ]);
+ $this->fileDriver
+ ->expects(static::exactly(2))
+ ->method('deleteFile')
+ ->withConsecutive(
+ [$savePath . '/654321translation.json'],
+ [$savePath . '/001122translation.json']
+ );
+ $this->date
+ ->method('date')
+ ->willReturn($now);
+ $this->serializer
+ ->method('serialize')
+ ->willReturn('{"keyValue":"transValue"}');
+ $this->fileDriver
+ ->method('filePutContents')
+ ->withConsecutive(
+ [$savePath . '/' . $version . 'translation.json', '{"keyValue":"transValue"}'],
+ [$savePath . '/deployed_version.txt', $version],
+ );
+
+ $this->fileManager->writeJsTranslationFileContent(
+ ['keyValue' => 'transValue'],
+ 'Theme/default/nl_BE'
+ );
+ }
+}
diff --git a/Test/Unit/bootstrap.php b/Test/Unit/bootstrap.php
new file mode 100644
index 0000000..c0ad674
--- /dev/null
+++ b/Test/Unit/bootstrap.php
@@ -0,0 +1,2 @@
+urlBuilder = $urlBuilder;
+ parent::__construct($context, $uiComponentFactory, $components, $data);
+ }
+
+ /**
+ * Prepare Data Source
+ *
+ * @param array $dataSource
+ * @return array
+ */
+ public function prepareDataSource(array $dataSource)
+ {
+ if (isset($dataSource['data']['items'])) {
+ foreach ($dataSource['data']['items'] as & $item) {
+ if (isset($item['key_id'])) {
+ $item[$this->getData('name')] = [
+ 'edit' => [
+ 'href' => $this->urlBuilder->getUrl(
+ static::URL_PATH_EDIT,
+ [
+ 'key_id' => $item['key_id']
+ ]
+ ),
+ 'label' => __('Edit')
+ ],
+ 'delete' => [
+ 'href' => $this->urlBuilder->getUrl(
+ static::URL_PATH_DELETE,
+ [
+ 'key_id' => $item['key_id']
+ ]
+ ),
+ 'label' => __('Delete'),
+ 'confirm' => [
+ 'title' =>__('Delete %1', $item['key_id']),
+ 'message' => __('Are you sure you want to delete record %1?', $item['key_id']),
+ ]
+ ]
+ ];
+ }
+ }
+ }
+
+ return $dataSource;
+ }
+}
diff --git a/ViewModel/JsTranslationConfig.php b/ViewModel/JsTranslationConfig.php
new file mode 100644
index 0000000..c1061e4
--- /dev/null
+++ b/ViewModel/JsTranslationConfig.php
@@ -0,0 +1,43 @@
+fileManager = $fileManager;
+ $this->design = $design;
+ }
+
+ public function getJsTranslationFilePath(): string
+ {
+ try {
+ return $this->fileManager->getJsTranslationUrl(
+ sprintf(
+ '%s/%s',
+ $this->design->getDesignTheme()->getThemePath(),
+ $this->design->getLocale()
+ )
+ );
+ } catch (\Exception $e) {
+ return '';
+ }
+ }
+}
diff --git a/composer.json b/composer.json
new file mode 100644
index 0000000..a42b6fe
--- /dev/null
+++ b/composer.json
@@ -0,0 +1,63 @@
+{
+ "name": "phpro/mage2-module-translations",
+ "description": "Manage translations via the Magento backend",
+ "type": "magento2-module",
+ "license": "MIT",
+ "authors": [
+ {
+ "name": "PHPro NV",
+ "email": "info@phpro.be",
+ "homepage": "https://www.phpro.be/"
+ }
+ ],
+ "require": {
+ "php": "^7.3",
+ "magento/framework": "^102.0|^103.0",
+ "magento/module-backend": "^101.0|^102.0",
+ "magento/module-ui": "^101.1"
+ },
+ "require-dev": {
+ "friendsofphp/php-cs-fixer": "~2.14",
+ "magento/magento-coding-standard": "5",
+ "php-parallel-lint/php-parallel-lint": "^1.3",
+ "phpro/grumphp-shim": "~1.5.0",
+ "phpunit/phpunit": "^8.5",
+ "squizlabs/php_codesniffer": "~3.4"
+ },
+ "autoload": {
+ "psr-4": {
+ "Phpro\\Translations\\": ""
+ },
+ "files": [
+ "registration.php"
+ ]
+ },
+ "repositories": [
+ {
+ "type": "composer",
+ "url": "https://repo.magento.com/"
+ }
+ ],
+ "config": {
+ "sort-packages": true,
+ "platform": {
+ "ext-gd": "7.3",
+ "ext-xsl": "7.3",
+ "ext-bcmath": "7.3",
+ "ext-pdo_mysql": "7.3",
+ "ext-soap": "7.3",
+ "ext-zip": "7.3"
+ },
+ "allow-plugins": {
+ "phpro/grumphp-shim": true
+ }
+ },
+ "scripts": {
+ "post-install-cmd": [
+ "([ $COMPOSER_DEV_MODE -eq 0 ] || vendor/bin/phpcs --config-set installed_paths ../../magento/magento-coding-standard,../../phpcompatibility/php-compatibility/PHPCompatibility)"
+ ],
+ "post-update-cmd": [
+ "([ $COMPOSER_DEV_MODE -eq 0 ] || vendor/bin/phpcs --config-set installed_paths ../../magento/magento-coding-standard,../../phpcompatibility/php-compatibility/PHPCompatibility)"
+ ]
+ }
+}
diff --git a/etc/acl.xml b/etc/acl.xml
new file mode 100644
index 0000000..84c2be1
--- /dev/null
+++ b/etc/acl.xml
@@ -0,0 +1,16 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/etc/adminhtml/di.xml b/etc/adminhtml/di.xml
new file mode 100644
index 0000000..ce4717c
--- /dev/null
+++ b/etc/adminhtml/di.xml
@@ -0,0 +1,19 @@
+
+
+
+
+
+
+
+
+ -
+
- \Magento\Framework\View\Element\Message\Renderer\BlockRenderer::CODE
+ -
+
- Phpro_Translations::messages/inlineGenerateSuccessMessage.phtml
+
+
+
+
+
+
diff --git a/etc/adminhtml/menu.xml b/etc/adminhtml/menu.xml
new file mode 100644
index 0000000..1b746be
--- /dev/null
+++ b/etc/adminhtml/menu.xml
@@ -0,0 +1,27 @@
+
+
+
+
diff --git a/etc/adminhtml/routes.xml b/etc/adminhtml/routes.xml
new file mode 100644
index 0000000..3785433
--- /dev/null
+++ b/etc/adminhtml/routes.xml
@@ -0,0 +1,8 @@
+
+
+
+
+
+
+
+
diff --git a/etc/db_schema.xml b/etc/db_schema.xml
new file mode 100644
index 0000000..c42c78b
--- /dev/null
+++ b/etc/db_schema.xml
@@ -0,0 +1,8 @@
+
+
+
+
diff --git a/etc/di.xml b/etc/di.xml
new file mode 100644
index 0000000..8036dc2
--- /dev/null
+++ b/etc/di.xml
@@ -0,0 +1,40 @@
+
+
+
+
+
+
+
+
+
+
+ translation
+ Phpro\Translations\Model\ResourceModel\Translation\Collection
+
+
+
+
+
+ - Phpro\Translations\Model\ResourceModel\Translation\Grid\Collection
+
+
+
+
+
+
+ - Phpro\Translations\Console\Command\Import
+ - Phpro\Translations\Console\Command\ImportFull
+ - Phpro\Translations\Console\Command\Export
+ - Phpro\Translations\Console\Command\GenerateFrontendTranslations
+ - Phpro\Translations\Console\Command\PrepareKeysCommand
+
+
+
+
+
+
+ - Phpro_Translations
+
+
+
+
diff --git a/etc/import.xml b/etc/import.xml
new file mode 100644
index 0000000..ab502a4
--- /dev/null
+++ b/etc/import.xml
@@ -0,0 +1,6 @@
+
+
+
diff --git a/etc/module.xml b/etc/module.xml
new file mode 100644
index 0000000..6c06b1d
--- /dev/null
+++ b/etc/module.xml
@@ -0,0 +1,8 @@
+
+
+
+
+
+
+
+
diff --git a/etc/webapi.xml b/etc/webapi.xml
new file mode 100644
index 0000000..fbadb54
--- /dev/null
+++ b/etc/webapi.xml
@@ -0,0 +1,33 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/grumphp.yml b/grumphp.yml
new file mode 100644
index 0000000..71259bd
--- /dev/null
+++ b/grumphp.yml
@@ -0,0 +1,24 @@
+grumphp:
+ tasks:
+ phpcsfixer:
+ config_contains_finder: true
+ config: .php_cs
+ triggered_by: [php]
+ phplint:
+ triggered_by: [php]
+ phpcs:
+ standard: [Magento2, PSR2]
+ ignore_patterns:
+ - "mage2-module-translations/vendor"
+ - "Test/Unit"
+ exclude:
+ - Magento2.Functions.StaticFunction
+ - Magento2.Exceptions.DirectThrow
+ - Magento2.Exceptions.ThrowCatch
+ - Magento2.Translation.ConstantUsage
+ - Magento2.NamingConvention.ReservedWords
+ triggered_by: [php]
+ phpunit:
+ always_execute: true
+ file_size:
+ max_size: 5M
diff --git a/phpunit.xml b/phpunit.xml
new file mode 100644
index 0000000..32c955b
--- /dev/null
+++ b/phpunit.xml
@@ -0,0 +1,7 @@
+
+
+
+ Test/Unit
+
+
+
diff --git a/registration.php b/registration.php
new file mode 100644
index 0000000..556395e
--- /dev/null
+++ b/registration.php
@@ -0,0 +1,6 @@
+
+
+
+
+
+
+
+
+
diff --git a/view/adminhtml/layout/phpro_translations_translation_generatejson.xml b/view/adminhtml/layout/phpro_translations_translation_generatejson.xml
new file mode 100644
index 0000000..495d6e0
--- /dev/null
+++ b/view/adminhtml/layout/phpro_translations_translation_generatejson.xml
@@ -0,0 +1,9 @@
+
+
+
+
+
+
+
+
+
diff --git a/view/adminhtml/layout/phpro_translations_translation_index.xml b/view/adminhtml/layout/phpro_translations_translation_index.xml
new file mode 100644
index 0000000..210d554
--- /dev/null
+++ b/view/adminhtml/layout/phpro_translations_translation_index.xml
@@ -0,0 +1,9 @@
+
+
+
+
+
+
+
+
+
diff --git a/view/adminhtml/layout/phpro_translations_translation_new.xml b/view/adminhtml/layout/phpro_translations_translation_new.xml
new file mode 100644
index 0000000..0c7152b
--- /dev/null
+++ b/view/adminhtml/layout/phpro_translations_translation_new.xml
@@ -0,0 +1,9 @@
+
+
+
+
+
+
+
+
+
diff --git a/view/adminhtml/templates/messages/inlineGenerateSuccessMessage.phtml b/view/adminhtml/templates/messages/inlineGenerateSuccessMessage.phtml
new file mode 100644
index 0000000..efe07ac
--- /dev/null
+++ b/view/adminhtml/templates/messages/inlineGenerateSuccessMessage.phtml
@@ -0,0 +1,11 @@
+getData('statsItems')
+?>
+= __('Inline frontend translations successfully re-generated for:'); ?>
+
diff --git a/view/adminhtml/ui_component/phpro_translations_create_form.xml b/view/adminhtml/ui_component/phpro_translations_create_form.xml
new file mode 100644
index 0000000..e62528b
--- /dev/null
+++ b/view/adminhtml/ui_component/phpro_translations_create_form.xml
@@ -0,0 +1,114 @@
+
+
diff --git a/view/adminhtml/ui_component/phpro_translations_edit_form.xml b/view/adminhtml/ui_component/phpro_translations_edit_form.xml
new file mode 100644
index 0000000..d3d8a32
--- /dev/null
+++ b/view/adminhtml/ui_component/phpro_translations_edit_form.xml
@@ -0,0 +1,117 @@
+
+
diff --git a/view/adminhtml/ui_component/phpro_translations_generate_translations_form.xml b/view/adminhtml/ui_component/phpro_translations_generate_translations_form.xml
new file mode 100644
index 0000000..3ccbff2
--- /dev/null
+++ b/view/adminhtml/ui_component/phpro_translations_generate_translations_form.xml
@@ -0,0 +1,60 @@
+
+
diff --git a/view/adminhtml/ui_component/phpro_translations_listing.xml b/view/adminhtml/ui_component/phpro_translations_listing.xml
new file mode 100644
index 0000000..7897946
--- /dev/null
+++ b/view/adminhtml/ui_component/phpro_translations_listing.xml
@@ -0,0 +1,140 @@
+
+
+
+ -
+
- phpro_translations_listing.phpro_translations_listing_data_source
+
+
+
+ phpro_translations_columns
+
+ phpro_translations_listing.phpro_translations_listing_data_source
+
+
+
+
+
+
+
+
+ key_id
+
+
+
+ Phpro_Translations::Translation
+
+
+ id
+ key_id
+
+
+
+
+
+ true
+
+
+
+
+
+
+
+
+
+ editSelected
+ phpro_translations_listing.phpro_translations_listing.phpro_translations_columns_editor
+
+ edit
+
+
+
+
+
+
+
+
+
+ phpro_translations_listing.phpro_translations_listing.phpro_translations_columns.ids
+ true
+ key_id
+
+
+ - false
+
+
+
+
+ - phpro_translations_listing.phpro_translations_listing.phpro_translations_columns_editor
+ - startEdit
+ -
+
- ${ $.$data.rowIndex }
+ - true
+
+
+
+
+
+
+ key_id
+ false
+ 55
+
+
+
+
+ asc
+
+
+
+
+
+ text
+
+
+ text
+
+ true
+
+
+
+
+
+
+ text
+
+
+ text
+
+ true
+
+
+
+
+
+
+
+ select
+
+ select
+
+ select
+
+ true
+
+
+
+
+
+
+ key_id
+ false
+ 107
+
+
+
+
diff --git a/view/adminhtml/web/css/source/_module.less b/view/adminhtml/web/css/source/_module.less
new file mode 100644
index 0000000..3392d11
--- /dev/null
+++ b/view/adminhtml/web/css/source/_module.less
@@ -0,0 +1 @@
+@import "module/_general";
diff --git a/view/adminhtml/web/css/source/module/_general.less b/view/adminhtml/web/css/source/module/_general.less
new file mode 100644
index 0000000..52a2c97
--- /dev/null
+++ b/view/adminhtml/web/css/source/module/_general.less
@@ -0,0 +1,16 @@
+/* SVG file ./globe-darkgray.svg converted to base64 */
+.admin__menu .item-translations.level-0 > a:before {
+ content: '';
+ background: url("") no-repeat center;
+ width: 25px;
+ height: 25px;
+ background-size: contain;
+ display: inline-block;
+}
+
+/* SVG file ./globe-lightgray.svg converted to base64 */
+.admin__menu .item-translations.level-0 > a:hover:before,
+.admin__menu .item-translations.level-0._active > a:before {
+ background: url("") no-repeat center;
+ background-size: contain;
+}
diff --git a/view/adminhtml/web/css/source/module/globe-darkgray.svg b/view/adminhtml/web/css/source/module/globe-darkgray.svg
new file mode 100644
index 0000000..5379a05
--- /dev/null
+++ b/view/adminhtml/web/css/source/module/globe-darkgray.svg
@@ -0,0 +1,7 @@
+
+
diff --git a/view/adminhtml/web/css/source/module/globe-lightgray.svg b/view/adminhtml/web/css/source/module/globe-lightgray.svg
new file mode 100644
index 0000000..0804907
--- /dev/null
+++ b/view/adminhtml/web/css/source/module/globe-lightgray.svg
@@ -0,0 +1,7 @@
+
+
diff --git a/view/frontend/layout/default.xml b/view/frontend/layout/default.xml
new file mode 100644
index 0000000..c104c9d
--- /dev/null
+++ b/view/frontend/layout/default.xml
@@ -0,0 +1,13 @@
+
+
+
+
+
+ Phpro\Translations\ViewModel\JsTranslationConfig
+
+
+
+
+
diff --git a/view/frontend/templates/page/js/require_js_maps.phtml b/view/frontend/templates/page/js/require_js_maps.phtml
new file mode 100644
index 0000000..b43eadb
--- /dev/null
+++ b/view/frontend/templates/page/js/require_js_maps.phtml
@@ -0,0 +1,19 @@
+getViewModel();
+$jsTranslationMapPath = $viewModel->getJsTranslationFilePath();
+?>
+
+
+