8 Commits

Author SHA1 Message Date
74cc56005e Add .claude/settings.json to gitignore
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-27 19:33:15 +01:00
136eed2bdd Document v1.3.1 session learnings in CLAUDE.md
- Update version to 1.3.1 and date to 2026-01-27
- Replace completed v1.3.1 roadmap with v1.3.2 placeholder
- Update license client dependency to dev-main
- Add v1.3.1 session history documenting SecureLicenseClient switch

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-27 19:31:06 +01:00
178f86f3e6 Update README for v1.3.1 release
- Added license management feature to admin features list
- Updated PHP requirement from 7.4 to 8.3
- Updated file structure version comment to v1.3.1
- Added v1.3.0 and v1.3.1 changelog entries

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-27 19:27:01 +01:00
7286459ff2 Version 1.3.1 - Switch to SecureLicenseClient with signature verification
- Upgraded from LicenseClient to SecureLicenseClient with HMAC-SHA256 response signature verification
- Added Server Secret configuration field for secure communication
- Added rate limit exception handling with retry time display
- Added signature verification error handling
- Added URL validation error handling (SSRF protection)
- Updated all translation files with new strings
- Compiled .mo files for all 7 language variants

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-27 19:23:42 +01:00
cbe758267e Document v1.3.0 session learnings in CLAUDE.md
- Updated PHP requirement from 7.4 to 8.3 in compatibility notes
- Added Session History section documenting v1.3.0 release session
- Documented key learnings about license client integration

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-25 19:44:16 +01:00
74c14581f1 Release version 1.3.0 - License management and settings sub-tabs
Breaking Changes:
- PHP 8.3+ now required (previously 7.4+)

Added:
- License management integration using magdev/wc-licensed-product-client
- Settings page split into General and License sub-tabs
- License validation and activation via AJAX
- PHP version check with admin notice

Changed:
- Refactored settings class to use modern WooCommerce patterns
- Updated all translations with new license-related strings

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-25 19:39:12 +01:00
0dbe18d954 Remove releases from git tracking and clean up MD5 checksums
- Remove all release packages from git tracking (kept locally)
- Delete MD5 checksum files, keeping only SHA256
- Update .gitignore to exclude releases/ directory
- Update CLAUDE.md release documentation

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-25 19:17:04 +01:00
63c8137f4e Document v1.2.9 learnings in CLAUDE.md
Added three new critical sections documenting lessons learned during v1.2.9 development:

1. WordPress Translation Functions with printf
   - Correct pattern: printf(esc_html__('Text (%s)', 'domain'), value)
   - Wrong pattern: printf(__('Text (%s)', 'domain'), value) - missing text domain
   - Explains why text domain must be in translation function
   - Shows proper output escaping with esc_html

2. Twig Translation Filters and HTML Entity Encoding
   - Explains why translation filters encode special characters in concatenated strings
   - Correct pattern: {{ 'text ' ~ currency_symbol }} (no translation filter)
   - Wrong pattern: {{ ('text ' ~ currency_symbol)|__() }} causes HTML entity encoding
   - Rule: Only translate static text, not concatenated dynamic values

3. Defensive Programming for POST Data Processing
   - Compares v1.2.8 (branching) vs v1.2.9 (defensive) patterns
   - Shows why single decision point is better than multiple branches
   - Key principles: initialize early, minimize branching, guaranteed execution
   - Pattern: initialize → conditionally populate → unconditionally act

Also corrected v1.2.8 examples that had wrong patterns, noting they were fixed in v1.2.9.

These patterns prevent future bugs and ensure consistent, secure implementation.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-30 05:08:09 +01:00
129 changed files with 1836 additions and 382 deletions

7
.gitignore vendored Normal file → Executable file
View File

@@ -33,3 +33,10 @@ notes.*
# local code
wordpress
core
wp-core
# Releases (not tracked in git)
/releases/
# Claude Code local settings
.claude/settings.json

View File

@@ -5,6 +5,85 @@ All notable changes to WooCommerce Tier and Package Prices will be documented in
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
## [1.3.1] - 2026-01-27
### Changed
- **Switched to SecureLicenseClient**: Upgraded from basic `LicenseClient` to `SecureLicenseClient` with HMAC-SHA256 response signature verification for enhanced security against tampering and replay attacks
- **Added Server Secret Configuration**: New "Server Secret" field in License settings for secure communication with the license server
### Added
- **Rate Limit Handling**: Added proper handling of `RateLimitExceededException` with user-friendly messages showing retry wait time
- **Signature Verification Error Handling**: Added dedicated handling for `SignatureException` when response signatures fail verification
- **URL Validation Error Handling**: Added handling for `InvalidArgumentException` from SSRF protection in the license client
### Security
- Response signatures are now verified using HMAC-SHA256 with license-specific derived keys (RFC 5869 HKDF)
- The license client now validates server URLs to prevent SSRF attacks (blocks private IP ranges)
- HTTP connections require HTTPS unless explicitly allowed for localhost testing
### Technical Details
**License Client Upgrade**:
- Changed from `LicenseClient` to `SecureLicenseClient`
- Added `serverSecret` parameter for signature verification
- Library updated from `v0.1.0` to `dev-main` with new security features
**New Exception Handling**:
- `RateLimitExceededException` - shows retry time to user
- `SignatureException` - indicates server secret mismatch
- `InvalidArgumentException` - invalid/blocked URL detected
**New Settings Field**:
- `wc_tpp_license_server_secret` (password type) for the shared secret
---
## [1.3.0] - 2026-01-25
### Breaking Changes
- **PHP 8.3 Required**: Minimum PHP version increased from 7.4 to 8.3 to support modern dependencies and the license client library. Users on older PHP versions will see an admin notice and the plugin will not load.
### Added
- **License Management**: Integrated `magdev/wc-licensed-product-client` library for license validation and activation
- New "License" settings tab for entering license server URL and license key
- License validation and activation via AJAX with visual feedback
- License status display showing active/inactive state, expiration date, and last check time
- Cached license status with daily auto-refresh
- **Settings Page Sub-tabs**: Split the settings page into "General" and "License" tabs using modern WooCommerce patterns
- Refactored to use `get_own_sections()` and `get_settings_for_{section}_section()` methods
- Improved navigation and organization of settings
- **PHP Version Check**: Added runtime PHP version validation with admin notice for incompatible servers
### Changed
- Updated composer.json to require PHP 8.3+ and added `magdev/wc-licensed-product-client` dependency
- Settings class now uses modern WooCommerce settings API patterns
### Technical Details
**New Dependencies**:
- `magdev/wc-licensed-product-client: ^0.1` (from private repository)
- `symfony/http-client: ^7.0` (transitive)
- `psr/log: ^3.0`, `psr/cache: ^3.0`, `psr/http-client: ^1.0` (transitive)
**License Client Integration**:
- Uses `LicenseClient` class for API communication
- AJAX endpoints: `wc_tpp_validate_license`, `wc_tpp_activate_license`
- License status cached in WordPress transient (`wc_tpp_license_status`)
---
## [1.2.9] - 2025-12-30
### Fixed

248
CLAUDE.md
View File

