38 Commits

Author SHA1 Message Date
6507f4d8bb v1.2.0 - Fix product selection, cart pricing, admin tabs + CI/CD
Fix three critical bugs that persisted through v1.1.11-v1.1.14:

- Product selection always empty: meta_query checked _product_type in
  postmeta, but WooCommerce uses the product_type taxonomy. Replaced
  with correct tax_query using NOT IN operator.
- Cart price always 0.00: composable_price_calculated flag persisted
  in session, preventing recalculation on page loads. Removed flag;
  static variable already handles per-request dedup.
- Admin tabs both visible on load: JS now triggers WooCommerce native
  tab click instead of manually toggling panel visibility.

Add Gitea CI/CD release workflow triggered on v* tags.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-01 12:01:20 +01:00
29a68b0be4 Add session learnings and context notes to v1.1.14 history
Key learnings documented:
- Diagnostic-first approach after multiple failed fixes
- User has no live access currently (testing delayed)
- Release workflow now well-established
- Shell context handling with absolute paths
- Debug logging best practices
- Context preservation from session summary

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2026-01-01 02:29:46 +01:00
fb8ddf903e Document v1.1.14 debug logging release in session history
Added comprehensive session notes for v1.1.14 including:
- Debug logging implementation strategy
- All 7 logging points in product retrieval
- How to use the debug release
- Expected log output and interpretation
- Next steps after receiving user logs

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2026-01-01 02:28:45 +01:00
33d2836de0 Add release package v1.1.14 with checksums
Debug release to diagnose product retrieval issues.

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2026-01-01 02:11:31 +01:00
c036a37602 Bump version to 1.1.14 - debug logging release
🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2026-01-01 02:06:42 +01:00
efedd1bf29 Add debug logging to product retrieval for troubleshooting
Added error_log statements to help diagnose why products aren't showing:
- Log selection criteria being used
- Log WP_Query arguments
- Log number of posts found by query
- Log each product being added (variable variations and simple)
- Log total products available at end

To enable: Set WP_DEBUG to true in wp-config.php
Then check debug.log or error.log for output

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2026-01-01 02:05:00 +01:00
12388af5a0 Document Session 13: v1.1.11-v1.1.13 variable product support journey
Added comprehensive session history covering:
- v1.1.11: Initial variable product support attempt
- v1.1.12: Fixed variation retrieval method + packaging
- v1.1.13: Removed overly strict stock filtering

Key learnings documented:
- Use get_children() not get_available_variations()
- Don't filter by is_in_stock() during retrieval
- Stock management is multi-layered (retrieval, display, validation)
- WooCommerce stock states (purchasable vs in_stock)
- Debug by elimination approach

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2026-01-01 02:02:56 +01:00
c39c13ffed more info about the admin rendering bug, style corrections 2025-12-31 23:33:19 +01:00
7931dbeef9 updated author uri and email address 2025-12-31 23:32:21 +01:00
ee81de86c2 Add release package v1.1.13 with checksums
🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 23:09:33 +01:00
669888817b Bump version to 1.1.13 - fix product retrieval issue
🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 23:08:53 +01:00
5564b888fc Fix product retrieval - remove strict stock check and add meta_query relation
- Removed is_in_stock() requirement to show all purchasable products
- Stock status still displayed on frontend, out-of-stock items disabled
- Added 'relation' => 'AND' to meta_query for proper multiple condition handling
- Should fix "No products available" issue

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 23:07:06 +01:00
91aca25169 Add release package v1.1.12 with checksums
🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 23:03:57 +01:00
4195fb2651 Bump version to 1.1.12 for variable product fix release
🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:59:46 +01:00
ba28ae174f Fix variable product variations retrieval
Changed from get_available_variations() to get_children() for more reliable
variation ID retrieval. The previous method returned variation data arrays
which may not have worked correctly in all contexts.

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:56:07 +01:00
755108a7d3 Add release package v1.1.11 with checksums
Feature release adding variable product support.

Package: wc-composable-product-v1.1.11.zip (414 KB, 379 files)
SHA-256: 214002a28a0426b4d2423f234d1dff63e4a8e58c6301cbd6eaed8db670db88c6
MD5: 63b105311dc1cc8ac67c05528ad02e30

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:53:00 +01:00
85983d5473 Bump version to 1.1.11 - Add variable product support
Feature release adding support for variable products and their variations.

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:51:49 +01:00
252b187600 Add variable product support to product selector
BREAKING CHANGE: Variable products now supported in composable products.

Changes:
- Modified get_available_products() to detect variable products
- Variable products now expand to show all available variations
- Each variation is checked individually for stock and purchasability
- Simple products continue to work as before

This allows customers to select specific variations (e.g., "T-Shirt - Red - Large")
instead of just seeing "T-Shirt" which wasn't selectable.

Fixes the issue where products weren't showing because all products
in the category/tag were variable products.

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:48:17 +01:00
8185a77697 Document v1.1.10 session history in CLAUDE.md
Added comprehensive Session 12 documentation covering:
- Two critical bug fixes (admin tabs + frontend empty state)
- Root cause analysis for both issues
- CSS panel hiding solution
- Twig empty state conditional
- Translation updates for all 6 locales
- Key lessons learned about CSS timing and empty states
- Debugging approach and testing recommendations

Fixed markdown linting warnings (MD036, MD032).

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:43:37 +01:00
6c2e317230 Add release package v1.1.10 with checksums
Critical bug fixes for admin and frontend issues discovered after v1.1.9.

Package: wc-composable-product-v1.1.10.zip (413 KB, 379 files)
SHA-256: 63bfe97aa9fd98e74750786ed0e1579b069505e85558316f7042787994c856ac
MD5: 271aad47684ee8318a8824861d5fc387

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:40:45 +01:00
58f5329bc4 Bump version to 1.1.10 - Fix admin and frontend critical bugs
Two critical bug fixes after v1.1.9 release:
1. Admin tabs both visible on initial page load
2. Frontend product selector showing no products

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:38:42 +01:00
0767016370 Update translations with new empty state message
Added translation for "No products available for selection" message in all 6 locales:
- de_DE (formal): Keine Produkte zur Auswahl verfügbar. Bitte konfigurieren Sie...
- de_DE_informal: ...Bitte konfiguriere...
- de_CH (formal): ...Bitte konfigurieren Sie...
- de_CH_informal: ...Bitte konfiguriere...
- fr_CH: Aucun produit disponible pour la sélection...
- it_CH: Nessun prodotto disponibile per la selezione...

