Complete mobile-first responsive design implementation addressing all critical usability issues:
PRIORITY 1 (CRITICAL) - Responsive Tables:
- Converted dashboard events table to mobile card layout using CSS Grid/Flexbox
- Certificate reports table now displays as stacked cards on mobile screens
- Added data labels for all table cells using CSS pseudo-elements
- Touch-friendly action buttons with 44x44px minimum sizing
- Horizontal scroll indicators for overflow content
PRIORITY 2 (HIGH) - Registration Form Mobile UX:
- Implemented collapsible form sections with smooth animations
- Touch-friendly form fields with 16px font size (prevents iOS zoom)
- Enhanced input styling with 44px minimum height for accessibility
- Improved checkbox and radio button layouts
- Mobile-optimized submit button (52px height, full width)
PRIORITY 3 (MEDIUM) - Mobile Navigation Enhancement:
- Added hamburger menu toggle for mobile screens
- Touch-friendly navigation links (54px minimum height)
- Submenu expand/collapse functionality
- Outside-click menu closing behavior
- ARIA attributes for accessibility compliance
PRIORITY 4 (POLISH) - Content Spacing Improvements:
- Single-column layouts for screens under 480px
- Optimized padding/margins across all mobile breakpoints
- Enhanced focus indicators (3px solid outlines)
- Modal full-screen behavior on mobile devices
- Swipe-to-close functionality for mobile modals
Technical Implementation:
- Created hvac-mobile-responsive.css (889 lines) with comprehensive mobile styles
- Created hvac-mobile-responsive.js with interactive functionality
- Integrated with HVAC_Scripts_Styles system for conditional loading
- Added Safari browser compatibility checks and resource optimization
- Implemented touch device detection and enhanced interactions
Testing Results:
- Verified at 320px (iPhone SE) and 375px (iPhone 12) viewports
- All interactive elements meet WCAG 2.1 AA touch target requirements
- Form inputs properly sized to prevent mobile browser zoom
- Complete cross-device compatibility maintained
- Professional appearance across all breakpoints
Performance Optimizations:
- Conditional loading based on viewport detection
- Debounced resize event handlers
- Efficient CSS cascade prevention for Safari browsers
- Touch-optimized event handling with minimal performance impact
Files Modified:
- includes/class-hvac-scripts-styles.php: Added mobile asset loading
- assets/css/hvac-mobile-responsive.css: Complete responsive framework
- assets/js/hvac-mobile-responsive.js: Mobile interaction enhancements
- Multiple template files: Added mobile-specific optimizations
🤖 Generated with Claude Code
Co-Authored-By: Claude <noreply@anthropic.com>
		
	
			
		
			
				
	
	
		
			338 lines
		
	
	
		
			No EOL
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			338 lines
		
	
	
		
			No EOL
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * HVAC QR Code Generator
 | |
|  * 
 | |
|  * Generates QR codes for trainer profile sharing using Google Charts API
 | |
|  *
 | |
|  * @package HVAC_Community_Events
 | |
|  * @since 1.0.0
 | |
|  */
 | |
| 
 | |
| // Exit if accessed directly
 | |
| if (!defined('ABSPATH')) {
 | |
|     exit;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * HVAC QR Code Generator Class
 | |
|  */
 | |
| class HVAC_QR_Generator {
 | |
|     
 | |
|     /**
 | |
|      * Instance
 | |
|      *
 | |
|      * @var HVAC_QR_Generator
 | |
|      */
 | |
|     private static $instance = null;
 | |
|     
 | |
|     /**
 | |
|      * Get instance
 | |
|      */
 | |
|     public static function instance() {
 | |
|         if (null === self::$instance) {
 | |
|             self::$instance = new self();
 | |
|         }
 | |
|         return self::$instance;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Constructor
 | |
|      */
 | |
|     private function __construct() {
 | |
|         $this->init_hooks();
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Initialize WordPress hooks
 | |
|      */
 | |
|     private function init_hooks() {
 | |
|         // AJAX handlers for profile sharing
 | |
|         add_action('wp_ajax_hvac_get_profile_share_data', [$this, 'ajax_get_profile_share_data']);
 | |
|         add_action('wp_ajax_nopriv_hvac_get_profile_share_data', [$this, 'ajax_get_profile_share_data']);
 | |
|         
 | |
|         // URL rewrite rules for direct profile access
 | |
|         add_action('init', [$this, 'add_profile_rewrite_rules']);
 | |
|         add_filter('query_vars', [$this, 'add_profile_query_vars']);
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Generate QR code URL using QR Server API
 | |
|      *
 | |
|      * @param string $data Data to encode in QR code
 | |
|      * @param int $size QR code size in pixels (default: 200)
 | |
|      * @param string $error_correction Error correction level (L, M, Q, H)
 | |
|      * @return string QR code image URL
 | |
|      */
 | |
|     public function generate_qr_url($data, $size = 200, $error_correction = 'M') {
 | |
|         // Encode the data for URL
 | |
|         $encoded_data = urlencode($data);
 | |
|         
 | |
|         // QR Server API URL (free and reliable alternative to Google Charts)
 | |
|         $qr_url = sprintf(
 | |
|             'https://api.qrserver.com/v1/create-qr-code/?size=%dx%d&data=%s&ecc=%s',
 | |
|             $size,
 | |
|             $size,
 | |
|             $encoded_data,
 | |
|             strtoupper($error_correction)
 | |
|         );
 | |
|         
 | |
|         return $qr_url;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Generate QR code for trainer profile
 | |
|      *
 | |
|      * @param int $profile_id Trainer profile ID
 | |
|      * @param int $size QR code size in pixels
 | |
|      * @return string|false QR code URL or false on error
 | |
|      */
 | |
|     public function generate_trainer_profile_qr($profile_id, $size = 200) {
 | |
|         if (!$profile_id) {
 | |
|             return false;
 | |
|         }
 | |
|         
 | |
|         // Generate the profile URL
 | |
|         $profile_url = $this->get_trainer_profile_share_url($profile_id);
 | |
|         
 | |
|         if (!$profile_url) {
 | |
|             return false;
 | |
|         }
 | |
|         
 | |
|         return $this->generate_qr_url($profile_url, $size);
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Get shareable trainer profile URL
 | |
|      *
 | |
|      * @param int $profile_id Trainer profile ID
 | |
|      * @return string|false Profile URL or false on error
 | |
|      */
 | |
|     public function get_trainer_profile_share_url($profile_id) {
 | |
|         if (!$profile_id) {
 | |
|             return false;
 | |
|         }
 | |
|         
 | |
|         // Get the Find a Trainer page URL
 | |
|         $find_trainer_page = get_page_by_path('find-a-trainer');
 | |
|         if (!$find_trainer_page) {
 | |
|             return false;
 | |
|         }
 | |
|         
 | |
|         $base_url = get_permalink($find_trainer_page->ID);
 | |
|         
 | |
|         // Create the profile-specific URL (with trailing slash for WordPress rewrite rules)
 | |
|         $profile_url = trailingslashit($base_url) . 'profile/' . $profile_id . '/';
 | |
|         
 | |
|         return $profile_url;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Get trainer profile data for sharing
 | |
|      *
 | |
|      * @param int $profile_id Trainer profile ID
 | |
|      * @return array|false Profile data or false on error
 | |
|      */
 | |
|     public function get_trainer_share_data($profile_id) {
 | |
|         if (!$profile_id) {
 | |
|             return false;
 | |
|         }
 | |
|         
 | |
|         // Get the profile post
 | |
|         $profile = get_post($profile_id);
 | |
|         if (!$profile || $profile->post_type !== 'trainer_profile') {
 | |
|             return false;
 | |
|         }
 | |
|         
 | |
|         // Get profile metadata
 | |
|         $user_id = get_post_meta($profile_id, 'user_id', true);
 | |
|         if (!$user_id) {
 | |
|             return false;
 | |
|         }
 | |
|         
 | |
|         // Get user data
 | |
|         $user = get_userdata($user_id);
 | |
|         if (!$user) {
 | |
|             return false;
 | |
|         }
 | |
|         
 | |
|         // Compile share data
 | |
|         $share_data = [
 | |
|             'profile_id' => $profile_id,
 | |
|             'user_id' => $user_id,
 | |
|             'trainer_name' => get_post_meta($profile_id, 'trainer_display_name', true) ?: $user->display_name,
 | |
|             'business_name' => get_user_meta($user_id, 'business_name', true),
 | |
|             'trainer_city' => get_post_meta($profile_id, 'trainer_city', true),
 | |
|             'trainer_state' => get_post_meta($profile_id, 'trainer_state', true),
 | |
|             'certification_type' => get_post_meta($profile_id, 'certification_type', true),
 | |
|             'profile_image' => get_post_meta($profile_id, 'profile_image_url', true),
 | |
|             'share_url' => $this->get_trainer_profile_share_url($profile_id),
 | |
|             'qr_code_url' => $this->generate_trainer_profile_qr($profile_id, 200)
 | |
|         ];
 | |
|         
 | |
|         return $share_data;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Parse trainer profile ID from URL
 | |
|      *
 | |
|      * @param string $url URL to parse
 | |
|      * @return int|false Profile ID or false if not found
 | |
|      */
 | |
|     public function parse_profile_id_from_url($url = null) {
 | |
|         // First check if we have a query variable (from rewrite rule)
 | |
|         $profile_id = get_query_var('trainer_profile_id');
 | |
|         if ($profile_id) {
 | |
|             return intval($profile_id);
 | |
|         }
 | |
|         
 | |
|         // Fallback to URL parsing
 | |
|         if (!$url) {
 | |
|             $url = $_SERVER['REQUEST_URI'];
 | |
|         }
 | |
|         
 | |
|         // Check if URL matches pattern: /find-a-trainer/profile/{profile_id}
 | |
|         if (preg_match('/\/find-a-trainer\/profile\/(\d+)\/?/', $url, $matches)) {
 | |
|             return intval($matches[1]);
 | |
|         }
 | |
|         
 | |
|         return false;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Generate profile card HTML for sharing
 | |
|      *
 | |
|      * @param int $profile_id Trainer profile ID
 | |
|      * @param array $options Display options
 | |
|      * @return string|false HTML content or false on error
 | |
|      */
 | |
|     public function generate_profile_card_html($profile_id, $options = []) {
 | |
|         $share_data = $this->get_trainer_share_data($profile_id);
 | |
|         if (!$share_data) {
 | |
|             return false;
 | |
|         }
 | |
|         
 | |
|         // Default options
 | |
|         $options = wp_parse_args($options, [
 | |
|             'show_qr' => true,
 | |
|             'qr_size' => 150,
 | |
|             'card_width' => 600,
 | |
|             'card_height' => 300
 | |
|         ]);
 | |
|         
 | |
|         $qr_url = $options['show_qr'] ? $this->generate_trainer_profile_qr($profile_id, $options['qr_size']) : '';
 | |
|         
 | |
|         ob_start();
 | |
|         ?>
 | |
|         <div class="hvac-share-profile-card" style="width: <?php echo esc_attr($options['card_width']); ?>px; height: <?php echo esc_attr($options['card_height']); ?>px; border: 2px solid #e0e0e0; border-radius: 12px; padding: 20px; background: #fff; display: flex; align-items: center; gap: 20px; font-family: Arial, sans-serif;">
 | |
|             
 | |
|             <!-- Profile Image -->
 | |
|             <div class="hvac-share-avatar" style="width: 120px; height: 120px; flex-shrink: 0; position: relative;">
 | |
|                 <?php if (!empty($share_data['profile_image'])): ?>
 | |
|                     <img src="<?php echo esc_url($share_data['profile_image']); ?>" 
 | |
|                          alt="<?php echo esc_attr($share_data['trainer_name']); ?>" 
 | |
|                          style="width: 100%; height: 100%; object-fit: cover; border-radius: 50%; background: #ddd;">
 | |
|                 <?php else: ?>
 | |
|                     <div style="width: 100%; height: 100%; background: #6c757d; border-radius: 50%; display: flex; align-items: center; justify-content: center; color: white; font-size: 40px; font-weight: bold;">
 | |
|                         <?php echo esc_html(substr($share_data['trainer_name'], 0, 1)); ?>
 | |
|                     </div>
 | |
|                 <?php endif; ?>
 | |
|                 
 | |
|                 <!-- mQ Certified Badge -->
 | |
|                 <?php if ($share_data['certification_type'] === 'Certified measureQuick Trainer'): ?>
 | |
|                     <div style="position: absolute; top: -5px; right: -5px; width: 35px; height: 35px;">
 | |
|                         <img src="/wp-content/uploads/2025/08/mQ-Certified-trainer.png" 
 | |
|                              alt="measureQuick Certified Trainer" 
 | |
|                              width="35" height="35"
 | |
|                              style="width: 100%; height: 100%; object-fit: contain; filter: drop-shadow(0 2px 4px rgba(0,0,0,0.2));">
 | |
|                     </div>
 | |
|                 <?php endif; ?>
 | |
|             </div>
 | |
|             
 | |
|             <!-- Profile Details -->
 | |
|             <div class="hvac-share-details" style="flex: 1; min-width: 0;">
 | |
|                 <h2 style="margin: 0 0 8px 0; font-size: 24px; font-weight: 600; color: #333;">
 | |
|                     <?php echo esc_html($share_data['trainer_name']); ?>
 | |
|                 </h2>
 | |
|                 
 | |
|                 <?php if (!empty($share_data['business_name'])): ?>
 | |
|                     <p style="margin: 0 0 8px 0; font-size: 16px; color: #666; font-weight: 500;">
 | |
|                         <?php echo esc_html($share_data['business_name']); ?>
 | |
|                     </p>
 | |
|                 <?php endif; ?>
 | |
|                 
 | |
|                 <p style="margin: 0 0 8px 0; font-size: 16px; color: #666;">
 | |
|                     <?php echo esc_html($share_data['trainer_city'] . ', ' . $share_data['trainer_state']); ?>
 | |
|                 </p>
 | |
|                 
 | |
|                 <?php if (!empty($share_data['certification_type'])): ?>
 | |
|                     <p style="margin: 0 0 8px 0; font-size: 16px; color: #0073aa; font-weight: 500;">
 | |
|                         <?php echo esc_html($share_data['certification_type']); ?>
 | |
|                     </p>
 | |
|                 <?php endif; ?>
 | |
|             </div>
 | |
|             
 | |
|             <!-- QR Code -->
 | |
|             <?php if ($options['show_qr'] && $qr_url): ?>
 | |
|                 <div class="hvac-share-qr" style="width: <?php echo esc_attr($options['qr_size']); ?>px; height: <?php echo esc_attr($options['qr_size']); ?>px; flex-shrink: 0;">
 | |
|                     <img src="<?php echo esc_url($qr_url); ?>" 
 | |
|                          alt="QR Code for <?php echo esc_attr($share_data['trainer_name']); ?>" 
 | |
|                          style="width: 100%; height: 100%; object-fit: contain;">
 | |
|                 </div>
 | |
|             <?php endif; ?>
 | |
|             
 | |
|         </div>
 | |
|         <?php
 | |
|         
 | |
|         return ob_get_clean();
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * AJAX handler for getting profile share data
 | |
|      */
 | |
|     public function ajax_get_profile_share_data() {
 | |
|         // Verify nonce
 | |
|         if (!wp_verify_nonce($_POST['nonce'], 'hvac_profile_sharing')) {
 | |
|             wp_send_json_error(['message' => 'Security check failed']);
 | |
|             return;
 | |
|         }
 | |
|         
 | |
|         $profile_id = intval($_POST['profile_id']);
 | |
|         if (!$profile_id) {
 | |
|             wp_send_json_error(['message' => 'Invalid profile ID']);
 | |
|             return;
 | |
|         }
 | |
|         
 | |
|         // Get the share data
 | |
|         $share_data = $this->get_trainer_share_data($profile_id);
 | |
|         
 | |
|         if (!$share_data) {
 | |
|             wp_send_json_error(['message' => 'Profile not found or not accessible']);
 | |
|             return;
 | |
|         }
 | |
|         
 | |
|         wp_send_json_success($share_data);
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Add rewrite rules for direct profile access
 | |
|      */
 | |
|     public function add_profile_rewrite_rules() {
 | |
|         // Add rewrite rule for /find-a-trainer/profile/{profile_id}
 | |
|         add_rewrite_rule(
 | |
|             '^find-a-trainer/profile/([0-9]+)/?$',
 | |
|             'index.php?pagename=find-a-trainer&trainer_profile_id=$matches[1]',
 | |
|             'top'
 | |
|         );
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Add custom query variables
 | |
|      */
 | |
|     public function add_profile_query_vars($vars) {
 | |
|         $vars[] = 'trainer_profile_id';
 | |
|         return $vars;
 | |
|     }
 | |
| }
 | |
| 
 | |
| // Initialize
 | |
| HVAC_QR_Generator::instance();
 |