@@ -1,7 +1,7 @@
# WooCommerce Tier and Package Prices - AI Context Document
**Last Updated:** 2025-12-30
**Current Version:** 1.2.9
**Last Updated:** 2026-01-27
**Current Version:** 1.3.1
**Author:** Marco Graetsch
**Project Status:** Production-ready WordPress plugin
@@ -16,9 +16,17 @@ This is a WooCommerce plugin that adds flexible pricing capabilities to products
This project is proudly **"vibe-coded"** using Claude.AI - the entire codebase was created through AI assistance.
## Temporary Roadmap
**Note for AI Assistants:** Clean this section after the specific features are done or new releases are made. Effective changes are tracked in `CHANGELOG.md`. Do not add completed versions here - document them in the Session History section at the end of this file. Always keep the `Known Bugs` section and create a section with the next bugfix and minor version after a release.
### Version 1.3.2
- No planned changes yet
## Technical Stack
- **Language:** PHP 7.4+
- **Language:** PHP 8.3+
- **Framework:** WordPress Plugin API
- **E-commerce:** WooCommerce 8.0+ (tested up to 10.x)
- **Template Engine:** Twig 3.0 (via Composer)
@@ -32,8 +40,11 @@ This project is proudly **"vibe-coded"** using Claude.AI - the entire codebase w
```json
{
"twig/twig": "^3.0",
"symfony/polyfill-ctype": "^1.x",
"symfony/polyfill-mbstring": "^1.x"
"magdev/wc-licensed-product-client": "dev-main",
"symfony/http-client": "^7.0",
"psr/log": "^3.0",
"psr/cache": "^3.0",
"psr/http-client": "^1.0"
}
```
@@ -379,9 +390,8 @@ zip -r wc-tier-and-package-prices/releases/wc-tier-and-package-prices-X.X.X.zip
# Return to project directory
cd wc-tier-and-package-prices
# Generate checksums
# Generate SHA256 checksum
cd releases
md5sum wc-tier-and-package-prices-X.X.X.zip > wc-tier-and-package-prices-X.X.X.zip.md5
sha256sum wc-tier-and-package-prices-X.X.X.zip > wc-tier-and-package-prices-X.X.X.zip.sha256
cd ..
```
@@ -676,9 +686,10 @@ Based on v1.1.22, v1.2.2, and v1.2.3 release experience, here's the complete wor
- `composer.json` - Version bump
- `CHANGELOG.md` - Release notes
- `CLAUDE.md` - Version and roadmap updates
- `releases/wc-tier-and-package-prices-X.X.X.zip*` - Package and checksums
- Feature-specific files (templates, PHP classes, etc.)
**Note:** Release packages (`releases/`) are not tracked in git - they are generated locally for distribution.
### Release Package Creation - Critical Notes
**IMPORTANT:** The zip command must be run from the **parent directory** to create proper archive structure.
@@ -821,15 +832,15 @@ WooCommerce has different hooks for different product types in the admin product
**Lesson:** When adding admin UI for variable product parents, use `woocommerce_product_options_general_product_data` and check `$product->is_type('variable')` to conditionally display. Using `woocommerce_product_options_pricing` will cause forms to never appear for variable products (as discovered in v1.2.6 → v1.2.7 fix).
#### CRITICAL: Currency Symbol Display (Learned in v1.2.8)
#### CRITICAL: Currency Symbol Display (Learned in v1.2.8, Corrected in v1.2.9)
When displaying currency symbols in admin interface table headers and input placeholders:
**Table Headers:**
```php
// ✅ Correct - Use printf with translation and WooCommerce currency function
<th><?php printf(__('Price (%s)', 'wc-tier-package-prices'), get_woocommerce_currency_symbol()); ?></th>
// ✅ Correct - Use printf with esc_html__ for translation (CORRECTED in v1.2.9)
<th><?php printf(esc_html__('Price (%s)', 'wc-tier-package-prices'), get_woocommerce_currency_symbol()); ?></th>
// ❌ Wrong - Hard-coded or missing currency
<th><?php _e('Price', 'wc-tier-package-prices'); ?></th>
@@ -839,8 +850,8 @@ When displaying currency symbols in admin interface table headers and input plac
**Twig Template Placeholders:**
```twig
{# ✅ Correct - Pass currency_symbol from PHP and concatenate in template #}
placeholder="{{ ('e.g., 9.99 ' ~ currency_symbol)|__('wc-tier-package-prices') }}"
{# ✅ Correct - Pass currency_symbol from PHP and concatenate (CORRECTED in v1.2.9 - no translation filter) #}
placeholder="{{ 'e.g., 9.99 ' ~ currency_symbol }}"
{# ❌ Wrong - Hard-coded or missing currency #}
placeholder="{{ 'e.g., 9.99'|__('wc-tier-package-prices') }}"
@@ -917,6 +928,133 @@ if (isset($_POST['_wc_tpp_tiers'])) {
**Rule:** Always check `if (!empty($array))` before calling `update_post_meta()` for array data. If empty, call `delete_post_meta()` instead.
#### CRITICAL: WordPress Translation Functions with printf (Learned in v1.2.9)
When using `printf()` with WordPress translation functions, the text domain must be passed to the translation function, NOT to printf:
**Wrong Pattern:**
```php
// ❌ WRONG - Text domain not passed to translation function
printf(__('Price (%s)', 'wc-tier-package-prices'), get_woocommerce_currency_symbol());
```
**Problem:** The `__()` function receives the text domain as a second parameter, but in this pattern it's missing. This causes the string "Price (%s)" to not be found in translation files, so it won't be translated.
**Correct Pattern:**
```php
// ✅ CORRECT - Text domain in translation function, with esc_html for output escaping
printf(esc_html__('Price (%s)', 'wc-tier-package-prices'), get_woocommerce_currency_symbol());
```
**Why This Works:**
- `esc_html__('Price (%s)', 'wc-tier-package-prices')` translates the string and returns it
- `printf()` then substitutes the `%s` placeholder with the currency symbol
- The translated string is used in the final output
- `esc_html` ensures proper output escaping
**Applied in v1.2.9:** All 6 table headers in `includes/class-wc-tpp-product-meta.php`
#### CRITICAL: Twig Translation Filters and HTML Entity Encoding (Learned in v1.2.9)
When concatenating dynamic values in Twig templates, applying the translation filter can cause HTML entity encoding issues:
**Wrong Pattern:**
```twig
{# ❌ WRONG - Translation filter encodes special characters in concatenated string #}
placeholder="{{ ('e.g., 9.99 ' ~ currency_symbol)|__('wc-tier-package-prices') }}"
```
**Problem:** When `currency_symbol` contains special characters (€, £, ¥, etc.), the concatenated string is passed through the translation function which treats it as a translatable string and encodes special characters as HTML entities (`&euro;`, `&pound;`, etc.).
**Correct Pattern:**
```twig
{# ✅ CORRECT - No translation filter, just concatenation #}
placeholder="{{ 'e.g., 9.99 ' ~ currency_symbol }}"
```
**Why This Works:**
- Placeholder examples don't need translation - they're illustrative values
- Direct concatenation preserves special characters
- Currency symbol displays correctly (€ instead of &euro;)
**Rule:** Only apply translation filters to static text that needs translation, not to concatenated strings with dynamic values that contain special characters.
**Applied in v1.2.9:**
- `templates/admin/tier-row.twig` - Price input placeholder
- `templates/admin/package-row.twig` - Price input placeholder
#### CRITICAL: Defensive Programming for POST Data Processing (Learned in v1.2.9)
The v1.2.8 fix for variation pricing deletion had the right logic but used a branching structure that could miss edge cases. The v1.2.9 refactor demonstrates a more defensive pattern:
**Less Defensive Pattern (v1.2.8):**
```php
// ❌ BRITTLE - Multiple branches, easy to miss edge cases
if (isset($_POST['wc_tpp_tiers'][$loop])) {
$tiers = array();
foreach ($_POST['wc_tpp_tiers'][$loop] as $tier) {
// ... populate tiers ...
}
if (!empty($tiers)) {
update_post_meta($variation_id, '_wc_tpp_tiers', $tiers);
} else {
delete_post_meta($variation_id, '_wc_tpp_tiers');
}
} else {
delete_post_meta($variation_id, '_wc_tpp_tiers');
}
```
**Problem:** Two separate code paths to `delete_post_meta()`. If logic changes, easy to update one path but forget the other.
**More Defensive Pattern (v1.2.9):**
```php
// ✅ DEFENSIVE - Single decision point, guaranteed cleanup
$tiers = array();
if (isset($_POST['wc_tpp_tiers'][$loop]) && is_array($_POST['wc_tpp_tiers'][$loop])) {
foreach ($_POST['wc_tpp_tiers'][$loop] as $tier) {
// ... populate tiers ...
}
}
// Always execute one of these based on final state
if (!empty($tiers)) {
update_post_meta($variation_id, '_wc_tpp_tiers', $tiers);
} else {
delete_post_meta($variation_id, '_wc_tpp_tiers');
}
```
**Why This Is Better:**
- Initialize array at the start (guaranteed initial state)
- Single conditional for populating (with extra `is_array()` safety check)
- Single decision point for save/delete (one place to maintain)
- Impossible to have a code path that doesn't call either update or delete
- Much easier to reason about and modify
**Key Principles:**
1. **Initialize variables early** - Establish known initial state
2. **Minimize branching** - Fewer code paths = fewer bugs
3. **Single decision point** - One place determines final action
4. **Add safety checks** - Validate assumptions (`is_array()`)
5. **Guaranteed execution** - Always perform one of update/delete, never neither
**Applied in v1.2.9:**
- `save_variation_pricing_fields()` - Both tier and package pricing logic refactored
**Rule:** When processing user input to decide between update and delete, prefer the pattern: initialize → conditionally populate → unconditionally act based on final state.
### When Adding Features
- Follow the existing pattern: add setting → add UI → add logic → add template
@@ -963,8 +1101,8 @@ if (isset($_POST['_wc_tpp_tiers'])) {
### PHP
- Minimum: 7.4
- Uses modern PHP features (type hints acceptable in new code)
- Minimum: 8.3 (breaking change in v1.3.0)
- Uses modern PHP features (type hints, named arguments, etc.)
- Composer autoloader handles namespacing
### Browsers
@@ -1000,4 +1138,84 @@ The plugin architecture is solid and well-tested. Most bugs arise from:
---
## Session History
### v1.3.0 Release Session (2026-01-25)
**Accomplished:**
1. Fixed known bugs from CLAUDE.md:
- Removed all releases from git tracking (`git rm --cached -r releases/`)
- Deleted all MD5 checksum files, keeping only SHA256
- Updated `.gitignore` to exclude `/releases/`
2. Implemented v1.3.0 features:
- Bumped PHP requirement from 7.4 to 8.3 (breaking change)
- Added PHP version check with admin notice for incompatible servers
- Added `magdev/wc-licensed-product-client` library integration
- Refactored settings page to use WooCommerce modern sub-tabs pattern (`get_own_sections()`)
- Created "General" and "License" sub-tabs
- Implemented license management with AJAX validation/activation
- Added license status caching via WordPress transients
- Added CSS styling for license status display
3. Updated all translation files:
- Added 28 new translation strings for license management
- Updated .pot template and all 7 .po files
- Compiled all .mo files
4. Created release package:
- Package size: 737KB (increased due to new dependencies)
- 642 files included (more due to license client library)
- SHA256 checksum generated
**Key Learnings:**
- WooCommerce settings sub-tabs use `get_own_sections()` and `get_settings_for_{section}_section()` pattern
- License client library `magdev/wc-licensed-product-client` is from private Gitea repo - requires `repositories` config in composer.json
- Package version was `^0.1` not `^1.0` - always check available versions before setting constraint
- Release package size increased from ~430KB to ~737KB due to new Composer dependencies
### v1.3.1 Release Session (2026-01-27)
**Accomplished:**
1. Reviewed server-implementation.md from license client library
2. Upgraded from `LicenseClient` to `SecureLicenseClient`:
- Added "Server Secret" configuration field in License settings tab
- Updated `get_license_client()` to use `SecureLicenseClient` with HMAC-SHA256 signature verification
- Updated JavaScript to pass `server_secret` in AJAX requests
3. Added comprehensive exception handling:
- `RateLimitExceededException` - Shows retry time to user
- `SignatureException` - Reports signature verification failures
- `InvalidArgumentException` - SSRF protection for private IP blocking
4. Updated Composer dependencies:
- Changed `magdev/wc-licensed-product-client` from `^0.1` to `dev-main`
- Library includes new security features (SSRF protection, response signatures)
5. Updated translations:
- Added 7 new translation strings for v1.3.1
- All 7 language files updated and compiled
6. Created release package:
- Package size: 745KB
- 642 files included
- SHA256 checksum generated
7. Updated README.md:
- Documented license management feature
- Updated PHP requirement to 8.3
- Added v1.3.0 and v1.3.1 changelog entries
**Key Learnings:**
- `SecureLicenseClient` requires `serverSecret` parameter for HMAC verification
- The library's exception classes are in `Magdev\WcLicensedProductClient\Exception\` and `Magdev\WcLicensedProductClient\Security\` namespaces
- `RateLimitExceededException` has a `retryAfter` property (int seconds)
- Always check library documentation in `docs/server-implementation.md` for implementation requirements
---
Always refer to this document when starting work on this project. Good luck!

View File

@@ -20,10 +20,11 @@ A powerful WooCommerce plugin that adds tier pricing and package pricing functio
### Admin Features
- Easy-to-use product meta boxes for adding tiers and packages
- Global settings page under WooCommerce menu
- Global settings page under WooCommerce menu with sub-tabs
- Configure display position (before/after add to cart, after price)
- Enable/disable tier or package pricing independently
- Sortable pricing rules
- License management with secure HMAC signature verification
### Frontend Features
- Beautiful pricing tables on product pages
@@ -110,7 +111,7 @@ When editing a product, scroll to the **Product data** panel:
```
wc-tier-and-package-prices/
├── wc-tier-and-package-prices.php # Main plugin file (v1.1.20)
├── wc-tier-and-package-prices.php # Main plugin file (v1.3.1)
├── includes/
│ ├── class-wc-tpp-admin.php # Admin settings integration
│ ├── class-wc-tpp-settings.php # WooCommerce settings page
@@ -149,7 +150,7 @@ wc-tier-and-package-prices/
- WordPress 6.0 or higher (tested up to 6.9.x)
- WooCommerce 8.0 or higher (tested up to 10.x)
- PHP 7.4 or higher
- PHP 8.3 or higher (required since v1.3.0)
### Compatibility
@@ -183,9 +184,32 @@ This plugin is licensed under the GPL v2 or later.
## Changelog
### Version 1.3.1 - 2026-01-27
__Current Release__ - Secure License Client
- __Changed__: Switched to `SecureLicenseClient` with HMAC-SHA256 response signature verification
- __New__: Server Secret configuration field for secure communication with license server
- __New__: Rate limit exception handling with retry time display
- __New__: Signature verification and URL validation error handling
- __Security__: Response signatures verified using HMAC-SHA256 with license-specific derived keys
See [CHANGELOG.md](CHANGELOG.md) for complete details.
### Version 1.3.0 - 2026-01-25
__Breaking Changes__ - PHP 8.3+ Required
- __Breaking__: Minimum PHP version increased from 7.4 to 8.3
- __New__: License management via `magdev/wc-licensed-product-client` library
- __New__: Settings page split into "General" and "License" sub-tabs
- __New__: AJAX-based license validation and activation with visual feedback
- __New__: License status caching with daily auto-refresh
- __New__: PHP version check with admin notice for incompatible servers
### Version 1.2.0 - 2025-12-29
__Current Release__ - Variable Product Support
Variable Product Support
- __New__: Full support for WooCommerce variable products with variation-level pricing
- __New__: Each variation can have independent tier and package pricing configuration
@@ -194,24 +218,6 @@ __Current Release__ - Variable Product Support
- __Fixed__: Quantity restrictions now work correctly per-variation
- 100% backward compatible - no breaking changes
See [CHANGELOG.md](CHANGELOG.md) for complete details.
### Version 1.1.22 - 2025-12-23
- Increased width of label input fields in admin interface
#### Fixed
- **CRITICAL:** WooCommerce Blocks fatal error in mini-cart and cart blocks
- Fixed `woocommerce_store_api_product_quantity_editable` filter signature mismatch
- Filter now correctly accepts `WC_Product` object instead of cart item array
- Resolves "Cannot use object of type WC_Product_Simple as array" fatal error
#### Technical Details
- Updated `block_quantity_editable()` method signature to accept product object
- Changed parameter from `$cart_item` array to `WC_Product $product`
- Uses `$product->get_id()` instead of array access for product ID
- Full compatibility with WooCommerce Store API and block-based cart/checkout
### Recent Major Updates
#### Version 1.1.7 - Enhanced Tier Pricing

View File

@@ -154,3 +154,52 @@ label[for^="wc_tpp_restrict_to_packages_"] {
input[id^="wc_tpp_restrict_to_packages_"] + .description {
display: none;
}
/* License Status Styling */
.wc-tpp-license-active {
color: #46b450;
font-weight: 600;
}
.wc-tpp-license-inactive {
color: #dc3232;
font-weight: 600;
}
.wc-tpp-license-expired {
color: #ffb900;
font-weight: 600;
}
#wc-tpp-license-spinner {
float: none;
margin-top: 0;
vertical-align: middle;
}
#wc-tpp-validate-license,
#wc-tpp-activate-license {
margin-right: 8px;
}
#wc-tpp-license-status-container {
margin-bottom: 10px;
padding: 10px 15px;
background: #f9f9f9;
border-left: 4px solid #ccc;
border-radius: 2px;
}
#wc-tpp-license-status-container.valid {
border-left-color: #46b450;
background: #f0fff0;
}
#wc-tpp-license-status-container.invalid {
border-left-color: #dc3232;
background: #fff0f0;
}
#wc-tpp-license-status-container small {
color: #666;
}

View File

@@ -1,7 +1,7 @@
{
"name": "magdev/wc-tier-package-prices",
"description": "WooCommerce plugin for tier pricing and package prices with Twig templates",
"version": "1.2.9",
"version": "1.3.1",
"type": "wordpress-plugin",
"license": "GPL-2.0-or-later",
"authors": [
@@ -10,9 +10,16 @@
"homepage": "https://src.bundespruefstelle.ch/magdev"
}
],
"repositories": [
{
"type": "vcs",
"url": "https://src.bundespruefstelle.ch/magdev/wc-licensed-product-client.git"
}
],
"require": {
"php": ">=7.4",
"twig/twig": "^3.0"
"php": ">=8.3",
"twig/twig": "^3.0",
"magdev/wc-licensed-product-client": "dev-main"
},
"autoload": {
"classmap": [

428
includes/class-wc-tpp-settings.php Normal file → Executable file
View File

@@ -2,7 +2,7 @@
/**
* WooCommerce Settings Integration
*
* Adds Tier & Package Prices settings to WooCommerce Settings > Advanced tab
* Adds Tier & Package Prices settings to WooCommerce Settings with sub-tabs
*
* @package WC_Tier_Package_Prices
*/
@@ -29,32 +29,31 @@ if (!class_exists('WC_TPP_Settings')) {
$this->label = __('Tier & Package Prices', 'wc-tier-package-prices');
parent::__construct();
// Add AJAX handlers for license validation
add_action('wp_ajax_wc_tpp_validate_license', array($this, 'ajax_validate_license'));
add_action('wp_ajax_wc_tpp_activate_license', array($this, 'ajax_activate_license'));
}
/**
* Get sections
* Get own sections - Modern WooCommerce pattern
*
* @return array
*/
public function get_sections() {
$sections = array(
protected function get_own_sections() {
return array(
'' => __('General', 'wc-tier-package-prices'),
'license' => __('License', 'wc-tier-package-prices'),
);
return apply_filters('woocommerce_get_sections_' . $this->id, $sections);
}
/**
* Get settings array
* Get settings for the default (General) section
*
* @param string $current_section Current section name.
* @return array
*/
public function get_settings($current_section = '') {
$settings = array();
if ('' === $current_section) {
$settings = array(
protected function get_settings_for_default_section() {
return array(
array(
'title' => __('Tier & Package Prices Settings', 'wc-tier-package-prices'),
'type' => 'title',
@@ -121,23 +120,410 @@ if (!class_exists('WC_TPP_Settings')) {
);
}
return apply_filters('woocommerce_get_settings_' . $this->id, $settings, $current_section);
/**
* Get settings for the License section
*
* @return array
*/
protected function get_settings_for_license_section() {
return array(
array(
'title' => __('License Management', 'wc-tier-package-prices'),
'type' => 'title',
'desc' => __('Enter your license key to receive updates and support.', 'wc-tier-package-prices'),
'id' => 'wc_tpp_license_settings',
),
array(
'title' => __('License Server URL', 'wc-tier-package-prices'),
'desc' => __('The URL of the license server.', 'wc-tier-package-prices'),
'id' => 'wc_tpp_license_server_url',
'type' => 'url',
'default' => '',
'css' => 'min-width:400px;',
'desc_tip' => true,
),
array(
'title' => __('License Key', 'wc-tier-package-prices'),
'desc' => __('Your license key for this plugin.', 'wc-tier-package-prices'),
'id' => 'wc_tpp_license_key',
'type' => 'text',
'default' => '',
'css' => 'min-width:400px;',
'desc_tip' => true,
),
array(
'title' => __('Server Secret', 'wc-tier-package-prices'),
'desc' => __('The shared secret for secure communication with the license server.', 'wc-tier-package-prices'),
'id' => 'wc_tpp_license_server_secret',
'type' => 'password',
'default' => '',
'css' => 'min-width:400px;',
'desc_tip' => true,
),
array(
'title' => __('License Status', 'wc-tier-package-prices'),
'type' => 'wc_tpp_license_status',
'id' => 'wc_tpp_license_status_display',
),
array(
'type' => 'sectionend',
'id' => 'wc_tpp_license_settings',
),
);
}
/**
* Get cached license status
*
* @return array|false
*/
private function get_cached_license_status() {
return get_transient('wc_tpp_license_status');
}
/**
* AJAX handler for license validation
*/
public function ajax_validate_license() {
check_ajax_referer('wc_tpp_license_nonce', 'nonce');
if (!current_user_can('manage_woocommerce')) {
wp_send_json_error(array('message' => __('Permission denied.', 'wc-tier-package-prices')));
}
$license_key = sanitize_text_field(wp_unslash($_POST['license_key'] ?? ''));
$server_url = esc_url_raw(wp_unslash($_POST['server_url'] ?? ''));
$server_secret = sanitize_text_field(wp_unslash($_POST['server_secret'] ?? ''));
if (empty($license_key) || empty($server_url) || empty($server_secret)) {
wp_send_json_error(array('message' => __('License key, server URL, and server secret are required.', 'wc-tier-package-prices')));
}
try {
$client = $this->get_license_client($server_url, $server_secret);
$domain = $this->get_current_domain();
$result = $client->validate($license_key, $domain);
// Cache the status
set_transient('wc_tpp_license_status', array(
'valid' => true,
'product_id' => $result->productId,
'expires_at' => $result->expiresAt?->format('Y-m-d H:i:s'),
'is_lifetime' => $result->isLifetime(),
'checked_at' => current_time('mysql'),
), DAY_IN_SECONDS);
wp_send_json_success(array(
'message' => __('License is valid!', 'wc-tier-package-prices'),
'status' => $this->get_cached_license_status(),
));
} catch (\Magdev\WcLicensedProductClient\Exception\RateLimitExceededException $e) {
wp_send_json_error(array(
'message' => sprintf(
/* translators: %d: Number of seconds to wait */
__('Rate limit exceeded. Please try again in %d seconds.', 'wc-tier-package-prices'),
$e->retryAfter ?? 60
),
'code' => 'rate_limit_exceeded',
'retry_after' => $e->retryAfter,
));
} catch (\Magdev\WcLicensedProductClient\Security\SignatureException $e) {
delete_transient('wc_tpp_license_status');
wp_send_json_error(array(
'message' => __('Response signature verification failed. Please check your server secret.', 'wc-tier-package-prices'),
'code' => 'signature_error',
));
} catch (\Magdev\WcLicensedProductClient\Exception\LicenseException $e) {
delete_transient('wc_tpp_license_status');
wp_send_json_error(array(
'message' => $e->getMessage(),
'code' => $e->errorCode ?? 'unknown',
));
} catch (\InvalidArgumentException $e) {
wp_send_json_error(array(
'message' => $e->getMessage(),
'code' => 'invalid_url',
));
} catch (\Exception $e) {
delete_transient('wc_tpp_license_status');
wp_send_json_error(array(
'message' => __('An unexpected error occurred. Please try again.', 'wc-tier-package-prices'),
'code' => 'exception',
));
}
}
/**
* AJAX handler for license activation
*/
public function ajax_activate_license() {
check_ajax_referer('wc_tpp_license_nonce', 'nonce');
if (!current_user_can('manage_woocommerce')) {
wp_send_json_error(array('message' => __('Permission denied.', 'wc-tier-package-prices')));
}
$license_key = sanitize_text_field(wp_unslash($_POST['license_key'] ?? ''));
$server_url = esc_url_raw(wp_unslash($_POST['server_url'] ?? ''));
$server_secret = sanitize_text_field(wp_unslash($_POST['server_secret'] ?? ''));
if (empty($license_key) || empty($server_url) || empty($server_secret)) {
wp_send_json_error(array('message' => __('License key, server URL, and server secret are required.', 'wc-tier-package-prices')));
}
try {
$client = $this->get_license_client($server_url, $server_secret);
$domain = $this->get_current_domain();
$result = $client->activate($license_key, $domain);
if ($result->success) {
// Validate to get full status after activation
$validate_result = $client->validate($license_key, $domain);
set_transient('wc_tpp_license_status', array(
'valid' => true,
'product_id' => $validate_result->productId,
'expires_at' => $validate_result->expiresAt?->format('Y-m-d H:i:s'),
'is_lifetime' => $validate_result->isLifetime(),
'checked_at' => current_time('mysql'),
), DAY_IN_SECONDS);
wp_send_json_success(array(
'message' => __('License activated successfully!', 'wc-tier-package-prices'),
'status' => $this->get_cached_license_status(),
));
}
wp_send_json_error(array('message' => $result->message));
} catch (\Magdev\WcLicensedProductClient\Exception\RateLimitExceededException $e) {
wp_send_json_error(array(
'message' => sprintf(
/* translators: %d: Number of seconds to wait */
__('Rate limit exceeded. Please try again in %d seconds.', 'wc-tier-package-prices'),
$e->retryAfter ?? 60
),
'code' => 'rate_limit_exceeded',
'retry_after' => $e->retryAfter,
));
} catch (\Magdev\WcLicensedProductClient\Security\SignatureException $e) {
wp_send_json_error(array(
'message' => __('Response signature verification failed. Please check your server secret.', 'wc-tier-package-prices'),
'code' => 'signature_error',
));
} catch (\Magdev\WcLicensedProductClient\Exception\LicenseException $e) {
wp_send_json_error(array(
'message' => $e->getMessage(),
'code' => $e->errorCode ?? 'unknown',
));
} catch (\InvalidArgumentException $e) {
wp_send_json_error(array(
'message' => $e->getMessage(),
'code' => 'invalid_url',
));
} catch (\Exception $e) {
wp_send_json_error(array(
'message' => __('An unexpected error occurred. Please try again.', 'wc-tier-package-prices'),
'code' => 'exception',
));
}
}
/**
* Get license client instance
*
* Uses SecureLicenseClient for HMAC signature verification.
*
* @param string $server_url License server URL.
* @param string $server_secret Shared secret for signature verification.
* @return \Magdev\WcLicensedProductClient\LicenseClientInterface
*/
private function get_license_client(string $server_url, string $server_secret): \Magdev\WcLicensedProductClient\LicenseClientInterface {
$httpClient = \Symfony\Component\HttpClient\HttpClient::create();
return new \Magdev\WcLicensedProductClient\SecureLicenseClient(
httpClient: $httpClient,
baseUrl: $server_url,
serverSecret: $server_secret,
);
}
/**
* Get current domain for license validation
*
* @return string
*/
private function get_current_domain(): string {
return wp_parse_url(home_url(), PHP_URL_HOST);
}
/**
* Output the settings
*/
public function output() {
$settings = $this->get_settings();
WC_Admin_Settings::output_fields($settings);
global $current_section;
// Register custom field type for license status display
add_action('woocommerce_admin_field_wc_tpp_license_status', array($this, 'output_license_status_field'));
parent::output();
// Add JavaScript for license section
if ('license' === $current_section) {
$this->output_license_scripts();
}
}
/**
* Save settings
* Output license status custom field
*
* @param array $value Field configuration.
*/
public function save() {
$settings = $this->get_settings();
WC_Admin_Settings::save_fields($settings);
public function output_license_status_field($value) {
$status = $this->get_cached_license_status();
?>
<tr valign="top">
<th scope="row" class="titledesc">
<label><?php esc_html_e('License Status', 'wc-tier-package-prices'); ?></label>
</th>
<td class="forminp">
<div id="wc-tpp-license-status-container" class="<?php echo !empty($status['valid']) ? 'valid' : 'invalid'; ?>">
<?php $this->render_license_status_html($status); ?>
</div>
<p class="description" style="margin-top: 10px;">
<button type="button" class="button" id="wc-tpp-validate-license">
<?php esc_html_e('Validate License', 'wc-tier-package-prices'); ?>
</button>
<button type="button" class="button" id="wc-tpp-activate-license">
<?php esc_html_e('Activate License', 'wc-tier-package-prices'); ?>
</button>
<span class="spinner" id="wc-tpp-license-spinner"></span>
</p>
</td>
</tr>
<?php
}
/**
* Render license status HTML
*
* @param array|false $status License status data.
*/
private function render_license_status_html($status) {
if (empty($status)) {
echo '<span class="wc-tpp-license-inactive">' . esc_html__('No license activated', 'wc-tier-package-prices') . '</span>';
return;
}
if (!empty($status['valid'])) {
echo '<span class="wc-tpp-license-active">' . esc_html__('License Active', 'wc-tier-package-prices') . '</span>';
if (!empty($status['expires_at']) && empty($status['is_lifetime'])) {
echo '<br><small>' . sprintf(
/* translators: %s: Expiration date */
esc_html__('Expires: %s', 'wc-tier-package-prices'),
esc_html($status['expires_at'])
) . '</small>';
} elseif (!empty($status['is_lifetime'])) {
echo '<br><small>' . esc_html__('Lifetime License', 'wc-tier-package-prices') . '</small>';
}
if (!empty($status['checked_at'])) {
echo '<br><small>' . sprintf(
/* translators: %s: Last check timestamp */
esc_html__('Last checked: %s', 'wc-tier-package-prices'),
esc_html($status['checked_at'])
) . '</small>';
}
} else {
echo '<span class="wc-tpp-license-inactive">' . esc_html__('License Invalid', 'wc-tier-package-prices') . '</span>';
}
}
/**
* Output JavaScript for license management
*/
private function output_license_scripts() {
$nonce = wp_create_nonce('wc_tpp_license_nonce');
?>
<script type="text/javascript">
jQuery(function($) {
var $validateBtn = $('#wc-tpp-validate-license');
var $activateBtn = $('#wc-tpp-activate-license');
var $spinner = $('#wc-tpp-license-spinner');
function getLicenseData() {
return {
license_key: $('#wc_tpp_license_key').val(),
server_url: $('#wc_tpp_license_server_url').val(),
server_secret: $('#wc_tpp_license_server_secret').val(),
nonce: '<?php echo esc_js($nonce); ?>'
};
}
function showSpinner() {
$spinner.addClass('is-active');
$validateBtn.prop('disabled', true);
$activateBtn.prop('disabled', true);
}
function hideSpinner() {
$spinner.removeClass('is-active');
$validateBtn.prop('disabled', false);
$activateBtn.prop('disabled', false);
}
$validateBtn.on('click', function() {
var data = getLicenseData();
if (!data.license_key || !data.server_url || !data.server_secret) {
alert('<?php echo esc_js(__('Please enter license server URL, license key, and server secret.', 'wc-tier-package-prices')); ?>');
return;
}
showSpinner();
$.post(ajaxurl, $.extend({action: 'wc_tpp_validate_license'}, data))
.done(function(response) {
if (response.success) {
alert(response.data.message);
location.reload();
} else {
alert(response.data.message || '<?php echo esc_js(__('Validation failed.', 'wc-tier-package-prices')); ?>');
}
})
.fail(function() {
alert('<?php echo esc_js(__('Request failed. Please try again.', 'wc-tier-package-prices')); ?>');
})
.always(hideSpinner);
});
$activateBtn.on('click', function() {
var data = getLicenseData();
if (!data.license_key || !data.server_url || !data.server_secret) {
alert('<?php echo esc_js(__('Please enter license server URL, license key, and server secret.', 'wc-tier-package-prices')); ?>');
return;
}
showSpinner();
$.post(ajaxurl, $.extend({action: 'wc_tpp_activate_license'}, data))
.done(function(response) {
if (response.success) {
alert(response.data.message);
location.reload();
} else {
alert(response.data.message || '<?php echo esc_js(__('Activation failed.', 'wc-tier-package-prices')); ?>');
}
})
.fail(function() {
alert('<?php echo esc_js(__('Request failed. Please try again.', 'wc-tier-package-prices')); ?>');
})
.always(hideSpinner);
});
});
</script>
<?php
}
}
}
}

View File

@@ -277,3 +277,124 @@ msgstr "Preis"
#: includes/class-wc-tpp-product-meta.php:165
msgid "Label (optional)"
msgstr "Beschriftung (optional)"
#: wc-tier-and-package-prices.php
msgid "WooCommerce Tier and Package Prices requires PHP 8.3 or higher. Your server is running PHP %s."
msgstr "WooCommerce Staffel- und Paketpreise erfordert PHP 8.3 oder höher. Ihr Server verwendet PHP %s."
#: includes/class-wc-tpp-settings.php
msgid "License"
msgstr "Lizenz"
#: includes/class-wc-tpp-settings.php
msgid "License Management"
msgstr "Lizenzverwaltung"
#: includes/class-wc-tpp-settings.php
msgid "Enter your license key to receive updates and support."
msgstr "Geben Sie Ihren Lizenzschlüssel ein, um Updates und Support zu erhalten."
#: includes/class-wc-tpp-settings.php
msgid "License Server URL"
msgstr "Lizenzserver-URL"
#: includes/class-wc-tpp-settings.php
msgid "The URL of the license server."
msgstr "Die URL des Lizenzservers."
#: includes/class-wc-tpp-settings.php
msgid "License Key"
msgstr "Lizenzschlüssel"
#: includes/class-wc-tpp-settings.php
msgid "Your license key for this plugin."
msgstr "Ihr Lizenzschlüssel für dieses Plugin."
#: includes/class-wc-tpp-settings.php
msgid "License Status"
msgstr "Lizenzstatus"
#: includes/class-wc-tpp-settings.php
msgid "Validate License"
msgstr "Lizenz überprüfen"
#: includes/class-wc-tpp-settings.php
msgid "Activate License"
msgstr "Lizenz aktivieren"
#: includes/class-wc-tpp-settings.php
msgid "No license activated"
msgstr "Keine Lizenz aktiviert"
#: includes/class-wc-tpp-settings.php
msgid "License Active"
msgstr "Lizenz aktiv"
#: includes/class-wc-tpp-settings.php
msgid "License Invalid"
msgstr "Lizenz ungültig"
#: includes/class-wc-tpp-settings.php
msgid "Lifetime License"
msgstr "Lebenslange Lizenz"
#: includes/class-wc-tpp-settings.php
msgid "Expires: %s"
msgstr "Läuft ab: %s"
#: includes/class-wc-tpp-settings.php
msgid "Last checked: %s"
msgstr "Zuletzt geprüft: %s"
#: includes/class-wc-tpp-settings.php
msgid "License is valid!"
msgstr "Lizenz ist gültig!"
#: includes/class-wc-tpp-settings.php
msgid "License activated successfully!"
msgstr "Lizenz erfolgreich aktiviert!"
#: includes/class-wc-tpp-settings.php
msgid "Permission denied."
msgstr "Zugriff verweigert."
#: includes/class-wc-tpp-settings.php
msgid "Server Secret"
msgstr "Server-Geheimnis"
#: includes/class-wc-tpp-settings.php
msgid "The shared secret for secure communication with the license server."
msgstr "Das gemeinsame Geheimnis für die sichere Kommunikation mit dem Lizenzserver."
#: includes/class-wc-tpp-settings.php
msgid "License key, server URL, and server secret are required."
msgstr "Lizenzschlüssel, Server-URL und Server-Geheimnis sind erforderlich."
#: includes/class-wc-tpp-settings.php
#. translators: %d: Number of seconds to wait
msgid "Rate limit exceeded. Please try again in %d seconds."
msgstr "Anfragelimit überschritten. Bitte versuchen Sie es in %d Sekunden erneut."
#: includes/class-wc-tpp-settings.php
msgid "Response signature verification failed. Please check your server secret."
msgstr "Überprüfung der Antwortsignatur fehlgeschlagen. Bitte überprüfen Sie Ihr Server-Geheimnis."
#: includes/class-wc-tpp-settings.php
msgid "An unexpected error occurred. Please try again."
msgstr "Ein unerwarteter Fehler ist aufgetreten. Bitte versuchen Sie es erneut."
#: includes/class-wc-tpp-settings.php
msgid "Please enter license server URL, license key, and server secret."
msgstr "Bitte geben Sie die Lizenzserver-URL, den Lizenzschlüssel und das Server-Geheimnis ein."
#: includes/class-wc-tpp-settings.php
msgid "Validation failed."
msgstr "Überprüfung fehlgeschlagen."
#: includes/class-wc-tpp-settings.php
msgid "Activation failed."
msgstr "Aktivierung fehlgeschlagen."
#: includes/class-wc-tpp-settings.php
msgid "Request failed. Please try again."
msgstr "Anfrage fehlgeschlagen. Bitte versuchen Sie es erneut."

View File

@@ -277,3 +277,124 @@ msgstr "Preis"
#: includes/class-wc-tpp-product-meta.php:165
msgid "Label (optional)"
msgstr "Beschriftung (optional)"
#: wc-tier-and-package-prices.php
msgid "WooCommerce Tier and Package Prices requires PHP 8.3 or higher. Your server is running PHP %s."
msgstr "WooCommerce Staffel- und Paketpreise erfordert PHP 8.3 oder höher. Dein Server verwendet PHP %s."
#: includes/class-wc-tpp-settings.php
msgid "License"
msgstr "Lizenz"
#: includes/class-wc-tpp-settings.php
msgid "License Management"
msgstr "Lizenzverwaltung"
#: includes/class-wc-tpp-settings.php
msgid "Enter your license key to receive updates and support."
msgstr "Gib deinen Lizenzschlüssel ein, um Updates und Support zu erhalten."
#: includes/class-wc-tpp-settings.php
msgid "License Server URL"
msgstr "Lizenzserver-URL"
#: includes/class-wc-tpp-settings.php
msgid "The URL of the license server."
msgstr "Die URL des Lizenzservers."
#: includes/class-wc-tpp-settings.php
msgid "License Key"
msgstr "Lizenzschlüssel"
#: includes/class-wc-tpp-settings.php
msgid "Your license key for this plugin."
msgstr "Dein Lizenzschlüssel für dieses Plugin."
#: includes/class-wc-tpp-settings.php
msgid "License Status"
msgstr "Lizenzstatus"
#: includes/class-wc-tpp-settings.php
msgid "Validate License"
msgstr "Lizenz überprüfen"
#: includes/class-wc-tpp-settings.php
msgid "Activate License"
msgstr "Lizenz aktivieren"
#: includes/class-wc-tpp-settings.php
msgid "No license activated"
msgstr "Keine Lizenz aktiviert"
#: includes/class-wc-tpp-settings.php
msgid "License Active"
msgstr "Lizenz aktiv"
#: includes/class-wc-tpp-settings.php
msgid "License Invalid"
msgstr "Lizenz ungültig"
#: includes/class-wc-tpp-settings.php
msgid "Lifetime License"
msgstr "Lebenslange Lizenz"
#: includes/class-wc-tpp-settings.php
msgid "Expires: %s"
msgstr "Läuft ab: %s"
#: includes/class-wc-tpp-settings.php
msgid "Last checked: %s"
msgstr "Zuletzt geprüft: %s"
#: includes/class-wc-tpp-settings.php
msgid "License is valid!"
msgstr "Lizenz ist gültig!"
#: includes/class-wc-tpp-settings.php
msgid "License activated successfully!"
msgstr "Lizenz erfolgreich aktiviert!"
#: includes/class-wc-tpp-settings.php
msgid "Permission denied."
msgstr "Zugriff verweigert."
#: includes/class-wc-tpp-settings.php
msgid "Server Secret"
msgstr "Server-Geheimnis"
#: includes/class-wc-tpp-settings.php
msgid "The shared secret for secure communication with the license server."
msgstr "Das gemeinsame Geheimnis für die sichere Kommunikation mit dem Lizenzserver."
#: includes/class-wc-tpp-settings.php
msgid "License key, server URL, and server secret are required."
msgstr "Lizenzschlüssel, Server-URL und Server-Geheimnis sind erforderlich."
#: includes/class-wc-tpp-settings.php
#. translators: %d: Number of seconds to wait
msgid "Rate limit exceeded. Please try again in %d seconds."
msgstr "Anfragelimit überschritten. Bitte versuche es in %d Sekunden erneut."
#: includes/class-wc-tpp-settings.php
msgid "Response signature verification failed. Please check your server secret."
msgstr "Überprüfung der Antwortsignatur fehlgeschlagen. Bitte überprüfe dein Server-Geheimnis."
#: includes/class-wc-tpp-settings.php
msgid "An unexpected error occurred. Please try again."
msgstr "Ein unerwarteter Fehler ist aufgetreten. Bitte versuche es erneut."
#: includes/class-wc-tpp-settings.php
msgid "Please enter license server URL, license key, and server secret."
msgstr "Bitte gib die Lizenzserver-URL, den Lizenzschlüssel und das Server-Geheimnis ein."
#: includes/class-wc-tpp-settings.php
msgid "Validation failed."
msgstr "Überprüfung fehlgeschlagen."
#: includes/class-wc-tpp-settings.php
msgid "Activation failed."
msgstr "Aktivierung fehlgeschlagen."
#: includes/class-wc-tpp-settings.php
msgid "Request failed. Please try again."
msgstr "Anfrage fehlgeschlagen. Bitte versuche es erneut."

View File

@@ -277,3 +277,124 @@ msgstr "Preis"
#: includes/class-wc-tpp-product-meta.php:165
msgid "Label (optional)"
msgstr "Beschriftung (optional)"
#: wc-tier-and-package-prices.php
msgid "WooCommerce Tier and Package Prices requires PHP 8.3 or higher. Your server is running PHP %s."
msgstr "WooCommerce Staffel- und Paketpreise erfordert PHP 8.3 oder höher. Ihr Server verwendet PHP %s."
#: includes/class-wc-tpp-settings.php
msgid "License"
msgstr "Lizenz"
#: includes/class-wc-tpp-settings.php
msgid "License Management"
msgstr "Lizenzverwaltung"
#: includes/class-wc-tpp-settings.php
msgid "Enter your license key to receive updates and support."
msgstr "Geben Sie Ihren Lizenzschlüssel ein, um Updates und Support zu erhalten."
#: includes/class-wc-tpp-settings.php
msgid "License Server URL"
msgstr "Lizenzserver-URL"
#: includes/class-wc-tpp-settings.php
msgid "The URL of the license server."
msgstr "Die URL des Lizenzservers."
#: includes/class-wc-tpp-settings.php
msgid "License Key"
msgstr "Lizenzschlüssel"
#: includes/class-wc-tpp-settings.php
msgid "Your license key for this plugin."
msgstr "Ihr Lizenzschlüssel für dieses Plugin."
#: includes/class-wc-tpp-settings.php
msgid "License Status"
msgstr "Lizenzstatus"
#: includes/class-wc-tpp-settings.php
msgid "Validate License"
msgstr "Lizenz überprüfen"
#: includes/class-wc-tpp-settings.php
msgid "Activate License"
msgstr "Lizenz aktivieren"
#: includes/class-wc-tpp-settings.php
msgid "No license activated"
msgstr "Keine Lizenz aktiviert"
#: includes/class-wc-tpp-settings.php
msgid "License Active"
msgstr "Lizenz aktiv"
#: includes/class-wc-tpp-settings.php
msgid "License Invalid"
msgstr "Lizenz ungültig"
#: includes/class-wc-tpp-settings.php
msgid "Lifetime License"
msgstr "Lebenslange Lizenz"
#: includes/class-wc-tpp-settings.php
msgid "Expires: %s"
msgstr "Läuft ab: %s"
#: includes/class-wc-tpp-settings.php
msgid "Last checked: %s"
msgstr "Zuletzt geprüft: %s"
#: includes/class-wc-tpp-settings.php
msgid "License is valid!"
msgstr "Lizenz ist gültig!"
#: includes/class-wc-tpp-settings.php
msgid "License activated successfully!"
msgstr "Lizenz erfolgreich aktiviert!"
#: includes/class-wc-tpp-settings.php
msgid "Permission denied."
msgstr "Zugriff verweigert."
#: includes/class-wc-tpp-settings.php
msgid "Server Secret"
msgstr "Server-Geheimnis"
#: includes/class-wc-tpp-settings.php
msgid "The shared secret for secure communication with the license server."
msgstr "Das gemeinsame Geheimnis für die sichere Kommunikation mit dem Lizenzserver."
#: includes/class-wc-tpp-settings.php
msgid "License key, server URL, and server secret are required."
msgstr "Lizenzschlüssel, Server-URL und Server-Geheimnis sind erforderlich."
#: includes/class-wc-tpp-settings.php
#. translators: %d: Number of seconds to wait
msgid "Rate limit exceeded. Please try again in %d seconds."
msgstr "Anfragelimit überschritten. Bitte versuchen Sie es in %d Sekunden erneut."
#: includes/class-wc-tpp-settings.php
msgid "Response signature verification failed. Please check your server secret."
msgstr "Überprüfung der Antwortsignatur fehlgeschlagen. Bitte überprüfen Sie Ihr Server-Geheimnis."
#: includes/class-wc-tpp-settings.php
msgid "An unexpected error occurred. Please try again."
msgstr "Ein unerwarteter Fehler ist aufgetreten. Bitte versuchen Sie es erneut."
#: includes/class-wc-tpp-settings.php
msgid "Please enter license server URL, license key, and server secret."
msgstr "Bitte geben Sie die Lizenzserver-URL, den Lizenzschlüssel und das Server-Geheimnis ein."
#: includes/class-wc-tpp-settings.php
msgid "Validation failed."
msgstr "Überprüfung fehlgeschlagen."
#: includes/class-wc-tpp-settings.php
msgid "Activation failed."
msgstr "Aktivierung fehlgeschlagen."
#: includes/class-wc-tpp-settings.php
msgid "Request failed. Please try again."
msgstr "Anfrage fehlgeschlagen. Bitte versuchen Sie es erneut."

View File

@@ -277,3 +277,124 @@ msgstr "Preis"
#: includes/class-wc-tpp-product-meta.php:165
msgid "Label (optional)"
msgstr "Beschriftung (optional)"
#: wc-tier-and-package-prices.php
msgid "WooCommerce Tier and Package Prices requires PHP 8.3 or higher. Your server is running PHP %s."
msgstr "WooCommerce Staffel- und Paketpreise erfordert PHP 8.3 oder höher. Dein Server verwendet PHP %s."
#: includes/class-wc-tpp-settings.php
msgid "License"
msgstr "Lizenz"
#: includes/class-wc-tpp-settings.php
msgid "License Management"
msgstr "Lizenzverwaltung"
#: includes/class-wc-tpp-settings.php
msgid "Enter your license key to receive updates and support."
msgstr "Gib deinen Lizenzschlüssel ein, um Updates und Support zu erhalten."
#: includes/class-wc-tpp-settings.php
msgid "License Server URL"
msgstr "Lizenzserver-URL"
#: includes/class-wc-tpp-settings.php
msgid "The URL of the license server."
msgstr "Die URL des Lizenzservers."
#: includes/class-wc-tpp-settings.php
msgid "License Key"
msgstr "Lizenzschlüssel"
#: includes/class-wc-tpp-settings.php
msgid "Your license key for this plugin."
msgstr "Dein Lizenzschlüssel für dieses Plugin."
#: includes/class-wc-tpp-settings.php
msgid "License Status"
msgstr "Lizenzstatus"
#: includes/class-wc-tpp-settings.php
msgid "Validate License"
msgstr "Lizenz überprüfen"
#: includes/class-wc-tpp-settings.php
msgid "Activate License"
msgstr "Lizenz aktivieren"
#: includes/class-wc-tpp-settings.php
msgid "No license activated"
msgstr "Keine Lizenz aktiviert"
#: includes/class-wc-tpp-settings.php
msgid "License Active"
msgstr "Lizenz aktiv"
#: includes/class-wc-tpp-settings.php
msgid "License Invalid"
msgstr "Lizenz ungültig"
#: includes/class-wc-tpp-settings.php
msgid "Lifetime License"
msgstr "Lebenslange Lizenz"
#: includes/class-wc-tpp-settings.php
msgid "Expires: %s"
msgstr "Läuft ab: %s"
#: includes/class-wc-tpp-settings.php
msgid "Last checked: %s"
msgstr "Zuletzt geprüft: %s"
#: includes/class-wc-tpp-settings.php
msgid "License is valid!"
msgstr "Lizenz ist gültig!"
#: includes/class-wc-tpp-settings.php
msgid "License activated successfully!"
msgstr "Lizenz erfolgreich aktiviert!"
#: includes/class-wc-tpp-settings.php
msgid "Permission denied."
msgstr "Zugriff verweigert."
#: includes/class-wc-tpp-settings.php
msgid "Server Secret"
msgstr "Server-Geheimnis"
#: includes/class-wc-tpp-settings.php
msgid "The shared secret for secure communication with the license server."
msgstr "Das gemeinsame Geheimnis für die sichere Kommunikation mit dem Lizenzserver."
#: includes/class-wc-tpp-settings.php
msgid "License key, server URL, and server secret are required."
msgstr "Lizenzschlüssel, Server-URL und Server-Geheimnis sind erforderlich."
#: includes/class-wc-tpp-settings.php
#. translators: %d: Number of seconds to wait
msgid "Rate limit exceeded. Please try again in %d seconds."
msgstr "Anfragelimit überschritten. Bitte versuche es in %d Sekunden erneut."
#: includes/class-wc-tpp-settings.php
msgid "Response signature verification failed. Please check your server secret."
msgstr "Überprüfung der Antwortsignatur fehlgeschlagen. Bitte überprüfe dein Server-Geheimnis."
#: includes/class-wc-tpp-settings.php
msgid "An unexpected error occurred. Please try again."
msgstr "Ein unerwarteter Fehler ist aufgetreten. Bitte versuche es erneut."
#: includes/class-wc-tpp-settings.php
msgid "Please enter license server URL, license key, and server secret."
msgstr "Bitte gib die Lizenzserver-URL, den Lizenzschlüssel und das Server-Geheimnis ein."
#: includes/class-wc-tpp-settings.php
msgid "Validation failed."
msgstr "Überprüfung fehlgeschlagen."
#: includes/class-wc-tpp-settings.php
msgid "Activation failed."
msgstr "Aktivierung fehlgeschlagen."
#: includes/class-wc-tpp-settings.php
msgid "Request failed. Please try again."
msgstr "Anfrage fehlgeschlagen. Bitte versuche es erneut."

View File

@@ -277,3 +277,124 @@ msgstr "Price"
#: includes/class-wc-tpp-product-meta.php:165
msgid "Label (optional)"
msgstr "Label (optional)"
#: wc-tier-and-package-prices.php
msgid "WooCommerce Tier and Package Prices requires PHP 8.3 or higher. Your server is running PHP %s."
msgstr "WooCommerce Tier and Package Prices requires PHP 8.3 or higher. Your server is running PHP %s."
#: includes/class-wc-tpp-settings.php
msgid "License"
msgstr "License"
#: includes/class-wc-tpp-settings.php
msgid "License Management"
msgstr "License Management"
#: includes/class-wc-tpp-settings.php
msgid "Enter your license key to receive updates and support."
msgstr "Enter your license key to receive updates and support."
#: includes/class-wc-tpp-settings.php
msgid "License Server URL"
msgstr "License Server URL"
#: includes/class-wc-tpp-settings.php
msgid "The URL of the license server."
msgstr "The URL of the license server."
#: includes/class-wc-tpp-settings.php
msgid "License Key"
msgstr "License Key"
#: includes/class-wc-tpp-settings.php
msgid "Your license key for this plugin."
msgstr "Your license key for this plugin."
#: includes/class-wc-tpp-settings.php
msgid "License Status"
msgstr "License Status"
#: includes/class-wc-tpp-settings.php
msgid "Validate License"
msgstr "Validate License"
#: includes/class-wc-tpp-settings.php
msgid "Activate License"
msgstr "Activate License"
#: includes/class-wc-tpp-settings.php
msgid "No license activated"
msgstr "No license activated"
#: includes/class-wc-tpp-settings.php
msgid "License Active"
msgstr "License Active"
#: includes/class-wc-tpp-settings.php
msgid "License Invalid"
msgstr "License Invalid"
#: includes/class-wc-tpp-settings.php
msgid "Lifetime License"
msgstr "Lifetime License"
#: includes/class-wc-tpp-settings.php
msgid "Expires: %s"
msgstr "Expires: %s"
#: includes/class-wc-tpp-settings.php
msgid "Last checked: %s"
msgstr "Last checked: %s"
#: includes/class-wc-tpp-settings.php
msgid "License is valid!"
msgstr "License is valid!"
#: includes/class-wc-tpp-settings.php
msgid "License activated successfully!"
msgstr "License activated successfully!"
#: includes/class-wc-tpp-settings.php
msgid "Permission denied."
msgstr "Permission denied."
#: includes/class-wc-tpp-settings.php
msgid "Server Secret"
msgstr "Server Secret"
#: includes/class-wc-tpp-settings.php
msgid "The shared secret for secure communication with the license server."
msgstr "The shared secret for secure communication with the license server."
#: includes/class-wc-tpp-settings.php
msgid "License key, server URL, and server secret are required."
msgstr "License key, server URL, and server secret are required."
#: includes/class-wc-tpp-settings.php
#. translators: %d: Number of seconds to wait
msgid "Rate limit exceeded. Please try again in %d seconds."
msgstr "Rate limit exceeded. Please try again in %d seconds."
#: includes/class-wc-tpp-settings.php
msgid "Response signature verification failed. Please check your server secret."
msgstr "Response signature verification failed. Please check your server secret."
#: includes/class-wc-tpp-settings.php
msgid "An unexpected error occurred. Please try again."
msgstr "An unexpected error occurred. Please try again."
#: includes/class-wc-tpp-settings.php
msgid "Please enter license server URL, license key, and server secret."
msgstr "Please enter license server URL, license key, and server secret."
#: includes/class-wc-tpp-settings.php
msgid "Validation failed."
msgstr "Validation failed."
#: includes/class-wc-tpp-settings.php
msgid "Activation failed."
msgstr "Activation failed."
#: includes/class-wc-tpp-settings.php
msgid "Request failed. Please try again."
msgstr "Request failed. Please try again."

View File

@@ -277,3 +277,124 @@ msgstr "Prix"
#: includes/class-wc-tpp-product-meta.php:165
msgid "Label (optional)"
msgstr "Étiquette (optionnel)"
#: wc-tier-and-package-prices.php
msgid "WooCommerce Tier and Package Prices requires PHP 8.3 or higher. Your server is running PHP %s."
msgstr "WooCommerce Prix par paliers et forfaits nécessite PHP 8.3 ou supérieur. Votre serveur utilise PHP %s."
#: includes/class-wc-tpp-settings.php
msgid "License"
msgstr "Licence"
#: includes/class-wc-tpp-settings.php
msgid "License Management"
msgstr "Gestion des licences"
#: includes/class-wc-tpp-settings.php
msgid "Enter your license key to receive updates and support."
msgstr "Entrez votre clé de licence pour recevoir des mises à jour et du support."
#: includes/class-wc-tpp-settings.php
msgid "License Server URL"
msgstr "URL du serveur de licence"
#: includes/class-wc-tpp-settings.php
msgid "The URL of the license server."
msgstr "L'URL du serveur de licence."
#: includes/class-wc-tpp-settings.php
msgid "License Key"
msgstr "Clé de licence"
#: includes/class-wc-tpp-settings.php
msgid "Your license key for this plugin."
msgstr "Votre clé de licence pour cette extension."
#: includes/class-wc-tpp-settings.php
msgid "License Status"
msgstr "Statut de la licence"
#: includes/class-wc-tpp-settings.php
msgid "Validate License"
msgstr "Valider la licence"
#: includes/class-wc-tpp-settings.php
msgid "Activate License"
msgstr "Activer la licence"
#: includes/class-wc-tpp-settings.php
msgid "No license activated"
msgstr "Aucune licence activée"
#: includes/class-wc-tpp-settings.php
msgid "License Active"
msgstr "Licence active"
#: includes/class-wc-tpp-settings.php
msgid "License Invalid"
msgstr "Licence invalide"
#: includes/class-wc-tpp-settings.php
msgid "Lifetime License"
msgstr "Licence à vie"
#: includes/class-wc-tpp-settings.php
msgid "Expires: %s"
msgstr "Expire le: %s"
#: includes/class-wc-tpp-settings.php
msgid "Last checked: %s"
msgstr "Dernière vérification: %s"
#: includes/class-wc-tpp-settings.php
msgid "License is valid!"
msgstr "La licence est valide!"
#: includes/class-wc-tpp-settings.php
msgid "License activated successfully!"
msgstr "Licence activée avec succès!"
#: includes/class-wc-tpp-settings.php
msgid "Permission denied."
msgstr "Accès refusé."
#: includes/class-wc-tpp-settings.php
msgid "Server Secret"
msgstr "Secret serveur"
#: includes/class-wc-tpp-settings.php
msgid "The shared secret for secure communication with the license server."
msgstr "Le secret partagé pour la communication sécurisée avec le serveur de licence."
#: includes/class-wc-tpp-settings.php
msgid "License key, server URL, and server secret are required."
msgstr "La clé de licence, l'URL du serveur et le secret serveur sont requis."
#: includes/class-wc-tpp-settings.php
#. translators: %d: Number of seconds to wait
msgid "Rate limit exceeded. Please try again in %d seconds."
msgstr "Limite de requêtes dépassée. Veuillez réessayer dans %d secondes."
#: includes/class-wc-tpp-settings.php
msgid "Response signature verification failed. Please check your server secret."
msgstr "La vérification de la signature a échoué. Veuillez vérifier votre secret serveur."
#: includes/class-wc-tpp-settings.php
msgid "An unexpected error occurred. Please try again."
msgstr "Une erreur inattendue s'est produite. Veuillez réessayer."
#: includes/class-wc-tpp-settings.php
msgid "Please enter license server URL, license key, and server secret."
msgstr "Veuillez entrer l'URL du serveur de licence, la clé de licence et le secret serveur."
#: includes/class-wc-tpp-settings.php
msgid "Validation failed."
msgstr "La validation a échoué."
#: includes/class-wc-tpp-settings.php
msgid "Activation failed."
msgstr "L'activation a échoué."
#: includes/class-wc-tpp-settings.php
msgid "Request failed. Please try again."
msgstr "La requête a échoué. Veuillez réessayer."

View File

@@ -277,3 +277,124 @@ msgstr "Prezzo"
#: includes/class-wc-tpp-product-meta.php:165
msgid "Label (optional)"
msgstr "Etichetta (facoltativo)"
#: wc-tier-and-package-prices.php
msgid "WooCommerce Tier and Package Prices requires PHP 8.3 or higher. Your server is running PHP %s."
msgstr "WooCommerce Prezzi a scaglioni e pacchetti richiede PHP 8.3 o superiore. Il tuo server utilizza PHP %s."
#: includes/class-wc-tpp-settings.php
msgid "License"
msgstr "Licenza"
#: includes/class-wc-tpp-settings.php
msgid "License Management"
msgstr "Gestione licenza"
#: includes/class-wc-tpp-settings.php
msgid "Enter your license key to receive updates and support."
msgstr "Inserisci la tua chiave di licenza per ricevere aggiornamenti e supporto."
#: includes/class-wc-tpp-settings.php
msgid "License Server URL"
msgstr "URL server licenza"
#: includes/class-wc-tpp-settings.php
msgid "The URL of the license server."
msgstr "L'URL del server di licenza."
#: includes/class-wc-tpp-settings.php
msgid "License Key"
msgstr "Chiave di licenza"
#: includes/class-wc-tpp-settings.php
msgid "Your license key for this plugin."
msgstr "La tua chiave di licenza per questo plugin."
#: includes/class-wc-tpp-settings.php
msgid "License Status"
msgstr "Stato licenza"
#: includes/class-wc-tpp-settings.php
msgid "Validate License"
msgstr "Verifica licenza"
#: includes/class-wc-tpp-settings.php
msgid "Activate License"
msgstr "Attiva licenza"
#: includes/class-wc-tpp-settings.php
msgid "No license activated"
msgstr "Nessuna licenza attivata"
#: includes/class-wc-tpp-settings.php
msgid "License Active"
msgstr "Licenza attiva"
#: includes/class-wc-tpp-settings.php
msgid "License Invalid"
msgstr "Licenza non valida"
#: includes/class-wc-tpp-settings.php
msgid "Lifetime License"
msgstr "Licenza a vita"
#: includes/class-wc-tpp-settings.php
msgid "Expires: %s"
msgstr "Scade il: %s"
#: includes/class-wc-tpp-settings.php
msgid "Last checked: %s"
msgstr "Ultima verifica: %s"
#: includes/class-wc-tpp-settings.php
msgid "License is valid!"
msgstr "La licenza è valida!"
#: includes/class-wc-tpp-settings.php
msgid "License activated successfully!"
msgstr "Licenza attivata con successo!"
#: includes/class-wc-tpp-settings.php
msgid "Permission denied."
msgstr "Accesso negato."
#: includes/class-wc-tpp-settings.php
msgid "Server Secret"
msgstr "Segreto del server"
#: includes/class-wc-tpp-settings.php
msgid "The shared secret for secure communication with the license server."
msgstr "Il segreto condiviso per la comunicazione sicura con il server di licenza."
#: includes/class-wc-tpp-settings.php
msgid "License key, server URL, and server secret are required."
msgstr "La chiave di licenza, l'URL del server e il segreto del server sono obbligatori."
#: includes/class-wc-tpp-settings.php
#. translators: %d: Number of seconds to wait
msgid "Rate limit exceeded. Please try again in %d seconds."
msgstr "Limite di richieste superato. Per favore riprova tra %d secondi."
#: includes/class-wc-tpp-settings.php
msgid "Response signature verification failed. Please check your server secret."
msgstr "Verifica della firma della risposta fallita. Per favore controlla il tuo segreto del server."
#: includes/class-wc-tpp-settings.php
msgid "An unexpected error occurred. Please try again."
msgstr "Si è verificato un errore imprevisto. Per favore riprova."
#: includes/class-wc-tpp-settings.php
msgid "Please enter license server URL, license key, and server secret."
msgstr "Inserisci l'URL del server di licenza, la chiave di licenza e il segreto del server."
#: includes/class-wc-tpp-settings.php
msgid "Validation failed."
msgstr "Verifica fallita."
#: includes/class-wc-tpp-settings.php
msgid "Activation failed."
msgstr "Attivazione fallita."
#: includes/class-wc-tpp-settings.php
msgid "Request failed. Please try again."
msgstr "Richiesta fallita. Per favore riprova."

View File

@@ -2,9 +2,9 @@
# This file is distributed under the GPL v2 or later.
msgid ""
msgstr ""
"Project-Id-Version: WooCommerce Tier and Package Prices 1.2.7\n"
"Project-Id-Version: WooCommerce Tier and Package Prices 1.3.1\n"
"Report-Msgid-Bugs-To: https://src.bundespruefstelle.ch/wc-tier-package-prices\n"
"POT-Creation-Date: 2025-12-30 00:00+0000\n"
"POT-Creation-Date: 2026-01-27 00:00+0000\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
@@ -250,3 +250,124 @@ msgstr ""
#: includes/class-wc-tpp-frontend.php:178
msgid "View options for %s"
msgstr ""
#: wc-tier-and-package-prices.php
msgid "WooCommerce Tier and Package Prices requires PHP 8.3 or higher. Your server is running PHP %s."
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "License"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "License Management"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Enter your license key to receive updates and support."
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "License Server URL"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "The URL of the license server."
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "License Key"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Your license key for this plugin."
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "License Status"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Validate License"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Activate License"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "No license activated"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "License Active"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "License Invalid"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Lifetime License"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Expires: %s"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Last checked: %s"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "License is valid!"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "License activated successfully!"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Permission denied."
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Server Secret"
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "The shared secret for secure communication with the license server."
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "License key, server URL, and server secret are required."
msgstr ""
#: includes/class-wc-tpp-settings.php
#. translators: %d: Number of seconds to wait
msgid "Rate limit exceeded. Please try again in %d seconds."
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Response signature verification failed. Please check your server secret."
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "An unexpected error occurred. Please try again."
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Please enter license server URL, license key, and server secret."
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Validation failed."
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Activation failed."
msgstr ""
#: includes/class-wc-tpp-settings.php
msgid "Request failed. Please try again."
msgstr ""

View File

@@ -1,219 +0,0 @@
# WooCommerce Tier and Package Prices - Releases
This directory contains production-ready releases of the WooCommerce Tier and Package Prices plugin.
## Latest Release
**Version 1.1.20** - December 23, 2025
### What's New in 1.1.20
- 🔧 **CRITICAL FIX:** Resolved WooCommerce Blocks fatal error in mini-cart and cart blocks
- ✅ Full WooCommerce Blocks compatibility (cart, mini-cart, checkout)
- ✅ Fixed `woocommerce_store_api_product_quantity_editable` filter signature
- ✅ Tier labels for enhanced UX (v1.1.7)
- ✅ Package quantity restrictions (v1.1.0)
- ✅ Clickable tier rows with auto-fill
- ✅ Enhanced "View Options" buttons
### Quick Install
```bash
# Download the package
wget https://your-domain.com/releases/wc-tier-and-package-prices-1.1.20.zip
# Verify checksum (optional but recommended)
sha256sum wc-tier-and-package-prices-1.1.20.zip
# Install via WordPress admin or WP-CLI
wp plugin install wc-tier-and-package-prices-1.1.20.zip --activate
```
## Files in This Directory
### Release Packages
| File | Description | Size |
|------|-------------|------|
| `wc-tier-and-package-prices-1.1.20.zip` | Production plugin package | ~400 KB |
| `wc-tier-and-package-prices-1.1.20.zip.sha256` | SHA-256 checksum | - |
| `wc-tier-and-package-prices-1.1.20.zip.md5` | MD5 checksum | - |
| `RELEASE-INFO-1.1.20.md` | Detailed release information | - |
## Verification
### Verify Package Integrity
**Using SHA-256:**
```bash
sha256sum -c wc-tier-and-package-prices-1.1.20.zip.sha256
```
**Using MD5:**
```bash
md5sum -c wc-tier-and-package-prices-1.1.20.zip.md5
```
### Expected Checksums
Checksums will be generated when the release package is created.
## Installation Methods
### Method 1: WordPress Admin (Recommended for most users)
1. Download `wc-tier-and-package-prices-1.1.20.zip`
2. Go to **WordPress Admin > Plugins > Add New**
3. Click **Upload Plugin**
4. Choose the downloaded ZIP file
5. Click **Install Now**
6. Click **Activate Plugin**
### Method 2: WP-CLI (For developers)
```bash
wp plugin install /path/to/wc-tier-and-package-prices-1.1.20.zip --activate
```
### Method 3: Manual Installation (Advanced)
```bash
# Extract to wp-content/plugins/
unzip wc-tier-and-package-prices-1.1.20.zip -d /path/to/wordpress/wp-content/plugins/
# Set correct permissions
chmod -R 755 /path/to/wordpress/wp-content/plugins/wc-tier-and-package-prices
# Activate via WordPress admin or WP-CLI
wp plugin activate wc-tier-and-package-prices
```
## What's Included
### Core Features
- ✅ Tier pricing with optional labels (quantity-based discounts)
- ✅ Package pricing with quantity restrictions (fixed-price bundles)
- ✅ Clickable tier rows with auto-quantity fill
- ✅ WooCommerce Blocks full support (cart, mini-cart, checkout)
- ✅ Twig template engine for secure templating
- ✅ WooCommerce HPOS (High-Performance Order Storage) compatible
- ✅ Quantity restriction enforcement
- ✅ "View Options" catalog buttons for restricted products
- ✅ Multilingual support (3 languages)
### Translations
- 🇺🇸 English (US)
- 🇩🇪 German (Germany)
- 🇨🇭 German (Switzerland, Informal)
### Production Ready
- ✅ Optimized autoloader
- ✅ No development dependencies
- ✅ Compiled Twig templates support
- ✅ Tested with WooCommerce 8.0 - 10.x
- ✅ Tested with WordPress 6.0 - 6.9.x
- ✅ PHP 7.4+ compatible
- ✅ Block-based themes compatible
## Package Contents
```
wc-tier-and-package-prices/
├── assets/ # CSS and JavaScript
│ ├── css/
│ │ ├── admin.css
│ │ └── frontend.css
│ └── js/
│ ├── admin.js
│ └── frontend.js
├── includes/ # PHP classes
│ ├── class-wc-tpp-admin.php
│ ├── class-wc-tpp-settings.php
│ ├── class-wc-tpp-cart.php
│ ├── class-wc-tpp-frontend.php
│ ├── class-wc-tpp-product-meta.php
│ └── class-wc-tpp-template-loader.php
├── languages/ # Translation files
│ ├── wc-tier-package-prices-de_CH_informal.po
│ ├── wc-tier-package-prices-de_CH_informal.mo
│ ├── wc-tier-package-prices-de_DE.po
│ ├── wc-tier-package-prices-de_DE.mo
│ ├── wc-tier-package-prices-en_US.po
│ ├── wc-tier-package-prices-en_US.mo
│ └── wc-tier-package-prices.pot
├── templates/ # Twig templates
│ ├── admin/
│ │ ├── tier-row.twig
│ │ └── package-row.twig
│ └── frontend/
│ ├── pricing-table.twig
│ ├── tier-pricing-table.twig
│ └── package-pricing-display.twig
├── vendor/ # Composer dependencies
│ └── twig/twig/
├── CHANGELOG.md
├── INSTALLATION.md
├── QUICKSTART.md
├── USAGE_EXAMPLES.md
├── README.md
├── composer.json
└── wc-tier-and-package-prices.php
```
## System Requirements
| Requirement | Minimum Version | Tested Up To |
|-------------|----------------|--------------|
| WordPress | 6.0+ | 6.9.x |
| PHP | 7.4+ | 8.x |
| WooCommerce | 8.0+ | 10.x |
| MySQL | 5.6+ | 8.x |
## Support
- **Documentation:** See README.md, INSTALLATION.md, QUICKSTART.md, USAGE_EXAMPLES.md
- **Repository:** https://src.bundespruefstelle.ch/magdev/wc-tier-package-prices
- **Author:** Marco Graetsch
## Version History
### 1.1.20 (2025-12-23) - Current Release
- **CRITICAL FIX:** WooCommerce Blocks fatal error resolved
- Fixed filter signature for `woocommerce_store_api_product_quantity_editable`
- Full compatibility with WooCommerce Store API and block-based cart/checkout
- Enhanced stability for block-based themes
### 1.1.7 (2025-12-22)
- Added optional tier labels for enhanced UX
- Clickable tier rows with auto-quantity fill
- Add to Cart button auto-disable for invalid quantities
### 1.1.4 (2025-12-22)
- WooCommerce Blocks support (cart, mini-cart, checkout)
- Enhanced "View Options" button styling
### 1.1.0 (2025-12-21)
- Package quantity restrictions (global and per-product)
- Quantity field hiding for restricted products
- Server-side validation for package quantities
### 1.0.1 (2025-12-21)
- Added Twig template engine
- Added Swiss German translation
- Improved template organization
- Enhanced security with auto-escaping
### 1.0.0 (2025-12-21)
- Initial release
- Tier pricing functionality
- Package pricing functionality
- German and English translations
For complete version history, see [CHANGELOG.md](../CHANGELOG.md)
## License
GPL v2 or later - https://www.gnu.org/licenses/gpl-2.0.html
---
**Note:** All packages are production-ready with optimized autoloaders and no development dependencies included.

View File

@@ -1 +0,0 @@
e6cfc9b88df9e7763be0cd56517ce8ab wc-tier-and-package-prices-1.0.1.zip

View File

@@ -1 +0,0 @@
92c1385d92527e77646e37f23c1bd1555a4290a5ec9314c0ee6ed896ded55e88 wc-tier-and-package-prices-1.0.1.zip

View File

@@ -1 +0,0 @@
830f443ce4b65e2ca9cfede3257bc4f5 wc-tier-and-package-prices-1.0.2.zip

View File

@@ -1 +0,0 @@
c1a5339da10b3625156b8fff4ec848e4a1318d6edc497bd5026cfe0a3ef39daa wc-tier-and-package-prices-1.0.2.zip

View File

@@ -1 +0,0 @@
ef68125c54b0c10f04ba82d48a98b4aa wc-tier-and-package-prices-1.1.0.zip

View File

@@ -1 +0,0 @@
da6b462f3dc297b282ed0da258b78fd9f2f82f3e76289c4c8fadd1ac9e02c55b wc-tier-and-package-prices-1.1.0.zip

View File

@@ -1 +0,0 @@
51c4f8a7c3ccede2d2005f2fe3ebe44e wc-tier-and-package-prices-1.1.1.zip

View File

@@ -1 +0,0 @@
b951f8b7ddd2bad6b3415d4583709fdf88f66aea4eae70110c903757ff53e045 wc-tier-and-package-prices-1.1.1.zip

View File

@@ -1 +0,0 @@
81be5283219cfa722f6d382a788e7dc1 releases/wc-tier-and-package-prices-1.1.10.zip

View File

@@ -1 +0,0 @@
2d3b01e61c8a03a8f20bc99b2019ca50fa08ecd68188feb2d2105dfe35d36f0d releases/wc-tier-and-package-prices-1.1.10.zip

View File

@@ -1 +0,0 @@
4a0c0b07b29d4b7046f9d3ff3f091321 releases/wc-tier-and-package-prices-1.1.11.zip

View File

@@ -1 +0,0 @@
3da9423d136a2ff254b61577ba1f84d4c0f0d1e57bae361ac29c90327feeeceb releases/wc-tier-and-package-prices-1.1.11.zip

View File

@@ -1 +0,0 @@
c1c0b5880636686227246be2c37dc42a releases/wc-tier-and-package-prices-1.1.12.zip

View File

@@ -1 +0,0 @@
05b32356d46803dbb7fa17c13a2d8da96f77126746e2895e2f5c6dd0e7b490ff releases/wc-tier-and-package-prices-1.1.12.zip

View File

@@ -1 +0,0 @@
8572eed399554905fbf331d18f0677a0 wc-tier-and-package-prices-1.1.13.zip

View File

@@ -1 +0,0 @@
83e29b2e40dd43e77bd83cd03d4ccc54ef53555b55544eba4d38161101f79f20 wc-tier-and-package-prices-1.1.13.zip

View File

@@ -1 +0,0 @@
e0cc51d1493ed35ab254220d9f46997b wc-tier-and-package-prices-1.1.14.zip

View File

@@ -1 +0,0 @@
8a2ce7438ee49baffdcaaf323b6426d73dd1cf704bea94a80fcce27a42c097ad wc-tier-and-package-prices-1.1.14.zip

View File

@@ -1 +0,0 @@
15fa0e0933c85b23f66940bf43810835 wc-tier-and-package-prices-1.1.15.zip

View File

@@ -1 +0,0 @@
a419579111ad20b127411e1078ca99187156d606381549e6bf147ffc3bd58de1 wc-tier-and-package-prices-1.1.15.zip

View File

@@ -1 +0,0 @@
d30a90715dbcd46a1dfd19f025897530 wc-tier-and-package-prices-1.1.16.zip

View File

@@ -1 +0,0 @@
e2ad36e049a902b8e287154867ef72c0e169766508781e223176a2a753b60915 wc-tier-and-package-prices-1.1.16.zip

View File

@@ -1 +0,0 @@
9adbb9aad13b8d141cfabfdf53643480 wc-tier-and-package-prices-1.1.17.zip

View File

@@ -1 +0,0 @@
985a195bf98d4dbc0a7afa90173efcda472f4c769adf3c833fa6a99ba9d44095 wc-tier-and-package-prices-1.1.17.zip

View File

@@ -1 +0,0 @@
ec4bb1d78a3c27488244b44971916ffd wc-tier-and-package-prices-1.1.18.zip

View File

@@ -1 +0,0 @@
7d942002edd866c2b6f3192ba010fe64058b7433c5ac776a48e9c3c41f4e2fda wc-tier-and-package-prices-1.1.18.zip

View File

@@ -1 +0,0 @@
c61c3a059429d8dacdce71d4acce401e wc-tier-and-package-prices-1.1.19.zip

View File

@@ -1 +0,0 @@
0e8bc4ccd233d388238e800cd0e0a129f9e8da14008e7164db7934a48ca8223a wc-tier-and-package-prices-1.1.19.zip

View File

@@ -1 +0,0 @@
eee69fcf391b3f3df9380306ffb31b1b wc-tier-and-package-prices-1.1.2.zip

View File

@@ -1 +0,0 @@
40ffd29ebc6af635f689472040acd220ae1c8df2f0d852fab4b43ce0fb5fe739 wc-tier-and-package-prices-1.1.2.zip

View File

@@ -1 +0,0 @@
bfdeee75bfe3795c9ab9abfe47f12a41 wc-tier-and-package-prices-1.1.20.zip

View File

@@ -1 +0,0 @@
953859241d15d76ec4783c72bac851ddd69e5a1f7b119ee4f9ebd30c7fabed17 wc-tier-and-package-prices-1.1.20.zip

View File

@@ -1 +0,0 @@
16813b3ed0d1001d5f60194d61d36fc2 wc-tier-and-package-prices-1.1.21.zip

View File

@@ -1 +0,0 @@
e0063852a9ac23b1fd994471a2829f9dcbe26316f00ddee2d00f77c7c6a47c8f wc-tier-and-package-prices-1.1.21.zip

View File

@@ -1 +0,0 @@
7d5a5c7980a91dff5167c90a6f3290b0 wc-tier-and-package-prices-1.1.22.zip

View File

@@ -1 +0,0 @@
f94dee838a3f288b4acb3b3d9a4e88ef987f9b1bc918403186014d8d43fee6d9 wc-tier-and-package-prices-1.1.22.zip

View File

@@ -1 +0,0 @@
dfec91be7e375b09613ba81cfebbe013 wc-tier-and-package-prices-1.1.3.zip

View File

@@ -1 +0,0 @@
7938542680b71a7b73269c96a4dff78f2222ac8409092011c5e40e97a5e465aa wc-tier-and-package-prices-1.1.3.zip

View File

@@ -1 +0,0 @@
3e5bc2cae17ecb81b729c3fdc979df23 wc-tier-and-package-prices-1.1.4.zip

View File

@@ -1 +0,0 @@
19553b2fed1c6ca20a8168eab8c570cb0302be801322cd41d86cec40b70ff162 wc-tier-and-package-prices-1.1.4.zip

View File

@@ -1 +0,0 @@
e9f8a69e4be107d857d3beb671d5a9fe wc-tier-and-package-prices-1.1.5.zip

View File

@@ -1 +0,0 @@
a13d71f3f65c7cf41613f88d7bcfcb112acfefb800fa6b95932f44a47cf764f3 wc-tier-and-package-prices-1.1.5.zip

View File

@@ -1 +0,0 @@
dbea10acffdc849f9aa387d128cb6d6e wc-tier-and-package-prices-1.1.6.zip

View File

@@ -1 +0,0 @@
730e366764449ac963bc85848ac8a91f654e4b35500ed3132a280ab4f215c80c wc-tier-and-package-prices-1.1.6.zip

View File

@@ -1 +0,0 @@
c593b6fc730133ecaf772f77d5659080 wc-tier-and-package-prices-1.1.7.zip

View File

@@ -1 +0,0 @@
00e7c473a8b0fc23eb1cd52f4b38db030618ed142a9a9fdfd9b1aa4c5849ec70 wc-tier-and-package-prices-1.1.7.zip

View File

@@ -1 +0,0 @@
666e1acad9f349982463b65d0e3e7fa3 wc-tier-and-package-prices-1.1.8.zip

View File

@@ -1 +0,0 @@
2c100d0d100a6fd7bd8f9c9b154d878a675bc795c5fddb19314bc9991b9a60ba wc-tier-and-package-prices-1.1.8.zip

View File

@@ -1 +0,0 @@
7421aceb8d1cc89b7d15b19d68cdfabe wc-tier-and-package-prices-1.1.9.zip

View File

@@ -1 +0,0 @@
8224bf8b9bfc3dc760d77c61700d27c31db1a67b70834d8b6a4581df66fd45bd wc-tier-and-package-prices-1.1.9.zip

View File

@@ -1 +0,0 @@
cee7ab535938b4096f225f0e0640c9b7 wc-tier-and-package-prices-1.2.0.zip

Some files were not shown because too many files have changed in this diff Show More