Recompiled all .mo files.

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:37:51 +01:00
fa7ec0e422 Fix two critical bugs in v1.1.9
Bug 1: Admin - Both General and Composable tabs visible on initial load
- Added explicit display:none to #composable_product_data panel
- Panel now hidden by default, shows only when product-type-composable class is present
- Prevents both tabs showing simultaneously on new product creation

Bug 2: Frontend - No feedback when no products configured
- Added empty state message when products array is empty
- Users now see helpful message instead of blank grid
- Cleared Twig cache to ensure template changes take effect

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:34:46 +01:00
f4d2543d4e Add release package v1.1.9 with checksums
Critical bug fix release - Admin rendering completely broken in v1.1.8.

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:22:54 +01:00
9e4513f911 Bump version to 1.1.9 - Fix critical admin rendering bug
Fix overly broad CSS selectors that broke admin tabs in v1.1.8.

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:21:30 +01:00
4dc7b767a8 Fix admin CSS - tabs disappeared due to overly aggressive selectors
CRITICAL FIX: The previous CSS change used .show_if_composable with !important
which hid ALL elements with that class, including the tab links themselves.

Changes:
- Changed from .show_if_composable to .options_group.show_if_composable
- Changed from .product_data_tabs .composable_options to li.composable_options
- Removed !important flags (not needed with specific selectors)
- Now only hides the general tab option groups, not the tab links

This fixes:
- Missing Composable Options tab in product edit screen
- Fields appearing out of context
- Tab navigation completely broken

The issue was that WooCommerce adds 'show_if_composable' class to BOTH:
1. The tab link (li.composable_options.show_if_composable)
2. The general tab fields (div.options_group.show_if_composable)

Now we specifically target only the option groups, leaving tabs alone.

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:16:37 +01:00
f763e35d19 Add release package v1.1.8 with checksums
Package details:
- Size: 411 KB (421,220 bytes)
- Files: 376 (all source + vendor + translations + .mo files)
- Directory structure: wc-composable-product/ (WordPress standard)
- SHA-256: d7d06e2a5d336609249f803b681cdf270dbe60d6fc28bdd6c451c6744d2fdab6
- MD5: 78eee5eee4762c308c5d37d1aac06b04

Release v1.1.8 includes:
- Critical admin rendering bug fix (tab visibility)
- Critical frontend product selector fix (WooCommerce button hiding)
- Critical price localization fix (wc_price integration)

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:13:59 +01:00
8b271c90c0 Merge branch 'dev' 2025-12-31 22:12:50 +01:00
0dd4408b23 Bump version to 1.1.8 for release
Version 1.1.8 includes critical UI bug fixes:
- Admin rendering bug (tab visibility)
- Frontend product selector not appearing
- Price formatting localization

Updated CHANGELOG.md with comprehensive v1.1.8 release notes.

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:12:43 +01:00
7a4a0a0135 Document v1.1.8 bug fixes in session history
Added comprehensive documentation for Session 11 covering three critical UI bug fixes:

1. Admin rendering bug - Fixed CSS specificity for proper tab visibility
2. Frontend product selector not appearing - Added woocommerce_is_purchasable filter
3. Localized price formatting - Implemented full WooCommerce price format support

Documentation includes:
- Detailed root cause analysis for each bug
- Complete code examples showing the fixes
- Links to specific file locations and line numbers
- Key lessons learned about WordPress/WooCommerce integration
- Testing recommendations for verification

Updated "Bugs found" section to mark all three issues as fixed in v1.1.8.

Added note about v1.1.6/v1.1.7 package structure fix (parent directory issue).

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:10:01 +01:00
c6a48d6404 Fix critical UI bugs in admin and frontend
Fixes three critical bugs reported in CLAUDE.md:

1. Admin rendering bug - Fixed CSS to prevent both General and Composable
   Options tabs from showing simultaneously on initial page load
   - Enhanced CSS specificity with !important flags
   - Added body.product-type-composable selectors for proper visibility control
   - Hides Composable Options tab by default, shows only when composable type selected

2. Frontend product selector not appearing - Fixed WooCommerce integration
   - Added hide_default_add_to_cart() method to Cart_Handler
   - Hooks woocommerce_is_purchasable filter to return false for composable products
   - This hides WooCommerce's default add-to-cart button
   - Allows our custom product selector to be the only interface

3. Localized price formatting - Implemented proper WooCommerce price formatting
   - Added wc_price Twig function in Plugin.php
   - Updated Product_Selector to pass formatted price HTML to template
   - Added price_format data to JavaScript localization
   - Implemented formatPrice() method in frontend.js
   - Supports all WooCommerce price formats (currency position, decimals, separators)
   - Template now uses {{ fixed_price_html|raw }} and {{ zero_price_html|raw }}
   - JavaScript dynamically formats prices using locale-specific settings

Technical improvements:
- Cart_Handler.php: +14 lines (hide_default_add_to_cart method)
- Plugin.php: +7 lines (wc_price function, price format localization)
- Product_Selector.php: +2 lines (formatted price HTML context)
- templates/product-selector.twig: Modified to use formatted price HTML
- assets/css/admin.css: +24 lines (enhanced tab visibility control)
- assets/js/frontend.js: +28 lines (formatPrice method with WooCommerce format support)

All PHP files pass lint checks. Frontend now properly displays localized prices
with correct currency symbols, decimal separators, and thousand separators for
all WooCommerce-supported locales (CHF for Switzerland, € for Europe, etc.).

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:08:08 +01:00
ac1cb9b135 Merge dev to main with corrected v1.1.6 package structure
🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 22:00:19 +01:00
f5bc0d0335 Fix v1.1.6 release package structure for WordPress compatibility
Corrected directory structure to match WordPress plugin standards.
Package now contains wc-composable-product/ parent directory with all files inside.
WordPress will extract to wp-content/plugins/wc-composable-product/ (correct!).

Package size: 410 KB (419,430 bytes)
New SHA-256: fb8f12486f19aef61f6e6ea4af63fe66f64adca66e2d42e1d17e9f05cb82f39f
New MD5: cc8c13780c4e8063c97b8632d0a43adb

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 21:57:49 +01:00
88a907c4dd Fix v1.1.7 release package structure for WordPress compatibility
CRITICAL FIX: WordPress plugin directory structure

Previous issue:
- ZIP contained files at root level
- WordPress created directory based on ZIP name: wc-composable-product-v1.1.7/
- This caused version-numbered directory in wp-content/plugins/

Corrected structure:
- ZIP now contains wc-composable-product/ parent directory
- All plugin files inside wc-composable-product/ directory
- WordPress extracts to wp-content/plugins/wc-composable-product/ (correct!)

Changes:
- Recreated ZIP with proper directory structure using rsync + zip
- Package size: 410 KB (419,697 bytes) - slightly larger due to parent dir
- New SHA-256: 866e7dd34431f4c881629fd8b59ddd3a27c7a45b7324a3d88cd064a3e01c1b83
- New MD5: 871fbb3b910380c0e43bcf1538408eda

WordPress standard: Plugin ZIP must contain single parent directory named after
the plugin slug, with all files inside that directory.

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 21:56:49 +01:00
03a7624564 Update CLAUDE.md with release package commit information
Added post-release updates section to v1.1.7 documentation:
- Both v1.1.6 and v1.1.7 release packages now committed to repository
- v1.1.6: 378 KB with .po files only (translations won't work)
- v1.1.7: 393 KB with .po + .mo files (translations functional)
- All packages include SHA-256 and MD5 checksums

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 21:50:54 +01:00
1c3f44f3c2 Add release package v1.1.6 with checksums
Release package for v1.1.6 including:
- Complete admin translations for Fixed Price field (v1.1.4 feature)
- All 6 locales with 56/56 strings translated (.po files)
- Package size: 378 KB (1,092,772 bytes)
- Files included: 370 files

Checksums:
- SHA-256: d64f4f5f1a00d392989cb613780e5726106a08c6aace08e0c74c80553a0b0f1e
- MD5: eae384e342450abd4ac83af0266ac764

Note: This version includes .po translation files but not compiled .mo files.
Users should upgrade to v1.1.7 for functional translations in WordPress.

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 21:48:47 +01:00
287f8b778b Add release package v1.1.7 with checksums
Release package for v1.1.7 including:
- Compiled .mo translation files for all 6 locales
- Critical fix: Translations now display in WordPress admin
- Package size: 393 KB (402,351 bytes)
- Files included: 376 files

Checksums:
- SHA-256: 518d411c8a35fff26f6cd07dd7548dd46dfc2d8452ce3735b96e10cd582bf3fc
- MD5: 2eb25087a470ff2cf7d36490ea34eed9

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 21:46:28 +01:00
63d8f9ed52 Document v1.1.7 release in CLAUDE.md session history
Added comprehensive documentation for Session 10:
- Critical bug fix: Missing translations due to lack of compiled .mo files
- Root cause analysis: WordPress requires .mo files, not just .po files
- Solution: Compiled all 6 .po files to .mo format using msgfmt
- Release details: v1.1.7 with 393 KB package, 376 files
- Key lessons: .po vs .mo files, WordPress i18n requirements, testing workflow

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

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 21:45:14 +01:00
44 changed files with 1675 additions and 43 deletions

View File

@@ -0,0 +1,205 @@
name: Create Release Package
on:
push:
tags:
- 'v*'
jobs:
build-release:
runs-on: ubuntu-latest
steps:
- name: Checkout code
uses: actions/checkout@v4
- name: Setup PHP
uses: shivammathur/setup-php@v2
with:
php-version: '8.3'
extensions: mbstring, xml, zip, intl, gettext
tools: composer:v2
- name: Get version from tag
id: version
run: |
VERSION=${GITHUB_REF_NAME#v}
echo "version=$VERSION" >> $GITHUB_OUTPUT
echo "Building version: $VERSION"
- name: Validate composer.json
run: composer validate --strict
- name: Install Composer dependencies (production)
run: |
composer config platform.php 8.3.0
composer install --no-dev --optimize-autoloader --no-interaction
- name: Install gettext
run: apt-get update && apt-get install -y gettext
- name: Compile translations
run: |
for po in languages/*.po; do
if [ -f "$po" ]; then
mo="${po%.po}.mo"
echo "Compiling $po to $mo"
msgfmt -o "$mo" "$po"
fi
done
- name: Verify plugin version matches tag
run: |
PLUGIN_VERSION=$(grep -oP "Version:\s*\K[0-9]+\.[0-9]+\.[0-9]+" wc-composable-product.php | head -1)
TAG_VERSION=${{ steps.version.outputs.version }}
if [ "$PLUGIN_VERSION" != "$TAG_VERSION" ]; then
echo "Error: Plugin version ($PLUGIN_VERSION) does not match tag version ($TAG_VERSION)"
exit 1
fi
echo "Version verified: $PLUGIN_VERSION"
- name: Create release directory
run: mkdir -p releases
- name: Build release package
run: |
VERSION=${{ steps.version.outputs.version }}
PLUGIN_NAME="wc-composable-product"
RELEASE_FILE="releases/${PLUGIN_NAME}-${VERSION}.zip"
# Move to parent directory for proper zip structure
cd ..
# Create zip with proper WordPress plugin structure
zip -r "${PLUGIN_NAME}/${RELEASE_FILE}" "${PLUGIN_NAME}" \
-x "${PLUGIN_NAME}/.git/*" \
-x "${PLUGIN_NAME}/.gitea/*" \
-x "${PLUGIN_NAME}/.github/*" \
-x "${PLUGIN_NAME}/.vscode/*" \
-x "${PLUGIN_NAME}/.claude/*" \
-x "${PLUGIN_NAME}/CLAUDE.md" \
-x "${PLUGIN_NAME}/wp-core" \
-x "${PLUGIN_NAME}/wp-core/*" \
-x "${PLUGIN_NAME}/wp-plugins" \
-x "${PLUGIN_NAME}/wp-plugins/*" \
-x "${PLUGIN_NAME}/releases/*" \
-x "${PLUGIN_NAME}/cache/*" \
-x "${PLUGIN_NAME}/composer.lock" \
-x "${PLUGIN_NAME}/*.log" \
-x "${PLUGIN_NAME}/.gitignore" \
-x "${PLUGIN_NAME}/.editorconfig" \
-x "${PLUGIN_NAME}/phpcs.xml*" \
-x "${PLUGIN_NAME}/phpunit.xml*" \
-x "${PLUGIN_NAME}/tests/*" \
-x "${PLUGIN_NAME}/*.po~" \
-x "${PLUGIN_NAME}/*.bak" \
-x "*.DS_Store"
cd "${PLUGIN_NAME}"
echo "Created: ${RELEASE_FILE}"
- name: Generate checksums
run: |
VERSION=${{ steps.version.outputs.version }}
cd releases
sha256sum "wc-composable-product-${VERSION}.zip" > "wc-composable-product-${VERSION}.zip.sha256"
echo "SHA256:"
cat "wc-composable-product-${VERSION}.zip.sha256"
- name: Verify package structure
run: |
set +o pipefail
VERSION=${{ steps.version.outputs.version }}
echo "Package contents:"
unzip -l "releases/wc-composable-product-${VERSION}.zip" | head -50 || true
# Verify main file is at correct location
if unzip -l "releases/wc-composable-product-${VERSION}.zip" | grep -q "wc-composable-product/wc-composable-product.php"; then
echo "✓ Main plugin file at correct location"
else
echo "✗ Error: Main plugin file not found at wc-composable-product/wc-composable-product.php"
exit 1
fi
# Verify vendor directory is included
if unzip -l "releases/wc-composable-product-${VERSION}.zip" | grep -q "wc-composable-product/vendor/"; then
echo "✓ Vendor directory included"
else
echo "✗ Error: Vendor directory not found"
exit 1
fi
- name: Extract changelog for release notes
id: changelog
run: |
VERSION=${{ steps.version.outputs.version }}
# Extract changelog section for this version
NOTES=$(sed -n "/^## \[${VERSION}\]/,/^## \[/p" CHANGELOG.md | sed '$ d' | tail -n +2)
if [ -z "$NOTES" ]; then
NOTES="Release version ${VERSION}"
fi
# Save to file for multi-line output
echo "$NOTES" > release_notes.txt
echo "Release notes extracted"
- name: Create Gitea Release
env:
GITEA_TOKEN: ${{ secrets.SRC_GITEA_TOKEN }}
run: |
VERSION=${{ steps.version.outputs.version }}
TAG_NAME=${{ github.ref_name }}
PRERELEASE="false"
if [[ "$TAG_NAME" == *-* ]]; then
PRERELEASE="true"
fi
# Read release notes
BODY=$(cat release_notes.txt)
# Check if release already exists for this tag and delete it
EXISTING_RELEASE=$(curl -s \
-H "Authorization: token ${GITEA_TOKEN}" \
"${GITHUB_SERVER_URL}/api/v1/repos/${GITHUB_REPOSITORY}/releases/tags/${TAG_NAME}")
EXISTING_ID=$(echo "$EXISTING_RELEASE" | jq -r '.id // empty')
if [ -n "$EXISTING_ID" ] && [ "$EXISTING_ID" != "null" ]; then
echo "Deleting existing release ID: $EXISTING_ID"
curl -s -X DELETE \
-H "Authorization: token ${GITEA_TOKEN}" \
"${GITHUB_SERVER_URL}/api/v1/repos/${GITHUB_REPOSITORY}/releases/${EXISTING_ID}"
fi
# Create release via Gitea API
RELEASE_RESPONSE=$(curl -s -X POST \
-H "Authorization: token ${GITEA_TOKEN}" \
-H "Content-Type: application/json" \
-d "{\"tag_name\": \"${TAG_NAME}\", \"name\": \"Release ${VERSION}\", \"body\": $(echo "$BODY" | jq -Rs .), \"draft\": false, \"prerelease\": ${PRERELEASE}}" \
"${GITHUB_SERVER_URL}/api/v1/repos/${GITHUB_REPOSITORY}/releases")
RELEASE_ID=$(echo "$RELEASE_RESPONSE" | jq -r '.id')
if [ "$RELEASE_ID" == "null" ] || [ -z "$RELEASE_ID" ]; then
echo "Failed to create release:"
echo "$RELEASE_RESPONSE"
exit 1
fi
echo "Created release ID: $RELEASE_ID"
# Upload attachments
for file in "releases/wc-composable-product-${VERSION}.zip" "releases/wc-composable-product-${VERSION}.zip.sha256"; do
if [ -f "$file" ]; then
FILENAME=$(basename "$file")
echo "Uploading $FILENAME..."
curl -s -X POST \
-H "Authorization: token ${GITEA_TOKEN}" \
-H "Content-Type: application/octet-stream" \
--data-binary "@$file" \
"${GITHUB_SERVER_URL}/api/v1/repos/${GITHUB_REPOSITORY}/releases/${RELEASE_ID}/assets?name=${FILENAME}"
echo "Uploaded $FILENAME"
fi
done
echo "Release created successfully: ${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY}/releases/tag/${TAG_NAME}"

0
.gitignore vendored Normal file → Executable file
View File

View File

@@ -5,6 +5,262 @@ All notable changes to this project will be documented in this file.
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.2.0] - 2026-03-01
### Fixed
- **CRITICAL**: Product selection always empty regardless of configuration (categories, tags, or SKUs)
- Root cause: Product query used a `meta_query` checking `_product_type` in `wp_postmeta`, but WooCommerce stores product types in the `product_type` **taxonomy** — the `!=` comparison with a non-existent meta key caused an INNER JOIN returning zero results
- Fix: Replaced broken `meta_query` with correct `tax_query` using `product_type` taxonomy to exclude composable products
- **CRITICAL**: Cart price always 0.00 despite correct frontend price calculation
- Root cause: `composable_price_calculated` flag was persisted to the cart session, preventing price recalculation on subsequent page loads — but `set_price()` only modifies the in-memory product object and is lost between requests
- Fix: Removed per-item session flag; the existing static `$already_calculated` flag already prevents duplicate calculation within a single request
- **Admin tab rendering**: Both General and Composable Options panels visible on initial page load
- Root cause: JavaScript manually showed `#composable_product_data` via `.show()` without hiding the General panel
- Fix: Trigger WooCommerce's native tab click instead, so the tab system handles panel visibility correctly
### Added
- **Gitea CI/CD release workflow** (`.gitea/workflows/release.yml`)
- Triggered on `v*` tags
- Installs PHP 8.3 with production Composer dependencies
- Compiles `.po``.mo` translations
- Verifies plugin version matches tag
- Builds release ZIP with proper WordPress directory structure
- Generates SHA-256 checksums
- Verifies package contains main plugin file and vendor directory
- Extracts changelog for release notes
- Creates Gitea release with attachments via API
### Removed
- Debug logging from v1.1.14 (no longer needed after root cause identified)
### Technical
- Modified files: includes/Product_Type.php, includes/Cart_Handler.php, assets/js/admin.js
- New file: .gitea/workflows/release.yml
- Product query now correctly uses `tax_query` with `product_type` taxonomy (`NOT IN` operator)
- Cart price recalculated on every request via `woocommerce_before_calculate_totals` hook
- Admin JS uses `$('ul.product_data_tabs li.composable_options a').trigger('click')` for native WooCommerce tab handling
## [1.1.14] - 2025-12-31
### Added
- **DEBUG**: Comprehensive debug logging to troubleshoot product retrieval issues
- Error log output shows selection criteria, query arguments, and results
- Logs each product/variation being added to help identify filtering issues
- Enable by setting WP_DEBUG to true in wp-config.php
### Technical
- Modified file: includes/Product_Type.php (added error_log statements throughout get_available_products())
- Logs criteria array (categories, tags, SKUs)
- Logs WP_Query arguments before execution
- Logs number of posts found by query
- Logs each variable product's variation count
- Logs each variation/simple product being added with name
- Logs total products available at end
- All logging wrapped in WP_DEBUG checks (no performance impact in production)
### Notes
- This is a debug release to help diagnose why products aren't showing
- No functional changes from v1.1.13
- User should enable WP_DEBUG and check debug.log or error.log
- Log output will show exactly where products are being filtered out
- All translation files remain at 100% completion (57/57 strings)
## [1.1.13] - 2025-12-31
### Fixed
- **CRITICAL**: "No products available for selection" message showing even when products are configured
- Removed overly strict `is_in_stock()` requirement that was filtering out all products
- Products now show regardless of stock status (out-of-stock items are displayed but disabled)
- Added `'relation' => 'AND'` to meta_query for proper handling of multiple meta conditions
### Changed
- Product retrieval now shows all purchasable products, not just in-stock ones
- Stock status still displayed on frontend with appropriate styling
- Out-of-stock items shown but disabled via checkbox and visual indicators
- Frontend stock management from v1.1.0 still fully functional
### Technical
- Modified file: includes/Product_Type.php (lines 117-124, 177, 181)
- Changed from `$variation->is_in_stock() && $variation->is_purchasable()` to just `$variation->is_purchasable()`
- Changed from `$product->is_in_stock() && $product->is_purchasable()` to just `$product->is_purchasable()`
- Added `'relation' => 'AND'` to meta_query array for WordPress query compatibility
### Notes
- This fixes the issue where NO products were showing in the selector
- Stock validation still occurs at add-to-cart time (Stock_Manager class)
- Frontend still displays stock badges (in stock, low stock, out of stock)
- Out-of-stock items remain non-selectable via disabled checkboxes
- All translation files remain at 100% completion (57/57 strings)
## [1.1.12] - 2025-12-31
### Fixed
- **CRITICAL**: Variable product variations still not appearing in product selector after v1.1.11 release
- Changed variation retrieval method from `get_available_variations()` to `get_children()` for more reliable variation ID retrieval
- `get_available_variations()` returns complex data arrays which may not work in all contexts
- `get_children()` returns simple array of variation IDs directly, ensuring consistent results
### Technical
- Modified file: includes/Product_Type.php (lines 171-184)
- Changed from `$product->get_available_variations()` to `$product->get_children()`
- More direct and reliable method for retrieving variation IDs
- Each variation ID passed to `wc_get_product()` for full product object
- Maintains all stock and purchasability checks from v1.1.11
### Notes
- This is a patch release fixing the variable product support introduced in v1.1.11
- User reported "nope, still no product selectable" after v1.1.11
- Root cause: `get_available_variations()` returns variation data arrays instead of clean IDs
- `get_children()` is the standard WooCommerce method for retrieving variation IDs
- All translation files remain at 100% completion (57/57 strings - no changes needed)
## [1.1.11] - 2025-12-31
### Added
- **FEATURE**: Variable product support - composable products can now include variable products and their variations
- Variable products automatically expand to show all available variations as selectable items
- Each variation displays with full attribute information (e.g., "Product - Color: Red, Size: Large")
### Fixed
- Products not showing in selector when all available products were variable products
- Variable products were being filtered out because parent products aren't directly purchasable
### Changed
- Modified `get_available_products()` to detect and handle variable products
- Variable products now expand into their individual variations
- Each variation checked individually for stock status and purchasability
- Simple products continue to work exactly as before
### Technical
- Modified file: includes/Product_Type.php (lines 160-188)
- Added logic to detect `is_type('variable')` products
- Uses `get_available_variations()` to retrieve all variations
- Each variation validated with `is_in_stock()` and `is_purchasable()`
- Maintains backward compatibility with simple products
### Notes
- This is a feature enhancement release, not a bug fix
- Resolves the issue where categories containing only variable products showed no selections
- Variations display with their parent product name plus selected attributes
- Stock management works correctly for both simple products and variations
- All translation files remain at 100% completion (57/57 strings - no new strings added)
## [1.1.10] - 2025-12-31
### Fixed
- **CRITICAL**: Admin panel - Both General and Composable tabs visible simultaneously on initial page load
- **CRITICAL**: Frontend - No products showing in product selector, only cart button and pricing visible
- Empty product grid now shows helpful message instead of blank space
### Changed
- Added explicit `display: none` to `#composable_product_data` panel for proper initial hiding
- Panel now only shows when `body.product-type-composable` class is present
- Added empty state message in product selector template when no products are configured
- Cleared Twig cache to ensure template changes take effect
### Added
- Empty state message: "No products available for selection. Please configure the product criteria in the admin panel."
- Translations for empty state message in all 6 supported locales (de_DE, de_DE_informal, de_CH, de_CH_informal, fr_CH, it_CH)
- Recompiled .mo translation files
### Technical
- Modified files: assets/css/admin.css (lines 7-16), templates/product-selector.twig (lines 12-15)
- Root cause (admin): Panel lacked explicit CSS hiding rule, relied only on `hidden` class
- Root cause (frontend): No feedback when products array is empty
- Solution: CSS specificity + empty state conditional in Twig template
### Notes
- This release fixes two critical bugs discovered immediately after v1.1.9
- Admin interface now correctly hides composable panel until product type is selected
- Frontend provides clear user feedback when product selection is unavailable
- All translation files now 100% complete (57/57 strings)
## [1.1.9] - 2025-12-31
### Fixed
- **CRITICAL**: Admin rendering completely broken - tabs disappeared and fields appeared out of context after v1.1.8 release
- CSS selectors were too broad, hiding tab navigation along with field groups
- Removed `!important` flags that caused overly aggressive hiding
### Changed
- Made CSS selectors more specific: `.options_group.show_if_composable` for field groups only
- Added separate rule for tab links: `.product_data_tabs li.composable_options`
- Tab navigation now works correctly without hiding itself
### Technical
- Modified files: assets/css/admin.css (lines 22-40)
- Root cause: `.show_if_composable` class used by WooCommerce for both tab links AND field groups
- Solution: Separate selectors for each use case to prevent unintended hiding
### Notes
- This release fixes critical regression introduced in v1.1.8
- Admin interface now renders correctly with visible tabs and properly positioned fields
- No `!important` flags needed with specific selectors
## [1.1.8] - 2025-12-31
### Fixed
- **CRITICAL**: Admin rendering bug where both General and Composable Options tabs showed simultaneously on initial page load
- **CRITICAL**: Frontend product selector not appearing on product pages - WooCommerce's default add-to-cart button now hidden for composable products
- **CRITICAL**: Price formatting not localized - prices now display with proper currency symbols, decimal separators, and thousand separators for all locales
### Added
- `wc_price()` Twig function for proper price formatting in templates
- `formatPrice()` JavaScript method with full WooCommerce locale support
- Price format localization data passed to frontend JavaScript (decimal/thousand separators, currency position, number of decimals)
- `hide_default_add_to_cart()` method to prevent WooCommerce's default purchase UI for composable products
### Changed
- Enhanced CSS specificity with `!important` flags for proper tab visibility control
- Template now uses `{{ fixed_price_html|raw }}` instead of raw currency concatenation
- Product selector passes pre-formatted price HTML from `wc_price()` function
- Frontend JavaScript updates prices dynamically using WooCommerce format settings
### Technical
- Modified files: assets/css/admin.css (+24 lines), includes/Cart_Handler.php (+14 lines), includes/Plugin.php (+7 lines), includes/Product_Selector.php (+2 lines), templates/product-selector.twig, assets/js/frontend.js (+28 lines)
- All PHP files pass syntax validation
- Supports Swiss format (CHF 50.-), European format (50,00 €), US format ($50.00), and all other WooCommerce locales
- Thousand separator support: comma (1,000), dot (1.000), apostrophe (1'000), space (1 000)
### Notes
- This release fixes all three critical UI bugs reported in CLAUDE.md
- Admin tabs now display correctly on initial page load without JavaScript flicker
- Frontend product selector is now the only purchase interface (no WooCommerce default button)
- All prices maintain proper locale formatting during dynamic updates
## [1.1.7] - 2025-12-31
### Added

1032
CLAUDE.md

File diff suppressed because it is too large Load Diff

View File

@@ -10,10 +10,13 @@ This plugin adds a new product type to WooCommerce that allows customers to buil
- **Custom Product Type**: New "Composable Product" type in WooCommerce
- **Flexible Selection**: Define available products by category, tag, or SKU
- **Variable Product Support**: Automatically expands variable products into selectable variations
- **Stock Management**: Real-time stock validation, visual indicators, and automatic inventory tracking
- **Configurable Limits**: Set global or per-product selection limits
- **Pricing Options**: Fixed price or sum of selected products
- **Multi-language Support**: Fully translatable with i18n support
- **Pricing Options**: Fixed price or sum of selected products with full locale-aware formatting
- **Multi-language Support**: Fully translated in 6 locales (de_DE, de_CH, fr_CH, it_CH + informal variants)
- **Modern UI**: Clean interface built with Twig templates and vanilla JavaScript
- **CI/CD**: Automated release workflow for Gitea
## Requirements
@@ -43,6 +46,7 @@ This plugin adds a new product type to WooCommerce that allows customers to buil
### Global Settings
Navigate to WooCommerce > Settings > Composable Products to configure:
- Default selection limit
- Default pricing mode
- Display options
@@ -60,10 +64,28 @@ composer install
### Translation
Generate POT file:
```bash
wp i18n make-pot . languages/wc-composable-product.pot
```
Compile translations:
```bash
for po in languages/*.po; do msgfmt -o "${po%.po}.mo" "$po"; done
```
### Creating Releases
Releases are automated via Gitea CI/CD. Push an annotated tag to trigger:
```bash
git tag -a v1.2.0 -m "Release v1.2.0"
git push origin v1.2.0
```
The workflow builds the release ZIP, compiles translations, generates checksums, and creates a Gitea release with attachments.
## License
GPL v3 or later - see LICENSE file for details

23
assets/css/admin.css Normal file → Executable file
View File

@@ -4,10 +4,17 @@
* @package WC_Composable_Product
*/
/* Hide composable panel by default */
#composable_product_data {
display: none;
padding: 12px;
}
/* Show composable panel when composable type is selected */
body.product-type-composable #composable_product_data {
display: block;
}
.composable_criteria_group {
border-top: 1px solid #eee;
padding-top: 12px;
@@ -19,11 +26,23 @@
min-height: 150px;
}
.show_if_composable {
/* Hide composable-specific elements by default (but not tabs) */
.options_group.show_if_composable {
display: none;
}
.product-type-composable .show_if_composable {
/* Show composable elements when composable product type is selected */
body.product-type-composable .options_group.show_if_composable {
display: block;
}
/* Hide the Composable Options tab link by default */
.product_data_tabs li.composable_options {
display: none;
}
/* Show the Composable Options tab when composable type selected */
body.product-type-composable .product_data_tabs li.composable_options {
display: block;
}

0
assets/css/frontend.css Normal file → Executable file
View File

8
assets/js/admin.js Normal file → Executable file
View File

@@ -17,12 +17,14 @@
if (productType === 'composable') {
$('.show_if_composable').show();
$('.hide_if_composable').hide();
$('#composable_product_data').show();
$('.product_data_tabs .composable_options a').show();
// Show the composable tab, then click it so WooCommerce's
// native tab system hides all other panels properly
$('.product_data_tabs li.composable_options').show();
$('ul.product_data_tabs li.composable_options a').trigger('click');
} else {
$('.show_if_composable').hide();
$('.product_data_tabs li.composable_options').hide();
$('#composable_product_data').hide();
$('.product_data_tabs .composable_options a').hide();
}
}).trigger('change');

34
assets/js/frontend.js Normal file → Executable file
View File

@@ -63,6 +63,36 @@
this.clearMessages($container);
},
/**
* Format price using WooCommerce settings
*
* @param {number} price Price amount
* @return {string} Formatted price HTML
*/
formatPrice: function(price) {
if (typeof wcComposableProduct === 'undefined' || !wcComposableProduct.price_format) {
return price.toFixed(2);
}
const format = wcComposableProduct.price_format;
const decimals = parseInt(format.decimals, 10);
const decimalSep = format.decimal_separator;
const thousandSep = format.thousand_separator;
// Format number
let priceStr = price.toFixed(decimals);
const parts = priceStr.split('.');
parts[0] = parts[0].replace(/\B(?=(\d{3})+(?!\d))/g, thousandSep);
priceStr = parts.join(decimalSep);
// Apply price format (e.g., "%1$s%2$s" for symbol+price or "%2$s%1$s" for price+symbol)
let formatted = format.price_format
.replace('%1$s', '<span class="woocommerce-Price-currencySymbol">' + format.currency_symbol + '</span>')
.replace('%2$s', priceStr);
return '<span class="woocommerce-Price-amount amount">' + formatted + '</span>';
},
/**
* Update total price
*
@@ -79,8 +109,8 @@
}
});
const currencySymbol = $container.find('.total-price').data('currency');
$container.find('.calculated-total').text(currencySymbol + total.toFixed(2));
const formattedPrice = this.formatPrice(total);
$container.find('.calculated-total').html(formattedPrice);
},
/**

View File

@@ -6,7 +6,7 @@
"authors": [
{
"name": "Marco Graetsch",
"email": "marco@example.com"
"email": "magdev3.0@gmail.com"
}
],
"require": {

View File

@@ -35,6 +35,21 @@ class Cart_Handler {
add_action('woocommerce_before_calculate_totals', [$this, 'calculate_cart_item_price']);
add_action('woocommerce_single_product_summary', [$this, 'render_product_selector'], 25);
add_action('woocommerce_checkout_create_order_line_item', [$this->stock_manager, 'store_selected_products_in_order'], 10, 3);
add_filter('woocommerce_is_purchasable', [$this, 'hide_default_add_to_cart'], 10, 2);
}
/**
* Hide default WooCommerce add to cart button for composable products
*
* @param bool $is_purchasable Is purchasable status
* @param \WC_Product $product Product object
* @return bool
*/
public function hide_default_add_to_cart($is_purchasable, $product) {
if ($product && $product->get_type() === 'composable') {
return false;
}
return $is_purchasable;
}
/**
@@ -185,7 +200,7 @@ class Cart_Handler {
return;
}
// Use static flag to prevent multiple executions
// Use static flag to prevent multiple executions within the same request
static $already_calculated = false;
if ($already_calculated) {
return;
@@ -193,13 +208,10 @@ class Cart_Handler {
foreach ($cart->get_cart() as $cart_item_key => $cart_item) {
if (isset($cart_item['data']) && $cart_item['data']->get_type() === 'composable') {
if (isset($cart_item['composable_products']) && !isset($cart_item['composable_price_calculated'])) {
if (isset($cart_item['composable_products'])) {
$product = $cart_item['data'];
$price = $product->calculate_composed_price($cart_item['composable_products']);
$cart_item['data']->set_price($price);
// Mark as calculated to prevent re-calculation by other plugins
$cart->cart_contents[$cart_item_key]['composable_price_calculated'] = true;
}
}
}

View File

@@ -84,6 +84,7 @@ class Plugin {
$this->twig->addFunction(new \Twig\TwigFunction('esc_html', 'esc_html'));
$this->twig->addFunction(new \Twig\TwigFunction('esc_attr', 'esc_attr'));
$this->twig->addFunction(new \Twig\TwigFunction('esc_url', 'esc_url'));
$this->twig->addFunction(new \Twig\TwigFunction('wc_price', 'wc_price'));
// Add WordPress escaping functions as Twig filters
$this->twig->addFilter(new \Twig\TwigFilter('esc_html', 'esc_html'));
@@ -161,6 +162,13 @@ class Plugin {
'max_items' => __('Maximum items selected', 'wc-composable-product'),
'min_items' => __('Please select at least one item', 'wc-composable-product'),
],
'price_format' => [
'currency_symbol' => get_woocommerce_currency_symbol(),
'decimal_separator' => wc_get_price_decimal_separator(),
'thousand_separator' => wc_get_price_thousand_separator(),
'decimals' => wc_get_price_decimals(),
'price_format' => get_woocommerce_price_format(),
],
]);
}
}

View File

@@ -65,6 +65,8 @@ class Product_Selector {
'show_prices' => $show_prices,
'show_total' => $show_total,
'fixed_price' => $product->get_price(),
'fixed_price_html' => wc_price($product->get_price()),
'zero_price_html' => wc_price(0),
'currency_symbol' => get_woocommerce_currency_symbol(),
];

View File

@@ -110,15 +110,17 @@ class Product_Type extends \WC_Product {
'post_status' => 'publish',
'orderby' => 'title',
'order' => 'ASC',
'tax_query' => [],
];
// Exclude composable products from selection
$args['meta_query'] = [
// Exclude composable products using the product_type taxonomy
// (WooCommerce stores product types as taxonomy terms, NOT as postmeta)
$args['tax_query'] = [
'relation' => 'AND',
[
'key' => '_product_type',
'value' => 'composable',
'compare' => '!=',
'taxonomy' => 'product_type',
'field' => 'slug',
'terms' => ['composable'],
'operator' => 'NOT IN',
],
];
@@ -148,10 +150,12 @@ class Product_Type extends \WC_Product {
case 'sku':
if (!empty($criteria['skus'])) {
$skus = array_map('trim', explode(',', $criteria['skus']));
$args['meta_query'][] = [
$args['meta_query'] = [
[
'key' => '_sku',
'value' => $skus,
'compare' => 'IN',
],
];
}
break;
@@ -163,13 +167,28 @@ class Product_Type extends \WC_Product {
if ($query->have_posts()) {
foreach ($query->posts as $post) {
$product = wc_get_product($post->ID);
if ($product && $product->is_in_stock() && $product->is_purchasable()) {
if (!$product) {
continue;
}
// Handle variable products by including their variations
if ($product->is_type('variable')) {
$variation_ids = $product->get_children();
foreach ($variation_ids as $variation_id) {
$variation = wc_get_product($variation_id);
if ($variation && $variation->is_purchasable()) {
$products[] = $variation;
}
}
} elseif ($product->is_purchasable()) {
$products[] = $product;
}
}
}
wp_reset_postdata();
return $products;
}

4
languages/wc-composable-product-de_CH.po Normal file → Executable file
View File

@@ -238,3 +238,7 @@ msgstr "übrig"
#: templates/product-selector.twig
msgid "In stock"
msgstr "An Lager"
#: templates/product-selector.twig
msgid "No products available for selection. Please configure the product criteria in the admin panel."
msgstr "Keine Produkte zur Auswahl verfügbar. Bitte konfigurieren Sie die Produktkriterien im Admin-Bereich."

4
languages/wc-composable-product-de_CH_informal.po Normal file → Executable file
View File

@@ -238,3 +238,7 @@ msgstr "übrig"
#: templates/product-selector.twig
msgid "In stock"
msgstr "An Lager"
#: templates/product-selector.twig
msgid "No products available for selection. Please configure the product criteria in the admin panel."
msgstr "Keine Produkte zur Auswahl verfügbar. Bitte konfiguriere die Produktkriterien im Admin-Bereich."

4
languages/wc-composable-product-de_DE.po Normal file → Executable file
View File

@@ -238,3 +238,7 @@ msgstr "übrig"
#: templates/product-selector.twig
msgid "In stock"
msgstr "Auf Lager"
#: templates/product-selector.twig
msgid "No products available for selection. Please configure the product criteria in the admin panel."
msgstr "Keine Produkte zur Auswahl verfügbar. Bitte konfigurieren Sie die Produktkriterien im Admin-Bereich."

4
languages/wc-composable-product-de_DE_informal.po Normal file → Executable file
View File

@@ -238,3 +238,7 @@ msgstr "übrig"
#: templates/product-selector.twig
msgid "In stock"
msgstr "Auf Lager"
#: templates/product-selector.twig
msgid "No products available for selection. Please configure the product criteria in the admin panel."
msgstr "Keine Produkte zur Auswahl verfügbar. Bitte konfiguriere die Produktkriterien im Admin-Bereich."

4
languages/wc-composable-product-fr_CH.po Normal file → Executable file
View File

@@ -238,3 +238,7 @@ msgstr "restant"
#: templates/product-selector.twig
msgid "In stock"
msgstr "En stock"
#: templates/product-selector.twig
msgid "No products available for selection. Please configure the product criteria in the admin panel."
msgstr "Aucun produit disponible pour la sélection. Veuillez configurer les critères de produit dans le panneau d'administration."

4
languages/wc-composable-product-it_CH.po Normal file → Executable file
View File

@@ -238,3 +238,7 @@ msgstr "rimasti"
#: templates/product-selector.twig
msgid "In stock"
msgstr "Disponibile"
#: templates/product-selector.twig
msgid "No products available for selection. Please configure the product criteria in the admin panel."
msgstr "Nessun prodotto disponibile per la selezione. Si prega di configurare i criteri del prodotto nel pannello di amministrazione."

4
languages/wc-composable-product.pot Normal file → Executable file
View File

@@ -237,3 +237,7 @@ msgstr ""
#: templates/product-selector.twig
msgid "In stock"
msgstr ""
#: templates/product-selector.twig
msgid "No products available for selection. Please configure the product criteria in the admin panel."
msgstr ""

View File

@@ -1 +0,0 @@
aec3bae001f0013322a73fa941169688 wc-composable-product-v1.0.0.zip

View File

@@ -1 +0,0 @@
4a0f7ec2171aeabfdfe155419fd6124f35f3e14501ee2ca324bbab447259a8bb wc-composable-product-v1.0.0.zip

View File

@@ -1 +0,0 @@
0a60816bbc5a01c0057c1ffa72679d93 releases/wc-composable-product-v1.1.0.zip

View File

@@ -1 +0,0 @@
645fdd68aca95cba77d961f3a48d41b9c12b3d17552572b7c039575dcfcab693 releases/wc-composable-product-v1.1.0.zip

View File

@@ -1 +0,0 @@
db09928aea6fffbf9c2e754d2264f2bc wc-composable-product-v1.1.1.zip

View File

@@ -1 +0,0 @@
761eef69da910ecfdb20ceeed70b5d0381c7cab895e81a040d132cb0f88d749b wc-composable-product-v1.1.1.zip

View File

@@ -1 +0,0 @@
37cef191778b448dcbd2ae10141f64c6 wc-composable-product-v1.1.2.zip

View File

@@ -1 +0,0 @@
191eae035b34ce8b33b90cf9d85ed54e493c1b471cda0efe5c992a512e91cc36 wc-composable-product-v1.1.2.zip

View File

@@ -1 +0,0 @@
9bbed416019a796b4d4a5ef72e016e1f wc-composable-product-v1.1.3.zip

View File

@@ -1 +0,0 @@
0ca23ca12570f0e9c518514ffc5209d78c76c3295954d10ec74a28013a762956 wc-composable-product-v1.1.3.zip

12
templates/product-selector.twig Normal file → Executable file
View File

@@ -9,6 +9,11 @@
</div>
<div class="composable-products-grid">
{% if products is empty %}
<div class="composable-no-products">
<p>{{ __('No products available for selection. Please configure the product criteria in the admin panel.') }}</p>
</div>
{% else %}
{% for product in products %}
<div class="composable-product-item{% if not product.in_stock %} out-of-stock{% endif %}" data-product-id="{{ product.id }}" data-price="{{ product.price }}" data-stock-status="{{ product.stock_status }}">
<div class="product-item-inner">
@@ -52,16 +57,17 @@
</div>
</div>
{% endfor %}
{% endif %}
</div>
{% if show_total %}
<div class="composable-total">
<div class="total-label">{{ __('Total Price:') }}</div>
<div class="total-price" data-currency="{{ currency_symbol }}">
<div class="total-price" data-currency="{{ currency_symbol }}" data-fixed-price="{{ fixed_price }}">
{% if pricing_mode == 'fixed' %}
{{ currency_symbol }}{{ fixed_price }}
{{ fixed_price_html|raw }}
{% else %}
<span class="calculated-total">{{ currency_symbol }}0.00</span>
<span class="calculated-total">{{ zero_price_html|raw }}</span>
{% endif %}
</div>
</div>

View File

@@ -1,11 +1,12 @@
<?php
/**
* Plugin Name: WooCommerce Composable Products
* Plugin URI: https://github.com/magdev/wc-composable-product
* Plugin URI: https://src.bundespruefstelle.ch/magdev/wc-composable-product
* Description: Create composable products where customers select a limited number of items from a configurable set
* Version: 1.1.7
* Version: 1.2.0
* Author: Marco Graetsch
* Author URI: https://example.com
* Author URI: https://src.bundespruefstelle.ch/magdev
* License: GPL v3 or later
* License URI: https://www.gnu.org/licenses/gpl-3.0.html
* Text Domain: wc-composable-product
@@ -19,7 +20,7 @@
defined('ABSPATH') || exit;
// Define plugin constants
define('WC_COMPOSABLE_PRODUCT_VERSION', '1.1.7');
define('WC_COMPOSABLE_PRODUCT_VERSION', '1.2.0');
define('WC_COMPOSABLE_PRODUCT_FILE', __FILE__);
define('WC_COMPOSABLE_PRODUCT_PATH', plugin_dir_path(__FILE__));
define('WC_COMPOSABLE_PRODUCT_URL', plugin_dir_url(__FILE__));