Implemented intelligent header rules based on user feedback.
Problem Analysis:
1. Dashboard submenu tabs already show page names (Overview, Revenue, Orders...)
2. Showing "Orders" header is ambiguous (Analytics or Management?)
3. Wasted vertical space for redundant information
4. FAB already handles actions on management pages
Solution: Headers ONLY When They Add Value
Rules Implemented:
1. Dashboard Pages: NO HEADERS
- Submenu tabs are sufficient
- Saves vertical space
- No ambiguity
Before:
Dashboard → Overview = "Dashboard" header (redundant!)
Dashboard → Orders = "Orders" header (confusing!)
After:
Dashboard → Overview = No header (tabs show "Overview")
Dashboard → Orders = No header (tabs show "Orders")
2. Settings Pages: HEADERS ONLY WITH ACTIONS
- Store Details + [Save] = Show header ✓
- Payments + [Refresh] = Show header ✓
- Pages without actions = No header (save space)
Logic: If there is an action button, we need a place to put it → header
If no action button, header is just wasting space → remove it
3. Management Pages: NO HEADERS
- FAB handles actions ([+ Add Order])
- No need for redundant header with action button
4. Payments Exception: REMOVED
- Treat Payments like any other settings page
- Has action (Refresh) = show header
- Consistent with other pages
Implementation:
Dashboard Pages (7 files):
- Removed usePageHeader hook
- Removed useEffect for setting header
- Removed unused imports (useEffect, usePageHeader)
- Result: Clean, no headers, tabs are enough
PageHeader Component:
- Removed Payments special case detection
- Removed useLocation import
- Simplified logic: hideOnDesktop prop only
SettingsLayout Component:
- Changed logic: Only set header when onSave OR action exists
- If no action: clearPageHeader() instead of setPageHeader(title)
- Result: Headers only appear when needed
Benefits:
✅ Saves vertical space (no redundant headers)
✅ No ambiguity (Dashboard Orders vs Orders Management)
✅ Consistent logic (action = header, no action = no header)
✅ Cleaner UI (less visual clutter)
✅ FAB handles management page actions
Files Modified:
- Dashboard/index.tsx (Overview)
- Dashboard/Revenue.tsx
- Dashboard/Orders.tsx
- Dashboard/Products.tsx
- Dashboard/Customers.tsx
- Dashboard/Coupons.tsx
- Dashboard/Taxes.tsx
- PageHeader.tsx
- SettingsLayout.tsx
Result: Smart headers that only appear when they add value! 🎯
Applied "bigger picture" thinking - added contextual headers to ALL submenu pages consistently.
Problem: Only some pages had headers, creating inconsistent UX
Issues Fixed:
1. Dashboard Submenu Pages - All Now Have Headers
Before: Only Overview had header
After: All 6 pages have headers (Revenue, Orders, Products, Customers, Coupons, Taxes)
2. Settings Pages Desktop - Show Headers (Except Payments)
Before: PageHeader was md:hidden on all pages
After: Shows on desktop for Settings pages, hidden only for Payments (special case)
Implementation:
- Added usePageHeader to 6 Dashboard submenu pages
- Modified PageHeader to show on desktop by default
- Auto-detect Payments page and hide header there
Result:
- ALL Dashboard pages have contextual headers
- ALL Settings pages have contextual headers on desktop
- Payments page special case handled
- Consistent UX across entire app
- No more bald pages!
Files Modified: 6 Dashboard pages + PageHeader.tsx
Fixed the layout hierarchy - PageHeader should be ABOVE submenu, not below.
Correct Information Architecture:
1. Page Title (Contextual Header) ← "Where am I?"
2. Submenu Tabs ← "What can I do here?"
3. Content ← "The actual data"
Changes Made:
1. ✅ Desktop Fullscreen Layout
Before: Submenu → PageHeader
After: PageHeader → Submenu
2. ✅ Mobile Fullscreen Layout
Before: Submenu → PageHeader (inside main)
After: PageHeader → Submenu (outside main)
3. ✅ Non-Fullscreen Layout
Before: TopNav → Submenu → PageHeader
After: TopNav → PageHeader → Submenu
4. ✅ Updated Z-Index
Before: PageHeader z-10 (below submenu)
After: PageHeader z-20 (same as submenu, but DOM order puts it on top)
Why This Order Makes Sense:
- User sees PAGE TITLE first ("Store Details")
- Then sees NAVIGATION OPTIONS (WooNooW, Store Details, Payments, Shipping)
- Then sees CONTENT (the actual form fields)
Visual Flow:
┌─────────────────────────────────┐
│ Store Details [Save] │ ← Contextual header (what page)
├─────────────────────────────────┤
│ WooNooW | Store Details | ... │ ← Submenu (navigation)
├─────────────────────────────────┤
│ Store Identity │
│ Store name * │ ← Content
│ [My Wordpress Store] │
└─────────────────────────────────┘
Before (Wrong):
User: "What are these tabs for?" (sees submenu first)
Then: "Oh, I'm on Store Details" (sees title after)
After (Correct):
User: "I'm on Store Details" (sees title first)
Then: "I can navigate to WooNooW, Payments, etc." (sees options)
Files Modified:
- App.tsx: Reordered PageHeader to be before SubmenuBar in all 3 layouts
- PageHeader.tsx: Updated z-index to z-20 (same as submenu)
Result: Proper information hierarchy! ✨
Fixed 3 issues and completed FAB implementation:
1. ✅ Dynamic Submenu Top Position
- Submenu now moves to top-0 when header is hidden
- Moves back to top-16 when header is visible
- Smooth transition based on scroll
Implementation:
- Added isHeaderVisible state in Shell
- Header notifies parent via onVisibilityChange callback
- Submenu receives headerVisible prop
- Dynamic topClass: headerVisible ? 'top-16' : 'top-0'
2. ✅ Hide Submenu on More Page
- More page now has no submenu bar
- Cleaner UI for navigation menu
Implementation:
- Added isMorePage check: location.pathname === '/more'
- Conditionally render submenu: {!isMorePage && (...)}
3. ✅ FAB Working on All Pages
- Dashboard: Quick Actions (placeholder)
- Orders: Create Order → /orders/new ✅
- Products: Add Product → /products/new
- Customers: Add Customer → /customers/new
- Coupons: Create Coupon → /coupons/new
Implementation:
- Added useFABConfig('orders') to Orders page
- FAB now visible and functional
- Clicking navigates to create page
Mobile Navigation Flow:
┌─────────────────────────────────┐
│ App Bar (hides on scroll) │
├─────────────────────────────────┤
│ Submenu (top-0 when bar hidden) │ ← Dynamic!
├─────────────────────────────────┤
│ Page Header (sticky) │
├─────────────────────────────────┤
│ Content (scrollable) │
│ [+] FAB │ ← Working!
├─────────────────────────────────┤
│ Bottom Nav (fixed) │
└─────────────────────────────────┘
More Page (Clean):
┌─────────────────────────────────┐
│ App Bar │
├─────────────────────────────────┤
│ (No submenu) │ ← Clean!
├─────────────────────────────────┤
│ More Page Content │
│ - Coupons │
│ - Settings │
├─────────────────────────────────┤
│ Bottom Nav │
└─────────────────────────────────┘
Files Modified:
- App.tsx: Added header visibility tracking, More page check
- SubmenuBar.tsx: Added headerVisible prop, dynamic top
- DashboardSubmenuBar.tsx: Added headerVisible prop, dynamic top
- Orders/index.tsx: Added useFABConfig('orders')
Next Steps:
- Add useFABConfig to Products, Customers, Coupons pages
- Implement speed dial menu for Dashboard FAB
- Test on real devices
Result:
✅ Submenu position responds to header visibility
✅ More page has clean layout
✅ FAB working on Orders page
✅ Ready to add FAB to remaining pages
THE BIGGER PICTURE - Root Cause Analysis:
Problem Chain:
1. FABContext value recreated every render
2. All FAB consumers re-render
3. Dashboard re-renders
4. useFABConfig runs
5. Creates new icon/callbacks
6. Triggers FABContext update
7. INFINITE LOOP!
The Bug (in BOTH contexts):
<Context.Provider value={{ config, setFAB, clearFAB }}>
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
NEW object every render!
Every time Provider re-renders:
- Creates NEW value object
- All consumers see "new" value
- All consumers re-render
- Causes more Provider re-renders
- INFINITE LOOP!
The Fix:
const setFAB = useCallback(..., []); // Stable function
const clearFAB = useCallback(..., []); // Stable function
const value = useMemo(() => ({ config, setFAB, clearFAB }), [config, setFAB, clearFAB]);
^^^^^^^
Only creates new object when dependencies actually change!
<Context.Provider value={value}>
^^^^^^^
Stable reference!
Why This is Critical:
Context is at the TOP of the component tree:
App
└─ FABProvider ← Bug here affects EVERYTHING below
└─ PageHeaderProvider ← Bug here too
└─ DashboardProvider
└─ Shell
└─ Dashboard ← Infinite re-renders
└─ Charts ← Break from constant re-renders
React Context Performance Rules:
1. ALWAYS memoize context value object
2. ALWAYS use useCallback for context functions
3. NEVER create inline objects in Provider value
4. Context updates trigger ALL consumers
Fixed Contexts:
1. FABContext - Memoized value, callbacks
2. PageHeaderContext - Memoized value, callbacks
Before:
Every render → new value object → all consumers re-render → LOOP
After:
Only config changes → new value object → consumers re-render once → done
Result:
✅ No infinite loops
✅ No unnecessary re-renders
✅ Clean console
✅ Smooth performance
✅ All features working
Files Modified:
- FABContext.tsx: Added useMemo and useCallback
- PageHeaderContext.tsx: Added useMemo and useCallback
- useFABConfig.tsx: Memoized icon and callbacks (previous fix)
- App.tsx: Fixed scroll detection with useRef (previous fix)
All infinite loop sources now eliminated!
Fixed all 5 issues:
1. ✅ FAB Now Shows
- Added useFABConfig('dashboard') to Dashboard page
- FAB renders and positioned correctly
2. ✅ Top Bar Scroll-Hide Working
- Changed from window.scrollY to scrollContainer.scrollTop
- Added scrollContainerRef to track correct scroll element
- Scroll detection now works on mobile layout
- Smooth slide animation (300ms)
3. ✅ Main Menu (TopNav) Hidden on Mobile
- Removed TopNav from mobile fullscreen layout
- Bottom nav is now the primary navigation
- Cleaner mobile UI with less clutter
4. ✅ Contextual Header Shows
- PageHeader component renders in mobile layout
- Sticky positioning below submenu
- Shows page title and action buttons
5. ✅ More Page Already Good
- No changes needed
Root Cause Analysis:
Issue #1 (FAB not shown):
- FAB component was created but no page was using useFABConfig()
- Fixed by adding useFABConfig('dashboard') to Dashboard
Issue #2 (Scroll not working):
- Was listening to window.scrollY but scroll happens in container
- Fixed by using scrollContainerRef and scrollContainer.scrollTop
Issue #3 (TopNav still visible):
- TopNav was redundant with BottomNav on mobile
- Removed from mobile layout entirely
Issue #4 (No contextual header):
- PageHeader was there but might not have been visible
- Confirmed it's rendering correctly now
Mobile Layout (Fixed):
┌─────────────────────────────────┐
│ My Store [Exit] │ ← Hides on scroll down
├─────────────────────────────────┤
│ [Overview] [Revenue] [Orders] │ ← Submenu (sticky)
├─────────────────────────────────┤
│ Dashboard │ ← Page header (sticky)
├─────────────────────────────────┤
│ │
│ Content Area │
│ (scrollable) │
│ [+] │ ← FAB (visible!)
│ │
├─────────────────────────────────┤
│ [🏠] [📋] [📦] [👥] [⋯] │ ← Bottom nav
└─────────────────────────────────┘
Files Modified:
- App.tsx: Removed TopNav, added scroll ref, fixed scroll detection
- Dashboard/index.tsx: Added useFABConfig('dashboard')
Test Results:
✅ FAB visible and clickable
✅ Header hides on scroll down
✅ Header shows on scroll up
✅ No TopNav on mobile
✅ PageHeader shows correctly
✅ Bottom nav works perfectly
Implemented mobile-optimized navigation structure:
1. Bottom Navigation (Mobile Only)
- 5 items: Dashboard, Orders, Products, Customers, More
- Fixed at bottom, always visible
- Thumb-friendly positioning
- Active state indication
- Hidden on desktop (md:hidden)
2. More Menu Page
- Overflow menu for Coupons and Settings
- Clean list layout with icons
- Descriptions for each item
- Chevron indicators
3. FAB (Floating Action Button)
- Context-aware system via FABContext
- Fixed bottom-right (72px from bottom)
- Hidden on desktop (md:hidden)
- Ready for contextual actions per page
4. FAB Context System
- Global state for FAB configuration
- setFAB() / clearFAB() methods
- Supports icon, label, onClick, visibility
- Allows pages to control FAB behavior
5. Layout Updates
- Added pb-14 to main for bottom nav spacing
- BottomNav and FAB in mobile fullscreen layout
- Wrapped app with FABProvider
Structure (Mobile):
┌─────────────────────────────────┐
│ App Bar (will hide on scroll) │
├─────────────────────────────────┤
│ Page Header (sticky, contextual)│
├─────────────────────────────────┤
│ Submenu (sticky) │
├─────────────────────────────────┤
│ Content (scrollable) │
│ [+] FAB │
├─────────────────────────────────┤
│ Bottom Nav (fixed) │
└─────────────────────────────────┘
Next Steps:
- Implement scroll-hide for app bar
- Add contextual FAB per page
- Test on real devices
Files Created:
- BottomNav.tsx: Bottom navigation component
- More/index.tsx: More menu page
- FABContext.tsx: FAB state management
- FAB.tsx: Floating action button component
- useScrollDirection.ts: Scroll detection hook
Files Modified:
- App.tsx: Added bottom nav, FAB, More route, providers
Problem:
- Content still not shrinking on narrow viewports
- Horizontal scrolling persists
- Header shrinks but body doesn't
Root Cause:
Missing min-w-0 on parent containers:
<main className="flex-1 flex flex-col"> ← No min-w-0!
<div className="overflow-auto p-4"> ← No min-w-0!
<AppRoutes />
Without min-w-0, flex containers won't shrink below their
content's natural width, even if children have min-w-0.
Solution:
Add min-w-0 to the entire container chain:
<main className="flex-1 flex flex-col min-h-0 min-w-0">
<div className="overflow-auto p-4 min-w-0">
<AppRoutes />
Container Chain (all need min-w-0):
┌────────────────────────────────────┐
│ <div flex> │
│ <Sidebar flex-shrink-0> │
│ <main flex-1 min-w-0> ✅ │ ← Added
│ <SubmenuBar> │
│ <PageHeader> │
│ <div overflow-auto min-w-0> ✅ │ ← Added
│ <AppRoutes> │
│ <SettingsLayout min-w-0> │
│ <PageHeader min-w-0> │
│ Content... │
└────────────────────────────────────┘
Applied to all 3 layouts:
1. Fullscreen Desktop (Sidebar + Main)
2. Fullscreen Mobile (TopNav + Main)
3. WP-Admin (TopNav + Main)
Why this works:
- min-w-0 must be on EVERY flex container in the chain
- Breaking the chain at any level prevents shrinking
- Now entire tree can shrink from root to leaf
Files Modified:
- App.tsx: Added min-w-0 to <main> and scrollable <div>
Result:
✅ Content shrinks properly on all viewports
✅ No horizontal scrolling
✅ Works from 320px to 1920px+
✅ All layouts (fullscreen, mobile, WP-Admin)
Problem Analysis:
1. Sticky header had no gap with first card
2. Sticky header not staying sticky when scrolling in WP-Admin
Root Cause:
The sticky header is inside a scrollable container:
<main className="flex-1 p-4 overflow-auto">
<SettingsLayout>
<div className="sticky top-[49px]"> ← Wrong!
When sticky is inside a scrollable container, it sticks relative
to that container, not the viewport. The top offset should be
relative to the scrollable container's top, not the viewport.
Solution:
1. Changed sticky position from top-[49px] to top-0
- Sticky is relative to scrollable parent (<main>)
- top-0 means stick to top of scrollable area
2. Added mb-6 for gap between header and content
- Prevents header from touching first card
- Maintains consistent spacing
Before:
<div className="sticky top-[49px] ...">
↑ Trying to offset from viewport (wrong context)
After:
<div className="sticky top-0 mb-6 ...">
↑ Stick to scrollable container top (correct)
↑ Add margin for gap
Layout Structure:
┌─────────────────────────────────────┐
│ WP Admin Bar (32px) │
├─────────────────────────────────────┤
│ WP Menu (112px) │
├─────────────────────────────────────┤
│ Submenu Bar (49px) - sticky │
├─────────────────────────────────────┤
│ <main overflow-auto> ← Scroll here │
│ ┌─────────────────────────────┐ │
│ │ Sticky Header (top-0) │ │ ← Sticks here
│ ├─────────────────────────────┤ │
│ │ Gap (mb-6) │ │
│ ├─────────────────────────────┤ │
│ │ First Card │ │
│ │ Content... │ │
│ └─────────────────────────────┘ │
└─────────────────────────────────────┘
Result:
✅ Sticky header stays at top when scrolling
✅ Gap between header and content (mb-6)
✅ Works in both fullscreen and WP-Admin modes
✅ Edge-to-edge background maintained
Files Modified:
- SettingsLayout.tsx: Simplified sticky positioning
Problem:
POST /payments/gateways/order → 404 'gateway_not_found'
Root Cause:
WordPress REST API matches routes in registration order.
The /gateways/order route was registered AFTER /gateways/{id}.
So /gateways/order was being matched by /gateways/{id} where id='order'.
Then get_gateway('order') returned 'gateway_not_found'.
Solution:
Register specific routes BEFORE dynamic routes:
1. /gateways (list)
2. /gateways/order (specific - NEW POSITION)
3. /gateways/{id} (dynamic)
4. /gateways/{id}/toggle (dynamic with action)
Route Priority Rules:
✅ Specific routes first
✅ Dynamic routes last
✅ More specific before less specific
Before:
/gateways → OK
/gateways/{id} → Matches everything including 'order'
/gateways/{id}/toggle → OK (more specific than {id})
/gateways/order → Never reached!
After:
/gateways → OK
/gateways/order → Matches 'order' specifically
/gateways/{id} → Matches other IDs
/gateways/{id}/toggle → OK
Result:
✅ /gateways/order now works correctly
✅ Sorting saves to database
✅ No more 'gateway_not_found' error
Files Modified:
- PaymentsController.php: Moved /order route before /{id} routes
1. Hide Drag Handle on Mobile ✅
Problem: Drag handle looks messy on mobile
Solution: Hide on mobile, show only on desktop
Changes:
- Added 'hidden md:block' to drag handle
- Added 'md:pl-8' to content wrapper
- Mobile: Clean list without drag handle
- Desktop: Drag handle visible for sorting
UX Priority: Better mobile experience > sorting on mobile
2. Persist Sort Order to Database ✅
Backend Implementation:
A. New API Endpoint
POST /woonoow/v1/payments/gateways/order
Body: { category: 'manual'|'online', order: ['id1', 'id2'] }
B. Save to WordPress Options
- woonoow_payment_gateway_order_manual
- woonoow_payment_gateway_order_online
C. Load Order on Page Load
GET /payments/gateways returns:
{
gateways: [...],
order: {
manual: ['bacs', 'cheque', 'cod'],
online: ['paypal', 'stripe']
}
}
Frontend Implementation:
A. Save on Drag End
- Calls API immediately after reorder
- Shows success toast
- Reverts on error with error toast
B. Load Saved Order
- Extracts order from API response
- Uses saved order if available
- Falls back to gateway order if no saved order
C. Error Handling
- Try/catch on save
- Revert order on failure
- User feedback via toast
3. Flow Diagram
Page Load:
┌─────────────────────────────────────┐
│ GET /payments/gateways │
├─────────────────────────────────────┤
│ Returns: { gateways, order } │
│ - order.manual: ['bacs', 'cod'] │
│ - order.online: ['paypal'] │
└─────────────────────────────────────┘
↓
┌─────────────────────────────────────┐
│ Initialize State │
│ - setManualOrder(order.manual) │
│ - setOnlineOrder(order.online) │
└─────────────────────────────────────┘
↓
┌─────────────────────────────────────┐
│ Display Sorted List │
│ - useMemo sorts by saved order │
└─────────────────────────────────────┘
User Drags:
┌─────────────────────────────────────┐
│ User drags item │
└─────────────────────────────────────┘
↓
┌─────────────────────────────────────┐
│ handleDragEnd │
│ - Calculate new order │
│ - Update state (optimistic) │
└─────────────────────────────────────┘
↓
┌─────────────────────────────────────┐
│ POST /payments/gateways/order │
│ Body: { category, order } │
└─────────────────────────────────────┘
↓
┌─────────────────────────────────────┐
│ Success: Toast notification │
│ Error: Revert + error toast │
└─────────────────────────────────────┘
4. Mobile vs Desktop
Mobile (< 768px):
✅ Clean list without drag handle
✅ No left padding
✅ Better UX
❌ No sorting (desktop only)
Desktop (≥ 768px):
✅ Drag handle visible
✅ Full sorting capability
✅ Visual feedback
✅ Keyboard accessible
Benefits:
✅ Order persists across sessions
✅ Order persists across page reloads
✅ Clean mobile UI
✅ Full desktop functionality
✅ Error handling with rollback
✅ Optimistic UI updates
Files Modified:
- PaymentsController.php: New endpoint + load order
- Payments.tsx: Save order + load order + mobile hide
- Database: 2 new options for order storage
Implemented sortable payment gateways using @dnd-kit
Features:
✅ Drag-and-drop for Manual Payment Methods
✅ Drag-and-drop for Online Payment Methods
✅ Visual drag handle (GripVertical icon)
✅ Smooth animations during drag
✅ Separate sorting for each category
✅ Order persists in component state
✅ Toast notification on reorder
UI Changes:
- Added drag handle on left side of each gateway card
- Cursor changes to grab/grabbing during drag
- Dragged item becomes semi-transparent (50% opacity)
- Smooth transitions between positions
Implementation:
1. DnD Context Setup
- PointerSensor for mouse/touch
- KeyboardSensor for accessibility
- closestCenter collision detection
2. Sortable Items
- SortableGatewayItem wrapper component
- Handles drag attributes and listeners
- Applies transform and transition styles
3. State Management
- manualOrder: Array of manual gateway IDs
- onlineOrder: Array of online gateway IDs
- Initialized from gateways on mount
- Updated on drag end
4. Sorting Logic
- useMemo to sort gateways by custom order
- arrayMove from @dnd-kit/sortable
- Separate handlers for each category
5. Visual Feedback
- GripVertical icon (left side, 8px from edge)
- Opacity 0.5 when dragging
- Smooth CSS transitions
- Cursor: grab/grabbing
TODO (Backend):
- Save order to WordPress options
- Load order on page load
- API endpoint: POST /payments/gateways/order
Benefits:
✅ Better UX for organizing payment methods
✅ Visual feedback during drag
✅ Accessible (keyboard support)
✅ Separate sorting per category
✅ No page reload needed
Files Modified:
- Payments.tsx: DnD implementation
- package.json: @dnd-kit dependencies (already installed)
Problem: Bank account cards too large, takes up too much space
Solution: Compact list view with expand/collapse functionality
UI Changes:
1. Compact View (Default)
Display: {BankName}: {AccountNumber} - {AccountName}
Example: "Bank BCA: 1234567890 - Dwindi Ramadhana"
Actions: Edit icon, Delete icon
Hover: Background highlight
2. Expanded View (On Edit/New)
Shows full form with all 6 fields
Collapse button to return to compact view
Remove Account button at bottom
Features:
✅ Click anywhere on row to expand
✅ Edit icon for explicit edit action
✅ Delete icon in compact view (quick delete)
✅ Auto-expand when adding new account
✅ Collapse button in expanded view
✅ Smooth transitions
✅ Space-efficient design
Benefits:
- 70% less vertical space
- Quick overview of all accounts
- Easy to scan multiple accounts
- Edit only when needed
- Better UX for managing many accounts
Icons Added:
- Edit2: Edit button
- ChevronUp: Collapse button
- ChevronDown: (reserved for future use)
Before: Each account = large card (200px height)
After: Each account = compact row (48px height)
Expands to form when editing
1. Added Support for More Field Types ✅
New field types:
- 'title': Heading/separator (renders as h3 with border)
- 'multiselect': Multiple select dropdown
- 'account': Bank account repeater (BACS)
Total supported: text, password, checkbox, select, textarea,
number, email, url, account, title, multiselect
2. Improved Account Field Handling ✅
Problem: WooCommerce might return serialized PHP or JSON string
Solution: Parse string values before rendering
Handles:
- JSON string: JSON.parse()
- Array: Use directly
- Empty/invalid: Default to []
This ensures bank accounts display correctly even if
backend returns different formats.
3. Added Title Field Support ✅
Renders as section heading:
┌─────────────────────────────┐
│ Account Details │ ← Title
│ Configure your bank... │ ← Description
├─────────────────────────────┤
│ [Account fields below] │
└─────────────────────────────┘
4. Installed DnD Kit for Sorting ✅
Packages installed:
- @dnd-kit/core
- @dnd-kit/sortable
- @dnd-kit/utilities
Prepared components:
- SortableGatewayItem wrapper
- Drag handle with GripVertical icon
- DnD sensors and context
Next: Wire up sorting logic and save order
Why This Matters:
✅ Bank account repeater will now work for BACS
✅ Supports all common WooCommerce field types
✅ Handles different data formats from backend
✅ Better organized settings with title separators
✅ Ready for drag-and-drop sorting
Files Modified:
- GenericGatewayForm.tsx: New field types + parsing
- Payments.tsx: DnD imports + sortable component
- package.json: DnD kit dependencies
1. Added Emoji Flags to Country/Region Select ✅
Before: Indonesia
After: 🇮🇩 Indonesia
Implementation:
- Uses same countryCodeToEmoji() helper
- Flags for all countries in dropdown
- Better visual identification
2. Implemented Bank Account Repeater Field ✅
New field type: 'account'
- Add/remove multiple bank accounts
- Each account has 6 fields:
* Account Name (required)
* Account Number (required)
* Bank Name (required)
* Sort Code / Branch Code (optional)
* IBAN (optional)
* BIC / SWIFT (optional)
UI Features:
✅ Compact card layout with muted background
✅ 2-column grid on desktop, 1-column on mobile
✅ Delete button per account (trash icon)
✅ Add button at bottom with plus icon
✅ Account numbering (Account 1, Account 2, etc.)
✅ Smaller inputs (h-9) for compact layout
✅ Clear labels with required indicators
Perfect for:
- Direct Bank Transfer (BACS)
- Manual payment methods
- Multiple bank account management
3. Updated GenericGatewayForm ✅
Added support:
- New 'account' field type
- BankAccount interface
- Repeater logic (add/remove/update)
- Plus and Trash2 icons from lucide-react
Data structure:
interface BankAccount {
account_name: string;
account_number: string;
bank_name: string;
sort_code?: string;
iban?: string;
bic?: string;
}
Benefits:
✅ Country select now has visual flags
✅ Bank accounts are easy to manage
✅ Compact, responsive UI
✅ Clear visual hierarchy
✅ Supports international formats (IBAN, BIC, Sort Code)
Files Modified:
- Store.tsx: Added flags to country select
- GenericGatewayForm.tsx: Bank account repeater
- SubmenuBar.tsx: Fullscreen prop (user change)
1. Made Settings Submenu Sticky ✅
Problem: Settings submenu wasn't sticky like Dashboard
Solution: Added sticky positioning to SubmenuBar
Added classes:
- sticky top-0 z-20
- bg-background/95 backdrop-blur
- supports-[backdrop-filter]:bg-background/60
Result: ✅ Settings submenu now stays at top when scrolling
2. Switched to Emoji Flags ✅
Problem: Base64 images not showing in select options
Better Solution: Use native emoji flags
Benefits:
- ✅ No image loading required
- ✅ Native OS rendering
- ✅ Smaller bundle size
- ✅ Better performance
- ✅ Always works (no broken images)
Implementation:
function countryCodeToEmoji(countryCode: string): string {
const codePoints = countryCode
.toUpperCase()
.split('')
.map(char => 127397 + char.charCodeAt(0));
return String.fromCodePoint(...codePoints);
}
// AE → 🇦🇪
// US → 🇺🇸
// ID → 🇮🇩
3. Updated Currency Select ✅
Before: [Image] United Arab Emirates dirham (AED)
After: 🇦🇪 United Arab Emirates dirham (AED)
- Emoji flag in label
- No separate icon prop needed
- Works immediately
4. Updated Store Summary ✅
Before: [Image] Your store is located in Indonesia
After: 🇮🇩 Your store is located in Indonesia
- Dynamic emoji flag based on currency
- Cleaner implementation
- No image loading
5. Simplified SearchableSelect ✅
- Removed icon prop (not needed with emoji)
- Removed image rendering code
- Simpler component API
Files Modified:
- SubmenuBar.tsx: Added sticky positioning
- Store.tsx: Emoji flags + helper function
- searchable-select.tsx: Removed icon support
Why Emoji > Images:
✅ Universal support (all modern browsers/OS)
✅ No loading time
✅ No broken images
✅ Smaller code
✅ Native rendering
✅ Accessibility friendly
1. Fixed Submenu Active State ✅
Problem: First submenu always active due to pathname.startsWith()
- /dashboard matches /dashboard/analytics
- Both items show as active
Solution: Use exact match instead
- const isActive = pathname === it.path
- Only clicked item shows as active
Files: DashboardSubmenuBar.tsx, SubmenuBar.tsx
2. Fixed Currency Symbol Display ✅
Problem: HTML entities showing (ءإ)
Solution: Use currency code when symbol has HTML entities
Before: United Arab Emirates dirham (ءإ)
After: United Arab Emirates dirham (AED)
Logic:
const displaySymbol = (!currency.symbol || currency.symbol.includes('&#'))
? currency.code
: currency.symbol;
3. Integrated Flags.json ✅
A. Moved flags.json to admin-spa/src/data/
B. Added flag support to SearchableSelect component
- New icon prop in Option interface
- Displays flag before label in trigger
- Displays flag before label in dropdown
C. Currency select now shows flags
- Flag icon next to each currency
- Visual country identification
- Better UX for currency selection
D. Dynamic store summary with flag
Before: 🇮🇩 Your store is located in Indonesia
After: [FLAG] Your store is located in Indonesia
- Flag based on selected currency
- Country name from flags.json
- Currency name (not just code)
- Dynamic updates when currency changes
Benefits:
✅ Clear submenu navigation
✅ Readable currency symbols
✅ Visual country flags
✅ Better currency selection UX
✅ Dynamic store location display
Files Modified:
- DashboardSubmenuBar.tsx: Exact match for active state
- SubmenuBar.tsx: Exact match for active state
- Store.tsx: Currency symbol fix + flags integration
- searchable-select.tsx: Icon support
- flags.json: Moved to admin-spa/src/data/
Problem: Payment gateway settings modal was using Dialog on all screen sizes
Solution: Split into responsive Dialog (desktop) and Drawer (mobile)
Changes:
1. Added Drawer and useMediaQuery imports
2. Added isDesktop hook: useMediaQuery("(min-width: 768px)")
3. Split modal into two conditional renders:
- Desktop (≥768px): Dialog with horizontal footer layout
- Mobile (<768px): Drawer with vertical footer layout
Desktop Layout (Dialog):
- Center modal overlay
- Horizontal footer: Cancel | View in WC | Save
- max-h-[80vh] for scrolling
Mobile Layout (Drawer):
- Bottom sheet (slides up from bottom)
- Vertical footer (full width buttons):
1. Save Settings (primary)
2. View in WooCommerce (ghost)
3. Cancel (outline)
- max-h-[90vh] for more screen space
- Swipe down to dismiss
Benefits:
✅ Native mobile experience with bottom sheet
✅ Easier to reach buttons on mobile (bottom of screen)
✅ Better one-handed use
✅ Swipe gesture to dismiss
✅ Desktop keeps familiar modal experience
User Changes Applied:
- AlertDialog z-index: z-50 → z-[999] (higher than other modals)
- Dialog max-height: max-h-[100vh] → max-h-[80vh] (better desktop UX)
Files Modified:
- Payments.tsx: Responsive Dialog/Drawer implementation
- alert-dialog.tsx: Increased z-index for proper layering
1. Reverted Accordion Grouping ✅
Problem: Payment titles are editable by users
- User renames "BNI Virtual Account" to "BNI VA 2"
- Grouping breaks - gateway moves to new accordion
- Confusing UX when titles change
Solution: Back to flat list
- All payment methods in one list
- Titles can be edited without breaking layout
- Simpler, more predictable behavior
2. Added AlertDialog Component ✅
Installed: @radix-ui/react-alert-dialog
Created: alert-dialog.tsx (shadcn pattern)
Use for confirmations:
- "Are you sure you want to delete?"
- "Discard unsaved changes?"
- "Disable payment method?"
Example:
<AlertDialog>
<AlertDialogTrigger>Delete</AlertDialogTrigger>
<AlertDialogContent>
<AlertDialogHeader>
<AlertDialogTitle>Are you sure?</AlertDialogTitle>
<AlertDialogDescription>
This action cannot be undone.
</AlertDialogDescription>
</AlertDialogHeader>
<AlertDialogFooter>
<AlertDialogCancel>Cancel</AlertDialogCancel>
<AlertDialogAction>Delete</AlertDialogAction>
</AlertDialogFooter>
</AlertDialogContent>
</AlertDialog>
Shadcn Dialog Components:
✅ Dialog - Forms, settings (@radix-ui/react-dialog)
✅ Drawer - Mobile bottom sheet (vaul)
✅ AlertDialog - Confirmations (@radix-ui/react-alert-dialog)
All three are official shadcn components!
Created responsive dialog pattern for better mobile UX:
Components Added:
1. drawer.tsx - Vaul-based drawer component (bottom sheet)
2. responsive-dialog.tsx - Smart wrapper that switches based on screen size
3. use-media-query.ts - Hook to detect screen size
Pattern:
- Desktop (≥768px): Use Dialog (modal overlay)
- Mobile (<768px): Use Drawer (bottom sheet)
- Provides consistent API for both
Usage Example:
<ResponsiveDialog
open={isOpen}
onOpenChange={setIsOpen}
title="Settings"
description="Configure your options"
footer={<Button>Save</Button>}
>
<FormContent />
</ResponsiveDialog>
Benefits:
- Better mobile UX with native-feeling bottom sheet
- Easier to reach buttons on mobile
- Consistent desktop experience
- Single component API
Dependencies:
- npm install vaul (drawer library)
- @radix-ui/react-dialog (already installed)
Next Steps:
- Convert payment gateway modal to use ResponsiveDialog
- Use AlertDialog for confirmations
- Apply pattern to other modals in project
Note: Payment gateway modal needs custom implementation
due to complex layout (scrollable body + sticky footer)
1. Remove Enable/Disable Checkbox ✅
- Already controlled by toggle in main UI
- Skip rendering 'enabled' field in GenericGatewayForm
- Cleaner form, less redundancy
2. Use Field Default as Default Value ✅
- Already working: field.value ?? field.default
- Backend sends current value, falls back to default
- No changes needed
3. Group Online Payments by Provider ✅
- Installed @radix-ui/react-accordion
- Created accordion.tsx component
- Group by gateway.title (provider name)
- Show provider with method count
- Expand to see individual methods
Structure:
TriPay (3 payment methods)
├─ BNI Virtual Account
├─ Mandiri Virtual Account
└─ BCA Virtual Account
PayPal (1 payment method)
└─ PayPal
Benefits:
- Cleaner UI with less clutter
- Easy to find specific provider
- Shows method count at a glance
- Multiple providers can be expanded
- Better organization for many gateways
Files Modified:
- GenericGatewayForm.tsx: Skip enabled field
- Payments.tsx: Accordion grouping by provider
- accordion.tsx: New component (shadcn pattern)
Next: Dialog/Drawer responsive pattern
Mobile Improvements:
1. Modal footer buttons now stack vertically on mobile
- Order: Save Settings (primary) -> View in WooCommerce -> Cancel
- Full width buttons on mobile for easier tapping
- Responsive padding: px-4 on mobile, px-6 on desktop
2. Refresh button moved inline with title
- Added action prop to SettingsLayout
- Refresh button now appears next to Payments title
- Cleaner, more compact layout
Payment Categories Simplified:
3. Removed Payment Providers section
- PayPal, Stripe are also 3rd party, not different
- Confusing to separate providers from other gateways
- All non-manual gateways now in single category
4. Renamed to Online Payment Methods
- Was: Manual + Payment Providers + 3rd Party
- Now: Manual + Online Payment Methods
- Clearer distinction: offline vs online payments
5. Unified styling for all online gateways
- Same card style as manual methods
- Status badges (Enabled/Disabled)
- Requirements alerts
- Manage button always visible
Mobile UX:
- Footer buttons: flex-col on mobile, flex-row on desktop
- Proper button ordering with CSS order utilities
- Responsive spacing and padding
- Touch-friendly button sizes
Files Modified:
- Payments.tsx: Mobile footer + simplified categories
- SettingsLayout.tsx: Added action prop for header actions
Result:
✅ Better mobile experience
✅ Clearer payment method organization
✅ Consistent styling across all gateways
✅ Issue 1: Modal Not Showing Current Values (FIXED!)
Problem: Opening modal showed defaults, not current saved values
Root Cause: Backend only sent field.default, not current value
Solution:
- Backend: Added field.value with current saved value
- normalize_field() now includes: value: $current_settings[$key]
- Frontend: Use field.value ?? field.default for initial data
- GenericGatewayForm initializes with current values
Result: ✅ Modal now shows "BNI Virtual Account 2" not "BNI Virtual Account"
✅ Issue 2: Sticky Modal Footer (FIXED!)
Problem: Footer scrolls away with long forms
Solution:
- Restructured modal: header + scrollable body + sticky footer
- DialogContent: flex flex-col with overflow on body only
- Footer: sticky bottom-0 with border-t
- Save button triggers form.requestSubmit()
Result: ✅ Cancel, View in WooCommerce, Save always visible
✅ Issue 3: HTML in Descriptions (FIXED!)
Problem: TriPay icon shows as raw HTML string
Solution:
- Changed: {field.description}
- To: dangerouslySetInnerHTML={{ __html: field.description }}
- Respects vendor creativity (images, formatting, links)
Result: ✅ TriPay icon image renders properly
📋 Technical Details:
Backend Changes (PaymentGatewaysProvider.php):
- get_gateway_settings() passes $current_settings to extractors
- normalize_field() adds 'value' => $current_settings[$key]
- All fields now have both default and current value
Frontend Changes:
- GatewayField interface: Added value?: string | boolean
- GenericGatewayForm: Initialize with field.value
- Modal structure: Header + Body (scroll) + Footer (sticky)
- Descriptions: Render as HTML with dangerouslySetInnerHTML
Files Modified:
- PaymentGatewaysProvider.php: Add current values to fields
- Payments.tsx: Restructure modal layout + add value to interface
- GenericGatewayForm.tsx: Use field.value + sticky footer + HTML descriptions
🎯 Result:
✅ Modal shows current saved values
✅ Footer always visible (no scrolling)
✅ Vendor HTML/images render properly
🔴 Issue 1: Toggle Loading State (CRITICAL FIX)
Problem: Optimistic update lies - toggle appears to work but fails
Solution:
- Removed ALL optimistic updates
- Added loading state tracking (togglingGateway)
- Disabled toggle during mutation
- Show real server state only
- User sees loading, not lies
Result: ✅ Honest UI - shows loading, then real state
🔴 Issue 2: 30s Save Time (CRITICAL FIX)
Problem: Saving gateway settings takes 30 seconds
Root Cause: WooCommerce analytics/tracking HTTP requests
Solution:
- Block HTTP during save: add_filter('pre_http_request', '__return_true', 999)
- Save settings (fast)
- Re-enable HTTP: remove_filter()
- Same fix as orders module
Result: ✅ Save now takes 1-2 seconds instead of 30s
🟡 Issue 3: Inconsistent Input Styling (FIXED)
Problem: email/tel inputs look different (browser defaults)
Solution:
- Added appearance-none to Input component
- Override -webkit-appearance
- Override -moz-appearance (for number inputs)
- Consistent styling for ALL input types
Result: ✅ All inputs look identical regardless of type
📋 Technical Details:
Toggle Flow (No More Lies):
User clicks → Disable toggle → Show loading → API call → Success → Refetch → Enable toggle
Save Flow (Fast):
Block HTTP → Save to DB → Unblock HTTP → Return (1-2s)
Input Styling:
text, email, tel, number, url, password → All identical appearance
Files Modified:
- Payments.tsx: Removed optimistic, added loading state
- PaymentGatewaysProvider.php: Block HTTP during save
- input.tsx: Override browser default styles
🎯 Result:
✅ No more lying optimistic updates
✅ 30s → 1-2s save time
✅ Consistent input styling
✅ Issue 1: Toggle Not Saving (CRITICAL FIX)
Problem: Toggle appeared to work but didn't persist
Root Cause: Missing query invalidation after toggle
Solution:
- Added queryClient.invalidateQueries after successful toggle
- Now fetches real server state after optimistic update
- Ensures SPA and WooCommerce stay in sync
✅ Issue 2: SearchableSelect Default Value
Problem: Showing 'Select country...' when Indonesia selected
Root Cause: WooCommerce stores country as 'ID:DKI_JAKARTA'
Solution:
- Split country:state format in backend
- Extract country code only for select
- Added timezone fallback to 'UTC' if empty
✅ Issue 3: 3rd Party Gateway Settings
Problem: TriPay showing 'Configure in WooCommerce' link
Solution:
- Replaced external link with Settings button
- Now opens GenericGatewayForm modal
- All WC form_fields render automatically
- TriPay fields (enable_icon, expired, checkout_method) work!
📋 Files Modified:
- Payments.tsx: Added invalidation + settings button
- StoreSettingsProvider.php: Split country format
- All 3rd party gateways now configurable in SPA
🎯 Result:
✅ Toggle saves correctly to WooCommerce
✅ Country/timezone show selected values
✅ All gateways with form_fields are editable
✅ No more 'Configure in WooCommerce' for compliant gateways
✅ Issue 1: Modal Z-Index Fixed
- Increased dialog z-index: z-[9999] → z-[99999]
- Now properly appears above fullscreen mode (z-50)
✅ Issue 2: Searchable Select for Large Lists
- Replaced Select with SearchableSelect for:
- Countries (200+ options)
- Currencies (100+ options)
- Timezones (400+ options)
- Users can now type to search instead of scrolling
- Better UX for large datasets
✅ Issue 3: Input Type Support
- Input component already supports type attribute
- No changes needed (already working)
✅ Issue 4: Timezone Options Fixed
- Replaced optgroup (not supported) with flat list
- SearchableSelect handles filtering by continent name
- Shows: 'Asia/Jakarta (UTC+7:00)'
- Search includes continent, city, and offset
📊 Result:
- ✅ Modal always on top
- ✅ Easy search for countries/currencies/timezones
- ✅ No more scrolling through hundreds of options
- ✅ Better accessibility
Addresses user feedback issues 1-4
✅ Store.tsx - Complete API Integration:
- Replaced mock data with real API calls
- useQuery for fetching settings, countries, timezones, currencies
- useMutation for saving settings
- Optimistic updates and error handling
✅ Real Data Sources:
- Countries: 200+ countries from WooCommerce (WC_Countries)
- Timezones: 400+ timezones from PHP with UTC offsets
- Currencies: 100+ currencies with symbols
- Settings: All WooCommerce store options
✅ UI Improvements:
- Country select: Full list instead of 5 hardcoded
- Timezone select: Grouped by continent with UTC offsets
- Currency select: Full list with symbols
- Already using shadcn components (Input, Select)
✅ Performance:
- 1 hour cache for static data (countries, timezones, currencies)
- 1 minute cache for settings
- Proper loading states
📋 Addresses user feedback:
- ✅ Wire real options for country and timezone
- ✅ Contact fields already use shadcn components
Next: Create custom BACS form with bank account repeater
✅ Generic form builder for payment gateways:
Features:
- Supports 8 field types: text, password, checkbox, select, textarea, number, email, url
- Auto-categorizes fields: Basic, API, Advanced
- Multi-page tabs for 20+ fields
- Single page for < 20 fields
- Unsupported field warning with link to WC settings
- Field validation (required, placeholder, etc.)
- Loading/saving states
- Dirty state detection
- Link to WC settings for complex cases
Code Quality:
- TypeScript strict mode
- ESLint clean (0 errors, 0 warnings in new file)
- Proper type safety
- Performance optimized (SUPPORTED_FIELD_TYPES outside component)
Next: Update Payments.tsx to use real API
- Installed @radix-ui/react-switch
- Created switch.tsx following existing UI component patterns
- Fixes import error in ToggleField component
- Dev server now running successfully