Commit 89229d97 by lmf

增加图片插件webp

优化产品主图显示
评论默认隐藏
parent 932207d2
...@@ -189,3 +189,13 @@ $isFilterApplied = $displayedCollection->getFlag(\Amasty\AdvancedReview\Model\To ...@@ -189,3 +189,13 @@ $isFilterApplied = $displayedCollection->getFlag(\Amasty\AdvancedReview\Model\To
} }
} }
</script> </script>
<script>
require([
'jquery'
], function ($) {
console.log("helo!");
$(".amreview-add-new").on(click,function () {
$(".block.review-add.amreview-submit-form").show();
});
});
</script>
...@@ -203,6 +203,8 @@ $thumbSizes = $responsiveImageHelper->getSizes('product_page_image_small'); ...@@ -203,6 +203,8 @@ $thumbSizes = $responsiveImageHelper->getSizes('product_page_image_small');
$(".thumbnails > a").mouseover(function (s) { $(".thumbnails > a").mouseover(function (s) {
var main_img = $(this).attr("name"); var main_img = $(this).attr("name");
$(".main-image").attr("src",main_img); $(".main-image").attr("src",main_img);
$(".main-image-wrapper > picture > source:first ").attr("srcset",main_img);
$(".main-image-wrapper > picture > source:last ").attr("srcset",main_img);
$(".thumbnails > a").removeClass("active"); $(".thumbnails > a").removeClass("active");
$(this).addClass("active"); $(this).addClass("active");
}); });
...@@ -219,6 +221,8 @@ $thumbSizes = $responsiveImageHelper->getSizes('product_page_image_small'); ...@@ -219,6 +221,8 @@ $thumbSizes = $responsiveImageHelper->getSizes('product_page_image_small');
sd = $(".thumbnails > a:first").attr('name'); sd = $(".thumbnails > a:first").attr('name');
} }
$(".main-image").attr("src",sd); $(".main-image").attr("src",sd);
$(".main-image-wrapper > picture > source:first ").attr("srcset",sd);
$(".main-image-wrapper > picture > source:last ").attr("srcset",sd);
}); });
$("a.prev ").on('click',function () { $("a.prev ").on('click',function () {
...@@ -233,6 +237,8 @@ $thumbSizes = $responsiveImageHelper->getSizes('product_page_image_small'); ...@@ -233,6 +237,8 @@ $thumbSizes = $responsiveImageHelper->getSizes('product_page_image_small');
sd = $(".thumbnails > a:last").attr('name'); sd = $(".thumbnails > a:last").attr('name');
} }
$(".main-image").attr("src",sd); $(".main-image").attr("src",sd);
$(".main-image-wrapper > picture > source:first ").attr("srcset",sd);
$(".main-image-wrapper > picture > source:last ").attr("srcset",sd);
}); });
}); });
</script> </script>
......
...@@ -1688,6 +1688,9 @@ button.action.submit.primary { ...@@ -1688,6 +1688,9 @@ button.action.submit.primary {
bottom: 0; bottom: 0;
z-index: 11; z-index: 11;
} }
.breeze-gallery .stage .main-image-wrapper > picture{
display: none;
}
} }
//详情页修改 //详情页修改
@media (min-width: 768px){ @media (min-width: 768px){
...@@ -1824,5 +1827,8 @@ background-color: #000; ...@@ -1824,5 +1827,8 @@ background-color: #000;
flex-direction: inherit !important; flex-direction: inherit !important;
} }
} }
.block.review-add.amreview-submit-form{
display: none;
}
...@@ -41,6 +41,7 @@ ...@@ -41,6 +41,7 @@
"composer/composer": "^1.9 || ^2.0", "composer/composer": "^1.9 || ^2.0",
"elasticsearch/elasticsearch": "~7.11.0", "elasticsearch/elasticsearch": "~7.11.0",
"guzzlehttp/guzzle": "^6.3.3", "guzzlehttp/guzzle": "^6.3.3",
"jajuma/module-webpimages": "^2.1",
"laminas/laminas-captcha": "^2.10", "laminas/laminas-captcha": "^2.10",
"laminas/laminas-code": "^3.5.1", "laminas/laminas-code": "^3.5.1",
"laminas/laminas-crypt": "^3.4.0", "laminas/laminas-crypt": "^3.4.0",
...@@ -73,6 +74,7 @@ ...@@ -73,6 +74,7 @@
"magento/magento-composer-installer": ">=0.1.11", "magento/magento-composer-installer": ">=0.1.11",
"magento/zendframework1": "~1.14.2", "magento/zendframework1": "~1.14.2",
"mageside/module-subscribe-at-checkout": "1.1.7", "mageside/module-subscribe-at-checkout": "1.1.7",
"mobiledetect/mobiledetectlib": "^2.8",
"monolog/monolog": "^1.17", "monolog/monolog": "^1.17",
"paragonie/sodium_compat": "^1.6", "paragonie/sodium_compat": "^1.6",
"pelago/emogrifier": "^5.0.0", "pelago/emogrifier": "^5.0.0",
......
...@@ -4,7 +4,7 @@ ...@@ -4,7 +4,7 @@
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
"This file is @generated automatically" "This file is @generated automatically"
], ],
"content-hash": "96d54abdc29e1aeacfdc595d5d248fb3", "content-hash": "3e6bf64bfbd2224fc8f617fcd165743f",
"packages": [ "packages": [
{ {
"name": "airwallex/payments-plugin-magento", "name": "airwallex/payments-plugin-magento",
...@@ -1545,6 +1545,34 @@ ...@@ -1545,6 +1545,34 @@
"time": "2022-06-20T21:43:03+00:00" "time": "2022-06-20T21:43:03+00:00"
}, },
{ {
"name": "jajuma/module-webpimages",
"version": "2.1.11",
"dist": {
"type": "zip",
"url": "https://repo.magento.com/archives/jajuma/module-webpimages/jajuma-module-webpimages-2.1.11.0.zip",
"shasum": "acc0eddedb94f5b235bcaeceddb215e3ed339cfb"
},
"type": "magento2-module",
"autoload": {
"psr-4": {
"Jajuma\\WebpImages\\": ""
},
"files": [
"registration.php"
]
},
"license": [
"MIT"
],
"authors": [
{
"name": "JaJuMa",
"email": "info@jajuma.de"
}
],
"description": "WebP Optimized Images"
},
{
"name": "justinrainbow/json-schema", "name": "justinrainbow/json-schema",
"version": "5.2.12", "version": "5.2.12",
"source": { "source": {
......
...@@ -130,6 +130,7 @@ return array( ...@@ -130,6 +130,7 @@ return array(
'940abd8fb01ee76a36b44f35dcf9783b' => $vendorDir . '/weew/helpers-array/src/array.php', '940abd8fb01ee76a36b44f35dcf9783b' => $vendorDir . '/weew/helpers-array/src/array.php',
'b598380463d27491e4810196bf363d15' => $vendorDir . '/airwallex/payments-plugin-magento/registration.php', 'b598380463d27491e4810196bf363d15' => $vendorDir . '/airwallex/payments-plugin-magento/registration.php',
'8592c7b0947d8a0965a9e8c3d16f9c24' => $vendorDir . '/elasticsearch/elasticsearch/src/autoload.php', '8592c7b0947d8a0965a9e8c3d16f9c24' => $vendorDir . '/elasticsearch/elasticsearch/src/autoload.php',
'0b2ea8cc52669e97f3bdd7d78dcd8eb5' => $vendorDir . '/jajuma/module-webpimages/registration.php',
'eda65932675b68b5aee4503e0762d64d' => $vendorDir . '/magento/magento2-functional-testing-framework/src/Magento/FunctionalTestingFramework/_bootstrap.php', 'eda65932675b68b5aee4503e0762d64d' => $vendorDir . '/magento/magento2-functional-testing-framework/src/Magento/FunctionalTestingFramework/_bootstrap.php',
'88ede69c0babb90b8d3e9e7a95ad0eb5' => $vendorDir . '/mageside/module-subscribe-at-checkout/registration.php', '88ede69c0babb90b8d3e9e7a95ad0eb5' => $vendorDir . '/mageside/module-subscribe-at-checkout/registration.php',
'3109cb1a231dcd04bee1f9f620d46975' => $vendorDir . '/paragonie/sodium_compat/autoload.php', '3109cb1a231dcd04bee1f9f620d46975' => $vendorDir . '/paragonie/sodium_compat/autoload.php',
......
...@@ -124,6 +124,7 @@ return array( ...@@ -124,6 +124,7 @@ return array(
'Jose\\Component\\Core\\Util\\Ecc\\' => array($vendorDir . '/web-token/jwt-framework/src/Ecc'), 'Jose\\Component\\Core\\Util\\Ecc\\' => array($vendorDir . '/web-token/jwt-framework/src/Ecc'),
'Jose\\' => array($vendorDir . '/web-token/jwt-framework/src'), 'Jose\\' => array($vendorDir . '/web-token/jwt-framework/src'),
'JmesPath\\' => array($vendorDir . '/mtdowling/jmespath.php/src'), 'JmesPath\\' => array($vendorDir . '/mtdowling/jmespath.php/src'),
'Jajuma\\WebpImages\\' => array($vendorDir . '/jajuma/module-webpimages'),
'JMS\\Serializer\\' => array($vendorDir . '/jms/serializer/src'), 'JMS\\Serializer\\' => array($vendorDir . '/jms/serializer/src'),
'Hoa\\Ustring\\' => array($vendorDir . '/hoa/ustring'), 'Hoa\\Ustring\\' => array($vendorDir . '/hoa/ustring'),
'Hoa\\Stream\\' => array($vendorDir . '/hoa/stream'), 'Hoa\\Stream\\' => array($vendorDir . '/hoa/stream'),
......
...@@ -131,6 +131,7 @@ class ComposerStaticInitb71ce7c407b65980cf51508f463c8dcf ...@@ -131,6 +131,7 @@ class ComposerStaticInitb71ce7c407b65980cf51508f463c8dcf
'940abd8fb01ee76a36b44f35dcf9783b' => __DIR__ . '/..' . '/weew/helpers-array/src/array.php', '940abd8fb01ee76a36b44f35dcf9783b' => __DIR__ . '/..' . '/weew/helpers-array/src/array.php',
'b598380463d27491e4810196bf363d15' => __DIR__ . '/..' . '/airwallex/payments-plugin-magento/registration.php', 'b598380463d27491e4810196bf363d15' => __DIR__ . '/..' . '/airwallex/payments-plugin-magento/registration.php',
'8592c7b0947d8a0965a9e8c3d16f9c24' => __DIR__ . '/..' . '/elasticsearch/elasticsearch/src/autoload.php', '8592c7b0947d8a0965a9e8c3d16f9c24' => __DIR__ . '/..' . '/elasticsearch/elasticsearch/src/autoload.php',
'0b2ea8cc52669e97f3bdd7d78dcd8eb5' => __DIR__ . '/..' . '/jajuma/module-webpimages/registration.php',
'eda65932675b68b5aee4503e0762d64d' => __DIR__ . '/..' . '/magento/magento2-functional-testing-framework/src/Magento/FunctionalTestingFramework/_bootstrap.php', 'eda65932675b68b5aee4503e0762d64d' => __DIR__ . '/..' . '/magento/magento2-functional-testing-framework/src/Magento/FunctionalTestingFramework/_bootstrap.php',
'88ede69c0babb90b8d3e9e7a95ad0eb5' => __DIR__ . '/..' . '/mageside/module-subscribe-at-checkout/registration.php', '88ede69c0babb90b8d3e9e7a95ad0eb5' => __DIR__ . '/..' . '/mageside/module-subscribe-at-checkout/registration.php',
'3109cb1a231dcd04bee1f9f620d46975' => __DIR__ . '/..' . '/paragonie/sodium_compat/autoload.php', '3109cb1a231dcd04bee1f9f620d46975' => __DIR__ . '/..' . '/paragonie/sodium_compat/autoload.php',
...@@ -290,6 +291,7 @@ class ComposerStaticInitb71ce7c407b65980cf51508f463c8dcf ...@@ -290,6 +291,7 @@ class ComposerStaticInitb71ce7c407b65980cf51508f463c8dcf
'Jose\\Component\\Core\\Util\\Ecc\\' => 29, 'Jose\\Component\\Core\\Util\\Ecc\\' => 29,
'Jose\\' => 5, 'Jose\\' => 5,
'JmesPath\\' => 9, 'JmesPath\\' => 9,
'Jajuma\\WebpImages\\' => 18,
'JMS\\Serializer\\' => 15, 'JMS\\Serializer\\' => 15,
), ),
'H' => 'H' =>
...@@ -846,6 +848,10 @@ class ComposerStaticInitb71ce7c407b65980cf51508f463c8dcf ...@@ -846,6 +848,10 @@ class ComposerStaticInitb71ce7c407b65980cf51508f463c8dcf
array ( array (
0 => __DIR__ . '/..' . '/mtdowling/jmespath.php/src', 0 => __DIR__ . '/..' . '/mtdowling/jmespath.php/src',
), ),
'Jajuma\\WebpImages\\' =>
array (
0 => __DIR__ . '/..' . '/jajuma/module-webpimages',
),
'JMS\\Serializer\\' => 'JMS\\Serializer\\' =>
array ( array (
0 => __DIR__ . '/..' . '/jms/serializer/src', 0 => __DIR__ . '/..' . '/jms/serializer/src',
......
...@@ -3426,6 +3426,37 @@ ...@@ -3426,6 +3426,37 @@
"install-path": "../hoa/ustring" "install-path": "../hoa/ustring"
}, },
{ {
"name": "jajuma/module-webpimages",
"version": "2.1.11",
"version_normalized": "2.1.11.0",
"dist": {
"type": "zip",
"url": "https://repo.magento.com/archives/jajuma/module-webpimages/jajuma-module-webpimages-2.1.11.0.zip",
"shasum": "acc0eddedb94f5b235bcaeceddb215e3ed339cfb"
},
"type": "magento2-module",
"installation-source": "dist",
"autoload": {
"psr-4": {
"Jajuma\\WebpImages\\": ""
},
"files": [
"registration.php"
]
},
"license": [
"MIT"
],
"authors": [
{
"name": "JaJuMa",
"email": "info@jajuma.de"
}
],
"description": "WebP Optimized Images",
"install-path": "../jajuma/module-webpimages"
},
{
"name": "jms/metadata", "name": "jms/metadata",
"version": "2.6.1", "version": "2.6.1",
"version_normalized": "2.6.1.0", "version_normalized": "2.6.1.0",
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
'name' => 'magento/magento2ce', 'name' => 'magento/magento2ce',
'pretty_version' => 'dev-master', 'pretty_version' => 'dev-master',
'version' => 'dev-master', 'version' => 'dev-master',
'reference' => '2e6888fd835e9bf29c26ecb13cab664e7a994ade', 'reference' => '932207d218c95c847882fd23a97bf06d6f69ff1d',
'type' => 'project', 'type' => 'project',
'install_path' => __DIR__ . '/../../', 'install_path' => __DIR__ . '/../../',
'aliases' => array(), 'aliases' => array(),
...@@ -487,6 +487,15 @@ ...@@ -487,6 +487,15 @@
'aliases' => array(), 'aliases' => array(),
'dev_requirement' => true, 'dev_requirement' => true,
), ),
'jajuma/module-webpimages' => array(
'pretty_version' => '2.1.11',
'version' => '2.1.11.0',
'reference' => NULL,
'type' => 'magento2-module',
'install_path' => __DIR__ . '/../jajuma/module-webpimages',
'aliases' => array(),
'dev_requirement' => false,
),
'jms/metadata' => array( 'jms/metadata' => array(
'pretty_version' => '2.6.1', 'pretty_version' => '2.6.1',
'version' => '2.6.1.0', 'version' => '2.6.1.0',
...@@ -949,7 +958,7 @@ ...@@ -949,7 +958,7 @@
'magento/magento2ce' => array( 'magento/magento2ce' => array(
'pretty_version' => 'dev-master', 'pretty_version' => 'dev-master',
'version' => 'dev-master', 'version' => 'dev-master',
'reference' => '2e6888fd835e9bf29c26ecb13cab664e7a994ade', 'reference' => '932207d218c95c847882fd23a97bf06d6f69ff1d',
'type' => 'project', 'type' => 'project',
'install_path' => __DIR__ . '/../../', 'install_path' => __DIR__ . '/../../',
'aliases' => array(), 'aliases' => array(),
......
<?php
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Block\Adminhtml\ConversionTool\Test;
use Magento\Framework\View\Element\Template;
use Magento\Framework\App\Filesystem\DirectoryList;
use Symfony\Component\Process\Process as SymfonyProcess;
use Symfony\Component\Process\Exception\ProcessFailedException as SymfonyProcessFailedException;
class Result extends \Magento\Framework\View\Element\Template
{
protected $productCollection;
protected $productRepository;
protected $directoryList;
protected $helper;
protected $moduleReader;
protected $error = null;
public function __construct(
Template\Context $context,
\Magento\Catalog\Model\ResourceModel\Product\Collection $productCollection,
\Magento\Catalog\Api\ProductRepositoryInterface $productRepository,
\Jajuma\WebpImages\Helper\Data $helper,
DirectoryList $directoryList,
\Magento\Framework\Module\Dir\Reader $moduleReader,
array $data = []
) {
$this->productCollection = $productCollection;
$this->productRepository = $productRepository;
$this->directoryList = $directoryList;
$this->helper = $helper;
$this->moduleReader = $moduleReader;
parent::__construct($context, $data);
}
public function getImageFromLastProduct()
{
$productId = $this->getRequest()->getParam('product');
if ($productId) {
try {
$product = $this->productRepository->getById($productId);
} catch (\Magento\Framework\Exception\NoSuchEntityException $exception) {
return null;
}
} else {
$lastProductSku = $this->productCollection->getLastItem()->getSku();
$product = $this->productRepository->get($lastProductSku);
}
return $product->getImage();
}
public function convert()
{
$convertTool = $this->getRequest()->getParam('convert_tool');
$productImage = $this->getImageFromLastProduct();
$this->helper->clearTestWebpFolder();
$this->helper->createFolderIfNotExist($this->getMediaPath() . '/webp_image/test/');
$t = time();
if ($productImage) {
$productOriginalImagePath = $this->getMediaPath() . '/catalog/product' . $productImage;
$webpImage = preg_replace('/\.(png|jpg|jpeg)$/i', $t . '.webp', $productImage);
$webpImage = explode('/', $webpImage);
$webpImage = end($webpImage);
$webpPath = $this->getMediaPath() . '/webp_image/test/' . $webpImage;
} else {
$modulePath = $this->moduleReader->getModuleDir(
\Magento\Framework\Module\Dir::MODULE_VIEW_DIR,
'Jajuma_WebpImages'
);
$productOriginalImagePath = $modulePath . '/adminhtml/web/images/test.png';
$webpPath = $this->getMediaPath() . '/webp_image/test/'. $t .'.webp';
}
if ($this->helper->hasCheckTransparency() && $this->helper->hasAlphaTransparency($productOriginalImagePath)) {
return __('Can not convert transparency image!');
}
switch ($convertTool) {
case 'cwebp':
$newFile = $this->convertToWebpViaCwebp($productOriginalImagePath, $webpPath);
break;
case 'convert':
$newFile = $this->convertToWebpViaImageMagick($productOriginalImagePath, $webpPath);
break;
case 'gd':
$newFile = $this->convertToWebpViaGd($productOriginalImagePath, $webpPath);
break;
default:
$newFile = false;
break;
}
if ($newFile) {
$originalUrl = $productImage
? $this->helper->getImageUrlFromPath($productOriginalImagePath)
: $this->_assetRepo->getUrl("Jajuma_WebpImages::images/test.png");
$webpUrl = $this->helper->getImageUrlFromPath($newFile);
return ['original' => $originalUrl, 'webp' => $webpUrl];
} else {
return false;
}
}
public function getMediaPath()
{
return $this->directoryList->getPath(DirectoryList::MEDIA);
}
/**
* Method to convert an image to WebP using the GD method
*
* @param $imagePath
* @param $webpPath
*
* @return bool
*/
private function convertToWebpViaCwebp($imagePath, $webpPath)
{
$customCommand = $this->getRequest()->getParam('cwebp_command');
$pathCommand = $this->getRequest()->getParam('path_to_cwebp');
$cmd = $pathCommand != null ? $pathCommand : 'cwebp';
if (!preg_match(\Jajuma\WebpImages\Helper\Data::REGX_CWEBP_PATH, $cmd)) {
$this->error = __('Invalid Cwepb Path. Path must only include
underscore (_), minus (-), dot (.), slash(/) and alphanumeric characters.');
return false;
}
try {
if ($customCommand != null) {
if (!preg_match(\Jajuma\WebpImages\Helper\Data::REGX_CWEBP, $customCommand)) {
$this->error = __('Invalid Cwepb Custom Command. Custom Command must only include
underscore (_), minus (-), space ( ) and alphanumeric characters.');
return false;
} else {
$process = new SymfonyProcess(
escapeshellarg($cmd) . ' '
. $imagePath . ' '
. $customCommand . ' '
. $webpPath
);
}
} else {
$process = new SymfonyProcess(
escapeShellArg($cmd) . ' '
. $imagePath . ' -q '
. $this->getRequest()->getParam('quality')
. ' -alpha_q 100 -z 9 -m 6 -segments 4 -sns 80 -f 25 -sharpness 0 -strong -pass 10'
. ' -mt -alpha_method 1 -alpha_filter fast -o '
. $webpPath
);
}
$process->mustRun();
} catch (SymfonyProcessFailedException $exception) {
$this->error = __('Conversion Failed. Please make sure your custom path and
command are correct and your quality configuration value is between 0 and 100!');
return false;
}
if (file_exists($webpPath)) {
return $webpPath;
} else {
return false;
}
}
/**
* Method to convert an image to WebP using the Imagemagick command
*
* @param $imagePath
* @param $webpPath
* @return string
*/
private function convertToWebpViaImageMagick($imagePath, $webpPath)
{
if ($this->helper->isLoadedImageMagick()) {
$customCommand = $this->getRequest()->getParam('imagemagick_command');
$pathCommand = $this->getRequest()->getParam('path_to_imagemagick');
$cmd = $pathCommand != null ? $pathCommand : 'convert';
if (!preg_match(\Jajuma\WebpImages\Helper\Data::REGX_IMAGEMAGICK_PATH, $cmd)) {
$this->error = __('Invalid ImageMagick Path. Path must only include
underscore (_), minus (-), dot (.), slash(/) and alphanumeric characters.');
return false;
}
try {
if ($customCommand != null) {
if (!preg_match(\Jajuma\WebpImages\Helper\Data::REGX_IMAGEMAGICK, $customCommand)) {
$this->error = __('Invalid ImageMagick Custom Command. Custom Command must only include
underscore (_), minus (-), space ( ), comma (,), colon (:),
equals sign (=) and alphanumeric characters.');
return false;
} else {
$process = new SymfonyProcess(
escapeshellarg($cmd) . ' '
. $imagePath . ' '
. $customCommand . ' '
. $webpPath
);
}
} else {
$process = new SymfonyProcess(
escapeshellarg($cmd) . ' ' . $imagePath
. ' -quality ' . $this->getRequest()->getParam('quality')
. ' -define webp:lossless=false,method=6,segments=4,sns-strength=80,auto-filter=true,'
. ' filter-sharpness=0,filter-strength=25,filter-type=1,'
. ' alpha-compression=1,alpha-filtering=fast,alpha-quality=100 '
. $webpPath
);
}
$process->mustRun();
} catch (SymfonyProcessFailedException $exception) {
$this->error = __('Conversion Failed. Please make sure your custom path and command are
correct and your quality configuration value is between 0 and 100!');
return false;
}
if (file_exists($webpPath)) {
return $webpPath;
} else {
return false;
}
} else {
return false;
}
}
/**
* Method to convert an image to WebP using the GD method
*
* @param $imagePath
* @param $webpPath
*
* @return bool
*/
private function convertToWebpViaGd($imagePath, $webpPath)
{
if ($this->helper->hasGdSupport() == false) {
return false;
}
$imageData = file_get_contents($imagePath);
try {
$image = imagecreatefromstring($imageData);
imagepalettetotruecolor($image);
} catch (\Exception $ex) {
return false;
}
imagewebp($image, $webpPath, $this->getRequest()->getParam('quality'));
return $webpPath;
}
public function getUnsupportWebpImage()
{
return $this->getViewFileUrl('Jajuma_WebpImages::images/unsupport_webp.jpg');
}
public function getError()
{
return $this->error;
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Block\Adminhtml\System\Config;
class AdvancedConfiguration extends \Magento\Config\Block\System\Config\Form\Field
{
/**
* Render text
*
* @param \Magento\Framework\Data\Form\Element\AbstractElement $element
*
* @return string
*/
public function render(\Magento\Framework\Data\Form\Element\AbstractElement $element)
{
$html = '';
if ($element->getComment()) {
$html .= '<div style="margin: auto; padding: 10px;">';
$html .= $element->getComment() . '</div>';
}
return $html;
}
/**
* Return element html
*
* @param \Magento\Framework\Data\Form\Element\AbstractElement $element
*
* @return string
* @SuppressWarnings(PHPMD.UnusedFormalParameter)
*/
protected function _getElementHtml(\Magento\Framework\Data\Form\Element\AbstractElement $element)
{
return $this->_toHtml();
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Block\Adminhtml\System\Config;
class ClearButton extends \Magento\Config\Block\System\Config\Form\Field
{
/** @var UrlInterface */
protected $_urlBuilder;
/**
* Constructor
*
* @param \Magento\Backend\Block\Template\Context $context
* @param array $data
*/
public function __construct(
\Magento\Backend\Block\Template\Context $context,
array $data = []
) {
$this->_urlBuilder = $context->getUrlBuilder();
parent::__construct($context, $data);
}
/**
* Set template
*
* @return void
*/
protected function _construct()
{
parent::_construct();
$this->setTemplate('Jajuma_WebpImages::system/config/clearbutton.phtml');
}
/**
* Generate button html
*
* @return string
* @throws \Magento\Framework\Exception\LocalizedException
*/
public function getButtonHtml()
{
$button = $this->getLayout()->createBlock(
\Magento\Backend\Block\Widget\Button::class
)->setData(
[
'id' => 'jajuma_webp_test_button',
'label' => __('Clear all webp image'),
'onclick' => 'javascript:clearWebpImage(); return false;',
]
);
return $button->toHtml();
}
/**
* Get admin url
*
* @return string
*/
public function getAdminUrl()
{
return $this->_urlBuilder->getUrl('webp/clear', ['store' => $this->_request->getParam('store')]);
}
/**
* Render button
*
* @param \Magento\Framework\Data\Form\Element\AbstractElement $element
* @return string
* @throws \Magento\Framework\Exception\LocalizedException
*/
public function render(\Magento\Framework\Data\Form\Element\AbstractElement $element)
{
// Remove scope label
$element->unsScope()->unsCanUseWebsiteValue()->unsCanUseDefaultValue();
return parent::render($element);
}
/**
* Return element html
*
* @param \Magento\Framework\Data\Form\Element\AbstractElement $element
* @return string
* @SuppressWarnings(PHPMD.UnusedFormalParameter)
*/
protected function _getElementHtml(\Magento\Framework\Data\Form\Element\AbstractElement $element)
{
return $this->_toHtml();
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Block\Adminhtml\System\Config;
class TestButton extends \Magento\Config\Block\System\Config\Form\Field
{
/** @var UrlInterface */
protected $_urlBuilder;
/**
* @var \Magento\Catalog\Model\ResourceModel\Product\Collection
*/
protected $productCollection;
/**
* TestButton constructor.
*
* @param \Magento\Backend\Block\Template\Context $context
* @param \Magento\Catalog\Model\ResourceModel\Product\Collection $productCollection
* @param array $data
*/
public function __construct(
\Magento\Backend\Block\Template\Context $context,
\Magento\Catalog\Model\ResourceModel\Product\Collection $productCollection,
array $data = []
) {
$this->_urlBuilder = $context->getUrlBuilder();
$this->productCollection = $productCollection;
parent::__construct($context, $data);
}
/**
* Set template
*
* @return void
*/
protected function _construct()
{
parent::_construct();
$this->setTemplate('Jajuma_WebpImages::system/config/testbutton.phtml');
}
/**
* Generate button html
*
* @return string
* @throws \Magento\Framework\Exception\LocalizedException
*/
public function getButtonHtml()
{
$button = $this->getLayout()->createBlock(
\Magento\Backend\Block\Widget\Button::class
)->setData(
[
'id' => 'jajuma_webp_test_button',
'label' => __('Test Conversion Tool'),
'onclick' => 'javascript:conversionToolTest(); return false;',
]
);
return $button->toHtml();
}
/**
* Get admin url
*
* @return string
*/
public function getAdminUrl()
{
return $this->_urlBuilder->getUrl('webp/test', ['store' => $this->_request->getParam('store')]);
}
/**
* Render button
*
* @param \Magento\Framework\Data\Form\Element\AbstractElement $element
* @return string
* @throws \Magento\Framework\Exception\LocalizedException
*/
public function render(\Magento\Framework\Data\Form\Element\AbstractElement $element)
{
// Remove scope label
$element->unsScope()->unsCanUseWebsiteValue()->unsCanUseDefaultValue();
return parent::render($element);
}
/**
* Return element html
*
* @param \Magento\Framework\Data\Form\Element\AbstractElement $element
* @return string
* @SuppressWarnings(PHPMD.UnusedFormalParameter)
*/
protected function _getElementHtml(\Magento\Framework\Data\Form\Element\AbstractElement $element)
{
return $this->_toHtml();
}
/**
* Get lastest product id
*
* @return int|null
*/
public function getLatestProductId()
{
if ($lastItem = $this->productCollection->getLastItem()) {
return $lastItem->getId();
}
return null;
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Block;
use Jajuma\WebpImages\Helper\Data as HelperModule;
class Picture extends \Magento\Framework\View\Element\Template
{
protected $_template = "picture-tag-format.phtml";
/**
* @var string
*/
private $webpImage = '';
/**
* @var string
*/
private $originalImage = '';
/**
* @var string
*/
private $originalTag = '';
/**
* @var string
*/
private $customSrcTag = '';
/**
* @var string
*/
private $customSrcSetTag = '';
/**
* Module helper data
*
* @var HelperModule
*/
protected $helper;
/**
* Picture constructor.
*
* @param \Magento\Framework\View\Element\Template\Context $context
* @param HelperModule $helperData
* @param array $data
*/
public function __construct(
\Magento\Framework\View\Element\Template\Context $context,
HelperModule $helperData,
array $data = []
) {
$this->helper = $helperData;
parent::__construct($context, $data);
}
/**
* Seti original image
*
* @param string $originalImage
*
* @return Picture
*/
public function setOriginalImage($originalImage)
{
$this->originalImage = $originalImage;
return $this;
}
/**
* Get original image
*
* @return string
*/
public function getOriginalImage()
{
return $this->originalImage;
}
/**
* Set Webp Image
*
* @param string $webpImage
*
* @return Picture
*/
public function setWebpImage($webpImage)
{
$this->webpImage = $webpImage;
return $this;
}
/**
* Get Webp Image
*
* @return string
*/
public function getWebpImage()
{
return $this->webpImage;
}
/**
* Set Original Tag
*
* @param string $originalTag
*
* @return Picture
*/
public function setOriginalTag($originalTag)
{
$this->originalTag = $originalTag;
return $this;
}
/**
* Get Original Tag
*
* @return string
*/
public function getOriginalTag()
{
return $this->originalTag;
}
/**
* Set custom src tag
*
* @param string $customSrcTag
*
* @return $this
*/
public function setCustomSrcTag($customSrcTag)
{
$this->customSrcTag = $customSrcTag;
return $this;
}
/**
* Get Custom Src Tag
*
* @return string
*/
public function getCustomSrcTag()
{
return $this->customSrcTag;
}
/**
* Set Custom Src Set Tag
*
* @param mixed $customSrcSetTag
*
* @return $this
*/
public function setCustomSrcSetTag($customSrcSetTag)
{
$this->customSrcSetTag = $customSrcSetTag;
return $this;
}
/**
* Get Custom Src Set Tag
*
* @return mixed
*/
public function getCustomSrcSetTag()
{
return $this->customSrcSetTag;
}
/**
* Get Original Image Type
*
* @return string
*/
public function getOriginalImageType()
{
if (preg_match('/\.(jpg|jpeg)$/i', $this->getOriginalImage())) {
return 'image/jpg';
}
if (preg_match('/\.(png)$/i', $this->getOriginalImage())) {
return 'image/png';
}
return '';
}
/**
* Is Native Lazy Loading Enabled
*
* @return bool
*/
public function isNativeLazyLoadingEnabled()
{
return $this->helper->isNativeLazyLoadingEnabled();
}
/**
* Get Exclude Native Lazy load Image Attributes
*
* @return mixed
*/
public function getExcludeNativeLazyloadImageAttributes()
{
return $this->helper->getExcludeNativeLazyloadImageAttributes();
}
}
<?php
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Block\System\Config\Form\Jajuma;
class Module extends \Magento\Config\Block\System\Config\Form\Field
{
/**
* Render text
*
* @param \Magento\Framework\Data\Form\Element\AbstractElement $element
*
* @return string
*/
public function render(\Magento\Framework\Data\Form\Element\AbstractElement $element)
{
$html = '';
if ($element->getComment()) {
$html = $element->getComment();
}
return $html;
}
/**
* Return element html
*
* @param \Magento\Framework\Data\Form\Element\AbstractElement $element
*
* @return string
* @SuppressWarnings(PHPMD.UnusedFormalParameter)
*/
protected function _getElementHtml(\Magento\Framework\Data\Form\Element\AbstractElement $element)
{
return $this->_toHtml();
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Controller\Adminhtml\Clear;
use Magento\Backend\App\Action;
use Magento\Framework\Controller\ResultFactory;
use Magento\Backend\App\Action\Context;
use Jajuma\WebpImages\Helper\Data;
class Index extends Action
{
/**
* Module helper
*
* @var Data
*/
protected $helper;
/**
* Constructor
*
* @param Data $helper
* @param Context $context
*/
public function __construct(
Data $helper,
Context $context
) {
$this->helper = $helper;
parent::__construct($context);
}
/**
* Execute
*
* @return \Magento\Framework\App\ResponseInterface|\Magento\Framework\Controller\ResultInterface
*/
public function execute()
{
$resultRedirect = $this->resultFactory->create(ResultFactory::TYPE_REDIRECT);
if ($this->helper->clearWebp() == 'nowebpFolder') {
$this->messageManager->addSuccessMessage(__('WebP Image Cache is empty'));
} elseif ($this->helper->clearWebp()) {
$this->messageManager->addSuccessMessage(
__('All WebP images have been removed (please also clear FPC to recreate WebP images)')
);
} else {
$this->messageManager->addErrorMessage(
__('Can not remove the webp_image folder (please check folder permissions)')
);
}
$resultRedirect->setUrl($this->_redirect->getRefererUrl());
return $resultRedirect;
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Controller\Adminhtml\Test;
use Magento\Backend\App\Action;
use Magento\Framework\Controller\ResultFactory;
use Magento\Backend\App\Action\Context;
use Magento\Framework\View\Result\PageFactory;
class Index extends Action
{
/**
* PageFactory
*
* @var PageFactory
*/
protected $resultPageFactory;
/**
* Test controller constructor
*
* @param Context $context
* @param PageFactory $resultPageFactory
*/
public function __construct(
Context $context,
PageFactory $resultPageFactory
) {
parent::__construct($context);
$this->resultPageFactory = $resultPageFactory;
}
/**
* Product list page
*
* @return \Magento\Backend\Model\View\Result\Page
*/
public function execute()
{
/** @var \Magento\Backend\Model\View\Result\Page $resultPage */
$resultPage = $this->resultPageFactory->create();
return $resultPage;
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Controller\Image;
use function GuzzleHttp\json_encode;
use Magento\Framework\App\Action\Context;
use Jajuma\WebpImages\Helper\Data;
use Magento\Framework\Controller\Result\JsonFactory;
class Convert extends \Magento\Framework\App\Action\Action
{
/**
* Context
*
* @var Context
*/
protected $context;
/**
* Module helper Data
*
* @var Data
*/
protected $helper;
/**
* JsonFactory
*
* @var JsonFactory
*/
protected $resultJsonFactory;
/**
* Constructor
*
* @param Context $context
* @param Data $helper
* @param JsonFactory $resultJsonFactory
*/
public function __construct(
Context $context,
Data $helper,
JsonFactory $resultJsonFactory
) {
$this->helper = $helper;
$this->resultJsonFactory = $resultJsonFactory;
parent::__construct($context);
}
/**
* Execute
*
* @return JsonFactory
*/
public function execute()
{
$resultJson = $this->resultJsonFactory->create();
$images = $this->getRequest()->getParam('images');
$isInProductView = $this->getRequest()->getParam('isInProductView');
if ($images) {
$webpUrls = [];
foreach ($images as $imageUrl) {
if (array_key_exists('thumb', $imageUrl)) {
$webpUrlThumb = $this->helper->convert($imageUrl['thumb']);
if ($webpUrlThumb) {
$imageUrl['thumb'] = $webpUrlThumb;
}
}
if (array_key_exists('img', $imageUrl)) {
$webpUrlImg = $this->helper->convert($imageUrl['img']);
if ($webpUrlImg) {
$imageUrl['img'] = $webpUrlImg;
}
}
if (array_key_exists('full', $imageUrl)) {
$webpUrlFull = $this->helper->convert($imageUrl['full']);
if ($webpUrlFull) {
$imageUrl['full'] = $webpUrlFull;
}
}
if (!empty($imageUrl)) {
array_push($webpUrls, $imageUrl);
}
}
}
return $resultJson->setData(['webpUrls' => $webpUrls]);
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Helper;
use Magento\Framework\App\Helper\AbstractHelper;
use Magento\Framework\App\Helper\Context;
use Magento\Framework\Filesystem;
use Magento\Framework\ObjectManager\ObjectManager;
use Magento\Store\Model\StoreManagerInterface;
use Magento\Framework\App\Filesystem\DirectoryList;
use Magento\Framework\Filesystem\Driver\File as DriverFile;
use Symfony\Component\Process\Process as SymfonyProcess;
use Symfony\Component\Process\Exception\ProcessFailedException as SymfonyProcessFailedException;
class Data extends AbstractHelper
{
/** Regx */
const REGX_CWEBP = '/^[a-zA-Z0-9-_\s]+$/mi';
const REGX_IMAGEMAGICK = '/^[a-zA-Z0-9-_\s,=:]+$/mi';
const REGX_CWEBP_PATH = '/(^[a-zA-Z0-9\/_\-\.]+cwebp$|^cwebp$)/mi';
const REGX_IMAGEMAGICK_PATH = '/(^[a-zA-Z0-9\/_\-\.]+convert$|^convert$)/mi';
/**
* StoreManager Interface
*
* @var StoreManagerInterface
*/
protected $storeManager;
/**
* Filesystem
*
* @var Filesystem
*/
protected $filesystem;
/**
* Driver file
*
* @var DriverFile
*/
protected $driverFile;
/**
* @var string|null
*/
protected $newFile;
/**
* IO file
*
* @var Filesystem\Io\File
*/
protected $ioFile;
/**
* @var null
*/
protected $imageQuality;
/**
* Module file helper
*
* @var File
*/
protected $fileHelper;
/**
* Constructor
*
* @param Context $context
* @param ObjectManager $objectManager
* @param StoreManagerInterface $storeManager
* @param Filesystem $filesystem
* @param DriverFile $driverFile
* @param Filesystem\Io\File $ioFile
* @param File $fileHelper
*/
public function __construct(
Context $context,
ObjectManager $objectManager,
StoreManagerInterface $storeManager,
Filesystem $filesystem,
DriverFile $driverFile,
Filesystem\Io\File $ioFile,
File $fileHelper
) {
$this->ioFile = $ioFile;
$this->storeManager = $storeManager;
$this->filesystem = $filesystem;
$this->driverFile = $driverFile;
$this->fileHelper = $fileHelper;
parent::__construct($context);
}
/**
* Convert Image to Webp Image
*
* @param string $imageUrl
*
* @return string
*/
public function convert($imageUrl)
{
$webpUrl = $this->getWebpNameFromImage($imageUrl);
$webpPath = $this->getImagePathFromUrl($webpUrl);
$this->newFile = $webpPath;
$folder = $this->driverFile->getParentDirectory($webpPath);
$this->createFolderIfNotExist($folder);
$imagePath = $this->getImagePathFromUrl($imageUrl);
if (empty($imagePath)) {
return $imageUrl;
}
if (!$this->isFileExists($imagePath)) {
return $imageUrl;
}
if (!$this->isEnabled()) {
return $imageUrl;
}
if ($this->isFileExists($webpPath)) {
// check if modified date of converted image is older than original image.
// then delete the old converted image.
$checkIsOldConvertedImage = $this->checkIsOldConvertedImage($imagePath, $webpPath);
if (!$checkIsOldConvertedImage) {
return $webpUrl;
}
}
// Detect alpha-transparency in PNG-images and skip it
if ($this->hasCheckTransparency() && $this->hasAlphaTransparency($imagePath)) {
return $imageUrl;
}
switch ($this->scopeConfig->getValue(
'webp/advance_mode/convert_tool',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE
)) {
case 'cwebp':
$this->newFile = $this->convertToWebpViaCwebp($imagePath, $webpPath);
break;
case 'convert':
$this->newFile = $this->convertToWebpViaImageMagick($imagePath, $webpPath);
break;
case 'gd':
$this->newFile = $this->convertToWebpViaGd($imagePath, $webpPath);
break;
}
$webpUrl = $this->getImageUrlFromPath($this->newFile);
return $webpUrl;
}
/**
* Check Is Old Converted Image
*
* @param $imagePath
* @param $webpPath
*
* @return bool
* @throws \Magento\Framework\Exception\FileSystemException
* @throws \Magento\Framework\Exception\NoSuchEntityException
*/
public function checkIsOldConvertedImage($imagePath, $webpPath)
{
// Get converted file modified date
$convertedFileModifiedDate = $this->driverFile->stat($webpPath)['mtime'];
// Get original file modified date
$originfileModifiedDate = $this->driverFile->stat($imagePath)['mtime'];
// check if modified date of converted image is older than original image.
if ($originfileModifiedDate && $convertedFileModifiedDate && $originfileModifiedDate > $convertedFileModifiedDate) {
// remove old converted file.
$this->driverFile->deleteFile($webpPath);
return true;
}
return false;
}
/**
* Method to convert an image to WebP using the GD method
*
* @param string $imagePath
* @param string $webpPath
*
* @return bool
*/
public function convertToWebpViaGd($imagePath, $webpPath)
{
if ($this->hasGdSupport() == false) {
return $imagePath;
}
$imageData = $this->fileGetContent($imagePath);
try {
$image = imagecreatefromstring($imageData);
imagepalettetotruecolor($image);
} catch (\Exception $ex) {
return false;
}
imagewebp($image, $webpPath, $this->imageQuality());
return $webpPath;
}
/**
* Method to convert an image to WebP using the Imagemagick command
*
* @param string $imagePath
* @param string $webpPath
*
* @return string
*/
public function convertToWebpViaImageMagick($imagePath, $webpPath)
{
if ($this->isLoadedImageMagick()) {
$customCommand = $this->scopeConfig->getValue(
'webp/advance_mode/imagemagick_command',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE
);
$pathCommand = $this->scopeConfig->getValue(
'webp/advance_mode/path_to_imagemagick',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE
);
$cmd = $pathCommand != null ? $pathCommand : 'convert';
try {
if ($customCommand != null) {
if (!preg_match(\Jajuma\WebpImages\Helper\Data::REGX_IMAGEMAGICK, $customCommand)) {
$this->error = __('Invalid ImageMagick Custom Command. Custom Command must only include
underscore (_), minus (-), space ( ), comma (,), colon (:),
equals sign (=) and alphanumeric characters.');
return false;
} else {
$process = new SymfonyProcess(
$this->escapeshellarg($cmd) . ' '
. $imagePath . ' '
. $customCommand . ' '
. $webpPath
);
}
} else {
$process = new SymfonyProcess(
$this->escapeShellArg($cmd) . ' '
. $imagePath
. ' -quality '
. $this->imageQuality()
. ' -define webp:lossless=false,method=6,segments=4,sns-strength=80,auto-filter=true,'
. ' filter-sharpness=0,filter-strength=25,filter-type=1,'
. ' alpha-compression=1,alpha-filtering=fast,alpha-quality=100 '
. $webpPath
);
}
$process->mustRun();
} catch (SymfonyProcessFailedException $exception) {
return $imagePath;
}
if ($this->isFileExists($webpPath)) {
return $webpPath;
}
}
return $imagePath;
}
/**
* Method to convert an image to WebP using the Cwebp command
*
* @param string $imagePath
* @param string $webpPath
*
* @return string
*/
public function convertToWebpViaCwebp($imagePath, $webpPath)
{
$customCommand = $this->scopeConfig->getValue(
'webp/advance_mode/cwebp_command',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE
);
$pathCommand = $this->scopeConfig->getValue(
'webp/advance_mode/path_to_cwebp',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE
);
$cmd = $pathCommand != null ? $pathCommand : 'cwebp';
try {
if ($customCommand != null) {
if (!preg_match(\Jajuma\WebpImages\Helper\Data::REGX_CWEBP, $customCommand)) {
$this->error = __('Invalid Cwepb Custom Command. Custom Command must only include
underscore (_), minus (-), space ( ) and alphanumeric characters.');
return false;
} else {
$process = new SymfonyProcess(
$this->escapeshellarg($cmd) . ' '
. $imagePath . ' '
. $customCommand . ' '
. $webpPath
);
}
} else {
$process = new SymfonyProcess(
$this->escapeShellArg($cmd) . ' '
. $imagePath . ' -q '
. $this->imageQuality()
. ' -alpha_q 100 -z 9 -m 6 -segments 4 -sns 80 -f 25 -sharpness 0 -strong -pass 10'
. ' -mt -alpha_method 1 -alpha_filter fast -o '
. $webpPath
);
}
$process->mustRun();
} catch (SymfonyProcessFailedException $exception) {
return $imagePath;
}
if ($this->isFileExists($webpPath)) {
return $webpPath;
}
return $imagePath;
}
/**
* Is Enabled
*
* @param int|null $store
*
* @return bool
*/
public function isEnabled($store = null)
{
return $this->scopeConfig->getValue(
'webp/setting/enable',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE,
$store
);
}
/**
* Image Quality
*
* @param int|null $store
*
* @return int
*/
public function imageQuality($store = null)
{
if (!$this->imageQuality) {
$this->imageQuality = $this->scopeConfig->getValue(
'webp/advance_mode/quality',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE,
$store
);
}
return (int)$this->imageQuality;
}
/**
* Has Gd Support
*
* @return bool
*/
public function hasGdSupport()
{
if (!function_exists('imagewebp')) {
return false;
}
return true;
}
/**
* Is Loaded Image Magick
*
* @return bool
*/
public function isLoadedImageMagick()
{
if (extension_loaded('imagick')) {
return true;
} else {
return false;
}
}
/**
* Get the WebP path equivalent of an image path
*
* @param mixed $image
*
* @return mixed
*/
public function getWebpNameFromImage($image)
{
$image = preg_replace('/\.(png|jpg|jpeg)$/i', '.webp', $image);
$image = str_replace('media/', 'media/webp_image/', $image);
return $image;
}
/**
* Get System Paths
*
* @return array
*/
public function getSystemPaths()
{
$baseUrl = $this->storeManager->getStore()->getBaseUrl();
$mediaUrl = $this->storeManager->getStore()->getBaseUrl(\Magento\Framework\UrlInterface::URL_TYPE_MEDIA);
$systemPaths = [
'pub' => [
'url' => $this->storeManager->getStore()->getBaseUrl(\Magento\Framework\UrlInterface::URL_TYPE_MEDIA),
'urlWithoutBaseUrl' => str_replace($baseUrl, '', $mediaUrl),
'path' => $this->filesystem->getDirectoryRead(DirectoryList::MEDIA)->getAbsolutePath()
]
];
return $systemPaths;
}
/**
* Get Image Path From Url
*
* @param string $imageUrl
*
* @return mixed
*/
public function getImagePathFromUrl($imageUrl)
{
$systemPaths = $this->getSystemPaths();
foreach ($systemPaths as $systemPath) {
if (strstr($imageUrl, $systemPath['url'])) {
// if url have base url. ex: https://example.com/pub/media/..
return str_replace($systemPath['url'], $systemPath['path'], $imageUrl);
} elseif (strstr($imageUrl, $systemPath['urlWithoutBaseUrl'])) {
// if url don't have base url. ex: /pub/media/..
if (strpos($imageUrl, '/media/') !== false) {
$imageUrl = str_replace('/media/', 'media/', $imageUrl);
}
// replace pub/media with system path
return str_replace($systemPath['urlWithoutBaseUrl'], $systemPath['path'], $imageUrl);
}
}
return false;
}
/**
* Get Image Url From Path
*
* @param string $imagePath
*
* @return mixed
*/
public function getImageUrlFromPath($imagePath)
{
$systemPaths = $this->getSystemPaths();
if (!preg_match('/^http/', $imagePath)) {
foreach ($systemPaths as $systemPath) {
if (strstr($imagePath, $systemPath['path'])) {
return str_replace($systemPath['path'], $systemPath['url'], $imagePath);
}
}
}
return false;
}
/**
* Has Check Transparency
*
* @param int|null $store
*
* @return mixed
*/
public function hasCheckTransparency($store = null)
{
return $this->scopeConfig->getValue(
'webp/setting/check_transparency',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE,
$store
);
}
/**
* Detect whether an image has PNG alpha transparency
*
* @param mixed $image
*
* @return bool
*/
public function hasAlphaTransparency($image)
{
if (empty($image)) {
return false;
}
if ($this->isFileExists($image) == false) {
return false;
}
if (preg_match('/\.(jpg|jpeg)$/', $image)) {
return false;
}
$fileIo = $this->fileHelper;
$fileIo->setCwd($this->driverFile->getParentDirectory($image));
$fileIo->setIwd($this->driverFile->getParentDirectory($image));
$imageContents = $fileIo->read($image);
$colorType = ord(substr($imageContents, 25, 1));
if ($colorType == 6 || $colorType == 4) {
return true;
} elseif (stripos($imageContents, 'PLTE') !== false && stripos($imageContents, 'tRNS') !== false) {
return true;
}
return false;
}
/**
* Create Folder If Not Exist
*
* @param string $path
*/
public function createFolderIfNotExist($path)
{
if (!$this->driverFile->isDirectory($path)) {
$ioAdapter = $this->ioFile;
$ioAdapter->mkdir($path, 0775);
}
}
/**
* Delete file
*
* @param string $filePath
*/
public function deleteFile($filePath)
{
if ($this->isFileExists($filePath)) {
$ioAdapter = $this->ioFile;
$ioAdapter->rm($filePath);
}
}
/**
* Remove Folder
*
* @param string $folderPath
*
* @return bool|string
*/
public function removeFolder($folderPath)
{
if ($this->driverFile->isDirectory($folderPath)) {
if ($this->driverFile->isWritable($folderPath)) {
$ioAdapter = $this->ioFile;
$ioAdapter->rmdir($folderPath, true);
} else {
return false;
}
} else {
return 'nowebpFolder';
}
}
/**
* Clear Test Webp Folder
*
* @return mixed
*/
public function clearTestWebpFolder()
{
$webpFolder = $this->filesystem->getDirectoryRead(DirectoryList::MEDIA)->getAbsolutePath() . 'webp_image/test';
return $this->removeFolder($webpFolder) ;
}
/**
* Clear webp
*
* @return mixed
*/
public function clearWebp()
{
$webpFolder = $this->filesystem->getDirectoryRead(DirectoryList::MEDIA)->getAbsolutePath() . 'webp_image';
return $this->removeFolder($webpFolder) ;
}
/**
* Get Exclude Image Attribute
*
* @param int|null $store
*
* @return mixed
*/
public function getExcludeImageAttribute($store = null)
{
return $this->scopeConfig->getValue(
'webp/professional_mode/exclude_img',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE,
$store
);
}
/**
* Get custom src tag
*
* @param int|null $store
*
* @return mixed
*/
public function getCustomSrcTag($store = null)
{
return $this->scopeConfig->getValue(
'webp/professional_mode/src_tag',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE,
$store
);
}
/**
* Get Custom Src Set Tag
*
* @param int|null $store
*
* @return mixed
*/
public function getCustomSrcSetTag($store = null)
{
return $this->scopeConfig->getValue(
'webp/professional_mode/srcset_tag',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE,
$store
);
}
/**
* Is Native Lazy Loading Enabled
*
* @param int|null $store
*
* @return mixed
*/
public function isNativeLazyLoadingEnabled($store = null)
{
return $this->scopeConfig->getValue(
'webp/native_lazy/enable_native_lazy',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE,
$store
);
}
/**
* Get Exclude Native Lazy load Image Attribute
*
* @param int|null $store
*
* @return mixed
*/
public function getExcludeNativeLazyloadImageAttribute($store = null)
{
return $this->scopeConfig->getValue(
'webp/native_lazy/exclude_native_lazy',
\Magento\Store\Model\ScopeInterface::SCOPE_STORE,
$store
);
}
/**
* Get Exclude Native Lazy load Image Attributes
*
* @return string
*/
public function getExcludeNativeLazyloadImageAttributes()
{
$excludeImageAttributes = $this->getExcludeNativeLazyloadImageAttribute();
if ($excludeImageAttributes) {
// Make sure unescaped slashes in blacklist get escaped.
$excludeImageAttributes = preg_replace('/([^\\\\]\\K\\/|^\\/)/', '\\/', $excludeImageAttributes);
$excludeImageAttributes = explode(',', $excludeImageAttributes);
$excludeImageAttributes = array_map('trim', $excludeImageAttributes);
$excludeImageAttributes = implode(".*|.*", $excludeImageAttributes);
$excludeImageAttributes = '/(.*data-nowebp=\"true\".*|.*' . $excludeImageAttributes . '.*)/mi';
} else {
$excludeImageAttributes = '/(.*data-nowebp=\"true\".*)/mi';
}
return $excludeImageAttributes;
}
/**
* Is File Exists
*
* @param string $path
*
* @return bool
*/
public function isFileExists($path)
{
return $this->ioFile->fileExists($path);
}
/**
* File Get Content
*
* @param string $path
*
* @return false|string
*/
public function fileGetContent($path)
{
return $this->driverFile->fileGetContents($path);
}
/**
* Escape Shell Arg
*
* @param string $str
*
* @return string
*/
public function escapeShellArg($str)
{
return escapeshellarg($str);
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Helper;
class File extends \Magento\Framework\Filesystem\Io\File
{
/**
* Set iwd
*
* @param mixed $iwd
*
* @return $this
*/
public function setIwd($iwd)
{
$this->_iwd = $iwd;
}
/**
* Set cwd
*
* @param mixed $cwd
*
* @return $this
*/
public function setCwd($cwd)
{
$this->_cwd = $cwd;
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Model\Config;
class Comment implements \Magento\Config\Model\Config\CommentInterface
{
/**
* @var $dir
*/
protected $dir;
/**
* Constructor
*
* @param \Magento\Framework\Filesystem\DirectoryList $dir
*/
public function __construct(
\Magento\Framework\Filesystem\DirectoryList $dir
) {
$this->dir = $dir;
}
/**
* Get comment text
*
* @param string $elementValue
*
* @return string
*/
public function getCommentText($elementValue)
{
return 'Define the specify path of cwebp command or leave it empty to use global command "cwebp". Example: '
. $this->dir->getPath('app') . '/code/Jajuma/WebpImages/bin/cwebp';
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Model\Config\Source;
class Tool implements \Magento\Framework\Option\ArrayInterface
{
/**
* Options getter
*
* @return array
*/
public function toOptionArray()
{
return [
['value' => 'cwebp', 'label' => __('Cwebp')],
['value' => 'convert', 'label' => __('Imagemagick')],
['value' => 'gd', 'label' => __('GD')]
];
}
}
<?php
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2022-present JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Plugin\Backend\Block\Menu;
use Magento\Backend\Block\Menu;
class AddLinkJs
{
/**
* @param Menu $subject
* @param string $html
* @return string
* @throws \Magento\Framework\Exception\LocalizedException
*/
public function afterToHtml(Menu $subject, $html)
{
$js = $subject->getLayout()->createBlock(\Magento\Backend\Block\Template::class)
->setTemplate('Jajuma_WebpImages::backend/menu/link_blank.phtml')
->toHtml();
return $html . $js;
}
}
\ No newline at end of file
<?php
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2022-present JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Plugin\Backend\Model\Menu\Item;
use Magento\Backend\Model\Menu\Item;
class ExtensionsLinkPlugin
{
/**
* @param Item $subject
* @param string $url
* @return string
*/
public function afterGetUrl(Item $subject, $url)
{
if ($subject->getId() === 'Jajuma_Extensions::extensions_link') {
return 'https://www.jajuma.de/en/jajuma-develop/magento-extensions?mtm_campaign=Extensions-Menu';
}
return $url;
}
}
\ No newline at end of file
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Plugin;
use Magento\Framework\Message\ManagerInterface;
use Magento\Framework\App\Config\ScopeConfigInterface;
class ConfigPlugin
{
/**
* Manager Interface
*
* @var ManagerInterface
*/
protected $messageManager;
/**
* Scope Config Interface
*
* @var ScopeConfigInterface
*/
protected $scopeConfig;
/**
* ConfigPlugin constructor.
*
* @param ManagerInterface $messageManager
* @param ScopeConfigInterface $scopeConfig
*/
public function __construct(
ManagerInterface $messageManager,
ScopeConfigInterface $scopeConfig
) {
$this->messageManager = $messageManager;
$this->scopeConfig = $scopeConfig;
}
/**
* Around save method
*
* @param \Magento\Config\Model\Config $subject
* @param \Closure $proceed
*
* @return \Closure $proceed
*/
public function aroundSave(
\Magento\Config\Model\Config $subject,
\Closure $proceed
) {
$data = $subject->getData();
if (isset($data['section']) && $data['section'] === 'webp') {
if (isset($data['groups']['advance_mode']['fields']['cwebp_command']['value'])
&& $data['groups']['advance_mode']['fields']['cwebp_command']['value'] !== '') {
$regexCwebp = \Jajuma\WebpImages\Helper\Data::REGX_CWEBP;
if (!preg_match(
$regexCwebp,
$data['groups']['advance_mode']['fields']['cwebp_command']['value']
)
) {
$this->messageManager->addNoticeMessage(__('Invalid Cwepb Custom Command. Custom Command
must only include underscore (_), minus (-), space ( ) and alphanumeric characters.'));
$data['groups']['advance_mode']['fields']['cwebp_command']['value']
= $this->scopeConfig->getValue('webp/advance_mode/cwebp_command');
}
}
if (isset($data['groups']['advance_mode']['fields']['imagemagick_command']['value'])
&& $data['groups']['advance_mode']['fields']['imagemagick_command']['value'] !== '') {
$regexImagemagick = \Jajuma\WebpImages\Helper\Data::REGX_IMAGEMAGICK;
if (!preg_match(
$regexImagemagick,
$data['groups']['advance_mode']['fields']['imagemagick_command']['value']
)
) {
$this->messageManager->addNoticeMessage(__('Invalid Imagemagick Custom Command.
Custom Command must only include underscore (_), minus (-), space ( ), comma (,), colon (:),
equals sign (=) and alphanumeric characters.'));
$data['groups']['advance_mode']['fields']['imagemagick_command']['value']
= $this->scopeConfig->getValue('webp/advance_mode/imagemagick_command');
}
}
if (isset($data['groups']['advance_mode']['fields']['path_to_cwebp']['value'])
&& $data['groups']['advance_mode']['fields']['path_to_cwebp']['value'] !== '') {
$regexCwebpPath = \Jajuma\WebpImages\Helper\Data::REGX_CWEBP_PATH;
if (!preg_match($regexCwebpPath, $data['groups']['advance_mode']['fields']['path_to_cwebp']['value'])) {
$this->messageManager->addNoticeMessage(__('Invalid Cwebp Path. Path must only include
underscore (_), minus (-), dot (.), slash(/) and alphanumeric characters.'));
$data['groups']['advance_mode']['fields']['path_to_cwebp']['value']
= $this->scopeConfig->getValue('webp/advance_mode/path_to_cwebp');
}
}
if (isset($data['groups']['advance_mode']['fields']['path_to_imagemagick']['value'])
&& $data['groups']['advance_mode']['fields']['path_to_imagemagick']['value'] !== '') {
$regexImageMagickPath = \Jajuma\WebpImages\Helper\Data::REGX_IMAGEMAGICK_PATH;
if (!preg_match(
$regexImageMagickPath,
$data['groups']['advance_mode']['fields']['path_to_imagemagick']['value']
)
) {
$this->messageManager->addNoticeMessage(__('Invalid ImageMagick Path.
Path must only include underscore (_), minus (-), dot (.),
slash(/) and alphanumeric characters.'));
$data['groups']['advance_mode']['fields']['path_to_imagemagick']['value']
= $this->scopeConfig->getValue('webp/advance_mode/path_to_imagemagick');
}
}
$subject->setData($data);
}
return $proceed();
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Plugin\Product\View;
use Jajuma\WebpImages\Helper\Data;
use Magento\Framework\Json\Helper\Data as JsonHelper;
class GalleryPlugin
{
/**
* Module helper data
*
* @var Data
*/
protected $helperWebp;
/**
* Json Helper
*
* @var JsonHelper
*/
protected $jsonHelper;
/**
* Constructor
*
* @param Data $webpImagesHelper
* @param JsonHelper $jsonHelper
*/
public function __construct(
Data $webpImagesHelper,
JsonHelper $jsonHelper
) {
$this->helperWebp = $webpImagesHelper;
$this->jsonHelper = $jsonHelper;
}
/**
* After Get Gallery Images Json
*
* @param \Magento\Catalog\Block\Product\View\Gallery $subject
* @param mixed $result
*/
public function afterGetGalleryImagesJson(\Magento\Catalog\Block\Product\View\Gallery $subject, $result)
{
$newImagesItems = [];
$imagesItems = $this->jsonHelper->jsonDecode($result);
foreach ($imagesItems as $itemImage) {
$thumbImage = $itemImage['thumb'];
$imgImage = $itemImage['img'];
$fullImage = $itemImage['full'];
$webpThumbImageUrl = $this->helperWebp->convert($thumbImage);
$itemImage['thumb_webp'] = $webpThumbImageUrl;
$webpImgImageUrl = $this->helperWebp->convert($imgImage);
$itemImage['img_webp'] = $webpImgImageUrl;
$webpFullImageUrl = $this->helperWebp->convert($fullImage);
$itemImage['full_webp'] = $webpFullImageUrl;
$newImagesItems[] = $itemImage;
}
return $this->jsonHelper->jsonEncode($newImagesItems);
}
}
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Plugin;
use Magento\Framework\View\LayoutInterface;
use Jajuma\WebpImages\Block\Picture;
use Jajuma\WebpImages\Helper\Data;
use Magento\Store\Model\StoreManagerInterface;
use Psr\Log\LoggerInterface;
class ReplaceImageTag
{
/**
* Module helper Data
*
* @var Data
*/
protected $helper;
/**
* StoreManagerInterface
*
* @var StoreManagerInterface
*/
protected $storeManager;
/**
* Constructor
*
* @param Data $helper
* @param StoreManagerInterface $storeManager
*/
public function __construct(
Data $helper,
StoreManagerInterface $storeManager,
LoggerInterface $logger
) {
$this->helper = $helper;
$this->storeManager = $storeManager;
$this->logger = $logger;
}
/**
* After get output
*
* @param LayoutInterface $layout
* @param string $output
*
* @return string
*/
public function afterGetOutput(LayoutInterface $layout, $output)
{
if (!$this->helper->isEnabled()) {
return $output;
}
$regex = '/<img([^<]+\s|\s)src=(\"|' . "\')([^<]+?\.(png|jpg|jpeg))[^<]+>(?!(<\/pic|\s*<\/pic))/mi";
if (preg_match_all($regex, $output, $images, PREG_OFFSET_CAPTURE) === false) {
return $output;
}
$accumulatedChange = 0;
$baseUrl = $this->storeManager->getStore()->getBaseUrl();
$mediaUrl = $this->storeManager->getStore()->getBaseUrl(
\Magento\Framework\UrlInterface::URL_TYPE_MEDIA
);
$mediaUrlWithoutBaseUrl = str_replace($baseUrl, '', $mediaUrl);
$excludeImageAttributes = $this->getExcludeImageAttributes();
$customSrcSetTag = $this->helper->getCustomSrcSetTag() ? $this->helper->getCustomSrcSetTag() : '';
foreach ($images[0] as $index => $image) {
$offset = $image[1] + $accumulatedChange;
$htmlTag = $images[0][$index][0];
$imageUrl = $images[3][$index][0];
/**
* Skip when image is not from same server
*/
if (strpos($imageUrl, $mediaUrl) === false && strpos($imageUrl, $mediaUrlWithoutBaseUrl) === false) {
continue;
}
/**
* Skip when image contains an excluded attribute
*/
$isValidRegex = false;
try {
preg_match($excludeImageAttributes, '');
$isValidRegex = true;
} catch (\Exception $e) {
$this->logger->info("Conversion Blacklist Configuration is invalid:" . $excludeImageAttributes);
$this->logger->info("Detail: " . $e->getMessage());
}
if ($isValidRegex) {
if (preg_match_all($excludeImageAttributes, $htmlTag)) {
continue;
}
}
$pictureTag = $this->convertImage($imageUrl, $htmlTag, $customSrcSetTag, $layout);
if (!$pictureTag) {
continue;
}
$output = substr_replace($output, $pictureTag, $offset, strlen($htmlTag));
$accumulatedChange = $accumulatedChange + (strlen($pictureTag) - strlen($htmlTag));
}
return $output;
}
/**
* Get picture tag format
*
* @param LayoutInterface $layout
*
* @return Picture
*/
private function getPicture(LayoutInterface $layout)
{
/** @var Picture $block */
$block = $layout->createBlock(Picture::class);
return $block;
}
/**
* Get exclude image attributes
*
* @return string
*/
private function getExcludeImageAttributes()
{
$excludeImageAttributes = $this->helper->getExcludeImageAttribute();
if ($excludeImageAttributes) {
// Make sure unescaped slashes in blacklist get escaped.
$excludeImageAttributes = preg_replace('/([^\\\\]\\K\\/|^\\/)/', '\\/', $excludeImageAttributes);
$excludeImageAttributes = explode(',', $excludeImageAttributes);
$excludeImageAttributes = array_map('trim', $excludeImageAttributes);
$excludeImageAttributes = implode(".*|.*", $excludeImageAttributes);
$excludeImageAttributes = '/(.*data-nowebp=\"true\".*|.*\/media\/captcha\/.*|.*' .
$excludeImageAttributes . '.*)/mi';
} else {
$excludeImageAttributes = '/(.*data-nowebp=\"true\".*|.*\/media\/captcha\/.*)/mi';
}
return $excludeImageAttributes;
}
/**
* Convert Image
*
* @param string $imageUrl
* @param string $htmlTag
* @param string $customSrcSetTag
* @param LayoutInterface $layout
*
* @return bool|string
*/
private function convertImage($imageUrl, $htmlTag, $customSrcSetTag, LayoutInterface $layout)
{
$lazyload = false;
if ($customSrcTag = $this->helper->getCustomSrcTag()) {
$expression = '/('.$customSrcTag.')=(\"|' . "\')([^<]+\.(png|jpg|jpeg))/mU";
if (preg_match_all($expression, $htmlTag, $match, PREG_OFFSET_CAPTURE)) {
$lazyload = true;
$imageUrl = $match[3][0][0];
}
}
$webpUrl = $this->helper->convert($imageUrl);
/**
* Skip when extension can not convert the image
*/
if ($webpUrl === $imageUrl) {
return false;
}
if ($lazyload) {
$pictureTag = $this->getPicture($layout)
->setOriginalImage($imageUrl)
->setWebpImage($webpUrl)
->setOriginalTag($htmlTag)
->setCustomSrcTag($customSrcTag)
->setCustomSrcSetTag($customSrcSetTag)
->toHtml();
} else {
$pictureTag = $this->getPicture($layout)
->setOriginalImage($imageUrl)
->setWebpImage($webpUrl)
->setOriginalTag($htmlTag)
->toHtml();
}
return $pictureTag;
}
}
<?php
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
namespace Jajuma\WebpImages\Setup\Patch\Data;
use Magento\Framework\App\ResourceConnection;
use Magento\Framework\Setup\Patch\DataPatchInterface;
use Magento\Framework\App\Config\Storage\WriterInterface;
use Magento\Framework\Setup\ModuleDataSetupInterface;
use Magento\Framework\App\Config\ScopeConfigInterface;
use Magento\Framework\Module\Dir;
class SetCwebpBinaryPath implements DataPatchInterface
{
/**
* @var ModuleDataSetupInterface
*/
private $moduleDataSetup;
/**
* @var WriterInterface
*/
protected $configWriter;
/**
* @var ScopeConfigInterface
*/
protected $scopeConfig;
/**
* @var Dir
*/
protected $dir;
/**
* PatchInitial constructor.
* @param ModuleDataSetupInterface $moduleDataSetup
* @param WriterInterface $configWriter
* @param ScopeConfigInterface $scopeConfig
* @param Dir $dir
*/
public function __construct(
ModuleDataSetupInterface $moduleDataSetup,
WriterInterface $configWriter,
ScopeConfigInterface $scopeConfig,
Dir $dir
) {
$this->moduleDataSetup = $moduleDataSetup;
$this->configWriter = $configWriter;
$this->scopeConfig = $scopeConfig;
$this->dir = $dir;
}
/**
* {@inheritdoc}
*/
public function apply()
{
if(!$this->scopeConfig->getValue('webp/advance_mode/path_to_cwebp')) {
$this->configWriter->save(
'webp/advance_mode/path_to_cwebp',
$this->dir->getDir('Jajuma_WebpImages') . '/bin/cwebp'
);
}
}
/**
* {@inheritdoc}
*/
public static function getDependencies()
{
return [];
}
/**
* {@inheritdoc}
*/
public function getAliases()
{
return [];
}
}
{
"name": "jajuma/module-webpimages",
"description": "WebP Optimized Images",
"type": "magento2-module",
"license": "MIT",
"version": "2.1.11",
"authors": [
{
"name": "JaJuMa",
"email": "info@jajuma.de"
}
],
"minimum-stability": "dev",
"require": {},
"autoload": {
"psr-4": {
"Jajuma\\WebpImages\\": ""
},
"files": [
"registration.php"
]
}
}
\ No newline at end of file
<?xml version="1.0" ?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:Acl/etc/acl.xsd">
<acl>
<resources>
<resource id="Magento_Backend::admin">
<resource id="Magento_Backend::stores">
<resource id="Magento_Backend::stores_settings">
<resource id="Magento_Config::config">
<resource id="Jajuma_General::jajuma_extensions" title="JaJuMa Extensions">
<resource id="Jajuma_WebpImages::config_jajuma_webp" title="JaJuMa - WebP Images" sortOrder="500"/>
</resource>
</resource>
</resource>
</resource>
</resource>
</resources>
</acl>
</config>
\ No newline at end of file
<?xml version="1.0"?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:ObjectManager/etc/config.xsd">
<type name="Magento\Config\Model\Config">
<plugin name="validate_webp_custom_command_before_save" type="Jajuma\WebpImages\Plugin\ConfigPlugin" sortOrder="1"/>
</type>
<type name="Magento\Backend\Model\Menu\Item">
<plugin name="jajumaExtensionsLink" type="Jajuma\WebpImages\Plugin\Backend\Model\Menu\Item\ExtensionsLinkPlugin"/>
</type>
<type name="Magento\Backend\Block\Menu">
<plugin name="jajumaExtensionsLinkJs" type="Jajuma\WebpImages\Plugin\Backend\Block\Menu\AddLinkJs"/>
</type>
</config>
\ No newline at end of file
<?xml version="1.0" ?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2022-present JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:module:Magento_Backend:etc/menu.xsd">
<menu>
<update id="Jajuma::top_level_extensions"
module="Jajuma_WebpImages"
resource="Jajuma_General::jajuma_extensions"
sortOrder="65"
title="JaJuMa"/>
<add id="Jajuma_WebpImages::jajuma_webp_images"
title="WebP Optimized Images"
module="Jajuma_WebpImages"
sortOrder="500"
resource="Jajuma_WebpImages::config_jajuma_webp"
parent="Jajuma::top_level_extensions"/>
<add id="Jajuma_WebpImages::jajuma_webp_images_configuration"
title="Configuration"
module="Jajuma_WebpImages"
sortOrder="510"
parent="Jajuma_WebpImages::jajuma_webp_images"
action="admin/system_config/edit/section/webp"
resource="Jajuma_WebpImages::config_jajuma_webp"/>
<update id="Jajuma::extensions"
module="Jajuma_WebpImages"
resource="Jajuma_General::jajuma_extensions"
sortOrder="10000"
parent="Jajuma::top_level_extensions"
title="JaJuMa Extensions"/>
<update id="Jajuma_Extensions::extensions_link"
title="More JaJuMa Extensions" module="Jajuma_WebpImages"
sortOrder="10010"
action="webpimages/extensions"
parent="Jajuma::extensions"
resource="Jajuma_General::jajuma_extensions"/>
</menu>
</config>
\ No newline at end of file
<?xml version="1.0"?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:App/etc/routes.xsd">
<router id="admin">
<route id="webp" frontName="webp">
<module name="Jajuma_WebpImages" before="Magento_Backend" />
</route>
</router>
</config>
<?xml version="1.0" ?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:module:Magento_Config:etc/system_file.xsd">
<system>
<tab id="jajuma" sortOrder="400" class="jajuma-tab" translate="label">
<label>Jajuma</label>
</tab>
<section id="webp" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="1" translate="label">
<label>WebP Image Optimization</label>
<tab>jajuma</tab>
<resource>Jajuma_WebpImages::config_jajuma_webp</resource>
<group id="extension_info" translate="label" type="text" sortOrder="10" showInDefault="1" showInWebsite="1" showInStore="1">
<label>Extension Information</label>
<attribute type="expanded">1</attribute>
<field id="extension_info_text" translate="label comment" sortOrder="10" showInDefault="1" showInWebsite="1" showInStore="1" type="text">
<frontend_model>Jajuma\WebpImages\Block\System\Config\Form\Jajuma\Module</frontend_model>
<comment><![CDATA[
<div style="margin:auto;padding:10px;background-color:#f1f1f1;display:inline-block;width:97%;">
<div class="box-img" style="width: 20%;float: left;">
<a href="https://www.jajuma.de/en" title="Magento 2 WebP Optimized Images Extension" target="_blank">
<div class="jajuma-extlogo"></div>
</a>
<strong style="font-size: 10px;">Copyright ©
<script>var today = new Date();
document.write(today.getFullYear());</script>
<a href="https://www.jajuma.de/en" target="_blank">www.jajuma.de</a>
</strong>
<div style="margin-top: 5px;">
<a href="https://www.jajuma.de/en/jajuma-shop/online-shop-with-magento-2-and-hyva-themes" target="_blank">
<svg xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 400 400"
class="w-10 h-10" width="25" height="25" style="position: relative;top: 5px;">
<rect width="400" height="400" fill="#F6F7FF"></rect>
<path d="M222.131 112.458L237.41 66.8783H290.403L275.043 112.458H222.131Z" fill="#14FFAF"></path>
<path fill-rule="evenodd" clip-rule="evenodd" d="M158.026 174.716C172.966 162.84 191.165
155.37 211.659 155.37C253.609 155.37 294.409 182.196 294.409
250.953V336H240.776V250.953C240.776 215.516 223.929 202.107 200.933 202.107C181.969
202.107 168.181 213.026 160.516 224.517V336H106.692V115.336H74L91.2077
64H158.026V174.716ZM272.988 109.567L286.407 69.7565H239.485L226.132
109.567H272.988ZM235.338 64H294.409L277.112 115.336H218.13L235.338 64Z"
fill="#0A23B9"></path>
</svg>
<span style="font-size: 16px;font-weight: 900;"> Theme Compatible</span>
</a>
</div>
</div>
<div style="width: 77%;float: left;padding-left: 3%;">
<div class="name" style="font-size: 16px;">
<b>Magento 2 WebP Optimized Images</b> by <a href="https://www.jajuma.de/en" target="_blank"><b>JaJuMa</b></a>
<div class="jajuma-minilogo"></div>
</div>
<hr>
<br>
<div class="more"><b>More Information About This Extension:</b>
<br>
<div><a href="https://www.jajuma.de/en/jajuma-develop/extensions/webp-optimized-images-extension-for-magento-2"
target="_blank"><svg xmlns="http://www.w3.org/2000/svg" fill="currentColor"
viewBox="0 0 576 512" class="w-10 h-10" width="15" height="15"
style="position: relative;top: 2px;">
<!--! Font Awesome Free 6.0.0 by @fontawesome - https://fontawesome.com License -
https://fontawesome.com/license/free (Icons: CC BY 4.0,
Fonts: SIL OFL 1.1, Code: MIT License) Copyright 2022 Fonticons, Inc. -->
<path d="M575.8 255.5C575.8 273.5 560.8 287.6 543.8 287.6H511.8L512.5 447.7C512.5
450.5 512.3 453.1 512 455.8V472C512 494.1 494.1 512 472 512H456C454.9 512 453.8
511.1 452.7 511.9C451.3 511.1 449.9 512 448.5 512H392C369.9 512 352 494.1 352
472V384C352 366.3 337.7 352 320 352H256C238.3 352 224 366.3 224 384V472C224 494.1
206.1 512 184 512H128.1C126.6 512 125.1 511.9 123.6 511.8C122.4 511.9 121.2 512 120
512H104C81.91 512 64 494.1 64 472V360C64 359.1 64.03 358.1 64.09 357.2V287.6H32.05C14.02
287.6 0 273.5 0 255.5C0 246.5 3.004 238.5 10.01 231.5L266.4 8.016C273.4 1.002 281.4 0
288.4 0C295.4 0 303.4 2.004 309.5 7.014L564.8 231.5C572.8 238.5 576.9 246.5 575.8
255.5L575.8 255.5z"></path></svg> Website</a> |
<a href="https://www.jajuma.de/sites/default/files/ckfinder/userfiles/images/jajuma-develop/webp-optimized-images-magento/JaJuMa_WebP_Optimized_Images_Manual_v003.pdf"
target="_blank"><svg xmlns="http://www.w3.org/2000/svg" fill="currentColor"
viewBox="0 0 448 512" class="w-10 h-10" width="15" height="15"
style="top: 2px;position: relative;">
<!--! Font Awesome Free 6.0.0 by @fontawesome - https://fontawesome.com License -
https://fontawesome.com/license/free (Icons: CC BY 4.0,
Fonts: SIL OFL 1.1, Code: MIT License) Copyright 2022 Fonticons, Inc. -->
<path d="M448 336v-288C448 21.49 426.5 0 400 0H96C42.98 0 0 42.98 0 96v320c0 53.02
42.98 96 96 96h320c17.67 0 32-14.33 32-31.1c0-11.72-6.607-21.52-16-27.1v-81.36C441.8
362.8 448 350.2 448 336zM143.1 128h192C344.8 128 352 135.2 352 144C352 152.8 344.8 160
336 160H143.1C135.2 160 128 152.8 128 144C128 135.2 135.2 128 143.1 128zM143.1
192h192C344.8 192 352 199.2 352 208C352 216.8 344.8 224 336 224H143.1C135.2 224 128
216.8 128 208C128 199.2 135.2 192 143.1 192zM384 448H96c-17.67 0-32-14.33-32-32c0-17.67
14.33-32 32-32h288V448z"></path></svg> Manual</a> |
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/webp-optimized-images-extension-for-magento-2/webp-images-faq"
target="_blank"><svg xmlns="http://www.w3.org/2000/svg" fill="currentColor"
viewBox="0 0 512 512" class="w-10 h-10" width="15" height="15"
style="position: relative;top: 2px;">
<!--! Font Awesome Free 6.0.0 by @fontawesome - https://fontawesome.com License -
https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1,
Code: MIT License) Copyright 2022 Fonticons, Inc. --><path d="M256 0C114.6 0 0 114.6 0
256s114.6 256 256 256s256-114.6 256-256S397.4 0 256 0zM256 464c-114.7
0-208-93.31-208-208S141.3 48 256 48s208 93.31 208 208S370.7 464 256 464zM256 336c-18
0-32 14-32 32s13.1 32 32 32c17.1 0 32-14 32-32S273.1 336 256 336zM289.1 128h-51.1C199
128 168 159 168 198c0 13 11 24 24 24s24-11 24-24C216 186 225.1 176 237.1 176h51.1C301.1
176 312 186 312 198c0 8-4 14.1-11 18.1L244 251C236 256 232 264 232 272V288c0 13 11 24
24 24S280 301 280 288V286l45.1-28c21-13 34-36 34-60C360 159 329 128 289.1 128z"></path>
</svg> FAQ</a> |
<a href="https://demo.extension.jajuma.de/"
target="_blank"><svg xmlns="http://www.w3.org/2000/svg" fill="currentColor"
viewBox="0 0 512 512" class="w-10 h-10" width="15" height="15"
style="position: relative;top: 2px;"><!--! Font Awesome Free 6.0.0 by @fontawesome -
https://fontawesome.com License - https://fontawesome.com/license/free
(Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License) Copyright 2022 Fonticons,
Inc. --><path d="M500.3 443.7l-119.7-119.7c27.22-40.41 40.65-90.9 33.46-144.7C401.8
87.79 326.8 13.32 235.2 1.723C99.01-15.51-15.51 99.01 1.724 235.2c11.6 91.64 86.08
166.7 177.6 178.9c53.8 7.189 104.3-6.236 144.7-33.46l119.7 119.7c15.62 15.62 40.95
15.62 56.57 0C515.9 484.7 515.9 459.3 500.3 443.7zM79.1 208c0-70.58 57.42-128
128-128s128 57.42 128 128c0 70.58-57.42 128-128 128S79.1 278.6 79.1 208z">
</path></svg> Luma Demo</a> |
<a href="https://hyva.extension.jajuma.de/"
target="_blank"><svg xmlns="http://www.w3.org/2000/svg" fill="currentColor"
viewBox="0 0 400 400" class="w-10 h-10" width="20" height="20" style="position:
relative;top: 5px;"> <rect fill="#fff"></rect> <path d="M222.131 112.458L237.41
66.8783H290.403L275.043 112.458H222.131Z"></path> <path fill-rule="evenodd"
clip-rule="evenodd" d="M158.026 174.716C172.966 162.84 191.165 155.37 211.659
155.37C253.609 155.37 294.409 182.196 294.409 250.953V336H240.776V250.953C240.776
215.516 223.929 202.107 200.933 202.107C181.969 202.107 168.181 213.026 160.516
224.517V336H106.692V115.336H74L91.2077 64H158.026V174.716ZM272.988 109.567L286.407
69.7565H239.485L226.132 109.567H272.988ZM235.338 64H294.409L277.112
115.336H218.13L235.338 64Z"></path></svg> Hyvä Theme Demo</a>
<div style="float: right;">Installed Version: <b>2.1.11</b></div>
</div>
</div>
<hr>
<div class="info"><b>Discover More Ideas To Improve Your Store:</b>
<div class="more-ideas-list">
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/ultimate-image-optimizer-extension-for-magento-2"
target="_blank">Ultimate Image Optimizer</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/shariff-social-share-buttons-extension-for-magento-2"
target="_blank">Shariff Social Share Buttons</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/prg-pattern-link-masking-for-magento-2"
target="_blank">PRG Pattern Link Masking</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/video-widget-gdpr-extension-for-magento-2"
target="_blank">Video Widget</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/page-preload-extension-for-magento-2"
target="_blank">Page Preload</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/matomo-analytics-extension-for-magento-2"
target="_blank">Matomo Analytics</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/honey-spam-anti-spam-extension-for-magento-2"
target="_blank">Honey Spam Anti-Spam</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/customer-registration-reminder-and-cleanup-extension-for-magento-2"
target="_blank">Customer Registration Reminder & Cleanup</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/category-grid-callouts-extension-for-magento-2"
target="_blank">Category Grid Callouts</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/customer-satisfaction-feedback-extension-for-magento-2"
target="_blank">Customer Satisfaction Feedback</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/customer-navigation-manager-extension-for-magento-2"
target="_blank"><b>Free:</b> Customer Navigation Manager</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/font-awesome-icons-for-hyva-themes-extension"
target="_blank"><b>Free:</b> Awesome Hyvä</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/dynamic-shipping-tax-extension-for-magento-2"
target="_blank"><b>Free:</b> Dynamic Shipping Tax</a>
<a href="https://www.jajuma.de/en/jajuma-develop/extensions/hyva-faq-widget-extension-for-hyva-themes"
target="_blank"><b>Free:</b> Hyvä FAQ Widget</a>
</div>
</div>
<hr>
</div>
</div>
]]></comment>
</field>
</group>
<group id="setting" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="10" translate="label">
<label>General Configuration</label>
<field id="enable" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="10" translate="label" type="select">
<label>Optimize Image</label>
<source_model>Magento\Config\Model\Config\Source\Yesno</source_model>
</field>
<field id="check_transparency" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="15" translate="label" type="select">
<label>Disable with transparent images</label>
<source_model>Magento\Config\Model\Config\Source\Yesno</source_model>
</field>
<field id="clear_cache_webp" translate="label comment" type="button" sortOrder="1000" showInDefault="1" showInWebsite="1" showInStore="1">
<label>Clear Webp Image</label>
<comment><![CDATA[This button will clear all the generated webp images.]]></comment>
<frontend_model>Jajuma\WebpImages\Block\Adminhtml\System\Config\ClearButton</frontend_model>
</field>
</group>
<group id="advance_mode" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="20" translate="label">
<label>Conversion Configuration</label>
<field id="convert_tool" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="20" translate="label" type="select">
<label>Conversion Tool</label>
<comment>Select the tool to be used for WebP image conversion</comment>
<source_model>Jajuma\WebpImages\Model\Config\Source\Tool</source_model>
</field>
<field id="path_to_cwebp" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="20" translate="label" type="text">
<label>Path to cwebp</label>
<validate>validate-cwebp-path</validate>
<comment><model>Jajuma\WebpImages\Model\Config\Comment</model></comment>
<depends>
<field id="convert_tool">cwebp</field>
</depends>
</field>
<field id="path_to_imagemagick" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="20" translate="label" type="text">
<label>Path to imagemagick</label>
<validate>validate-imagemagick-path</validate>
<comment>Define the path of imagemagick command or leave it empty to use global command "convert". Example: "/usr/local/bin/convert"</comment>
<depends>
<field id="convert_tool">convert</field>
</depends>
</field>
<field id="quality" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="30" translate="label" type="text">
<label>WebP Quality</label>
<validate>validate-number validate-zero-or-greater validate-digits</validate>
<comment>Define the compression factor applied for webp conversion (from 0 to 100)</comment>
</field>
<field id="cwebp_command" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="100" translate="label" type="text">
<label>Cwebp Custom Command</label>
<validate>validate-cwebp-command</validate>
<comment><![CDATA[Example command: -alpha_q 100 -z 9 -m 6 -segments 4 -sns 80 -f 25 -sharpness 0 -strong -pass 10 -mt -alpha_method 1 -alpha_filter fast -o]]></comment>
<depends>
<field id="convert_tool">cwebp</field>
</depends>
</field>
<field id="imagemagick_command" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="200" translate="label" type="text">
<label>Imagemagick Custom Command</label>
<validate>validate-imagemagick-command</validate>
<comment><![CDATA[Example command: -quality 100 -define webp:lossless=true,method=6]]></comment>
<depends>
<field id="convert_tool">convert</field>
</depends>
</field>
<field id="test_button" translate="label comment" type="button" sortOrder="1000" showInDefault="1" showInWebsite="1" showInStore="1">
<frontend_model>Jajuma\WebpImages\Block\Adminhtml\System\Config\TestButton</frontend_model>
</field>
</group>
<group id="native_lazy" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="30" translate="label">
<label>Native Lazy Loading</label>
<field id="enable_native_lazy" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="10" translate="label" type="select">
<label>Enable Native Lazy Loading</label>
<comment><![CDATA[If enabled, ' loading="lazy" ' will be added for images converted.]]></comment>
<source_model>Magento\Config\Model\Config\Source\Yesno</source_model>
</field>
<field id="exclude_native_lazy" translate="label comment" sortOrder="100" showInDefault="1" showInWebsite="1" showInStore="1" type="text">
<label>Native Lazy Loading Blacklist</label>
<comment><![CDATA[<span><b>Exclude images from Native Lazy Loading</b> by a list of comma separated strings (or RegEx).<br>
If the img tag match with any of the strings above, the ' loading="lazy" ' won't be added to that img tag.</span>]]></comment>
</field>
</group>
<group id="professional_mode" showInDefault="1" showInStore="1" showInWebsite="1" sortOrder="40" translate="label">
<label>Advanced Configuration</label>
<field id="message" translate="label comment" sortOrder="10" showInDefault="1" showInWebsite="1" showInStore="1" type="text">
<frontend_model>Jajuma\WebpImages\Block\Adminhtml\System\Config\AdvancedConfiguration</frontend_model>
<comment><![CDATA[
<div class="messages">
<div class="message message-warning message-demo-mode">
<b>Note: </b><br><span>For most sites following configs are not needed.<br>
Please use only after checking the manual <a href="https://www.jajuma.de/en/jajuma-develop/extensions/webp-optimized-images-extension-for-magento-2" target="_blank">(Link)</a> and understand what you are doing.
</span> </div>
</div>
]]></comment>
</field>
<field id="exclude_img" translate="label comment" sortOrder="100" showInDefault="1" showInWebsite="1" showInStore="1" type="text">
<label>Blacklist</label>
<comment><![CDATA[<span><b>Exclude images</b> from WebP Conversion by a list of comma separated strings (or RegEx).<br>
If the img tag match with any of the strings above, that img tag won't be converted.<br>
Note: img-tags with <i>data-nowebp="true"</i> are excluded from conversion by default.</span>]]></comment>
</field>
<field id="src_tag" translate="label comment" sortOrder="200" showInDefault="1" showInWebsite="1" showInStore="1" type="text">
<label>Custom src-tag</label>
<comment><![CDATA[<span>By default we look at the <b>src</b> attribute to get the image URL and use this for WebP conversion.<br>
If you use some custom attribute for img URL, e. g. in case you are using lazyload on your site, you can use this config to have this converted first.<br>
Example: If your lazyload function uses <b>data-src</b>, just input <i>data-src</i> into this config.<br>
For img tags having a data-src we will then use the data-src image URL for WebP conversion.<br>
For img tags having no data-src, we will still use src attribute for conversion.
</span>]]></comment>
</field>
<field id="srcset_tag" translate="label comment" sortOrder="300" showInDefault="1" showInWebsite="1" showInStore="1" type="text">
<label>Custom srcset-tag</label>
<comment><![CDATA[<span>By default we add picture tags using <b>srcset</b> tags.<br>
You can use this config to change this behaviour, e. g. in case you are using lazyload on your site.<br>
Example: If your lazyload function uses <b>data-srcset</b>, just input <i>data-srcset</i> into this config.<br>
The extension will then add picture tags using data-srcset.<br>
!!! Please ensure your lazyload script does support lazyloading for picture-tags !!!
</span>]]></comment>
</field>
</group>
</section>
</system>
</config>
\ No newline at end of file
<?xml version="1.0" ?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="urn:magento:module:Magento_Store:etc/config.xsd">
<default>
<webp>
<setting>
<enable>1</enable>
<quality>75</quality>
</setting>
</webp>
</default>
</config>
\ No newline at end of file
<?xml version="1.0"?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="urn:magento:framework:ObjectManager/etc/config.xsd">
<type name="Magento\Catalog\Block\Product\View\Gallery">
<plugin name="jajuma-webpimages-product-view-gallery" type="Jajuma\WebpImages\Plugin\Product\View\GalleryPlugin" sortOrder="5"/>
</type>
<type name="Magento\Framework\View\LayoutInterface">
<plugin name="jajuma-webpimages-replace-image-tag" type="Jajuma\WebpImages\Plugin\ReplaceImageTag"/>
</type>
</config>
\ No newline at end of file
<?xml version="1.0"?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="../../../../../../lib/internal/Magento/Framework/App/etc/routes.xsd">
<router id="standard">
<route id="webp" frontName="webp">
<module name="Jajuma_WebpImages" />
</route>
</router>
</config>
\ No newline at end of file
<?xml version="1.0" ?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="urn:magento:framework:Module/etc/module.xsd">
<module name="Jajuma_WebpImages" setup_version="2.1.4"/>
</config>
<?php declare(strict_types = 1);
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
\Magento\Framework\Component\ComponentRegistrar::register(
\Magento\Framework\Component\ComponentRegistrar::MODULE,
'Jajuma_WebpImages',
__DIR__
);
<?xml version="1.0"?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<page xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:View/Layout/etc/page_configuration.xsd">
<body>
<referenceContainer name="header">
<block class="Magento\Backend\Block\Template" template="Jajuma_WebpImages::system/config/js.phtml"/>
</referenceContainer>
</body>
</page>
\ No newline at end of file
<?xml version="1.0"?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<page xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:View/Layout/etc/page_configuration.xsd">
<head>
<css src="Jajuma_WebpImages::css/main.css"/>
</head>
</page>
<?xml version="1.0"?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<page xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" layout="admin-1column" xsi:noNamespaceSchemaLocation="urn:magento:framework:View/Layout/etc/page_configuration.xsd">
<head>
<css src="Jajuma_WebpImages::css/twentytwenty.css"/>
</head>
<body>
<referenceContainer name="header" remove="true"/>
<referenceContainer name="page.menu" remove="true"/>
<referenceContainer name="page.breadcrumbs" remove="true"/>
<referenceContainer name="footer" remove="true"/>
<referenceContainer name="content">
<block class="Jajuma\WebpImages\Block\Adminhtml\ConversionTool\Test\Result" name="conversion.tool.test.result" template="Jajuma_WebpImages::conversion_tool/test/result.phtml"/>
</referenceContainer>
</body>
</page>
var config = {
paths: {
'twentytwenty': 'Jajuma_WebpImages/js/lib/jquery.twentytwenty',
'eventmove': 'Jajuma_WebpImages/js/lib/jquery.event.move',
},
shim: {
'twentytwenty': {
deps: ['jquery', 'eventmove']
}
}
};
\ No newline at end of file
<script>
require([
'jquery'
], function ($) {
$(document).ready(function () {
$('a[href*="https://www.jajuma.de"]').attr('target', '_blank');
});
});
</script>
<?php
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
?>
<style>
body .menu-wrapper {
display: none;
}
.webp-container-compare {
max-width: 700px;
margin: 0 auto;
}
body {
background: #ffffff;
}
</style>
<?php
/**
* @var $block \Jajuma\WebpImages\Block\Adminhtml\ConversionTool\Test\Result
*/
$result = $block->convert();
if (is_array($result)):
?>
<script type="text/javascript">
require([
'jquery',
'twentytwenty',
], function () {
jQuery(document).ready(function() {
jQuery("#compare-image").twentytwenty({
default_offset_pct: 0.5,
orientation: 'horizontal',
before_label: 'ORIGINAL',
after_label: 'WEBP',
no_overlay: false,
move_slider_on_hover: false,
move_with_handle_only: true,
click_to_move: false
});
});
});
</script>
<div id="test-success-message">
<div class="messages">
<div class="message message-success success">
<div data-ui-id="messages-message-success">
<p><?= /* @noEscape */ __('Success!') ?></p>
<p><?= /* @noEscape */ __('Selected conversion tool works.') ?></p>
<p><?= /* @noEscape */__('WebP Image File was created
using configured quality level / custom conversion command.') ?></p>
<p><?= /* @noEscape */ __('See below for comparison Original vs WebP:') ?></p>
</div>
</div>
</div>
</div>
<?php if (strpos($result['original'], 'Jajuma_WebpImages/images/test.png') !== false): ?>
<div id="test-notice-message">
<div class="messages">
<div class="message message-notice notice">
<div data-ui-id="messages-message-notice">
<p><?= /* @noEscape */__('Product ID used for testing
is invalid or that product has no image.<br>') ?>
<?= /* @noEscape */__('Hence, below we can only show some sample image.<br>') ?>
<?= /* @noEscape */ __('Please use another Product ID,
so we can show the conversion result with your own image.') ?></p>
</div>
</div>
</div>
</div>
<?php endif; ?>
<div class="webp-container-compare">
<div id="compare-image" class="twentytwenty-container">
<!-- The before image is first -->
<img src="<?= /* @noEscape */ $result['original'] ?>" />
<!-- The after image is last -->
<picture>
<source srcset="<?= /* @noEscape */ $result['webp'] ?>" type="image/webp">
<img src="<?= /* @noEscape */ $block->getUnsupportWebpImage() ?>" />
</picture>
</div>
</div>
<?php else: ?>
<div id="test-notice-message">
<div class="messages">
<div class="message message-warning">
<div data-ui-id="message message-warning">
<?php if ($message = $block->getError()): ?>
<p><?= /* @noEscape */ $message ?></p>
<?php else: ?>
<p><?= /* @noEscape */ __('Sorry, Image conversion failed.') ?></p>
<p><?= /* @noEscape */__('Seems selected conversion tool is not available on your server.') ?></p>
<p><?= /* @noEscape */ __('Can you please:') ?></p>
<ul style="margin-left:20px;">
<li><?= /* @noEscape */ __('Try another conversion tool') ?></li>
<li><?= /* @noEscape */ __('And/or test with another image') ?></li>
<li><?= /* @noEscape */ __('Check the extension manual
regrading conversion tool requirements') ?></li>
</ul>
<a
href="https://www.jajuma.de/en/jajuma-develop/
extensions/webp-optimized-images-extension-for-magento-2"
target="_blank">
Link To Manual
</a>
<?php endif; ?>
</div>
</div>
</div>
</div>
<?php endif; ?>
<?php
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
?>
<script type="text/javascript">
require([
'jquery',
'Magento_Ui/js/modal/confirm'
], function ($, confirm) {
window.clearWebpImage = function() {
confirm({
title: '<?= /* @noEscape */__("Clear Image") ?>',
content: '<?= /* @noEscape */__("Do you want to delete all generated webp images") ?>',
actions: {
confirm: function(){ window.location.href = "<?= /* @noEscape */ $block->getAdminUrl() ?>"},
cancel: function(){ return false },
always: function(){ return false }
}
});
}
});
</script>
<?= $block->getButtonHtml() ?>
\ No newline at end of file
<?php
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
?>
<script type="text/javascript">
requirejs([
'jquery',
'jquery/ui',
'jquery/validate',
'mage/translate'
], function($){
'use strict';
$.validator.addMethod(
"validate-cwebp-command",
function(value, element) {
return this.optional(element) || <?= \Jajuma\WebpImages\Helper\Data::REGX_CWEBP ?>.test(value);
},
$.mage.__("Incorrect cwebp custom command. Custom Command must only include " +
"underscore (_), minus (-), " +
"space ( ) and alphanumeric characters.")
);
$.validator.addMethod(
"validate-imagemagick-command",
function(value, element) {
return this.optional(element) || <?= \Jajuma\WebpImages\Helper\Data::REGX_IMAGEMAGICK ?>.test(value);
},
$.mage.__("Incorrect ImageMagick custom command. Custom Command must only include " +
"underscore (_), minus (-), space ( ), comma (,), colon (:), " +
"equals sign (=) and alphanumeric characters.")
);
$.validator.addMethod(
"validate-cwebp-path",
function(value, element) {
return this.optional(element) || <?= \Jajuma\WebpImages\Helper\Data::REGX_CWEBP_PATH ?>.test(value);
},
$.mage.__("Incorrect cwebp path. Path must only include " +
"underscore (_), minus (-), dot (.), slash(/) and alphanumeric characters.")
);
$.validator.addMethod(
"validate-imagemagick-path",
function(value, element) {
return this.optional(element)
|| <?= \Jajuma\WebpImages\Helper\Data::REGX_IMAGEMAGICK_PATH ?>.test(value);
},
$.mage.__("Incorrect ImageMagick path. Path must only include " +
"underscore (_), minus (-), dot (.), slash(/) and alphanumeric characters.")
);
});
</script>
<?php
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
/**
* @var $block Jajuma\WebpPlus\Block\Adminhtml\System\Config\TestButton
*/
?>
<style>
.test-conversion-tool-wrapper {
display: flex;
}
.test-conversion-tool-wrapper .product-input-wrapper {
margin-left: 10px;
display: flex;
}
.test-conversion-tool-wrapper .product-input-wrapper .product {
width: 80px;
}
.test-conversion-tool-wrapper .product-input-wrapper .tooltip {
line-height: 26px;
}
</style>
<script type="text/javascript">
require([
'jquery',
'Magento_Ui/js/modal/modal',
'Magento_Ui/js/modal/alert'
], function ($, modal, alert) {
window.conversionToolTest = function() {
if (!$.isNumeric($('#webp_advance_mode_quality').val())) {
alert({
title: 'Error',
content: "<?= /* @noEscape */ __('Quality should be a numeric value') ?>",
actions: {
always: function(){}
}
});
return;
}
$('body').trigger('processStart');
$('#iFrame').remove();
let productId = $('.product-input-wrapper .product').val();
let postData = {form_key: FORM_KEY, product: productId};
configForm.find('[id^=webp_advance_mode]').find(':input').serializeArray().map(function(field) {
var name = field.name.match(/groups\[advance_mode\]?(\[groups\]\[debug\])?\[fields\]\[(.*)\]\[value]/);
if(name && name.length === 3){
postData[name[2]] = field.value;
}
});
let options ={
type: 'popup',
title: '<?= /* @noEscape */ __('Test Conversion Tool') ?>',
responsive: true,
innerScroll: false,
buttons: [{
text: $.mage.__('Close'),
class: 'close-modal',
click: function () {
this.closeModal();
}
}]
};
let popup = modal(options, $('#quickViewContainer'));
let modalContainer = $("#quickViewContainer");
let iframe = $('<iframe />', {
id: 'iFrame',
src: "<?= /* @noEscape */ $block->getAdminUrl() ?>?" + $.param( postData, true )
});
modalContainer.html(iframe);
$('#iFrame').on("load", function () {
$('body').trigger('processStop');
modalContainer.addClass("quick-view-image");
modalContainer.modal('openModal');
this.style.height = this.contentWindow.document.body.scrollHeight+ 10 + 'px';
this.style.border = '0';
this.style.width = '100%';
});
}
});
</script>
<div class="test-conversion-tool-wrapper">
<div class="button-wrapper">
<?= $block->getButtonHtml() ?>
</div>
<div class="product-input-wrapper">
<input type="text" name="product_id" class="product" value="<?= /* @noEscape */ $block->getLatestProductId()?>">
<div class="tooltip">
<span class="help"><span></span></span>
<div class="tooltip-content">Fill your product id to test the conversion tool</div>
</div>
</div>
</div>
<div id="quickViewContainer">
</div>
li#menu-jajuma-top-level-extensions > a:before {
background-image: url('');
background-size: contain;
background-repeat: no-repeat;
margin-bottom: 10px;
content: '';
background-position: center;
filter: brightness(1.7);
opacity: .65;
}
li#menu-jajuma-top-level-extensions:hover > a:before {
opacity: 1;
}
li[class^=item-jajuma] strong.submenu-group-title {
display: block!important;
}
li.item-extensions.parent.level-1 {
border-top: 1px solid #aaa6a0;
}
#menu-jajuma-top-level-extensions .item-extensions-link span:before {
background-image: url(../images/idea.svg);
display: inline-block;
content: '';
width: 25px;
height: 25px;
margin-right: 15px;
vertical-align: middle;
background-size: contain;
background-repeat: no-repeat;
}
#nav li#menu-jajuma-top-level-extensions .submenu .submenu-title {
background-image: url('');
background-size: contain;
background-repeat: no-repeat;
height: 55px;
font-size: 0px;
margin-bottom: 15px;
filter: brightness(1.3);
}
li.item-jajuma-webp-images.parent.level-1 {
min-width: 280px;
}
.config-nav-block.jajuma-tab strong:before, .admin__menu [data-ui-id^=menu-jajuma-].item-jajuma-webp-images.parent.level-1 .submenu-group-title span:before {
background-image: url(../images/images.svg);
display: inline-block;
content: '';
width: 30px;
height: 30px;
margin-right: 15px;
vertical-align: middle;
background-size: contain;
background-repeat: no-repeat;
position: relative;
top: -2px;
}
.config-nav-block.jajuma-tab strong:before, .admin__menu [data-ui-id^=menu-jajuma-] .submenu-group-title span:before {
background: url('') no-repeat center;
width: 30px;
height: 30px;
content: '';
display: inline-block;
margin-right: 15px;
vertical-align: middle;
background-size: contain;
}
#webp_extension_info .jajuma-extlogo {
background-image: url('');
width: 185px;
height: 185px;
background-size: cover;
}
.jajuma-minilogo {
background-image: url('');
width: 75px;
height: 25px;
background-size: cover;
margin-left: 10px;
float: right;
}
.more-ideas-list {
display: flex;
flex-wrap: wrap;
display: grid;
grid-template-columns: repeat(auto-fill, minmax(250px, 1fr));
grid-gap: 3px;
}
.twentytwenty-horizontal .twentytwenty-handle:before, .twentytwenty-horizontal .twentytwenty-handle:after, .twentytwenty-vertical .twentytwenty-handle:before, .twentytwenty-vertical .twentytwenty-handle:after {
content: " ";
display: block;
background: white;
position: absolute;
z-index: 30;
-webkit-box-shadow: 0px 0px 12px rgba(51, 51, 51, 0.5);
-moz-box-shadow: 0px 0px 12px rgba(51, 51, 51, 0.5);
box-shadow: 0px 0px 12px rgba(51, 51, 51, 0.5); }
.twentytwenty-horizontal .twentytwenty-handle:before, .twentytwenty-horizontal .twentytwenty-handle:after {
width: 3px;
height: 9999px;
left: 50%;
margin-left: -1.5px; }
.twentytwenty-vertical .twentytwenty-handle:before, .twentytwenty-vertical .twentytwenty-handle:after {
width: 9999px;
height: 3px;
top: 50%;
margin-top: -1.5px; }
.twentytwenty-before-label, .twentytwenty-after-label, .twentytwenty-overlay {
position: absolute;
top: 0;
width: 100%;
height: 100%; }
.twentytwenty-before-label, .twentytwenty-after-label, .twentytwenty-overlay {
-webkit-transition-duration: 0.5s;
-moz-transition-duration: 0.5s;
transition-duration: 0.5s; }
.twentytwenty-before-label, .twentytwenty-after-label {
-webkit-transition-property: opacity;
-moz-transition-property: opacity;
transition-property: opacity; }
.twentytwenty-before-label:before, .twentytwenty-after-label:before {
color: white;
font-size: 13px;
letter-spacing: 0.1em; }
.twentytwenty-before-label:before, .twentytwenty-after-label:before {
position: absolute;
background: rgba(255, 255, 255, 0.2);
line-height: 38px;
padding: 0 20px;
-webkit-border-radius: 2px;
-moz-border-radius: 2px;
border-radius: 2px; }
.twentytwenty-horizontal .twentytwenty-before-label:before, .twentytwenty-horizontal .twentytwenty-after-label:before {
top: 50%;
margin-top: -19px; }
.twentytwenty-vertical .twentytwenty-before-label:before, .twentytwenty-vertical .twentytwenty-after-label:before {
left: 50%;
margin-left: -45px;
text-align: center;
width: 90px; }
.twentytwenty-left-arrow, .twentytwenty-right-arrow, .twentytwenty-up-arrow, .twentytwenty-down-arrow {
width: 0;
height: 0;
border: 6px inset transparent;
position: absolute; }
.twentytwenty-left-arrow, .twentytwenty-right-arrow {
top: 50%;
margin-top: -6px; }
.twentytwenty-up-arrow, .twentytwenty-down-arrow {
left: 50%;
margin-left: -6px; }
.twentytwenty-container {
-webkit-box-sizing: content-box;
-moz-box-sizing: content-box;
box-sizing: content-box;
z-index: 0;
overflow: hidden;
position: relative;
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none; }
.twentytwenty-container img {
max-width: 100%;
position: absolute;
top: 0;
display: block; }
.twentytwenty-container.active .twentytwenty-overlay, .twentytwenty-container.active :hover.twentytwenty-overlay {
background: rgba(0, 0, 0, 0); }
.twentytwenty-container.active .twentytwenty-overlay .twentytwenty-before-label,
.twentytwenty-container.active .twentytwenty-overlay .twentytwenty-after-label, .twentytwenty-container.active :hover.twentytwenty-overlay .twentytwenty-before-label,
.twentytwenty-container.active :hover.twentytwenty-overlay .twentytwenty-after-label {
opacity: 0; }
.twentytwenty-container * {
-webkit-box-sizing: content-box;
-moz-box-sizing: content-box;
box-sizing: content-box; }
.twentytwenty-before-label {
opacity: 0; }
.twentytwenty-before-label:before {
content: attr(data-content); }
.twentytwenty-after-label {
opacity: 0; }
.twentytwenty-after-label:before {
content: attr(data-content); }
.twentytwenty-horizontal .twentytwenty-before-label:before {
left: 10px; }
.twentytwenty-horizontal .twentytwenty-after-label:before {
right: 10px; }
.twentytwenty-vertical .twentytwenty-before-label:before {
top: 10px; }
.twentytwenty-vertical .twentytwenty-after-label:before {
bottom: 10px; }
.twentytwenty-overlay {
-webkit-transition-property: background;
-moz-transition-property: background;
transition-property: background;
background: rgba(0, 0, 0, 0);
z-index: 25; }
.twentytwenty-overlay:hover {
background: rgba(0, 0, 0, 0.5); }
.twentytwenty-overlay:hover .twentytwenty-after-label {
opacity: 1; }
.twentytwenty-overlay:hover .twentytwenty-before-label {
opacity: 1; }
.twentytwenty-before {
z-index: 20; }
.twentytwenty-after {
z-index: 10; }
.twentytwenty-handle {
height: 38px;
width: 38px;
position: absolute;
left: 50%;
top: 50%;
margin-left: -22px;
margin-top: -22px;
border: 3px solid white;
-webkit-border-radius: 1000px;
-moz-border-radius: 1000px;
border-radius: 1000px;
-webkit-box-shadow: 0px 0px 12px rgba(51, 51, 51, 0.5);
-moz-box-shadow: 0px 0px 12px rgba(51, 51, 51, 0.5);
box-shadow: 0px 0px 12px rgba(51, 51, 51, 0.5);
z-index: 40;
cursor: pointer; }
.twentytwenty-horizontal .twentytwenty-handle:before {
bottom: 50%;
margin-bottom: 22px;
-webkit-box-shadow: 0 3px 0 white, 0px 0px 12px rgba(51, 51, 51, 0.5);
-moz-box-shadow: 0 3px 0 white, 0px 0px 12px rgba(51, 51, 51, 0.5);
box-shadow: 0 3px 0 white, 0px 0px 12px rgba(51, 51, 51, 0.5); }
.twentytwenty-horizontal .twentytwenty-handle:after {
top: 50%;
margin-top: 22px;
-webkit-box-shadow: 0 -3px 0 white, 0px 0px 12px rgba(51, 51, 51, 0.5);
-moz-box-shadow: 0 -3px 0 white, 0px 0px 12px rgba(51, 51, 51, 0.5);
box-shadow: 0 -3px 0 white, 0px 0px 12px rgba(51, 51, 51, 0.5); }
.twentytwenty-vertical .twentytwenty-handle:before {
left: 50%;
margin-left: 22px;
-webkit-box-shadow: 3px 0 0 white, 0px 0px 12px rgba(51, 51, 51, 0.5);
-moz-box-shadow: 3px 0 0 white, 0px 0px 12px rgba(51, 51, 51, 0.5);
box-shadow: 3px 0 0 white, 0px 0px 12px rgba(51, 51, 51, 0.5); }
.twentytwenty-vertical .twentytwenty-handle:after {
right: 50%;
margin-right: 22px;
-webkit-box-shadow: -3px 0 0 white, 0px 0px 12px rgba(51, 51, 51, 0.5);
-moz-box-shadow: -3px 0 0 white, 0px 0px 12px rgba(51, 51, 51, 0.5);
box-shadow: -3px 0 0 white, 0px 0px 12px rgba(51, 51, 51, 0.5); }
.twentytwenty-left-arrow {
border-right: 6px solid white;
left: 50%;
margin-left: -17px; }
.twentytwenty-right-arrow {
border-left: 6px solid white;
right: 50%;
margin-right: -17px; }
.twentytwenty-up-arrow {
border-bottom: 6px solid white;
top: 50%;
margin-top: -17px; }
.twentytwenty-down-arrow {
border-top: 6px solid white;
bottom: 50%;
margin-bottom: -17px; }
<svg xmlns="http://www.w3.org/2000/svg" fill="#fff" viewBox="0 0 384 512"><!--! Font Awesome Pro 6.1.1 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license (Commercial License) Copyright 2022 Fonticons, Inc. --><path d="M112.1 454.3c0 6.297 1.816 12.44 5.284 17.69l17.14 25.69c5.25 7.875 17.17 14.28 26.64 14.28h61.67c9.438 0 21.36-6.401 26.61-14.28l17.08-25.68c2.938-4.438 5.348-12.37 5.348-17.7L272 415.1h-160L112.1 454.3zM192 0C90.02 .3203 16 82.97 16 175.1c0 44.38 16.44 84.84 43.56 115.8c16.53 18.84 42.34 58.23 52.22 91.45c.0313 .25 .0938 .5166 .125 .7823h160.2c.0313-.2656 .0938-.5166 .125-.7823c9.875-33.22 35.69-72.61 52.22-91.45C351.6 260.8 368 220.4 368 175.1C368 78.8 289.2 .0039 192 0zM288.4 260.1c-15.66 17.85-35.04 46.3-49.05 75.89h-94.61c-14.01-29.59-33.39-58.04-49.04-75.88C75.24 236.8 64 206.1 64 175.1C64 113.3 112.1 48.25 191.1 48C262.6 48 320 105.4 320 175.1C320 206.1 308.8 236.8 288.4 260.1zM176 80C131.9 80 96 115.9 96 160c0 8.844 7.156 16 16 16S128 168.8 128 160c0-26.47 21.53-48 48-48c8.844 0 16-7.148 16-15.99S184.8 80 176 80z"/></svg>
\ No newline at end of file
<svg xmlns="http://www.w3.org/2000/svg" fill="#a79d95" viewBox="0 0 576 512" class="w-10 h-10" width="25" height="25"><!--! Font Awesome Free 6.0.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License) Copyright 2022 Fonticons, Inc. --><path d="M512 32H160c-35.35 0-64 28.65-64 64v224c0 35.35 28.65 64 64 64H512c35.35 0 64-28.65 64-64V96C576 60.65 547.3 32 512 32zM528 320c0 8.822-7.178 16-16 16h-16l-109.3-160.9C383.7 170.7 378.7 168 373.3 168c-5.352 0-10.35 2.672-13.31 7.125l-62.74 94.11L274.9 238.6C271.9 234.4 267.1 232 262 232c-5.109 0-9.914 2.441-12.93 6.574L176 336H160c-8.822 0-16-7.178-16-16V96c0-8.822 7.178-16 16-16H512c8.822 0 16 7.178 16 16V320zM224 112c-17.67 0-32 14.33-32 32s14.33 32 32 32c17.68 0 32-14.33 32-32S241.7 112 224 112zM456 480H120C53.83 480 0 426.2 0 360v-240C0 106.8 10.75 96 24 96S48 106.8 48 120v240c0 39.7 32.3 72 72 72h336c13.25 0 24 10.75 24 24S469.3 480 456 480z"></path></svg>
\ No newline at end of file
// DOM.event.move
//
// 2.0.0
//
// Stephen Band
//
// Triggers 'movestart', 'move' and 'moveend' events after
// mousemoves following a mousedown cross a distance threshold,
// similar to the native 'dragstart', 'drag' and 'dragend' events.
// Move events are throttled to animation frames. Move event objects
// have the properties:
//
// pageX:
// pageY: Page coordinates of pointer.
// startX:
// startY: Page coordinates of pointer at movestart.
// distX:
// distY: Distance the pointer has moved since movestart.
// deltaX:
// deltaY: Distance the finger has moved since last event.
// velocityX:
// velocityY: Average velocity over last few events.
(function(fn) {
if (typeof define === 'function' && define.amd) {
define([], fn);
} else if ((typeof module !== "undefined" && module !== null) && module.exports) {
module.exports = fn;
} else {
fn();
}
})(function(){
var assign = Object.assign || window.jQuery && jQuery.extend;
// Number of pixels a pressed pointer travels before movestart
// event is fired.
var threshold = 8;
// Shim for requestAnimationFrame, falling back to timer. See:
// see http://paulirish.com/2011/requestanimationframe-for-smart-animating/
var requestFrame = (function(){
return (
window.requestAnimationFrame ||
window.webkitRequestAnimationFrame ||
window.mozRequestAnimationFrame ||
window.oRequestAnimationFrame ||
window.msRequestAnimationFrame ||
function(fn, element){
return window.setTimeout(function(){
fn();
}, 25);
}
);
})();
// Shim for customEvent
// see https://developer.mozilla.org/en-US/docs/Web/API/CustomEvent/CustomEvent#Polyfill
(function () {
if ( typeof window.CustomEvent === "function" ) return false;
function CustomEvent ( event, params ) {
params = params || { bubbles: false, cancelable: false, detail: undefined };
var evt = document.createEvent( 'CustomEvent' );
evt.initCustomEvent( event, params.bubbles, params.cancelable, params.detail );
return evt;
}
CustomEvent.prototype = window.Event.prototype;
window.CustomEvent = CustomEvent;
})();
var ignoreTags = {
textarea: true,
input: true,
select: true,
button: true
};
var mouseevents = {
move: 'mousemove',
cancel: 'mouseup dragstart',
end: 'mouseup'
};
var touchevents = {
move: 'touchmove',
cancel: 'touchend',
end: 'touchend'
};
var rspaces = /\s+/;
// DOM Events
var eventOptions = { bubbles: true, cancelable: true };
var eventsSymbol = typeof Symbol === "function" ? Symbol('events') : {};
function createEvent(type) {
return new CustomEvent(type, eventOptions);
}
function getEvents(node) {
return node[eventsSymbol] || (node[eventsSymbol] = {});
}
function on(node, types, fn, data, selector) {
types = types.split(rspaces);
var events = getEvents(node);
var i = types.length;
var handlers, type;
function handler(e) { fn(e, data); }
while (i--) {
type = types[i];
handlers = events[type] || (events[type] = []);
handlers.push([fn, handler]);
node.addEventListener(type, handler);
}
}
function off(node, types, fn, selector) {
types = types.split(rspaces);
var events = getEvents(node);
var i = types.length;
var type, handlers, k;
if (!events) { return; }
while (i--) {
type = types[i];
handlers = events[type];
if (!handlers) { continue; }
k = handlers.length;
while (k--) {
if (handlers[k][0] === fn) {
node.removeEventListener(type, handlers[k][1]);
handlers.splice(k, 1);
}
}
}
}
function trigger(node, type, properties) {
// Don't cache events. It prevents you from triggering an event of a
// given type from inside the handler of another event of that type.
var event = createEvent(type);
if (properties) { assign(event, properties); }
node.dispatchEvent(event);
}
// Constructors
function Timer(fn){
var callback = fn,
active = false,
running = false;
function trigger(time) {
if (active){
callback();
requestFrame(trigger);
running = true;
active = false;
}
else {
running = false;
}
}
this.kick = function(fn) {
active = true;
if (!running) { trigger(); }
};
this.end = function(fn) {
var cb = callback;
if (!fn) { return; }
// If the timer is not running, simply call the end callback.
if (!running) {
fn();
}
// If the timer is running, and has been kicked lately, then
// queue up the current callback and the end callback, otherwise
// just the end callback.
else {
callback = active ?
function(){ cb(); fn(); } :
fn ;
active = true;
}
};
}
// Functions
function noop() {}
function preventDefault(e) {
e.preventDefault();
}
function isIgnoreTag(e) {
return !!ignoreTags[e.target.tagName.toLowerCase()];
}
function isPrimaryButton(e) {
// Ignore mousedowns on any button other than the left (or primary)
// mouse button, or when a modifier key is pressed.
return (e.which === 1 && !e.ctrlKey && !e.altKey);
}
function identifiedTouch(touchList, id) {
var i, l;
if (touchList.identifiedTouch) {
return touchList.identifiedTouch(id);
}
// touchList.identifiedTouch() does not exist in
// webkit yet… we must do the search ourselves...
i = -1;
l = touchList.length;
while (++i < l) {
if (touchList[i].identifier === id) {
return touchList[i];
}
}
}
function changedTouch(e, data) {
var touch = identifiedTouch(e.changedTouches, data.identifier);
// This isn't the touch you're looking for.
if (!touch) { return; }
// Chrome Android (at least) includes touches that have not
// changed in e.changedTouches. That's a bit annoying. Check
// that this touch has changed.
if (touch.pageX === data.pageX && touch.pageY === data.pageY) { return; }
return touch;
}
// Handlers that decide when the first movestart is triggered
function mousedown(e){
// Ignore non-primary buttons
if (!isPrimaryButton(e)) { return; }
// Ignore form and interactive elements
if (isIgnoreTag(e)) { return; }
on(document, mouseevents.move, mousemove, e);
on(document, mouseevents.cancel, mouseend, e);
}
function mousemove(e, data){
checkThreshold(e, data, e, removeMouse);
}
function mouseend(e, data) {
removeMouse();
}
function removeMouse() {
off(document, mouseevents.move, mousemove);
off(document, mouseevents.cancel, mouseend);
}
function touchstart(e) {
// Don't get in the way of interaction with form elements
if (ignoreTags[e.target.tagName.toLowerCase()]) { return; }
var touch = e.changedTouches[0];
// iOS live updates the touch objects whereas Android gives us copies.
// That means we can't trust the touchstart object to stay the same,
// so we must copy the data. This object acts as a template for
// movestart, move and moveend event objects.
var data = {
target: touch.target,
pageX: touch.pageX,
pageY: touch.pageY,
identifier: touch.identifier,
// The only way to make handlers individually unbindable is by
// making them unique.
touchmove: function(e, data) { touchmove(e, data); },
touchend: function(e, data) { touchend(e, data); }
};
on(document, touchevents.move, data.touchmove, data);
on(document, touchevents.cancel, data.touchend, data);
}
function touchmove(e, data) {
var touch = changedTouch(e, data);
if (!touch) { return; }
checkThreshold(e, data, touch, removeTouch);
}
function touchend(e, data) {
var touch = identifiedTouch(e.changedTouches, data.identifier);
if (!touch) { return; }
removeTouch(data);
}
function removeTouch(data) {
off(document, touchevents.move, data.touchmove);
off(document, touchevents.cancel, data.touchend);
}
function checkThreshold(e, data, touch, fn) {
var distX = touch.pageX - data.pageX;
var distY = touch.pageY - data.pageY;
// Do nothing if the threshold has not been crossed.
if ((distX * distX) + (distY * distY) < (threshold * threshold)) { return; }
triggerStart(e, data, touch, distX, distY, fn);
}
function triggerStart(e, data, touch, distX, distY, fn) {
var touches = e.targetTouches;
var time = e.timeStamp - data.timeStamp;
// Create a movestart object with some special properties that
// are passed only to the movestart handlers.
var template = {
altKey: e.altKey,
ctrlKey: e.ctrlKey,
shiftKey: e.shiftKey,
startX: data.pageX,
startY: data.pageY,
distX: distX,
distY: distY,
deltaX: distX,
deltaY: distY,
pageX: touch.pageX,
pageY: touch.pageY,
velocityX: distX / time,
velocityY: distY / time,
identifier: data.identifier,
targetTouches: touches,
finger: touches ? touches.length : 1,
enableMove: function() {
this.moveEnabled = true;
this.enableMove = noop;
e.preventDefault();
}
};
// Trigger the movestart event.
trigger(data.target, 'movestart', template);
// Unbind handlers that tracked the touch or mouse up till now.
fn(data);
}
// Handlers that control what happens following a movestart
function activeMousemove(e, data) {
var timer = data.timer;
data.touch = e;
data.timeStamp = e.timeStamp;
timer.kick();
}
function activeMouseend(e, data) {
var target = data.target;
var event = data.event;
var timer = data.timer;
removeActiveMouse();
endEvent(target, event, timer, function() {
// Unbind the click suppressor, waiting until after mouseup
// has been handled.
setTimeout(function(){
off(target, 'click', preventDefault);
}, 0);
});
}
function removeActiveMouse() {
off(document, mouseevents.move, activeMousemove);
off(document, mouseevents.end, activeMouseend);
}
function activeTouchmove(e, data) {
var event = data.event;
var timer = data.timer;
var touch = changedTouch(e, event);
if (!touch) { return; }
// Stop the interface from gesturing
e.preventDefault();
event.targetTouches = e.targetTouches;
data.touch = touch;
data.timeStamp = e.timeStamp;
timer.kick();
}
function activeTouchend(e, data) {
var target = data.target;
var event = data.event;
var timer = data.timer;
var touch = identifiedTouch(e.changedTouches, event.identifier);
// This isn't the touch you're looking for.
if (!touch) { return; }
removeActiveTouch(data);
endEvent(target, event, timer);
}
function removeActiveTouch(data) {
off(document, touchevents.move, data.activeTouchmove);
off(document, touchevents.end, data.activeTouchend);
}
// Logic for triggering move and moveend events
function updateEvent(event, touch, timeStamp) {
var time = timeStamp - event.timeStamp;
event.distX = touch.pageX - event.startX;
event.distY = touch.pageY - event.startY;
event.deltaX = touch.pageX - event.pageX;
event.deltaY = touch.pageY - event.pageY;
// Average the velocity of the last few events using a decay
// curve to even out spurious jumps in values.
event.velocityX = 0.3 * event.velocityX + 0.7 * event.deltaX / time;
event.velocityY = 0.3 * event.velocityY + 0.7 * event.deltaY / time;
event.pageX = touch.pageX;
event.pageY = touch.pageY;
}
function endEvent(target, event, timer, fn) {
timer.end(function(){
trigger(target, 'moveend', event);
return fn && fn();
});
}
// Set up the DOM
function movestart(e) {
if (e.defaultPrevented) { return; }
if (!e.moveEnabled) { return; }
var event = {
startX: e.startX,
startY: e.startY,
pageX: e.pageX,
pageY: e.pageY,
distX: e.distX,
distY: e.distY,
deltaX: e.deltaX,
deltaY: e.deltaY,
velocityX: e.velocityX,
velocityY: e.velocityY,
identifier: e.identifier,
targetTouches: e.targetTouches,
finger: e.finger
};
var data = {
target: e.target,
event: event,
timer: new Timer(update),
touch: undefined,
timeStamp: e.timeStamp
};
function update(time) {
updateEvent(event, data.touch, data.timeStamp);
trigger(data.target, 'move', event);
}
if (e.identifier === undefined) {
// We're dealing with a mouse event.
// Stop clicks from propagating during a move
on(e.target, 'click', preventDefault);
on(document, mouseevents.move, activeMousemove, data);
on(document, mouseevents.end, activeMouseend, data);
}
else {
// In order to unbind correct handlers they have to be unique
data.activeTouchmove = function(e, data) { activeTouchmove(e, data); };
data.activeTouchend = function(e, data) { activeTouchend(e, data); };
// We're dealing with a touch.
on(document, touchevents.move, data.activeTouchmove, data);
on(document, touchevents.end, data.activeTouchend, data);
}
}
on(document, 'mousedown', mousedown);
on(document, 'touchstart', touchstart);
on(document, 'movestart', movestart);
// jQuery special events
//
// jQuery event objects are copies of DOM event objects. They need
// a little help copying the move properties across.
if (!window.jQuery) { return; }
var properties = ("startX startY pageX pageY distX distY deltaX deltaY velocityX velocityY").split(' ');
function enableMove1(e) { e.enableMove(); }
function enableMove2(e) { e.enableMove(); }
function enableMove3(e) { e.enableMove(); }
function add(handleObj) {
var handler = handleObj.handler;
handleObj.handler = function(e) {
// Copy move properties across from originalEvent
var i = properties.length;
var property;
while(i--) {
property = properties[i];
e[property] = e.originalEvent[property];
}
handler.apply(this, arguments);
};
}
jQuery.event.special.movestart = {
setup: function() {
// Movestart must be enabled to allow other move events
on(this, 'movestart', enableMove1);
// Do listen to DOM events
return false;
},
teardown: function() {
off(this, 'movestart', enableMove1);
return false;
},
add: add
};
jQuery.event.special.move = {
setup: function() {
on(this, 'movestart', enableMove2);
return false;
},
teardown: function() {
off(this, 'movestart', enableMove2);
return false;
},
add: add
};
jQuery.event.special.moveend = {
setup: function() {
on(this, 'movestart', enableMove3);
return false;
},
teardown: function() {
off(this, 'movestart', enableMove3);
return false;
},
add: add
};
});
(function($){
$.fn.twentytwenty = function(options) {
var options = $.extend({
default_offset_pct: 0.5,
orientation: 'horizontal',
before_label: 'Before',
after_label: 'After',
no_overlay: false,
move_slider_on_hover: false,
move_with_handle_only: true,
click_to_move: false
}, options);
return this.each(function() {
var sliderPct = options.default_offset_pct;
var container = $(this);
var sliderOrientation = options.orientation;
var beforeDirection = (sliderOrientation === 'vertical') ? 'down' : 'left';
var afterDirection = (sliderOrientation === 'vertical') ? 'up' : 'right';
container.wrap("<div class='twentytwenty-wrapper twentytwenty-" + sliderOrientation + "'></div>");
if(!options.no_overlay) {
container.append("<div class='twentytwenty-overlay'></div>");
var overlay = container.find(".twentytwenty-overlay");
overlay.append("<div class='twentytwenty-before-label' data-content='"+options.before_label+"'></div>");
overlay.append("<div class='twentytwenty-after-label' data-content='"+options.after_label+"'></div>");
}
var beforeImg = container.find("img:first");
var afterImg = container.find("img:last");
container.append("<div class='twentytwenty-handle'></div>");
var slider = container.find(".twentytwenty-handle");
slider.append("<span class='twentytwenty-" + beforeDirection + "-arrow'></span>");
slider.append("<span class='twentytwenty-" + afterDirection + "-arrow'></span>");
container.addClass("twentytwenty-container");
beforeImg.addClass("twentytwenty-before");
afterImg.addClass("twentytwenty-after");
var calcOffset = function(dimensionPct) {
var w = beforeImg.width();
var h = beforeImg.height();
return {
w: w+"px",
h: h+"px",
cw: (dimensionPct*w)+"px",
ch: (dimensionPct*h)+"px"
};
};
var adjustContainer = function(offset) {
if (sliderOrientation === 'vertical') {
beforeImg.css("clip", "rect(0,"+offset.w+","+offset.ch+",0)");
afterImg.css("clip", "rect("+offset.ch+","+offset.w+","+offset.h+",0)");
}
else {
beforeImg.css("clip", "rect(0,"+offset.cw+","+offset.h+",0)");
afterImg.css("clip", "rect(0,"+offset.w+","+offset.h+","+offset.cw+")");
}
container.css("height", offset.h);
};
var adjustSlider = function(pct) {
var offset = calcOffset(pct);
slider.css((sliderOrientation==="vertical") ? "top" : "left", (sliderOrientation==="vertical") ? offset.ch : offset.cw);
adjustContainer(offset);
};
// Return the number specified or the min/max number if it outside the range given.
var minMaxNumber = function(num, min, max) {
return Math.max(min, Math.min(max, num));
};
// Calculate the slider percentage based on the position.
var getSliderPercentage = function(positionX, positionY) {
var sliderPercentage = (sliderOrientation === 'vertical') ?
(positionY-offsetY)/imgHeight :
(positionX-offsetX)/imgWidth;
return minMaxNumber(sliderPercentage, 0, 1);
};
$(window).on("resize.twentytwenty", function(e) {
adjustSlider(sliderPct);
});
var offsetX = 0;
var offsetY = 0;
var imgWidth = 0;
var imgHeight = 0;
var onMoveStart = function(e) {
if (((e.distX > e.distY && e.distX < -e.distY) || (e.distX < e.distY && e.distX > -e.distY)) && sliderOrientation !== 'vertical') {
e.preventDefault();
}
else if (((e.distX < e.distY && e.distX < -e.distY) || (e.distX > e.distY && e.distX > -e.distY)) && sliderOrientation === 'vertical') {
e.preventDefault();
}
container.addClass("active");
offsetX = container.offset().left;
offsetY = container.offset().top;
imgWidth = beforeImg.width();
imgHeight = beforeImg.height();
};
var onMove = function(e) {
if (container.hasClass("active")) {
sliderPct = getSliderPercentage(e.pageX, e.pageY);
adjustSlider(sliderPct);
}
};
var onMoveEnd = function() {
container.removeClass("active");
};
var moveTarget = options.move_with_handle_only ? slider : container;
moveTarget.on("movestart",onMoveStart);
moveTarget.on("move",onMove);
moveTarget.on("moveend",onMoveEnd);
if (options.move_slider_on_hover) {
container.on("mouseenter", onMoveStart);
container.on("mousemove", onMove);
container.on("mouseleave", onMoveEnd);
}
slider.on("touchmove", function(e) {
e.preventDefault();
});
container.find("img").on("mousedown", function(event) {
event.preventDefault();
});
if (options.click_to_move) {
container.on('click', function(e) {
offsetX = container.offset().left;
offsetY = container.offset().top;
imgWidth = beforeImg.width();
imgHeight = beforeImg.height();
sliderPct = getSliderPercentage(e.pageX, e.pageY);
adjustSlider(sliderPct);
});
}
$(window).trigger("resize.twentytwenty");
});
};
})(jQuery);
<?xml version="1.0"?>
<!--
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
-->
<page xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:View/Layout/etc/page_configuration.xsd">
<head>
<css src="Jajuma_WebpImages::css/web.css"/>
</head>
</page>
var config = {
deps: ["Jajuma_WebpImages/js/webpimages"],
config: {
mixins: {
'mage/gallery/gallery': {
'Jajuma_WebpImages/js/gallery/gallery-mixin': true
},
'Magento_Swatches/js/swatch-renderer': {
'Jajuma_WebpImages/js/swatch-renderer-mixin': true
},
}
}
};
\ No newline at end of file
<?php
/**
* @author JaJuMa GmbH <info@jajuma.de>
* @copyright Copyright (c) 2020 JaJuMa GmbH <https://www.jajuma.de>. All rights reserved.
* @license http://opensource.org/licenses/mit-license.php MIT License
*/
/** @var $block \Jajuma\WebpImages\Block\Picture */
$customSrcTag = $block->getCustomSrcTag();
$customSrcSetTag = $block->getCustomSrcSetTag();
$originalTag = $block->getOriginalTag();
$isLazyLoadEnabled = $block->isNativeLazyLoadingEnabled();
$excludeNativeLazyloadImageAttributes = $block->getExcludeNativeLazyloadImageAttributes();
$isImageInExcludeList = preg_match_all($excludeNativeLazyloadImageAttributes, $originalTag);
if ($isLazyLoadEnabled && !$isImageInExcludeList) {
// add loading="lazy" at the end of the img tag
$originalTag = preg_replace('/>$/', ' loading="lazy" />', $originalTag);
}
?>
<?php if ($customSrcTag): ?>
<picture>
<source type="image/webp" <?= /* @noEscape */ $customSrcSetTag ? $customSrcSetTag : 'srcset' ?>="
<?= /* @noEscape */ $block->getWebpImage() ?>">
<source type="<?= /* @noEscape */ $block->getOriginalImageType() ?>"
<?= /* @noEscape */ $customSrcSetTag ? $customSrcSetTag : 'srcset' ?>="
<?= /* @noEscape */ $block->getOriginalImage() ?>">
<?= /* @noEscape */ $originalTag ?>
</picture>
<?php else: ?>
<picture>
<source type="image/webp" srcset="<?= /* @noEscape */ $block->getWebpImage() ?>">
<source type="<?= /* @noEscape */ $block->getOriginalImageType() ?>"
srcset="<?= /* @noEscape */ $block->getOriginalImage() ?>">
<?= /* @noEscape */ $originalTag ?>
</picture>
<?php endif; ?>
.swatch-option-loading-webp {
content: url(../../images/loader-2.gif);
}
\ No newline at end of file
define(['jquery','mage/utils/wrapper','Jajuma_WebpImages/js/lib/modernizr-webp'], function ($, wrapper) {
'use strict';
return function (initialize) {
return wrapper.wrap(initialize, function (initialize, config, element) {
ModernizrJajuma.on('webp', function(result) {
if (result) {
$.each(config.data, function (key, value) {
value.full = value.full_webp;
value.thumb = value.thumb_webp;
value.img = value.img_webp;
config.data[key] = value;
});
}
initialize(config, element);
});
});
};
});
\ No newline at end of file
/*! modernizr 3.6.0 (Custom Build) | MIT *
* https://modernizr.com/download/?-webp-setclasses !*/
!function(e,n,A){function o(e,n){return typeof e===n}function t(){var e,n,A,t,a,i,l;for(var f in r)if(r.hasOwnProperty(f)){if(e=[],n=r[f],n.name&&(e.push(n.name.toLowerCase()),n.options&&n.options.aliases&&n.options.aliases.length))for(A=0;A<n.options.aliases.length;A++)e.push(n.options.aliases[A].toLowerCase());for(t=o(n.fn,"function")?n.fn():n.fn,a=0;a<e.length;a++)i=e[a],l=i.split("."),1===l.length?ModernizrJajuma[l[0]]=t:(!ModernizrJajuma[l[0]]||ModernizrJajuma[l[0]]instanceof Boolean||(ModernizrJajuma[l[0]]=new Boolean(ModernizrJajuma[l[0]])),ModernizrJajuma[l[0]][l[1]]=t),s.push((t?"":"no-")+l.join("-"))}}function a(e){var n=u.className,A=ModernizrJajuma._config.classPrefix||"";if(c&&(n=n.baseVal),ModernizrJajuma._config.enableJSClass){var o=new RegExp("(^|\\s)"+A+"no-js(\\s|$)");n=n.replace(o,"$1"+A+"js$2")}ModernizrJajuma._config.enableClasses&&(n+=" "+A+e.join(" "+A),c?u.className.baseVal=n:u.className=n)}function i(e,n){if("object"==typeof e)for(var A in e)f(e,A)&&i(A,e[A]);else{e=e.toLowerCase();var o=e.split("."),t=ModernizrJajuma[o[0]];if(2==o.length&&(t=t[o[1]]),"undefined"!=typeof t)return ModernizrJajuma;n="function"==typeof n?n():n,1==o.length?ModernizrJajuma[o[0]]=n:(!ModernizrJajuma[o[0]]||ModernizrJajuma[o[0]]instanceof Boolean||(ModernizrJajuma[o[0]]=new Boolean(ModernizrJajuma[o[0]])),ModernizrJajuma[o[0]][o[1]]=n),a([(n&&0!=n?"":"no-")+o.join("-")]),ModernizrJajuma._trigger(e,n)}return ModernizrJajuma}var s=[],r=[],l={_version:"3.6.0",_config:{classPrefix:"",enableClasses:!0,enableJSClass:!0,usePrefixes:!0},_q:[],on:function(e,n){var A=this;setTimeout(function(){n(A[e])},0)},addTest:function(e,n,A){r.push({name:e,fn:n,options:A})},addAsyncTest:function(e){r.push({name:null,fn:e})}},ModernizrJajuma=function(){};ModernizrJajuma.prototype=l,ModernizrJajuma=new ModernizrJajuma;var f,u=n.documentElement,c="svg"===u.nodeName.toLowerCase();!function(){var e={}.hasOwnProperty;f=o(e,"undefined")||o(e.call,"undefined")?function(e,n){return n in e&&o(e.constructor.prototype[n],"undefined")}:function(n,A){return e.call(n,A)}}(),l._l={},l.on=function(e,n){this._l[e]||(this._l[e]=[]),this._l[e].push(n),ModernizrJajuma.hasOwnProperty(e)&&setTimeout(function(){ModernizrJajuma._trigger(e,ModernizrJajuma[e])},0)},l._trigger=function(e,n){if(this._l[e]){var A=this._l[e];setTimeout(function(){var e,o;for(e=0;e<A.length;e++)(o=A[e])(n)},0),delete this._l[e]}},ModernizrJajuma._q.push(function(){l.addTest=i}),ModernizrJajuma.addAsyncTest(function(){function e(e,n,A){function o(n){var o=n&&"load"===n.type?1==t.width:!1,a="webp"===e;i(e,a&&o?new Boolean(o):o),A&&A(n)}var t=new Image;t.onerror=o,t.onload=o,t.src=n}var n=[{uri:"",name:"webp"},{uri:"",name:"webp.alpha"},{uri:"",name:"webp.animation"},{uri:"",name:"webp.lossless"}],A=n.shift();e(A.name,A.uri,function(A){if(A&&"load"===A.type)for(var o=0;o<n.length;o++)e(n[o].name,n[o].uri)})}),t(),a(s),delete l.addTest,delete l.addAsyncTest;for(var p=0;p<ModernizrJajuma._q.length;p++)ModernizrJajuma._q[p]();e.ModernizrJajuma=ModernizrJajuma}(window,document);
\ No newline at end of file
define([
'jquery',
'jquery/ui',
"domReady!",
'Jajuma_WebpImages/js/lib/modernizr-webp'
], function ($) {
'use strict';
return function (widget) {
$.widget('mage.SwatchRenderer', widget, {
_OnClick: function ($this, $widget) {
var $parent = $this.parents('.' + $widget.options.classes.attributeClass),
$wrapper = $this.parents('.' + $widget.options.classes.attributeOptionsWrapper),
$label = $parent.find('.' + $widget.options.classes.attributeSelectedOptionLabelClass),
attributeId = (typeof $parent.attr('attribute-id') != 'undefined') ? $parent.attr('attribute-id') : (typeof $parent.attr('data-attribute-id') != 'undefined') ? $parent.attr('data-attribute-id') : false,
optionId = (typeof $this.attr('option-id') != 'undefined') ? $this.attr('option-id') : (typeof $this.attr('data-option-id') != 'undefined') ? $this.attr('data-option-id') : false,
optionLabel = (typeof $this.attr('option-label') != 'undefined') ? $this.attr('option-label') : (typeof $this.attr('data-option-label') != 'undefined') ? $this.attr('data-option-label') : false,
$input = $parent.find('.' + $widget.options.classes.attributeInput);
if (typeof this.options.jsonSwatchConfig[attributeId]['additional_data'] != 'undefined') {
var checkAdditionalData = JSON.parse(this.options.jsonSwatchConfig[attributeId]['additional_data']);
}
if ($widget.inProductList) {
$input = $widget.productForm.find(
'.' + $widget.options.classes.attributeInput + '[name="super_attribute[' + attributeId + ']"]'
);
}
if ($this.hasClass('disabled')) {
return;
}
if ($this.hasClass('selected')) {
$parent.removeAttr('option-selected').find('.selected').removeClass('selected');
$input.val('');
$label.text('');
$this.attr('aria-checked', false);
} else {
if ((typeof $this.attr('option-id') != 'undefined') ) {
$parent.attr('option-selected', $this.attr('option-id')).find('.selected').removeClass('selected');
}
if ((typeof $this.attr('data-option-id') != 'undefined') ) {
$parent.attr('data-option-selected', $this.attr('data-option-id')).find('.selected').removeClass('selected');
}
$label.text(optionLabel);
$input.val(optionId);
$input.attr('data-attr-name', this._getAttributeCodeById(attributeId));
$this.addClass('selected');
$widget._toggleCheckedAttributes($this, $wrapper);
}
$widget._Rebuild();
if ($widget.element.parents($widget.options.selectorProduct)
.find(this.options.selectorProductPrice).is(':data(mage-priceBox)')
) {
$widget._UpdatePrice();
}
$(document).trigger('updateMsrpPriceBlock',
[
parseInt($this.attr('index'), 10) + 1,
$widget.options.jsonConfig.optionPrices
]);
if (typeof checkAdditionalData != 'undefined') {
if (checkAdditionalData['update_product_preview_image'] == '1') {
$widget._loadMedia();
}
} else {
$widget._loadMedia();
}
$input.trigger('change');
},
updateBaseImage: function (images, context, isInProductView) {
var justAnImage = images[0],
initialImages = this.options.mediaGalleryInitial,
imagesToUpdate,
gallery = context.find(this.options.mediaGallerySelector).data('gallery'),
isInitial,
self = this;
if (isInProductView) {
imagesToUpdate = images.length ? this._setImageType($.extend(true, [], images)) : [];
isInitial = _.isEqual(imagesToUpdate, initialImages);
if (this.options.gallerySwitchStrategy === 'prepend' && !isInitial) {
imagesToUpdate = imagesToUpdate.concat(initialImages);
}
imagesToUpdate = this._setImageIndex(imagesToUpdate);
if (!_.isUndefined(gallery)) {
ModernizrJajuma.on('webp', function(result) {
if (result) {
// custom replace images with webp images
self.convertImgToWebp(imagesToUpdate, context, isInProductView, justAnImage);
}
});
} else {
context.find(this.options.mediaGallerySelector).on('gallery:loaded', function (loadedGallery) {
loadedGallery = context.find(this.options.mediaGallerySelector).data('gallery');
loadedGallery.updateData(imagesToUpdate);
}.bind(this));
}
if (isInitial) {
$(this.options.mediaGallerySelector).AddFotoramaVideoEvents();
} else {
$(this.options.mediaGallerySelector).AddFotoramaVideoEvents({
selectedOption: this.getProduct(),
dataMergeStrategy: this.options.gallerySwitchStrategy
});
}
gallery.first();
} else if (justAnImage && justAnImage.img) {
ModernizrJajuma.on('webp', function(result) {
if (result) {
// custom replace images with webp images
self.convertImgToWebp(images, context, isInProductView, justAnImage);
} else {
context.find('.product-image-photo').attr('src', justAnImage.img);
context.find('[type="image/jpg"]').attr('srcset', justAnImage.img);
}
});
}
},
convertImgToWebp: function (images, context, isInProductView, justAnImage) {
var gallery = context.find(this.options.mediaGallerySelector).data('gallery');
if (justAnImage.img.indexOf('.webp') !== -1) {
if (!isInProductView) {
context.find('[type="image/webp"]').attr('srcset', justAnImage.img);
}
} else {
$.ajax({
url: BASE_URL + 'webp/image/convert',
data: {
images: images,
isInProductView: isInProductView
},
type: 'POST',
beforeSend: function () {
// add loading
if (isInProductView) {
// add loading for main .fotorama__active
$('.fotorama__stage .fotorama__img').addClass('swatch-option-loading-webp');
// add loading for thumb .fotorama__active
$('.fotorama__thumb img').addClass('swatch-option-loading-webp');
} else {
// add loading for main image in category page
context.find('.product-image-photo').addClass('swatch-option-loading-webp');
}
},
success: function (response) {
if (response.webpUrls.length != 0) {
// replace jpg/png image with webp image in category page only
if (!isInProductView) {
// remove loading for main image in category page
context.find('.product-image-photo').removeClass('swatch-option-loading-webp');
// load image
context.find('[type="image/webp"]').attr('srcset', response.webpUrls[0]['img']);
}
// replace jpg/png image with webp image in fotorama images in product page only
if (isInProductView) {
// load image
gallery.updateData(response.webpUrls);
// remove loading for main .fotorama__active
$('.fotorama__stage .fotorama__img').removeClass('swatch-option-loading-webp');
// remove loading for thumb .fotorama__active
$('.fotorama__thumb img').removeClass('swatch-option-loading-webp');
}
}
}
});
}
}
});
return $.mage.SwatchRenderer;
}
});
require([
'jquery',
'Jajuma_WebpImages/js/lib/modernizr-webp'
],function($){
$(document).ready(function() {
ModernizrJajuma.on('webp', function(result) {
if (!result) {
$('body').addClass('no-webp');
}
});
})
})
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment