fix
22
wp-content/plugins/robin-image-optimizer/vendor/autoload.php
vendored
Normal file
@@ -0,0 +1,22 @@
|
||||
<?php
|
||||
|
||||
// autoload.php @generated by Composer
|
||||
|
||||
if (PHP_VERSION_ID < 50600) {
|
||||
if (!headers_sent()) {
|
||||
header('HTTP/1.1 500 Internal Server Error');
|
||||
}
|
||||
$err = 'Composer 2.3.0 dropped support for autoloading on PHP <5.6 and you are running '.PHP_VERSION.', please upgrade PHP or use Composer 2.2 LTS via "composer self-update --2.2". Aborting.'.PHP_EOL;
|
||||
if (!ini_get('display_errors')) {
|
||||
if (PHP_SAPI === 'cli' || PHP_SAPI === 'phpdbg') {
|
||||
fwrite(STDERR, $err);
|
||||
} elseif (!headers_sent()) {
|
||||
echo $err;
|
||||
}
|
||||
}
|
||||
throw new RuntimeException($err);
|
||||
}
|
||||
|
||||
require_once __DIR__ . '/composer/autoload_real.php';
|
||||
|
||||
return ComposerAutoloaderInit5ee79b07cb3e29b73ecedab0cd3eb850::getLoader();
|
||||
592
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/CHANGELOG.md
vendored
Normal file
@@ -0,0 +1,592 @@
|
||||
##### [Version 3.3.50](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.49...v3.3.50) (2025-11-25)
|
||||
|
||||
> Things are getting better every day. 🚀
|
||||
|
||||
##### [Version 3.3.49](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.48...v3.3.49) (2025-09-18)
|
||||
|
||||
> Things are getting better every day. 🚀
|
||||
|
||||
##### [Version 3.3.48](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.47...v3.3.48) (2025-08-11)
|
||||
|
||||
Development
|
||||
|
||||
##### [Version 3.3.47](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.46...v3.3.47) (2025-07-21)
|
||||
|
||||
- Fixed review link
|
||||
- Fixed plugins ranking
|
||||
|
||||
##### [Version 3.3.46](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.45...v3.3.46) (2025-05-16)
|
||||
|
||||
- Add masteriyo recommandation
|
||||
- Add bf helpers
|
||||
- update formbricks
|
||||
|
||||
##### [Version 3.3.45](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.44...v3.3.45) (2025-04-28)
|
||||
|
||||
- feat: add review_link param to about_us filter
|
||||
|
||||
##### [Version 3.3.44](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.43...v3.3.44) (2025-02-18)
|
||||
|
||||
- Fix variable mismatch in the install category function.
|
||||
|
||||
##### [Version 3.3.43](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.42...v3.3.43) (2025-02-18)
|
||||
|
||||
- Compute survey install category as common data.
|
||||
- Add auto-trigger cancel event
|
||||
|
||||
##### [Version 3.3.42](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.41...v3.3.42) (2025-02-17)
|
||||
|
||||
- Updated Formbricks package version
|
||||
- Added support for Formbrick script loading via Internal Pages hooks
|
||||
|
||||
##### [Version 3.3.41](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.40...v3.3.41) (2025-02-05)
|
||||
|
||||
- fixed a minor issue that prevented the translation checking for Hestia PRO
|
||||
|
||||
##### [Version 3.3.40](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.39...v3.3.40) (2024-12-29)
|
||||
|
||||
Adds feadzy features.
|
||||
|
||||
##### [Version 3.3.39](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.38...v3.3.39) (2024-11-19)
|
||||
|
||||
- Adds WPFS promo to About Page
|
||||
- Changes to Licenser and Updater to accommodate WPFS
|
||||
|
||||
##### [Version 3.3.38](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.37...v3.3.38) (2024-11-13)
|
||||
|
||||
- fix: new SDK slug on wp env
|
||||
- fix: wfp promo not checking for donation page
|
||||
|
||||
##### [Version 3.3.37](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.36...v3.3.37) (2024-11-06)
|
||||
|
||||
- Hotfix for Black Friday Global notice
|
||||
|
||||
##### [Version 3.3.36](https://github.com/Codeinwp/themeisle-sdk-main/compare/v3.3.35...v3.3.36) (2024-11-06)
|
||||
|
||||
- Update Blackfriday
|
||||
- Add WP Full Pay
|
||||
|
||||
##### [Version 3.3.35](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.34...v3.3.35) (2024-10-29)
|
||||
|
||||
ignore dev files
|
||||
|
||||
##### [Version 3.3.34](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.33...v3.3.34) (2024-10-25)
|
||||
|
||||
- Add initial promo delay to 6 hours
|
||||
- Add 21 days delay between two promos
|
||||
|
||||
##### [Version 3.3.33](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.32...v3.3.33) (2024-10-24)
|
||||
|
||||
- chore: disable otter promos by default
|
||||
- feat: add lang code for Formbricks survey
|
||||
|
||||
##### [Version 3.3.32](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.31...v3.3.32) (2024-10-16)
|
||||
|
||||
- Fix translations bugs and adapt mechanism to work on themes
|
||||
- Adds product review links in About page
|
||||
- Fix About page layout on Safari browser
|
||||
|
||||
##### [Version 3.3.31](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.30...v3.3.31) (2024-10-14)
|
||||
|
||||
- Update minimum PHP version required for Hyve promo
|
||||
- add E2E test support with wp-scripts and Playwright
|
||||
|
||||
##### [Version 3.3.30](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.29...v3.3.30) (2024-09-10)
|
||||
|
||||
- Remove margin auto from ThemeIsle Woo tab
|
||||
- Add Hyve Promo
|
||||
|
||||
##### [Version 3.3.29](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.28...v3.3.29) (2024-08-21)
|
||||
|
||||
- feat: improve the Promotions appearance timing
|
||||
|
||||
##### [Version 3.3.28](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.27...v3.3.28) (2024-08-09)
|
||||
|
||||
|
||||
- fix: remove the delay on woo plugins promos [closes Codeinwp/themeisle[#1655](https://github.com/Codeinwp/themeisle-sdk/issues/1655)]
|
||||
- fix: swap Neve FSE notice with Neve [closes Codeinwp/themeisle[#1654](https://github.com/Codeinwp/themeisle-sdk/issues/1654)]
|
||||
- fix: rework optimole promo [closes Codeinwp/themeisle[#1656](https://github.com/Codeinwp/themeisle-sdk/issues/1656)]
|
||||
- fix: translations workflow always requiring updates [closes Codeinwp/neve-pro-addon[#2840](https://github.com/Codeinwp/themeisle-sdk/issues/2840)]
|
||||
|
||||
##### [Version 3.3.27](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.26...v3.3.27) (2024-07-30)
|
||||
|
||||
- fix: array out of bounds index
|
||||
|
||||
##### [Version 3.3.26](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.25...v3.3.26) (2024-07-29)
|
||||
|
||||
- fix: JS error in core editor
|
||||
- twak: change ROP product name
|
||||
- tweak: updated developers list
|
||||
- fix: removed duplicate feed URL
|
||||
- feat: add private translation functionality
|
||||
|
||||
##### [Version 3.3.25](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.24...v3.3.25) (2024-07-08)
|
||||
|
||||
improve upgrade link
|
||||
|
||||
##### [Version 3.3.24](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.23...v3.3.24) (2024-06-25)
|
||||
|
||||
- Feat: Allow affiliate url change via DB option.
|
||||
- Fix: Featured plugins displaying on some non-pro products.
|
||||
|
||||
##### [Version 3.3.23](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.22...v3.3.23) (2024-06-19)
|
||||
|
||||
- fix: compress images
|
||||
- fix: replace huge SVG logo with png version
|
||||
- fix: improve text on float widget
|
||||
|
||||
##### [Version 3.3.22](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.21...v3.3.22) (2024-05-17)
|
||||
|
||||
- Updated NPM packages
|
||||
- Fixed the default state of the telemetry logger flag for PRO users
|
||||
- Added floating widget functionality
|
||||
- Updated the remote get check for WordPress VIP env
|
||||
|
||||
##### [Version 3.3.21](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.20...v3.3.21) (2024-04-29)
|
||||
|
||||
- fixes non-existant string being used in Licenser
|
||||
|
||||
##### [Version 3.3.20](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.19...v3.3.20) (2024-04-16)
|
||||
|
||||
fix required params on sprintf
|
||||
|
||||
##### [Version 3.3.19](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.18...v3.3.19) (2024-04-15)
|
||||
|
||||
Add full translation support
|
||||
|
||||
##### [Version 3.3.18](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.17...v3.3.18) (2024-04-05)
|
||||
|
||||
### Big Fixes
|
||||
- Fix rollback for Themes
|
||||
|
||||
##### [Version 3.3.17](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.16...v3.3.17) (2024-04-01)
|
||||
|
||||
### Improvements
|
||||
- Add safeguard against type mutation for featured plugin results.
|
||||
|
||||
##### [Version 3.3.16](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.15...v3.3.16) (2024-03-29)
|
||||
|
||||
### Improvements
|
||||
- Formbricks on multiple products.
|
||||
- List Pro Products in featured tab.
|
||||
|
||||
##### [Version 3.3.15](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.14...v3.3.15) (2024-03-26)
|
||||
|
||||
### Improvements
|
||||
- Add a filter to disallow promotions on specific products.
|
||||
- Update Formbricks
|
||||
|
||||
##### [Version 3.3.14](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.13...v3.3.14) (2024-02-27)
|
||||
|
||||
- Add Announcement Module
|
||||
- Add Script Loader Module
|
||||
|
||||
##### [Version 3.3.13](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.12...v3.3.13) (2024-02-01)
|
||||
|
||||
- Updated nonce check
|
||||
|
||||
##### [Version 3.3.12](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.11...v3.3.12) (2024-01-31)
|
||||
|
||||
- fix: prevent reference key set for users with no capability
|
||||
|
||||
##### [Version 3.3.11](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.10...v3.3.11) (2023-12-12)
|
||||
|
||||
- fix: cached requests for wp options
|
||||
- fix: php notice on failed xml feed
|
||||
|
||||
##### [Version 3.3.10](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.9...v3.3.10) (2023-12-11)
|
||||
|
||||
feat: add new filter for tsdk_utmify query arguments
|
||||
|
||||
##### [Version 3.3.9](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.8...v3.3.9) (2023-11-16)
|
||||
|
||||
- Fix: a debugging error when you activate Neve Pro after installing Otter & Otter Pro without the license.
|
||||
|
||||
##### [Version 3.3.8](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.7...v3.3.8) (2023-11-14)
|
||||
|
||||
- Add Product Telemetry
|
||||
|
||||
##### [Version 3.3.7](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.6...v3.3.7) (2023-10-31)
|
||||
|
||||
- fix: deprecating notice in Dashboard Widget
|
||||
|
||||
##### [Version 3.3.6](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.5...v3.3.6) (2023-10-05)
|
||||
|
||||
- Fix duplicate notification in Neve FSE
|
||||
- Fix SDK breaking Enfold theme builder
|
||||
|
||||
##### [Version 3.3.5](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.4...v3.3.5) (2023-09-29)
|
||||
|
||||
Fix TPC message.
|
||||
Fix TPC typo.
|
||||
|
||||
##### [Version 3.3.4](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.3...v3.3.4) (2023-09-18)
|
||||
|
||||
- Allow users to activate plugins from the About page
|
||||
|
||||
##### [Version 3.3.3](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.2...v3.3.3) (2023-08-22)
|
||||
|
||||
- Disable install buttons on the About page if users can not install plugins
|
||||
|
||||
##### [Version 3.3.2](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.1...v3.3.2) (2023-08-02)
|
||||
|
||||
- Added a new product page for Otter
|
||||
|
||||
##### [Version 3.3.1](https://github.com/Codeinwp/themeisle-sdk/compare/v3.3.0...v3.3.1) (2023-06-21)
|
||||
|
||||
- Fix Neve FSE promo typo
|
||||
|
||||
#### [Version 3.3.0](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.41...v3.3.0) (2023-05-30)
|
||||
|
||||
- Adds About Page Integration
|
||||
- Adds promos for Neve FSE, Sparks & PPOM
|
||||
|
||||
##### [Version 3.2.41](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.40...v3.2.41) (2023-05-10)
|
||||
|
||||
- Delay product recommendation mentions
|
||||
- Allow upselling notifications for new users
|
||||
|
||||
##### [Version 3.2.40](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.39...v3.2.40) (2023-03-30)
|
||||
|
||||
- Add ROP upsell to all products
|
||||
|
||||
##### [Version 3.2.39](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.38...v3.2.39) (2023-03-17)
|
||||
|
||||
* Adds direct utility function for a direct support link.
|
||||
|
||||
##### [Version 3.2.38](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.37...v3.2.38) (2023-03-10)
|
||||
|
||||
Fix promotions path-breaking
|
||||
|
||||
##### [Version 3.2.37](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.36...v3.2.37) (2023-03-01)
|
||||
|
||||
Fix array casting
|
||||
|
||||
##### [Version 3.2.36](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.35...v3.2.36) (2023-03-01)
|
||||
|
||||
fix perfomance issues on attachments count https://github.com/Codeinwp/themeisle-sdk/issues/159
|
||||
|
||||
##### [Version 3.2.35](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.34...v3.2.35) (2023-02-22)
|
||||
|
||||
Added Codeinwp and wpshout feeds to dashboard widget
|
||||
|
||||
##### [Version 3.2.34](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.33...v3.2.34) (2023-01-31)
|
||||
|
||||
Improve promotions
|
||||
|
||||
##### [Version 3.2.33](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.32...v3.2.33) (2023-01-30)
|
||||
|
||||
* Adds PHP 8.2 compatibility
|
||||
* Update promotions
|
||||
|
||||
##### [Version 3.2.32](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.31...v3.2.32) (2022-11-30)
|
||||
|
||||
Release
|
||||
|
||||
##### [Version 3.2.31](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.30...v3.2.31) (2022-11-23)
|
||||
|
||||
- improve the promotions module
|
||||
|
||||
##### [Version 3.2.30](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.29...v3.2.30) (2022-09-15)
|
||||
|
||||
- fix filesystem wrong use - ref [#138](https://github.com/Codeinwp/themeisle-sdk/issues/138), props [@ethanclevenger91](https://github.com/ethanclevenger91) for reporting
|
||||
|
||||
##### [Version 3.2.29](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.28...v3.2.29) (2022-09-08)
|
||||
|
||||
* Adds compatibility mechanism
|
||||
* Adds content utms
|
||||
* Adds usage time on uninstall feedback
|
||||
|
||||
##### [Version 3.2.28](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.27...v3.2.28) (2022-08-30)
|
||||
|
||||
* Adds utm handler
|
||||
* Improve promotions
|
||||
|
||||
##### [Version 3.2.27](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.26...v3.2.27) (2022-08-23)
|
||||
|
||||
- Add Promotion Module
|
||||
Add the Promotion module for free plugins
|
||||
|
||||
##### [Version 3.2.26](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.25...v3.2.26) (2022-05-12)
|
||||
|
||||
- [Fix] Solve rollback sometimes not available
|
||||
|
||||
##### [Version 3.2.25](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.24...v3.2.25) (2022-03-28)
|
||||
|
||||
- Force update request after rollback
|
||||
|
||||
##### [Version 3.2.24](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.23...v3.2.24) (2022-02-09)
|
||||
|
||||
Fix edge case issue on dismiss
|
||||
Avoid issues with open_basedir restrictions
|
||||
|
||||
##### [Version 3.2.23](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.22...v3.2.23) (2022-02-02)
|
||||
|
||||
Fix php 8.1 issues
|
||||
Fix edge case when update_themes site transient was empty and a fatal error was thrown
|
||||
|
||||
##### [Version 3.2.22](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.21...v3.2.22) (2021-10-27)
|
||||
|
||||
Fix edge case when reset failed checks was not working properly
|
||||
|
||||
##### [Version 3.2.21](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.20...v3.2.21) (2021-06-30)
|
||||
|
||||
review and improve compatibility with auto-updates on custom updates endpoint
|
||||
|
||||
##### [Version 3.2.20](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.19...v3.2.20) (2021-03-30)
|
||||
|
||||
add wp-config support
|
||||
|
||||
##### [Version 3.2.19](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.18...v3.2.19) (2021-03-12)
|
||||
|
||||
* Adds compatibility with latest PHPCS coding standards.
|
||||
* Adds compatibility with core auto-update.
|
||||
|
||||
##### [Version 3.2.18](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.17...v3.2.18) (2021-03-04)
|
||||
|
||||
* Fix regression on rollback order
|
||||
|
||||
##### [Version 3.2.17](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.16...v3.2.17) (2021-03-04)
|
||||
|
||||
* Fix compatibility with PHP 8 due to usort
|
||||
|
||||
##### [Version 3.2.16](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.15...v3.2.16) (2020-11-17)
|
||||
|
||||
* Fix long texts on rollback.
|
||||
* Fix RTL mode for uninstall feedback.
|
||||
|
||||
##### [Version 3.2.15](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.14...v3.2.15) (2020-07-23)
|
||||
|
||||
* remove no redundant module
|
||||
|
||||
##### [Version 3.2.14](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.13...v3.2.14) (2020-06-10)
|
||||
|
||||
> Things are getting better every day. 🚀
|
||||
|
||||
##### [Version 3.2.13](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.12...v3.2.13) (2020-06-10)
|
||||
|
||||
Adds plan logic and expiration
|
||||
|
||||
##### [Version 3.2.12](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.11...v3.2.12) (2020-06-10)
|
||||
|
||||
Adds key filter
|
||||
|
||||
##### [Version 3.2.11](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.10...v3.2.11) (2020-06-04)
|
||||
|
||||
* remove non-printable chars
|
||||
|
||||
##### [Version 3.2.10](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.9...v3.2.10) (2020-05-28)
|
||||
|
||||
* Remove extra files on export
|
||||
|
||||
##### [Version 3.2.9](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.8...v3.2.9) (2020-05-18)
|
||||
|
||||
adds new endpoints
|
||||
|
||||
##### [Version 3.2.8](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.7...v3.2.8) (2020-03-24)
|
||||
|
||||
* change license handler method access
|
||||
|
||||
##### [Version 3.2.7](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.6...v3.2.7) (2020-03-24)
|
||||
|
||||
* fix callback for license processing hook
|
||||
|
||||
##### [Version 3.2.6](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.5...v3.2.6) (2020-03-23)
|
||||
|
||||
* Fix notice on license deactivation
|
||||
|
||||
##### [Version 3.2.5](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.4...v3.2.5) (2020-03-23)
|
||||
|
||||
* always load notification manager last
|
||||
|
||||
##### [Version 3.2.4](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.3...v3.2.4) (2020-03-21)
|
||||
|
||||
* Cast version response to array for icons
|
||||
|
||||
##### [Version 3.2.3](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.2...v3.2.3) (2020-03-21)
|
||||
|
||||
* use product slug instead of the one from api
|
||||
|
||||
##### [Version 3.2.2](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.1...v3.2.2) (2020-03-13)
|
||||
|
||||
* improve notice dismiss mechanism
|
||||
|
||||
##### [Version 3.2.1](https://github.com/Codeinwp/themeisle-sdk/compare/v3.2.0...v3.2.1) (2020-03-05)
|
||||
|
||||
Fix rollback call for private products
|
||||
|
||||
#### [Version 3.2.0](https://github.com/Codeinwp/themeisle-sdk/compare/v3.1.9...v3.2.0) (2020-03-04)
|
||||
|
||||
* adds license activation/deactivation handlers for wp cli
|
||||
* adds compatibility with the newest license API
|
||||
|
||||
##### [Version 3.1.9](https://github.com/Codeinwp/themeisle-sdk/compare/v3.1.8...v3.1.9) (2020-02-24)
|
||||
|
||||
* Add integration with GitHub actions
|
||||
|
||||
## [3.1.8](https://github.com/Codeinwp/themeisle-sdk/compare/v3.1.7...v3.1.8) (2019-11-18)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* update developers name ([6aca63e](https://github.com/Codeinwp/themeisle-sdk/commit/6aca63e))
|
||||
|
||||
## [3.1.7](https://github.com/Codeinwp/themeisle-sdk/compare/v3.1.6...v3.1.7) (2019-11-07)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* license field style on wp5.3 ([0239997](https://github.com/Codeinwp/themeisle-sdk/commit/0239997))
|
||||
* license field style on wp5.3 ([86d3a1b](https://github.com/Codeinwp/themeisle-sdk/commit/86d3a1b))
|
||||
|
||||
## [3.1.6](https://github.com/Codeinwp/themeisle-sdk/compare/v3.1.5...v3.1.6) (2019-09-24)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* remove license related options when deactivated ([02cd6ce](https://github.com/Codeinwp/themeisle-sdk/commit/02cd6ce))
|
||||
* remove license related options when deactivated ([d3c1a1f](https://github.com/Codeinwp/themeisle-sdk/commit/d3c1a1f))
|
||||
|
||||
## [3.1.5](https://github.com/Codeinwp/themeisle-sdk/compare/v3.1.4...v3.1.5) (2019-09-11)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* allow unloading certain module features ([2a2559a](https://github.com/Codeinwp/themeisle-sdk/commit/2a2559a))
|
||||
* license activation workflow, show error message when failed to a… ([ade795c](https://github.com/Codeinwp/themeisle-sdk/commit/ade795c))
|
||||
* license activation workflow, show error message when failed to activate ([2f5cbae](https://github.com/Codeinwp/themeisle-sdk/commit/2f5cbae))
|
||||
|
||||
## [3.1.4](https://github.com/Codeinwp/themeisle-sdk/compare/v3.1.3...v3.1.4) (2019-08-23)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* license key was missing on get_version call ([365cde6](https://github.com/Codeinwp/themeisle-sdk/commit/365cde6))
|
||||
* license key was missing on get_version call ([c02f225](https://github.com/Codeinwp/themeisle-sdk/commit/c02f225))
|
||||
|
||||
## [3.1.3](https://github.com/Codeinwp/themeisle-sdk/compare/v3.1.2...v3.1.3) (2019-08-20)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* license deactivation behaviour https://github.com/Codeinwp/visua… ([59c4afe](https://github.com/Codeinwp/themeisle-sdk/commit/59c4afe))
|
||||
* license deactivation behaviour https://github.com/Codeinwp/visualizer-pro/issues/192 ([f641e18](https://github.com/Codeinwp/themeisle-sdk/commit/f641e18))
|
||||
|
||||
## [3.1.2](https://github.com/Codeinwp/themeisle-sdk/compare/v3.1.1...v3.1.2) (2019-08-12)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* phpunit test case ([efe851c](https://github.com/Codeinwp/themeisle-sdk/commit/efe851c))
|
||||
* url format for license endpoint, improve changelog handling and license checks ([a492c68](https://github.com/Codeinwp/themeisle-sdk/commit/a492c68))
|
||||
|
||||
## [3.1.1](https://github.com/Codeinwp/themeisle-sdk/compare/v3.1.0...v3.1.1) (2019-08-08)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* adds is_file for file existence check ([d1205c4](https://github.com/Codeinwp/themeisle-sdk/commit/d1205c4))
|
||||
* adds is_file for file existence check ([be119c1](https://github.com/Codeinwp/themeisle-sdk/commit/be119c1))
|
||||
|
||||
# [3.1.0](https://github.com/Codeinwp/themeisle-sdk/compare/v3.0.10...v3.1.0) (2019-08-05)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* adds extra comments for rest of the options, fix [#64](https://github.com/Codeinwp/themeisle-sdk/issues/64) ([018b22f](https://github.com/Codeinwp/themeisle-sdk/commit/018b22f))
|
||||
* hide license key when active under a password mask, fix [#67](https://github.com/Codeinwp/themeisle-sdk/issues/67) ([c0633c2](https://github.com/Codeinwp/themeisle-sdk/commit/c0633c2))
|
||||
* new uninstall feedback popup issues ([5bda4bd](https://github.com/Codeinwp/themeisle-sdk/commit/5bda4bd))
|
||||
* phpcs indentation errors ([d59ed4f](https://github.com/Codeinwp/themeisle-sdk/commit/d59ed4f))
|
||||
* undefined notices on license check, fix [#60](https://github.com/Codeinwp/themeisle-sdk/issues/60) ([7f56a97](https://github.com/Codeinwp/themeisle-sdk/commit/7f56a97))
|
||||
* uninstall feedback popup placement [[#61](https://github.com/Codeinwp/themeisle-sdk/issues/61)] ([1102d6c](https://github.com/Codeinwp/themeisle-sdk/commit/1102d6c))
|
||||
|
||||
|
||||
### Features
|
||||
|
||||
* new product feedback popup ([f0dbab3](https://github.com/Codeinwp/themeisle-sdk/commit/f0dbab3))
|
||||
* new uninstall feedback form for themes ([8a29f21](https://github.com/Codeinwp/themeisle-sdk/commit/8a29f21))
|
||||
|
||||
## [3.0.10](https://github.com/Codeinwp/themeisle-sdk/compare/v3.0.9...v3.0.10) (2019-07-16)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* compatibility with lower PHP versions ([065ac8e](https://github.com/Codeinwp/themeisle-sdk/commit/065ac8e))
|
||||
* not loading licenser when SDK comes from theme [[#62](https://github.com/Codeinwp/themeisle-sdk/issues/62)] ([b706ca7](https://github.com/Codeinwp/themeisle-sdk/commit/b706ca7))
|
||||
* not loading licenser when SDK comes from theme [[#65](https://github.com/Codeinwp/themeisle-sdk/issues/65) ([419d8e6](https://github.com/Codeinwp/themeisle-sdk/commit/419d8e6))
|
||||
* preserve loaded when adding the licenser one ([cd50434](https://github.com/Codeinwp/themeisle-sdk/commit/cd50434))
|
||||
|
||||
## [3.0.9](https://github.com/Codeinwp/themeisle-sdk/compare/v3.0.8...v3.0.9) (2019-06-26)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* adds new icon for dashboard widget ([de78068](https://github.com/Codeinwp/themeisle-sdk/commit/de78068))
|
||||
* anchor element on license activation message which should link to the license field, fix [#57](https://github.com/Codeinwp/themeisle-sdk/issues/57) ([2e78856](https://github.com/Codeinwp/themeisle-sdk/commit/2e78856))
|
||||
* change uninstall feedback logo with new version, fix [#58](https://github.com/Codeinwp/themeisle-sdk/issues/58) ([2554a4f](https://github.com/Codeinwp/themeisle-sdk/commit/2554a4f))
|
||||
* remove soon to expire notice, fix https://github.com/Codeinwp/themeisle/issues/752 ([a126225](https://github.com/Codeinwp/themeisle-sdk/commit/a126225))
|
||||
|
||||
## [3.0.8](https://github.com/Codeinwp/themeisle-sdk/compare/v3.0.7...v3.0.8) (2019-05-28)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* undefined class on diff module which should check the class on global namespace ([df6bb12](https://github.com/Codeinwp/themeisle-sdk/commit/df6bb12))
|
||||
|
||||
## [3.0.7](https://github.com/Codeinwp/themeisle-sdk/compare/v3.0.6...v3.0.7) (2019-05-27)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* change store url with the new domain ([6bdbe1e](https://github.com/Codeinwp/themeisle-sdk/commit/6bdbe1e))
|
||||
|
||||
## [3.0.6](https://github.com/Codeinwp/themeisle-sdk/compare/v3.0.5...v3.0.6) (2019-05-21)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* build php version for deployment stage ([a785699](https://github.com/Codeinwp/themeisle-sdk/commit/a785699))
|
||||
* uninstall feedback should load only on the proper pages ([259e78f](https://github.com/Codeinwp/themeisle-sdk/commit/259e78f))
|
||||
|
||||
## [3.0.5](https://github.com/Codeinwp/themeisle-sdk/compare/v3.0.4...v3.0.5) (2019-03-07)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* dashboard widget issues and recommended module inconsistency fix [#50](https://github.com/Codeinwp/themeisle-sdk/issues/50), [#49](https://github.com/Codeinwp/themeisle-sdk/issues/49), [#47](https://github.com/Codeinwp/themeisle-sdk/issues/47) ([757eb02](https://github.com/Codeinwp/themeisle-sdk/commit/757eb02))
|
||||
|
||||
## [3.0.4](https://github.com/Codeinwp/themeisle-sdk/compare/v3.0.3...v3.0.4) (2019-01-28)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* uninstall feedback disclosure issues when one of the feedback fields is open ([4631eef](https://github.com/Codeinwp/themeisle-sdk/commit/4631eef))
|
||||
|
||||
## [3.0.3](https://github.com/Codeinwp/themeisle-sdk/compare/v3.0.2...v3.0.3) (2019-01-07)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* **build:** fix exit code when is running outside wordpress context ([d298bb5](https://github.com/Codeinwp/themeisle-sdk/commit/d298bb5))
|
||||
|
||||
## [3.0.2](https://github.com/Codeinwp/themeisle-sdk/compare/v3.0.1...v3.0.2) (2018-12-28)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* remove composer/installers from package requirements ([a0ad543](https://github.com/Codeinwp/themeisle-sdk/commit/a0ad543))
|
||||
|
||||
## [3.0.1](https://github.com/Codeinwp/themeisle-sdk/compare/v3.0.0...v3.0.1) (2018-12-24)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* notifications setup triggers after all products register their n… ([999a944](https://github.com/Codeinwp/themeisle-sdk/commit/999a944))
|
||||
* notifications setup triggers after all products register their notices ([ec3cacc](https://github.com/Codeinwp/themeisle-sdk/commit/ec3cacc))
|
||||
|
||||
# 1.0.0 (2018-12-21)
|
||||
|
||||
|
||||
### Features
|
||||
|
||||
* adds uninstall feedback privacy policy info ([ed17943](https://github.com/Codeinwp/themeisle-sdk/commit/ed17943))
|
||||
674
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/LICENSE
vendored
Normal file
@@ -0,0 +1,674 @@
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 3, 29 June 2007
|
||||
|
||||
Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The GNU General Public License is a free, copyleft license for
|
||||
software and other kinds of works.
|
||||
|
||||
The licenses for most software and other practical works are designed
|
||||
to take away your freedom to share and change the works. By contrast,
|
||||
the GNU General Public License is intended to guarantee your freedom to
|
||||
share and change all versions of a program--to make sure it remains free
|
||||
software for all its users. We, the Free Software Foundation, use the
|
||||
GNU General Public License for most of our software; it applies also to
|
||||
any other work released this way by its authors. You can apply it to
|
||||
your programs, too.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
them if you wish), that you receive source code or can get it if you
|
||||
want it, that you can change the software or use pieces of it in new
|
||||
free programs, and that you know you can do these things.
|
||||
|
||||
To protect your rights, we need to prevent others from denying you
|
||||
these rights or asking you to surrender the rights. Therefore, you have
|
||||
certain responsibilities if you distribute copies of the software, or if
|
||||
you modify it: responsibilities to respect the freedom of others.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must pass on to the recipients the same
|
||||
freedoms that you received. You must make sure that they, too, receive
|
||||
or can get the source code. And you must show them these terms so they
|
||||
know their rights.
|
||||
|
||||
Developers that use the GNU GPL protect your rights with two steps:
|
||||
(1) assert copyright on the software, and (2) offer you this License
|
||||
giving you legal permission to copy, distribute and/or modify it.
|
||||
|
||||
For the developers' and authors' protection, the GPL clearly explains
|
||||
that there is no warranty for this free software. For both users' and
|
||||
authors' sake, the GPL requires that modified versions be marked as
|
||||
changed, so that their problems will not be attributed erroneously to
|
||||
authors of previous versions.
|
||||
|
||||
Some devices are designed to deny users access to install or run
|
||||
modified versions of the software inside them, although the manufacturer
|
||||
can do so. This is fundamentally incompatible with the aim of
|
||||
protecting users' freedom to change the software. The systematic
|
||||
pattern of such abuse occurs in the area of products for individuals to
|
||||
use, which is precisely where it is most unacceptable. Therefore, we
|
||||
have designed this version of the GPL to prohibit the practice for those
|
||||
products. If such problems arise substantially in other domains, we
|
||||
stand ready to extend this provision to those domains in future versions
|
||||
of the GPL, as needed to protect the freedom of users.
|
||||
|
||||
Finally, every program is threatened constantly by software patents.
|
||||
States should not allow patents to restrict development and use of
|
||||
software on general-purpose computers, but in those that do, we wish to
|
||||
avoid the special danger that patents applied to a free program could
|
||||
make it effectively proprietary. To prevent this, the GPL assures that
|
||||
patents cannot be used to render the program non-free.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
TERMS AND CONDITIONS
|
||||
|
||||
0. Definitions.
|
||||
|
||||
"This License" refers to version 3 of the GNU General Public License.
|
||||
|
||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||
works, such as semiconductor masks.
|
||||
|
||||
"The Program" refers to any copyrightable work licensed under this
|
||||
License. Each licensee is addressed as "you". "Licensees" and
|
||||
"recipients" may be individuals or organizations.
|
||||
|
||||
To "modify" a work means to copy from or adapt all or part of the work
|
||||
in a fashion requiring copyright permission, other than the making of an
|
||||
exact copy. The resulting work is called a "modified version" of the
|
||||
earlier work or a work "based on" the earlier work.
|
||||
|
||||
A "covered work" means either the unmodified Program or a work based
|
||||
on the Program.
|
||||
|
||||
To "propagate" a work means to do anything with it that, without
|
||||
permission, would make you directly or secondarily liable for
|
||||
infringement under applicable copyright law, except executing it on a
|
||||
computer or modifying a private copy. Propagation includes copying,
|
||||
distribution (with or without modification), making available to the
|
||||
public, and in some countries other activities as well.
|
||||
|
||||
To "convey" a work means any kind of propagation that enables other
|
||||
parties to make or receive copies. Mere interaction with a user through
|
||||
a computer network, with no transfer of a copy, is not conveying.
|
||||
|
||||
An interactive user interface displays "Appropriate Legal Notices"
|
||||
to the extent that it includes a convenient and prominently visible
|
||||
feature that (1) displays an appropriate copyright notice, and (2)
|
||||
tells the user that there is no warranty for the work (except to the
|
||||
extent that warranties are provided), that licensees may convey the
|
||||
work under this License, and how to view a copy of this License. If
|
||||
the interface presents a list of user commands or options, such as a
|
||||
menu, a prominent item in the list meets this criterion.
|
||||
|
||||
1. Source Code.
|
||||
|
||||
The "source code" for a work means the preferred form of the work
|
||||
for making modifications to it. "Object code" means any non-source
|
||||
form of a work.
|
||||
|
||||
A "Standard Interface" means an interface that either is an official
|
||||
standard defined by a recognized standards body, or, in the case of
|
||||
interfaces specified for a particular programming language, one that
|
||||
is widely used among developers working in that language.
|
||||
|
||||
The "System Libraries" of an executable work include anything, other
|
||||
than the work as a whole, that (a) is included in the normal form of
|
||||
packaging a Major Component, but which is not part of that Major
|
||||
Component, and (b) serves only to enable use of the work with that
|
||||
Major Component, or to implement a Standard Interface for which an
|
||||
implementation is available to the public in source code form. A
|
||||
"Major Component", in this context, means a major essential component
|
||||
(kernel, window system, and so on) of the specific operating system
|
||||
(if any) on which the executable work runs, or a compiler used to
|
||||
produce the work, or an object code interpreter used to run it.
|
||||
|
||||
The "Corresponding Source" for a work in object code form means all
|
||||
the source code needed to generate, install, and (for an executable
|
||||
work) run the object code and to modify the work, including scripts to
|
||||
control those activities. However, it does not include the work's
|
||||
System Libraries, or general-purpose tools or generally available free
|
||||
programs which are used unmodified in performing those activities but
|
||||
which are not part of the work. For example, Corresponding Source
|
||||
includes interface definition files associated with source files for
|
||||
the work, and the source code for shared libraries and dynamically
|
||||
linked subprograms that the work is specifically designed to require,
|
||||
such as by intimate data communication or control flow between those
|
||||
subprograms and other parts of the work.
|
||||
|
||||
The Corresponding Source need not include anything that users
|
||||
can regenerate automatically from other parts of the Corresponding
|
||||
Source.
|
||||
|
||||
The Corresponding Source for a work in source code form is that
|
||||
same work.
|
||||
|
||||
2. Basic Permissions.
|
||||
|
||||
All rights granted under this License are granted for the term of
|
||||
copyright on the Program, and are irrevocable provided the stated
|
||||
conditions are met. This License explicitly affirms your unlimited
|
||||
permission to run the unmodified Program. The output from running a
|
||||
covered work is covered by this License only if the output, given its
|
||||
content, constitutes a covered work. This License acknowledges your
|
||||
rights of fair use or other equivalent, as provided by copyright law.
|
||||
|
||||
You may make, run and propagate covered works that you do not
|
||||
convey, without conditions so long as your license otherwise remains
|
||||
in force. You may convey covered works to others for the sole purpose
|
||||
of having them make modifications exclusively for you, or provide you
|
||||
with facilities for running those works, provided that you comply with
|
||||
the terms of this License in conveying all material for which you do
|
||||
not control copyright. Those thus making or running the covered works
|
||||
for you must do so exclusively on your behalf, under your direction
|
||||
and control, on terms that prohibit them from making any copies of
|
||||
your copyrighted material outside their relationship with you.
|
||||
|
||||
Conveying under any other circumstances is permitted solely under
|
||||
the conditions stated below. Sublicensing is not allowed; section 10
|
||||
makes it unnecessary.
|
||||
|
||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||
|
||||
No covered work shall be deemed part of an effective technological
|
||||
measure under any applicable law fulfilling obligations under article
|
||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||
similar laws prohibiting or restricting circumvention of such
|
||||
measures.
|
||||
|
||||
When you convey a covered work, you waive any legal power to forbid
|
||||
circumvention of technological measures to the extent such circumvention
|
||||
is effected by exercising rights under this License with respect to
|
||||
the covered work, and you disclaim any intention to limit operation or
|
||||
modification of the work as a means of enforcing, against the work's
|
||||
users, your or third parties' legal rights to forbid circumvention of
|
||||
technological measures.
|
||||
|
||||
4. Conveying Verbatim Copies.
|
||||
|
||||
You may convey verbatim copies of the Program's source code as you
|
||||
receive it, in any medium, provided that you conspicuously and
|
||||
appropriately publish on each copy an appropriate copyright notice;
|
||||
keep intact all notices stating that this License and any
|
||||
non-permissive terms added in accord with section 7 apply to the code;
|
||||
keep intact all notices of the absence of any warranty; and give all
|
||||
recipients a copy of this License along with the Program.
|
||||
|
||||
You may charge any price or no price for each copy that you convey,
|
||||
and you may offer support or warranty protection for a fee.
|
||||
|
||||
5. Conveying Modified Source Versions.
|
||||
|
||||
You may convey a work based on the Program, or the modifications to
|
||||
produce it from the Program, in the form of source code under the
|
||||
terms of section 4, provided that you also meet all of these conditions:
|
||||
|
||||
a) The work must carry prominent notices stating that you modified
|
||||
it, and giving a relevant date.
|
||||
|
||||
b) The work must carry prominent notices stating that it is
|
||||
released under this License and any conditions added under section
|
||||
7. This requirement modifies the requirement in section 4 to
|
||||
"keep intact all notices".
|
||||
|
||||
c) You must license the entire work, as a whole, under this
|
||||
License to anyone who comes into possession of a copy. This
|
||||
License will therefore apply, along with any applicable section 7
|
||||
additional terms, to the whole of the work, and all its parts,
|
||||
regardless of how they are packaged. This License gives no
|
||||
permission to license the work in any other way, but it does not
|
||||
invalidate such permission if you have separately received it.
|
||||
|
||||
d) If the work has interactive user interfaces, each must display
|
||||
Appropriate Legal Notices; however, if the Program has interactive
|
||||
interfaces that do not display Appropriate Legal Notices, your
|
||||
work need not make them do so.
|
||||
|
||||
A compilation of a covered work with other separate and independent
|
||||
works, which are not by their nature extensions of the covered work,
|
||||
and which are not combined with it such as to form a larger program,
|
||||
in or on a volume of a storage or distribution medium, is called an
|
||||
"aggregate" if the compilation and its resulting copyright are not
|
||||
used to limit the access or legal rights of the compilation's users
|
||||
beyond what the individual works permit. Inclusion of a covered work
|
||||
in an aggregate does not cause this License to apply to the other
|
||||
parts of the aggregate.
|
||||
|
||||
6. Conveying Non-Source Forms.
|
||||
|
||||
You may convey a covered work in object code form under the terms
|
||||
of sections 4 and 5, provided that you also convey the
|
||||
machine-readable Corresponding Source under the terms of this License,
|
||||
in one of these ways:
|
||||
|
||||
a) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by the
|
||||
Corresponding Source fixed on a durable physical medium
|
||||
customarily used for software interchange.
|
||||
|
||||
b) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by a
|
||||
written offer, valid for at least three years and valid for as
|
||||
long as you offer spare parts or customer support for that product
|
||||
model, to give anyone who possesses the object code either (1) a
|
||||
copy of the Corresponding Source for all the software in the
|
||||
product that is covered by this License, on a durable physical
|
||||
medium customarily used for software interchange, for a price no
|
||||
more than your reasonable cost of physically performing this
|
||||
conveying of source, or (2) access to copy the
|
||||
Corresponding Source from a network server at no charge.
|
||||
|
||||
c) Convey individual copies of the object code with a copy of the
|
||||
written offer to provide the Corresponding Source. This
|
||||
alternative is allowed only occasionally and noncommercially, and
|
||||
only if you received the object code with such an offer, in accord
|
||||
with subsection 6b.
|
||||
|
||||
d) Convey the object code by offering access from a designated
|
||||
place (gratis or for a charge), and offer equivalent access to the
|
||||
Corresponding Source in the same way through the same place at no
|
||||
further charge. You need not require recipients to copy the
|
||||
Corresponding Source along with the object code. If the place to
|
||||
copy the object code is a network server, the Corresponding Source
|
||||
may be on a different server (operated by you or a third party)
|
||||
that supports equivalent copying facilities, provided you maintain
|
||||
clear directions next to the object code saying where to find the
|
||||
Corresponding Source. Regardless of what server hosts the
|
||||
Corresponding Source, you remain obligated to ensure that it is
|
||||
available for as long as needed to satisfy these requirements.
|
||||
|
||||
e) Convey the object code using peer-to-peer transmission, provided
|
||||
you inform other peers where the object code and Corresponding
|
||||
Source of the work are being offered to the general public at no
|
||||
charge under subsection 6d.
|
||||
|
||||
A separable portion of the object code, whose source code is excluded
|
||||
from the Corresponding Source as a System Library, need not be
|
||||
included in conveying the object code work.
|
||||
|
||||
A "User Product" is either (1) a "consumer product", which means any
|
||||
tangible personal property which is normally used for personal, family,
|
||||
or household purposes, or (2) anything designed or sold for incorporation
|
||||
into a dwelling. In determining whether a product is a consumer product,
|
||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||
product received by a particular user, "normally used" refers to a
|
||||
typical or common use of that class of product, regardless of the status
|
||||
of the particular user or of the way in which the particular user
|
||||
actually uses, or expects or is expected to use, the product. A product
|
||||
is a consumer product regardless of whether the product has substantial
|
||||
commercial, industrial or non-consumer uses, unless such uses represent
|
||||
the only significant mode of use of the product.
|
||||
|
||||
"Installation Information" for a User Product means any methods,
|
||||
procedures, authorization keys, or other information required to install
|
||||
and execute modified versions of a covered work in that User Product from
|
||||
a modified version of its Corresponding Source. The information must
|
||||
suffice to ensure that the continued functioning of the modified object
|
||||
code is in no case prevented or interfered with solely because
|
||||
modification has been made.
|
||||
|
||||
If you convey an object code work under this section in, or with, or
|
||||
specifically for use in, a User Product, and the conveying occurs as
|
||||
part of a transaction in which the right of possession and use of the
|
||||
User Product is transferred to the recipient in perpetuity or for a
|
||||
fixed term (regardless of how the transaction is characterized), the
|
||||
Corresponding Source conveyed under this section must be accompanied
|
||||
by the Installation Information. But this requirement does not apply
|
||||
if neither you nor any third party retains the ability to install
|
||||
modified object code on the User Product (for example, the work has
|
||||
been installed in ROM).
|
||||
|
||||
The requirement to provide Installation Information does not include a
|
||||
requirement to continue to provide support service, warranty, or updates
|
||||
for a work that has been modified or installed by the recipient, or for
|
||||
the User Product in which it has been modified or installed. Access to a
|
||||
network may be denied when the modification itself materially and
|
||||
adversely affects the operation of the network or violates the rules and
|
||||
protocols for communication across the network.
|
||||
|
||||
Corresponding Source conveyed, and Installation Information provided,
|
||||
in accord with this section must be in a format that is publicly
|
||||
documented (and with an implementation available to the public in
|
||||
source code form), and must require no special password or key for
|
||||
unpacking, reading or copying.
|
||||
|
||||
7. Additional Terms.
|
||||
|
||||
"Additional permissions" are terms that supplement the terms of this
|
||||
License by making exceptions from one or more of its conditions.
|
||||
Additional permissions that are applicable to the entire Program shall
|
||||
be treated as though they were included in this License, to the extent
|
||||
that they are valid under applicable law. If additional permissions
|
||||
apply only to part of the Program, that part may be used separately
|
||||
under those permissions, but the entire Program remains governed by
|
||||
this License without regard to the additional permissions.
|
||||
|
||||
When you convey a copy of a covered work, you may at your option
|
||||
remove any additional permissions from that copy, or from any part of
|
||||
it. (Additional permissions may be written to require their own
|
||||
removal in certain cases when you modify the work.) You may place
|
||||
additional permissions on material, added by you to a covered work,
|
||||
for which you have or can give appropriate copyright permission.
|
||||
|
||||
Notwithstanding any other provision of this License, for material you
|
||||
add to a covered work, you may (if authorized by the copyright holders of
|
||||
that material) supplement the terms of this License with terms:
|
||||
|
||||
a) Disclaiming warranty or limiting liability differently from the
|
||||
terms of sections 15 and 16 of this License; or
|
||||
|
||||
b) Requiring preservation of specified reasonable legal notices or
|
||||
author attributions in that material or in the Appropriate Legal
|
||||
Notices displayed by works containing it; or
|
||||
|
||||
c) Prohibiting misrepresentation of the origin of that material, or
|
||||
requiring that modified versions of such material be marked in
|
||||
reasonable ways as different from the original version; or
|
||||
|
||||
d) Limiting the use for publicity purposes of names of licensors or
|
||||
authors of the material; or
|
||||
|
||||
e) Declining to grant rights under trademark law for use of some
|
||||
trade names, trademarks, or service marks; or
|
||||
|
||||
f) Requiring indemnification of licensors and authors of that
|
||||
material by anyone who conveys the material (or modified versions of
|
||||
it) with contractual assumptions of liability to the recipient, for
|
||||
any liability that these contractual assumptions directly impose on
|
||||
those licensors and authors.
|
||||
|
||||
All other non-permissive additional terms are considered "further
|
||||
restrictions" within the meaning of section 10. If the Program as you
|
||||
received it, or any part of it, contains a notice stating that it is
|
||||
governed by this License along with a term that is a further
|
||||
restriction, you may remove that term. If a license document contains
|
||||
a further restriction but permits relicensing or conveying under this
|
||||
License, you may add to a covered work material governed by the terms
|
||||
of that license document, provided that the further restriction does
|
||||
not survive such relicensing or conveying.
|
||||
|
||||
If you add terms to a covered work in accord with this section, you
|
||||
must place, in the relevant source files, a statement of the
|
||||
additional terms that apply to those files, or a notice indicating
|
||||
where to find the applicable terms.
|
||||
|
||||
Additional terms, permissive or non-permissive, may be stated in the
|
||||
form of a separately written license, or stated as exceptions;
|
||||
the above requirements apply either way.
|
||||
|
||||
8. Termination.
|
||||
|
||||
You may not propagate or modify a covered work except as expressly
|
||||
provided under this License. Any attempt otherwise to propagate or
|
||||
modify it is void, and will automatically terminate your rights under
|
||||
this License (including any patent licenses granted under the third
|
||||
paragraph of section 11).
|
||||
|
||||
However, if you cease all violation of this License, then your
|
||||
license from a particular copyright holder is reinstated (a)
|
||||
provisionally, unless and until the copyright holder explicitly and
|
||||
finally terminates your license, and (b) permanently, if the copyright
|
||||
holder fails to notify you of the violation by some reasonable means
|
||||
prior to 60 days after the cessation.
|
||||
|
||||
Moreover, your license from a particular copyright holder is
|
||||
reinstated permanently if the copyright holder notifies you of the
|
||||
violation by some reasonable means, this is the first time you have
|
||||
received notice of violation of this License (for any work) from that
|
||||
copyright holder, and you cure the violation prior to 30 days after
|
||||
your receipt of the notice.
|
||||
|
||||
Termination of your rights under this section does not terminate the
|
||||
licenses of parties who have received copies or rights from you under
|
||||
this License. If your rights have been terminated and not permanently
|
||||
reinstated, you do not qualify to receive new licenses for the same
|
||||
material under section 10.
|
||||
|
||||
9. Acceptance Not Required for Having Copies.
|
||||
|
||||
You are not required to accept this License in order to receive or
|
||||
run a copy of the Program. Ancillary propagation of a covered work
|
||||
occurring solely as a consequence of using peer-to-peer transmission
|
||||
to receive a copy likewise does not require acceptance. However,
|
||||
nothing other than this License grants you permission to propagate or
|
||||
modify any covered work. These actions infringe copyright if you do
|
||||
not accept this License. Therefore, by modifying or propagating a
|
||||
covered work, you indicate your acceptance of this License to do so.
|
||||
|
||||
10. Automatic Licensing of Downstream Recipients.
|
||||
|
||||
Each time you convey a covered work, the recipient automatically
|
||||
receives a license from the original licensors, to run, modify and
|
||||
propagate that work, subject to this License. You are not responsible
|
||||
for enforcing compliance by third parties with this License.
|
||||
|
||||
An "entity transaction" is a transaction transferring control of an
|
||||
organization, or substantially all assets of one, or subdividing an
|
||||
organization, or merging organizations. If propagation of a covered
|
||||
work results from an entity transaction, each party to that
|
||||
transaction who receives a copy of the work also receives whatever
|
||||
licenses to the work the party's predecessor in interest had or could
|
||||
give under the previous paragraph, plus a right to possession of the
|
||||
Corresponding Source of the work from the predecessor in interest, if
|
||||
the predecessor has it or can get it with reasonable efforts.
|
||||
|
||||
You may not impose any further restrictions on the exercise of the
|
||||
rights granted or affirmed under this License. For example, you may
|
||||
not impose a license fee, royalty, or other charge for exercise of
|
||||
rights granted under this License, and you may not initiate litigation
|
||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||
any patent claim is infringed by making, using, selling, offering for
|
||||
sale, or importing the Program or any portion of it.
|
||||
|
||||
11. Patents.
|
||||
|
||||
A "contributor" is a copyright holder who authorizes use under this
|
||||
License of the Program or a work on which the Program is based. The
|
||||
work thus licensed is called the contributor's "contributor version".
|
||||
|
||||
A contributor's "essential patent claims" are all patent claims
|
||||
owned or controlled by the contributor, whether already acquired or
|
||||
hereafter acquired, that would be infringed by some manner, permitted
|
||||
by this License, of making, using, or selling its contributor version,
|
||||
but do not include claims that would be infringed only as a
|
||||
consequence of further modification of the contributor version. For
|
||||
purposes of this definition, "control" includes the right to grant
|
||||
patent sublicenses in a manner consistent with the requirements of
|
||||
this License.
|
||||
|
||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||
patent license under the contributor's essential patent claims, to
|
||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||
propagate the contents of its contributor version.
|
||||
|
||||
In the following three paragraphs, a "patent license" is any express
|
||||
agreement or commitment, however denominated, not to enforce a patent
|
||||
(such as an express permission to practice a patent or covenant not to
|
||||
sue for patent infringement). To "grant" such a patent license to a
|
||||
party means to make such an agreement or commitment not to enforce a
|
||||
patent against the party.
|
||||
|
||||
If you convey a covered work, knowingly relying on a patent license,
|
||||
and the Corresponding Source of the work is not available for anyone
|
||||
to copy, free of charge and under the terms of this License, through a
|
||||
publicly available network server or other readily accessible means,
|
||||
then you must either (1) cause the Corresponding Source to be so
|
||||
available, or (2) arrange to deprive yourself of the benefit of the
|
||||
patent license for this particular work, or (3) arrange, in a manner
|
||||
consistent with the requirements of this License, to extend the patent
|
||||
license to downstream recipients. "Knowingly relying" means you have
|
||||
actual knowledge that, but for the patent license, your conveying the
|
||||
covered work in a country, or your recipient's use of the covered work
|
||||
in a country, would infringe one or more identifiable patents in that
|
||||
country that you have reason to believe are valid.
|
||||
|
||||
If, pursuant to or in connection with a single transaction or
|
||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||
covered work, and grant a patent license to some of the parties
|
||||
receiving the covered work authorizing them to use, propagate, modify
|
||||
or convey a specific copy of the covered work, then the patent license
|
||||
you grant is automatically extended to all recipients of the covered
|
||||
work and works based on it.
|
||||
|
||||
A patent license is "discriminatory" if it does not include within
|
||||
the scope of its coverage, prohibits the exercise of, or is
|
||||
conditioned on the non-exercise of one or more of the rights that are
|
||||
specifically granted under this License. You may not convey a covered
|
||||
work if you are a party to an arrangement with a third party that is
|
||||
in the business of distributing software, under which you make payment
|
||||
to the third party based on the extent of your activity of conveying
|
||||
the work, and under which the third party grants, to any of the
|
||||
parties who would receive the covered work from you, a discriminatory
|
||||
patent license (a) in connection with copies of the covered work
|
||||
conveyed by you (or copies made from those copies), or (b) primarily
|
||||
for and in connection with specific products or compilations that
|
||||
contain the covered work, unless you entered into that arrangement,
|
||||
or that patent license was granted, prior to 28 March 2007.
|
||||
|
||||
Nothing in this License shall be construed as excluding or limiting
|
||||
any implied license or other defenses to infringement that may
|
||||
otherwise be available to you under applicable patent law.
|
||||
|
||||
12. No Surrender of Others' Freedom.
|
||||
|
||||
If conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot convey a
|
||||
covered work so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you may
|
||||
not convey it at all. For example, if you agree to terms that obligate you
|
||||
to collect a royalty for further conveying from those to whom you convey
|
||||
the Program, the only way you could satisfy both those terms and this
|
||||
License would be to refrain entirely from conveying the Program.
|
||||
|
||||
13. Use with the GNU Affero General Public License.
|
||||
|
||||
Notwithstanding any other provision of this License, you have
|
||||
permission to link or combine any covered work with a work licensed
|
||||
under version 3 of the GNU Affero General Public License into a single
|
||||
combined work, and to convey the resulting work. The terms of this
|
||||
License will continue to apply to the part which is the covered work,
|
||||
but the special requirements of the GNU Affero General Public License,
|
||||
section 13, concerning interaction through a network will apply to the
|
||||
combination as such.
|
||||
|
||||
14. Revised Versions of this License.
|
||||
|
||||
The Free Software Foundation may publish revised and/or new versions of
|
||||
the GNU General Public License from time to time. Such new versions will
|
||||
be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the
|
||||
Program specifies that a certain numbered version of the GNU General
|
||||
Public License "or any later version" applies to it, you have the
|
||||
option of following the terms and conditions either of that numbered
|
||||
version or of any later version published by the Free Software
|
||||
Foundation. If the Program does not specify a version number of the
|
||||
GNU General Public License, you may choose any version ever published
|
||||
by the Free Software Foundation.
|
||||
|
||||
If the Program specifies that a proxy can decide which future
|
||||
versions of the GNU General Public License can be used, that proxy's
|
||||
public statement of acceptance of a version permanently authorizes you
|
||||
to choose that version for the Program.
|
||||
|
||||
Later license versions may give you additional or different
|
||||
permissions. However, no additional obligations are imposed on any
|
||||
author or copyright holder as a result of your choosing to follow a
|
||||
later version.
|
||||
|
||||
15. Disclaimer of Warranty.
|
||||
|
||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||
|
||||
16. Limitation of Liability.
|
||||
|
||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||
SUCH DAMAGES.
|
||||
|
||||
17. Interpretation of Sections 15 and 16.
|
||||
|
||||
If the disclaimer of warranty and limitation of liability provided
|
||||
above cannot be given local legal effect according to their terms,
|
||||
reviewing courts shall apply local law that most closely approximates
|
||||
an absolute waiver of all civil liability in connection with the
|
||||
Program, unless a warranty or assumption of liability accompanies a
|
||||
copy of the Program in return for a fee.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
state the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
{one line to give the program's name and a brief idea of what it does.}
|
||||
Copyright (C) {year} {name of author}
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program does terminal interaction, make it output a short
|
||||
notice like this when it starts in an interactive mode:
|
||||
|
||||
{project} Copyright (C) {year} {fullname}
|
||||
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, your program's commands
|
||||
might be different; for a GUI interface, you would use an "about box".
|
||||
|
||||
You should also get your employer (if you work as a programmer) or school,
|
||||
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
||||
For more information on this, and how to apply and follow the GNU GPL, see
|
||||
<http://www.gnu.org/licenses/>.
|
||||
|
||||
The GNU General Public License does not permit incorporating your program
|
||||
into proprietary programs. If your program is a subroutine library, you
|
||||
may consider it more useful to permit linking proprietary applications with
|
||||
the library. If this is what you want to do, use the GNU Lesser General
|
||||
Public License instead of this License. But first, please read
|
||||
<http://www.gnu.org/philosophy/why-not-lgpl.html>.
|
||||
BIN
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/assets/images/animation.jpg
vendored
Normal file
|
After Width: | Height: | Size: 4.7 KiB |
BIN
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/assets/images/conditions.jpg
vendored
Normal file
|
After Width: | Height: | Size: 20 KiB |
BIN
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/assets/images/css.jpg
vendored
Normal file
|
After Width: | Height: | Size: 8.7 KiB |
BIN
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/assets/images/neve.png
vendored
Normal file
|
After Width: | Height: | Size: 1.6 KiB |
|
After Width: | Height: | Size: 36 KiB |
|
After Width: | Height: | Size: 26 KiB |
|
After Width: | Height: | Size: 173 KiB |
|
After Width: | Height: | Size: 88 KiB |
BIN
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/assets/images/sparks.png
vendored
Normal file
|
After Width: | Height: | Size: 2.6 KiB |
BIN
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/assets/images/team.jpg
vendored
Normal file
|
After Width: | Height: | Size: 142 KiB |
|
After Width: | Height: | Size: 9.4 KiB |
BIN
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/assets/images/wplk.png
vendored
Normal file
|
After Width: | Height: | Size: 1.9 KiB |
@@ -0,0 +1 @@
|
||||
<?php return array('dependencies' => array('react', 'wp-components', 'wp-element'), 'version' => '8d9c74cada5a40e4082b');
|
||||
@@ -0,0 +1 @@
|
||||
<?php return array('dependencies' => array('react', 'wp-components', 'wp-element'), 'version' => '80cf3f304af2c0a33ac3');
|
||||
@@ -0,0 +1 @@
|
||||
#ti-sdk-float-widget .ti-float-button-enter{opacity:.01}#ti-sdk-float-widget .ti-float-button-enter.ti-float-button-enter-active{opacity:1;transition:opacity 200ms ease-in}#ti-sdk-float-widget .ti-float-button-leave{opacity:1}#ti-sdk-float-widget .ti-float-button-leave.ti-float-button-leave-active{opacity:.01;transition:opacity 200ms ease-in}#ti-sdk-float-widget .ti-float-button{-webkit-appearance:none;-moz-appearance:none;appearance:none;border:0;padding:0;width:51px;height:51px;background-color:#f0f0f1;border-radius:50%;display:flex;align-items:center;justify-content:center;position:fixed;bottom:32px;right:24px}#ti-sdk-float-widget .ti-float-button:hover,#ti-sdk-float-widget .ti-float-button:focus{cursor:pointer}#ti-sdk-float-widget .ti-float-button>img,#ti-sdk-float-widget .ti-float-button>span{position:absolute;opacity:0;transition:opacity 200ms ease-in}#ti-sdk-float-widget .ti-float-button>img.active,#ti-sdk-float-widget .ti-float-button>span.active{opacity:1}#ti-sdk-float-widget .ti-float-button>img.active:hover,#ti-sdk-float-widget .ti-float-button>img.active:focus,#ti-sdk-float-widget .ti-float-button>span.active:hover,#ti-sdk-float-widget .ti-float-button>span.active:focus{opacity:.9}#ti-sdk-float-widget .ti-float-button>span{--ti-icon-size: 42px;font-size:var(--ti-icon-size);width:var(--ti-icon-size);height:var(--ti-icon-size);border-radius:50%;color:#fff;background-color:var(--ti-float-primary-color, #2271b1)}#ti-sdk-float-widget .ti-float-button>img{width:42px;height:42px}.ti-float-widget-panel{width:275px;border-radius:8px;background-color:#fff;box-shadow:0px 2px 15px 0px rgba(0,0,0,.1);margin:12px}.ti-float-widget-panel .ti-float-widget-panel__header{border-radius:8px 8px 0 0;padding:12px 12px 48px 12px;background:linear-gradient(75deg, var(--ti-float-primary-color, #2271b1) 15%, rgba(255, 255, 255, 0.65) 500%);display:flex;flex-direction:column;align-items:flex-start;gap:8px}.ti-float-widget-panel .ti-float-widget-panel__header img{background:#fff;border-radius:50%;width:42px;height:42px;box-shadow:0 0 0px 5px #fff;margin:5px}.ti-float-widget-panel .ti-float-widget-panel__header p,.ti-float-widget-panel .ti-float-widget-panel__header h3{color:#fff}.ti-float-widget-panel .ti-float-widget-panel__header p{font-size:14px}.ti-float-widget-panel .ti-float-widget-panel__header h3{font-size:20px;margin:0}.ti-float-widget-panel .ti-float-widget-panel__content{padding:12px;display:flex;flex-direction:column;gap:8px;margin-top:-36px}.ti-float-widget-panel .ti-float-widget-panel__content>a{display:flex;gap:8px;align-items:center;justify-content:flex-start;background:linear-gradient(-75deg, rgb(255, 255, 255) 0%, rgb(255, 255, 255) 100%);padding:12px;border-radius:8px;box-shadow:0px 2px 15px 0px rgba(0,0,0,.15);color:#000;font-size:14px;text-decoration:none;transition:background 200ms ease-in}.ti-float-widget-panel .ti-float-widget-panel__content>a:hover{background:linear-gradient(-75deg, var(--ti-float-primary-color, #2271b1) -150%, rgb(255, 255, 255) 50%)}.ti-float-widget-panel .ti-float-widget-panel__content>a:focus-visible{box-shadow:0 0 0 2px var(--ti-float-primary-color, #2271b1)}.ti-float-widget-panel .ti-float-widget-panel__content>a .dashicons{background:#2271b1;color:#efefef;font-size:16px;width:24px;height:24px;padding:4px;border-radius:4px}.ti-float-widget-panel .ti-float-widget-panel__content>a .dashicons.dashicons-book-alt{background:#fdd7ae;color:#ff9d32}.ti-float-widget-panel .ti-float-widget-panel__content>a .dashicons.dashicons-format-status{background:#96d7c7;color:#30955e}.ti-float-widget-panel .ti-float-widget-panel__content>a .dashicons.dashicons-admin-tools{background:#dac8ff;color:#783ff2}.ti-float-widget-panel .ti-float-widget-panel__content>a .dashicons.dashicons-superhero-alt{background:#ffe39e;color:#fdb600}.ti-float-widget-panel .ti-float-widget-panel__content>a .dashicons.dashicons-star-filled{background:#bfd0ff;color:#3068ff}
|
||||
@@ -0,0 +1 @@
|
||||
(()=>{"use strict";const e=window.React,t=window.wp.element;function l({isActive:t,onToggle:l}){const{logoUrl:n,primaryColor:a,strings:o,links:i}=window.tiSDKFloatData;return(0,e.createElement)("div",{className:"ti-float-widget-panel",style:{"--ti-float-primary-color":a}},(0,e.createElement)("div",{className:"ti-float-widget-panel__header"},(0,e.createElement)("img",{src:n,alt:"Site Logo"}),o.panelGreet&&(0,e.createElement)("p",null,o.panelGreet),o.panelTitle&&(0,e.createElement)("h3",null,o.panelTitle)),(0,e.createElement)("div",{className:"ti-float-widget-panel__content"},i&&i.map(((t,l)=>(0,e.createElement)("a",{key:"ti-float_link_"+l,href:t.link,target:!0===t?.internal?"_self":"_blank"},(0,e.createElement)("span",{className:"dashicons "+t.icon}),t.title))),(0,e.createElement)("a",{href:"#ti-toggle-widget-float",className:"screen-reader-shortcut"},o.closeToggle?o.closeToggle:"Close")))}const n=window.wp.components;function a({isActive:a,onToggle:o,onClose:i}){const{logoUrl:s,strings:c}=window.tiSDKFloatData,r=c.toggleButton,m=a?"":"active",d=a?"active":"";return(0,e.createElement)(t.Fragment,null,(0,e.createElement)("button",{id:"ti-toggle-widget-float",tabIndex:"0",className:"ti-float-button",onClick:o,"aria-label":"","aria-pressed":a,"aria-disabled":!1,title:r},(0,e.createElement)("img",{className:"ti-float-logo "+m,src:s}),(0,e.createElement)("span",{className:"dashicons dashicons-no-alt ti-float-close-icon "+d}),(0,e.createElement)("span",{className:"screen-reader-text"},r),a&&(0,e.createElement)(n.Popover,{variant:"unstyled",placement:"top-end",onFocusOutside:()=>{console.log("Focus Outside"),i()}},(0,e.createElement)(l,null))))}function o(){const[l,n]=(0,t.useState)(!1);return(0,e.createElement)(t.Fragment,null,(0,e.createElement)(a,{isActive:l,onToggle:()=>{n(!l)},onClose:()=>{n(!1)}}))}document.addEventListener("DOMContentLoaded",(()=>{const l=document.querySelector("#ti-sdk-float-widget");l&&(0,t.render)((0,e.createElement)(o,null),l)}))})();
|
||||
@@ -0,0 +1 @@
|
||||
<?php return array('dependencies' => array('react', 'wp-block-editor', 'wp-components', 'wp-compose', 'wp-data', 'wp-edit-post', 'wp-element', 'wp-hooks', 'wp-plugins'), 'version' => '1e086c2d21f2850672d5');
|
||||
@@ -0,0 +1 @@
|
||||
.ti-sdk-om-notice{--wp-admin-theme-color: #3858E9;--wp-admin-theme-color-darker-10: #2e47ba;position:relative;padding:0;border-left-color:#3858e9}.ti-sdk-om-notice .content{background:rgba(255,255,255,.75);display:flex;align-items:center;padding:15px 20px}.ti-sdk-om-notice img{max-width:100px;margin-right:20px;display:none}.ti-sdk-om-notice .description{font-size:14px;margin-bottom:10px;color:#000}.ti-sdk-om-notice .actions{margin-top:auto;display:flex;margin-bottom:0;gap:20px}.ti-sdk-om-notice form{display:flex;align-items:center;gap:10px}.ti-sdk-om-notice .om-progress span:not(.dashicons){font-weight:500}.ti-sdk-om-notice input{border-radius:0;min-width:250px}.ti-sdk-om-notice a.components-button{display:flex;align-items:center;justify-content:center}.ti-sdk-om-notice .is-link{text-decoration:none;display:flex;align-items:center}.ti-sdk-om-notice .is-link span{line-height:normal}.ti-sdk-om-notice .dashicons{margin-right:2px;text-decoration:none}.ti-sdk-om-notice .done{display:flex;flex-direction:column;align-items:flex-start}.ti-sdk-om-notice .done a{width:auto}.compat-field-optimole th{display:none !important}.compat-field-optimole td{width:100% !important}.compat-field-optimole .ti-sdk-om-notice{margin:0}.om-notice-dismiss{right:10px;top:10px;text-decoration:none !important;position:absolute}.om-notice-dismiss:before{content:none}.ti-om-stack-wrap .om-stack-notice{--wp-admin-theme-color: #3858E9;--wp-admin-theme-color-darker-10: #2e47ba;gap:10px;border-left:4px solid #3858e9;display:flex;flex-direction:column;align-items:start;position:relative;padding:20px 10px}.ti-om-stack-wrap .om-stack-notice>span{display:none}.ti-om-stack-wrap .om-stack-notice img{max-width:90px !important}.ti-om-stack-wrap .om-stack-notice h2{font-size:18px;margin:30px auto 10px;font-weight:600}.ti-om-stack-wrap .om-stack-notice p{font-size:13px;max-width:250px;margin:0;line-height:normal}.ti-om-stack-wrap .om-stack-notice i{margin:10px 0 0;font-size:12px;color:#757575}.ti-om-stack-wrap .om-stack-notice .cta{padding:10px 25px !important}.ti-om-stack-wrap .om-stack-notice .om-notice-dismiss{color:inherit}.ti-om-stack-wrap .om-stack-notice input{border-radius:0}.ti-om-stack-wrap .om-stack-notice .done{margin-top:15px;display:grid;gap:10px}.ti-om-stack-wrap .om-stack-notice .done p{font-size:15px;font-weight:500}.ti-om-stack-wrap .om-stack-notice .om-progress{margin:10px 0}.block-editor-block-inspector .ti-om-stack-wrap{border-top:1px solid #e0e0e0}.om-progress{gap:5px;font-size:13px;display:flex;align-items:center}.om-progress .spin{animation:om-rotation 2s infinite linear}@keyframes om-rotation{0%{transform:rotate(0deg)}100%{transform:rotate(359deg)}}.ti-sdk-om-promo.hidden{display:none}.media-sidebar .ti-sdk-om-notice input{min-width:unset;flex-grow:1}.media-sidebar .ti-sdk-om-notice .description{margin-bottom:10px}.media-sidebar .ti-sdk-om-notice .content{padding:15px 10px}.media-sidebar .ti-sdk-om-notice .actions{gap:10px}.media-sidebar .ti-sdk-om-notice form{flex-wrap:wrap;justify-content:center}.attachment-info .ti-sdk-om-notice input{min-width:unset;flex-grow:1}.attachment-info .ti-sdk-om-notice form{flex-wrap:wrap;justify-content:center}.ti-sdk-rop-notice{position:relative;padding:10px}.ti-sdk-rop-notice .rop-notice-actions{display:flex;gap:10px}.ti-sdk-rop-notice p{padding:0 10px 0 0}.ti-sdk-optimole-post-publish{padding:0 !important}@media screen and (min-width: 768px){.ti-sdk-om-notice img{display:block}}@media screen and (min-width: 1200px){.attachment-info .ti-sdk-om-notice form{flex-wrap:unset}}
|
||||
@@ -0,0 +1 @@
|
||||
<?php return array('dependencies' => array(), 'version' => '44eb6f2a376d36991f76');
|
||||
@@ -0,0 +1 @@
|
||||
(()=>{"use strict";let r=!1,e=!1;const t=[],o=new Proxy({},{get:(o,i,n)=>(...o)=>(async(o,...i)=>{if(e){if(window.formbricks){const r=window.formbricks,e=o;await r[e](...i)}}else if("setup"===o){if(r)return void console.warn("🧱 Formbricks - Warning: Formbricks is already initializing.");r=!0;const o=i[0],{appUrl:n,environmentId:s}=o;if(!n)return void console.error("🧱 Formbricks - Error: appUrl is required");if(!s)return void console.error("🧱 Formbricks - Error: environmentId is required");if((await(async r=>{if(!window.formbricks){const e=document.createElement("script");e.type="text/javascript",e.src=`${r}/js/formbricks.umd.cjs`,e.async=!0;const t=async()=>new Promise(((r,t)=>{const o=setTimeout((()=>{t(new Error("Formbricks SDK loading timed out"))}),1e4);e.onload=()=>{clearTimeout(o),r()},e.onerror=()=>{clearTimeout(o),t(new Error("Failed to load Formbricks SDK"))}}));document.head.appendChild(e);try{return await t(),{ok:!0,data:void 0}}catch(r){return{ok:!1,error:new Error(r.message??"Failed to load Formbricks SDK")}}}return{ok:!0,data:void 0}})(n)).ok&&window.formbricks){const o=window.formbricks;o.setup(...i),r=!1,e=!0;for(const{prop:r,args:e}of t)"function"==typeof o[r]?o[r](...e):console.error(`🧱 Formbricks - Error: Method ${r} does not exist on formbricks`)}}else console.warn("🧱 Formbricks - Warning: Formbricks not initialized. This method will be queued and executed after initialization."),t.push({prop:o,args:i})})(i,...o)});document.addEventListener("DOMContentLoaded",(()=>{window.tsdk_formbricks={init:async r=>{var e,t;"object"==typeof r&&null!==r||(r={});const i={...window.tsdk_survey_data,...r,attributes:{...null!==(e=window.tsdk_survey_data.attributes)&&void 0!==e?e:{},...null!==(t=r.attributes)&&void 0!==t?t:{}}},{environmentId:n,appUrl:s,attributes:a,userId:d}=i;await(o?.setup({environmentId:n,appUrl:s})),o?.setAttributes(a),function(){const r=localStorage.getItem("formbricks-js");if(!r)return!0;try{const e=JSON.parse(r);if(e?.user?.data?.userId)return!1}catch(r){console.warn(r)}return!0}()&&o?.setUserId(d)}};let r=null;var e;e=window.tsdk_survey_data?.attributes?.install_days_number,isNaN(e)||"boolean"==typeof e||(r=setTimeout((()=>{window.tsdk_formbricks?.init()}),350)),window.addEventListener("themeisle:survey:trigger:cancel",(()=>{clearTimeout(r)})),window.dispatchEvent(new Event("themeisle:survey:loaded"))}))})();
|
||||
@@ -0,0 +1 @@
|
||||
<?php return array('dependencies' => array(), 'version' => 'f35e2c927fed8015fa08');
|
||||
35
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/composer.json
vendored
Normal file
@@ -0,0 +1,35 @@
|
||||
{
|
||||
"name": "codeinwp/themeisle-sdk",
|
||||
"description": "Themeisle SDK.",
|
||||
"type": "library",
|
||||
"keywords": [
|
||||
"wordpress"
|
||||
],
|
||||
"homepage": "https://github.com/Codeinwp/themeisle-sdk",
|
||||
"license": "GPL-2.0+",
|
||||
"authors": [
|
||||
{
|
||||
"name": "ThemeIsle team",
|
||||
"email": "friends@themeisle.com",
|
||||
"homepage": "https://themeisle.com"
|
||||
}
|
||||
],
|
||||
"version": "3.3.50",
|
||||
"require-dev": {
|
||||
"codeinwp/phpcs-ruleset": "dev-main",
|
||||
"yoast/phpunit-polyfills": "^2.0"
|
||||
},
|
||||
"scripts": {
|
||||
"format": "phpcbf --standard=phpcs.xml --report-summary --report-source -s --runtime-set testVersion 7.2- ",
|
||||
"phpcs": "phpcs --standard=phpcs.xml -s --runtime-set testVersion 7.2-",
|
||||
"lint": "composer run-script phpcs"
|
||||
},
|
||||
"support": {
|
||||
"issues": "https://github.com/Codeinwp/themeisle-sdk/issues"
|
||||
},
|
||||
"config": {
|
||||
"allow-plugins": {
|
||||
"dealerdirect/phpcodesniffer-composer-installer": true
|
||||
}
|
||||
}
|
||||
}
|
||||
3
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/index.php
vendored
Normal file
@@ -0,0 +1,3 @@
|
||||
<?php
|
||||
// phpcs:ignoreFile
|
||||
// Nothing here.
|
||||
284
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/load.php
vendored
Normal file
@@ -0,0 +1,284 @@
|
||||
<?php
|
||||
/**
|
||||
* Loader for the ThemeIsleSDK
|
||||
*
|
||||
* Logic for loading always the latest SDK from the installed themes/plugins.
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-2.0.php GNU Public License
|
||||
* @since 1.1.0
|
||||
*/
|
||||
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
return;
|
||||
}
|
||||
// Current SDK version and path.
|
||||
$themeisle_sdk_version = '3.3.50';
|
||||
$themeisle_sdk_path = dirname( __FILE__ );
|
||||
|
||||
global $themeisle_sdk_max_version;
|
||||
global $themeisle_sdk_max_path;
|
||||
|
||||
// If this is the latest SDK and it comes from a theme, we should load licenser separately.
|
||||
$themeisle_sdk_relative_licenser_path = '/src/Modules/Licenser.php';
|
||||
|
||||
global $themeisle_sdk_abs_licenser_path;
|
||||
if ( ! is_file( $themeisle_sdk_path . $themeisle_sdk_relative_licenser_path ) && ! empty( $themeisle_sdk_max_path ) && is_file( $themeisle_sdk_max_path . $themeisle_sdk_relative_licenser_path ) ) {
|
||||
$themeisle_sdk_abs_licenser_path = $themeisle_sdk_max_path . $themeisle_sdk_relative_licenser_path;
|
||||
add_filter( 'themeisle_sdk_required_files', 'themeisle_sdk_load_licenser_if_present' );
|
||||
}
|
||||
|
||||
if ( ( is_null( $themeisle_sdk_max_path ) || version_compare( $themeisle_sdk_version, $themeisle_sdk_max_version ) == 0 ) &&
|
||||
apply_filters( 'themeisle_sdk_should_overwrite_path', false, $themeisle_sdk_path, $themeisle_sdk_max_path ) ) {
|
||||
$themeisle_sdk_max_path = $themeisle_sdk_path;
|
||||
}
|
||||
|
||||
if ( is_null( $themeisle_sdk_max_version ) || version_compare( $themeisle_sdk_version, $themeisle_sdk_max_version ) > 0 ) {
|
||||
$themeisle_sdk_max_version = $themeisle_sdk_version;
|
||||
$themeisle_sdk_max_path = $themeisle_sdk_path;
|
||||
}
|
||||
|
||||
// load the latest sdk version from the active Themeisle products.
|
||||
if ( ! function_exists( 'themeisle_sdk_load_licenser_if_present' ) ) :
|
||||
/**
|
||||
* Always load the licenser, if present.
|
||||
*
|
||||
* @param array $to_load Previously files to load.
|
||||
*/
|
||||
function themeisle_sdk_load_licenser_if_present( $to_load ) {
|
||||
global $themeisle_sdk_abs_licenser_path;
|
||||
$to_load[] = $themeisle_sdk_abs_licenser_path;
|
||||
|
||||
return $to_load;
|
||||
}
|
||||
endif;
|
||||
|
||||
// load the latest sdk version from the active Themeisle products.
|
||||
if ( ! function_exists( 'themeisle_sdk_load_latest' ) ) :
|
||||
/**
|
||||
* Always load the latest sdk version.
|
||||
*/
|
||||
function themeisle_sdk_load_latest() {
|
||||
/**
|
||||
* Don't load the library if we are on < 5.4.
|
||||
*/
|
||||
if ( version_compare( PHP_VERSION, '5.4.32', '<' ) ) {
|
||||
return;
|
||||
}
|
||||
global $themeisle_sdk_max_path;
|
||||
require_once $themeisle_sdk_max_path . '/start.php';
|
||||
}
|
||||
endif;
|
||||
add_action( 'init', 'themeisle_sdk_load_latest' );
|
||||
|
||||
if ( ! function_exists( 'tsdk_utmify' ) ) {
|
||||
/**
|
||||
* Utmify a link.
|
||||
*
|
||||
* @param string $url URL to add utms.
|
||||
* @param string $area Area in page where this is used ( CTA, image, section name).
|
||||
* @param string $location Location, such as customizer, about page.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
function tsdk_utmify( $url, $area, $location = null ) {
|
||||
static $current_page = null;
|
||||
if ( $location === null && $current_page === null ) {
|
||||
global $pagenow;
|
||||
$screen = function_exists( 'get_current_screen' ) ? get_current_screen() : $pagenow;
|
||||
$current_page = isset( $screen->id ) ? $screen->id : ( ( $screen === null ) ? 'non-admin' : $screen );
|
||||
$current_page = sanitize_key( str_replace( '.php', '', $current_page ) );
|
||||
}
|
||||
$location = $location === null ? $current_page : $location;
|
||||
$is_upgrade_url = strpos( $url, '/upgrade' ) !== false;
|
||||
$content = sanitize_key(
|
||||
trim(
|
||||
str_replace(
|
||||
[
|
||||
'https://',
|
||||
'themeisle.com',
|
||||
'/themes/',
|
||||
'/plugins/',
|
||||
'/upgrade',
|
||||
],
|
||||
'',
|
||||
$url
|
||||
),
|
||||
'/'
|
||||
)
|
||||
);
|
||||
$filter_key = sanitize_key( $content );
|
||||
$url_args = [
|
||||
'utm_source' => 'wpadmin',
|
||||
'utm_medium' => $location,
|
||||
'utm_campaign' => $area,
|
||||
'utm_content' => $content,
|
||||
];
|
||||
$query_arguments = apply_filters( 'tsdk_utmify_' . $filter_key, $url_args, $url );
|
||||
$utmify_url = esc_url_raw(
|
||||
add_query_arg(
|
||||
$query_arguments,
|
||||
$url
|
||||
)
|
||||
);
|
||||
|
||||
/**
|
||||
* Check if there is an affiliate URL for this upgrade link, if so use it.
|
||||
*/
|
||||
if ( $is_upgrade_url ) {
|
||||
$option_content_key = str_replace( '-', '_', $filter_key );
|
||||
$theme_upgrade_option_name = 'themeisle_af_' . $option_content_key . '_themes_upgrade';
|
||||
$plugin_upgrade_option_name = 'themeisle_af_' . $option_content_key . '_plugins_upgrade';
|
||||
|
||||
$theme_option_url = get_option( $theme_upgrade_option_name, false );
|
||||
if ( ! empty( $theme_option_url ) ) {
|
||||
$utmify_url = esc_url_raw( $theme_option_url );
|
||||
}
|
||||
$plugin_option_url = get_option( $plugin_upgrade_option_name, false );
|
||||
if ( ! empty( $plugin_option_url ) ) {
|
||||
$utmify_url = esc_url_raw( $plugin_option_url );
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
return apply_filters( 'tsdk_utmify_url_' . $filter_key, $utmify_url, $url );
|
||||
}
|
||||
|
||||
add_filter( 'tsdk_utmify', 'tsdk_utmify', 10, 3 );
|
||||
}
|
||||
|
||||
|
||||
if ( ! function_exists( 'tsdk_lstatus' ) ) {
|
||||
/**
|
||||
* Check license status.
|
||||
*
|
||||
* @param string $file Product basefile.
|
||||
*
|
||||
* @return string Status.
|
||||
*/
|
||||
function tsdk_lstatus( $file ) {
|
||||
return \ThemeisleSDK\Modules\Licenser::status( $file );
|
||||
}
|
||||
}
|
||||
if ( ! function_exists( 'tsdk_lis_valid' ) ) {
|
||||
/**
|
||||
* Check if license is valid.
|
||||
*
|
||||
* @param string $file Product basefile.
|
||||
*
|
||||
* @return bool Validness.
|
||||
*/
|
||||
function tsdk_lis_valid( $file ) {
|
||||
return \ThemeisleSDK\Modules\Licenser::is_valid( $file );
|
||||
}
|
||||
}
|
||||
if ( ! function_exists( 'tsdk_lplan' ) ) {
|
||||
/**
|
||||
* Get license plan.
|
||||
*
|
||||
* @param string $file Product basefile.
|
||||
*
|
||||
* @return string Plan.
|
||||
*/
|
||||
function tsdk_lplan( $file ) {
|
||||
return \ThemeisleSDK\Modules\Licenser::plan( $file );
|
||||
}
|
||||
}
|
||||
|
||||
if ( ! function_exists( 'tsdk_lkey' ) ) {
|
||||
/**
|
||||
* Get license key.
|
||||
*
|
||||
* @param string $file Product basefile.
|
||||
*
|
||||
* @return string Key.
|
||||
*/
|
||||
function tsdk_lkey( $file ) {
|
||||
return \ThemeisleSDK\Modules\Licenser::key( $file );
|
||||
}
|
||||
}
|
||||
|
||||
if ( ! function_exists( 'tsdk_translate_link' ) ) {
|
||||
|
||||
/**
|
||||
* Function to translate a link based on the current language.
|
||||
*
|
||||
* @param string $url URL to translate.
|
||||
* @param string{'path'|'query'|'domain'} $type Type of localization. Supports path, query and domain.
|
||||
* @param array $available_languages Available language to choose from.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
function tsdk_translate_link(
|
||||
$url, $type = 'path', $available_languages = [
|
||||
'de_DE' => 'de',
|
||||
'de_DE_formal' => 'de',
|
||||
]
|
||||
) {
|
||||
$language = get_user_locale();
|
||||
if ( ! isset( $available_languages[ $language ] ) ) {
|
||||
return $url;
|
||||
}
|
||||
$code = $available_languages[ $language ];
|
||||
// We asume that false is based on query and add the code via query arg.
|
||||
if ( $type === 'query' ) {
|
||||
return add_query_arg( 'lang', $code, $url );
|
||||
}
|
||||
|
||||
$parsed_url = wp_parse_url( $url );
|
||||
// we replace the domain here based on the localized one.
|
||||
if ( $type === 'domain' ) {
|
||||
return $parsed_url['scheme'] . '://' . $code . ( isset( $parsed_url['path'] ) ? $parsed_url['path'] : '' ) . ( isset( $parsed_url['query'] ) ? '?' . $parsed_url['query'] : '' ) . ( isset( $parsed_url['fragment'] ) ? '#' . $parsed_url['fragment'] : '' );
|
||||
}
|
||||
// default is the path based approach.
|
||||
$new_path = isset( $parsed_url['path'] ) ? "/$code" . $parsed_url['path'] : "/$code";
|
||||
|
||||
return $parsed_url['scheme'] . '://' . $parsed_url['host'] . $new_path . ( isset( $parsed_url['query'] ) ? '?' . $parsed_url['query'] : '' ) . ( isset( $parsed_url['fragment'] ) ? '#' . $parsed_url['fragment'] : '' );
|
||||
|
||||
}
|
||||
}
|
||||
if ( ! function_exists( 'tsdk_support_link' ) ) {
|
||||
|
||||
/**
|
||||
* Get Themeisle Support URL.
|
||||
*
|
||||
* @param string $file Product basefile.
|
||||
*
|
||||
* @return false|string Return support URL or false if no license is active.
|
||||
*/
|
||||
function tsdk_support_link( $file ) {
|
||||
|
||||
if ( ! did_action( 'init' ) ) {
|
||||
_doing_it_wrong( __FUNCTION__, 'tsdk_support_link() should not be called before the init action.', '3.2.39' );
|
||||
}
|
||||
$params = [];
|
||||
if ( ! tsdk_lis_valid( $file ) ) {
|
||||
return false;
|
||||
}
|
||||
$product = \ThemeisleSDK\Product::get( $file );
|
||||
if ( ! $product->requires_license() ) {
|
||||
return false;
|
||||
}
|
||||
static $site_params = null;
|
||||
if ( $site_params === null ) {
|
||||
if ( is_user_logged_in() && function_exists( 'wp_get_current_user' ) ) {
|
||||
$current_user = wp_get_current_user();
|
||||
$site_params['semail'] = urlencode( $current_user->user_email );
|
||||
}
|
||||
$site_params['swb'] = urlencode( home_url() );
|
||||
global $wp_version;
|
||||
$site_params['snv'] = urlencode( sprintf( 'WP-%s-PHP-%s', $wp_version, ( PHP_MAJOR_VERSION . '.' . PHP_MINOR_VERSION ) ) );
|
||||
}
|
||||
$params['slkey'] = tsdk_lkey( $file );
|
||||
$params['sprd'] = urlencode( $product->get_name() );
|
||||
$params['svrs'] = urlencode( $product->get_version() );
|
||||
|
||||
|
||||
return add_query_arg(
|
||||
array_merge( $site_params, $params ),
|
||||
'https://store.themeisle.com/direct-support/'
|
||||
);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,233 @@
|
||||
<?php
|
||||
/**
|
||||
* The abstract class for module definition.
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Loader
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 3.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Common;
|
||||
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Class Abstract_Module.
|
||||
*
|
||||
* @package ThemeisleSDK\Common
|
||||
*/
|
||||
abstract class Abstract_Module {
|
||||
/**
|
||||
* Plugin paths.
|
||||
*
|
||||
* @var string[] $plugin_paths Plugin paths.
|
||||
*/
|
||||
private $plugin_paths = [
|
||||
'otter-blocks' => 'otter-blocks/otter-blocks.php',
|
||||
'optimole-wp' => 'optimole-wp/optimole-wp.php',
|
||||
'tweet-old-post' => 'tweet-old-post/tweet-old-post.php',
|
||||
'feedzy-rss-feeds' => 'feedzy-rss-feeds/feedzy-rss-feed.php',
|
||||
'woocommerce-product-addon' => 'woocommerce-product-addon/woocommerce-product-addon.php',
|
||||
'visualizer' => 'visualizer/index.php',
|
||||
'wp-landing-kit' => 'wp-landing-kit/wp-landing-kit.php',
|
||||
'multiple-pages-generator-by-porthas' => 'multiple-pages-generator-by-porthas/porthas-multi-pages-generator.php',
|
||||
'sparks-for-woocommerce' => 'sparks-for-woocommerce/sparks-for-woocommerce.php',
|
||||
'templates-patterns-collection' => 'templates-patterns-collection/templates-patterns-collection.php',
|
||||
'wpcf7-redirect' => 'wpcf7-redirect/wpcf7-redirect.php',
|
||||
'wp-full-stripe-free' => 'wp-full-stripe-free/wp-full-stripe.php',
|
||||
'learning-management-system' => 'learning-management-system/lms.php',
|
||||
];
|
||||
|
||||
/**
|
||||
* Product which use the module.
|
||||
*
|
||||
* @var Product $product Product object.
|
||||
*/
|
||||
protected $product = null;
|
||||
|
||||
/**
|
||||
* Can load the module for the selected product.
|
||||
*
|
||||
* @param Product $product Product data.
|
||||
*
|
||||
* @return bool Should load module?
|
||||
*/
|
||||
abstract public function can_load( $product );
|
||||
|
||||
/**
|
||||
* Bootstrap the module.
|
||||
*
|
||||
* @param Product $product Product object.
|
||||
*/
|
||||
abstract public function load( $product );
|
||||
|
||||
/**
|
||||
* Check if the product is from partner.
|
||||
*
|
||||
* @param Product $product Product data.
|
||||
*
|
||||
* @return bool Is product from partner.
|
||||
*/
|
||||
public function is_from_partner( $product ) {
|
||||
foreach ( Module_Factory::$domains as $partner_domain ) {
|
||||
if ( strpos( $product->get_store_url(), $partner_domain ) !== false ) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return array_key_exists( $product->get_slug(), Module_Factory::$slugs );
|
||||
}
|
||||
|
||||
/**
|
||||
* Wrapper for wp_remote_get on VIP environments.
|
||||
*
|
||||
* @param string $url Url to check.
|
||||
* @param array $args Option params.
|
||||
*
|
||||
* @return array|\WP_Error
|
||||
*/
|
||||
public function safe_get( $url, $args = array() ) {
|
||||
return function_exists( 'vip_safe_wp_remote_get' )
|
||||
? vip_safe_wp_remote_get( $url )
|
||||
: wp_remote_get( //phpcs:ignore WordPressVIPMinimum.Functions.RestrictedFunctions.wp_remote_get_wp_remote_get, Already used.
|
||||
$url,
|
||||
$args
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the SDK base url.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function get_sdk_uri() {
|
||||
global $themeisle_sdk_max_path;
|
||||
|
||||
/**
|
||||
* $themeisle_sdk_max_path can point to the theme when the theme version is higher.
|
||||
* hence we also need to check that the path does not point to the theme else this will break the URL.
|
||||
* References: https://github.com/Codeinwp/neve-pro-addon/issues/2403
|
||||
*/
|
||||
if ( ( $this->product->is_plugin() || $this->product->is_theme() ) && false === strpos( $themeisle_sdk_max_path, get_template_directory() ) ) {
|
||||
return plugins_url( '/', $themeisle_sdk_max_path . '/themeisle-sdk/' );
|
||||
};
|
||||
|
||||
return get_template_directory_uri() . '/vendor/codeinwp/themeisle-sdk/';
|
||||
}
|
||||
|
||||
/**
|
||||
* Call plugin api
|
||||
*
|
||||
* @param string $slug plugin slug.
|
||||
*
|
||||
* @return array|mixed|object
|
||||
*/
|
||||
public function call_plugin_api( $slug ) {
|
||||
include_once ABSPATH . 'wp-admin/includes/plugin-install.php';
|
||||
|
||||
$call_api = get_transient( 'ti_plugin_info_' . $slug );
|
||||
|
||||
if ( false === $call_api ) {
|
||||
$call_api = plugins_api(
|
||||
'plugin_information',
|
||||
array(
|
||||
'slug' => $slug,
|
||||
'fields' => array(
|
||||
'downloaded' => false,
|
||||
'rating' => false,
|
||||
'description' => false,
|
||||
'short_description' => true,
|
||||
'donate_link' => false,
|
||||
'tags' => false,
|
||||
'sections' => true,
|
||||
'homepage' => true,
|
||||
'added' => false,
|
||||
'last_updated' => false,
|
||||
'compatibility' => false,
|
||||
'tested' => false,
|
||||
'requires' => false,
|
||||
'downloadlink' => false,
|
||||
'icons' => true,
|
||||
'banners' => true,
|
||||
),
|
||||
)
|
||||
);
|
||||
set_transient( 'ti_plugin_info_' . $slug, $call_api, 30 * MINUTE_IN_SECONDS );
|
||||
}
|
||||
|
||||
return $call_api;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the plugin status.
|
||||
*
|
||||
* @param string $plugin Plugin slug.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function is_plugin_installed( $plugin ) {
|
||||
if ( ! isset( $this->plugin_paths[ $plugin ] ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( file_exists( WP_CONTENT_DIR . '/plugins/' . $this->plugin_paths[ $plugin ] ) ) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get plugin activation link.
|
||||
*
|
||||
* @param string $slug The plugin slug.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function get_plugin_activation_link( $slug ) {
|
||||
$reference_key = $slug === 'otter-blocks' ? 'reference_key' : 'optimole_reference_key';
|
||||
$plugin = isset( $this->plugin_paths[ $slug ] ) ? $this->plugin_paths[ $slug ] : $slug . '/' . $slug . '.php';
|
||||
|
||||
return add_query_arg(
|
||||
array(
|
||||
'plugin_status' => 'all',
|
||||
'paged' => '1',
|
||||
'action' => 'activate',
|
||||
$reference_key => $this->product->get_key(),
|
||||
'plugin' => rawurlencode( $plugin ),
|
||||
'_wpnonce' => wp_create_nonce( 'activate-plugin_' . $plugin ),
|
||||
),
|
||||
admin_url( 'plugins.php' )
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if a plugin is active.
|
||||
*
|
||||
* @param string $plugin plugin slug.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function is_plugin_active( $plugin ) {
|
||||
include_once ABSPATH . 'wp-admin/includes/plugin.php';
|
||||
|
||||
$plugin = isset( $this->plugin_paths[ $plugin ] ) ? $this->plugin_paths[ $plugin ] : $plugin . '/' . $plugin . '.php';
|
||||
|
||||
return is_plugin_active( $plugin );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the current date.
|
||||
*
|
||||
* @return \DateTime The date time.
|
||||
*/
|
||||
public function get_current_date() {
|
||||
return apply_filters( 'themeisle_sdk_current_date', new \DateTime( 'now' ) );
|
||||
}
|
||||
}
|
||||
108
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Common/Module_factory.php
vendored
Normal file
@@ -0,0 +1,108 @@
|
||||
<?php
|
||||
/**
|
||||
* The module factory class.
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Loader
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 3.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Common;
|
||||
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Class Job_Factory
|
||||
*
|
||||
* @package ThemeisleSDK\Common
|
||||
*/
|
||||
class Module_Factory {
|
||||
/**
|
||||
* Partners slugs.
|
||||
*
|
||||
* @var array $SLUGS Partners product slugs.
|
||||
*/
|
||||
public static $slugs = [
|
||||
'zermatt' => true,
|
||||
'neto' => true,
|
||||
'olsen' => true,
|
||||
'benson' => true,
|
||||
'romero' => true,
|
||||
'carmack' => true,
|
||||
'puzzle' => true,
|
||||
'broadsheet' => true,
|
||||
'girlywp' => true,
|
||||
'veggie' => true,
|
||||
'zeko' => true,
|
||||
'maishawp' => true,
|
||||
'didi' => true,
|
||||
'liber' => true,
|
||||
'medicpress-pt' => true,
|
||||
'adrenaline-pt' => true,
|
||||
'consultpress-pt' => true,
|
||||
'legalpress-pt' => true,
|
||||
'gympress-pt' => true,
|
||||
'readable-pt' => true,
|
||||
'bolts-pt' => true,
|
||||
];
|
||||
/**
|
||||
* Partners domains.
|
||||
*
|
||||
* @var array $DOMAINS Partners domains.
|
||||
*/
|
||||
public static $domains = [
|
||||
'proteusthemes.com',
|
||||
'anarieldesign.com',
|
||||
'prothemedesign.com',
|
||||
'cssigniter.com',
|
||||
];
|
||||
/**
|
||||
* Map which contains all the modules loaded for each product.
|
||||
*
|
||||
* @var array Mapping array.
|
||||
*/
|
||||
private static $modules_attached = [];
|
||||
|
||||
/**
|
||||
* Load availabe modules for the selected product.
|
||||
*
|
||||
* @param Product $product Loaded product.
|
||||
* @param array $modules List of modules.
|
||||
*/
|
||||
public static function attach( $product, $modules ) {
|
||||
|
||||
if ( ! isset( self::$modules_attached[ $product->get_slug() ] ) ) {
|
||||
self::$modules_attached[ $product->get_slug() ] = [];
|
||||
}
|
||||
|
||||
foreach ( $modules as $module ) {
|
||||
$class = 'ThemeisleSDK\\Modules\\' . ucwords( $module, '_' );
|
||||
/**
|
||||
* Module object.
|
||||
*
|
||||
* @var Abstract_Module $module_object Module instance.
|
||||
*/
|
||||
$module_object = new $class( $product );
|
||||
|
||||
if ( ! $module_object->can_load( $product ) ) {
|
||||
continue;
|
||||
}
|
||||
self::$modules_attached[ $product->get_slug() ][ $module ] = $module_object->load( $product );
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Products/Modules loaded map.
|
||||
*
|
||||
* @return array Modules map.
|
||||
*/
|
||||
public static function get_modules_map() {
|
||||
return self::$modules_attached;
|
||||
}
|
||||
}
|
||||
522
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Loader.php
vendored
Normal file
@@ -0,0 +1,522 @@
|
||||
<?php
|
||||
/**
|
||||
* The main loader class for ThemeIsle SDK
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Loader
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 1.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK;
|
||||
|
||||
use ThemeisleSDK\Common\Module_Factory;
|
||||
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Singleton loader for ThemeIsle SDK.
|
||||
*/
|
||||
final class Loader {
|
||||
/**
|
||||
* Singleton instance.
|
||||
*
|
||||
* @var Loader instance The singleton instance
|
||||
*/
|
||||
private static $instance;
|
||||
/**
|
||||
* Current loader version.
|
||||
*
|
||||
* @var string $version The class version.
|
||||
*/
|
||||
private static $version = '2.0.0';
|
||||
/**
|
||||
* Holds registered products.
|
||||
*
|
||||
* @var array<Product> The products which use the SDK.
|
||||
*/
|
||||
private static $products = [];
|
||||
/**
|
||||
* Holds available modules to load.
|
||||
*
|
||||
* @var array The modules which SDK will be using.
|
||||
*/
|
||||
private static $available_modules = [
|
||||
'script_loader',
|
||||
'dashboard_widget',
|
||||
'rollback',
|
||||
'uninstall_feedback',
|
||||
'licenser',
|
||||
'logger',
|
||||
'translate',
|
||||
'translations',
|
||||
'review',
|
||||
'recommendation',
|
||||
'notification',
|
||||
'promotions',
|
||||
'welcome',
|
||||
'compatibilities',
|
||||
'about_us',
|
||||
'announcements',
|
||||
'featured_plugins',
|
||||
'float_widget',
|
||||
];
|
||||
/**
|
||||
* Holds the labels for the modules.
|
||||
*
|
||||
* @var array The labels for the modules.
|
||||
*/
|
||||
public static $labels = [
|
||||
'announcements' => [
|
||||
'notice_link_label' => 'See the Offer',
|
||||
'max_savings' => 'Our biggest sale of the year: <strong>%s OFF everything!</strong> Don\'t miss this limited-time offer.',
|
||||
'black_friday' => 'Black Friday Sale',
|
||||
'time_left' => '%s left',
|
||||
],
|
||||
'compatibilities' => [
|
||||
'notice' => '%s requires a newer version of %s. Please %supdate%s %s %s to the latest version.',
|
||||
'notice2' => '%s update requires a newer version of %s. Please %supdate%s %s %s.',
|
||||
'notice_theme' => '%1$sWarning:%2$s This theme has not been tested with your current version of %1$s%3$s%2$s. Please update %3$s plugin.',
|
||||
'notice_plugin' => '%1$sWarning:%2$s This plugin has not been tested with your current version of %1$s%3$s%2$s. Please update %3$s %4$s.',
|
||||
'theme' => 'theme',
|
||||
'plugin' => 'plugin',
|
||||
],
|
||||
'dashboard_widget' => [
|
||||
'title' => 'WordPress Guides/Tutorials',
|
||||
'popular' => 'Popular %s',
|
||||
'install' => 'Install',
|
||||
'powered' => 'Powered by %s',
|
||||
],
|
||||
'licenser' => [
|
||||
'activate' => 'Activate',
|
||||
'invalid_msg' => 'Invalid license.',
|
||||
'error_notice' => 'ERROR: Failed to connect to the license service. Please try again later. Reason: %s',
|
||||
'error_notice2' => 'ERROR: Failed to validate license. Please try again in one minute.',
|
||||
'error_invalid' => 'ERROR: Invalid license provided.',
|
||||
'update_license' => 'Updating this theme will lose any customizations you have made. Cancel to stop, OK to update.',
|
||||
'invalid_msg' => 'Invalid license.',
|
||||
'already_active' => 'License is already active.',
|
||||
'notice_update' => '%1$s is available. %2$sCheck out what\'s%3$s new or %4$supdate now%3$s.',
|
||||
'not_active' => 'License not active.',
|
||||
'deactivate' => 'Deactivate',
|
||||
'renew_cta' => 'Renew license to update',
|
||||
'autoactivate_notice' => '%s has been successfully activated using %s license !',
|
||||
'valid' => 'Valid',
|
||||
'invalid' => 'Invalid',
|
||||
'notice' => 'Enter your license from %s purchase history in order to get %s updates',
|
||||
'expired' => 'Your %s\'s License Key has expired. In order to continue receiving support and software updates you must %srenew%s your license key.',
|
||||
|
||||
'inactive' => 'In order to benefit from updates and support for %s, please add your license code from your %spurchase history%s and validate it %shere%s.',
|
||||
'no_activations' => 'No more activations left for %s. You need to upgrade your plan in order to use %s on more websites. If you need assistance, please get in touch with %s staff.',
|
||||
],
|
||||
'promotions' => [
|
||||
'recommended' => 'Recommended by %s',
|
||||
'installActivate' => 'Install & Activate',
|
||||
'preview' => 'Preview',
|
||||
'installing' => 'Installing',
|
||||
'activating' => 'Activating',
|
||||
'connecting' => 'Connecting to API',
|
||||
'learnmore' => 'Learn More',
|
||||
'activate' => 'Activate',
|
||||
'all_set' => 'Awesome! You are all set!',
|
||||
'woo' => [
|
||||
'title' => 'More extensions from Themeisle',
|
||||
'title2' => 'Recommended extensions',
|
||||
'cta_install' => 'Install',
|
||||
'learn_more' => 'Learn More',
|
||||
'dismiss' => 'Dismiss this suggestion',
|
||||
'ppom_title' => 'Product Add-Ons',
|
||||
'ppom_desc' => 'Add extra custom fields & add-ons on your product pages, like sizes, colors & more.',
|
||||
'spark_title1' => 'Wishlist',
|
||||
'spark_title2' => 'Multi-Announcement Bars',
|
||||
'spark_title3' => 'Advanced Product Review',
|
||||
'spark_desc1' => 'Loyalize your customers by allowing them to save their favorite products.',
|
||||
'spark_desc2' => 'Add a top notification bar on your website to highlight the latest products, offers, or upcoming events.',
|
||||
'spark_desc3' => 'Enable an advanced review section, enlarging the basic review options with lots of capabilities.',
|
||||
],
|
||||
|
||||
'feedzy' => [
|
||||
'import_desc' => 'Schedule automatic content imports from any RSS feed directly to your site. %sBuilt by %s%s',
|
||||
'install_now' => 'Install Now',
|
||||
'by' => 'by %s',
|
||||
'editor_recommends' => '%s recommends %sFeedzy%s to display entries from any RSS feed with more advanced styling and filtering options.',
|
||||
],
|
||||
'optimole' => [
|
||||
'installOptimole' => 'Install Optimole',
|
||||
'gotodash' => 'Go to Optimole dashboard',
|
||||
'dismisscta' => 'Dismiss this notice.',
|
||||
'message1' => 'Increase this page speed and SEO ranking by optimizing images with Optimole.',
|
||||
'message3' => 'Save your server space by storing images to Optimole and deliver them optimized from 400 locations around the globe. Unlimited images, Unlimited traffic.',
|
||||
'message4' => 'This image looks to be too large and would affect your site speed, we recommend you to install Optimole to optimize your images.',
|
||||
'message2' => 'Leverage Optimole\'s full integration with Elementor to automatically lazyload, resize, compress to AVIF/WebP and deliver from 400 locations around the globe!',
|
||||
],
|
||||
'redirectionCF7' => [
|
||||
'gotodash' => 'Go to Contact Forms',
|
||||
'dismisscta' => 'Dismiss this notice.',
|
||||
'gst' => 'Get Started Free',
|
||||
'message' => 'Add URL redirects, spam protection, execute JavaScript after submissions, and more with the Redirection for CF7 free plugin.',
|
||||
],
|
||||
'hyve' => [
|
||||
'gotodash' => 'Go to Hyve Dashboard',
|
||||
'install' => 'Install Hyve',
|
||||
'dismisscta' => 'Dismiss this notice.',
|
||||
'message' => 'Hyve is an AI-powered chatbot that turns your WordPress content into interactive conversations, helping you efficiently handle user inquiries.',
|
||||
],
|
||||
'wp_full_pay' => [
|
||||
'gotodash' => 'Go to WP Full Pay Settings',
|
||||
'install' => 'Install WP Full Pay',
|
||||
'dismisscta' => 'Dismiss this notice.',
|
||||
'message' => 'Enhance your donation page with WP Full Pay—create custom Stripe forms for one-time and recurring donations, manage transactions easily, and boost support with a seamless setup.',
|
||||
],
|
||||
'masteriyo' => [
|
||||
'gotodash' => 'Go to Masteriyo Dashboard',
|
||||
'install' => 'Install Masteriyo',
|
||||
'dismisscta' => 'Dismiss this notice.',
|
||||
'message' => 'Transform your site into a learning hub with Masteriyo LMS. Build engaging courses with intuitive tools, track student progress effortlessly, and grow your education business with powerful marketing features and seamless payment integration.',
|
||||
],
|
||||
],
|
||||
'welcome' => [
|
||||
'ctan' => 'No, thanks.',
|
||||
'ctay' => 'Upgrade Now!',
|
||||
'message' => '<p>You\'ve been using <b>{product}</b> for 7 days now and we appreciate your loyalty! We also want to make sure you\'re getting the most out of our product. That\'s why we\'re offering you a special deal - upgrade to <b>{pro_product}</b> in the next 5 days and receive a discount of <b>up to 30%</b>. <a href="{cta_link}" target="_blank">Upgrade now</a> and unlock all the amazing features of <b>{pro_product}</b>!</p>',
|
||||
],
|
||||
'uninstall' => [
|
||||
'heading_plugin' => 'What\'s wrong?',
|
||||
'heading_theme' => 'What does not work for you in {theme}?',
|
||||
'submit' => 'Submit',
|
||||
'cta_info' => 'What info do we collect?',
|
||||
'button_submit' => 'Submit & Deactivate',
|
||||
'button_cancel' => 'Skip & Deactivate',
|
||||
'disclosure' => [
|
||||
'title' => 'Below is a detailed view of all data that Themeisle will receive if you fill in this survey. No email address or IP addresses are transmitted after you submit the survey.',
|
||||
'version' => '%s %s version %s %s %s %s',
|
||||
'website' => '%sCurrent website:%s %s %s %s',
|
||||
'usage' => '%sUsage time:%s %s %s%s',
|
||||
'reason' => '%s Uninstall reason %s %s Selected reason from the above survey %s ',
|
||||
],
|
||||
|
||||
'options' => [
|
||||
'id3' => [
|
||||
'title' => 'I found a better plugin',
|
||||
'placeholder' => 'What\'s the plugin\'s name?',
|
||||
],
|
||||
'id4' => [
|
||||
|
||||
'title' => 'I could not get the plugin to work',
|
||||
'placeholder' => 'What problem are you experiencing?',
|
||||
],
|
||||
'id5' => [
|
||||
|
||||
'title' => 'I no longer need the plugin',
|
||||
'placeholder' => 'If you could improve one thing about our product, what would it be?',
|
||||
],
|
||||
'id6' => [
|
||||
'title' => 'It\'s a temporary deactivation. I\'m just debugging an issue.',
|
||||
'placeholder' => 'What problem are you experiencing?',
|
||||
],
|
||||
'id7' => [
|
||||
'title' => 'I don\'t know how to make it look like demo',
|
||||
],
|
||||
'id8' => [
|
||||
|
||||
'placeholder' => 'What option is missing?',
|
||||
'title' => 'It lacks options',
|
||||
],
|
||||
'id9' => [
|
||||
'title' => 'Is not working with a plugin that I need',
|
||||
'placeholder' => 'What is the name of the plugin',
|
||||
],
|
||||
'id10' => [
|
||||
'title' => 'I want to try a new design, I don\'t like {theme} style',
|
||||
],
|
||||
'id999' => [
|
||||
'title' => 'Other',
|
||||
'placeholder' => 'What can we do better?',
|
||||
],
|
||||
],
|
||||
],
|
||||
'review' => [
|
||||
'notice' => '<p>Hey, it\'s great to see you have <b>{product}</b> active for a few days now. How is everything going? If you can spare a few moments to rate it on WordPress.org it would help us a lot (and boost my motivation). Cheers! <br/> <br/>~ {developer}, developer of {product}</p>',
|
||||
'ctay' => 'Ok, I will gladly help.',
|
||||
'ctan' => 'No, thanks.',
|
||||
|
||||
],
|
||||
'rollback' => [
|
||||
'cta' => 'Rollback to v%s',
|
||||
],
|
||||
'logger' => [
|
||||
'notice' => 'Do you enjoy <b>{product}</b>? Become a contributor by opting in to our anonymous data tracking. We guarantee no sensitive data is collected.',
|
||||
'cta_y' => 'Sure, I would love to help.',
|
||||
'cta_n' => 'No, thanks.',
|
||||
],
|
||||
'about_us' => [
|
||||
'title' => 'About Us',
|
||||
'heroHeader' => 'Our Story',
|
||||
'heroTextFirst' => 'Themeisle was founded in 2012 by a group of passionate developers who wanted to create beautiful and functional WordPress themes and plugins. Since then, we have grown into a team of over 20 dedicated professionals who are committed to delivering the best possible products to our customers.',
|
||||
'heroTextSecond' => 'At Themeisle, we offer a wide range of WordPress themes and plugins that are designed to meet the needs of both beginners and advanced users. Our products are feature-rich, easy to use, and are designed to help you create beautiful and functional websites.',
|
||||
'teamImageCaption' => 'Our team in WCEU2022 in Portugal',
|
||||
'newsHeading' => 'Stay connected for news & updates!',
|
||||
'emailPlaceholder' => 'Your email address',
|
||||
'signMeUp' => 'Sign me up',
|
||||
'installNow' => 'Install Now',
|
||||
'activate' => 'Activate',
|
||||
'learnMore' => 'Learn More',
|
||||
'installed' => 'Installed',
|
||||
'notInstalled' => 'Not Installed',
|
||||
'active' => 'Active',
|
||||
'others' => [
|
||||
'optimole_desc' => 'Optimole is an image optimization service that automatically optimizes your images and serves them to your visitors via a global CDN, making your website lighter, faster and helping you reduce your bandwidth usage.',
|
||||
'neve_desc' => 'A fast, lightweight, customizable WordPress theme offering responsive design, speed, and flexibility for various website types.',
|
||||
'landingkit_desc' => 'Turn WordPress into a landing page powerhouse with Landing Kit, map domains to pages or any other published resource.',
|
||||
'sparks_desc' => 'Extend your store functionality with 8 ultra-performant features like product comparisons, variation swatches, wishlist, and more.',
|
||||
'tpc_desc' => 'Design, save, and revisit your templates anytime with your personal vault on Templates Cloud.',
|
||||
],
|
||||
'otter-page' => [
|
||||
'heading' => 'Build innovative layouts with Otter Blocks and Gutenberg',
|
||||
'text' => 'Otter is a lightweight, dynamic collection of page building blocks and templates for the WordPress block editor.',
|
||||
'buttons' => [
|
||||
'install_otter_free' => "Install Otter - It's free!",
|
||||
'install_now' => 'Install Now',
|
||||
'learn_more' => 'Learn More',
|
||||
],
|
||||
'features' => [
|
||||
'advancedTitle' => 'Advanced Features',
|
||||
'advancedDesc' => 'Add features such as Custom CSS, Animations & Visibility Conditions to all blocks.',
|
||||
'fastTitle' => 'Lightweight and Fast',
|
||||
'fastDesc' => 'Otter enhances WordPress site building experience without impacting site speed.',
|
||||
'mobileTitle' => 'Mobile-Friendly',
|
||||
'mobileDesc' => 'Each block can be tweaked to provide a consistent experience across all devices.',
|
||||
],
|
||||
'details' => [
|
||||
's1Title' => 'A Better Page Building Experience',
|
||||
's1Text' => 'Otter can be used to build everything from a personal blog to an e-commerce site without losing the personal touch. Otter’s ease of use transforms basic blocks into expressive layouts in seconds.',
|
||||
's2Title' => 'A New Collection of Patterns',
|
||||
's2Text' => 'A New Patterns Library, containing a range of different elements in a variety of styles to help you build great pages. All of your website’s most important areas are covered: headers, testimonials, pricing tables, sections and more.',
|
||||
's3Title' => 'Advanced Blocks',
|
||||
's3Text' => 'Enhance your website’s design with powerful blocks, like the Add to Cart, Business Hours, Review Comparison, and dozens of WooCommerce blocks.',
|
||||
],
|
||||
'testimonials' => [
|
||||
'heading' => 'Trusted by more than 300K website owners',
|
||||
'users' => [
|
||||
'user_1' => 'Loved the collection of blocks. If you want to create nice Gutenberg Pages, this plugin will be very handy and useful.',
|
||||
'user_2' => 'I am very satisfied with Otter – a fantastic collection of blocks. And the plugin is perfectly integrated with Gutenberg and complete enough for my needs.',
|
||||
'user_3' => 'Otter Blocks work really well and I like the customization options. Easy to use and format to fit in with my site theme – and I’ve not encountered any compatibility or speed issues.',
|
||||
],
|
||||
],
|
||||
],
|
||||
],
|
||||
'float_widget' => [
|
||||
'button' => 'Toggle Help Widget for %s',
|
||||
'panel' => [
|
||||
'greeting' => 'Thank you for using %s',
|
||||
'title' => 'How can we help you?',
|
||||
'close' => 'Close Toggle Help Widget',
|
||||
],
|
||||
'links' => [
|
||||
'documentation' => 'Documentation',
|
||||
'support' => 'Get Support',
|
||||
'wizard' => 'Run Setup Wizard',
|
||||
'upgrade' => 'Upgrade to Pro',
|
||||
'feature_request' => 'Suggest a Feature',
|
||||
'rate' => 'Rate Us',
|
||||
],
|
||||
],
|
||||
];
|
||||
|
||||
/**
|
||||
* Initialize the sdk logic.
|
||||
*/
|
||||
public static function init() {
|
||||
self::localize_labels();
|
||||
if ( ! isset( self::$instance ) && ! ( self::$instance instanceof Loader ) ) {
|
||||
self::$instance = new Loader();
|
||||
$modules = array_merge( self::$available_modules, apply_filters( 'themeisle_sdk_modules', [] ) );
|
||||
foreach ( $modules as $key => $module ) {
|
||||
if ( ! class_exists( 'ThemeisleSDK\\Modules\\' . ucwords( $module, '_' ) ) ) {
|
||||
unset( $modules[ $key ] );
|
||||
}
|
||||
}
|
||||
self::$available_modules = $modules;
|
||||
|
||||
add_action( 'themeisle_sdk_first_activation', array( __CLASS__, 'activate' ) );
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Localize the labels.
|
||||
*/
|
||||
public static function localize_labels() {
|
||||
$originals = self::$labels;
|
||||
$all_translations = [];
|
||||
|
||||
global $wp_filter;
|
||||
if ( isset( $wp_filter['themeisle_sdk_labels'] ) ) {
|
||||
foreach ( $wp_filter['themeisle_sdk_labels']->callbacks as $priority => $hooks ) {
|
||||
foreach ( $hooks as $hook ) {
|
||||
// Each callback gets fresh originals, not previous callback's output
|
||||
$result = call_user_func( $hook['function'], $originals );
|
||||
$all_translations[] = $result;
|
||||
}
|
||||
}
|
||||
|
||||
// Remove the filter so it doesn't run again via apply_filters
|
||||
remove_all_filters( 'themeisle_sdk_labels' );
|
||||
}
|
||||
|
||||
// Merge all results, first real translation wins
|
||||
self::$labels = self::merge_all_translations( $originals, $all_translations );
|
||||
}
|
||||
/**
|
||||
* Merge all translations.
|
||||
*
|
||||
* @param array $originals The original labels.
|
||||
* @param array $all_translations The all translations.
|
||||
*
|
||||
* @return array The merged labels.
|
||||
*/
|
||||
private static function merge_all_translations( $originals, $all_translations ) {
|
||||
$result = $originals;
|
||||
|
||||
foreach ( $all_translations as $translations ) {
|
||||
$result = self::merge_if_translated( $result, $translations, $originals );
|
||||
}
|
||||
|
||||
return $result;
|
||||
}
|
||||
/**
|
||||
* Merge if translated.
|
||||
*
|
||||
* @param array $current The current labels.
|
||||
* @param array $new The new labels.
|
||||
* @param array $originals The original labels.
|
||||
* @return array The merged labels.
|
||||
*/
|
||||
private static function merge_if_translated( $current, $new, $originals ) {
|
||||
foreach ( $new as $key => $value ) {
|
||||
if ( ! isset( $originals[ $key ] ) ) {
|
||||
// New key, accept it
|
||||
if ( ! isset( $current[ $key ] ) ) {
|
||||
$current[ $key ] = $value;
|
||||
}
|
||||
continue;
|
||||
}
|
||||
|
||||
if ( is_array( $value ) && is_array( $originals[ $key ] ) ) {
|
||||
$current[ $key ] = self::merge_if_translated(
|
||||
$current[ $key ],
|
||||
$value,
|
||||
$originals[ $key ]
|
||||
);
|
||||
} else {
|
||||
// Only accept if:
|
||||
// 1. New value is actually translated (differs from original)
|
||||
// 2. Current value is NOT already translated
|
||||
$is_new_translated = ( $value !== $originals[ $key ] );
|
||||
$is_current_untranslated = ( $current[ $key ] === $originals[ $key ] );
|
||||
|
||||
if ( $is_new_translated && $is_current_untranslated ) {
|
||||
$current[ $key ] = $value;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return $current;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get cache token used in API requests.
|
||||
*
|
||||
* @return string Cache token.
|
||||
*/
|
||||
public static function get_cache_token() {
|
||||
$cache_token = get_transient( 'themeisle_sdk_cache_token' );
|
||||
if ( false === $cache_token ) {
|
||||
$cache_token = wp_generate_password( 6, false );
|
||||
set_transient( $cache_token, WEEK_IN_SECONDS );
|
||||
}
|
||||
|
||||
return $cache_token;
|
||||
}
|
||||
|
||||
/**
|
||||
* Clear cache token.
|
||||
*/
|
||||
public static function clear_cache_token() {
|
||||
delete_transient( 'themeisle_sdk_cache_token' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Register product into SDK.
|
||||
*
|
||||
* @param string $base_file The product base file.
|
||||
*
|
||||
* @return Loader The singleton object.
|
||||
*/
|
||||
public static function add_product( $base_file ) {
|
||||
|
||||
if ( ! is_file( $base_file ) ) {
|
||||
return self::$instance;
|
||||
}
|
||||
$product = new Product( $base_file );
|
||||
|
||||
Module_Factory::attach( $product, self::get_modules() );
|
||||
|
||||
self::$products[ $product->get_slug() ] = $product;
|
||||
|
||||
return self::$instance;
|
||||
}
|
||||
|
||||
/**
|
||||
* Activate the product routine.
|
||||
*
|
||||
* @param string $file The base file of the product.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public static function activate( $file ) {
|
||||
|
||||
$dirname = trailingslashit( dirname( ( $file ) ) );
|
||||
if ( ! file_exists( $dirname . '_reference.php' ) ) {
|
||||
return;
|
||||
}
|
||||
$reference_data = require_once $dirname . '_reference.php';
|
||||
if ( ! is_array( $reference_data ) ||
|
||||
! isset( $reference_data['key'] ) ||
|
||||
! isset( $reference_data['value'] ) ||
|
||||
! preg_match( '/^[a-zA-Z0-9_]+_reference_key$/', $reference_data['key'] ) ) {
|
||||
return;
|
||||
}
|
||||
add_option( $reference_data['key'], sanitize_key( $reference_data['value'] ) );
|
||||
}
|
||||
/**
|
||||
* Get all registered modules by the SDK.
|
||||
*
|
||||
* @return array Modules available.
|
||||
*/
|
||||
public static function get_modules() {
|
||||
return self::$available_modules;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get all products using the SDK.
|
||||
*
|
||||
* @return array<Product> Products available.
|
||||
*/
|
||||
public static function get_products() {
|
||||
return self::$products;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the version of the SDK.
|
||||
*
|
||||
* @return string The version.
|
||||
*/
|
||||
public static function get_version() {
|
||||
return self::$version;
|
||||
}
|
||||
}
|
||||
450
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/About_us.php
vendored
Normal file
@@ -0,0 +1,450 @@
|
||||
<?php
|
||||
/**
|
||||
* The about page model class for ThemeIsle SDK
|
||||
*
|
||||
* Here's how to hook it in your plugin:
|
||||
*
|
||||
* add_filter( <product_slug>_about_us_metadata', 'add_about_meta' );
|
||||
*
|
||||
* function add_about_meta($data) {
|
||||
* return [
|
||||
* 'location' => <top level page - e.g. themes.php>,
|
||||
* 'logo' => <logo url>,
|
||||
* 'page_menu' => [['text' => '', 'url' => '']], // optional
|
||||
* 'has_upgrade_menu' => <condition>,
|
||||
* 'upgrade_link' => <url>,
|
||||
* 'upgrade_text' => 'Get Pro Version',
|
||||
* 'review_link' => false, // Leave it empty for default WPorg link or false to hide it.
|
||||
* ]
|
||||
* }
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2023, Andrei Baicus
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 3.2.42
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Loader;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Promotions module for ThemeIsle SDK.
|
||||
*/
|
||||
class About_Us extends Abstract_Module {
|
||||
/**
|
||||
* About data.
|
||||
*
|
||||
* @var array $about_data About page data, received from the filter.
|
||||
*
|
||||
* Shape of the $about_data property array:
|
||||
* [
|
||||
* 'location' => 'top level page',
|
||||
* 'logo' => 'logo path',
|
||||
* 'page_menu' => [['text' => '', 'url' => '']], // Optional
|
||||
* 'has_upgrade_menu' => !defined('NEVE_PRO_VERSION'),
|
||||
* 'upgrade_link' => 'upgrade url',
|
||||
* 'upgrade_text' => 'Get Pro Version',
|
||||
* ]
|
||||
*/
|
||||
private $about_data = array();
|
||||
|
||||
/**
|
||||
* Should we load this module.
|
||||
*
|
||||
* @param Product $product Product object.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$this->about_data = apply_filters( $product->get_key() . '_about_us_metadata', array() );
|
||||
|
||||
return ! empty( $this->about_data );
|
||||
}
|
||||
|
||||
/**
|
||||
* Registers the hooks.
|
||||
*
|
||||
* @param Product $product Product to load.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
$this->product = $product;
|
||||
|
||||
add_action( 'admin_menu', [ $this, 'add_submenu_pages' ] );
|
||||
add_action( 'admin_enqueue_scripts', [ $this, 'enqueue_about_page_script' ] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Adds submenu pages.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function add_submenu_pages() {
|
||||
if ( ! isset( $this->about_data['location'] ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
add_submenu_page(
|
||||
$this->about_data['location'],
|
||||
Loader::$labels['about_us']['title'],
|
||||
Loader::$labels['about_us']['title'],
|
||||
'manage_options',
|
||||
$this->get_about_page_slug(),
|
||||
array( $this, 'render_about_us_page' ),
|
||||
100
|
||||
);
|
||||
|
||||
if ( ! isset( $this->about_data['has_upgrade_menu'] ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( $this->about_data['has_upgrade_menu'] !== true ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( ! isset( $this->about_data['upgrade_link'] ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( ! isset( $this->about_data['upgrade_text'] ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
add_submenu_page(
|
||||
$this->about_data['location'],
|
||||
$this->about_data['upgrade_text'],
|
||||
'<span class="tsdk-upg-menu-item">' . $this->about_data['upgrade_text'] . '</span>',
|
||||
'manage_options',
|
||||
$this->about_data['upgrade_link'],
|
||||
'',
|
||||
101
|
||||
);
|
||||
add_action(
|
||||
'admin_footer',
|
||||
function () {
|
||||
?>
|
||||
<style>
|
||||
.tsdk-upg-menu-item {
|
||||
color: #009528;
|
||||
}
|
||||
|
||||
.tsdk-upg-menu-item:hover {
|
||||
color: #008a20;
|
||||
}
|
||||
</style>
|
||||
<script type="text/javascript">
|
||||
jQuery(document).ready(function ($) {
|
||||
$('.tsdk-upg-menu-item').parent().attr('target', '_blank');
|
||||
});
|
||||
</script>
|
||||
<?php
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Render page content.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function render_about_us_page() {
|
||||
echo '<div id="ti-sdk-about"></div>';
|
||||
}
|
||||
|
||||
/**
|
||||
* Enqueue scripts & styles.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function enqueue_about_page_script() {
|
||||
$current_screen = get_current_screen();
|
||||
|
||||
if ( ! isset( $current_screen->id ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( strpos( $current_screen->id, $this->get_about_page_slug() ) === false ) {
|
||||
return;
|
||||
}
|
||||
global $themeisle_sdk_max_path;
|
||||
$handle = 'ti-sdk-about-' . $this->product->get_key();
|
||||
$asset_file = require $themeisle_sdk_max_path . '/assets/js/build/about/about.asset.php';
|
||||
$deps = array_merge( $asset_file['dependencies'], [ 'updates' ] );
|
||||
|
||||
do_action( 'themeisle_internal_page', $this->product->get_slug(), 'about_us' );
|
||||
|
||||
wp_register_script( $handle, $this->get_sdk_uri() . 'assets/js/build/about/about.js', $deps, $asset_file['version'], true );
|
||||
wp_localize_script( $handle, 'tiSDKAboutData', $this->get_about_localization_data() );
|
||||
|
||||
wp_enqueue_script( $handle );
|
||||
wp_enqueue_style( $handle, $this->get_sdk_uri() . 'assets/js/build/about/about.css', [ 'wp-components' ], $asset_file['version'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get localized data.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
private function get_about_localization_data() {
|
||||
$links = isset( $this->about_data['page_menu'] ) ? $this->about_data['page_menu'] : [];
|
||||
$product_pages = isset( $this->about_data['product_pages'] ) ? $this->about_data['product_pages'] : [];
|
||||
|
||||
return [
|
||||
'links' => $links,
|
||||
'logoUrl' => $this->about_data['logo'],
|
||||
'productPages' => $this->get_product_pages_data( $product_pages ),
|
||||
'products' => $this->get_other_products_data(),
|
||||
'homeUrl' => esc_url( home_url() ),
|
||||
'pageSlug' => $this->get_about_page_slug(),
|
||||
'currentProduct' => [
|
||||
'slug' => $this->product->get_key(),
|
||||
'name' => $this->product->get_name(),
|
||||
],
|
||||
'teamImage' => $this->get_sdk_uri() . 'assets/images/team.jpg',
|
||||
'strings' => [
|
||||
'aboutUs' => Loader::$labels['about_us']['title'],
|
||||
'heroHeader' => Loader::$labels['about_us']['heroHeader'],
|
||||
'heroTextFirst' => Loader::$labels['about_us']['heroTextFirst'],
|
||||
'heroTextSecond' => Loader::$labels['about_us']['heroTextSecond'],
|
||||
'teamImageCaption' => Loader::$labels['about_us']['teamImageCaption'],
|
||||
'newsHeading' => Loader::$labels['about_us']['newsHeading'],
|
||||
'emailPlaceholder' => Loader::$labels['about_us']['emailPlaceholder'],
|
||||
'signMeUp' => Loader::$labels['about_us']['signMeUp'],
|
||||
'installNow' => Loader::$labels['about_us']['installNow'],
|
||||
'activate' => Loader::$labels['about_us']['activate'],
|
||||
'learnMore' => Loader::$labels['about_us']['learnMore'],
|
||||
'installed' => Loader::$labels['about_us']['installed'],
|
||||
'notInstalled' => Loader::$labels['about_us']['notInstalled'],
|
||||
'active' => Loader::$labels['about_us']['active'],
|
||||
],
|
||||
'canInstallPlugins' => current_user_can( 'install_plugins' ),
|
||||
'canActivatePlugins' => current_user_can( 'activate_plugins' ),
|
||||
'showReviewLink' => ! ( isset( $this->about_data['review_link'] ) && false === $this->about_data['review_link'] ),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Get product pages data.
|
||||
*
|
||||
* @param array $product_pages Product pages.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
private function get_product_pages_data( $product_pages ) {
|
||||
|
||||
$otter_slug = 'otter-blocks';
|
||||
$otter_plugin = [
|
||||
'status' => 'not-installed',
|
||||
];
|
||||
$otter_plugin['status'] = $this->is_plugin_installed( $otter_slug ) ? 'installed' : 'not-installed';
|
||||
$otter_plugin['status'] = $this->is_plugin_active( $otter_slug ) ? 'active' : $otter_plugin['status'];
|
||||
$otter_plugin['activationLink'] = $this->get_plugin_activation_link( $otter_slug );
|
||||
|
||||
$pages = [
|
||||
'otter-page' => [
|
||||
'name' => 'Otter Blocks',
|
||||
'hash' => '#otter-page',
|
||||
'product' => $otter_slug,
|
||||
'plugin' => $otter_plugin,
|
||||
'strings' => [
|
||||
'heading' => Loader::$labels['about_us']['otter-page']['heading'],
|
||||
'text' => Loader::$labels['about_us']['otter-page']['text'],
|
||||
'buttons' => [
|
||||
'install_otter_free' => Loader::$labels['about_us']['otter-page']['buttons']['install_otter_free'],
|
||||
'install_now' => Loader::$labels['about_us']['otter-page']['buttons']['install_now'],
|
||||
'learn_more' => Loader::$labels['about_us']['otter-page']['buttons']['learn_more'],
|
||||
'learn_more_link' => tsdk_utmify( 'https://themeisle.com/plugins/otter-blocks/', 'otter-page', 'about-us' ),
|
||||
],
|
||||
'features' => [
|
||||
'advancedTitle' => Loader::$labels['about_us']['otter-page']['features']['advancedTitle'],
|
||||
'advancedDesc' => Loader::$labels['about_us']['otter-page']['features']['advancedDesc'],
|
||||
'fastTitle' => Loader::$labels['about_us']['otter-page']['features']['fastTitle'],
|
||||
'fastDesc' => Loader::$labels['about_us']['otter-page']['features']['fastDesc'],
|
||||
'mobileTitle' => Loader::$labels['about_us']['otter-page']['features']['mobileTitle'],
|
||||
'mobileDesc' => Loader::$labels['about_us']['otter-page']['features']['mobileDesc'],
|
||||
],
|
||||
'details' => [
|
||||
's1Title' => Loader::$labels['about_us']['otter-page']['details']['s1Title'],
|
||||
's1Text' => Loader::$labels['about_us']['otter-page']['details']['s1Text'],
|
||||
's2Title' => Loader::$labels['about_us']['otter-page']['details']['s2Title'],
|
||||
's2Text' => Loader::$labels['about_us']['otter-page']['details']['s2Text'],
|
||||
's3Title' => Loader::$labels['about_us']['otter-page']['details']['s3Title'],
|
||||
's3Text' => Loader::$labels['about_us']['otter-page']['details']['s3Text'],
|
||||
's1Image' => $this->get_sdk_uri() . 'assets/images/otter/otter-builder.png',
|
||||
's2Image' => $this->get_sdk_uri() . 'assets/images/otter/otter-patterns.png',
|
||||
's3Image' => $this->get_sdk_uri() . 'assets/images/otter/otter-library.png',
|
||||
],
|
||||
'testimonials' => [
|
||||
'heading' => Loader::$labels['about_us']['otter-page']['testimonials']['heading'],
|
||||
'users' => [
|
||||
[
|
||||
'avatar' => 'https://mllj2j8xvfl0.i.optimole.com/cb:3970~373ad/w:80/h:80/q:mauto/https://themeisle.com/wp-content/uploads/2021/05/avatar-03.png',
|
||||
'name' => 'Michael Burry',
|
||||
'text' => Loader::$labels['about_us']['otter-page']['testimonials']['users']['user_1'],
|
||||
],
|
||||
[
|
||||
'avatar' => 'https://mllj2j8xvfl0.i.optimole.com/cb:3970~373ad/w:80/h:80/q:mauto/https://themeisle.com/wp-content/uploads/2022/04/avatar-04.png',
|
||||
'name' => 'Maria Gonzales',
|
||||
'text' => Loader::$labels['about_us']['otter-page']['testimonials']['users']['user_2'],
|
||||
],
|
||||
[
|
||||
'avatar' => 'https://mllj2j8xvfl0.i.optimole.com/cb:3970~373ad/w:80/h:80/q:mauto/https://themeisle.com/wp-content/uploads/2022/04/avatar-05.png',
|
||||
'name' => 'Florian Henckel',
|
||||
'text' => Loader::$labels['about_us']['otter-page']['testimonials']['users']['user_3'],
|
||||
],
|
||||
],
|
||||
],
|
||||
],
|
||||
],
|
||||
];
|
||||
|
||||
return array_filter(
|
||||
$pages,
|
||||
function ( $page_data, $page_key ) use ( $product_pages ) {
|
||||
return in_array( $page_key, $product_pages, true ) &&
|
||||
isset( $page_data['plugin']['status'] ) &&
|
||||
$page_data['plugin']['status'] === 'not-installed';
|
||||
},
|
||||
ARRAY_FILTER_USE_BOTH
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get products data.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
private function get_other_products_data() {
|
||||
$products = [
|
||||
'optimole-wp' => [
|
||||
'name' => 'Optimole',
|
||||
'description' => Loader::$labels['about_us']['others']['optimole_desc'],
|
||||
],
|
||||
'neve' => [
|
||||
'skip_api' => true,
|
||||
'name' => 'Neve',
|
||||
'description' => Loader::$labels['about_us']['others']['neve_desc'],
|
||||
'icon' => $this->get_sdk_uri() . 'assets/images/neve.png',
|
||||
],
|
||||
'learning-management-system' => [
|
||||
'name' => 'Masteriyo LMS',
|
||||
],
|
||||
'otter-blocks' => [
|
||||
'name' => 'Otter',
|
||||
],
|
||||
'tweet-old-post' => [
|
||||
'name' => 'Revive Social',
|
||||
],
|
||||
'feedzy-rss-feeds' => [
|
||||
'name' => 'Feedzy',
|
||||
],
|
||||
'woocommerce-product-addon' => [
|
||||
'name' => 'PPOM',
|
||||
'condition' => class_exists( 'WooCommerce', false ),
|
||||
],
|
||||
'visualizer' => [
|
||||
'name' => 'Visualizer',
|
||||
],
|
||||
'wp-landing-kit' => [
|
||||
'skip_api' => true,
|
||||
'premiumUrl' => tsdk_utmify( 'https://themeisle.com/plugins/wp-landing-kit', $this->get_about_page_slug() ),
|
||||
'name' => 'WP Landing Kit',
|
||||
'description' => Loader::$labels['about_us']['others']['landingkit_desc'],
|
||||
'icon' => $this->get_sdk_uri() . 'assets/images/wplk.png',
|
||||
],
|
||||
'multiple-pages-generator-by-porthas' => [
|
||||
'name' => 'MPG',
|
||||
],
|
||||
'sparks-for-woocommerce' => [
|
||||
'skip_api' => true,
|
||||
'premiumUrl' => tsdk_utmify( 'https://themeisle.com/plugins/sparks-for-woocommerce', $this->get_about_page_slug() ),
|
||||
'name' => 'Sparks',
|
||||
'description' => Loader::$labels['about_us']['others']['sparks_desc'],
|
||||
'icon' => $this->get_sdk_uri() . 'assets/images/sparks.png',
|
||||
'condition' => class_exists( 'WooCommerce', false ),
|
||||
],
|
||||
'templates-patterns-collection' => [
|
||||
'name' => 'Templates Cloud',
|
||||
'description' => Loader::$labels['about_us']['others']['tpc_desc'],
|
||||
],
|
||||
'wp-cloudflare-page-cache' => [
|
||||
'name' => 'Super Page Cache',
|
||||
],
|
||||
'hyve-lite' => [
|
||||
'name' => 'Hyve Lite',
|
||||
],
|
||||
'wp-full-stripe-free' => [
|
||||
'name' => 'WP Full Pay',
|
||||
],
|
||||
];
|
||||
|
||||
foreach ( $products as $slug => $product ) {
|
||||
if ( isset( $product['condition'] ) && ! $product['condition'] ) {
|
||||
unset( $products[ $slug ] );
|
||||
continue;
|
||||
}
|
||||
|
||||
if ( $slug === 'neve' ) {
|
||||
$theme = get_template();
|
||||
$themes = wp_get_themes();
|
||||
|
||||
$products[ $slug ]['status'] = isset( $themes['neve'] ) ? 'installed' : 'not-installed';
|
||||
$products[ $slug ]['status'] = $theme === 'neve' ? 'active' : $products[ $slug ]['status'];
|
||||
|
||||
$products[ $slug ]['activationLink'] = add_query_arg(
|
||||
[
|
||||
'stylesheet' => 'neve',
|
||||
'action' => 'activate',
|
||||
'_wpnonce' => wp_create_nonce( 'switch-theme_neve' ),
|
||||
],
|
||||
admin_url( 'themes.php' )
|
||||
);
|
||||
|
||||
continue;
|
||||
}
|
||||
|
||||
$products[ $slug ]['status'] = $this->is_plugin_installed( $slug ) ? 'installed' : 'not-installed';
|
||||
$products[ $slug ]['status'] = $this->is_plugin_active( $slug ) ? 'active' : $products[ $slug ]['status'];
|
||||
$products[ $slug ]['activationLink'] = $this->get_plugin_activation_link( $slug );
|
||||
|
||||
|
||||
if ( isset( $product['skip_api'] ) ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$api_data = $this->call_plugin_api( $slug );
|
||||
if ( ! isset( $product['icon'] ) && ( isset( $api_data->icons['2x'] ) || $api_data->icons['1x'] ) ) {
|
||||
$products[ $slug ]['icon'] = isset( $api_data->icons['2x'] ) ? $api_data->icons['2x'] : $api_data->icons['1x'];
|
||||
}
|
||||
if ( ! isset( $product['description'] ) && isset( $api_data->short_description ) ) {
|
||||
$products[ $slug ]['description'] = $api_data->short_description;
|
||||
}
|
||||
if ( ! isset( $product['name'] ) && isset( $api_data->name ) ) {
|
||||
$products[ $slug ]['name'] = $api_data->name;
|
||||
}
|
||||
}
|
||||
|
||||
return $products;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the page slug.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
private function get_about_page_slug() {
|
||||
return 'ti-about-' . $this->product->get_key();
|
||||
}
|
||||
}
|
||||
412
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/Announcements.php
vendored
Normal file
@@ -0,0 +1,412 @@
|
||||
<?php
|
||||
/**
|
||||
* File responsible for announcements.
|
||||
*
|
||||
* This is used to display information about limited events, such as Black Friday.
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 3.3.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use DateTime;
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Loader;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
/**
|
||||
* Announcement module for the ThemeIsle SDK.
|
||||
*/
|
||||
class Announcements extends Abstract_Module {
|
||||
|
||||
const SALE_DURATION_BLACK_FRIDAY = '+7 days'; // DateTime modifier. (Include Cyber Monday)
|
||||
|
||||
/**
|
||||
* Mark if the notice was already loaded.
|
||||
*
|
||||
* @var boolean
|
||||
*/
|
||||
private static $notice_loaded = false;
|
||||
|
||||
/**
|
||||
* The product to be used.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
private static $current_product = '';
|
||||
|
||||
/**
|
||||
* Check if the module can be loaded.
|
||||
*
|
||||
* @param Product $product Product data.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Load the module for the selected product.
|
||||
*
|
||||
* @param Product $product Product data.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function load( $product ) {
|
||||
$this->product = $product;
|
||||
|
||||
add_filter(
|
||||
'themeisle_sdk_is_black_friday_sale',
|
||||
function( $is_black_friday ) {
|
||||
return $this->is_black_friday_sale( $this->get_current_date() );
|
||||
}
|
||||
);
|
||||
|
||||
add_action( 'admin_init', array( $this, 'load_announcements' ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Load all valid announcements.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function load_announcements() {
|
||||
if ( ! $this->is_black_friday_sale( $this->get_current_date() ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
add_action( 'admin_notices', array( $this, 'black_friday_notice_render' ) );
|
||||
add_action( 'wp_ajax_themeisle_sdk_dismiss_black_friday_notice', array( $this, 'disable_notification_ajax' ) );
|
||||
add_action(
|
||||
'themeisle_internal_page',
|
||||
function( $plugin, $page_slug ) {
|
||||
self::$current_product = $plugin;
|
||||
},
|
||||
10,
|
||||
2
|
||||
);
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* Get the remaining time for the event in a human-readable format.
|
||||
*
|
||||
* @param DateTime $end_date The end date for event.
|
||||
*
|
||||
* @return string Remaining time for the event.
|
||||
*/
|
||||
public function get_remaining_time_for_event( $end_date ) {
|
||||
return human_time_diff( $this->get_current_date()->getTimestamp(), $end_date->getTimestamp() );
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if the announcement can be shown.
|
||||
*
|
||||
* @param DateTime $current_date The announcement to check.
|
||||
* @param int $user_id The user id to show the notice.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function can_show_notice( $current_date, $user_id ) {
|
||||
$current_year = $current_date->format( 'Y' );
|
||||
$user_notice_dismiss_timestamp = get_user_meta( $user_id, 'themeisle_sdk_dismissed_notice_black_friday', true );
|
||||
|
||||
if ( empty( $user_notice_dismiss_timestamp ) ) {
|
||||
return true;
|
||||
}
|
||||
|
||||
$dismissed_year = wp_date( 'Y', $user_notice_dismiss_timestamp );
|
||||
|
||||
return $current_year !== $dismissed_year;
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculate the start date for Black Friday based on the year of the given date.
|
||||
*
|
||||
* Black Friday is the day after the Thanksgiving and the sale starts on the Monday of that week.
|
||||
*
|
||||
* @param DateTime $date The current date object, used to determine the year.
|
||||
* @return DateTime The start date of Black Friday for the given year.
|
||||
*/
|
||||
public function get_start_date( $date ) {
|
||||
$year = $date->format( 'Y' );
|
||||
$black_friday = new DateTime( "last friday of november {$year}" );
|
||||
|
||||
$sale_start = clone $black_friday;
|
||||
$sale_start->modify( 'monday this week' );
|
||||
$sale_start->setTime( 0, 0 );
|
||||
|
||||
return $sale_start;
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculate the event end date.
|
||||
*
|
||||
* @param DateTime $start_date The start date.
|
||||
* @return DateTime The end date.
|
||||
*/
|
||||
public function get_end_date( $start_date ) {
|
||||
$black_friday_end = clone $start_date;
|
||||
$black_friday_end->modify( self::SALE_DURATION_BLACK_FRIDAY );
|
||||
$black_friday_end->setTime( 23, 59, 59 );
|
||||
return $black_friday_end;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if the current date falls within the Black Friday sale period.
|
||||
*
|
||||
* @param DateTime $current_date The date to check.
|
||||
* @return bool True if the date is within the Black Friday sale period, false otherwise.
|
||||
*/
|
||||
public function is_black_friday_sale( $current_date ) {
|
||||
$black_friday_start_date = $this->get_start_date( $current_date );
|
||||
$black_friday_end = $this->get_end_date( $black_friday_start_date );
|
||||
return $black_friday_start_date <= $current_date && $current_date <= $black_friday_end;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the notice data.
|
||||
*
|
||||
* @return array The notice data.
|
||||
*/
|
||||
public function get_notice_data() {
|
||||
$time_left_label = $this->get_remaining_time_for_event( $this->get_end_date( $this->get_start_date( $this->get_current_date() ) ) );
|
||||
$time_left_label = sprintf( Loader::$labels['announcements']['time_left'], $time_left_label );
|
||||
|
||||
$utm_location = 'globalnotice';
|
||||
if ( ! empty( $this->product ) ) {
|
||||
$utm_location = $this->product->get_friendly_name();
|
||||
}
|
||||
|
||||
$sale_title = Loader::$labels['announcements']['black_friday'];
|
||||
$sale_url = tsdk_translate_link( tsdk_utmify( 'https://themeisle.com/blackfriday/', 'bfcm25', $utm_location ) );
|
||||
$sale_message = sprintf( Loader::$labels['announcements']['max_savings'], '50%' );
|
||||
|
||||
return array(
|
||||
'title' => $sale_title,
|
||||
'sale_url' => $sale_url,
|
||||
'message' => $sale_message,
|
||||
'time_left' => $time_left_label,
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Render the Black Friday notice.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function black_friday_notice_render() {
|
||||
|
||||
// Prevent the notice from being rendered twice.
|
||||
if ( self::$notice_loaded ) {
|
||||
return;
|
||||
}
|
||||
self::$notice_loaded = true;
|
||||
|
||||
$all_configs = apply_filters( 'themeisle_sdk_blackfriday_data', array( 'default' => $this->get_notice_data() ) );
|
||||
|
||||
if ( empty( $all_configs ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$data = end( $all_configs );
|
||||
|
||||
if ( ! empty( self::$current_product ) && isset( $all_configs[ self::$current_product ] ) ) {
|
||||
$data = $all_configs[ self::$current_product ];
|
||||
}
|
||||
|
||||
if ( empty( $data ) || ! is_array( $data ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$current_user_id = get_current_user_id();
|
||||
$can_dismiss = true;
|
||||
|
||||
if ( ! empty( $data['dismiss'] ) ) {
|
||||
$can_dismiss = $data['dismiss'];
|
||||
} else {
|
||||
// Disable by default if we are on a product page.
|
||||
if ( 0 < did_action( 'themeisle_internal_page' ) ) {
|
||||
$can_dismiss = false;
|
||||
}
|
||||
}
|
||||
|
||||
if ( $can_dismiss && ! $this->can_show_notice( $this->get_current_date(), $current_user_id ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$logo_url = ! empty( $data['logo_url'] ) ? $data['logo_url'] : $this->get_sdk_uri() . 'assets/images/themeisle-logo.png';
|
||||
$cta_label = ! empty( $data['cta_label'] ) ? $data['cta_label'] : Loader::$labels['announcements']['notice_link_label'];
|
||||
$sale_url = ! empty( $data['sale_url'] ) ? $data['sale_url'] : '';
|
||||
$hide_other_notices = ! empty( $data['hide_other_notices'] ) ? $data['hide_other_notices'] : ! $can_dismiss;
|
||||
$dismiss_notice_url = wp_nonce_url(
|
||||
add_query_arg(
|
||||
array( 'action' => 'themeisle_sdk_dismiss_black_friday_notice' ),
|
||||
admin_url( 'admin-ajax.php' )
|
||||
),
|
||||
'dismiss_themeisle_event_notice'
|
||||
);
|
||||
|
||||
if ( empty( $sale_url ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( ! current_user_can( 'install_plugins' ) ) {
|
||||
$sale_url = remove_query_arg( 'lkey', $sale_url );
|
||||
}
|
||||
|
||||
?>
|
||||
<style>
|
||||
.themeisle-sale {
|
||||
border-left-color: #0466CB;
|
||||
}
|
||||
.themeisle-sale :is(.themeisle-sale-title, p) {
|
||||
margin: 0;
|
||||
}
|
||||
.themeisle-sale-container {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
padding: 0.5rem 0;
|
||||
gap: 0.5rem;
|
||||
padding-right: 10px;
|
||||
}
|
||||
.themeisle-sale-content {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 0.2rem;
|
||||
}
|
||||
.themeisle-sale a {
|
||||
text-decoration: none;
|
||||
}
|
||||
.themeisle-sale p a {
|
||||
margin-left: 1rem;
|
||||
padding: 7px 12px;
|
||||
border-radius: 4px;
|
||||
background: #0466CB;
|
||||
color: white;
|
||||
font-weight: 700;
|
||||
}
|
||||
.themeisle-sale-dismiss {
|
||||
padding-top: 5px;
|
||||
}
|
||||
.themeisle-sale-dismiss span {
|
||||
color: #787c82;
|
||||
font-size: 16px;
|
||||
}
|
||||
.notice.themeisle-sale {
|
||||
padding: 0;
|
||||
}
|
||||
.themeisle-sale-logo {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
margin-left: 5px;
|
||||
}
|
||||
.themeisle-sale-time-left {
|
||||
margin-left: 5px;
|
||||
padding: 3px 5px;
|
||||
border-radius: 4px;
|
||||
background-color: #dfdfdf;
|
||||
font-weight: 600;
|
||||
font-size: x-small;
|
||||
line-height: 1;
|
||||
}
|
||||
.themeisle-sale-title {
|
||||
font-size: 14px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
}
|
||||
.themeisle-sale-action {
|
||||
flex-grow: 1;
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
}
|
||||
<?php if ( $hide_other_notices ) : ?>
|
||||
.notice:not(.themeisle-sale) {
|
||||
display: none;
|
||||
}
|
||||
<?php endif; ?>
|
||||
</style>
|
||||
<div class="themeisle-sale notice notice-info" data-event-slug="black_friday">
|
||||
<div class="themeisle-sale-container">
|
||||
<div class="themeisle-sale-logo">
|
||||
<img
|
||||
width="45"
|
||||
src="<?php echo esc_url( $logo_url ); ?>"
|
||||
/>
|
||||
</div>
|
||||
<div class="themeisle-sale-content">
|
||||
<h4 class="themeisle-sale-title">
|
||||
<?php echo esc_html( $data['title'] ); ?>
|
||||
<span class="themeisle-sale-time-left">
|
||||
<?php echo esc_html( $data['time_left'] ); ?>
|
||||
</span>
|
||||
</h4>
|
||||
<p>
|
||||
<?php echo wp_kses_post( $data['message'] ); ?>
|
||||
</p>
|
||||
</div>
|
||||
<div class="themeisle-sale-action">
|
||||
<a
|
||||
href="<?php echo esc_url( $sale_url ); ?>"
|
||||
target="_blank"
|
||||
class="button button-primary themeisle-sale-button"
|
||||
>
|
||||
<?php echo esc_html( $cta_label ); ?>
|
||||
</a>
|
||||
</div>
|
||||
<?php if ( $can_dismiss ) : ?>
|
||||
<a href="<?php echo esc_url( $dismiss_notice_url ); ?>" class="themeisle-sale-dismiss">
|
||||
<span class="dashicons dashicons-dismiss"></span>
|
||||
</a>
|
||||
<?php endif; ?>
|
||||
</div>
|
||||
</div>
|
||||
<script>
|
||||
// Note: Some plugins use React and the content is ready after the `DOMContentLoaded` event. Use this function to reposition the notice after components have rendered.
|
||||
window.tsdk_reposition_notice = function() {
|
||||
const bannerRoot = document.getElementById('tsdk_banner');
|
||||
const saleNotice = document.querySelector('.themeisle-sale');
|
||||
if ( ! bannerRoot || ! saleNotice ) {
|
||||
return;
|
||||
}
|
||||
|
||||
bannerRoot.appendChild(saleNotice);
|
||||
};
|
||||
|
||||
document.addEventListener( 'DOMContentLoaded', function() {
|
||||
window.tsdk_reposition_notice();
|
||||
} );
|
||||
</script>
|
||||
<?php
|
||||
}
|
||||
|
||||
/**
|
||||
* Disable the notification via ajax.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function disable_notification_ajax() {
|
||||
check_ajax_referer( 'dismiss_themeisle_event_notice' );
|
||||
|
||||
update_user_meta( get_current_user_id(), 'themeisle_sdk_dismissed_notice_black_friday', $this->get_current_date()->getTimestamp() );
|
||||
|
||||
$return_page_url = wp_get_referer();
|
||||
if ( empty( $return_page_url ) ) {
|
||||
$return_page_url = admin_url();
|
||||
}
|
||||
|
||||
wp_safe_redirect( $return_page_url );
|
||||
exit;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,235 @@
|
||||
<?php
|
||||
/**
|
||||
* The compatibilities model class for ThemeIsle SDK
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 1.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Loader;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Promotions module for ThemeIsle SDK.
|
||||
*/
|
||||
class Compatibilities extends Abstract_Module {
|
||||
const REQUIRED = 'required';
|
||||
const TESTED_UP = 'tested_up';
|
||||
|
||||
/**
|
||||
* Should we load this module.
|
||||
*
|
||||
* @param Product $product Product object.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
if ( $product->is_theme() && ! current_user_can( 'switch_themes' ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( $product->is_plugin() && ! current_user_can( 'install_plugins' ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Registers the hooks.
|
||||
*
|
||||
* @param Product $product Product to load.
|
||||
*
|
||||
* @return Compatibilities Module instance.
|
||||
* @throws \Exception If the configuration is invalid.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
|
||||
|
||||
$this->product = $product;
|
||||
|
||||
$compatibilities = apply_filters( 'themeisle_sdk_compatibilities/' . $this->product->get_slug(), [] );
|
||||
if ( empty( $compatibilities ) ) {
|
||||
return $this;
|
||||
}
|
||||
$requirement = null;
|
||||
$check_type = null;
|
||||
foreach ( $compatibilities as $compatibility ) {
|
||||
|
||||
if ( empty( $compatibility['basefile'] ) ) {
|
||||
return $this;
|
||||
}
|
||||
$requirement = new Product( $compatibility['basefile'] );
|
||||
$tested_up = isset( $compatibility[ self::TESTED_UP ] ) ? $compatibility[ self::TESTED_UP ] : '999';
|
||||
$required = $compatibility[ self::REQUIRED ];
|
||||
if ( ! version_compare( $required, $tested_up, '<' ) ) {
|
||||
throw new \Exception( sprintf( 'Invalid required/tested_up configuration. Required version %s should be lower than tested_up %s.', $required, $tested_up ) );
|
||||
}
|
||||
$check_type = self::REQUIRED;
|
||||
if ( ! version_compare( $requirement->get_version(), $required, '<' ) ) {
|
||||
$check_type = self::TESTED_UP;
|
||||
if ( version_compare( $requirement->get_version(), $tested_up . '.9999', '<' ) ) {
|
||||
return $this;
|
||||
}
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
if ( empty( $requirement ) ) {
|
||||
return $this;
|
||||
}
|
||||
if ( $check_type === self::REQUIRED ) {
|
||||
$this->mark_required( $product, $requirement );
|
||||
}
|
||||
if ( $check_type === self::TESTED_UP ) {
|
||||
$this->mark_testedup( $product, $requirement );
|
||||
}
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Mark the product tested up.
|
||||
*
|
||||
* @param Product $product Product object.
|
||||
* @param Product $requirement Requirement object.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function mark_testedup( $product, $requirement ) {
|
||||
add_action(
|
||||
'admin_head',
|
||||
function () use ( $product, $requirement ) {
|
||||
$screen = function_exists( 'get_current_screen' ) ? get_current_screen() : '';
|
||||
|
||||
if ( empty( $screen ) || ! isset( $screen->id ) ) {
|
||||
return;
|
||||
}
|
||||
if ( $requirement->is_theme() && $screen->id === 'themes' ) {
|
||||
?>
|
||||
<script type="text/javascript">
|
||||
jQuery(document).ready(function ($) {
|
||||
setInterval(checkTheme, 500);
|
||||
|
||||
function checkTheme() {
|
||||
var theme = jQuery('.theme.active[data-slug="<?php echo esc_attr( $requirement->get_slug() ); ?>"]');
|
||||
var notice_id = 'testedup<?php echo esc_attr( $requirement->get_slug() . $product->get_slug() ); ?>';
|
||||
if (theme.length > 0 && jQuery('#' + notice_id).length === 0) {
|
||||
theme.find('.theme-id-container').prepend('<div style="bottom:100%;top:auto;" id="' + notice_id + '" class="notice notice-warning"><?php echo sprintf( esc_html( Loader::$labels['compatibilities']['notice_theme'] ), '<strong>', '</strong>', esc_attr( $product->get_friendly_name() ) ); ?></div>');
|
||||
}
|
||||
if (theme.length > 0 && jQuery('#' + notice_id + 'overlay').length === 0) {
|
||||
jQuery('.theme-overlay.active .theme-author').after('<div style="bottom:100%;top:auto;" id="' + notice_id + 'overlay" class="notice notice-warning"><p><?php echo sprintf( esc_html( Loader::$labels['compatibilities']['notice_theme'] ), '<strong>', '</strong>', esc_attr( $product->get_friendly_name() ) ); ?></p></div>');
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
</script>
|
||||
<?php
|
||||
}
|
||||
if ( $requirement->is_plugin() && $screen->id === 'plugins' ) {
|
||||
?>
|
||||
<script type="text/javascript">
|
||||
jQuery(document).ready(function ($) {
|
||||
setInterval(checkPlugin, 500);
|
||||
|
||||
function checkPlugin() {
|
||||
var plugin = jQuery('.plugins .active[data-slug="<?php echo esc_attr( $requirement->get_slug() ); ?>"]');
|
||||
var notice_id = 'testedup<?php echo esc_attr( $requirement->get_slug() . $product->get_slug() ); ?>';
|
||||
if (plugin.length > 0 && jQuery('#' + notice_id).length === 0) {
|
||||
plugin.find('.column-description').append('<div style="bottom:100%;top:auto;" id="' + notice_id + '" class="notice notice-warning notice-alt notice-inline"><?php echo sprintf( esc_html( Loader::$labels['compatibilities']['notice_plugin'] ), '<strong>', '</strong>', esc_attr( $product->get_friendly_name() ), esc_attr( $product->is_plugin() ? Loader::$labels['compatibilities']['plugin'] : Loader::$labels['compatibilities']['theme'] ) ); ?></div>');
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
</script>
|
||||
<?php
|
||||
}
|
||||
}
|
||||
);
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Mark the product requirements.
|
||||
*
|
||||
* @param Product $product Product object.
|
||||
* @param Product $requirement Requirement object.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function mark_required( $product, $requirement ) {
|
||||
add_filter(
|
||||
'upgrader_pre_download',
|
||||
function ( $return, $package, $upgrader ) use ( $product, $requirement ) {
|
||||
/**
|
||||
* Upgrader object.
|
||||
*
|
||||
* @var \WP_Upgrader $upgrader Upgrader object.
|
||||
*/
|
||||
$should_block = false;
|
||||
if ( $product->is_theme()
|
||||
&& property_exists( $upgrader, 'skin' )
|
||||
&& property_exists( $upgrader->skin, 'theme_info' )
|
||||
&& $upgrader->skin->theme_info->template === $product->get_slug() ) {
|
||||
$should_block = true;
|
||||
|
||||
}
|
||||
if ( ! $should_block && $product->is_plugin()
|
||||
&& property_exists( $upgrader, 'skin' )
|
||||
&& property_exists( $upgrader->skin, 'plugin_info' )
|
||||
&& $upgrader->skin->plugin_info['Name'] === $product->get_name() ) {
|
||||
$should_block = true;
|
||||
}
|
||||
if ( $should_block ) {
|
||||
echo( sprintf(
|
||||
esc_html( Loader::$labels['compatibilities']['notice2'] ),
|
||||
esc_attr( $product->get_friendly_name() ),
|
||||
esc_attr( $requirement->get_friendly_name() ),
|
||||
'<a href="' . esc_url( admin_url( $requirement->is_theme() ? 'themes.php' : 'plugins.php' ) ) . '">',
|
||||
'</a>',
|
||||
esc_attr( $requirement->get_friendly_name() ),
|
||||
esc_attr( $requirement->is_theme() ? Loader::$labels['compatibilities']['theme'] : Loader::$labels['compatibilities']['plugin'] )
|
||||
) );
|
||||
$upgrader->maintenance_mode( false );
|
||||
die();
|
||||
}
|
||||
|
||||
return $return;
|
||||
},
|
||||
10,
|
||||
3
|
||||
);
|
||||
|
||||
add_action(
|
||||
'admin_notices',
|
||||
function () use ( $product, $requirement ) {
|
||||
echo '<div class="notice notice-error "><p>';
|
||||
echo( sprintf(
|
||||
esc_html( Loader::$labels['compatibilities']['notice'] ),
|
||||
'<strong>' . esc_attr( $product->get_friendly_name() ) . '</strong>',
|
||||
'<strong>' . esc_attr( $requirement->get_friendly_name() ) . '</strong>',
|
||||
'<a href="' . esc_url( admin_url( $requirement->is_theme() ? 'themes.php' : 'plugins.php' ) ) . '">',
|
||||
'</a>',
|
||||
'<strong>' . esc_attr( $requirement->get_friendly_name() ) . '</strong>',
|
||||
esc_attr( $requirement->is_theme() ? Loader::$labels['compatibilities']['theme'] : Loader::$labels['compatibilities']['plugin'] )
|
||||
) );
|
||||
echo '</p></div>';
|
||||
}
|
||||
);
|
||||
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,506 @@
|
||||
<?php
|
||||
/**
|
||||
* The blog dashboard model class for ThemeIsle SDK
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 1.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Loader;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Blog dashboard widget module for ThemeIsle SDK.
|
||||
*/
|
||||
class Dashboard_Widget extends Abstract_Module {
|
||||
|
||||
/**
|
||||
* Fetched feeds items.
|
||||
*
|
||||
* @var array Feed items.
|
||||
*/
|
||||
private $items = array();
|
||||
|
||||
/**
|
||||
* Dashboard widget title.
|
||||
*
|
||||
* @var string $dashboard_name Dashboard name.
|
||||
*/
|
||||
private $dashboard_name = '';
|
||||
|
||||
/**
|
||||
* Dashboard widget feed sources.
|
||||
*
|
||||
* @var array $feeds Feed url.
|
||||
*/
|
||||
private $feeds = [];
|
||||
|
||||
/**
|
||||
* Should we load this module.
|
||||
*
|
||||
* @param Product $product Product object.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( ! apply_filters( $product->get_slug() . '_load_dashboard_widget', true ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Registers the hooks.
|
||||
*
|
||||
* @param Product $product Product to load.
|
||||
*
|
||||
* @return Dashboard_Widget Module instance.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
if ( apply_filters( 'themeisle_sdk_hide_dashboard_widget', false ) ) {
|
||||
return;
|
||||
}
|
||||
$this->product = $product;
|
||||
$this->dashboard_name = apply_filters( 'themeisle_sdk_dashboard_widget_name', Loader::$labels['dashboard_widget']['title'] );
|
||||
$this->feeds = apply_filters(
|
||||
'themeisle_sdk_dashboard_widget_feeds',
|
||||
[
|
||||
'https://themeisle.com/blog/feed',
|
||||
'https://wpshout.com/feed',
|
||||
]
|
||||
);
|
||||
add_action( 'wp_dashboard_setup', array( &$this, 'add_widget' ) );
|
||||
add_action( 'wp_network_dashboard_setup', array( &$this, 'add_widget' ) );
|
||||
add_filter( 'themeisle_sdk_recommend_plugin_or_theme', array( &$this, 'recommend_plugin_or_theme' ) );
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Add widget to the dashboard
|
||||
*
|
||||
* @return string|void
|
||||
*/
|
||||
public function add_widget() {
|
||||
global $wp_meta_boxes;
|
||||
if ( isset( $wp_meta_boxes['dashboard']['normal']['core']['themeisle'] ) ) {
|
||||
return;
|
||||
}
|
||||
wp_add_dashboard_widget(
|
||||
'themeisle',
|
||||
$this->dashboard_name,
|
||||
[
|
||||
$this,
|
||||
'render_dashboard_widget',
|
||||
]
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Render widget content
|
||||
*/
|
||||
public function render_dashboard_widget() {
|
||||
$this->setup_feeds();
|
||||
if ( empty( $this->items ) || ! is_array( $this->items ) ) {
|
||||
return;
|
||||
}
|
||||
?>
|
||||
<style type="text/css">
|
||||
#themeisle ul li.ti-dw-recommend-item {
|
||||
padding-left: 7px;
|
||||
border-top: 1px solid #eee;
|
||||
margin-bottom: 0px;
|
||||
padding-top: 6px;
|
||||
}
|
||||
|
||||
#themeisle h2.hndle {
|
||||
background-image: url('data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAEgAAABbCAMAAADncTNAAAAAtFBMVEVHcEyAgIB/f3+xsbGgoaGBgYGCgoKKioqAgIC1tbW5ubnFx8iAgIDU1taBgYGCgoKAgIC0tLXW19jW2NiAgIC3uLiBgYHLzMy4uLhycnLW19d/f3/T1NW0tLTX19mVlZWvr6+BgYHl5eWKiottbW5JSUnW2Nm5ubnh4eHT1NWVlZVjY2N4eHh9fX6pqqq+v79PT0/39/fu7u7Nzc7Z2ttYWFgBAQHDw8P////JysoZGRk0NTZqJc/sAAAAIXRSTlMA0FL7/oEnEPL6eibivm9gwJya76/enFq2CXI+2lFAyM8GATmPAAADj0lEQVR4Xu2YaW/iOhSGAwRCWDosnXa6znjJvm8svf//f12TuARyhiR2pfnUR6gSEnr0+uT4xK7yRb755pvhHePli5K7Bfpkuhoq8ozRJdMH+WWha6Z3sqYparCSLRJqspjImVbANJU03cNMMpofAwQZCGsmpQYyFvVM0Q00OQ9koMl5IPcCoro+RA1Dt2Ea9n9eZ0+YHJLkgIlkDywQx00wCTyaReiKH8LbNU9ybJOdkchV6QFxyCFLbVvdfaREqgUWg/tx2UbqIcK2Hex2TdGLwFTjIj3XP3YfCZFsb23KRZn/3263oymSFI0/a5S4PqUBjoBIJBDjeEhCN0wxQSRybIxtJ3K5SGzuE/vAwIQc8ZmMMJFAIM4oikZItfEFtorGgoE43FObwqHU68OtPCnOz8KZ2Jbl5LgkSW0Tc7YyIz/EFWmS4jMbiZU5mJOmKRaJpKGGyLZtDJh3iyaNUu/3+xyKnrtFL71EG+FTiMpENhQtxUQ8kSOXCIr2tnCNhg/gTX0SHYFp0t7TCwQZ7U841yoHrW6rtGroUwTWVnLMssxx+H4bgZcSOFf5MYx0Ae8FghomMDyC2EBNImBywPkNTDNqGLQpIg2TjUNU8tBy9DQMo0DAZF16rAi7vJAtFTIYFAHUc6hIRW6OuOhJgaCSwmDEAYK4oa7ro+qIEyJU/US7KTJKPNSFT9tFgVFBu0SF1y7yjX4masRA9Da7EFGj28R/BkQz6xGIOurkx38T/bKs9Uk8aIiMwm/Jw0VP1yLrJwt13xAxvABBgsK4KWLov35DkRF7ZaqgzuZ7MQ8MOntmVYyAqKTwaICKqvSUFnVccMN5sziEP/5+xGDTahbH5Q3ZB76zr8fI+nJtvUUU3t3ml5GKviK/npCg3CGodnuJ4JVkfRFJYGVDBZrqKnn9RLf+CzDTS5PaN5J38+auzX4ykU4Qoj0rdKfcYs5ijfo9OL/uRUgZyQr7NCWtWwiUSLc4arfJa7lpszTA1OJZAQ8w8dXFrR5YHzCWSnS3pZ18tOi4Ps4vl/c7i/6qomjRecN+UubrPyPGn/VEMU3T0UFHkaPzpgjxmJsnjmrtionlMDZiog0TsY/DPtn8SXtlBvbtxKtwopy7lqW3smQO+yoGE1Uu55GJ3pmI8ygoejZNnqj0vnIRCyTKfLstRdtStGQi09myUsvwvlkuzSUXbV+Xz5ryBebV33fln/A/moud69FZiEYAAAAASUVORK5CYII=');
|
||||
background-repeat: no-repeat;
|
||||
background-position: 2% 50%;
|
||||
background-size: 25px;
|
||||
padding-left: 39px;
|
||||
}
|
||||
|
||||
#themeisle .inside {
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
.ti-feed-list {
|
||||
padding: 0 12px 5px;
|
||||
margin-bottom: 10px;
|
||||
border-bottom: 1px solid #eee;
|
||||
}
|
||||
|
||||
.ti-dw-feed-item a {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
margin-bottom: 5px;
|
||||
padding: 5px;
|
||||
transition: .2s ease;
|
||||
border-radius: 3px;
|
||||
}
|
||||
|
||||
.ti-dw-feed-item a:hover {
|
||||
background-color: #f8f8f8;
|
||||
}
|
||||
|
||||
.ti-dw-feed-item a:hover .ti-dw-date-container {
|
||||
opacity: .9;
|
||||
}
|
||||
|
||||
.ti-dw-feed-item .ti-dw-month-container {
|
||||
margin-top: -5px;
|
||||
text-transform: uppercase;
|
||||
font-size: 10px;
|
||||
letter-spacing: 1px;
|
||||
font-weight: 700;
|
||||
}
|
||||
|
||||
.ti-dw-feed-item .ti-dw-date-container {
|
||||
border-radius: 3px;
|
||||
transition: .2s ease;
|
||||
min-height: 35px;
|
||||
margin-right: 5px;
|
||||
min-width: 35px;
|
||||
text-align: center;
|
||||
border: 1px solid #2a6f97;
|
||||
color: #fff;
|
||||
background: #2ea2cc;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.ti-dw-footer {
|
||||
padding: 0 12px 5px;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.ti-dw-recommend-item {
|
||||
display: block;
|
||||
}
|
||||
|
||||
.ti-dw-recommend-item span {
|
||||
color: #72777c;
|
||||
}
|
||||
|
||||
.ti-dw-powered-by {
|
||||
font-size: 11px;
|
||||
margin-top: 3px;
|
||||
display: block;
|
||||
color: #72777c;
|
||||
}
|
||||
|
||||
.ti-dw-powered-by span {
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
</style>
|
||||
<?php do_action( 'themeisle_sdk_dashboard_widget_before', $this->product ); ?>
|
||||
|
||||
<ul class="ti-feed-list">
|
||||
<?php
|
||||
|
||||
foreach ( $this->items as $item ) {
|
||||
?>
|
||||
<li class="ti-dw-feed-item">
|
||||
<a href="
|
||||
<?php
|
||||
echo esc_url(
|
||||
add_query_arg(
|
||||
array(
|
||||
'utm_source' => 'wpadmin',
|
||||
'utm_campaign' => 'feed',
|
||||
'utm_medium' => 'dashboard_widget',
|
||||
),
|
||||
$item['link']
|
||||
)
|
||||
);
|
||||
?>
|
||||
" target="_blank">
|
||||
<span class="ti-dw-date-container"><span
|
||||
class="ti-dw-day-container"><?php echo esc_attr( gmdate( 'd', $item['date'] ) ); ?></span> <span
|
||||
class="ti-dw-month-container"><?php echo esc_attr( substr( gmdate( 'M', $item['date'] ), 0, 3 ) ); ?></span></span><?php echo esc_attr( $item['title'] ); ?>
|
||||
</a>
|
||||
</li>
|
||||
<?php
|
||||
}
|
||||
?>
|
||||
</ul>
|
||||
<?php
|
||||
$recommend = apply_filters( 'themeisle_sdk_recommend_plugin_or_theme', array() );
|
||||
if ( ! is_array( $recommend ) || empty( $recommend ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$type = $recommend['type'];
|
||||
|
||||
if ( ( 'theme' === $type && ! current_user_can( 'install_themes' ) ) ) {
|
||||
return;
|
||||
}
|
||||
if ( ( 'plugin' === $type && ! current_user_can( 'install_plugins' ) ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
add_thickbox();
|
||||
$url = add_query_arg(
|
||||
[
|
||||
'theme' => $recommend['slug'],
|
||||
],
|
||||
network_admin_url( 'theme-install.php' )
|
||||
);
|
||||
|
||||
if ( 'plugin' === $type ) {
|
||||
|
||||
$url = add_query_arg(
|
||||
array(
|
||||
'tab' => 'plugin-information',
|
||||
'plugin' => $recommend['slug'],
|
||||
),
|
||||
network_admin_url( 'plugin-install.php' )
|
||||
);
|
||||
}
|
||||
?>
|
||||
<div class="ti-dw-footer">
|
||||
<span class="ti-dw-recommend-item ">
|
||||
<span class="ti-dw-recommend"><?php echo esc_attr( apply_filters( 'themeisle_sdk_dashboard_popular_label', sprintf( Loader::$labels['dashboard_widget']['popular'], ucwords( $type ) ) ) ); ?>
|
||||
: </span>
|
||||
<?php
|
||||
echo esc_attr(
|
||||
trim(
|
||||
str_replace(
|
||||
array(
|
||||
'lite',
|
||||
'Lite',
|
||||
'(Lite)',
|
||||
'(lite)',
|
||||
),
|
||||
'',
|
||||
$recommend['name']
|
||||
)
|
||||
)
|
||||
);
|
||||
?>
|
||||
(<a class="thickbox open-plugin-details-modal"
|
||||
href="<?php echo esc_url( $url . '&TB_iframe=true&width=600&height=500' ); ?>"><?php echo esc_attr( apply_filters( 'themeisle_sdk_dashboard_install_label', Loader::$labels['dashboard_widget']['install'] ) ); ?></a>)
|
||||
</span>
|
||||
<span class="ti-dw-powered-by"><span><?php echo esc_attr( apply_filters( 'themeisle_sdk_dashboard_widget_powered_by', sprintf( Loader::$labels['dashboard_widget']['powered'], $this->product->get_friendly_name() ) ) ); ?></span></span>
|
||||
</div>
|
||||
|
||||
<?php
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Setup feed items.
|
||||
*/
|
||||
private function setup_feeds() {
|
||||
if ( false === ( $items_normalized = get_transient( 'themeisle_sdk_feed_items' ) ) ) { //phpcs:ignore Squiz.PHP.DisallowMultipleAssignments.FoundInControlStructure
|
||||
// Do not force the feed for the items in the sdk feeds list.
|
||||
// this prevents showing notices if another plugin will force all SimplePie feeds to load, instead it will
|
||||
// use the regular SimplePie validation and abort early if the feed is not valid.
|
||||
$sdk_feeds = $this->feeds;
|
||||
add_action(
|
||||
'wp_feed_options',
|
||||
function ( $feed, $url ) use ( $sdk_feeds ) {
|
||||
if ( defined( 'TI_SDK_PHPUNIT' ) && true === TI_SDK_PHPUNIT ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( ! is_string( $url ) && in_array( $url, $sdk_feeds, true ) ) {
|
||||
$feed->force_feed( false );
|
||||
|
||||
return;
|
||||
}
|
||||
if ( is_array( $url ) ) {
|
||||
foreach ( $url as $feed_url ) {
|
||||
if ( in_array( $feed_url, $sdk_feeds, true ) ) {
|
||||
$feed->force_feed( false );
|
||||
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
PHP_INT_MAX,
|
||||
2
|
||||
);
|
||||
// Load SimplePie Instance.
|
||||
$feed = fetch_feed( $sdk_feeds );
|
||||
// TODO report error when is an error loading the feed.
|
||||
if ( is_wp_error( $feed ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$items = $feed->get_items( 0, 5 );
|
||||
$items = is_array( $items ) ? $items : [];
|
||||
|
||||
$items_normalized = [];
|
||||
|
||||
foreach ( $items as $item ) {
|
||||
$items_normalized[] = array(
|
||||
'title' => $item->get_title(),
|
||||
'date' => $item->get_date( 'U' ),
|
||||
'link' => $item->get_permalink(),
|
||||
);
|
||||
}
|
||||
set_transient( 'themeisle_sdk_feed_items', $items_normalized, 48 * HOUR_IN_SECONDS );
|
||||
}
|
||||
$this->items = $items_normalized;
|
||||
}
|
||||
|
||||
/**
|
||||
* Either the current product is installed or not.
|
||||
*
|
||||
* @param array $val The current recommended product.
|
||||
*
|
||||
* @return bool Either we should exclude the plugin or not.
|
||||
*/
|
||||
public function remove_current_products( $val ) {
|
||||
if ( 'theme' === $val['type'] ) {
|
||||
$exist = wp_get_theme( $val['slug'] );
|
||||
|
||||
return ! $exist->exists();
|
||||
} else {
|
||||
$all_plugins = array_keys( get_plugins() );
|
||||
foreach ( $all_plugins as $slug ) {
|
||||
if ( strpos( $slug, $val['slug'] ) !== false ) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Contact the API and fetch the recommended plugins/themes
|
||||
*/
|
||||
public function recommend_plugin_or_theme() {
|
||||
$products = $this->get_product_from_api();
|
||||
if ( ! is_array( $products ) ) {
|
||||
$products = array();
|
||||
}
|
||||
$products = array_filter( $products, array( $this, 'remove_current_products' ) );
|
||||
$products = array_merge( $products );
|
||||
if ( count( $products ) > 1 ) {
|
||||
shuffle( $products );
|
||||
$products = array_slice( $products, 0, 1 );
|
||||
}
|
||||
$to_recommend = isset( $products[0] ) ? $products[0] : $products;
|
||||
|
||||
return $to_recommend;
|
||||
}
|
||||
|
||||
/**
|
||||
* Fetch products from the recomended section.
|
||||
*
|
||||
* @return array|mixed The list of products to use in recomended section.
|
||||
*/
|
||||
public function get_product_from_api() {
|
||||
if ( false === ( $products = get_transient( 'themeisle_sdk_products' ) ) ) { //phpcs:ignore Squiz.PHP.DisallowMultipleAssignments.FoundInControlStructure
|
||||
$products = array();
|
||||
$all_themes = $this->get_themes_from_wporg( 'themeisle' );
|
||||
$all_plugins = $this->get_plugins_from_wporg( 'themeisle' );
|
||||
static $allowed_products = [
|
||||
'hestia' => true,
|
||||
'neve' => true,
|
||||
'visualizer' => true,
|
||||
'feedzy-rss-feeds' => true,
|
||||
'wp-product-review' => true,
|
||||
'otter-blocks' => true,
|
||||
'themeisle-companion' => true,
|
||||
];
|
||||
foreach ( $all_themes as $theme ) {
|
||||
if ( $theme->active_installs < 4999 ) {
|
||||
continue;
|
||||
}
|
||||
if ( ! isset( $allowed_products[ $theme->slug ] ) ) {
|
||||
continue;
|
||||
}
|
||||
$products[] = array(
|
||||
'name' => $theme->name,
|
||||
'type' => 'theme',
|
||||
'slug' => $theme->slug,
|
||||
'installs' => $theme->active_installs,
|
||||
);
|
||||
}
|
||||
foreach ( $all_plugins as $plugin ) {
|
||||
if ( $plugin->active_installs < 4999 ) {
|
||||
continue;
|
||||
}
|
||||
if ( ! isset( $allowed_products[ $plugin->slug ] ) ) {
|
||||
continue;
|
||||
}
|
||||
$products[] = array(
|
||||
'name' => $plugin->name,
|
||||
'type' => 'plugin',
|
||||
'slug' => $plugin->slug,
|
||||
'installs' => $plugin->active_installs,
|
||||
);
|
||||
}
|
||||
set_transient( 'themeisle_sdk_products', $products, 6 * HOUR_IN_SECONDS );
|
||||
}
|
||||
|
||||
return $products;
|
||||
}
|
||||
|
||||
/**
|
||||
* Fetch themes from wporg api.
|
||||
*
|
||||
* @param string $author The author name.
|
||||
*
|
||||
* @return array The list of themes.
|
||||
*/
|
||||
public function get_themes_from_wporg( $author ) {
|
||||
$products = $this->safe_get(
|
||||
'https://api.wordpress.org/themes/info/1.1/?action=query_themes&request[author]=' . $author . '&request[per_page]=30&request[fields][active_installs]=true'
|
||||
);
|
||||
$products = json_decode( wp_remote_retrieve_body( $products ) );
|
||||
if ( is_object( $products ) ) {
|
||||
$products = isset( $products->themes ) ? $products->themes : array();
|
||||
} else {
|
||||
$products = array();
|
||||
}
|
||||
|
||||
return (array) $products;
|
||||
}
|
||||
|
||||
/**
|
||||
* Fetch plugin from wporg api.
|
||||
*
|
||||
* @param string $author The author slug.
|
||||
*
|
||||
* @return array The list of plugins for the selected author.
|
||||
*/
|
||||
public function get_plugins_from_wporg( $author ) {
|
||||
$products = $this->safe_get(
|
||||
'https://api.wordpress.org/plugins/info/1.1/?action=query_plugins&request[author]=' . $author . '&request[per_page]=40&request[fields][active_installs]=true'
|
||||
);
|
||||
$products = json_decode( wp_remote_retrieve_body( $products ) );
|
||||
if ( is_object( $products ) ) {
|
||||
$products = isset( $products->plugins ) ? $products->plugins : array();
|
||||
} else {
|
||||
$products = array();
|
||||
}
|
||||
|
||||
return (array) $products;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,263 @@
|
||||
<?php
|
||||
/**
|
||||
* File responsible for showing plugins inside the featured tab.
|
||||
*
|
||||
* This is used to display information about limited events, such as Black Friday.
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 3.3.0
|
||||
*/
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Loader;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Featured_Plugins module for the ThemeIsle SDK.
|
||||
*/
|
||||
class Featured_Plugins extends Abstract_Module {
|
||||
|
||||
/**
|
||||
* The transient key prefix.
|
||||
*
|
||||
* @var string $transient_key
|
||||
*/
|
||||
private $transient_key = 'themeisle_sdk_featured_plugins_';
|
||||
|
||||
/**
|
||||
* The current product instance.
|
||||
*
|
||||
* @var Product|null
|
||||
*/
|
||||
protected $product = null;
|
||||
|
||||
/**
|
||||
* Check if the module can be loaded.
|
||||
*
|
||||
* @param Product $product Product data.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( $product->is_wordpress_available() ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return ! apply_filters( 'themeisle_sdk_disable_featured_plugins', false );
|
||||
}
|
||||
|
||||
/**
|
||||
* Load the module for the selected product.
|
||||
*
|
||||
* @param Product $product Product data.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function load( $product ) {
|
||||
$this->product = $product;
|
||||
|
||||
if ( ! current_user_can( 'install_plugins' ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
// bail if we already registered a filter for the plugin API.
|
||||
if ( apply_filters( 'themeisle_sdk_plugin_api_filter_registered', false ) ) {
|
||||
return;
|
||||
}
|
||||
add_filter( 'themeisle_sdk_plugin_api_filter_registered', '__return_true' );
|
||||
|
||||
add_filter( 'plugins_api_result', [ $this, 'filter_plugin_api_results' ], 11, 3 );
|
||||
|
||||
// Enqueue inline JS only on plugin-install.php.
|
||||
add_action( 'admin_enqueue_scripts', [ $this, 'maybe_add_inline_js' ] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Enqueue inline JavaScript only on plugin-install.php.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function maybe_add_inline_js() {
|
||||
$screen = get_current_screen();
|
||||
if ( isset( $screen->base ) && 'plugin-install' === $screen->base ) {
|
||||
add_action(
|
||||
'admin_footer',
|
||||
function() {
|
||||
$text = esc_html( sprintf( Loader::$labels['promotions']['recommended'], $this->product->get_friendly_name() ) );
|
||||
|
||||
echo '<script>(function(){
|
||||
function onPluginCardFound(card) {
|
||||
var recommendedDiv = document.createElement("div");
|
||||
Object.assign(recommendedDiv.style, {
|
||||
display: "block",
|
||||
textAlign: "center",
|
||||
padding: "0 12px 12px",
|
||||
background: "#f6f7f7"
|
||||
});
|
||||
recommendedDiv.innerHTML = "' . esc_html( $text ) . '";
|
||||
card.appendChild(recommendedDiv);
|
||||
}
|
||||
|
||||
function checkAndRun() {
|
||||
var card = document.querySelector(".plugin-card-learning-management-system");
|
||||
if (card && !card.dataset.recommendedAdded) {
|
||||
onPluginCardFound(card);
|
||||
card.dataset.recommendedAdded = "true";
|
||||
}
|
||||
}
|
||||
|
||||
var observer = new MutationObserver(function(mutations) {
|
||||
checkAndRun();
|
||||
});
|
||||
|
||||
observer.observe(document.body, { childList: true, subtree: true });
|
||||
|
||||
// Initial check in case the card is already present.
|
||||
checkAndRun();
|
||||
})();</script>';
|
||||
}
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Filter the plugin API results to include the featured plugins.
|
||||
*
|
||||
* @param object $res The result object.
|
||||
* @param string $action The type of information being requested from the Plugin Install API.
|
||||
* @param object $args Plugin API arguments.
|
||||
*
|
||||
* @return object
|
||||
*/
|
||||
public function filter_plugin_api_results( $res, $action, $args ) {
|
||||
|
||||
if ( 'query_plugins' !== $action ) {
|
||||
return $res;
|
||||
}
|
||||
|
||||
if ( isset( $args->page ) && 1 === (int) $args->page && isset( $args->search ) && ! empty( $args->search ) ) {
|
||||
$res->plugins = $this->maybe_prepend_lms_plugin( $res->plugins, $args );
|
||||
return $res;
|
||||
}
|
||||
|
||||
if ( ! isset( $args->browse ) || $args->browse !== 'featured' ) {
|
||||
return $res;
|
||||
}
|
||||
|
||||
$featured = $this->query_plugins_by_author( $args );
|
||||
|
||||
$plugins = array_merge( $featured, (array) $res->plugins );
|
||||
$plugins = array_slice( $plugins, 0, $res->info['results'] );
|
||||
$res->plugins = $plugins;
|
||||
|
||||
return $res;
|
||||
}
|
||||
|
||||
/**
|
||||
* Prepend the LMS plugin if the search query matches LMS-related terms.
|
||||
*
|
||||
* @param array $plugins The plugins array.
|
||||
* @param object $args The plugin API arguments.
|
||||
* @return array
|
||||
*/
|
||||
private function maybe_prepend_lms_plugin( $plugins, $args ) {
|
||||
$search = isset( $args->search ) ? strtolower( $args->search ) : '';
|
||||
if (
|
||||
strpos( $search, 'lms' ) !== false ||
|
||||
strpos( $search, 'learn' ) !== false
|
||||
) {
|
||||
$filter_slugs = apply_filters( 'themeisle_sdk_masteriyo_filter_slugs', [ 'learning-management-system' ] );
|
||||
$masteriyo = $this->get_plugins_filtered_from_author( $args, $filter_slugs, 'masteriyo' );
|
||||
|
||||
if ( ! empty( $masteriyo ) ) {
|
||||
// Remove existing LMS plugin if present to avoid duplicates.
|
||||
$plugins = array_filter(
|
||||
$plugins,
|
||||
function( $plugin ) {
|
||||
return ( is_object( $plugin ) && isset( $plugin->slug ) && $plugin->slug !== 'learning-management-system' ) ||
|
||||
( is_array( $plugin ) && isset( $plugin['slug'] ) && $plugin['slug'] !== 'learning-management-system' );
|
||||
}
|
||||
);
|
||||
|
||||
$plugins = array_merge( $masteriyo, $plugins );
|
||||
}
|
||||
}
|
||||
return $plugins;
|
||||
}
|
||||
|
||||
/**
|
||||
* Query plugins by author.
|
||||
*
|
||||
* @param object $args The arguments for the query.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
private function query_plugins_by_author( $args ) {
|
||||
$featured = [];
|
||||
|
||||
$optimole_filter_slugs = apply_filters( 'themeisle_sdk_optimole_filter_slugs', [ 'optimole-wp' ] );
|
||||
$filtered_from_optimole = $this->get_plugins_filtered_from_author( $args, $optimole_filter_slugs, 'Optimole' );
|
||||
$featured = array_merge( $featured, $filtered_from_optimole );
|
||||
|
||||
$themeisle_filter_slugs = apply_filters( 'themeisle_sdk_themeisle_filter_slugs', [ 'otter-blocks', 'wp-cloudflare-page-cache' ] );
|
||||
$filtered_from_themeisle = $this->get_plugins_filtered_from_author( $args, $themeisle_filter_slugs );
|
||||
$featured = array_merge( $featured, $filtered_from_themeisle );
|
||||
|
||||
return $featured;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get plugins filtered from an author.
|
||||
*
|
||||
* @param object $args The arguments for the query.
|
||||
* @param array $filter_slugs The slugs to filter.
|
||||
* @param string $author The author to filter.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
protected function get_plugins_filtered_from_author( $args, $filter_slugs = [], $author = 'Themeisle' ) {
|
||||
|
||||
$cached = get_transient( $this->transient_key . $author );
|
||||
if ( $cached ) {
|
||||
return $cached;
|
||||
}
|
||||
|
||||
$new_args = [
|
||||
'page' => 1,
|
||||
'per_page' => 36,
|
||||
'locale' => get_user_locale(),
|
||||
'author' => $author,
|
||||
'wp_version' => isset( $args->wp_version ) ? $args->wp_version : get_bloginfo( 'version' ),
|
||||
];
|
||||
|
||||
$api = plugins_api( 'query_plugins', $new_args );
|
||||
if ( is_wp_error( $api ) ) {
|
||||
return [];
|
||||
}
|
||||
|
||||
$filtered = array_filter(
|
||||
$api->plugins,
|
||||
function( $plugin ) use ( $filter_slugs ) {
|
||||
$array_plugin = (array) $plugin;
|
||||
return in_array( $array_plugin['slug'], $filter_slugs );
|
||||
}
|
||||
);
|
||||
|
||||
set_transient( $this->transient_key . $author, $filtered, 12 * HOUR_IN_SECONDS );
|
||||
|
||||
return $filtered;
|
||||
}
|
||||
}
|
||||
274
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/Float_widget.php
vendored
Normal file
@@ -0,0 +1,274 @@
|
||||
<?php
|
||||
/**
|
||||
* The float widget model class for ThemeIsle SDK
|
||||
*
|
||||
* Here's how to hook it in your plugin:
|
||||
*
|
||||
* add_filter( <product_slug>_float_widget_metadata', 'add_float_widget_meta' );
|
||||
*
|
||||
* function add_float_widget_meta($data) {
|
||||
* return [
|
||||
* 'logo' => <logo url>,
|
||||
* 'nice_name' => <nice name>, // optional, will default to product name
|
||||
* 'primary_color' => <hex_color_value>, // optional
|
||||
* 'pages' => [ 'page-slugs' ], //pages where the float widget should be displayed
|
||||
* 'has_upgrade_menu' => <condition>,
|
||||
* 'upgrade_link' => <url>,
|
||||
* 'documentation_link' => <url>,
|
||||
* 'premium_support_link' => <url>, // optional, provide from pro version
|
||||
* 'feature_request_link' => <url>, // optional, provide from pro version
|
||||
* 'wizard_link' => <url>, // optional, provide if a user is available
|
||||
* ]
|
||||
* }
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2024, Bogdan Preda
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 3.2.42
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Loader;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Float widget module for ThemeIsle SDK.
|
||||
*/
|
||||
class Float_Widget extends Abstract_Module {
|
||||
/**
|
||||
* Float widget data.
|
||||
*
|
||||
* @var array $float_widget_data Float widget data, received from the filter.
|
||||
*
|
||||
* Shape of the $about_data property array:
|
||||
* [
|
||||
* 'logo' => <logo url>,
|
||||
* 'nice_name' => <nice name>, // optional, will default to product name
|
||||
* 'primary_color' => <hex_color_value>, // optional
|
||||
* 'pages' => [ 'page-slugs' ], //pages where the float widget should be displayed
|
||||
* 'has_upgrade_menu' => <condition>,
|
||||
* 'upgrade_link' => <url>,
|
||||
* 'documentation_link' => <url>,
|
||||
* 'premium_support_link' => <url>, // optional, provide from pro version
|
||||
* 'feature_request_link' => <url>, // optional, provide from pro version
|
||||
* 'wizard_link' => <url>, // optional, provide if a user is available
|
||||
* ]
|
||||
*/
|
||||
private $float_widget_data = array();
|
||||
|
||||
/**
|
||||
* Should we load this module.
|
||||
*
|
||||
* @param Product $product Product object.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$this->float_widget_data = apply_filters( $product->get_key() . '_float_widget_metadata', array() );
|
||||
|
||||
$can_load = ! empty( $this->float_widget_data );
|
||||
|
||||
$this->float_widget_data = array_merge(
|
||||
[
|
||||
'logo' => '',
|
||||
'primary_color' => '#2271b1', // Default color.
|
||||
'nice_name' => $product->get_name(),
|
||||
'documentation_link' => '',
|
||||
'premium_support_link' => '',
|
||||
'feature_request_link' => '',
|
||||
'wizard_link' => '',
|
||||
],
|
||||
$this->float_widget_data
|
||||
);
|
||||
|
||||
return $can_load;
|
||||
}
|
||||
|
||||
/**
|
||||
* Registers the hooks.
|
||||
*
|
||||
* @param Product $product Product to load.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
$this->product = $product;
|
||||
|
||||
add_action( 'in_admin_footer', [ $this, 'render_float_placeholder' ] );
|
||||
add_action( 'admin_enqueue_scripts', [ $this, 'enqueue_float_widget_script' ] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the allowed pages for the float widget.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
private function get_allowed_pages() {
|
||||
if ( ! isset( $this->float_widget_data['pages'] ) || ! is_array( $this->float_widget_data['pages'] ) ) {
|
||||
return [];
|
||||
}
|
||||
return $this->float_widget_data['pages'];
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if the current screen is allowed for the float widget.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
private function is_current_screen_allowed() {
|
||||
$current_screen = get_current_screen();
|
||||
|
||||
if ( ! isset( $current_screen->id ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( ! in_array( $current_screen->id, $this->get_allowed_pages(), true ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Renders the float widget placeholder.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function render_float_placeholder() {
|
||||
if ( ! $this->is_current_screen_allowed() ) {
|
||||
return;
|
||||
}
|
||||
|
||||
echo '<div id="ti-sdk-float-widget" style="--ti-float-primary-color: ' . esc_attr( $this->float_widget_data['primary_color'] ) . '"></div>';
|
||||
}
|
||||
|
||||
/**
|
||||
* Enqueue scripts & styles.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function enqueue_float_widget_script() {
|
||||
|
||||
if ( ! $this->is_current_screen_allowed() ) {
|
||||
return;
|
||||
}
|
||||
|
||||
global $themeisle_sdk_max_path;
|
||||
$handle = 'ti-sdk-float-' . $this->product->get_key();
|
||||
$asset_file = require $themeisle_sdk_max_path . '/assets/js/build/float_widget/float.asset.php';
|
||||
$deps = array_merge( $asset_file['dependencies'], [ 'updates' ] );
|
||||
|
||||
wp_register_script( $handle, $this->get_sdk_uri() . 'assets/js/build/float_widget/float.js', $deps, $asset_file['version'], true );
|
||||
wp_localize_script( $handle, 'tiSDKFloatData', $this->get_float_localization_data() );
|
||||
|
||||
wp_enqueue_script( $handle );
|
||||
wp_enqueue_style( $handle, $this->get_sdk_uri() . 'assets/js/build/float_widget/float.css', [ 'wp-components' ], $asset_file['version'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the float widget localization data.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
private function get_float_localization_data() {
|
||||
return [
|
||||
'logoUrl' => $this->float_widget_data['logo'],
|
||||
'primaryColor' => esc_attr( $this->float_widget_data['primary_color'] ),
|
||||
'strings' => [
|
||||
'toggleButton' => sprintf( Loader::$labels['float_widget']['button'], $this->float_widget_data['nice_name'] ),
|
||||
'panelGreet' => sprintf( Loader::$labels['float_widget']['panel']['greeting'], $this->float_widget_data['nice_name'] ),
|
||||
'panelTitle' => Loader::$labels['float_widget']['panel']['title'],
|
||||
'closeToggle' => Loader::$labels['float_widget']['panel']['close'],
|
||||
],
|
||||
'links' => $this->get_links(),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Generates the links for the float widget.
|
||||
*
|
||||
* For Free:
|
||||
* - Documentation (redirects to Themeisle doc page)
|
||||
* - Get Support (redirects to WP free support forum)
|
||||
* - Run Setup Wizard (this will trigger the setup wizard) if available
|
||||
* - Upgrade to Pro (redirects to Themeisle upgrade page)
|
||||
* - Rate Us (redirects to WP rating page)
|
||||
*
|
||||
* For Pro:
|
||||
* - Documentation (redirects to Themeisle doc page)
|
||||
* - Get Support (redirects to Themeisle support page to open a ticket)
|
||||
* - Run Setup Wizard (this will trigger the setup wizard) if available
|
||||
* - Feature Request (if available redirect to collect feedback requests)
|
||||
* - Rate Us (redirects to WP rating page)
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
private function get_links() {
|
||||
$links = [];
|
||||
|
||||
if ( ! empty( $this->float_widget_data['documentation_link'] ) ) {
|
||||
$links[] = [
|
||||
'icon' => 'dashicons-book-alt',
|
||||
'title' => Loader::$labels['float_widget']['links']['documentation'],
|
||||
'link' => $this->float_widget_data['documentation_link'],
|
||||
];
|
||||
}
|
||||
|
||||
$support_link = [
|
||||
'icon' => 'dashicons-format-status',
|
||||
'title' => Loader::$labels['float_widget']['links']['support'],
|
||||
'link' => 'https://wordpress.org/support/' . $this->product->get_type() . '/' . $this->product->get_slug() . '/',
|
||||
];
|
||||
if ( ! $this->float_widget_data['has_upgrade_menu'] && ! empty( $this->float_widget_data['premium_support_link'] ) ) {
|
||||
$support_link['link'] = $this->float_widget_data['premium_support_link'];
|
||||
}
|
||||
$links[] = $support_link;
|
||||
|
||||
if ( ! empty( $this->float_widget_data['wizard_link'] ) ) {
|
||||
$links[] = [
|
||||
'icon' => 'dashicons-admin-tools',
|
||||
'title' => Loader::$labels['float_widget']['links']['wizard'],
|
||||
'link' => $this->float_widget_data['wizard_link'],
|
||||
'internal' => true,
|
||||
];
|
||||
}
|
||||
|
||||
$pro = [
|
||||
'icon' => 'dashicons-superhero-alt',
|
||||
'title' => Loader::$labels['float_widget']['links']['upgrade'],
|
||||
'link' => $this->float_widget_data['upgrade_link'],
|
||||
];
|
||||
$featured_or_pro = $pro;
|
||||
if ( ! $this->float_widget_data['has_upgrade_menu'] ) {
|
||||
$featured_or_pro = []; // we remove the upgrade link
|
||||
$featured = $pro;
|
||||
$featured['title'] = Loader::$labels['float_widget']['links']['feature_request'];
|
||||
$featured['link'] = $this->float_widget_data['feature_request_link'];
|
||||
if ( ! empty( $featured['link'] ) ) {
|
||||
$featured_or_pro = $featured;
|
||||
}
|
||||
}
|
||||
|
||||
if ( ! empty( $featured_or_pro ) ) {
|
||||
$links[] = $featured_or_pro;
|
||||
}
|
||||
|
||||
$links[] = [
|
||||
'icon' => 'dashicons-star-filled',
|
||||
'title' => Loader::$labels['float_widget']['links']['rate'],
|
||||
'link' => 'https://wordpress.org/support/' . $this->product->get_type() . '/' . $this->product->get_slug() . '/reviews/#new-post',
|
||||
];
|
||||
|
||||
return $links;
|
||||
}
|
||||
}
|
||||
1193
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/Licenser.php
vendored
Normal file
280
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/Logger.php
vendored
Normal file
@@ -0,0 +1,280 @@
|
||||
<?php
|
||||
/**
|
||||
* The logger model class for ThemeIsle SDK
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 1.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Loader;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Logger module for ThemeIsle SDK.
|
||||
*/
|
||||
class Logger extends Abstract_Module {
|
||||
/**
|
||||
* Endpoint where to collect logs.
|
||||
*/
|
||||
const TRACKING_ENDPOINT = 'https://api.themeisle.com/tracking/log';
|
||||
|
||||
/**
|
||||
* Endpoint where to collect telemetry.
|
||||
*/
|
||||
const TELEMETRY_ENDPOINT = 'https://api.themeisle.com/tracking/events';
|
||||
|
||||
|
||||
/**
|
||||
* Check if we should load the module for this product.
|
||||
*
|
||||
* @param Product $product Product to load the module for.
|
||||
*
|
||||
* @return bool Should we load ?
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
return apply_filters( $product->get_slug() . '_sdk_enable_logger', true );
|
||||
}
|
||||
|
||||
/**
|
||||
* Load module logic.
|
||||
*
|
||||
* @param Product $product Product to load.
|
||||
*
|
||||
* @return Logger Module object.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
$this->product = $product;
|
||||
add_action( 'wp_loaded', array( $this, 'setup_actions' ) );
|
||||
add_action( 'admin_init', array( $this, 'setup_notification' ) );
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Setup notification on admin.
|
||||
*/
|
||||
public function setup_notification() {
|
||||
if ( $this->is_logger_active() ) {
|
||||
return;
|
||||
}
|
||||
add_filter( 'themeisle_sdk_registered_notifications', [ $this, 'add_notification' ] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Setup tracking actions.
|
||||
*/
|
||||
public function setup_actions() {
|
||||
if ( ! $this->is_logger_active() ) {
|
||||
return;
|
||||
}
|
||||
add_action(
|
||||
'admin_enqueue_scripts',
|
||||
function() {
|
||||
if ( ! apply_filters( 'themeisle_sdk_enable_telemetry', false ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$this->load_telemetry();
|
||||
},
|
||||
PHP_INT_MAX
|
||||
);
|
||||
|
||||
$action_key = $this->product->get_key() . '_log_activity';
|
||||
if ( ! wp_next_scheduled( $action_key ) ) {
|
||||
wp_schedule_single_event( time() + ( wp_rand( 1, 24 ) * 3600 ), $action_key );
|
||||
}
|
||||
add_action( $action_key, array( $this, 'send_log' ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if the logger is active.
|
||||
*
|
||||
* @return bool Is logger active?
|
||||
*/
|
||||
private function is_logger_active() {
|
||||
if ( apply_filters( 'themeisle_sdk_disable_telemetry', false ) ) {
|
||||
return false;
|
||||
}
|
||||
$default = 'no';
|
||||
|
||||
if ( ! $this->product->is_wordpress_available() ) {
|
||||
$default = 'yes';
|
||||
} else {
|
||||
$all_products = Loader::get_products();
|
||||
foreach ( $all_products as $product ) {
|
||||
if ( $product->requires_license() ) {
|
||||
$default = 'yes';
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
return ( get_option( $this->product->get_key() . '_logger_flag', $default ) === 'yes' );
|
||||
}
|
||||
/**
|
||||
* Add notification to queue.
|
||||
*
|
||||
* @param array $all_notifications Previous notification.
|
||||
*
|
||||
* @return array All notifications.
|
||||
*/
|
||||
public function add_notification( $all_notifications ) {
|
||||
|
||||
$message = apply_filters( $this->product->get_key() . '_logger_heading', Loader::$labels['logger']['notice'] );
|
||||
|
||||
$message = str_replace(
|
||||
array( '{product}' ),
|
||||
$this->product->get_friendly_name(),
|
||||
$message
|
||||
);
|
||||
$button_submit = apply_filters( $this->product->get_key() . '_logger_button_submit', Loader::$labels['logger']['cta_y'] );
|
||||
$button_cancel = apply_filters( $this->product->get_key() . '_logger_button_cancel', Loader::$labels['logger']['cta_n'] );
|
||||
|
||||
$all_notifications[] = [
|
||||
'id' => $this->product->get_key() . '_logger_flag',
|
||||
'message' => $message,
|
||||
'ctas' => [
|
||||
'confirm' => [
|
||||
'link' => '#',
|
||||
'text' => $button_submit,
|
||||
],
|
||||
'cancel' => [
|
||||
'link' => '#',
|
||||
'text' => $button_cancel,
|
||||
],
|
||||
],
|
||||
];
|
||||
|
||||
return $all_notifications;
|
||||
}
|
||||
|
||||
/**
|
||||
* Send the statistics to the api endpoint.
|
||||
*/
|
||||
public function send_log() {
|
||||
$environment = array();
|
||||
$theme = wp_get_theme();
|
||||
$environment['theme'] = array();
|
||||
$environment['theme']['name'] = $theme->get( 'Name' );
|
||||
$environment['theme']['author'] = $theme->get( 'Author' );
|
||||
$environment['theme']['parent'] = $theme->parent() !== false ? $theme->parent()->get( 'Name' ) : $theme->get( 'Name' );
|
||||
$environment['plugins'] = get_option( 'active_plugins' );
|
||||
global $wp_version;
|
||||
wp_remote_post(
|
||||
self::TRACKING_ENDPOINT,
|
||||
array(
|
||||
'method' => 'POST',
|
||||
'timeout' => 3,
|
||||
'redirection' => 5,
|
||||
'body' => array(
|
||||
'site' => get_site_url(),
|
||||
'slug' => $this->product->get_slug(),
|
||||
'version' => $this->product->get_version(),
|
||||
'wp_version' => $wp_version,
|
||||
'install_time' => $this->product->get_install_time(),
|
||||
'locale' => get_locale(),
|
||||
'data' => apply_filters( $this->product->get_key() . '_logger_data', array() ),
|
||||
'environment' => $environment,
|
||||
'license' => apply_filters( $this->product->get_key() . '_license_status', '' ),
|
||||
),
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Load telemetry.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function load_telemetry() {
|
||||
// See which products have telemetry enabled.
|
||||
try {
|
||||
$products_with_telemetry = array();
|
||||
$all_products = Loader::get_products();
|
||||
$all_products[ $this->product->get_slug() ] = $this->product; // Add current product to the list of products to check for telemetry.
|
||||
|
||||
// Register telemetry params for eligible products.
|
||||
foreach ( $all_products as $product_slug => $product ) {
|
||||
|
||||
// Ignore PRO products.
|
||||
if ( false !== strstr( $product_slug, 'pro' ) ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$pro_slug = $product->get_pro_slug();
|
||||
$logger_key = $product->get_key() . '_logger_flag';
|
||||
|
||||
// If the product is not available in the WordPress store, or it's PRO version is installed, activate the logger if it was not initialized -- Pro users are opted in by default.
|
||||
if ( ! $product->is_wordpress_available() || ( ! empty( $pro_slug ) && isset( $all_products[ $pro_slug ] ) ) ) {
|
||||
$logger_flag = get_option( $logger_key );
|
||||
|
||||
if ( false === $logger_flag ) {
|
||||
update_option( $logger_key, 'yes' );
|
||||
}
|
||||
}
|
||||
|
||||
if ( 'yes' === get_option( $product->get_key() . '_logger_flag', 'no' ) ) {
|
||||
|
||||
$main_slug = explode( '-', $product_slug );
|
||||
$main_slug = $main_slug[0];
|
||||
$track_hash = Licenser::create_license_hash( str_replace( '-', '_', ! empty( $pro_slug ) ? $pro_slug : $product_slug ) );
|
||||
|
||||
// Check if product was already tracked.
|
||||
$active_telemetry = false;
|
||||
foreach ( $products_with_telemetry as $product_with_telemetry ) {
|
||||
if ( $product_with_telemetry['slug'] === $main_slug ) {
|
||||
$active_telemetry = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if ( $active_telemetry ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$products_with_telemetry[] = array(
|
||||
'slug' => $main_slug,
|
||||
'trackHash' => $track_hash ? $track_hash : 'free',
|
||||
'consent' => true,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
$products_with_telemetry = apply_filters( 'themeisle_sdk_telemetry_products', $products_with_telemetry );
|
||||
|
||||
if ( 0 === count( $products_with_telemetry ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$tracking_handler = apply_filters( 'themeisle_sdk_dependency_script_handler', 'tracking' );
|
||||
if ( ! empty( $tracking_handler ) ) {
|
||||
do_action( 'themeisle_sdk_dependency_enqueue_script', 'tracking' );
|
||||
wp_localize_script(
|
||||
$tracking_handler,
|
||||
'tiTelemetry',
|
||||
array(
|
||||
'products' => $products_with_telemetry,
|
||||
'endpoint' => self::TELEMETRY_ENDPOINT,
|
||||
)
|
||||
);
|
||||
}
|
||||
} catch ( \Exception $e ) {
|
||||
if ( defined( 'WP_DEBUG' ) && WP_DEBUG && defined( 'WP_DEBUG_LOG' ) && WP_DEBUG_LOG ) {
|
||||
error_log( $e->getMessage() ); // phpcs:ignore
|
||||
}
|
||||
} finally {
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
515
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/Notification.php
vendored
Normal file
@@ -0,0 +1,515 @@
|
||||
<?php
|
||||
/**
|
||||
* The notification model class for ThemeIsle SDK
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 1.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Notification module for ThemeIsle SDK.
|
||||
*/
|
||||
class Notification extends Abstract_Module {
|
||||
/**
|
||||
* Show notifications only after the user has the product installed after this amount of time, in hours.
|
||||
*/
|
||||
const MIN_INSTALL_TIME = 100;
|
||||
/**
|
||||
* How much time should we show the notification, in days.
|
||||
*/
|
||||
const MAX_TIME_TO_LIVE = 7;
|
||||
|
||||
/**
|
||||
* Number of days between notifications.
|
||||
*/
|
||||
const TIME_BETWEEN_NOTIFICATIONS = 5;
|
||||
|
||||
/**
|
||||
* Holds a possible notification list.
|
||||
*
|
||||
* @var array Notifications list.
|
||||
*/
|
||||
private static $notifications = [];
|
||||
|
||||
/**
|
||||
* Show notification data.
|
||||
*/
|
||||
public static function show_notification() {
|
||||
|
||||
$current_notification = self::get_last_notification();
|
||||
|
||||
$notification_details = [];
|
||||
// Check if the saved notification is still present among the possible ones.
|
||||
if ( ! empty( $current_notification ) ) {
|
||||
$notification_details = self::get_notification_details( $current_notification );
|
||||
if ( empty( $notification_details ) ) {
|
||||
$current_notification = [];
|
||||
}
|
||||
}
|
||||
// Check if the notificatin is expired.
|
||||
if ( ! empty( $current_notification ) && self::is_notification_expired( $current_notification ) ) {
|
||||
update_option( $current_notification['id'], 'no' );
|
||||
self::set_last_active_notification_timestamp();
|
||||
$current_notification = [];
|
||||
}
|
||||
// If we don't have any saved notification, get a new one.
|
||||
if ( empty( $current_notification ) ) {
|
||||
$notification_details = self::get_random_notification();
|
||||
if ( empty( $notification_details ) ) {
|
||||
return;
|
||||
}
|
||||
self::set_active_notification(
|
||||
[
|
||||
'id' => $notification_details['id'],
|
||||
'display_at' => time(),
|
||||
]
|
||||
);
|
||||
}
|
||||
if ( empty( $notification_details ) ) {
|
||||
return;
|
||||
}
|
||||
$notification_html = self::get_notification_html( $notification_details );
|
||||
do_action( $notification_details['id'] . '_before_render' );
|
||||
|
||||
echo $notification_html; //phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped, already escaped internally.
|
||||
|
||||
do_action( $notification_details['id'] . '_after_render' );
|
||||
self::render_snippets();
|
||||
}
|
||||
|
||||
/**
|
||||
* Get last notification details.
|
||||
*
|
||||
* @return array Last notification details.
|
||||
*/
|
||||
private static function get_last_notification() {
|
||||
$notification = self::get_notifications_metadata();
|
||||
|
||||
return isset( $notification['last_notification'] ) ? $notification['last_notification'] : [];
|
||||
}
|
||||
|
||||
/**
|
||||
* Get notification center details.
|
||||
*
|
||||
* @return array Notification center details.
|
||||
*/
|
||||
private static function get_notifications_metadata() {
|
||||
|
||||
$data = get_option(
|
||||
'themeisle_sdk_notifications',
|
||||
[
|
||||
'last_notification' => [],
|
||||
'last_notification_active' => 0,
|
||||
]
|
||||
);
|
||||
|
||||
return $data;
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if the notification is still possible.
|
||||
*
|
||||
* @param array $notification Notification to check.
|
||||
*
|
||||
* @return array Either is still active or not.
|
||||
*/
|
||||
private static function get_notification_details( $notification ) {
|
||||
$notifications = array_filter(
|
||||
self::$notifications,
|
||||
function ( $value ) use ( $notification ) {
|
||||
if ( isset( $value['id'] ) && isset( $notification['id'] ) && $value['id'] === $notification['id'] ) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
);
|
||||
|
||||
return ! empty( $notifications ) ? reset( $notifications ) : [];
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if the notification is expired.
|
||||
*
|
||||
* @param array $notification Notification to check.
|
||||
*
|
||||
* @return bool Either the notification is due.
|
||||
*/
|
||||
private static function is_notification_expired( $notification ) {
|
||||
if ( ! isset( $notification['display_at'] ) ) {
|
||||
return true;
|
||||
}
|
||||
|
||||
$notifications = array_filter(
|
||||
self::$notifications,
|
||||
function ( $value ) use ( $notification ) {
|
||||
if ( isset( $value['id'] ) && isset( $notification['id'] ) && $value['id'] === $notification['id'] ) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
);
|
||||
|
||||
if ( empty( $notifications ) ) {
|
||||
return true;
|
||||
}
|
||||
$notification_definition = reset( $notifications );
|
||||
|
||||
$when_to_expire = isset( $notification_definition['expires_at'] )
|
||||
? $notification_definition['expires_at'] :
|
||||
( isset( $notification_definition['expires'] )
|
||||
? ( $notification['display_at'] + $notification_definition['expires'] ) :
|
||||
( $notification['display_at'] + self::MAX_TIME_TO_LIVE * DAY_IN_SECONDS )
|
||||
);
|
||||
|
||||
return ( $when_to_expire - time() ) < 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set last notification details.
|
||||
*/
|
||||
private static function set_last_active_notification_timestamp() {
|
||||
$metadata = self::get_notifications_metadata();
|
||||
$metadata['last_notification_active'] = time();
|
||||
update_option( 'themeisle_sdk_notifications', $metadata );
|
||||
}
|
||||
|
||||
/**
|
||||
* Return notification to show.
|
||||
*
|
||||
* @return array Notification data.
|
||||
*/
|
||||
public static function get_random_notification() {
|
||||
if ( ( time() - self::get_last_active_notification_timestamp() ) < self::TIME_BETWEEN_NOTIFICATIONS * DAY_IN_SECONDS ) {
|
||||
return [];
|
||||
}
|
||||
|
||||
$notifications = self::$notifications;
|
||||
$notifications = array_filter(
|
||||
$notifications,
|
||||
function ( $value ) {
|
||||
if ( isset( $value['sticky'] ) && true === $value['sticky'] ) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
);
|
||||
// No priority notifications, use all.
|
||||
if ( empty( $notifications ) ) {
|
||||
$notifications = self::$notifications;
|
||||
}
|
||||
if ( empty( $notifications ) ) {
|
||||
return [];
|
||||
}
|
||||
$notifications = array_values( $notifications );
|
||||
|
||||
return $notifications[ array_rand( $notifications, 1 ) ];
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Get last notification details.
|
||||
*
|
||||
* @return int Last notification details.
|
||||
*/
|
||||
private static function get_last_active_notification_timestamp() {
|
||||
$notification = self::get_notifications_metadata();
|
||||
|
||||
return isset( $notification['last_notification_active'] ) ? $notification['last_notification_active'] : 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get last notification details.
|
||||
*
|
||||
* @param array $notification Notification data.
|
||||
*/
|
||||
private static function set_active_notification( $notification ) {
|
||||
$metadata = self::get_notifications_metadata();
|
||||
$metadata['last_notification'] = $notification;
|
||||
update_option( 'themeisle_sdk_notifications', $metadata );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get notification html.
|
||||
*
|
||||
* @param array $notification_details Notification details.
|
||||
*
|
||||
* @return string Html for notice.
|
||||
*/
|
||||
public static function get_notification_html( $notification_details ) {
|
||||
$default = [
|
||||
'id' => '',
|
||||
'heading' => '',
|
||||
'img_src' => '',
|
||||
'message' => '',
|
||||
'ctas' => [
|
||||
'confirm' => [
|
||||
'link' => '#',
|
||||
'text' => '',
|
||||
],
|
||||
'cancel' => [
|
||||
'link' => '#',
|
||||
'text' => '',
|
||||
],
|
||||
],
|
||||
'type' => 'success',
|
||||
];
|
||||
$notification_details = wp_parse_args( $notification_details, $default );
|
||||
global $pagenow;
|
||||
$type = in_array( $notification_details['type'], [ 'success', 'info', 'warning', 'error' ], true ) ? $notification_details['type'] : 'success';
|
||||
$notification_details['ctas']['cancel']['link'] = wp_nonce_url( add_query_arg( [ 'nid' => $notification_details['id'] ], admin_url( $pagenow ) ), $notification_details['id'], 'tsdk_dismiss_nonce' );
|
||||
$notification_html = '<div class="notice notice-' . $type . ' is-dismissible themeisle-sdk-notice" data-notification-id="' . esc_attr( $notification_details['id'] ) . '" id="' . esc_attr( $notification_details['id'] ) . '-notification"> <div class="themeisle-sdk-notification-box">';
|
||||
|
||||
if ( ! empty( $notification_details['heading'] ) ) {
|
||||
$notification_html .= sprintf( '<h4>%s</h4>', wp_kses_post( $notification_details['heading'] ) );
|
||||
}
|
||||
if ( ! empty( $notification_details['img_src'] ) ) {
|
||||
$notification_html .= '<div class="wrap-flex">';
|
||||
$notification_html .= sprintf( '<img src="%s" alt="%s" />', esc_attr( $notification_details['img_src'] ), esc_attr( $notification_details['heading'] ) );
|
||||
}
|
||||
if ( ! empty( $notification_details['message'] ) ) {
|
||||
$notification_html .= wp_kses_post( $notification_details['message'] );
|
||||
if ( ! empty( $notification_details['img_src'] ) ) {
|
||||
$notification_html .= '</div>';
|
||||
}
|
||||
}
|
||||
$notification_html .= '<div class="actions">';
|
||||
|
||||
if ( ! empty( $notification_details['ctas']['confirm']['text'] ) ) {
|
||||
$notification_html .= sprintf(
|
||||
'<a href="%s" target="_blank" class=" button button-primary %s" data-confirm="yes" >%s</a>',
|
||||
esc_url( $notification_details['ctas']['confirm']['link'] ),
|
||||
esc_attr( $notification_details['id'] . '_confirm' ),
|
||||
wp_kses_post( $notification_details['ctas']['confirm']['text'] )
|
||||
);
|
||||
}
|
||||
|
||||
if ( ! empty( $notification_details['ctas']['cancel']['text'] ) ) {
|
||||
$notification_html .= sprintf(
|
||||
'<a href="%s" class=" button %s" data-confirm="no">%s</a>',
|
||||
esc_url( $notification_details['ctas']['cancel']['link'] ),
|
||||
esc_attr( $notification_details['id'] ) . '_cancel',
|
||||
wp_kses_post( $notification_details['ctas']['cancel']['text'] )
|
||||
);
|
||||
}
|
||||
|
||||
$notification_html .= '</div>';
|
||||
$notification_html .= ' </div>';
|
||||
$notification_html .= ' </div>';
|
||||
|
||||
return $notification_html;
|
||||
}
|
||||
|
||||
/**
|
||||
* Adds js snippet for hiding the notice.
|
||||
*/
|
||||
public static function render_snippets() {
|
||||
|
||||
?>
|
||||
<style type="text/css">
|
||||
.themeisle-sdk-notification-box {
|
||||
padding: 3px;
|
||||
}
|
||||
|
||||
.themeisle-sdk-notification-box .wrap-flex {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: start;
|
||||
gap: 12px;
|
||||
}
|
||||
|
||||
.themeisle-sdk-notification-box .wrap-flex img {
|
||||
width: 42px;
|
||||
object-fit: cover;
|
||||
}
|
||||
|
||||
.themeisle-sdk-notification-box .actions {
|
||||
margin-top: 6px;
|
||||
margin-bottom: 4px;
|
||||
}
|
||||
|
||||
.themeisle-sdk-notification-box .button {
|
||||
margin-right: 5px;
|
||||
}
|
||||
</style>
|
||||
<script type="text/javascript">
|
||||
(function ($) {
|
||||
$(document).ready(function () {
|
||||
$('#wpbody-content').on('click', ".themeisle-sdk-notice a.button, .themeisle-sdk-notice .notice-dismiss", function (e) {
|
||||
|
||||
var container = $('.themeisle-sdk-notice');
|
||||
var link = $(this);
|
||||
var notification_id = container.attr('data-notification-id');
|
||||
var confirm = link.attr('data-confirm');
|
||||
if (typeof confirm === "undefined") {
|
||||
confirm = 'no';
|
||||
}
|
||||
$.post(
|
||||
ajaxurl,
|
||||
{
|
||||
'nonce': '<?php echo esc_attr( wp_create_nonce( (string) __CLASS__ ) ); ?>',
|
||||
'action': 'themeisle_sdk_dismiss_notice',
|
||||
'id': notification_id,
|
||||
'confirm': confirm,
|
||||
},
|
||||
).fail(function() {
|
||||
location.href = encodeURI(link.attr('href'));
|
||||
});
|
||||
if (confirm === 'yes') {
|
||||
$(this).trigger('themeisle-sdk:confirmed');
|
||||
} else {
|
||||
$(this).trigger('themeisle-sdk:canceled');
|
||||
}
|
||||
container.hide();
|
||||
if (confirm === 'no' || link.attr('href') === '#') {
|
||||
return false;
|
||||
}
|
||||
});
|
||||
});
|
||||
})(jQuery);
|
||||
</script>
|
||||
<?php
|
||||
}
|
||||
|
||||
/**
|
||||
* Dismiss the notification.
|
||||
*/
|
||||
public static function dismiss() {
|
||||
check_ajax_referer( (string) __CLASS__, 'nonce' );
|
||||
|
||||
$id = isset( $_POST['id'] ) ? sanitize_text_field( $_POST['id'] ) : '';
|
||||
$confirm = isset( $_POST['confirm'] ) ? sanitize_text_field( $_POST['confirm'] ) : 'no';
|
||||
|
||||
if ( empty( $id ) ) {
|
||||
wp_send_json( [] );
|
||||
}
|
||||
self::setup_notifications();
|
||||
$ids = wp_list_pluck( self::$notifications, 'id' );
|
||||
if ( ! in_array( $id, $ids, true ) ) {
|
||||
wp_send_json( [] );
|
||||
}
|
||||
self::set_last_active_notification_timestamp();
|
||||
update_option( $id, $confirm );
|
||||
do_action( $id . '_process_confirm', $confirm );
|
||||
wp_send_json( [] );
|
||||
}
|
||||
/**
|
||||
* Dismiss the notification.
|
||||
*/
|
||||
public static function dismiss_get() {
|
||||
$is_nonce_dismiss = sanitize_text_field( isset( $_GET['tsdk_dismiss_nonce'] ) ? $_GET['tsdk_dismiss_nonce'] : '' );
|
||||
if ( strlen( $is_nonce_dismiss ) < 5 ) {
|
||||
return;
|
||||
}
|
||||
$id = sanitize_text_field( isset( $_GET['nid'] ) ? $_GET['nid'] : '' );
|
||||
if ( empty( $id ) ) {
|
||||
return;
|
||||
}
|
||||
$nonce = wp_verify_nonce( sanitize_text_field( $_GET['tsdk_dismiss_nonce'] ), $id );
|
||||
if ( $nonce !== 1 ) {
|
||||
return;
|
||||
}
|
||||
$ids = wp_list_pluck( self::$notifications, 'id' );
|
||||
if ( ! in_array( $id, $ids, true ) ) {
|
||||
return;
|
||||
}
|
||||
$confirm = 'no';
|
||||
self::set_last_active_notification_timestamp();
|
||||
update_option( $id, $confirm );
|
||||
do_action( $id . '_process_confirm', $confirm );
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if we should load the notification module.
|
||||
*
|
||||
* @param Product $product Product to check.
|
||||
*
|
||||
* @return bool Should we load this?
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
if ( ! current_user_can( 'manage_options' ) ) {
|
||||
return false;
|
||||
}
|
||||
if ( ( time() - $product->get_install_time() ) < ( self::MIN_INSTALL_TIME * HOUR_IN_SECONDS ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Setup notifications queue.
|
||||
*/
|
||||
public static function setup_notifications() {
|
||||
$notifications = apply_filters( 'themeisle_sdk_registered_notifications', [] );
|
||||
$notifications = array_filter(
|
||||
$notifications,
|
||||
function ( $value ) {
|
||||
if ( ! isset( $value['id'] ) ) {
|
||||
return false;
|
||||
}
|
||||
if ( get_option( $value['id'], '' ) !== '' ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return apply_filters( $value['id'] . '_should_show', true );
|
||||
}
|
||||
);
|
||||
self::$notifications = $notifications;
|
||||
}
|
||||
/**
|
||||
* Load the module logic.
|
||||
*
|
||||
* @param Product $product Product to load the module for.
|
||||
*
|
||||
* @return Notification Module instance.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
if ( apply_filters( 'themeisle_sdk_hide_notifications', false ) ) {
|
||||
return;
|
||||
}
|
||||
$this->product = $product;
|
||||
|
||||
$notifications = apply_filters( 'themeisle_sdk_registered_notifications', [] );
|
||||
$notifications = array_filter(
|
||||
$notifications,
|
||||
function ( $value ) {
|
||||
if ( ! isset( $value['id'] ) ) {
|
||||
return false;
|
||||
}
|
||||
if ( get_option( $value['id'], '' ) !== '' ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return apply_filters( $value['id'] . '_should_show', true );
|
||||
}
|
||||
);
|
||||
self::$notifications = $notifications;
|
||||
add_action( 'admin_notices', array( __CLASS__, 'show_notification' ) );
|
||||
add_action( 'wp_ajax_themeisle_sdk_dismiss_notice', array( __CLASS__, 'dismiss' ) );
|
||||
add_action( 'admin_head', array( __CLASS__, 'dismiss_get' ) );
|
||||
add_action( 'admin_head', array( __CLASS__, 'setup_notifications' ) );
|
||||
|
||||
return $this;
|
||||
}
|
||||
}
|
||||
1472
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/Promotions.php
vendored
Normal file
@@ -0,0 +1,334 @@
|
||||
<?php
|
||||
/**
|
||||
* The class that exposes hooks for recommend.
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Rollback
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 1.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
// Exit if accessed directly.
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Expose endpoints for ThemeIsle SDK.
|
||||
*/
|
||||
class Recommendation extends Abstract_Module {
|
||||
|
||||
|
||||
/**
|
||||
* Load module logic.
|
||||
*
|
||||
* @param Product $product Product to load.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
$this->product = $product;
|
||||
$this->setup_hooks();
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Setup endpoints.
|
||||
*/
|
||||
private function setup_hooks() {
|
||||
add_action( $this->product->get_key() . '_recommend_products', array( $this, 'render_products_box' ), 10, 4 );
|
||||
add_action( 'admin_head', array( $this, 'enqueue' ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if we should load the module for this product.
|
||||
*
|
||||
* @param Product $product Product data.
|
||||
*
|
||||
* @return bool Should we load the module?
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Render products box content.
|
||||
*
|
||||
* @param array $plugins_list - list of useful plugins (in slug => nicename format).
|
||||
* @param array $themes_list - list of useful themes (in slug => nicename format).
|
||||
* @param array $strings - list of translated strings.
|
||||
* @param array $preferences - list of preferences.
|
||||
*/
|
||||
public function render_products_box( $plugins_list, $themes_list, $strings, $preferences = array() ) {
|
||||
|
||||
if ( empty( $plugins_list ) && empty( $themes_list ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( ! empty( $plugins_list ) && ! current_user_can( 'install_plugins' ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( ! empty( $themes_list ) && ! current_user_can( 'install_themes' ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
add_thickbox();
|
||||
|
||||
if ( ! empty( $themes_list ) ) {
|
||||
$list = $this->get_themes( $themes_list, $preferences );
|
||||
|
||||
if ( has_action( $this->product->get_key() . '_recommend_products_theme_template' ) ) {
|
||||
do_action( $this->product->get_key() . '_recommend_products_theme_template', $list, $strings, $preferences );
|
||||
} else {
|
||||
echo '<div class="recommend-product">';
|
||||
|
||||
foreach ( $list as $theme ) {
|
||||
echo '<div class="plugin_box">';
|
||||
echo ' <img class="theme-banner" src="' . esc_url( $theme->screenshot_url ) . '">';
|
||||
echo ' <div class="title-action-wrapper">';
|
||||
echo ' <span class="plugin-name">' . esc_html( $theme->custom_name ) . '</span>';
|
||||
if ( ! isset( $preferences['description'] ) || ( isset( $preferences['description'] ) && $preferences['description'] ) ) {
|
||||
echo '<span class="plugin-desc">' . esc_html( substr( $theme->description, 0, strpos( $theme->description, '.' ) ) ) . '.</span>';
|
||||
}
|
||||
echo ' </div>';
|
||||
echo '<div class="plugin-box-footer">';
|
||||
echo ' <div class="button-wrap">';
|
||||
echo ' <a class="button button-primary " href="' . esc_url( $theme->custom_url ) . '"><span class="dashicons dashicons-external"></span>' . esc_html( $strings['install'] ) . '</a>';
|
||||
echo ' </div>';
|
||||
echo ' </div>';
|
||||
echo '</div>';
|
||||
}
|
||||
|
||||
echo '</div>';
|
||||
}
|
||||
}
|
||||
if ( ! empty( $plugins_list ) ) {
|
||||
$list = $this->get_plugins( $plugins_list, $preferences );
|
||||
|
||||
if ( has_action( $this->product->get_key() . '_recommend_products_plugin_template' ) ) {
|
||||
do_action( $this->product->get_key() . '_recommend_products_plugin_template', $list, $strings, $preferences );
|
||||
} else {
|
||||
echo '<div class="recommend-product">';
|
||||
|
||||
foreach ( $list as $current_plugin ) {
|
||||
echo '<div class="plugin_box">';
|
||||
echo ' <img class="plugin-banner" src="' . esc_url( $current_plugin->custom_image ) . '">';
|
||||
echo ' <div class="title-action-wrapper">';
|
||||
echo ' <span class="plugin-name">' . esc_html( $current_plugin->custom_name ) . '</span>';
|
||||
if ( ! isset( $preferences['description'] ) || ( isset( $preferences['description'] ) && $preferences['description'] ) ) {
|
||||
echo '<span class="plugin-desc">' . esc_html( substr( $current_plugin->short_description, 0, strpos( $current_plugin->short_description, '.' ) ) ) . '. </span>';
|
||||
}
|
||||
echo ' </div>';
|
||||
echo ' <div class="plugin-box-footer">';
|
||||
echo ' <a class="button button-primary thickbox open-plugin-details-modal" href="' . esc_url( $current_plugin->custom_url ) . '"><span class="dashicons dashicons-external"></span>' . esc_html( $strings['install'] ) . '</a>';
|
||||
echo ' </div>';
|
||||
echo '</div>';
|
||||
}
|
||||
|
||||
echo '</div>';
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Collect all the information for the themes list.
|
||||
*
|
||||
* @param array $themes_list - list of useful themes (in slug => nicename format).
|
||||
* @param array $preferences - list of preferences.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
private function get_themes( $themes_list, $preferences ) {
|
||||
$list = array();
|
||||
foreach ( $themes_list as $slug => $nicename ) {
|
||||
$theme = $this->call_theme_api( $slug );
|
||||
if ( ! $theme ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$url = add_query_arg(
|
||||
array(
|
||||
'theme' => $theme->slug,
|
||||
),
|
||||
network_admin_url( 'theme-install.php' )
|
||||
);
|
||||
|
||||
$name = empty( $nicename ) ? $theme->name : $nicename;
|
||||
|
||||
$theme->custom_url = $url;
|
||||
$theme->custom_name = $name;
|
||||
|
||||
$list[] = $theme;
|
||||
}
|
||||
|
||||
return $list;
|
||||
}
|
||||
|
||||
/**
|
||||
* Call theme api
|
||||
*
|
||||
* @param string $slug theme slug.
|
||||
*
|
||||
* @return array|mixed|object
|
||||
*/
|
||||
private function call_theme_api( $slug ) {
|
||||
$theme = get_transient( 'ti_theme_info_' . $slug );
|
||||
|
||||
if ( false !== $theme ) {
|
||||
return $theme;
|
||||
}
|
||||
|
||||
$products = $this->safe_get(
|
||||
'https://api.wordpress.org/themes/info/1.1/?action=query_themes&request[theme]=' . $slug . '&request[per_page]=1'
|
||||
);
|
||||
$products = json_decode( wp_remote_retrieve_body( $products ) );
|
||||
if ( is_object( $products ) ) {
|
||||
$theme = $products->themes[0];
|
||||
set_transient( 'ti_theme_info_' . $slug, $theme, 6 * HOUR_IN_SECONDS );
|
||||
}
|
||||
|
||||
return $theme;
|
||||
}
|
||||
|
||||
/**
|
||||
* Collect all the information for the plugins list.
|
||||
*
|
||||
* @param array $plugins_list - list of useful plugins (in slug => nicename format).
|
||||
* @param array $preferences - list of preferences.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
private function get_plugins( $plugins_list, $preferences ) {
|
||||
$list = array();
|
||||
foreach ( $plugins_list as $plugin => $nicename ) {
|
||||
$current_plugin = $this->call_plugin_api( $plugin );
|
||||
|
||||
$name = empty( $nicename ) ? $current_plugin->name : $nicename;
|
||||
|
||||
$image = $current_plugin->banners['low'];
|
||||
if ( isset( $preferences['image'] ) && 'icon' === $preferences['image'] ) {
|
||||
$image = $current_plugin->icons['1x'];
|
||||
}
|
||||
|
||||
$url = add_query_arg(
|
||||
array(
|
||||
'tab' => 'plugin-information',
|
||||
'plugin' => $current_plugin->slug,
|
||||
'TB_iframe' => true,
|
||||
'width' => 800,
|
||||
'height' => 800,
|
||||
),
|
||||
network_admin_url( 'plugin-install.php' )
|
||||
);
|
||||
|
||||
$current_plugin->custom_url = $url;
|
||||
$current_plugin->custom_name = $name;
|
||||
$current_plugin->custom_image = $image;
|
||||
|
||||
$list[] = $current_plugin;
|
||||
}
|
||||
|
||||
return $list;
|
||||
}
|
||||
|
||||
/**
|
||||
* Load css and scripts for the plugin recommend page.
|
||||
*/
|
||||
public function enqueue() {
|
||||
$screen = get_current_screen();
|
||||
|
||||
if ( ! isset( $screen->id ) ) {
|
||||
return;
|
||||
}
|
||||
if ( false === apply_filters( $this->product->get_key() . '_enqueue_recommend', false, $screen->id ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
?>
|
||||
<style type="text/css">
|
||||
.recommend-product {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
flex-wrap: wrap;
|
||||
}
|
||||
|
||||
.recommend-product .theme-banner {
|
||||
width: 200px;
|
||||
margin: auto;
|
||||
}
|
||||
|
||||
.recommend-product .plugin-banner {
|
||||
width: 100px;
|
||||
margin: auto;
|
||||
}
|
||||
|
||||
.recommend-product .plugin_box .button span {
|
||||
|
||||
margin-top: 2px;
|
||||
margin-right: 7px;
|
||||
}
|
||||
|
||||
.recommend-product .plugin_box .button {
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.recommend-product .plugin_box {
|
||||
margin-bottom: 20px;
|
||||
padding-top: 5px;
|
||||
display: flex;
|
||||
box-shadow: 0px 0px 10px -5px rgba(0, 0, 0, 0.55);
|
||||
background: #fff;
|
||||
border-radius: 5px;
|
||||
flex-direction: column;
|
||||
justify-content: flex-start;
|
||||
width: 95%;
|
||||
}
|
||||
|
||||
.recommend-product .title-action-wrapper {
|
||||
padding: 15px 20px 5px 20px;
|
||||
}
|
||||
|
||||
.recommend-product .plugin-name {
|
||||
font-size: 18px;
|
||||
display: block;
|
||||
white-space: nowrap;
|
||||
text-overflow: ellipsis;
|
||||
margin-bottom: 10px;
|
||||
overflow: hidden;
|
||||
line-height: normal;
|
||||
}
|
||||
|
||||
|
||||
.recommend-product .plugin-desc {
|
||||
display: block;
|
||||
margin-bottom: 10px;
|
||||
font-size: 13px;
|
||||
color: #777;
|
||||
line-height: 1.6;
|
||||
}
|
||||
|
||||
.recommend-product .button-wrap > div {
|
||||
padding: 0;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.plugin-box-footer {
|
||||
display: flex;
|
||||
justify-content: space-around;
|
||||
vertical-align: middle;
|
||||
align-items: center;
|
||||
padding: 0px 10px 5px;
|
||||
flex: 1;
|
||||
margin-top: auto;
|
||||
}
|
||||
</style>
|
||||
<?php
|
||||
}
|
||||
}
|
||||
112
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/Review.php
vendored
Normal file
@@ -0,0 +1,112 @@
|
||||
<?php
|
||||
/**
|
||||
* The Review model class for ThemeIsle SDK
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 1.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Loader;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Review module for ThemeIsle SDK.
|
||||
*/
|
||||
class Review extends Abstract_Module {
|
||||
|
||||
/**
|
||||
* Check if we should load module for this.
|
||||
*
|
||||
* @param Product $product Product to check.
|
||||
*
|
||||
* @return bool Should load ?
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
if ( ! $product->is_wordpress_available() ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return apply_filters( $product->get_slug() . '_sdk_should_review', true );
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Add notification to queue.
|
||||
*
|
||||
* @param array $all_notifications Previous notification.
|
||||
*
|
||||
* @return array All notifications.
|
||||
*/
|
||||
public function add_notification( $all_notifications ) {
|
||||
|
||||
$developers = [
|
||||
'Marius',
|
||||
'Hardeep',
|
||||
'Andrei',
|
||||
'Robert',
|
||||
];
|
||||
|
||||
$link = 'https://wordpress.org/support/' . $this->product->get_type() . '/' . $this->product->get_slug() . '/reviews/#wporg-footer';
|
||||
|
||||
$message = apply_filters( $this->product->get_key() . '_feedback_review_message', Loader::$labels['review']['notice'] );
|
||||
|
||||
$button_submit = apply_filters( $this->product->get_key() . '_feedback_review_button_do', Loader::$labels['review']['ctay'] );
|
||||
$button_cancel = apply_filters( $this->product->get_key() . '_feedback_review_button_cancel', Loader::$labels['review']['ctan'] );
|
||||
$message = str_replace(
|
||||
[ '{product}', '{developer}' ],
|
||||
[
|
||||
$this->product->get_friendly_name(),
|
||||
$developers[ strlen( get_site_url() ) % count( $developers ) ],
|
||||
],
|
||||
$message
|
||||
);
|
||||
|
||||
$all_notifications[] = [
|
||||
'id' => $this->product->get_key() . '_review_flag',
|
||||
'message' => $message,
|
||||
'ctas' => [
|
||||
'confirm' => [
|
||||
'link' => $link,
|
||||
'text' => $button_submit,
|
||||
],
|
||||
'cancel' => [
|
||||
'link' => '#',
|
||||
'text' => $button_cancel,
|
||||
],
|
||||
],
|
||||
];
|
||||
|
||||
return $all_notifications;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Load module logic.
|
||||
*
|
||||
* @param Product $product Product to load.
|
||||
*
|
||||
* @return Review Module instance.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
|
||||
$this->product = $product;
|
||||
|
||||
add_filter( 'themeisle_sdk_registered_notifications', [ $this, 'add_notification' ] );
|
||||
|
||||
return $this;
|
||||
}
|
||||
}
|
||||
417
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/Rollback.php
vendored
Normal file
@@ -0,0 +1,417 @@
|
||||
<?php
|
||||
/**
|
||||
* The rollback class for ThemeIsle SDK.
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Rollback
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 1.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
// Exit if accessed directly.
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Loader;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Rollback for ThemeIsle SDK.
|
||||
*/
|
||||
class Rollback extends Abstract_Module {
|
||||
|
||||
/**
|
||||
* Add js scripts for themes rollback.
|
||||
*/
|
||||
public function add_footer() {
|
||||
$screen = get_current_screen();
|
||||
if ( ! isset( $screen->parent_file ) ) {
|
||||
return;
|
||||
}
|
||||
if ( 'themes.php' !== $screen->parent_file ) {
|
||||
return;
|
||||
}
|
||||
if ( ! $this->product->is_theme() ) {
|
||||
return;
|
||||
}
|
||||
$version = $this->get_rollback();
|
||||
if ( empty( $version ) ) {
|
||||
return;
|
||||
}
|
||||
?>
|
||||
<script type="text/javascript">
|
||||
jQuery(document).ready(function ($) {
|
||||
setInterval(checkTheme, 500);
|
||||
|
||||
function checkTheme() {
|
||||
var theme = '<?php echo esc_attr( $this->product->get_slug() ); ?>-action';
|
||||
|
||||
if (jQuery('#' + theme).length > 0) {
|
||||
if (jQuery('.theme-overlay.active').is(':visible')) {
|
||||
if (jQuery('#' + theme + '-rollback').length === 0) {
|
||||
jQuery('.theme-actions .active-theme').prepend('<a class="button" style="float:left" id="' + theme + '-rollback" href="<?php echo esc_url( wp_nonce_url( admin_url( 'admin-post.php?action=' . $this->product->get_key() . '_rollback' ), $this->product->get_key() . '_rollback' ) ); ?>">Rollback to v<?php echo esc_attr( $version['version'] ); ?></a>')
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
</script>
|
||||
<?php
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the last rollback for this product.
|
||||
*
|
||||
* @return array The rollback version.
|
||||
*/
|
||||
public function get_rollback() {
|
||||
$rollback = array();
|
||||
$versions = $this->get_api_versions();
|
||||
$versions = apply_filters( $this->product->get_key() . '_rollbacks', $versions );
|
||||
if ( empty( $versions ) ) {
|
||||
return $rollback;
|
||||
}
|
||||
if ( $versions ) {
|
||||
usort( $versions, array( $this, 'sort_rollback_array' ) );
|
||||
foreach ( $versions as $version ) {
|
||||
if ( isset( $version['version'] ) && isset( $version['url'] ) && version_compare( $this->product->get_version(), $version['version'], '>' ) ) {
|
||||
$rollback = $version;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return $rollback;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get versions array from wp.org
|
||||
*
|
||||
* @return array Array of versions.
|
||||
*/
|
||||
private function get_api_versions() {
|
||||
|
||||
$cache_key = $this->product->get_cache_key();
|
||||
$cache_versions = get_transient( $cache_key );
|
||||
if ( false === $cache_versions ) {
|
||||
$versions = $this->get_remote_versions();
|
||||
set_transient( $cache_key, $versions, 5 * DAY_IN_SECONDS );
|
||||
} else {
|
||||
$versions = is_array( $cache_versions ) ? $cache_versions : array();
|
||||
}
|
||||
|
||||
return $versions;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get remote versions zips.
|
||||
*
|
||||
* @return array Array of available versions.
|
||||
*/
|
||||
private function get_remote_versions() {
|
||||
$url = $this->get_versions_api_url();
|
||||
if ( empty( $url ) ) {
|
||||
return [];
|
||||
}
|
||||
$response = function_exists( 'vip_safe_wp_remote_get' )
|
||||
? vip_safe_wp_remote_get( $url )
|
||||
: wp_remote_get( $url ); //phpcs:ignore WordPressVIPMinimum.Functions.RestrictedFunctions.wp_remote_get_wp_remote_get
|
||||
if ( is_wp_error( $response ) ) {
|
||||
return array();
|
||||
}
|
||||
$response = wp_remote_retrieve_body( $response );
|
||||
|
||||
if ( is_serialized( $response ) ) {
|
||||
$response = maybe_unserialize( $response );
|
||||
} else {
|
||||
$response = json_decode( $response );
|
||||
}
|
||||
|
||||
if ( ! is_object( $response ) ) {
|
||||
return array();
|
||||
}
|
||||
if ( ! isset( $response->versions ) ) {
|
||||
return array();
|
||||
}
|
||||
|
||||
$versions = array();
|
||||
foreach ( $response->versions as $key => $value ) {
|
||||
$versions[] = array(
|
||||
'version' => is_object( $value ) ? $value->version : $key,
|
||||
'url' => is_object( $value ) ? $value->file : $value,
|
||||
);
|
||||
}
|
||||
|
||||
return $versions;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return url where to check for versions.
|
||||
*
|
||||
* @return string Url where to check for versions.
|
||||
*/
|
||||
private function get_versions_api_url() {
|
||||
if ( $this->product->is_wordpress_available() && $this->product->is_plugin() ) {
|
||||
return sprintf( 'https://api.wordpress.org/plugins/info/1.0/%s', $this->product->get_slug() );
|
||||
}
|
||||
if ( $this->product->is_wordpress_available() && $this->product->is_theme() ) {
|
||||
return sprintf( 'https://api.wordpress.org/themes/info/1.1/?action=theme_information&request[slug]=%s&request[fields][versions]=true', $this->product->get_slug() );
|
||||
}
|
||||
$license = $this->product->get_license();
|
||||
if ( $this->product->requires_license() && strlen( $license ) < 10 ) {
|
||||
return '';
|
||||
}
|
||||
|
||||
return sprintf( '%slicense/versions/%s/%s/%s/%s', Product::API_URL, rawurlencode( $this->product->get_name() ), $license, urlencode( get_site_url() ), $this->product->get_version() );
|
||||
}
|
||||
|
||||
/**
|
||||
* Show the rollback links in the plugin page.
|
||||
*
|
||||
* @param array $links Plugin links.
|
||||
*
|
||||
* @return array $links Altered links.
|
||||
*/
|
||||
public function add_rollback_link( $links ) {
|
||||
$version = $this->get_rollback();
|
||||
if ( empty( $version ) ) {
|
||||
return $links;
|
||||
}
|
||||
$links[] = '<a href="' . wp_nonce_url( admin_url( 'admin-post.php?action=' . $this->product->get_key() . '_rollback' ), $this->product->get_key() . '_rollback' ) . '">' . sprintf( apply_filters( $this->product->get_key() . '_rollback_label', Loader::$labels['rollback']['cta'] ), $version['version'] ) . '</a>';
|
||||
|
||||
return $links;
|
||||
}
|
||||
|
||||
/**
|
||||
* Start the rollback operation.
|
||||
*/
|
||||
public function start_rollback() {
|
||||
if ( ! isset( $_GET['_wpnonce'] ) || ! wp_verify_nonce( $_GET['_wpnonce'], $this->product->get_key() . '_rollback' ) ) { //phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized
|
||||
wp_nonce_ays( '' );
|
||||
}
|
||||
|
||||
if ( $this->product->is_plugin() ) {
|
||||
$this->start_rollback_plugin();
|
||||
|
||||
return;
|
||||
}
|
||||
if ( $this->product->is_theme() ) {
|
||||
$this->start_rollback_theme();
|
||||
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Start the rollback operation for the plugin.
|
||||
*/
|
||||
private function start_rollback_plugin() {
|
||||
$rollback = $this->get_rollback();
|
||||
$plugin_transient = get_site_transient( 'update_plugins' );
|
||||
$plugin_folder = $this->product->get_slug();
|
||||
$plugin_file = $this->product->get_file();
|
||||
$version = $rollback['version'];
|
||||
$temp_array = array(
|
||||
'slug' => $plugin_folder,
|
||||
'new_version' => $version,
|
||||
'package' => $rollback['url'],
|
||||
);
|
||||
|
||||
$temp_object = (object) $temp_array;
|
||||
$plugin_transient->response[ $plugin_folder . '/' . $plugin_file ] = $temp_object;
|
||||
set_site_transient( 'update_plugins', $plugin_transient );
|
||||
|
||||
$transient = get_transient( $this->product->get_key() . '_warning_rollback' );
|
||||
|
||||
// Style fix for the api link that gets outside the content.
|
||||
echo '<style>body#error-page{word-break:break-word;}</style>';
|
||||
|
||||
if ( false === $transient ) {
|
||||
set_transient( $this->product->get_key() . '_warning_rollback', 'in progress', 30 );
|
||||
require_once ABSPATH . 'wp-admin/includes/class-wp-upgrader.php';
|
||||
$title = sprintf( apply_filters( $this->product->get_key() . '_rollback_message', 'Rolling back %s to v%s' ), $this->product->get_name(), $version );
|
||||
$plugin = $plugin_folder . '/' . $plugin_file;
|
||||
$nonce = 'upgrade-plugin_' . $plugin;
|
||||
$url = 'update.php?action=upgrade-plugin&plugin=' . urlencode( $plugin );
|
||||
$upgrader_skin = new \Plugin_Upgrader_Skin( compact( 'title', 'nonce', 'url', 'plugin' ) );
|
||||
$upgrader = new \Plugin_Upgrader( $upgrader_skin );
|
||||
$upgrader->upgrade( $plugin );
|
||||
delete_transient( $this->product->get_key() . '_warning_rollback' );
|
||||
wp_die(
|
||||
'',
|
||||
esc_attr( $title ),
|
||||
array(
|
||||
'response' => 200,
|
||||
)
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Start the rollback operation for the theme.
|
||||
*/
|
||||
private function start_rollback_theme() {
|
||||
add_filter( 'update_theme_complete_actions', array( $this, 'alter_links_theme_upgrade' ) );
|
||||
$rollback = $this->get_rollback();
|
||||
$transient = get_site_transient( 'update_themes' );
|
||||
$folder = $this->product->get_slug();
|
||||
$version = $rollback['version'];
|
||||
$temp_array = array(
|
||||
'new_version' => $version,
|
||||
'package' => $rollback['url'],
|
||||
);
|
||||
|
||||
$transient->response[ $folder . '/style.css' ] = $temp_array;
|
||||
set_site_transient( 'update_themes', $transient );
|
||||
|
||||
$transient = get_transient( $this->product->get_key() . '_warning_rollback' );
|
||||
|
||||
// Style fix for the api link that gets outside the content.
|
||||
echo '<style>body#error-page{word-break:break-word;}</style>';
|
||||
|
||||
if ( false === $transient ) {
|
||||
set_transient( $this->product->get_key() . '_warning_rollback', 'in progress', 30 );
|
||||
require_once ABSPATH . 'wp-admin/includes/class-wp-upgrader.php';
|
||||
$title = sprintf( apply_filters( $this->product->get_key() . '_rollback_message', 'Rolling back %s to v%s' ), $this->product->get_name(), $version );
|
||||
$theme = $folder . '/style.css';
|
||||
$nonce = 'upgrade-theme_' . $theme;
|
||||
$url = 'update.php?action=upgrade-theme&theme=' . urlencode( $theme );
|
||||
|
||||
/**
|
||||
* The rollback will attach a temporary theme for the rollback to the transient.
|
||||
* However, when executing the upgrade for the attached theme we need to change the slug to the original theme slug.
|
||||
* This is because it will use the slug to create a temp folder for the theme used during the upgrade.
|
||||
*/
|
||||
add_filter(
|
||||
'upgrader_package_options',
|
||||
function ( $options ) use ( $folder, $theme ) {
|
||||
if ( isset( $options['hook_extra']['theme'] ) && $options['hook_extra']['theme'] === $theme && isset( $options['hook_extra']['temp_backup']['slug'] ) ) {
|
||||
$options['hook_extra']['temp_backup']['slug'] = $folder;
|
||||
}
|
||||
|
||||
return $options;
|
||||
}
|
||||
);
|
||||
|
||||
$upgrader = new \Theme_Upgrader( new \Theme_Upgrader_Skin( compact( 'title', 'nonce', 'url', 'theme' ) ) );
|
||||
$upgrader->upgrade( $theme );
|
||||
delete_transient( $this->product->get_key() . '_warning_rollback' );
|
||||
wp_die(
|
||||
'',
|
||||
esc_attr( $title ),
|
||||
array(
|
||||
'response' => 200,
|
||||
)
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Alter links and remove duplicate customize message.
|
||||
*
|
||||
* @param array $links Array of old links.
|
||||
*
|
||||
* @return mixed Array of links.
|
||||
*/
|
||||
public function alter_links_theme_upgrade( $links ) {
|
||||
if ( isset( $links['preview'] ) ) {
|
||||
$links['preview'] = str_replace( '<span aria-hidden="true">Customize</span>', '', $links['preview'] );
|
||||
}
|
||||
|
||||
return $links;
|
||||
}
|
||||
|
||||
/**
|
||||
* Loads product object.
|
||||
*
|
||||
* @param Product $product Product object.
|
||||
*
|
||||
* @return bool Should we load the module?
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
if ( $product->is_theme() && ! current_user_can( 'switch_themes' ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( $product->is_plugin() && ! current_user_can( 'install_plugins' ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Sort the rollbacks array in descending order.
|
||||
*
|
||||
* @param mixed $a First version to compare.
|
||||
* @param mixed $b Second version to compare.
|
||||
*
|
||||
* @return bool Which version is greater?
|
||||
*/
|
||||
public function sort_rollback_array( $a, $b ) {
|
||||
return version_compare( $b['version'], $a['version'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Load module logic.
|
||||
*
|
||||
* @param Product $product Product object.
|
||||
*
|
||||
* @return $this Module object.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
$this->product = $product;
|
||||
$this->show_link();
|
||||
$this->add_hooks();
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* If product can be rolled back, show the link to rollback.
|
||||
*/
|
||||
private function show_link() {
|
||||
add_filter(
|
||||
'plugin_action_links_' . plugin_basename( $this->product->get_basefile() ),
|
||||
array(
|
||||
$this,
|
||||
'add_rollback_link',
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Fires after the option has been updated.
|
||||
*
|
||||
* @param mixed $old_value The old option value.
|
||||
* @param mixed $value The new option value.
|
||||
* @param string $option Option name.
|
||||
*/
|
||||
public function update_active_plugins_action( $old_value, $value, $option ) {
|
||||
delete_site_transient( 'update_plugins' );
|
||||
wp_cache_delete( 'plugins', 'plugins' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Set the rollback hook. Strangely, this does not work if placed in the ThemeIsle_SDK_Rollback class, so it is being called from there instead.
|
||||
*/
|
||||
public function add_hooks() {
|
||||
add_action( 'admin_post_' . $this->product->get_key() . '_rollback', array( $this, 'start_rollback' ) );
|
||||
add_action( 'admin_footer', array( $this, 'add_footer' ) );
|
||||
|
||||
// This hook will be invoked after the plugin activation.
|
||||
// We use this to force an update of the cache so that Update is present immediate after a rollback.
|
||||
add_action( 'update_option_active_plugins', array( $this, 'update_active_plugins_action' ), 10, 3 );
|
||||
}
|
||||
}
|
||||
253
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/Script_loader.php
vendored
Normal file
@@ -0,0 +1,253 @@
|
||||
<?php
|
||||
/**
|
||||
* The dependency model class for ThemeIsle SDK
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 3.3
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Script loader module for ThemeIsle SDK.
|
||||
*/
|
||||
class Script_Loader extends Abstract_Module {
|
||||
/**
|
||||
* Check if we should load the module for this product.
|
||||
*
|
||||
* @param Product $product Product to load the module for.
|
||||
*
|
||||
* @return bool Should we load ?
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Load module logic.
|
||||
*
|
||||
* @param Product $product Product to load.
|
||||
*
|
||||
* @return Dependancy Module object.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
$this->product = $product;
|
||||
$this->setup_actions();
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Setup actions. Once for all products.
|
||||
*/
|
||||
private function setup_actions() {
|
||||
|
||||
if ( apply_filters( 'themeisle_sdk_script_setup', false ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
add_filter( 'themeisle_sdk_dependency_script_handler', [ $this, 'get_script_handler' ], 10, 1 );
|
||||
add_action( 'themeisle_sdk_dependency_enqueue_script', [ $this, 'enqueue_script' ], 10, 1 );
|
||||
add_filter( 'themeisle_sdk_secret_masking', [ $this, 'secret_masking' ], 10, 1 );
|
||||
|
||||
add_filter( 'themeisle_sdk_script_setup', '__return_true' );
|
||||
|
||||
add_action( 'themeisle_internal_page', [ $this, 'load_survey_for_product' ], 10, 2 );
|
||||
}
|
||||
|
||||
/**
|
||||
* Load survey for product using internal pages.
|
||||
*
|
||||
* @param string $product_slug Product slug.
|
||||
* @param string $page_slug Page slug.
|
||||
*/
|
||||
public function load_survey_for_product( $product_slug, $page_slug ) {
|
||||
$data = apply_filters( 'themeisle-sdk/survey/' . $product_slug, [], $page_slug );
|
||||
|
||||
if ( empty( $data ) || ! is_array( $data ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$handler = $this->get_script_handler( 'survey' );
|
||||
$this->load_survey( $handler, $data );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the script handler.
|
||||
*
|
||||
* @param string $slug The slug of the script.
|
||||
*
|
||||
* @return string The script handler. Empty if slug is not a string or not implemented.
|
||||
*/
|
||||
public function get_script_handler( $slug ) {
|
||||
if ( ! is_string( $slug ) ) {
|
||||
return '';
|
||||
}
|
||||
|
||||
if ( 'tracking' !== $slug && 'survey' !== $slug ) {
|
||||
return '';
|
||||
}
|
||||
|
||||
return apply_filters( 'themeisle_sdk_dependency_script_handler_name', 'themeisle_sdk_' . $slug . '_script', $slug );
|
||||
}
|
||||
|
||||
/**
|
||||
* Enqueue the script.
|
||||
*
|
||||
* @param string $slug The slug of the script.
|
||||
*/
|
||||
public function enqueue_script( $slug ) {
|
||||
$handler = apply_filters( 'themeisle_sdk_dependency_script_handler', $slug );
|
||||
if ( empty( $handler ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( 'tracking' === $slug ) {
|
||||
$this->load_tracking( $handler );
|
||||
} elseif ( 'survey' === $slug ) {
|
||||
$this->load_survey( $handler );
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Load the survey script.
|
||||
*
|
||||
* @param string $handler The script handler.
|
||||
* @param array $data The survey data.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function load_survey( $handler, $data = array() ) {
|
||||
global $themeisle_sdk_max_path;
|
||||
$asset_file = require $themeisle_sdk_max_path . '/assets/js/build/survey/survey_deps.asset.php';
|
||||
|
||||
wp_enqueue_script(
|
||||
$handler,
|
||||
$this->get_sdk_uri() . 'assets/js/build/survey/survey_deps.js',
|
||||
$asset_file['dependencies'],
|
||||
$asset_file['version'],
|
||||
true
|
||||
);
|
||||
|
||||
$data = array_replace_recursive( $this->get_survey_common_data( $data ), $data );
|
||||
|
||||
wp_localize_script( $handler, 'tsdk_survey_data', $data );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the common data in the Formbrick survey format.
|
||||
*
|
||||
* @param array $reference_data Reference data to extrapolate common properties.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function get_survey_common_data( $reference_data = array() ) {
|
||||
$language = apply_filters( 'themeisle_sdk_current_lang', get_user_locale() );
|
||||
$available_languages = [
|
||||
'de_DE' => 'de',
|
||||
'de_DE_formal' => 'de',
|
||||
];
|
||||
$lang_code = isset( $available_languages[ $language ] ) ? $available_languages[ $language ] : 'en';
|
||||
|
||||
$url_parts = wp_parse_url( apply_filters( 'themeisle_sdk_current_site_url', get_site_url() ) );
|
||||
$clean_url = str_replace( 'www.', '', $url_parts['host'] );
|
||||
if ( isset( $url_parts['path'] ) ) {
|
||||
$clean_url .= $url_parts['path'];
|
||||
}
|
||||
$user_id = 'u_' . hash( 'crc32b', $clean_url );
|
||||
|
||||
$common_data = [
|
||||
'userId' => $user_id,
|
||||
'appUrl' => 'https://app.formbricks.com',
|
||||
'attributes' => [
|
||||
'language' => $lang_code,
|
||||
],
|
||||
];
|
||||
|
||||
if (
|
||||
isset( $reference_data['attributes'], $reference_data['attributes']['install_days_number'] )
|
||||
&& is_int( $reference_data['attributes']['install_days_number'] )
|
||||
) {
|
||||
$common_data['attributes']['days_since_install'] = $this->install_time_category( $reference_data['attributes']['install_days_number'] );
|
||||
}
|
||||
|
||||
return $common_data;
|
||||
}
|
||||
|
||||
/**
|
||||
* Compute the install time category.
|
||||
*
|
||||
* @param int $install_days_number The number of days passed since installation.
|
||||
*
|
||||
* @return int The category.
|
||||
*/
|
||||
private function install_time_category( $install_days_number ) {
|
||||
if ( 1 < $install_days_number && 8 > $install_days_number ) {
|
||||
return 7;
|
||||
}
|
||||
|
||||
if ( 8 <= $install_days_number && 31 > $install_days_number ) {
|
||||
return 30;
|
||||
}
|
||||
|
||||
if ( 30 < $install_days_number && 90 > $install_days_number ) {
|
||||
return 90;
|
||||
}
|
||||
|
||||
if ( 90 <= $install_days_number ) {
|
||||
return 91;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Load the tracking script.
|
||||
*
|
||||
* @param string $handler The script handler.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function load_tracking( $handler ) {
|
||||
global $themeisle_sdk_max_path;
|
||||
$asset_file = require $themeisle_sdk_max_path . '/assets/js/build/tracking/tracking.asset.php';
|
||||
|
||||
wp_enqueue_script(
|
||||
$handler,
|
||||
$this->get_sdk_uri() . 'assets/js/build/tracking/tracking.js',
|
||||
$asset_file['dependencies'],
|
||||
$asset_file['version'],
|
||||
true
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Mask a secret with `*` for half of its length.
|
||||
*
|
||||
* @param mixed $secret The secret.
|
||||
*
|
||||
* @return mixed The masked secret if secret is a valid string.
|
||||
*/
|
||||
public function secret_masking( $secret ) {
|
||||
if ( empty( $secret ) || ! is_string( $secret ) ) {
|
||||
return $secret;
|
||||
}
|
||||
|
||||
$half_len = intval( strlen( $secret ) / 2 );
|
||||
return str_repeat( '*', $half_len ) . substr( $secret, $half_len );
|
||||
}
|
||||
}
|
||||
918
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/Translate.php
vendored
Normal file
@@ -0,0 +1,918 @@
|
||||
<?php
|
||||
/**
|
||||
* The translate model class for ThemeIsle SDK
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 1.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Translate module for ThemeIsle SDK.
|
||||
*/
|
||||
class Translate extends Abstract_Module {
|
||||
/**
|
||||
* List of available locales.
|
||||
*
|
||||
* @var array Array of available locals.
|
||||
*/
|
||||
private static $locales = array(
|
||||
'af' => array(
|
||||
'slug' => 'af',
|
||||
'name' => 'Afrikaans',
|
||||
),
|
||||
'ak' => array(
|
||||
'slug' => 'ak',
|
||||
'name' => 'Akan',
|
||||
),
|
||||
'am' => array(
|
||||
'slug' => 'am',
|
||||
'name' => 'Amharic',
|
||||
),
|
||||
'ar' => array(
|
||||
'slug' => 'ar',
|
||||
'name' => 'Arabic',
|
||||
),
|
||||
'arq' => array(
|
||||
'slug' => 'arq',
|
||||
'name' => 'Algerian Arabic',
|
||||
),
|
||||
'ary' => array(
|
||||
'slug' => 'ary',
|
||||
'name' => 'Moroccan Arabic',
|
||||
),
|
||||
'as' => array(
|
||||
'slug' => 'as',
|
||||
'name' => 'Assamese',
|
||||
),
|
||||
'ast' => array(
|
||||
'slug' => 'ast',
|
||||
'name' => 'Asturian',
|
||||
),
|
||||
'az' => array(
|
||||
'slug' => 'az',
|
||||
'name' => 'Azerbaijani',
|
||||
),
|
||||
'azb' => array(
|
||||
'slug' => 'azb',
|
||||
'name' => 'South Azerbaijani',
|
||||
),
|
||||
'az_TR' => array(
|
||||
'slug' => 'az-tr',
|
||||
'name' => 'Azerbaijani (Turkey)',
|
||||
),
|
||||
'ba' => array(
|
||||
'slug' => 'ba',
|
||||
'name' => 'Bashkir',
|
||||
),
|
||||
'bal' => array(
|
||||
'slug' => 'bal',
|
||||
'name' => 'Catalan (Balear)',
|
||||
),
|
||||
'bcc' => array(
|
||||
'slug' => 'bcc',
|
||||
'name' => 'Balochi Southern',
|
||||
),
|
||||
'bel' => array(
|
||||
'slug' => 'bel',
|
||||
'name' => 'Belarusian',
|
||||
),
|
||||
'bg_BG' => array(
|
||||
'slug' => 'bg',
|
||||
'name' => 'Bulgarian',
|
||||
),
|
||||
'bn_BD' => array(
|
||||
'slug' => 'bn',
|
||||
'name' => 'Bengali',
|
||||
),
|
||||
'bo' => array(
|
||||
'slug' => 'bo',
|
||||
'name' => 'Tibetan',
|
||||
),
|
||||
'bre' => array(
|
||||
'slug' => 'br',
|
||||
'name' => 'Breton',
|
||||
),
|
||||
'bs_BA' => array(
|
||||
'slug' => 'bs',
|
||||
'name' => 'Bosnian',
|
||||
),
|
||||
'ca' => array(
|
||||
'slug' => 'ca',
|
||||
'name' => 'Catalan',
|
||||
),
|
||||
'ceb' => array(
|
||||
'slug' => 'ceb',
|
||||
'name' => 'Cebuano',
|
||||
),
|
||||
'ckb' => array(
|
||||
'slug' => 'ckb',
|
||||
'name' => 'Kurdish (Sorani)',
|
||||
),
|
||||
'co' => array(
|
||||
'slug' => 'co',
|
||||
'name' => 'Corsican',
|
||||
),
|
||||
'cs_CZ' => array(
|
||||
'slug' => 'cs',
|
||||
'name' => 'Czech',
|
||||
),
|
||||
'cy' => array(
|
||||
'slug' => 'cy',
|
||||
'name' => 'Welsh',
|
||||
),
|
||||
'da_DK' => array(
|
||||
'slug' => 'da',
|
||||
'name' => 'Danish',
|
||||
),
|
||||
'de_DE' => array(
|
||||
'slug' => 'de',
|
||||
'name' => 'German',
|
||||
),
|
||||
'de_CH' => array(
|
||||
'slug' => 'de-ch',
|
||||
'name' => 'German (Switzerland)',
|
||||
),
|
||||
'dv' => array(
|
||||
'slug' => 'dv',
|
||||
'name' => 'Dhivehi',
|
||||
),
|
||||
'dzo' => array(
|
||||
'slug' => 'dzo',
|
||||
'name' => 'Dzongkha',
|
||||
),
|
||||
'el' => array(
|
||||
'slug' => 'el',
|
||||
'name' => 'Greek',
|
||||
),
|
||||
'art_xemoji' => array(
|
||||
'slug' => 'art-xemoji',
|
||||
'name' => 'Emoji',
|
||||
),
|
||||
'en_US' => array(
|
||||
'slug' => 'en',
|
||||
'name' => 'English',
|
||||
),
|
||||
'en_AU' => array(
|
||||
'slug' => 'en-au',
|
||||
'name' => 'English (Australia)',
|
||||
),
|
||||
'en_CA' => array(
|
||||
'slug' => 'en-ca',
|
||||
'name' => 'English (Canada)',
|
||||
),
|
||||
'en_GB' => array(
|
||||
'slug' => 'en-gb',
|
||||
'name' => 'English (UK)',
|
||||
),
|
||||
'en_NZ' => array(
|
||||
'slug' => 'en-nz',
|
||||
'name' => 'English (New Zealand)',
|
||||
),
|
||||
'en_ZA' => array(
|
||||
'slug' => 'en-za',
|
||||
'name' => 'English (South Africa)',
|
||||
),
|
||||
'eo' => array(
|
||||
'slug' => 'eo',
|
||||
'name' => 'Esperanto',
|
||||
),
|
||||
'es_ES' => array(
|
||||
'slug' => 'es',
|
||||
'name' => 'Spanish (Spain)',
|
||||
),
|
||||
'es_AR' => array(
|
||||
'slug' => 'es-ar',
|
||||
'name' => 'Spanish (Argentina)',
|
||||
),
|
||||
'es_CL' => array(
|
||||
'slug' => 'es-cl',
|
||||
'name' => 'Spanish (Chile)',
|
||||
),
|
||||
'es_CO' => array(
|
||||
'slug' => 'es-co',
|
||||
'name' => 'Spanish (Colombia)',
|
||||
),
|
||||
'es_CR' => array(
|
||||
'slug' => 'es-cr',
|
||||
'name' => 'Spanish (Costa Rica)',
|
||||
),
|
||||
'es_GT' => array(
|
||||
'slug' => 'es-gt',
|
||||
'name' => 'Spanish (Guatemala)',
|
||||
),
|
||||
'es_MX' => array(
|
||||
'slug' => 'es-mx',
|
||||
'name' => 'Spanish (Mexico)',
|
||||
),
|
||||
'es_PE' => array(
|
||||
'slug' => 'es-pe',
|
||||
'name' => 'Spanish (Peru)',
|
||||
),
|
||||
'es_PR' => array(
|
||||
'slug' => 'es-pr',
|
||||
'name' => 'Spanish (Puerto Rico)',
|
||||
),
|
||||
'es_VE' => array(
|
||||
'slug' => 'es-ve',
|
||||
'name' => 'Spanish (Venezuela)',
|
||||
),
|
||||
'et' => array(
|
||||
'slug' => 'et',
|
||||
'name' => 'Estonian',
|
||||
),
|
||||
'eu' => array(
|
||||
'slug' => 'eu',
|
||||
'name' => 'Basque',
|
||||
),
|
||||
'fa_IR' => array(
|
||||
'slug' => 'fa',
|
||||
'name' => 'Persian',
|
||||
),
|
||||
'fa_AF' => array(
|
||||
'slug' => 'fa-af',
|
||||
'name' => 'Persian (Afghanistan)',
|
||||
),
|
||||
'fuc' => array(
|
||||
'slug' => 'fuc',
|
||||
'name' => 'Fulah',
|
||||
),
|
||||
'fi' => array(
|
||||
'slug' => 'fi',
|
||||
'name' => 'Finnish',
|
||||
),
|
||||
'fo' => array(
|
||||
'slug' => 'fo',
|
||||
'name' => 'Faroese',
|
||||
),
|
||||
'fr_FR' => array(
|
||||
'slug' => 'fr',
|
||||
'name' => 'French (France)',
|
||||
),
|
||||
'fr_BE' => array(
|
||||
'slug' => 'fr-be',
|
||||
'name' => 'French (Belgium)',
|
||||
),
|
||||
'fr_CA' => array(
|
||||
'slug' => 'fr-ca',
|
||||
'name' => 'French (Canada)',
|
||||
),
|
||||
'frp' => array(
|
||||
'slug' => 'frp',
|
||||
'name' => 'Arpitan',
|
||||
),
|
||||
'fur' => array(
|
||||
'slug' => 'fur',
|
||||
'name' => 'Friulian',
|
||||
),
|
||||
'fy' => array(
|
||||
'slug' => 'fy',
|
||||
'name' => 'Frisian',
|
||||
),
|
||||
'ga' => array(
|
||||
'slug' => 'ga',
|
||||
'name' => 'Irish',
|
||||
),
|
||||
'gd' => array(
|
||||
'slug' => 'gd',
|
||||
'name' => 'Scottish Gaelic',
|
||||
),
|
||||
'gl_ES' => array(
|
||||
'slug' => 'gl',
|
||||
'name' => 'Galician',
|
||||
),
|
||||
'gn' => array(
|
||||
'slug' => 'gn',
|
||||
'name' => 'Guarani',
|
||||
),
|
||||
'gsw' => array(
|
||||
'slug' => 'gsw',
|
||||
'name' => 'Swiss German',
|
||||
),
|
||||
'gu' => array(
|
||||
'slug' => 'gu',
|
||||
'name' => 'Gujarati',
|
||||
),
|
||||
'hat' => array(
|
||||
'slug' => 'hat',
|
||||
'name' => 'Haitian Creole',
|
||||
),
|
||||
'hau' => array(
|
||||
'slug' => 'hau',
|
||||
'name' => 'Hausa',
|
||||
),
|
||||
'haw_US' => array(
|
||||
'slug' => 'haw',
|
||||
'name' => 'Hawaiian',
|
||||
),
|
||||
'haz' => array(
|
||||
'slug' => 'haz',
|
||||
'name' => 'Hazaragi',
|
||||
),
|
||||
'he_IL' => array(
|
||||
'slug' => 'he',
|
||||
'name' => 'Hebrew',
|
||||
),
|
||||
'hi_IN' => array(
|
||||
'slug' => 'hi',
|
||||
'name' => 'Hindi',
|
||||
),
|
||||
'hr' => array(
|
||||
'slug' => 'hr',
|
||||
'name' => 'Croatian',
|
||||
),
|
||||
'hu_HU' => array(
|
||||
'slug' => 'hu',
|
||||
'name' => 'Hungarian',
|
||||
),
|
||||
'hy' => array(
|
||||
'slug' => 'hy',
|
||||
'name' => 'Armenian',
|
||||
),
|
||||
'id_ID' => array(
|
||||
'slug' => 'id',
|
||||
'name' => 'Indonesian',
|
||||
),
|
||||
'ido' => array(
|
||||
'slug' => 'ido',
|
||||
'name' => 'Ido',
|
||||
),
|
||||
'is_IS' => array(
|
||||
'slug' => 'is',
|
||||
'name' => 'Icelandic',
|
||||
),
|
||||
'it_IT' => array(
|
||||
'slug' => 'it',
|
||||
'name' => 'Italian',
|
||||
),
|
||||
'ja' => array(
|
||||
'slug' => 'ja',
|
||||
'name' => 'Japanese',
|
||||
),
|
||||
'jv_ID' => array(
|
||||
'slug' => 'jv',
|
||||
'name' => 'Javanese',
|
||||
),
|
||||
'ka_GE' => array(
|
||||
'slug' => 'ka',
|
||||
'name' => 'Georgian',
|
||||
),
|
||||
'kab' => array(
|
||||
'slug' => 'kab',
|
||||
'name' => 'Kabyle',
|
||||
),
|
||||
'kal' => array(
|
||||
'slug' => 'kal',
|
||||
'name' => 'Greenlandic',
|
||||
),
|
||||
'kin' => array(
|
||||
'slug' => 'kin',
|
||||
'name' => 'Kinyarwanda',
|
||||
),
|
||||
'kk' => array(
|
||||
'slug' => 'kk',
|
||||
'name' => 'Kazakh',
|
||||
),
|
||||
'km' => array(
|
||||
'slug' => 'km',
|
||||
'name' => 'Khmer',
|
||||
),
|
||||
'kn' => array(
|
||||
'slug' => 'kn',
|
||||
'name' => 'Kannada',
|
||||
),
|
||||
'ko_KR' => array(
|
||||
'slug' => 'ko',
|
||||
'name' => 'Korean',
|
||||
),
|
||||
'kir' => array(
|
||||
'slug' => 'kir',
|
||||
'name' => 'Kyrgyz',
|
||||
),
|
||||
'lb_LU' => array(
|
||||
'slug' => 'lb',
|
||||
'name' => 'Luxembourgish',
|
||||
),
|
||||
'li' => array(
|
||||
'slug' => 'li',
|
||||
'name' => 'Limburgish',
|
||||
),
|
||||
'lin' => array(
|
||||
'slug' => 'lin',
|
||||
'name' => 'Lingala',
|
||||
),
|
||||
'lo' => array(
|
||||
'slug' => 'lo',
|
||||
'name' => 'Lao',
|
||||
),
|
||||
'lt_LT' => array(
|
||||
'slug' => 'lt',
|
||||
'name' => 'Lithuanian',
|
||||
),
|
||||
'lv' => array(
|
||||
'slug' => 'lv',
|
||||
'name' => 'Latvian',
|
||||
),
|
||||
'me_ME' => array(
|
||||
'slug' => 'me',
|
||||
'name' => 'Montenegrin',
|
||||
),
|
||||
'mg_MG' => array(
|
||||
'slug' => 'mg',
|
||||
'name' => 'Malagasy',
|
||||
),
|
||||
'mk_MK' => array(
|
||||
'slug' => 'mk',
|
||||
'name' => 'Macedonian',
|
||||
),
|
||||
'ml_IN' => array(
|
||||
'slug' => 'ml',
|
||||
'name' => 'Malayalam',
|
||||
),
|
||||
'mlt' => array(
|
||||
'slug' => 'mlt',
|
||||
'name' => 'Maltese',
|
||||
),
|
||||
'mn' => array(
|
||||
'slug' => 'mn',
|
||||
'name' => 'Mongolian',
|
||||
),
|
||||
'mr' => array(
|
||||
'slug' => 'mr',
|
||||
'name' => 'Marathi',
|
||||
),
|
||||
'mri' => array(
|
||||
'slug' => 'mri',
|
||||
'name' => 'Maori',
|
||||
),
|
||||
'ms_MY' => array(
|
||||
'slug' => 'ms',
|
||||
'name' => 'Malay',
|
||||
),
|
||||
'my_MM' => array(
|
||||
'slug' => 'mya',
|
||||
'name' => 'Myanmar (Burmese)',
|
||||
),
|
||||
'ne_NP' => array(
|
||||
'slug' => 'ne',
|
||||
'name' => 'Nepali',
|
||||
),
|
||||
'nb_NO' => array(
|
||||
'slug' => 'nb',
|
||||
'name' => 'Norwegian (Bokmal)',
|
||||
),
|
||||
'nl_NL' => array(
|
||||
'slug' => 'nl',
|
||||
'name' => 'Dutch',
|
||||
),
|
||||
'nl_BE' => array(
|
||||
'slug' => 'nl-be',
|
||||
'name' => 'Dutch (Belgium)',
|
||||
),
|
||||
'nn_NO' => array(
|
||||
'slug' => 'nn',
|
||||
'name' => 'Norwegian (Nynorsk)',
|
||||
),
|
||||
'oci' => array(
|
||||
'slug' => 'oci',
|
||||
'name' => 'Occitan',
|
||||
),
|
||||
'ory' => array(
|
||||
'slug' => 'ory',
|
||||
'name' => 'Oriya',
|
||||
),
|
||||
'os' => array(
|
||||
'slug' => 'os',
|
||||
'name' => 'Ossetic',
|
||||
),
|
||||
'pa_IN' => array(
|
||||
'slug' => 'pa',
|
||||
'name' => 'Punjabi',
|
||||
),
|
||||
'pl_PL' => array(
|
||||
'slug' => 'pl',
|
||||
'name' => 'Polish',
|
||||
),
|
||||
'pt_BR' => array(
|
||||
'slug' => 'pt-br',
|
||||
'name' => 'Portuguese (Brazil)',
|
||||
),
|
||||
'pt_PT' => array(
|
||||
'slug' => 'pt',
|
||||
'name' => 'Portuguese (Portugal)',
|
||||
),
|
||||
'ps' => array(
|
||||
'slug' => 'ps',
|
||||
'name' => 'Pashto',
|
||||
),
|
||||
'rhg' => array(
|
||||
'slug' => 'rhg',
|
||||
'name' => 'Rohingya',
|
||||
),
|
||||
'ro_RO' => array(
|
||||
'slug' => 'ro',
|
||||
'name' => 'Romanian',
|
||||
),
|
||||
'roh' => array(
|
||||
'slug' => 'roh',
|
||||
'name' => 'Romansh',
|
||||
),
|
||||
'ru_RU' => array(
|
||||
'slug' => 'ru',
|
||||
'name' => 'Russian',
|
||||
),
|
||||
'rue' => array(
|
||||
'slug' => 'rue',
|
||||
'name' => 'Rusyn',
|
||||
),
|
||||
'rup_MK' => array(
|
||||
'slug' => 'rup',
|
||||
'name' => 'Aromanian',
|
||||
),
|
||||
'sah' => array(
|
||||
'slug' => 'sah',
|
||||
'name' => 'Sakha',
|
||||
),
|
||||
'sa_IN' => array(
|
||||
'slug' => 'sa-in',
|
||||
'name' => 'Sanskrit',
|
||||
),
|
||||
'scn' => array(
|
||||
'slug' => 'scn',
|
||||
'name' => 'Sicilian',
|
||||
),
|
||||
'si_LK' => array(
|
||||
'slug' => 'si',
|
||||
'name' => 'Sinhala',
|
||||
),
|
||||
'sk_SK' => array(
|
||||
'slug' => 'sk',
|
||||
'name' => 'Slovak',
|
||||
),
|
||||
'sl_SI' => array(
|
||||
'slug' => 'sl',
|
||||
'name' => 'Slovenian',
|
||||
),
|
||||
'sna' => array(
|
||||
'slug' => 'sna',
|
||||
'name' => 'Shona',
|
||||
),
|
||||
'snd' => array(
|
||||
'slug' => 'snd',
|
||||
'name' => 'Sindhi',
|
||||
),
|
||||
'so_SO' => array(
|
||||
'slug' => 'so',
|
||||
'name' => 'Somali',
|
||||
),
|
||||
'sq' => array(
|
||||
'slug' => 'sq',
|
||||
'name' => 'Albanian',
|
||||
),
|
||||
'sq_XK' => array(
|
||||
'slug' => 'sq-xk',
|
||||
'name' => 'Shqip (Kosovo)',
|
||||
),
|
||||
'sr_RS' => array(
|
||||
'slug' => 'sr',
|
||||
'name' => 'Serbian',
|
||||
),
|
||||
'srd' => array(
|
||||
'slug' => 'srd',
|
||||
'name' => 'Sardinian',
|
||||
),
|
||||
'su_ID' => array(
|
||||
'slug' => 'su',
|
||||
'name' => 'Sundanese',
|
||||
),
|
||||
'sv_SE' => array(
|
||||
'slug' => 'sv',
|
||||
'name' => 'Swedish',
|
||||
),
|
||||
'sw' => array(
|
||||
'slug' => 'sw',
|
||||
'name' => 'Swahili',
|
||||
),
|
||||
'syr' => array(
|
||||
'slug' => 'syr',
|
||||
'name' => 'Syriac',
|
||||
),
|
||||
'szl' => array(
|
||||
'slug' => 'szl',
|
||||
'name' => 'Silesian',
|
||||
),
|
||||
'ta_IN' => array(
|
||||
'slug' => 'ta',
|
||||
'name' => 'Tamil',
|
||||
),
|
||||
'ta_LK' => array(
|
||||
'slug' => 'ta-lk',
|
||||
'name' => 'Tamil (Sri Lanka)',
|
||||
),
|
||||
'tah' => array(
|
||||
'slug' => 'tah',
|
||||
'name' => 'Tahitian',
|
||||
),
|
||||
'te' => array(
|
||||
'slug' => 'te',
|
||||
'name' => 'Telugu',
|
||||
),
|
||||
'tg' => array(
|
||||
'slug' => 'tg',
|
||||
'name' => 'Tajik',
|
||||
),
|
||||
'th' => array(
|
||||
'slug' => 'th',
|
||||
'name' => 'Thai',
|
||||
),
|
||||
'tir' => array(
|
||||
'slug' => 'tir',
|
||||
'name' => 'Tigrinya',
|
||||
),
|
||||
'tl' => array(
|
||||
'slug' => 'tl',
|
||||
'name' => 'Tagalog',
|
||||
),
|
||||
'tr_TR' => array(
|
||||
'slug' => 'tr',
|
||||
'name' => 'Turkish',
|
||||
),
|
||||
'tt_RU' => array(
|
||||
'slug' => 'tt',
|
||||
'name' => 'Tatar',
|
||||
),
|
||||
'tuk' => array(
|
||||
'slug' => 'tuk',
|
||||
'name' => 'Turkmen',
|
||||
),
|
||||
'twd' => array(
|
||||
'slug' => 'twd',
|
||||
'name' => 'Tweants',
|
||||
),
|
||||
'tzm' => array(
|
||||
'slug' => 'tzm',
|
||||
'name' => 'Tamazight (Central Atlas)',
|
||||
),
|
||||
'ug_CN' => array(
|
||||
'slug' => 'ug',
|
||||
'name' => 'Uighur',
|
||||
),
|
||||
'uk' => array(
|
||||
'slug' => 'uk',
|
||||
'name' => 'Ukrainian',
|
||||
),
|
||||
'ur' => array(
|
||||
'slug' => 'ur',
|
||||
'name' => 'Urdu',
|
||||
),
|
||||
'uz_UZ' => array(
|
||||
'slug' => 'uz',
|
||||
'name' => 'Uzbek',
|
||||
),
|
||||
'vi' => array(
|
||||
'slug' => 'vi',
|
||||
'name' => 'Vietnamese',
|
||||
),
|
||||
'wa' => array(
|
||||
'slug' => 'wa',
|
||||
'name' => 'Walloon',
|
||||
),
|
||||
'xho' => array(
|
||||
'slug' => 'xho',
|
||||
'name' => 'Xhosa',
|
||||
),
|
||||
'xmf' => array(
|
||||
'slug' => 'xmf',
|
||||
'name' => 'Mingrelian',
|
||||
),
|
||||
'yor' => array(
|
||||
'slug' => 'yor',
|
||||
'name' => 'Yoruba',
|
||||
),
|
||||
'zh_CN' => array(
|
||||
'slug' => 'zh-cn',
|
||||
'name' => 'Chinese (China)',
|
||||
),
|
||||
'zh_HK' => array(
|
||||
'slug' => 'zh-hk',
|
||||
'name' => 'Chinese (Hong Kong)',
|
||||
),
|
||||
'zh_TW' => array(
|
||||
'slug' => 'zh-tw',
|
||||
'name' => 'Chinese (Taiwan)',
|
||||
),
|
||||
'de_DE_formal' => array(
|
||||
'slug' => 'de/formal',
|
||||
'name' => 'German (Formal)',
|
||||
),
|
||||
'nl_NL_formal' => array(
|
||||
'slug' => 'nl/formal',
|
||||
'name' => 'Dutch (Formal)',
|
||||
),
|
||||
'de_CH_informal' => array(
|
||||
'slug' => 'de-ch/informal',
|
||||
'name' => 'Chinese (Taiwan)',
|
||||
),
|
||||
'pt_PT_ao90' => array(
|
||||
'slug' => 'pt/ao90',
|
||||
'name' => 'Portuguese (Portugal, AO90)',
|
||||
),
|
||||
);
|
||||
|
||||
/**
|
||||
* Check if we should load module for this.
|
||||
*
|
||||
* @param Product $product Product to check.
|
||||
*
|
||||
* @return bool Should load ?
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
if ( ! $product->is_wordpress_available() ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$lang = $this->get_user_locale();
|
||||
|
||||
if ( 'en_US' === $lang ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$languages = $this->get_translations( $product );
|
||||
|
||||
if ( ! is_array( $languages ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( ! isset( $languages['translations'] ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$languages = $languages['translations'];
|
||||
|
||||
$available = wp_list_pluck( $languages, 'language' );
|
||||
|
||||
if ( in_array( $lang, $available ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( ! isset( self::$locales[ $lang ] ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return apply_filters( $product->get_slug() . '_sdk_enable_translate', true );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the user's locale.
|
||||
*/
|
||||
private function get_user_locale() {
|
||||
global $wp_version;
|
||||
if ( version_compare( $wp_version, '4.7.0', '>=' ) ) {
|
||||
return get_user_locale();
|
||||
}
|
||||
$user = wp_get_current_user();
|
||||
if ( $user ) {
|
||||
$locale = $user->locale;
|
||||
}
|
||||
|
||||
return $locale ? $locale : get_locale();
|
||||
}
|
||||
|
||||
/**
|
||||
* Fetch translations from api.
|
||||
*
|
||||
* @param Product $product Product to check.
|
||||
*
|
||||
* @return mixed Translation array.
|
||||
*/
|
||||
private function get_translations( $product ) {
|
||||
$cache_key = $product->get_key() . '_all_languages';
|
||||
$translations = get_transient( $cache_key );
|
||||
|
||||
if ( false === $translations ) {
|
||||
require_once ABSPATH . 'wp-admin/includes/translation-install.php';
|
||||
$translations = translations_api(
|
||||
$product->get_type() . 's',
|
||||
array(
|
||||
'slug' => $product->get_slug(),
|
||||
'version' => $product->get_version(),
|
||||
)
|
||||
);
|
||||
set_transient( $cache_key, $translations, WEEK_IN_SECONDS );
|
||||
}
|
||||
|
||||
return $translations;
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Add notification to queue.
|
||||
*
|
||||
* @param array $all_notifications Previous notification.
|
||||
*
|
||||
* @return array All notifications.
|
||||
*/
|
||||
public function add_notification( $all_notifications ) {
|
||||
|
||||
$lang = $this->get_user_locale();
|
||||
$link = $this->get_locale_paths( $lang );
|
||||
$language_meta = self::$locales[ $lang ];
|
||||
|
||||
$heading = apply_filters( $this->product->get_key() . '_feedback_translate_heading', 'Improve {product}' );
|
||||
$heading = str_replace(
|
||||
array( '{product}' ),
|
||||
$this->product->get_friendly_name(),
|
||||
$heading
|
||||
);
|
||||
$message = apply_filters(
|
||||
$this->product->get_key() . '_feedback_translation',
|
||||
'Translating <b>{product}</b> into as many languages as possible is a huge project. We still need help with a lot of them, so if you are good at translating into <b>{language}</b>, it would be greatly appreciated.
|
||||
The process is easy, and you can join by following the link below!'
|
||||
);
|
||||
|
||||
$message = str_replace(
|
||||
[ '{product}', '{language}' ],
|
||||
[
|
||||
$this->product->get_friendly_name(),
|
||||
$language_meta['name'],
|
||||
],
|
||||
$message
|
||||
);
|
||||
|
||||
$button_submit = apply_filters( $this->product->get_key() . '_feedback_translate_button_do', 'Ok, I will gladly help.' );
|
||||
$button_cancel = apply_filters( $this->product->get_key() . '_feedback_translate_button_cancel', 'No, thanks.' );
|
||||
|
||||
$all_notifications[] = [
|
||||
'id' => $this->product->get_key() . '_translate_flag',
|
||||
'heading' => $heading,
|
||||
'message' => $message,
|
||||
'ctas' => [
|
||||
'confirm' => [
|
||||
'link' => $link,
|
||||
'text' => $button_submit,
|
||||
],
|
||||
'cancel' => [
|
||||
'link' => '#',
|
||||
'text' => $button_cancel,
|
||||
],
|
||||
],
|
||||
];
|
||||
|
||||
return $all_notifications;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return the locale path.
|
||||
*
|
||||
* @param string $locale Locale code.
|
||||
*
|
||||
* @return string Locale path.
|
||||
*/
|
||||
private function get_locale_paths( $locale ) {
|
||||
if ( empty( $locale ) ) {
|
||||
return '';
|
||||
}
|
||||
|
||||
$slug = isset( self::$locales[ $locale ] ) ? self::$locales[ $locale ]['slug'] : '';
|
||||
if ( empty( $slug ) ) {
|
||||
return '';
|
||||
}
|
||||
if ( strpos( $slug, '/' ) === false ) {
|
||||
$slug .= '/default';
|
||||
}
|
||||
$url = 'https://translate.wordpress.org/projects/wp-' . $this->product->get_type() . 's/' . $this->product->get_slug() . '/' . ( $this->product->get_type() === 'plugin' ? 'dev/' : '' ) . $slug . '?filters%5Bstatus%5D=untranslated&sort%5Bby%5D=random';
|
||||
|
||||
return $url;
|
||||
}
|
||||
|
||||
/**
|
||||
* Load module logic.
|
||||
*
|
||||
* @param Product $product Product to load.
|
||||
*
|
||||
* @return Translate Module instance.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
|
||||
$this->product = $product;
|
||||
|
||||
add_filter( 'themeisle_sdk_registered_notifications', [ $this, 'add_notification' ] );
|
||||
|
||||
return $this;
|
||||
}
|
||||
}
|
||||
247
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/Translations.php
vendored
Normal file
@@ -0,0 +1,247 @@
|
||||
<?php
|
||||
/**
|
||||
* The translations model class for ThemeIsle SDK
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2024, Bogdan Preda
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 3.3.23
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Translations module for ThemeIsle SDK.
|
||||
*/
|
||||
class Translations extends Abstract_Module {
|
||||
|
||||
const API_URL = 'https://translations.themeisle.com/wp-json/gpb-themeisle/';
|
||||
const CACHE_KEY = 'ti_translations_data';
|
||||
|
||||
/**
|
||||
* Check if we should load module for this.
|
||||
*
|
||||
* @param Product $product Product to check.
|
||||
*
|
||||
* @return bool Should load ?
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( $product->is_wordpress_available() ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return apply_filters( $product->get_slug() . '_sdk_enable_private_translations', false );
|
||||
}
|
||||
|
||||
/**
|
||||
* Load module logic.
|
||||
*
|
||||
* @param Product $product Product to load.
|
||||
*
|
||||
* @return Translations Module instance.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
|
||||
$this->product = $product;
|
||||
|
||||
if ( $this->product->is_plugin() ) {
|
||||
add_filter( 'pre_set_site_transient_update_plugins', [ $this, 'add_plugin_translations' ], 11 );
|
||||
} else {
|
||||
add_filter( 'pre_set_site_transient_update_themes', [ $this, 'add_theme_translations' ], 11 );
|
||||
}
|
||||
|
||||
add_filter( 'http_request_host_is_external', [ $this, 'allow_translations_api' ], 10, 3 );
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Allow external downloads for the translations API.
|
||||
*
|
||||
* @param bool $external Whether the host is external.
|
||||
* @param string $host The host being checked.
|
||||
* @param string $url The URL being checked.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function allow_translations_api( $external, $host, $url ) {
|
||||
return strpos( $url, self::API_URL ) === 0 ? true : $external;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get translations from API.
|
||||
*
|
||||
* @return bool | array
|
||||
*/
|
||||
private function get_api_translations() {
|
||||
$translation_data = $this->get_translation_data();
|
||||
|
||||
return empty( $translation_data ) ? false : $translation_data;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get translation data from API.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
private function get_translation_data() {
|
||||
$cached = get_transient( self::CACHE_KEY );
|
||||
|
||||
if ( $cached ) {
|
||||
return $cached;
|
||||
}
|
||||
|
||||
$response = $this->safe_get(
|
||||
self::API_URL . 'translations',
|
||||
array(
|
||||
'timeout' => 15, //phpcs:ignore WordPressVIPMinimum.Performance.RemoteRequestTimeout.timeout_timeout, Inherited by wp_remote_get only, for vip environment we use defaults.
|
||||
'sslverify' => false,
|
||||
)
|
||||
);
|
||||
if ( is_wp_error( $response ) || 200 != wp_remote_retrieve_response_code( $response ) ) {
|
||||
return [];
|
||||
}
|
||||
|
||||
$data = json_decode( wp_remote_retrieve_body( $response ) );
|
||||
|
||||
if ( ! is_array( $data ) ) {
|
||||
return [];
|
||||
}
|
||||
|
||||
set_transient( self::CACHE_KEY, $data, 12 * HOUR_IN_SECONDS );
|
||||
|
||||
return $data;
|
||||
}
|
||||
|
||||
/**
|
||||
* Add translations to the transient data.
|
||||
*
|
||||
* @param array $_transient_data incoming transient data.
|
||||
* @param string $type plugins or themes.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function add_translations( $_transient_data, $type = 'plugins' ) {
|
||||
$translations = $this->get_api_translations();
|
||||
|
||||
if ( ! is_array( $translations ) ) {
|
||||
return $_transient_data;
|
||||
}
|
||||
|
||||
if ( ! isset( $_transient_data->translations ) ) {
|
||||
return $_transient_data;
|
||||
}
|
||||
|
||||
if ( ! in_array( $type, [ 'plugins', 'themes' ] ) ) {
|
||||
return $_transient_data;
|
||||
}
|
||||
|
||||
$installed_translations = wp_get_installed_translations( $type );
|
||||
|
||||
foreach ( $translations as $translation ) {
|
||||
$translation = (array) $translation;
|
||||
|
||||
|
||||
if ( ! $this->is_valid_translation( $translation, $type ) ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$latest_translation = strtotime( $translation['updated'] );
|
||||
|
||||
if ( ! is_int( $latest_translation ) ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$existing = (int) get_option( $this->get_translation_option_key( $translation ) );
|
||||
$has_translation = isset( $installed_translations[ $translation['slug'] ][ $translation['language'] ] );
|
||||
|
||||
// If we already have the latest translation, skip.
|
||||
if ( $existing >= $latest_translation && $has_translation ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$_transient_data->translations[] = $translation;
|
||||
|
||||
update_option( $this->get_translation_option_key( $translation ), $latest_translation );
|
||||
}
|
||||
|
||||
return $_transient_data;
|
||||
}
|
||||
|
||||
/**
|
||||
* Add theme translations to the transient.
|
||||
*
|
||||
* @param array $_transient_data incoming transient data.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function add_theme_translations( $_transient_data ) {
|
||||
return $this->add_translations( $_transient_data, 'themes' );
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Add plugin translations to the transient.
|
||||
*
|
||||
* @param array $_transient_data The transient data.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function add_plugin_translations( $_transient_data ) {
|
||||
return $this->add_translations( $_transient_data );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the option key for storing translations.
|
||||
*
|
||||
* @param array $translation the translation data from the API.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
private function get_translation_option_key( $translation ) {
|
||||
return $translation['slug'] . '_translation_' . $translation['language'];
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if a translation is valid and applies for the current site.
|
||||
*
|
||||
* @param array $translation The translation data.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
private function is_valid_translation( $translation, $type = 'plugins' ) {
|
||||
if ( ! isset( $translation['slug'] ) || $translation['slug'] !== $this->product->get_slug() ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$locales = apply_filters( $type . '_update_check_locales', array_values( get_available_languages() ) );
|
||||
|
||||
if ( ! is_array( $locales ) ) {
|
||||
return false;
|
||||
}
|
||||
$locales = array_unique( $locales );
|
||||
|
||||
if ( ! isset( $translation['language'], $translation['updated'] ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( ! in_array( $translation['language'], $locales, true ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,838 @@
|
||||
<?php
|
||||
/**
|
||||
* The deactivate feedback model class for ThemeIsle SDK
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Feedback
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 1.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Loader;
|
||||
use ThemeisleSDK\Product;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Uninstall feedback module for ThemeIsle SDK.
|
||||
*/
|
||||
class Uninstall_Feedback extends Abstract_Module {
|
||||
/**
|
||||
* How many seconds before the deactivation window is triggered for themes?
|
||||
*
|
||||
* @var int Number of days.
|
||||
*/
|
||||
const AUTO_TRIGGER_DEACTIVATE_WINDOW_SECONDS = 3;
|
||||
/**
|
||||
* How many days before the deactivation window pops up again for the theme?
|
||||
*
|
||||
* @var int Number of days.
|
||||
*/
|
||||
const PAUSE_DEACTIVATE_WINDOW_DAYS = 100;
|
||||
/**
|
||||
* Where to send the data.
|
||||
*
|
||||
* @var string Endpoint url.
|
||||
*/
|
||||
const FEEDBACK_ENDPOINT = 'https://api.themeisle.com/tracking/uninstall';
|
||||
|
||||
/**
|
||||
* Default options for plugins.
|
||||
*
|
||||
* @var array $options_plugin The main options list for plugins.
|
||||
*/
|
||||
private $options_plugin = array(
|
||||
'id3' => array(
|
||||
'id' => 3,
|
||||
'type' => 'text',
|
||||
|
||||
),
|
||||
'id4' => array(
|
||||
'type' => 'textarea',
|
||||
'id' => 4,
|
||||
),
|
||||
'id5' => array(
|
||||
'id' => 5,
|
||||
'type' => 'textarea',
|
||||
),
|
||||
'id6' => array(
|
||||
'type' => 'textarea',
|
||||
'id' => 6,
|
||||
),
|
||||
);
|
||||
/**
|
||||
* Default options for theme.
|
||||
*
|
||||
* @var array $options_theme The main options list for themes.
|
||||
*/
|
||||
private $options_theme = array(
|
||||
'id7' => array(
|
||||
'id' => 7,
|
||||
),
|
||||
'id8' => array(
|
||||
'type' => 'text',
|
||||
'id' => 8,
|
||||
),
|
||||
'id9' => array(
|
||||
'id' => 9,
|
||||
'type' => 'text',
|
||||
),
|
||||
'id10' => array(
|
||||
|
||||
'title' => '',
|
||||
'id' => 10,
|
||||
),
|
||||
);
|
||||
/**
|
||||
* Default other option.
|
||||
*
|
||||
* @var array $other The other option
|
||||
*/
|
||||
private $other = array(
|
||||
'id999' => array(
|
||||
'id' => 999,
|
||||
'type' => 'textarea',
|
||||
),
|
||||
);
|
||||
|
||||
/**
|
||||
* Loads the additional resources
|
||||
*/
|
||||
public function load_resources() {
|
||||
$screen = get_current_screen();
|
||||
|
||||
if ( ! $screen || ! in_array( $screen->id, array( 'theme-install', 'plugins' ) ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$this->add_feedback_popup_style();
|
||||
|
||||
if ( $this->product->get_type() === 'theme' ) {
|
||||
$this->add_theme_feedback_drawer_js();
|
||||
$this->render_theme_feedback_popup();
|
||||
|
||||
return;
|
||||
}
|
||||
$this->add_plugin_feedback_popup_js();
|
||||
$this->render_plugin_feedback_popup();
|
||||
}
|
||||
|
||||
/**
|
||||
* Render theme feedback drawer.
|
||||
*/
|
||||
private function render_theme_feedback_popup() {
|
||||
$heading = str_replace( '{theme}', $this->product->get_name(), Loader::$labels['uninstall']['heading_theme'] );
|
||||
$button_submit = apply_filters( $this->product->get_key() . '_feedback_deactivate_button_submit', Loader::$labels['uninstall']['submit'] );
|
||||
$options = $this->options_theme;
|
||||
$options = $this->randomize_options( apply_filters( $this->product->get_key() . '_feedback_deactivate_options', $options ) );
|
||||
$info_disclosure_link = '<a href="#" class="info-disclosure-link">' . apply_filters( $this->product->get_slug() . '_themeisle_sdk_info_collect_cta', Loader::$labels['uninstall']['cta_info'] ) . '</a>';
|
||||
|
||||
$options += $this->other;
|
||||
|
||||
?>
|
||||
<div class="ti-theme-uninstall-feedback-drawer ti-feedback">
|
||||
<div class="popup--header">
|
||||
<h5><?php echo wp_kses( $heading, array( 'span' => true ) ); ?> </h5>
|
||||
<button class="toggle"><span>×</span></button>
|
||||
</div><!--/.popup--header-->
|
||||
<div class="popup--body">
|
||||
<?php $this->render_options_list( $options ); ?>
|
||||
</div><!--/.popup--body-->
|
||||
<div class="popup--footer">
|
||||
<div class="actions">
|
||||
<?php
|
||||
echo wp_kses_post( $info_disclosure_link );
|
||||
echo wp_kses_post( $this->get_disclosure_labels() );
|
||||
echo '<div class="buttons">';
|
||||
echo get_submit_button( //phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped, Function has an internal sanitization.
|
||||
$button_submit,
|
||||
'secondary',
|
||||
$this->product->get_key() . 'ti-deactivate-yes',
|
||||
false,
|
||||
array(
|
||||
'data-after-text' => $button_submit,
|
||||
'disabled' => true,
|
||||
)
|
||||
);
|
||||
echo '</div>';
|
||||
?>
|
||||
</div><!--/.actions-->
|
||||
</div><!--/.popup--footer-->
|
||||
</div>
|
||||
<?php
|
||||
}
|
||||
|
||||
/**
|
||||
* Add feedback styles.
|
||||
*/
|
||||
private function add_feedback_popup_style() {
|
||||
?>
|
||||
<style>
|
||||
.ti-feedback {
|
||||
background: #fff;
|
||||
max-width: 400px;
|
||||
z-index: 10000;
|
||||
box-shadow: 0 0 15px -5px rgba(0, 0, 0, .5);
|
||||
transition: all .3s ease-out;
|
||||
}
|
||||
|
||||
|
||||
.ti-feedback .popup--header {
|
||||
position: relative;
|
||||
background-color: #23A1CE;
|
||||
}
|
||||
|
||||
.ti-feedback .popup--header h5 {
|
||||
margin: 0;
|
||||
font-size: 16px;
|
||||
padding: 15px;
|
||||
color: #fff;
|
||||
font-weight: 600;
|
||||
text-align: center;
|
||||
letter-spacing: .3px;
|
||||
}
|
||||
|
||||
.ti-feedback .popup--body {
|
||||
padding: 15px;
|
||||
}
|
||||
|
||||
.ti-feedback .popup--form {
|
||||
margin: 0;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.ti-feedback .popup--form input[type="radio"] {
|
||||
<?php echo is_rtl() ? 'margin: 0 0 0 10px;' : 'margin: 0 10px 0 0;'; ?>
|
||||
}
|
||||
|
||||
.ti-feedback .popup--form input[type="radio"]:checked ~ textarea {
|
||||
display: block;
|
||||
}
|
||||
|
||||
.ti-feedback .popup--form textarea {
|
||||
width: 100%;
|
||||
margin: 10px 0 0;
|
||||
display: none;
|
||||
max-height: 150px;
|
||||
}
|
||||
|
||||
.ti-feedback li {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
margin-bottom: 15px;
|
||||
flex-wrap: wrap;
|
||||
}
|
||||
|
||||
.ti-feedback li label {
|
||||
max-width: 90%;
|
||||
}
|
||||
|
||||
.ti-feedback li:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
.ti-feedback .popup--footer {
|
||||
padding: 0 15px 15px;
|
||||
}
|
||||
|
||||
.ti-feedback .actions {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
}
|
||||
|
||||
.info-disclosure-link {
|
||||
width: 100%;
|
||||
margin-bottom: 15px;
|
||||
}
|
||||
|
||||
.ti-feedback .info-disclosure-content {
|
||||
max-height: 0;
|
||||
overflow: hidden;
|
||||
width: 100%;
|
||||
transition: .3s ease;
|
||||
}
|
||||
|
||||
.ti-feedback .info-disclosure-content.active {
|
||||
max-height: 300px;
|
||||
}
|
||||
|
||||
.ti-feedback .info-disclosure-content p {
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.ti-feedback .info-disclosure-content ul {
|
||||
margin: 10px 0;
|
||||
border-radius: 3px;
|
||||
}
|
||||
|
||||
.ti-feedback .info-disclosure-content ul li {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
margin-bottom: 0;
|
||||
padding: 5px 0;
|
||||
border-bottom: 1px solid #ccc;
|
||||
}
|
||||
|
||||
.ti-feedback .buttons {
|
||||
display: flex;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.ti-feedback .buttons input:last-child {
|
||||
<?php echo is_rtl() ? 'margin-right: auto;' : 'margin-left: auto;'; ?>
|
||||
}
|
||||
|
||||
.ti-theme-uninstall-feedback-drawer {
|
||||
border-top-left-radius: 5px;
|
||||
position: fixed;
|
||||
top: 100%;
|
||||
right: 15px;
|
||||
}
|
||||
|
||||
.ti-theme-uninstall-feedback-drawer.active {
|
||||
transform: translateY(-100%);
|
||||
}
|
||||
|
||||
.ti-theme-uninstall-feedback-drawer .popup--header {
|
||||
border-top-left-radius: 5px;
|
||||
}
|
||||
|
||||
.ti-theme-uninstall-feedback-drawer .popup--header .toggle {
|
||||
position: absolute;
|
||||
padding: 3px 0;
|
||||
width: 30px;
|
||||
top: -26px;
|
||||
right: 0;
|
||||
cursor: pointer;
|
||||
border-top-left-radius: 5px;
|
||||
border-top-right-radius: 5px;
|
||||
font-size: 20px;
|
||||
background-color: #23A1CE;
|
||||
color: #fff;
|
||||
border: none;
|
||||
line-height: 20px;
|
||||
}
|
||||
|
||||
.ti-theme-uninstall-feedback-drawer .toggle span {
|
||||
margin: 0;
|
||||
display: inline-block;
|
||||
}
|
||||
|
||||
.ti-theme-uninstall-feedback-drawer:not(.active) .toggle span {
|
||||
transform: rotate(45deg);
|
||||
}
|
||||
|
||||
.ti-theme-uninstall-feedback-drawer .popup--header .toggle:hover {
|
||||
background-color: #1880a5;
|
||||
}
|
||||
|
||||
|
||||
.ti-plugin-uninstall-feedback-popup .popup--header:before {
|
||||
content: "";
|
||||
display: block;
|
||||
position: absolute;
|
||||
top: 50%;
|
||||
transform: translateY(-50%);
|
||||
<?php
|
||||
echo is_rtl() ?
|
||||
'right: -10px;
|
||||
border-top: 20px solid transparent;
|
||||
border-left: 20px solid #23A1CE;
|
||||
border-bottom: 20px solid transparent;' :
|
||||
'left: -10px;
|
||||
border-top: 20px solid transparent;
|
||||
border-right: 20px solid #23A1CE;
|
||||
border-bottom: 20px solid transparent;';
|
||||
?>
|
||||
}
|
||||
|
||||
.ti-plugin-uninstall-feedback-popup {
|
||||
display: none;
|
||||
position: absolute;
|
||||
white-space: normal;
|
||||
width: 400px;
|
||||
<?php echo is_rtl() ? 'right: calc( 100% + 15px );' : 'left: calc( 100% + 15px );'; ?> top: -15px;
|
||||
}
|
||||
|
||||
.ti-plugin-uninstall-feedback-popup.sending-feedback .popup--body i {
|
||||
animation: rotation 2s infinite linear;
|
||||
display: block;
|
||||
float: none;
|
||||
align-items: center;
|
||||
width: 100%;
|
||||
margin: 0 auto;
|
||||
height: 100%;
|
||||
background: transparent;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
.ti-plugin-uninstall-feedback-popup.sending-feedback .popup--body i:before {
|
||||
padding: 0;
|
||||
background: transparent;
|
||||
box-shadow: none;
|
||||
color: #b4b9be
|
||||
}
|
||||
|
||||
|
||||
.ti-plugin-uninstall-feedback-popup.active {
|
||||
display: block;
|
||||
}
|
||||
|
||||
tr[data-plugin^="<?php echo esc_attr( $this->product->get_slug() ); ?>"] .deactivate {
|
||||
position: relative;
|
||||
}
|
||||
|
||||
body.ti-feedback-open .ti-feedback-overlay {
|
||||
content: "";
|
||||
display: block;
|
||||
background-color: rgba(0, 0, 0, 0.5);
|
||||
top: 0;
|
||||
bottom: 0;
|
||||
right: 0;
|
||||
left: 0;
|
||||
z-index: 10000;
|
||||
position: fixed;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.ti-plugin-uninstall-feedback-popup {
|
||||
position: fixed;
|
||||
max-width: 100%;
|
||||
margin: 0 auto;
|
||||
left: 50%;
|
||||
top: 50px;
|
||||
transform: translateX(-50%);
|
||||
}
|
||||
|
||||
.ti-plugin-uninstall-feedback-popup .popup--header:before {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
<?php
|
||||
}
|
||||
|
||||
/**
|
||||
* Theme feedback drawer JS.
|
||||
*/
|
||||
private function add_theme_feedback_drawer_js() {
|
||||
$key = $this->product->get_key();
|
||||
?>
|
||||
<script type="text/javascript" id="ti-deactivate-js">
|
||||
(function ($) {
|
||||
$(document).ready(function () {
|
||||
setTimeout(function () {
|
||||
$('.ti-theme-uninstall-feedback-drawer').addClass('active');
|
||||
}, <?php echo absint( self::AUTO_TRIGGER_DEACTIVATE_WINDOW_SECONDS * 1000 ); ?> );
|
||||
|
||||
$('.ti-theme-uninstall-feedback-drawer .toggle').on('click', function (e) {
|
||||
e.preventDefault();
|
||||
$('.ti-theme-uninstall-feedback-drawer').toggleClass('active');
|
||||
});
|
||||
|
||||
$('.info-disclosure-link').on('click', function (e) {
|
||||
e.preventDefault();
|
||||
$('.info-disclosure-content').toggleClass('active');
|
||||
});
|
||||
|
||||
$('.ti-theme-uninstall-feedback-drawer input[type="radio"]').on('change', function () {
|
||||
var radio = $(this);
|
||||
if (radio.parent().find('textarea').length > 0 &&
|
||||
radio.parent().find('textarea').val().length === 0) {
|
||||
$('#<?php echo esc_attr( $key ); ?>ti-deactivate-yes').attr('disabled', 'disabled');
|
||||
radio.parent().find('textarea').on('keyup', function (e) {
|
||||
if ($(this).val().length === 0) {
|
||||
$('#<?php echo esc_attr( $key ); ?>ti-deactivate-yes').attr('disabled', 'disabled');
|
||||
} else {
|
||||
$('#<?php echo esc_attr( $key ); ?>ti-deactivate-yes').removeAttr('disabled');
|
||||
}
|
||||
});
|
||||
} else {
|
||||
$('#<?php echo esc_attr( $key ); ?>ti-deactivate-yes').removeAttr('disabled');
|
||||
}
|
||||
});
|
||||
|
||||
$('#<?php echo esc_attr( $key ); ?>ti-deactivate-yes').on('click', function (e) {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
|
||||
var selectedOption = $(
|
||||
'.ti-theme-uninstall-feedback-drawer input[name="ti-deactivate-option"]:checked');
|
||||
$.post(ajaxurl, {
|
||||
'action': '<?php echo esc_attr( $key ) . '_uninstall_feedback'; ?>',
|
||||
'nonce': '<?php echo esc_attr( wp_create_nonce( (string) __CLASS__ ) ); ?>',
|
||||
'id': selectedOption.parent().attr('ti-option-id'),
|
||||
'msg': selectedOption.parent().find('textarea').val(),
|
||||
'type': 'theme',
|
||||
'key': '<?php echo esc_attr( $key ); ?>'
|
||||
});
|
||||
$('.ti-theme-uninstall-feedback-drawer').fadeOut();
|
||||
});
|
||||
});
|
||||
})(jQuery);
|
||||
|
||||
</script>
|
||||
<?php
|
||||
do_action( $this->product->get_key() . '_uninstall_feedback_after_js' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Render the options list.
|
||||
*
|
||||
* @param array $options the options for the feedback form.
|
||||
*/
|
||||
private function render_options_list( $options ) {
|
||||
$key = $this->product->get_key();
|
||||
$inputs_row_map = [
|
||||
'text' => 1,
|
||||
'textarea' => 2,
|
||||
];
|
||||
?>
|
||||
<ul class="popup--form">
|
||||
<?php
|
||||
foreach ( $options as $idx => $attributes ) {
|
||||
$title = Loader::$labels['uninstall']['options'][ $idx ]['title'];
|
||||
$placeholder = array_key_exists( 'placeholder', Loader::$labels['uninstall']['options'][ $idx ] ) ? Loader::$labels['uninstall']['options'][ $idx ]['placeholder'] : '';
|
||||
?>
|
||||
<li ti-option-id="<?php echo esc_attr( $attributes['id'] ); ?>">
|
||||
<input type="radio" name="ti-deactivate-option"
|
||||
id="<?php echo esc_attr( $key . $attributes['id'] ); ?>">
|
||||
<label for="<?php echo esc_attr( $key . $attributes['id'] ); ?>">
|
||||
<?php echo esc_attr( str_replace( '{theme}', $this->product->get_name(), $title ) ); ?>
|
||||
</label>
|
||||
<?php
|
||||
if ( array_key_exists( 'type', $attributes ) ) {
|
||||
|
||||
echo '<textarea width="100%" rows="' . esc_attr( $inputs_row_map[ $attributes['type'] ] ) . '" name="comments" placeholder="' . esc_attr( $placeholder ) . '"></textarea>';
|
||||
}
|
||||
?>
|
||||
</li>
|
||||
<?php } ?>
|
||||
</ul>
|
||||
<?php
|
||||
}
|
||||
|
||||
/**
|
||||
* Render plugin feedback popup.
|
||||
*/
|
||||
private function render_plugin_feedback_popup() {
|
||||
$button_cancel = apply_filters( $this->product->get_key() . '_feedback_deactivate_button_cancel', Loader::$labels['uninstall']['button_cancel'] );
|
||||
$button_submit = apply_filters( $this->product->get_key() . '_feedback_deactivate_button_submit', Loader::$labels['uninstall']['button_submit'] );
|
||||
$options = $this->randomize_options( apply_filters( $this->product->get_key() . '_feedback_deactivate_options', $this->options_plugin ) );
|
||||
$info_disclosure_link = '<a href="#" class="info-disclosure-link">' . apply_filters( $this->product->get_slug() . '_themeisle_sdk_info_collect_cta', Loader::$labels['uninstall']['cta_info'] ) . '</a>';
|
||||
|
||||
$options += $this->other;
|
||||
?>
|
||||
<div class="ti-plugin-uninstall-feedback-popup ti-feedback"
|
||||
id="<?php echo esc_attr( $this->product->get_slug() . '_uninstall_feedback_popup' ); ?>">
|
||||
<div class="popup--header">
|
||||
<h5><?php echo wp_kses( Loader::$labels['uninstall']['heading_plugin'], array( 'span' => true ) ); ?> </h5>
|
||||
</div><!--/.popup--header-->
|
||||
<div class="popup--body">
|
||||
<?php $this->render_options_list( $options ); ?>
|
||||
</div><!--/.popup--body-->
|
||||
<div class="popup--footer">
|
||||
<div class="actions">
|
||||
<?php
|
||||
echo wp_kses_post( $info_disclosure_link );
|
||||
echo wp_kses_post( $this->get_disclosure_labels() );
|
||||
echo '<div class="buttons">';
|
||||
echo get_submit_button( //phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped, Function internals are escaped.
|
||||
$button_cancel,
|
||||
'secondary',
|
||||
$this->product->get_key() . 'ti-deactivate-no',
|
||||
false
|
||||
);
|
||||
echo get_submit_button( //phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped, Function internals are escaped.
|
||||
$button_submit,
|
||||
'primary',
|
||||
$this->product->get_key() . 'ti-deactivate-yes',
|
||||
false,
|
||||
array(
|
||||
'data-after-text' => $button_submit,
|
||||
'disabled' => true,
|
||||
)
|
||||
);
|
||||
echo '</div>';
|
||||
?>
|
||||
</div><!--/.actions-->
|
||||
</div><!--/.popup--footer-->
|
||||
</div>
|
||||
|
||||
<?php
|
||||
}
|
||||
|
||||
/**
|
||||
* Add plugin feedback popup JS
|
||||
*/
|
||||
private function add_plugin_feedback_popup_js() {
|
||||
$popup_id = '#' . $this->product->get_slug() . '_uninstall_feedback_popup';
|
||||
$key = $this->product->get_key();
|
||||
?>
|
||||
<script type="text/javascript" id="ti-deactivate-js">
|
||||
(function ($) {
|
||||
$(document).ready(function () {
|
||||
var targetElement = 'tr[data-plugin^="<?php echo esc_attr( $this->product->get_slug() ); ?>/"] span.deactivate a';
|
||||
var redirectUrl = $(targetElement).attr('href');
|
||||
if ($('.ti-feedback-overlay').length === 0) {
|
||||
$('body').prepend('<div class="ti-feedback-overlay"></div>');
|
||||
}
|
||||
$('<?php echo esc_attr( $popup_id ); ?> ').appendTo($(targetElement).parent());
|
||||
|
||||
$(targetElement).on('click', function (e) {
|
||||
e.preventDefault();
|
||||
$('<?php echo esc_attr( $popup_id ); ?> ').addClass('active');
|
||||
$('body').addClass('ti-feedback-open');
|
||||
$('.ti-feedback-overlay').on('click', function () {
|
||||
$('<?php echo esc_attr( $popup_id ); ?> ').removeClass('active');
|
||||
$('body').removeClass('ti-feedback-open');
|
||||
});
|
||||
});
|
||||
|
||||
$('<?php echo esc_attr( $popup_id ); ?> .info-disclosure-link').on('click', function (e) {
|
||||
e.preventDefault();
|
||||
$(this).parent().find('.info-disclosure-content').toggleClass('active');
|
||||
});
|
||||
|
||||
$('<?php echo esc_attr( $popup_id ); ?> input[type="radio"]').on('change', function () {
|
||||
var radio = $(this);
|
||||
if (radio.parent().find('textarea').length > 0 &&
|
||||
radio.parent().find('textarea').val().length === 0) {
|
||||
$('<?php echo esc_attr( $popup_id ); ?> #<?php echo esc_attr( $key ); ?>ti-deactivate-yes').attr('disabled', 'disabled');
|
||||
radio.parent().find('textarea').on('keyup', function (e) {
|
||||
if ($(this).val().length === 0) {
|
||||
$('<?php echo esc_attr( $popup_id ); ?> #<?php echo esc_attr( $key ); ?>ti-deactivate-yes').attr('disabled', 'disabled');
|
||||
} else {
|
||||
$('<?php echo esc_attr( $popup_id ); ?> #<?php echo esc_attr( $key ); ?>ti-deactivate-yes').removeAttr('disabled');
|
||||
}
|
||||
});
|
||||
} else {
|
||||
$('<?php echo esc_attr( $popup_id ); ?> #<?php echo esc_attr( $key ); ?>ti-deactivate-yes').removeAttr('disabled');
|
||||
}
|
||||
});
|
||||
|
||||
$('<?php echo esc_attr( $popup_id ); ?> #<?php echo esc_attr( $key ); ?>ti-deactivate-no').on('click', function (e) {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
$(targetElement).unbind('click');
|
||||
$('body').removeClass('ti-feedback-open');
|
||||
$('<?php echo esc_attr( $popup_id ); ?>').remove();
|
||||
if (redirectUrl !== '') {
|
||||
location.href = redirectUrl;
|
||||
}
|
||||
});
|
||||
|
||||
$('<?php echo esc_attr( $popup_id ); ?> #<?php echo esc_attr( $key ); ?>ti-deactivate-yes').on('click', function (e) {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
$(targetElement).unbind('click');
|
||||
var selectedOption = $(
|
||||
'<?php echo esc_attr( $popup_id ); ?> input[name="ti-deactivate-option"]:checked');
|
||||
var data = {
|
||||
'action': '<?php echo esc_attr( $key ) . '_uninstall_feedback'; ?>',
|
||||
'nonce': '<?php echo esc_attr( wp_create_nonce( (string) __CLASS__ ) ); ?>',
|
||||
'id': selectedOption.parent().attr('ti-option-id'),
|
||||
'msg': selectedOption.parent().find('textarea').val(),
|
||||
'type': 'plugin',
|
||||
'key': '<?php echo esc_attr( $key ); ?>'
|
||||
};
|
||||
$.ajax({
|
||||
type: 'POST',
|
||||
url: ajaxurl,
|
||||
data: data,
|
||||
complete() {
|
||||
$('body').removeClass('ti-feedback-open');
|
||||
$('<?php echo esc_attr( $popup_id ); ?>').remove();
|
||||
if (redirectUrl !== '') {
|
||||
location.href = redirectUrl;
|
||||
}
|
||||
},
|
||||
beforeSend() {
|
||||
$('<?php echo esc_attr( $popup_id ); ?>').addClass('sending-feedback');
|
||||
$('<?php echo esc_attr( $popup_id ); ?> .popup--footer').remove();
|
||||
$('<?php echo esc_attr( $popup_id ); ?> .popup--body').html('<i class="dashicons dashicons-update-alt"></i>');
|
||||
}
|
||||
});
|
||||
});
|
||||
});
|
||||
})(jQuery);
|
||||
|
||||
</script>
|
||||
<?php
|
||||
do_action( $this->product->get_key() . '_uninstall_feedback_after_js' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the disclosure labels markup.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
private function get_disclosure_labels() {
|
||||
$disclosure_new_labels = apply_filters( $this->product->get_slug() . '_themeisle_sdk_disclosure_content_labels', [], $this->product );
|
||||
$disclosure_labels = array_merge(
|
||||
[
|
||||
'title' => Loader::$labels['uninstall']['disclosure']['title'],
|
||||
'items' => [
|
||||
sprintf( Loader::$labels['uninstall']['disclosure']['version'], '<strong>', ucwords( $this->product->get_type() ), '</strong>', '<code>', $this->product->get_version(), '</code>' ),
|
||||
sprintf( Loader::$labels['uninstall']['disclosure']['website'], '<strong>', '</strong>', '<code>', get_site_url(), '</code>' ),
|
||||
sprintf( Loader::$labels['uninstall']['disclosure']['usage'], '<strong>', '</strong>', '<code>', ( time() - $this->product->get_install_time() ), 's</code>' ),
|
||||
sprintf( Loader::$labels['uninstall']['disclosure']['reason'], '<strong>', '</strong>', '<i>', '</i>' ),
|
||||
],
|
||||
],
|
||||
$disclosure_new_labels
|
||||
);
|
||||
|
||||
$info_disclosure_content = '<div class="info-disclosure-content"><p>' . wp_kses_post( $disclosure_labels['title'] ) . '</p><ul>';
|
||||
foreach ( $disclosure_labels['items'] as $disclosure_item ) {
|
||||
$info_disclosure_content .= sprintf( '<li>%s</li>', wp_kses_post( $disclosure_item ) );
|
||||
}
|
||||
$info_disclosure_content .= '</ul></div>';
|
||||
|
||||
return $info_disclosure_content;
|
||||
}
|
||||
|
||||
/**
|
||||
* Randomizes the options array.
|
||||
*
|
||||
* @param array $options The options array.
|
||||
*/
|
||||
public function randomize_options( $options ) {
|
||||
$new = array();
|
||||
$keys = array_keys( $options );
|
||||
shuffle( $keys );
|
||||
|
||||
foreach ( $keys as $key ) {
|
||||
$new[ $key ] = $options[ $key ];
|
||||
}
|
||||
|
||||
return $new;
|
||||
}
|
||||
|
||||
/**
|
||||
* Called when the deactivate button is clicked.
|
||||
*/
|
||||
public function post_deactivate() {
|
||||
check_ajax_referer( (string) __CLASS__, 'nonce' );
|
||||
|
||||
$this->post_deactivate_or_cancel();
|
||||
|
||||
if ( empty( $_POST['id'] ) ) {
|
||||
|
||||
wp_send_json( [] );
|
||||
|
||||
return;
|
||||
}
|
||||
$this->call_api(
|
||||
array(
|
||||
'type' => 'deactivate',
|
||||
'id' => sanitize_key( $_POST['id'] ),
|
||||
'comment' => isset( $_POST['msg'] ) ? sanitize_textarea_field( $_POST['msg'] ) : '',
|
||||
)
|
||||
);
|
||||
wp_send_json( [] );
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Called when the deactivate/cancel button is clicked.
|
||||
*/
|
||||
private function post_deactivate_or_cancel() {
|
||||
if ( ! isset( $_POST['type'] ) || ! isset( $_POST['key'] ) ) { //phpcs:ignore WordPress.Security.NonceVerification.Missing, Nonce already present in caller function.
|
||||
return;
|
||||
}
|
||||
if ( 'theme' !== $_POST['type'] ) { //phpcs:ignore WordPress.Security.NonceVerification.Missing, Nonce already present in caller function.
|
||||
return;
|
||||
}
|
||||
|
||||
set_transient( 'ti_sdk_pause_' . sanitize_text_field( $_POST['key'] ), true, self::PAUSE_DEACTIVATE_WINDOW_DAYS * DAY_IN_SECONDS );//phpcs:ignore WordPress.Security.NonceVerification.Missing, Nonce already present in caller function.
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Calls the API
|
||||
*
|
||||
* @param array $attributes The attributes of the post body.
|
||||
*
|
||||
* @return bool Is the request succesfull?
|
||||
*/
|
||||
protected function call_api( $attributes ) {
|
||||
$slug = $this->product->get_slug();
|
||||
$version = $this->product->get_version();
|
||||
$attributes['slug'] = $slug;
|
||||
$attributes['version'] = $version;
|
||||
$attributes['url'] = get_site_url();
|
||||
$attributes['active_time'] = ( time() - $this->product->get_install_time() );
|
||||
|
||||
$response = wp_remote_post(
|
||||
self::FEEDBACK_ENDPOINT,
|
||||
array(
|
||||
'body' => $attributes,
|
||||
)
|
||||
);
|
||||
|
||||
return is_wp_error( $response );
|
||||
}
|
||||
|
||||
/**
|
||||
* Should we load this object?.
|
||||
*
|
||||
* @param Product $product Product object.
|
||||
*
|
||||
* @return bool Should we load the module?
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
if ( $this->is_from_partner( $product ) ) {
|
||||
return false;
|
||||
}
|
||||
if ( $product->is_theme() && ( false !== get_transient( 'ti_sdk_pause_' . $product->get_key(), false ) ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( defined( 'DOING_AJAX' ) && DOING_AJAX ) {
|
||||
return true;
|
||||
}
|
||||
global $pagenow;
|
||||
|
||||
if ( ! isset( $pagenow ) || empty( $pagenow ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( $product->is_plugin() && 'plugins.php' !== $pagenow ) {
|
||||
return false;
|
||||
|
||||
}
|
||||
if ( $product->is_theme() && 'theme-install.php' !== $pagenow ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Loads module hooks.
|
||||
*
|
||||
* @param Product $product Product details.
|
||||
*
|
||||
* @return Uninstall_Feedback Current module instance.
|
||||
*/
|
||||
public function load( $product ) {
|
||||
|
||||
if ( apply_filters( $product->get_key() . '_hide_uninstall_feedback', false ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$this->product = $product;
|
||||
add_action( 'admin_head', array( $this, 'load_resources' ) );
|
||||
add_action( 'wp_ajax_' . $this->product->get_key() . '_uninstall_feedback', array( $this, 'post_deactivate' ) );
|
||||
|
||||
return $this;
|
||||
}
|
||||
}
|
||||
194
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Modules/Welcome.php
vendored
Normal file
@@ -0,0 +1,194 @@
|
||||
<?php
|
||||
/**
|
||||
* The welcome model class for ThemeIsle SDK
|
||||
*
|
||||
* Here's how to hook it in your plugin or theme:
|
||||
* ```php
|
||||
* add_filter( '<product_slug>_welcome_metadata', function() {
|
||||
* return [
|
||||
* 'is_enabled' => <condition_if_pro_available>,
|
||||
* 'pro_name' => 'Product PRO name',
|
||||
* 'logo' => '<path_to_logo>',
|
||||
* 'cta_link' => tsdk_utmify( 'https://link_to_upgrade.with/?discount=<discountCode>')
|
||||
* ];
|
||||
* } );
|
||||
* ```
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Modules
|
||||
* @copyright Copyright (c) 2023, Bogdan Preda
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 1.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK\Modules;
|
||||
|
||||
// Exit if accessed directly.
|
||||
use ThemeisleSDK\Common\Abstract_Module;
|
||||
use ThemeisleSDK\Loader;
|
||||
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Promotions module for ThemeIsle SDK.
|
||||
*/
|
||||
class Welcome extends Abstract_Module {
|
||||
|
||||
/**
|
||||
* Debug mode.
|
||||
*
|
||||
* @var bool
|
||||
*/
|
||||
private $debug = false;
|
||||
|
||||
/**
|
||||
* Welcome metadata.
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
private $welcome_discounts = array();
|
||||
|
||||
/**
|
||||
* Check that we can load this module.
|
||||
*
|
||||
* @param \ThemeisleSDK\Product $product The product.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function can_load( $product ) {
|
||||
$this->debug = apply_filters( 'themeisle_sdk_welcome_debug', $this->debug );
|
||||
$welcome_metadata = apply_filters( $product->get_key() . '_welcome_metadata', array() );
|
||||
|
||||
$is_welcome_enabled = $this->is_welcome_meta_valid( $welcome_metadata );
|
||||
|
||||
if ( $is_welcome_enabled ) {
|
||||
$this->welcome_discounts[ $product->get_key() ] = $welcome_metadata;
|
||||
}
|
||||
|
||||
return $this->debug || $is_welcome_enabled;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check that the metadata is valid and the welcome is enabled.
|
||||
*
|
||||
* @param array $welcome_metadata The metadata to validate.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
private function is_welcome_meta_valid( $welcome_metadata ) {
|
||||
return ! empty( $welcome_metadata ) && isset( $welcome_metadata['is_enabled'] ) && $welcome_metadata['is_enabled'];
|
||||
}
|
||||
|
||||
/**
|
||||
* Load the module.
|
||||
*
|
||||
* @param \ThemeisleSDK\Product $product The product.
|
||||
*
|
||||
* @return $this
|
||||
*/
|
||||
public function load( $product ) {
|
||||
if ( ! current_user_can( 'install_plugins' ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$this->product = $product;
|
||||
if ( ! $this->is_time_to_show_welcome() && $this->debug === false ) {
|
||||
return;
|
||||
}
|
||||
|
||||
add_filter( 'themeisle_sdk_registered_notifications', [ $this, 'add_notification' ], 99, 1 );
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if it's time to show the welcome.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
private function is_time_to_show_welcome() {
|
||||
// if 7 days from install have not passed, don't show the welcome.
|
||||
if ( $this->product->get_install_time() + 7 * DAY_IN_SECONDS > time() ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// if 12 days from install have passed, don't show the welcome ( after 7 days for 5 days ).
|
||||
if ( $this->product->get_install_time() + 12 * DAY_IN_SECONDS < time() ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Add the welcome notification.
|
||||
* Will block all other notifications if a welcome notification is present.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function add_notification( $all_notifications ) {
|
||||
if ( empty( $this->welcome_discounts ) ) {
|
||||
return $all_notifications;
|
||||
}
|
||||
|
||||
if ( ! isset( $this->welcome_discounts[ $this->product->get_key() ] ) ) {
|
||||
return $all_notifications;
|
||||
}
|
||||
|
||||
// filter out the notifications that are not welcome upsells
|
||||
// if we arrived here we will have at least one welcome upsell
|
||||
$all_notifications = array_filter(
|
||||
$all_notifications,
|
||||
function ( $notification ) {
|
||||
return strpos( $notification['id'], '_welcome_upsell_flag' ) !== false;
|
||||
}
|
||||
);
|
||||
|
||||
$offer = $this->welcome_discounts[ $this->product->get_key() ];
|
||||
|
||||
$response = [];
|
||||
$logo = isset( $offer['logo'] ) ? $offer['logo'] : '';
|
||||
$pro_name = isset( $offer['pro_name'] ) ? $offer['pro_name'] : $this->product->get_friendly_name() . ' PRO';
|
||||
|
||||
$link = $offer['cta_link'];
|
||||
|
||||
$message = apply_filters( $this->product->get_key() . '_welcome_upsell_message', Loader::$labels['welcome']['message'] );
|
||||
|
||||
$button_submit = apply_filters( $this->product->get_key() . '_feedback_review_button_do', Loader::$labels['welcome']['ctay'] );
|
||||
$button_cancel = apply_filters( $this->product->get_key() . '_feedback_review_button_cancel', Loader::$labels['welcome']['ctan'] );
|
||||
$message = str_replace(
|
||||
[ '{product}', '{pro_product}', '{cta_link}' ],
|
||||
[
|
||||
$this->product->get_friendly_name(),
|
||||
$pro_name,
|
||||
$link,
|
||||
],
|
||||
$message
|
||||
);
|
||||
|
||||
$all_notifications[] = [
|
||||
'id' => $this->product->get_key() . '_welcome_upsell_flag',
|
||||
'message' => $message,
|
||||
'img_src' => $logo,
|
||||
'ctas' => [
|
||||
'confirm' => [
|
||||
'link' => $link,
|
||||
'text' => $button_submit,
|
||||
],
|
||||
'cancel' => [
|
||||
'link' => '#',
|
||||
'text' => $button_cancel,
|
||||
],
|
||||
],
|
||||
'type' => 'info',
|
||||
];
|
||||
|
||||
$key = array_rand( $all_notifications );
|
||||
$response[] = $all_notifications[ $key ];
|
||||
|
||||
return $response;
|
||||
}
|
||||
|
||||
}
|
||||
497
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/src/Product.php
vendored
Normal file
@@ -0,0 +1,497 @@
|
||||
<?php
|
||||
/**
|
||||
* The product model class for ThemeIsle SDK
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @subpackage Product
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-3.0.php GNU Public License
|
||||
* @since 1.0.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK;
|
||||
|
||||
// Exit if accessed directly.
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Product model for ThemeIsle SDK.
|
||||
*/
|
||||
class Product {
|
||||
/**
|
||||
* Define plugin type string.
|
||||
*/
|
||||
const PLUGIN_TYPE = 'plugin';
|
||||
/**
|
||||
* Define theme type string.
|
||||
*/
|
||||
const THEME_TYPE = 'theme';
|
||||
/**
|
||||
* If the product has a pro version, contains the pro slug.
|
||||
*
|
||||
* @var string $pro_slug Pro slug, if available.
|
||||
*/
|
||||
public $pro_slug;
|
||||
/**
|
||||
* Current product slug.
|
||||
*
|
||||
* @var string $slug THe product slug.
|
||||
*/
|
||||
private $slug;
|
||||
/**
|
||||
* Product basefile, with the proper metadata.
|
||||
*
|
||||
* @var string $basefile The file with headers.
|
||||
*/
|
||||
private $basefile;
|
||||
/**
|
||||
* Type of the product.
|
||||
*
|
||||
* @var string $type The product type ( plugin | theme ).
|
||||
*/
|
||||
private $type;
|
||||
/**
|
||||
* The file name.
|
||||
*
|
||||
* @var string $file The file name.
|
||||
*/
|
||||
private $file;
|
||||
/**
|
||||
* Product name, fetched from the file headers.
|
||||
*
|
||||
* @var string $name The product name.
|
||||
*/
|
||||
private $name;
|
||||
/**
|
||||
* Product normalized key.
|
||||
*
|
||||
* @var string $key The product ready key.
|
||||
*/
|
||||
private $key;
|
||||
/**
|
||||
* Author URL
|
||||
*
|
||||
* @var string $author_url The author url.
|
||||
*/
|
||||
private $author_url;
|
||||
/**
|
||||
* Product store url.
|
||||
*
|
||||
* @var string $store_url The store url.
|
||||
*/
|
||||
private $store_url;
|
||||
/**
|
||||
* Product install timestamp.
|
||||
*
|
||||
* @var int $install The date of install.
|
||||
*/
|
||||
private $install;
|
||||
/**
|
||||
* Product store/author name.
|
||||
*
|
||||
* @var string $store_name The store name.
|
||||
*/
|
||||
private $store_name;
|
||||
/**
|
||||
* Does the product requires license.
|
||||
*
|
||||
* @var bool $requires_license Either user needs to activate it with license.
|
||||
*/
|
||||
private $requires_license;
|
||||
/**
|
||||
* Is the product available on wordpress.org
|
||||
*
|
||||
* @var bool $wordpress_available Either is available on WordPress or not.
|
||||
*/
|
||||
private $wordpress_available;
|
||||
/**
|
||||
* Current version of the product.
|
||||
*
|
||||
* @var string $version The product version.
|
||||
*/
|
||||
private $version;
|
||||
/**
|
||||
* Holds a map of loaded products objects.
|
||||
*
|
||||
* @var array Array of loaded products.
|
||||
*/
|
||||
private static $cached_products = [];
|
||||
/**
|
||||
* Root api endpoint.
|
||||
*/
|
||||
const API_URL = 'https://api.themeisle.com/';
|
||||
|
||||
/**
|
||||
* ThemeIsle_SDK_Product constructor.
|
||||
*
|
||||
* @param string $basefile Product basefile.
|
||||
*/
|
||||
public function __construct( $basefile ) {
|
||||
if ( ! empty( $basefile ) ) {
|
||||
if ( is_file( $basefile ) ) {
|
||||
$this->basefile = $basefile;
|
||||
$this->setup_from_path();
|
||||
$this->setup_from_fileheaders();
|
||||
}
|
||||
}
|
||||
$install = get_option( $this->get_key() . '_install', 0 );
|
||||
if ( 0 === $install ) {
|
||||
$install = time();
|
||||
/**
|
||||
* Action to be triggered when the product is first activated.
|
||||
*
|
||||
* @param string $basefile The basefile of the product.
|
||||
*/
|
||||
do_action( 'themeisle_sdk_first_activation', $basefile );
|
||||
|
||||
update_option( $this->get_key() . '_install', time() );
|
||||
}
|
||||
$this->install = $install;
|
||||
self::$cached_products[ crc32( $basefile ) ] = $this;
|
||||
$current_version = get_option( $this->slug . '_version', '' );
|
||||
|
||||
if ( $current_version !== $this->version && wp_cache_get( "{$this->slug}_version_upgrade" ) === false ) {
|
||||
// Set the cache lock to avoid multiple calls.
|
||||
wp_cache_set( "{$this->slug}_version_upgrade", true, HOUR_IN_SECONDS );
|
||||
/**
|
||||
* Action to be triggered when the product is updated.
|
||||
*
|
||||
* @param string $current_version The current version of the product.
|
||||
* @param string $new_version The new version of the product.
|
||||
* @param string $basefile The basefile of the product.
|
||||
*/
|
||||
do_action( "themeisle_sdk_update_{$this->slug}", $current_version, $this->version, $basefile );
|
||||
|
||||
// Update the version of the product.
|
||||
update_option( "{$this->slug}_version", $this->version );
|
||||
// Delete the cache lock.
|
||||
wp_cache_delete( "{$this->slug}_version_upgrade" );
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Return a product.
|
||||
*
|
||||
* @param string $basefile Product basefile.
|
||||
*
|
||||
* @return Product Product Object.
|
||||
*/
|
||||
public static function get( $basefile ) {
|
||||
$key = crc32( $basefile );
|
||||
if ( isset( self::$cached_products[ $key ] ) ) {
|
||||
return self::$cached_products[ $key ];
|
||||
}
|
||||
self::$cached_products[ $key ] = new Product( $basefile );
|
||||
|
||||
return self::$cached_products[ $key ];
|
||||
}
|
||||
|
||||
/**
|
||||
* Setup props from path.
|
||||
*/
|
||||
public function setup_from_path() {
|
||||
$this->file = basename( $this->basefile );
|
||||
$dir = dirname( $this->basefile );
|
||||
$this->slug = basename( $dir );
|
||||
$exts = explode( '.', $this->basefile );
|
||||
$ext = $exts[ count( $exts ) - 1 ];
|
||||
if ( 'css' === $ext ) {
|
||||
$this->type = 'theme';
|
||||
}
|
||||
if ( 'php' === $ext ) {
|
||||
$this->type = 'plugin';
|
||||
}
|
||||
$this->key = self::key_ready_name( $this->slug );
|
||||
}
|
||||
|
||||
/**
|
||||
* Normalize string.
|
||||
*
|
||||
* @param string $string the String to be normalized for cron handler.
|
||||
*
|
||||
* @return string $name The normalized string.
|
||||
*/
|
||||
public static function key_ready_name( $string ) {
|
||||
return str_replace( '-', '_', strtolower( trim( $string ) ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Setup props from fileheaders.
|
||||
*/
|
||||
public function setup_from_fileheaders() {
|
||||
$file_headers = array(
|
||||
'Requires License' => 'Requires License',
|
||||
'WordPress Available' => 'WordPress Available',
|
||||
'Pro Slug' => 'Pro Slug',
|
||||
'Version' => 'Version',
|
||||
);
|
||||
if ( 'plugin' === $this->type ) {
|
||||
$file_headers['Name'] = 'Plugin Name';
|
||||
$file_headers['AuthorName'] = 'Author';
|
||||
$file_headers['AuthorURI'] = 'Author URI';
|
||||
}
|
||||
if ( 'theme' === $this->type ) {
|
||||
$file_headers['Name'] = 'Theme Name';
|
||||
$file_headers['AuthorName'] = 'Author';
|
||||
$file_headers['AuthorURI'] = 'Author URI';
|
||||
}
|
||||
$file_headers = get_file_data( $this->basefile, $file_headers );
|
||||
|
||||
$this->name = $file_headers['Name'];
|
||||
$this->store_name = $file_headers['AuthorName'];
|
||||
$this->author_url = $file_headers['AuthorURI'];
|
||||
$this->store_url = $file_headers['AuthorURI'];
|
||||
|
||||
$this->requires_license = ( 'yes' === $file_headers['Requires License'] ) ? true : false;
|
||||
$this->wordpress_available = ( 'yes' === $file_headers['WordPress Available'] ) ? true : false;
|
||||
$this->pro_slug = ! empty( $file_headers['Pro Slug'] ) ? $file_headers['Pro Slug'] : '';
|
||||
$this->version = $file_headers['Version'];
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Return the product key.
|
||||
*
|
||||
* @return string The product key.
|
||||
*/
|
||||
public function get_key() {
|
||||
return $this->key;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if the product is either theme or plugin.
|
||||
*
|
||||
* @return string Product type.
|
||||
*/
|
||||
public function get_type() {
|
||||
return $this->type;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return if the product is used as a plugin.
|
||||
*
|
||||
* @return bool Is plugin?
|
||||
*/
|
||||
public function is_plugin() {
|
||||
return self::PLUGIN_TYPE === $this->type;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return if the product is used as a theme.
|
||||
*
|
||||
* @return bool Is theme ?
|
||||
*/
|
||||
public function is_theme() {
|
||||
return self::THEME_TYPE === $this->type;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the product slug.
|
||||
*
|
||||
* @return string The product slug.
|
||||
*/
|
||||
public function get_slug() {
|
||||
return $this->slug;
|
||||
}
|
||||
|
||||
/**
|
||||
* The magic var_dump info method.
|
||||
*
|
||||
* @return array Debug info.
|
||||
*/
|
||||
public function __debugInfo() {
|
||||
return array(
|
||||
'name' => $this->name,
|
||||
'slug' => $this->slug,
|
||||
'version' => $this->version,
|
||||
'basefile' => $this->basefile,
|
||||
'key' => $this->key,
|
||||
'type' => $this->type,
|
||||
'store_name' => $this->store_name,
|
||||
'store_url' => $this->store_url,
|
||||
'wordpress_available' => $this->wordpress_available,
|
||||
'requires_license' => $this->requires_license,
|
||||
);
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Getter for product version.
|
||||
*
|
||||
* @return string The product version.
|
||||
*/
|
||||
public function get_version() {
|
||||
return $this->version;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns current product license, if available.
|
||||
*
|
||||
* @return string Return license key, if available.
|
||||
*/
|
||||
public function get_license() {
|
||||
|
||||
if ( ! $this->requires_license() && ! $this->is_wordpress_available() ) {
|
||||
return 'free';
|
||||
}
|
||||
$license_data = get_option( $this->get_key() . '_license_data', '' );
|
||||
|
||||
if ( empty( $license_data ) ) {
|
||||
return get_option( $this->get_key() . '_license', '' );
|
||||
}
|
||||
if ( ! isset( $license_data->key ) ) {
|
||||
return get_option( $this->get_key() . '_license', '' );
|
||||
}
|
||||
|
||||
return $license_data->key;
|
||||
}
|
||||
|
||||
/**
|
||||
* Either the product requires license or not.
|
||||
*
|
||||
* @return bool Either requires license or not.
|
||||
*/
|
||||
public function requires_license() {
|
||||
return $this->requires_license;
|
||||
}
|
||||
|
||||
/**
|
||||
* If product is available on wordpress.org or not.
|
||||
*
|
||||
* @return bool Either is wp available or not.
|
||||
*/
|
||||
public function is_wordpress_available() {
|
||||
return $this->wordpress_available;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return friendly name.
|
||||
*
|
||||
* @return string Friendly name.
|
||||
*/
|
||||
public function get_friendly_name() {
|
||||
$name = trim( str_replace( 'Lite', '', $this->get_name() ) );
|
||||
if ( defined( 'OTTER_BLOCKS_BASEFILE' ) && OTTER_BLOCKS_BASEFILE === $this->basefile ) {
|
||||
$name = 'Otter Blocks';
|
||||
}
|
||||
if ( defined( 'OPTML_BASEFILE' ) && OPTML_BASEFILE === $this->basefile ) {
|
||||
$name = 'Optimole';
|
||||
}
|
||||
if ( defined( 'WPMM_FILE' ) && WPMM_FILE === $this->basefile ) {
|
||||
$name = 'LightStart';
|
||||
}
|
||||
$name = apply_filters( $this->get_key() . '_friendly_name', $name );
|
||||
$name = rtrim( $name, '- ()' );
|
||||
|
||||
return $name;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return the product version cache key.
|
||||
*
|
||||
* @return string The product version cache key.
|
||||
*/
|
||||
public function get_cache_key() {
|
||||
return $this->get_key() . '_' . preg_replace( '/[^0-9a-zA-Z ]/m', '', $this->get_version() ) . 'versions';
|
||||
}
|
||||
|
||||
/**
|
||||
* Getter for product name.
|
||||
*
|
||||
* @return string The product name.
|
||||
*/
|
||||
public function get_name() {
|
||||
return $this->name;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the Store name.
|
||||
*
|
||||
* @return string Store name.
|
||||
*/
|
||||
public function get_store_name() {
|
||||
return $this->store_name;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the store url.
|
||||
*
|
||||
* @return string The store url.
|
||||
*/
|
||||
public function get_store_url() {
|
||||
|
||||
if ( strpos( $this->store_url, '/themeisle.com' ) !== false ) {
|
||||
return 'https://store.themeisle.com/';
|
||||
}
|
||||
|
||||
return $this->store_url;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns product basefile, which holds the metaheaders.
|
||||
*
|
||||
* @return string The product basefile.
|
||||
*/
|
||||
public function get_basefile() {
|
||||
return $this->basefile;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get changelog url.
|
||||
*
|
||||
* @return string Changelog url.
|
||||
*/
|
||||
public function get_changelog() {
|
||||
return add_query_arg(
|
||||
[
|
||||
'name' => rawurlencode( $this->get_name() ),
|
||||
'edd_action' => 'view_changelog',
|
||||
'locale' => get_user_locale(),
|
||||
],
|
||||
$this->get_store_url()
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns product filename.
|
||||
*
|
||||
* @return string The product filename.
|
||||
*/
|
||||
public function get_file() {
|
||||
return $this->file;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the pro slug, if available.
|
||||
*
|
||||
* @return string The pro slug.
|
||||
*/
|
||||
public function get_pro_slug() {
|
||||
return $this->pro_slug;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return the install timestamp.
|
||||
*
|
||||
* @return int The install timestamp.
|
||||
*/
|
||||
public function get_install_time() {
|
||||
return $this->install;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the URL of the product base file.
|
||||
*
|
||||
* @param string $path The path to the file.
|
||||
*
|
||||
* @return string The URL of the product base file.
|
||||
*/
|
||||
public function get_base_url( $path = '/' ) {
|
||||
if ( $this->type ) {
|
||||
return plugins_url( $path, $this->basefile );
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
57
wp-content/plugins/robin-image-optimizer/vendor/codeinwp/themeisle-sdk/start.php
vendored
Normal file
@@ -0,0 +1,57 @@
|
||||
<?php
|
||||
/**
|
||||
* File responsible for sdk files loading.
|
||||
*
|
||||
* @package ThemeIsleSDK
|
||||
* @copyright Copyright (c) 2017, Marius Cristea
|
||||
* @license http://opensource.org/licenses/gpl-2.0.php GNU Public License
|
||||
* @since 1.1.0
|
||||
*/
|
||||
|
||||
namespace ThemeisleSDK;
|
||||
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
$products = apply_filters( 'themeisle_sdk_products', array() );
|
||||
$themeisle_library_path = dirname( __FILE__ );
|
||||
$files_to_load = [
|
||||
$themeisle_library_path . '/src/Loader.php',
|
||||
$themeisle_library_path . '/src/Product.php',
|
||||
|
||||
$themeisle_library_path . '/src/Common/Abstract_module.php',
|
||||
$themeisle_library_path . '/src/Common/Module_factory.php',
|
||||
|
||||
$themeisle_library_path . '/src/Modules/Script_loader.php',
|
||||
$themeisle_library_path . '/src/Modules/Dashboard_widget.php',
|
||||
$themeisle_library_path . '/src/Modules/Rollback.php',
|
||||
$themeisle_library_path . '/src/Modules/Uninstall_feedback.php',
|
||||
$themeisle_library_path . '/src/Modules/Licenser.php',
|
||||
$themeisle_library_path . '/src/Modules/Endpoint.php',
|
||||
$themeisle_library_path . '/src/Modules/Notification.php',
|
||||
$themeisle_library_path . '/src/Modules/Logger.php',
|
||||
$themeisle_library_path . '/src/Modules/Translate.php',
|
||||
$themeisle_library_path . '/src/Modules/Translations.php',
|
||||
$themeisle_library_path . '/src/Modules/Review.php',
|
||||
$themeisle_library_path . '/src/Modules/Recommendation.php',
|
||||
$themeisle_library_path . '/src/Modules/Promotions.php',
|
||||
$themeisle_library_path . '/src/Modules/Welcome.php',
|
||||
$themeisle_library_path . '/src/Modules/Compatibilities.php',
|
||||
$themeisle_library_path . '/src/Modules/About_us.php',
|
||||
$themeisle_library_path . '/src/Modules/Announcements.php',
|
||||
$themeisle_library_path . '/src/Modules/Featured_plugins.php',
|
||||
$themeisle_library_path . '/src/Modules/Float_widget.php',
|
||||
];
|
||||
|
||||
$files_to_load = array_merge( $files_to_load, apply_filters( 'themeisle_sdk_required_files', [] ) );
|
||||
|
||||
foreach ( $files_to_load as $file ) {
|
||||
if ( is_file( $file ) ) {
|
||||
require_once $file;
|
||||
}
|
||||
}
|
||||
Loader::init();
|
||||
|
||||
foreach ( $products as $product ) {
|
||||
Loader::add_product( $product );
|
||||
}
|
||||
579
wp-content/plugins/robin-image-optimizer/vendor/composer/ClassLoader.php
vendored
Normal file
@@ -0,0 +1,579 @@
|
||||
<?php
|
||||
|
||||
/*
|
||||
* This file is part of Composer.
|
||||
*
|
||||
* (c) Nils Adermann <naderman@naderman.de>
|
||||
* Jordi Boggiano <j.boggiano@seld.be>
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
|
||||
namespace Composer\Autoload;
|
||||
|
||||
/**
|
||||
* ClassLoader implements a PSR-0, PSR-4 and classmap class loader.
|
||||
*
|
||||
* $loader = new \Composer\Autoload\ClassLoader();
|
||||
*
|
||||
* // register classes with namespaces
|
||||
* $loader->add('Symfony\Component', __DIR__.'/component');
|
||||
* $loader->add('Symfony', __DIR__.'/framework');
|
||||
*
|
||||
* // activate the autoloader
|
||||
* $loader->register();
|
||||
*
|
||||
* // to enable searching the include path (eg. for PEAR packages)
|
||||
* $loader->setUseIncludePath(true);
|
||||
*
|
||||
* In this example, if you try to use a class in the Symfony\Component
|
||||
* namespace or one of its children (Symfony\Component\Console for instance),
|
||||
* the autoloader will first look for the class under the component/
|
||||
* directory, and it will then fallback to the framework/ directory if not
|
||||
* found before giving up.
|
||||
*
|
||||
* This class is loosely based on the Symfony UniversalClassLoader.
|
||||
*
|
||||
* @author Fabien Potencier <fabien@symfony.com>
|
||||
* @author Jordi Boggiano <j.boggiano@seld.be>
|
||||
* @see https://www.php-fig.org/psr/psr-0/
|
||||
* @see https://www.php-fig.org/psr/psr-4/
|
||||
*/
|
||||
class ClassLoader
|
||||
{
|
||||
/** @var \Closure(string):void */
|
||||
private static $includeFile;
|
||||
|
||||
/** @var string|null */
|
||||
private $vendorDir;
|
||||
|
||||
// PSR-4
|
||||
/**
|
||||
* @var array<string, array<string, int>>
|
||||
*/
|
||||
private $prefixLengthsPsr4 = array();
|
||||
/**
|
||||
* @var array<string, list<string>>
|
||||
*/
|
||||
private $prefixDirsPsr4 = array();
|
||||
/**
|
||||
* @var list<string>
|
||||
*/
|
||||
private $fallbackDirsPsr4 = array();
|
||||
|
||||
// PSR-0
|
||||
/**
|
||||
* List of PSR-0 prefixes
|
||||
*
|
||||
* Structured as array('F (first letter)' => array('Foo\Bar (full prefix)' => array('path', 'path2')))
|
||||
*
|
||||
* @var array<string, array<string, list<string>>>
|
||||
*/
|
||||
private $prefixesPsr0 = array();
|
||||
/**
|
||||
* @var list<string>
|
||||
*/
|
||||
private $fallbackDirsPsr0 = array();
|
||||
|
||||
/** @var bool */
|
||||
private $useIncludePath = false;
|
||||
|
||||
/**
|
||||
* @var array<string, string>
|
||||
*/
|
||||
private $classMap = array();
|
||||
|
||||
/** @var bool */
|
||||
private $classMapAuthoritative = false;
|
||||
|
||||
/**
|
||||
* @var array<string, bool>
|
||||
*/
|
||||
private $missingClasses = array();
|
||||
|
||||
/** @var string|null */
|
||||
private $apcuPrefix;
|
||||
|
||||
/**
|
||||
* @var array<string, self>
|
||||
*/
|
||||
private static $registeredLoaders = array();
|
||||
|
||||
/**
|
||||
* @param string|null $vendorDir
|
||||
*/
|
||||
public function __construct($vendorDir = null)
|
||||
{
|
||||
$this->vendorDir = $vendorDir;
|
||||
self::initializeIncludeClosure();
|
||||
}
|
||||
|
||||
/**
|
||||
* @return array<string, list<string>>
|
||||
*/
|
||||
public function getPrefixes()
|
||||
{
|
||||
if (!empty($this->prefixesPsr0)) {
|
||||
return call_user_func_array('array_merge', array_values($this->prefixesPsr0));
|
||||
}
|
||||
|
||||
return array();
|
||||
}
|
||||
|
||||
/**
|
||||
* @return array<string, list<string>>
|
||||
*/
|
||||
public function getPrefixesPsr4()
|
||||
{
|
||||
return $this->prefixDirsPsr4;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return list<string>
|
||||
*/
|
||||
public function getFallbackDirs()
|
||||
{
|
||||
return $this->fallbackDirsPsr0;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return list<string>
|
||||
*/
|
||||
public function getFallbackDirsPsr4()
|
||||
{
|
||||
return $this->fallbackDirsPsr4;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return array<string, string> Array of classname => path
|
||||
*/
|
||||
public function getClassMap()
|
||||
{
|
||||
return $this->classMap;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param array<string, string> $classMap Class to filename map
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function addClassMap(array $classMap)
|
||||
{
|
||||
if ($this->classMap) {
|
||||
$this->classMap = array_merge($this->classMap, $classMap);
|
||||
} else {
|
||||
$this->classMap = $classMap;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Registers a set of PSR-0 directories for a given prefix, either
|
||||
* appending or prepending to the ones previously set for this prefix.
|
||||
*
|
||||
* @param string $prefix The prefix
|
||||
* @param list<string>|string $paths The PSR-0 root directories
|
||||
* @param bool $prepend Whether to prepend the directories
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function add($prefix, $paths, $prepend = false)
|
||||
{
|
||||
$paths = (array) $paths;
|
||||
if (!$prefix) {
|
||||
if ($prepend) {
|
||||
$this->fallbackDirsPsr0 = array_merge(
|
||||
$paths,
|
||||
$this->fallbackDirsPsr0
|
||||
);
|
||||
} else {
|
||||
$this->fallbackDirsPsr0 = array_merge(
|
||||
$this->fallbackDirsPsr0,
|
||||
$paths
|
||||
);
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
$first = $prefix[0];
|
||||
if (!isset($this->prefixesPsr0[$first][$prefix])) {
|
||||
$this->prefixesPsr0[$first][$prefix] = $paths;
|
||||
|
||||
return;
|
||||
}
|
||||
if ($prepend) {
|
||||
$this->prefixesPsr0[$first][$prefix] = array_merge(
|
||||
$paths,
|
||||
$this->prefixesPsr0[$first][$prefix]
|
||||
);
|
||||
} else {
|
||||
$this->prefixesPsr0[$first][$prefix] = array_merge(
|
||||
$this->prefixesPsr0[$first][$prefix],
|
||||
$paths
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Registers a set of PSR-4 directories for a given namespace, either
|
||||
* appending or prepending to the ones previously set for this namespace.
|
||||
*
|
||||
* @param string $prefix The prefix/namespace, with trailing '\\'
|
||||
* @param list<string>|string $paths The PSR-4 base directories
|
||||
* @param bool $prepend Whether to prepend the directories
|
||||
*
|
||||
* @throws \InvalidArgumentException
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function addPsr4($prefix, $paths, $prepend = false)
|
||||
{
|
||||
$paths = (array) $paths;
|
||||
if (!$prefix) {
|
||||
// Register directories for the root namespace.
|
||||
if ($prepend) {
|
||||
$this->fallbackDirsPsr4 = array_merge(
|
||||
$paths,
|
||||
$this->fallbackDirsPsr4
|
||||
);
|
||||
} else {
|
||||
$this->fallbackDirsPsr4 = array_merge(
|
||||
$this->fallbackDirsPsr4,
|
||||
$paths
|
||||
);
|
||||
}
|
||||
} elseif (!isset($this->prefixDirsPsr4[$prefix])) {
|
||||
// Register directories for a new namespace.
|
||||
$length = strlen($prefix);
|
||||
if ('\\' !== $prefix[$length - 1]) {
|
||||
throw new \InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator.");
|
||||
}
|
||||
$this->prefixLengthsPsr4[$prefix[0]][$prefix] = $length;
|
||||
$this->prefixDirsPsr4[$prefix] = $paths;
|
||||
} elseif ($prepend) {
|
||||
// Prepend directories for an already registered namespace.
|
||||
$this->prefixDirsPsr4[$prefix] = array_merge(
|
||||
$paths,
|
||||
$this->prefixDirsPsr4[$prefix]
|
||||
);
|
||||
} else {
|
||||
// Append directories for an already registered namespace.
|
||||
$this->prefixDirsPsr4[$prefix] = array_merge(
|
||||
$this->prefixDirsPsr4[$prefix],
|
||||
$paths
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Registers a set of PSR-0 directories for a given prefix,
|
||||
* replacing any others previously set for this prefix.
|
||||
*
|
||||
* @param string $prefix The prefix
|
||||
* @param list<string>|string $paths The PSR-0 base directories
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function set($prefix, $paths)
|
||||
{
|
||||
if (!$prefix) {
|
||||
$this->fallbackDirsPsr0 = (array) $paths;
|
||||
} else {
|
||||
$this->prefixesPsr0[$prefix[0]][$prefix] = (array) $paths;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Registers a set of PSR-4 directories for a given namespace,
|
||||
* replacing any others previously set for this namespace.
|
||||
*
|
||||
* @param string $prefix The prefix/namespace, with trailing '\\'
|
||||
* @param list<string>|string $paths The PSR-4 base directories
|
||||
*
|
||||
* @throws \InvalidArgumentException
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function setPsr4($prefix, $paths)
|
||||
{
|
||||
if (!$prefix) {
|
||||
$this->fallbackDirsPsr4 = (array) $paths;
|
||||
} else {
|
||||
$length = strlen($prefix);
|
||||
if ('\\' !== $prefix[$length - 1]) {
|
||||
throw new \InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator.");
|
||||
}
|
||||
$this->prefixLengthsPsr4[$prefix[0]][$prefix] = $length;
|
||||
$this->prefixDirsPsr4[$prefix] = (array) $paths;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Turns on searching the include path for class files.
|
||||
*
|
||||
* @param bool $useIncludePath
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function setUseIncludePath($useIncludePath)
|
||||
{
|
||||
$this->useIncludePath = $useIncludePath;
|
||||
}
|
||||
|
||||
/**
|
||||
* Can be used to check if the autoloader uses the include path to check
|
||||
* for classes.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function getUseIncludePath()
|
||||
{
|
||||
return $this->useIncludePath;
|
||||
}
|
||||
|
||||
/**
|
||||
* Turns off searching the prefix and fallback directories for classes
|
||||
* that have not been registered with the class map.
|
||||
*
|
||||
* @param bool $classMapAuthoritative
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function setClassMapAuthoritative($classMapAuthoritative)
|
||||
{
|
||||
$this->classMapAuthoritative = $classMapAuthoritative;
|
||||
}
|
||||
|
||||
/**
|
||||
* Should class lookup fail if not found in the current class map?
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function isClassMapAuthoritative()
|
||||
{
|
||||
return $this->classMapAuthoritative;
|
||||
}
|
||||
|
||||
/**
|
||||
* APCu prefix to use to cache found/not-found classes, if the extension is enabled.
|
||||
*
|
||||
* @param string|null $apcuPrefix
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function setApcuPrefix($apcuPrefix)
|
||||
{
|
||||
$this->apcuPrefix = function_exists('apcu_fetch') && filter_var(ini_get('apc.enabled'), FILTER_VALIDATE_BOOLEAN) ? $apcuPrefix : null;
|
||||
}
|
||||
|
||||
/**
|
||||
* The APCu prefix in use, or null if APCu caching is not enabled.
|
||||
*
|
||||
* @return string|null
|
||||
*/
|
||||
public function getApcuPrefix()
|
||||
{
|
||||
return $this->apcuPrefix;
|
||||
}
|
||||
|
||||
/**
|
||||
* Registers this instance as an autoloader.
|
||||
*
|
||||
* @param bool $prepend Whether to prepend the autoloader or not
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function register($prepend = false)
|
||||
{
|
||||
spl_autoload_register(array($this, 'loadClass'), true, $prepend);
|
||||
|
||||
if (null === $this->vendorDir) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ($prepend) {
|
||||
self::$registeredLoaders = array($this->vendorDir => $this) + self::$registeredLoaders;
|
||||
} else {
|
||||
unset(self::$registeredLoaders[$this->vendorDir]);
|
||||
self::$registeredLoaders[$this->vendorDir] = $this;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Unregisters this instance as an autoloader.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function unregister()
|
||||
{
|
||||
spl_autoload_unregister(array($this, 'loadClass'));
|
||||
|
||||
if (null !== $this->vendorDir) {
|
||||
unset(self::$registeredLoaders[$this->vendorDir]);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Loads the given class or interface.
|
||||
*
|
||||
* @param string $class The name of the class
|
||||
* @return true|null True if loaded, null otherwise
|
||||
*/
|
||||
public function loadClass($class)
|
||||
{
|
||||
if ($file = $this->findFile($class)) {
|
||||
$includeFile = self::$includeFile;
|
||||
$includeFile($file);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Finds the path to the file where the class is defined.
|
||||
*
|
||||
* @param string $class The name of the class
|
||||
*
|
||||
* @return string|false The path if found, false otherwise
|
||||
*/
|
||||
public function findFile($class)
|
||||
{
|
||||
// class map lookup
|
||||
if (isset($this->classMap[$class])) {
|
||||
return $this->classMap[$class];
|
||||
}
|
||||
if ($this->classMapAuthoritative || isset($this->missingClasses[$class])) {
|
||||
return false;
|
||||
}
|
||||
if (null !== $this->apcuPrefix) {
|
||||
$file = apcu_fetch($this->apcuPrefix.$class, $hit);
|
||||
if ($hit) {
|
||||
return $file;
|
||||
}
|
||||
}
|
||||
|
||||
$file = $this->findFileWithExtension($class, '.php');
|
||||
|
||||
// Search for Hack files if we are running on HHVM
|
||||
if (false === $file && defined('HHVM_VERSION')) {
|
||||
$file = $this->findFileWithExtension($class, '.hh');
|
||||
}
|
||||
|
||||
if (null !== $this->apcuPrefix) {
|
||||
apcu_add($this->apcuPrefix.$class, $file);
|
||||
}
|
||||
|
||||
if (false === $file) {
|
||||
// Remember that this class does not exist.
|
||||
$this->missingClasses[$class] = true;
|
||||
}
|
||||
|
||||
return $file;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the currently registered loaders keyed by their corresponding vendor directories.
|
||||
*
|
||||
* @return array<string, self>
|
||||
*/
|
||||
public static function getRegisteredLoaders()
|
||||
{
|
||||
return self::$registeredLoaders;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $class
|
||||
* @param string $ext
|
||||
* @return string|false
|
||||
*/
|
||||
private function findFileWithExtension($class, $ext)
|
||||
{
|
||||
// PSR-4 lookup
|
||||
$logicalPathPsr4 = strtr($class, '\\', DIRECTORY_SEPARATOR) . $ext;
|
||||
|
||||
$first = $class[0];
|
||||
if (isset($this->prefixLengthsPsr4[$first])) {
|
||||
$subPath = $class;
|
||||
while (false !== $lastPos = strrpos($subPath, '\\')) {
|
||||
$subPath = substr($subPath, 0, $lastPos);
|
||||
$search = $subPath . '\\';
|
||||
if (isset($this->prefixDirsPsr4[$search])) {
|
||||
$pathEnd = DIRECTORY_SEPARATOR . substr($logicalPathPsr4, $lastPos + 1);
|
||||
foreach ($this->prefixDirsPsr4[$search] as $dir) {
|
||||
if (file_exists($file = $dir . $pathEnd)) {
|
||||
return $file;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// PSR-4 fallback dirs
|
||||
foreach ($this->fallbackDirsPsr4 as $dir) {
|
||||
if (file_exists($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr4)) {
|
||||
return $file;
|
||||
}
|
||||
}
|
||||
|
||||
// PSR-0 lookup
|
||||
if (false !== $pos = strrpos($class, '\\')) {
|
||||
// namespaced class name
|
||||
$logicalPathPsr0 = substr($logicalPathPsr4, 0, $pos + 1)
|
||||
. strtr(substr($logicalPathPsr4, $pos + 1), '_', DIRECTORY_SEPARATOR);
|
||||
} else {
|
||||
// PEAR-like class name
|
||||
$logicalPathPsr0 = strtr($class, '_', DIRECTORY_SEPARATOR) . $ext;
|
||||
}
|
||||
|
||||
if (isset($this->prefixesPsr0[$first])) {
|
||||
foreach ($this->prefixesPsr0[$first] as $prefix => $dirs) {
|
||||
if (0 === strpos($class, $prefix)) {
|
||||
foreach ($dirs as $dir) {
|
||||
if (file_exists($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr0)) {
|
||||
return $file;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// PSR-0 fallback dirs
|
||||
foreach ($this->fallbackDirsPsr0 as $dir) {
|
||||
if (file_exists($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr0)) {
|
||||
return $file;
|
||||
}
|
||||
}
|
||||
|
||||
// PSR-0 include paths.
|
||||
if ($this->useIncludePath && $file = stream_resolve_include_path($logicalPathPsr0)) {
|
||||
return $file;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return void
|
||||
*/
|
||||
private static function initializeIncludeClosure()
|
||||
{
|
||||
if (self::$includeFile !== null) {
|
||||
return;
|
||||
}
|
||||
|
||||
/**
|
||||
* Scope isolated include.
|
||||
*
|
||||
* Prevents access to $this/self from included files.
|
||||
*
|
||||
* @param string $file
|
||||
* @return void
|
||||
*/
|
||||
self::$includeFile = \Closure::bind(static function($file) {
|
||||
include $file;
|
||||
}, null, null);
|
||||
}
|
||||
}
|
||||
396
wp-content/plugins/robin-image-optimizer/vendor/composer/InstalledVersions.php
vendored
Normal file
@@ -0,0 +1,396 @@
|
||||
<?php
|
||||
|
||||
/*
|
||||
* This file is part of Composer.
|
||||
*
|
||||
* (c) Nils Adermann <naderman@naderman.de>
|
||||
* Jordi Boggiano <j.boggiano@seld.be>
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
|
||||
namespace Composer;
|
||||
|
||||
use Composer\Autoload\ClassLoader;
|
||||
use Composer\Semver\VersionParser;
|
||||
|
||||
/**
|
||||
* This class is copied in every Composer installed project and available to all
|
||||
*
|
||||
* See also https://getcomposer.org/doc/07-runtime.md#installed-versions
|
||||
*
|
||||
* To require its presence, you can require `composer-runtime-api ^2.0`
|
||||
*
|
||||
* @final
|
||||
*/
|
||||
class InstalledVersions
|
||||
{
|
||||
/**
|
||||
* @var string|null if set (by reflection by Composer), this should be set to the path where this class is being copied to
|
||||
* @internal
|
||||
*/
|
||||
private static $selfDir = null;
|
||||
|
||||
/**
|
||||
* @var mixed[]|null
|
||||
* @psalm-var array{root: array{name: string, pretty_version: string, version: string, reference: string|null, type: string, install_path: string, aliases: string[], dev: bool}, versions: array<string, array{pretty_version?: string, version?: string, reference?: string|null, type?: string, install_path?: string, aliases?: string[], dev_requirement: bool, replaced?: string[], provided?: string[]}>}|array{}|null
|
||||
*/
|
||||
private static $installed;
|
||||
|
||||
/**
|
||||
* @var bool
|
||||
*/
|
||||
private static $installedIsLocalDir;
|
||||
|
||||
/**
|
||||
* @var bool|null
|
||||
*/
|
||||
private static $canGetVendors;
|
||||
|
||||
/**
|
||||
* @var array[]
|
||||
* @psalm-var array<string, array{root: array{name: string, pretty_version: string, version: string, reference: string|null, type: string, install_path: string, aliases: string[], dev: bool}, versions: array<string, array{pretty_version?: string, version?: string, reference?: string|null, type?: string, install_path?: string, aliases?: string[], dev_requirement: bool, replaced?: string[], provided?: string[]}>}>
|
||||
*/
|
||||
private static $installedByVendor = array();
|
||||
|
||||
/**
|
||||
* Returns a list of all package names which are present, either by being installed, replaced or provided
|
||||
*
|
||||
* @return string[]
|
||||
* @psalm-return list<string>
|
||||
*/
|
||||
public static function getInstalledPackages()
|
||||
{
|
||||
$packages = array();
|
||||
foreach (self::getInstalled() as $installed) {
|
||||
$packages[] = array_keys($installed['versions']);
|
||||
}
|
||||
|
||||
if (1 === \count($packages)) {
|
||||
return $packages[0];
|
||||
}
|
||||
|
||||
return array_keys(array_flip(\call_user_func_array('array_merge', $packages)));
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a list of all package names with a specific type e.g. 'library'
|
||||
*
|
||||
* @param string $type
|
||||
* @return string[]
|
||||
* @psalm-return list<string>
|
||||
*/
|
||||
public static function getInstalledPackagesByType($type)
|
||||
{
|
||||
$packagesByType = array();
|
||||
|
||||
foreach (self::getInstalled() as $installed) {
|
||||
foreach ($installed['versions'] as $name => $package) {
|
||||
if (isset($package['type']) && $package['type'] === $type) {
|
||||
$packagesByType[] = $name;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return $packagesByType;
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks whether the given package is installed
|
||||
*
|
||||
* This also returns true if the package name is provided or replaced by another package
|
||||
*
|
||||
* @param string $packageName
|
||||
* @param bool $includeDevRequirements
|
||||
* @return bool
|
||||
*/
|
||||
public static function isInstalled($packageName, $includeDevRequirements = true)
|
||||
{
|
||||
foreach (self::getInstalled() as $installed) {
|
||||
if (isset($installed['versions'][$packageName])) {
|
||||
return $includeDevRequirements || !isset($installed['versions'][$packageName]['dev_requirement']) || $installed['versions'][$packageName]['dev_requirement'] === false;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks whether the given package satisfies a version constraint
|
||||
*
|
||||
* e.g. If you want to know whether version 2.3+ of package foo/bar is installed, you would call:
|
||||
*
|
||||
* Composer\InstalledVersions::satisfies(new VersionParser, 'foo/bar', '^2.3')
|
||||
*
|
||||
* @param VersionParser $parser Install composer/semver to have access to this class and functionality
|
||||
* @param string $packageName
|
||||
* @param string|null $constraint A version constraint to check for, if you pass one you have to make sure composer/semver is required by your package
|
||||
* @return bool
|
||||
*/
|
||||
public static function satisfies(VersionParser $parser, $packageName, $constraint)
|
||||
{
|
||||
$constraint = $parser->parseConstraints((string) $constraint);
|
||||
$provided = $parser->parseConstraints(self::getVersionRanges($packageName));
|
||||
|
||||
return $provided->matches($constraint);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a version constraint representing all the range(s) which are installed for a given package
|
||||
*
|
||||
* It is easier to use this via isInstalled() with the $constraint argument if you need to check
|
||||
* whether a given version of a package is installed, and not just whether it exists
|
||||
*
|
||||
* @param string $packageName
|
||||
* @return string Version constraint usable with composer/semver
|
||||
*/
|
||||
public static function getVersionRanges($packageName)
|
||||
{
|
||||
foreach (self::getInstalled() as $installed) {
|
||||
if (!isset($installed['versions'][$packageName])) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$ranges = array();
|
||||
if (isset($installed['versions'][$packageName]['pretty_version'])) {
|
||||
$ranges[] = $installed['versions'][$packageName]['pretty_version'];
|
||||
}
|
||||
if (array_key_exists('aliases', $installed['versions'][$packageName])) {
|
||||
$ranges = array_merge($ranges, $installed['versions'][$packageName]['aliases']);
|
||||
}
|
||||
if (array_key_exists('replaced', $installed['versions'][$packageName])) {
|
||||
$ranges = array_merge($ranges, $installed['versions'][$packageName]['replaced']);
|
||||
}
|
||||
if (array_key_exists('provided', $installed['versions'][$packageName])) {
|
||||
$ranges = array_merge($ranges, $installed['versions'][$packageName]['provided']);
|
||||
}
|
||||
|
||||
return implode(' || ', $ranges);
|
||||
}
|
||||
|
||||
throw new \OutOfBoundsException('Package "' . $packageName . '" is not installed');
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $packageName
|
||||
* @return string|null If the package is being replaced or provided but is not really installed, null will be returned as version, use satisfies or getVersionRanges if you need to know if a given version is present
|
||||
*/
|
||||
public static function getVersion($packageName)
|
||||
{
|
||||
foreach (self::getInstalled() as $installed) {
|
||||
if (!isset($installed['versions'][$packageName])) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if (!isset($installed['versions'][$packageName]['version'])) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return $installed['versions'][$packageName]['version'];
|
||||
}
|
||||
|
||||
throw new \OutOfBoundsException('Package "' . $packageName . '" is not installed');
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $packageName
|
||||
* @return string|null If the package is being replaced or provided but is not really installed, null will be returned as version, use satisfies or getVersionRanges if you need to know if a given version is present
|
||||
*/
|
||||
public static function getPrettyVersion($packageName)
|
||||
{
|
||||
foreach (self::getInstalled() as $installed) {
|
||||
if (!isset($installed['versions'][$packageName])) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if (!isset($installed['versions'][$packageName]['pretty_version'])) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return $installed['versions'][$packageName]['pretty_version'];
|
||||
}
|
||||
|
||||
throw new \OutOfBoundsException('Package "' . $packageName . '" is not installed');
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $packageName
|
||||
* @return string|null If the package is being replaced or provided but is not really installed, null will be returned as reference
|
||||
*/
|
||||
public static function getReference($packageName)
|
||||
{
|
||||
foreach (self::getInstalled() as $installed) {
|
||||
if (!isset($installed['versions'][$packageName])) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if (!isset($installed['versions'][$packageName]['reference'])) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return $installed['versions'][$packageName]['reference'];
|
||||
}
|
||||
|
||||
throw new \OutOfBoundsException('Package "' . $packageName . '" is not installed');
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $packageName
|
||||
* @return string|null If the package is being replaced or provided but is not really installed, null will be returned as install path. Packages of type metapackages also have a null install path.
|
||||
*/
|
||||
public static function getInstallPath($packageName)
|
||||
{
|
||||
foreach (self::getInstalled() as $installed) {
|
||||
if (!isset($installed['versions'][$packageName])) {
|
||||
continue;
|
||||
}
|
||||
|
||||
return isset($installed['versions'][$packageName]['install_path']) ? $installed['versions'][$packageName]['install_path'] : null;
|
||||
}
|
||||
|
||||
throw new \OutOfBoundsException('Package "' . $packageName . '" is not installed');
|
||||
}
|
||||
|
||||
/**
|
||||
* @return array
|
||||
* @psalm-return array{name: string, pretty_version: string, version: string, reference: string|null, type: string, install_path: string, aliases: string[], dev: bool}
|
||||
*/
|
||||
public static function getRootPackage()
|
||||
{
|
||||
$installed = self::getInstalled();
|
||||
|
||||
return $installed[0]['root'];
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the raw installed.php data for custom implementations
|
||||
*
|
||||
* @deprecated Use getAllRawData() instead which returns all datasets for all autoloaders present in the process. getRawData only returns the first dataset loaded, which may not be what you expect.
|
||||
* @return array[]
|
||||
* @psalm-return array{root: array{name: string, pretty_version: string, version: string, reference: string|null, type: string, install_path: string, aliases: string[], dev: bool}, versions: array<string, array{pretty_version?: string, version?: string, reference?: string|null, type?: string, install_path?: string, aliases?: string[], dev_requirement: bool, replaced?: string[], provided?: string[]}>}
|
||||
*/
|
||||
public static function getRawData()
|
||||
{
|
||||
@trigger_error('getRawData only returns the first dataset loaded, which may not be what you expect. Use getAllRawData() instead which returns all datasets for all autoloaders present in the process.', E_USER_DEPRECATED);
|
||||
|
||||
if (null === self::$installed) {
|
||||
// only require the installed.php file if this file is loaded from its dumped location,
|
||||
// and not from its source location in the composer/composer package, see https://github.com/composer/composer/issues/9937
|
||||
if (substr(__DIR__, -8, 1) !== 'C') {
|
||||
self::$installed = include __DIR__ . '/installed.php';
|
||||
} else {
|
||||
self::$installed = array();
|
||||
}
|
||||
}
|
||||
|
||||
return self::$installed;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the raw data of all installed.php which are currently loaded for custom implementations
|
||||
*
|
||||
* @return array[]
|
||||
* @psalm-return list<array{root: array{name: string, pretty_version: string, version: string, reference: string|null, type: string, install_path: string, aliases: string[], dev: bool}, versions: array<string, array{pretty_version?: string, version?: string, reference?: string|null, type?: string, install_path?: string, aliases?: string[], dev_requirement: bool, replaced?: string[], provided?: string[]}>}>
|
||||
*/
|
||||
public static function getAllRawData()
|
||||
{
|
||||
return self::getInstalled();
|
||||
}
|
||||
|
||||
/**
|
||||
* Lets you reload the static array from another file
|
||||
*
|
||||
* This is only useful for complex integrations in which a project needs to use
|
||||
* this class but then also needs to execute another project's autoloader in process,
|
||||
* and wants to ensure both projects have access to their version of installed.php.
|
||||
*
|
||||
* A typical case would be PHPUnit, where it would need to make sure it reads all
|
||||
* the data it needs from this class, then call reload() with
|
||||
* `require $CWD/vendor/composer/installed.php` (or similar) as input to make sure
|
||||
* the project in which it runs can then also use this class safely, without
|
||||
* interference between PHPUnit's dependencies and the project's dependencies.
|
||||
*
|
||||
* @param array[] $data A vendor/composer/installed.php data set
|
||||
* @return void
|
||||
*
|
||||
* @psalm-param array{root: array{name: string, pretty_version: string, version: string, reference: string|null, type: string, install_path: string, aliases: string[], dev: bool}, versions: array<string, array{pretty_version?: string, version?: string, reference?: string|null, type?: string, install_path?: string, aliases?: string[], dev_requirement: bool, replaced?: string[], provided?: string[]}>} $data
|
||||
*/
|
||||
public static function reload($data)
|
||||
{
|
||||
self::$installed = $data;
|
||||
self::$installedByVendor = array();
|
||||
|
||||
// when using reload, we disable the duplicate protection to ensure that self::$installed data is
|
||||
// always returned, but we cannot know whether it comes from the installed.php in __DIR__ or not,
|
||||
// so we have to assume it does not, and that may result in duplicate data being returned when listing
|
||||
// all installed packages for example
|
||||
self::$installedIsLocalDir = false;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return string
|
||||
*/
|
||||
private static function getSelfDir()
|
||||
{
|
||||
if (self::$selfDir === null) {
|
||||
self::$selfDir = strtr(__DIR__, '\\', '/');
|
||||
}
|
||||
|
||||
return self::$selfDir;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return array[]
|
||||
* @psalm-return list<array{root: array{name: string, pretty_version: string, version: string, reference: string|null, type: string, install_path: string, aliases: string[], dev: bool}, versions: array<string, array{pretty_version?: string, version?: string, reference?: string|null, type?: string, install_path?: string, aliases?: string[], dev_requirement: bool, replaced?: string[], provided?: string[]}>}>
|
||||
*/
|
||||
private static function getInstalled()
|
||||
{
|
||||
if (null === self::$canGetVendors) {
|
||||
self::$canGetVendors = method_exists('Composer\Autoload\ClassLoader', 'getRegisteredLoaders');
|
||||
}
|
||||
|
||||
$installed = array();
|
||||
$copiedLocalDir = false;
|
||||
|
||||
if (self::$canGetVendors) {
|
||||
$selfDir = self::getSelfDir();
|
||||
foreach (ClassLoader::getRegisteredLoaders() as $vendorDir => $loader) {
|
||||
$vendorDir = strtr($vendorDir, '\\', '/');
|
||||
if (isset(self::$installedByVendor[$vendorDir])) {
|
||||
$installed[] = self::$installedByVendor[$vendorDir];
|
||||
} elseif (is_file($vendorDir.'/composer/installed.php')) {
|
||||
/** @var array{root: array{name: string, pretty_version: string, version: string, reference: string|null, type: string, install_path: string, aliases: string[], dev: bool}, versions: array<string, array{pretty_version?: string, version?: string, reference?: string|null, type?: string, install_path?: string, aliases?: string[], dev_requirement: bool, replaced?: string[], provided?: string[]}>} $required */
|
||||
$required = require $vendorDir.'/composer/installed.php';
|
||||
self::$installedByVendor[$vendorDir] = $required;
|
||||
$installed[] = $required;
|
||||
if (self::$installed === null && $vendorDir.'/composer' === $selfDir) {
|
||||
self::$installed = $required;
|
||||
self::$installedIsLocalDir = true;
|
||||
}
|
||||
}
|
||||
if (self::$installedIsLocalDir && $vendorDir.'/composer' === $selfDir) {
|
||||
$copiedLocalDir = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (null === self::$installed) {
|
||||
// only require the installed.php file if this file is loaded from its dumped location,
|
||||
// and not from its source location in the composer/composer package, see https://github.com/composer/composer/issues/9937
|
||||
if (substr(__DIR__, -8, 1) !== 'C') {
|
||||
/** @var array{root: array{name: string, pretty_version: string, version: string, reference: string|null, type: string, install_path: string, aliases: string[], dev: bool}, versions: array<string, array{pretty_version?: string, version?: string, reference?: string|null, type?: string, install_path?: string, aliases?: string[], dev_requirement: bool, replaced?: string[], provided?: string[]}>} $required */
|
||||
$required = require __DIR__ . '/installed.php';
|
||||
self::$installed = $required;
|
||||
} else {
|
||||
self::$installed = array();
|
||||
}
|
||||
}
|
||||
|
||||
if (self::$installed !== array() && !$copiedLocalDir) {
|
||||
$installed[] = self::$installed;
|
||||
}
|
||||
|
||||
return $installed;
|
||||
}
|
||||
}
|
||||
21
wp-content/plugins/robin-image-optimizer/vendor/composer/LICENSE
vendored
Normal file
@@ -0,0 +1,21 @@
|
||||
|
||||
Copyright (c) Nils Adermann, Jordi Boggiano
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is furnished
|
||||
to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
THE SOFTWARE.
|
||||
|
||||
10
wp-content/plugins/robin-image-optimizer/vendor/composer/autoload_classmap.php
vendored
Normal file
@@ -0,0 +1,10 @@
|
||||
<?php
|
||||
|
||||
// autoload_classmap.php @generated by Composer
|
||||
|
||||
$vendorDir = dirname(__DIR__);
|
||||
$baseDir = dirname($vendorDir);
|
||||
|
||||
return array(
|
||||
'Composer\\InstalledVersions' => $vendorDir . '/composer/InstalledVersions.php',
|
||||
);
|
||||
10
wp-content/plugins/robin-image-optimizer/vendor/composer/autoload_files.php
vendored
Normal file
@@ -0,0 +1,10 @@
|
||||
<?php
|
||||
|
||||
// autoload_files.php @generated by Composer
|
||||
|
||||
$vendorDir = dirname(__DIR__);
|
||||
$baseDir = dirname($vendorDir);
|
||||
|
||||
return array(
|
||||
'e80b779300b5353d4cf50195f3b144e9' => $vendorDir . '/codeinwp/themeisle-sdk/load.php',
|
||||
);
|
||||
9
wp-content/plugins/robin-image-optimizer/vendor/composer/autoload_namespaces.php
vendored
Normal file
@@ -0,0 +1,9 @@
|
||||
<?php
|
||||
|
||||
// autoload_namespaces.php @generated by Composer
|
||||
|
||||
$vendorDir = dirname(__DIR__);
|
||||
$baseDir = dirname($vendorDir);
|
||||
|
||||
return array(
|
||||
);
|
||||
9
wp-content/plugins/robin-image-optimizer/vendor/composer/autoload_psr4.php
vendored
Normal file
@@ -0,0 +1,9 @@
|
||||
<?php
|
||||
|
||||
// autoload_psr4.php @generated by Composer
|
||||
|
||||
$vendorDir = dirname(__DIR__);
|
||||
$baseDir = dirname($vendorDir);
|
||||
|
||||
return array(
|
||||
);
|
||||
48
wp-content/plugins/robin-image-optimizer/vendor/composer/autoload_real.php
vendored
Normal file
@@ -0,0 +1,48 @@
|
||||
<?php
|
||||
|
||||
// autoload_real.php @generated by Composer
|
||||
|
||||
class ComposerAutoloaderInit5ee79b07cb3e29b73ecedab0cd3eb850
|
||||
{
|
||||
private static $loader;
|
||||
|
||||
public static function loadClassLoader($class)
|
||||
{
|
||||
if ('Composer\Autoload\ClassLoader' === $class) {
|
||||
require __DIR__ . '/ClassLoader.php';
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @return \Composer\Autoload\ClassLoader
|
||||
*/
|
||||
public static function getLoader()
|
||||
{
|
||||
if (null !== self::$loader) {
|
||||
return self::$loader;
|
||||
}
|
||||
|
||||
spl_autoload_register(array('ComposerAutoloaderInit5ee79b07cb3e29b73ecedab0cd3eb850', 'loadClassLoader'), true, true);
|
||||
self::$loader = $loader = new \Composer\Autoload\ClassLoader(\dirname(__DIR__));
|
||||
spl_autoload_unregister(array('ComposerAutoloaderInit5ee79b07cb3e29b73ecedab0cd3eb850', 'loadClassLoader'));
|
||||
|
||||
require __DIR__ . '/autoload_static.php';
|
||||
call_user_func(\Composer\Autoload\ComposerStaticInit5ee79b07cb3e29b73ecedab0cd3eb850::getInitializer($loader));
|
||||
|
||||
$loader->register(true);
|
||||
|
||||
$filesToLoad = \Composer\Autoload\ComposerStaticInit5ee79b07cb3e29b73ecedab0cd3eb850::$files;
|
||||
$requireFile = \Closure::bind(static function ($fileIdentifier, $file) {
|
||||
if (empty($GLOBALS['__composer_autoload_files'][$fileIdentifier])) {
|
||||
$GLOBALS['__composer_autoload_files'][$fileIdentifier] = true;
|
||||
|
||||
require $file;
|
||||
}
|
||||
}, null, null);
|
||||
foreach ($filesToLoad as $fileIdentifier => $file) {
|
||||
$requireFile($fileIdentifier, $file);
|
||||
}
|
||||
|
||||
return $loader;
|
||||
}
|
||||
}
|
||||
24
wp-content/plugins/robin-image-optimizer/vendor/composer/autoload_static.php
vendored
Normal file
@@ -0,0 +1,24 @@
|
||||
<?php
|
||||
|
||||
// autoload_static.php @generated by Composer
|
||||
|
||||
namespace Composer\Autoload;
|
||||
|
||||
class ComposerStaticInit5ee79b07cb3e29b73ecedab0cd3eb850
|
||||
{
|
||||
public static $files = array (
|
||||
'e80b779300b5353d4cf50195f3b144e9' => __DIR__ . '/..' . '/codeinwp/themeisle-sdk/load.php',
|
||||
);
|
||||
|
||||
public static $classMap = array (
|
||||
'Composer\\InstalledVersions' => __DIR__ . '/..' . '/composer/InstalledVersions.php',
|
||||
);
|
||||
|
||||
public static function getInitializer(ClassLoader $loader)
|
||||
{
|
||||
return \Closure::bind(function () use ($loader) {
|
||||
$loader->classMap = ComposerStaticInit5ee79b07cb3e29b73ecedab0cd3eb850::$classMap;
|
||||
|
||||
}, null, ClassLoader::class);
|
||||
}
|
||||
}
|
||||
50
wp-content/plugins/robin-image-optimizer/vendor/composer/installed.json
vendored
Normal file
@@ -0,0 +1,50 @@
|
||||
{
|
||||
"packages": [
|
||||
{
|
||||
"name": "codeinwp/themeisle-sdk",
|
||||
"version": "3.3.50",
|
||||
"version_normalized": "3.3.50.0",
|
||||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/Codeinwp/themeisle-sdk.git",
|
||||
"reference": "3c1f8dfc2390e667bbc086c5d660900a7985efa6"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/Codeinwp/themeisle-sdk/zipball/3c1f8dfc2390e667bbc086c5d660900a7985efa6",
|
||||
"reference": "3c1f8dfc2390e667bbc086c5d660900a7985efa6",
|
||||
"shasum": ""
|
||||
},
|
||||
"require-dev": {
|
||||
"codeinwp/phpcs-ruleset": "dev-main",
|
||||
"yoast/phpunit-polyfills": "^2.0"
|
||||
},
|
||||
"time": "2025-11-25T19:36:35+00:00",
|
||||
"type": "library",
|
||||
"installation-source": "dist",
|
||||
"notification-url": "https://packagist.org/downloads/",
|
||||
"license": [
|
||||
"GPL-2.0+"
|
||||
],
|
||||
"authors": [
|
||||
{
|
||||
"name": "ThemeIsle team",
|
||||
"email": "friends@themeisle.com",
|
||||
"homepage": "https://themeisle.com"
|
||||
}
|
||||
],
|
||||
"description": "Themeisle SDK.",
|
||||
"homepage": "https://github.com/Codeinwp/themeisle-sdk",
|
||||
"keywords": [
|
||||
"wordpress"
|
||||
],
|
||||
"support": {
|
||||
"issues": "https://github.com/Codeinwp/themeisle-sdk/issues",
|
||||
"source": "https://github.com/Codeinwp/themeisle-sdk/tree/v3.3.50"
|
||||
},
|
||||
"install-path": "../codeinwp/themeisle-sdk"
|
||||
}
|
||||
],
|
||||
"dev": false,
|
||||
"dev-package-names": []
|
||||
}
|
||||
32
wp-content/plugins/robin-image-optimizer/vendor/composer/installed.php
vendored
Normal file
@@ -0,0 +1,32 @@
|
||||
<?php return array(
|
||||
'root' => array(
|
||||
'name' => 'codeinwp/robin-image-optimizer',
|
||||
'pretty_version' => 'v2.0.4',
|
||||
'version' => '2.0.4.0',
|
||||
'reference' => '33329022874ad4309bad39c217cf66107c8c77d0',
|
||||
'type' => 'library',
|
||||
'install_path' => __DIR__ . '/../../',
|
||||
'aliases' => array(),
|
||||
'dev' => false,
|
||||
),
|
||||
'versions' => array(
|
||||
'codeinwp/robin-image-optimizer' => array(
|
||||
'pretty_version' => 'v2.0.4',
|
||||
'version' => '2.0.4.0',
|
||||
'reference' => '33329022874ad4309bad39c217cf66107c8c77d0',
|
||||
'type' => 'library',
|
||||
'install_path' => __DIR__ . '/../../',
|
||||
'aliases' => array(),
|
||||
'dev_requirement' => false,
|
||||
),
|
||||
'codeinwp/themeisle-sdk' => array(
|
||||
'pretty_version' => '3.3.50',
|
||||
'version' => '3.3.50.0',
|
||||
'reference' => '3c1f8dfc2390e667bbc086c5d660900a7985efa6',
|
||||
'type' => 'library',
|
||||
'install_path' => __DIR__ . '/../codeinwp/themeisle-sdk',
|
||||
'aliases' => array(),
|
||||
'dev_requirement' => false,
|
||||
),
|
||||
),
|
||||
);
|
||||