Commit Graph

2 Commits

Author SHA1 Message Date
dwindown
824266044d fix: CRITICAL - Memoize all context values to stop infinite loops
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!
2025-11-06 21:27:44 +07:00
dwindown
4d2469f826 feat: Add scroll-hide header and contextual FAB system
Implemented:

1. Scroll-Hide App Bar (Mobile)
   - Hides on scroll down (past 50px)
   - Shows on scroll up
   - Chrome URL bar behavior
   - Smooth slide animation (300ms)
   - Desktop always visible (md:translate-y-0)

2. Contextual FAB Hook
   - useFABConfig() hook for pages
   - Pre-configured for: orders, products, customers, coupons, dashboard
   - Automatic cleanup on unmount
   - Easy to use: useFABConfig('orders')

3. Removed Focus Styles
   - Bottom nav links: focus:outline-none
   - Cleaner mobile UX

Header Scroll Behavior:
- Scroll down > 50px: Header slides up (-translate-y-full)
- Scroll up: Header slides down (translate-y-0)
- Desktop: Always visible (md:translate-y-0)
- Smooth transition (duration-300)

FAB Configuration:
const configs = {
  orders: 'Create Order' → /orders/new
  products: 'Add Product' → /products/new
  customers: 'Add Customer' → /customers/new
  coupons: 'Create Coupon' → /coupons/new
  dashboard: 'Quick Actions' → (future speed dial)
  none: Hide FAB
}

Usage in Pages:
import { useFABConfig } from '@/hooks/useFABConfig';

function OrdersPage() {
  useFABConfig('orders'); // Sets up FAB automatically
  return <div>...</div>;
}

Next Steps:
- Add useFABConfig to actual pages
- Test scroll behavior on devices
- Implement speed dial for dashboard

Files Created:
- useFABConfig.tsx: Contextual FAB configuration hook

Files Modified:
- App.tsx: Scroll detection and header animation
- BottomNav.tsx: Removed focus outline styles
2025-11-06 20:27:19 +07:00