Compare commits
13 Commits
refactor/r
...
feature/on
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
6f9081cb52 | ||
|
|
7f35dc57c6 | ||
|
|
7d63f4fbcd | ||
|
|
b78b081d29 | ||
|
|
676dda4642 | ||
|
|
c8d184f9dc | ||
|
|
322c4c805b | ||
|
|
bcf74e1fd1 | ||
|
|
7ca756fc5a | ||
|
|
ca363a2406 | ||
|
|
5ec0e6c92c | ||
|
|
ee574cb736 | ||
|
|
f66f9b87ab |
367
ONE-PAGE-CHECKOUT-PLAN.md
Normal file
367
ONE-PAGE-CHECKOUT-PLAN.md
Normal file
@@ -0,0 +1,367 @@
|
|||||||
|
# One-Page Checkout Implementation Plan
|
||||||
|
|
||||||
|
**Branch:** `feature/one-page-checkout`
|
||||||
|
**Status:** In Development
|
||||||
|
**Priority:** High
|
||||||
|
**Phone Requirement:** Required (not optional)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Convert the current two-phase checkout into a streamlined one-page checkout experience where customers can see all fields at once and complete their order in a single action.
|
||||||
|
|
||||||
|
### Current State
|
||||||
|
- **Phase 1:** Collect email, shipping address → fetch shipping methods
|
||||||
|
- **Phase 2:** Select shipping method, billing address → complete order
|
||||||
|
- **Total API calls:** 6-7 sequential requests across 2 user interactions
|
||||||
|
|
||||||
|
### Target State
|
||||||
|
- **Single Page:** All fields visible simultaneously
|
||||||
|
- **Dynamic updates:** Shipping methods fetch automatically when address changes
|
||||||
|
- **Single submit:** One "Complete Order" button
|
||||||
|
- **Optimized API:** 3-4 sequential steps (parallel where possible)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Requirements
|
||||||
|
|
||||||
|
### Must-Have
|
||||||
|
- [ ] All checkout fields visible on single page
|
||||||
|
- [ ] Phone number is **required** (strict validation)
|
||||||
|
- [ ] Shipping methods fetch automatically (debounced) when address changes
|
||||||
|
- [ ] Real-time total calculation (updates when shipping method selected)
|
||||||
|
- [ ] Single "Complete Order" submit button
|
||||||
|
- [ ] Section-based validation with inline errors
|
||||||
|
- [ ] Auto-scroll to first error on validation failure
|
||||||
|
- [ ] Preserve form data on error
|
||||||
|
|
||||||
|
### UX Requirements
|
||||||
|
- [ ] Clear visual hierarchy (Contact → Shipping → Billing → Shipping Method → Payment)
|
||||||
|
- [ ] Collapsible sections (optional - all expanded by default)
|
||||||
|
- [ ] Loading states for shipping method fetching
|
||||||
|
- [ ] Disabled submit button until all required fields valid
|
||||||
|
- [ ] Success confirmation page (existing)
|
||||||
|
|
||||||
|
### Technical Requirements
|
||||||
|
- [ ] Debounced shipping method API calls (500ms)
|
||||||
|
- [ ] Optimistic UI updates where possible
|
||||||
|
- [ ] Proper error handling per section
|
||||||
|
- [ ] Analytics events for checkout steps
|
||||||
|
- [ ] Mobile-responsive layout
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## UI Layout
|
||||||
|
|
||||||
|
### Left Column (Form - 60% width on desktop)
|
||||||
|
|
||||||
|
```
|
||||||
|
┌─────────────────────────────────────┐
|
||||||
|
│ 1. Contact Information │
|
||||||
|
│ ├─ Email * [________________] │
|
||||||
|
│ └─ Phone * [________________] │
|
||||||
|
│ [+381... format hint] │
|
||||||
|
├─────────────────────────────────────┤
|
||||||
|
│ 2. Shipping Address │
|
||||||
|
│ ├─ First Name * [____________] │
|
||||||
|
│ ├─ Last Name * [_____________] │
|
||||||
|
│ ├─ Country * [▼ Serbia ▼] │
|
||||||
|
│ ├─ Street Address * [________] │
|
||||||
|
│ ├─ Apt/Suite [______________] │
|
||||||
|
│ ├─ City * [_________________] │
|
||||||
|
│ └─ Postal Code * [__________] │
|
||||||
|
├─────────────────────────────────────┤
|
||||||
|
│ 3. Billing Address │
|
||||||
|
│ [✓] Same as shipping address │
|
||||||
|
│ (Fields hidden when checked) │
|
||||||
|
├─────────────────────────────────────┤
|
||||||
|
│ 4. Shipping Method │
|
||||||
|
│ (Loading... / Select to see │
|
||||||
|
│ available options) │
|
||||||
|
│ ○ Standard (2-3 days) 400 RSD │
|
||||||
|
│ ○ Express (1-2 days) 800 RSD │
|
||||||
|
├─────────────────────────────────────┤
|
||||||
|
│ 5. Payment Method │
|
||||||
|
│ ● Cash on Delivery │
|
||||||
|
│ (Additional payment methods TBD) │
|
||||||
|
├─────────────────────────────────────┤
|
||||||
|
│ [ Complete Order - 3,600 RSD ] │
|
||||||
|
│ Loading spinner when processing │
|
||||||
|
└─────────────────────────────────────┘
|
||||||
|
```
|
||||||
|
|
||||||
|
### Right Column (Order Summary - 40% width on desktop)
|
||||||
|
|
||||||
|
```
|
||||||
|
┌─────────────────────────────────────┐
|
||||||
|
│ Order Summary │
|
||||||
|
├─────────────────────────────────────┤
|
||||||
|
│ Product Image Serum x1 3,200 │
|
||||||
|
│ RSD │
|
||||||
|
├─────────────────────────────────────┤
|
||||||
|
│ Subtotal 3,200 RSD │
|
||||||
|
│ Shipping 400 RSD │
|
||||||
|
│ ───────────────────────────────── │
|
||||||
|
│ Total 3,600 RSD │
|
||||||
|
└─────────────────────────────────────┘
|
||||||
|
```
|
||||||
|
|
||||||
|
### Mobile Layout
|
||||||
|
Single column, stacked sections with sticky order summary at bottom.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Technical Implementation
|
||||||
|
|
||||||
|
### State Management
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
// Form state (existing)
|
||||||
|
const [shippingAddress, setShippingAddress] = useState<AddressForm>({...});
|
||||||
|
const [billingAddress, setBillingAddress] = useState<AddressForm>({...});
|
||||||
|
const [sameAsShipping, setSameAsShipping] = useState(true);
|
||||||
|
|
||||||
|
// New state
|
||||||
|
const [paymentMethod, setPaymentMethod] = useState<string>("cod");
|
||||||
|
const [errors, setErrors] = useState<ValidationErrors>({
|
||||||
|
contact: null,
|
||||||
|
shipping: null,
|
||||||
|
billing: null,
|
||||||
|
shippingMethod: null,
|
||||||
|
general: null,
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
### Debounced Shipping Method Fetching
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
useEffect(() => {
|
||||||
|
if (!isAddressComplete(shippingAddress)) return;
|
||||||
|
|
||||||
|
const timer = setTimeout(() => {
|
||||||
|
fetchShippingMethods();
|
||||||
|
}, 500); // 500ms debounce
|
||||||
|
|
||||||
|
return () => clearTimeout(timer);
|
||||||
|
}, [shippingAddress]);
|
||||||
|
```
|
||||||
|
|
||||||
|
### Validation Schema
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
const validationRules = {
|
||||||
|
email: (value) => /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(value),
|
||||||
|
phone: (value) => {
|
||||||
|
// Country-specific validation
|
||||||
|
// Serbia: +381 XX XXX XXXX
|
||||||
|
// Bosnia: +387 XX XXX XXX
|
||||||
|
// etc.
|
||||||
|
},
|
||||||
|
required: (value) => value.trim().length > 0,
|
||||||
|
postalCode: (value, country) => {
|
||||||
|
// Country-specific postal code validation
|
||||||
|
},
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
### API Call Sequence
|
||||||
|
|
||||||
|
**Optimized Flow (parallel + sequential):**
|
||||||
|
|
||||||
|
```
|
||||||
|
Step 1: Validation (client-side)
|
||||||
|
├─ Validate all fields
|
||||||
|
└─ Show inline errors
|
||||||
|
|
||||||
|
Step 2: Parallel Independent Calls
|
||||||
|
├─ Update Email
|
||||||
|
└─ Update Shipping Address
|
||||||
|
(Both can run simultaneously)
|
||||||
|
|
||||||
|
Step 3: Conditional Call
|
||||||
|
└─ Update Billing Address (if different from shipping)
|
||||||
|
|
||||||
|
Step 4: Sequential Dependent Calls
|
||||||
|
├─ Update Shipping Method
|
||||||
|
├─ Update Metadata (phone, language, payment method)
|
||||||
|
└─ Complete Checkout
|
||||||
|
|
||||||
|
Total: 4 sequential steps vs current 7+
|
||||||
|
```
|
||||||
|
|
||||||
|
### Error Handling Strategy
|
||||||
|
|
||||||
|
**Field-level:**
|
||||||
|
- Real-time validation on blur
|
||||||
|
- Visual indicators (red border, error message)
|
||||||
|
- Prevent submit if validation fails
|
||||||
|
|
||||||
|
**Section-level:**
|
||||||
|
- Group errors by section
|
||||||
|
- Show section header in red if has errors
|
||||||
|
- Expand section if collapsed and has errors
|
||||||
|
|
||||||
|
**Form-level:**
|
||||||
|
- On submit: validate all fields
|
||||||
|
- If errors: scroll to first error, show summary
|
||||||
|
- If API error: show in relevant section, preserve data
|
||||||
|
|
||||||
|
**API-level:**
|
||||||
|
- Map Saleor errors to form fields when possible
|
||||||
|
- Generic error: show at top of form
|
||||||
|
- Network error: show retry button
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Files to Modify
|
||||||
|
|
||||||
|
### Primary Files
|
||||||
|
|
||||||
|
1. **`/src/app/[locale]/checkout/page.tsx`**
|
||||||
|
- Major refactor of checkout flow
|
||||||
|
- Combine Phase 1 & Phase 2 into single component
|
||||||
|
- Add debounced shipping method fetching
|
||||||
|
- Implement section-based validation
|
||||||
|
- Optimize API call sequence
|
||||||
|
|
||||||
|
2. **`/src/lib/saleor/mutations/Checkout.ts`**
|
||||||
|
- Ensure all mutations available
|
||||||
|
- Add metadata update mutation if needed
|
||||||
|
|
||||||
|
3. **`/src/lib/saleor/queries/Checkout.ts`**
|
||||||
|
- Ensure checkout query returns shipping methods
|
||||||
|
|
||||||
|
### Translation Files
|
||||||
|
|
||||||
|
4. **`/messages/sr.json`** (and other language files)
|
||||||
|
- Add new translation keys for one-page checkout
|
||||||
|
- Section headers
|
||||||
|
- Validation messages
|
||||||
|
- Button labels
|
||||||
|
|
||||||
|
### Styling
|
||||||
|
|
||||||
|
5. **`/src/app/globals.css`** (or Tailwind config)
|
||||||
|
- Ensure consistent form styling
|
||||||
|
- Add validation state styles
|
||||||
|
- Loading spinner styles
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Implementation Phases
|
||||||
|
|
||||||
|
### Phase 1: Core Structure (Day 1-2)
|
||||||
|
- [ ] Refactor checkout page layout
|
||||||
|
- [ ] Display all sections simultaneously
|
||||||
|
- [ ] Keep existing form logic working
|
||||||
|
- [ ] Test existing flow still works
|
||||||
|
|
||||||
|
### Phase 2: Dynamic Shipping Methods (Day 3)
|
||||||
|
- [ ] Implement debounced fetching
|
||||||
|
- [ ] Add loading states
|
||||||
|
- [ ] Display shipping methods inline
|
||||||
|
- [ ] Update total when method selected
|
||||||
|
|
||||||
|
### Phase 3: Validation & Error Handling (Day 4)
|
||||||
|
- [ ] Implement field-level validation
|
||||||
|
- [ ] Add section-based error display
|
||||||
|
- [ ] Auto-scroll to errors
|
||||||
|
- [ ] Test all validation scenarios
|
||||||
|
|
||||||
|
### Phase 4: Optimization (Day 5)
|
||||||
|
- [ ] Optimize API call sequence
|
||||||
|
- [ ] Add parallel mutation execution
|
||||||
|
- [ ] Improve loading states
|
||||||
|
- [ ] Add optimistic updates
|
||||||
|
|
||||||
|
### Phase 5: Polish (Day 6)
|
||||||
|
- [ ] Mobile responsiveness
|
||||||
|
- [ ] Analytics events
|
||||||
|
- [ ] Accessibility improvements
|
||||||
|
- [ ] Final testing
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Testing Checklist
|
||||||
|
|
||||||
|
### Functionality Tests
|
||||||
|
- [ ] Fill all fields, submit successfully
|
||||||
|
- [ ] Verify order created in Saleor
|
||||||
|
- [ ] Verify emails sent
|
||||||
|
- [ ] Change shipping method, verify total updates
|
||||||
|
- [ ] Change address, verify shipping methods refetch
|
||||||
|
|
||||||
|
### Validation Tests
|
||||||
|
- [ ] Submit with empty email → email error
|
||||||
|
- [ ] Submit with empty phone → phone error
|
||||||
|
- [ ] Submit with invalid email format → format error
|
||||||
|
- [ ] Submit with invalid phone → format error
|
||||||
|
- [ ] Submit with empty required fields → field errors
|
||||||
|
- [ ] Submit without selecting shipping method → shipping error
|
||||||
|
|
||||||
|
### Edge Cases
|
||||||
|
- [ ] Slow network (test debouncing)
|
||||||
|
- [ ] No shipping methods available
|
||||||
|
- [ ] API failure during submission
|
||||||
|
- [ ] Partial API failure (some mutations succeed)
|
||||||
|
- [ ] Browser refresh (preserve data?)
|
||||||
|
|
||||||
|
### Mobile Tests
|
||||||
|
- [ ] Layout works on iPhone SE
|
||||||
|
- [ ] Layout works on iPhone 14 Pro Max
|
||||||
|
- [ ] Touch targets large enough
|
||||||
|
- [ ] Scroll behavior smooth
|
||||||
|
|
||||||
|
### Accessibility Tests
|
||||||
|
- [ ] Tab navigation works
|
||||||
|
- [ ] Screen reader friendly
|
||||||
|
- [ ] Error announcements
|
||||||
|
- [ ] Focus management
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Rollout Strategy
|
||||||
|
|
||||||
|
1. **Development:** Complete on feature branch
|
||||||
|
2. **Testing:** Local testing with all scenarios
|
||||||
|
3. **Staging:** Deploy to dev.manoonoils.com
|
||||||
|
4. **Monitoring:** Check for errors, conversion rates
|
||||||
|
5. **Production:** Merge to master and deploy
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Success Metrics
|
||||||
|
|
||||||
|
- **Conversion Rate:** Should increase (fewer steps = less drop-off)
|
||||||
|
- **Time to Complete:** Should decrease (single page vs two phases)
|
||||||
|
- **Error Rate:** Should decrease (better validation)
|
||||||
|
- **Mobile Completion:** Should improve (optimized for mobile)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Future Enhancements (Out of Scope)
|
||||||
|
|
||||||
|
- [ ] Save addresses for logged-in users
|
||||||
|
- [ ] Address autocomplete (Google Maps)
|
||||||
|
- [ ] Multiple payment methods (Stripe, etc.)
|
||||||
|
- [ ] Guest checkout improvements
|
||||||
|
- [ ] Order notes/comments field
|
||||||
|
- [ ] Gift wrapping options
|
||||||
|
- [ ] Promo code input
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Notes
|
||||||
|
|
||||||
|
- Phone number is **strictly required** - validate format per country
|
||||||
|
- Keep existing checkout success page
|
||||||
|
- Maintain multi-language support
|
||||||
|
- Ensure analytics tracking works
|
||||||
|
- Don't break existing cart functionality
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Created:** March 28, 2026
|
||||||
|
**Branch:** feature/one-page-checkout
|
||||||
|
**Next Step:** Start Phase 1 - Core Structure
|
||||||
@@ -129,11 +129,78 @@ export default function CheckoutPage() {
|
|||||||
|
|
||||||
const [shippingMethods, setShippingMethods] = useState<ShippingMethod[]>([]);
|
const [shippingMethods, setShippingMethods] = useState<ShippingMethod[]>([]);
|
||||||
const [selectedShippingMethod, setSelectedShippingMethod] = useState<string>("");
|
const [selectedShippingMethod, setSelectedShippingMethod] = useState<string>("");
|
||||||
const [showShippingMethods, setShowShippingMethods] = useState(false);
|
const [isLoadingShipping, setIsLoadingShipping] = useState(false);
|
||||||
|
|
||||||
const lines = getLines();
|
const lines = getLines();
|
||||||
const total = getTotal();
|
const total = getTotal();
|
||||||
|
|
||||||
|
// Debounced shipping method fetching
|
||||||
|
useEffect(() => {
|
||||||
|
if (!checkout) return;
|
||||||
|
|
||||||
|
// Check if address is complete enough to fetch shipping methods
|
||||||
|
const isAddressComplete =
|
||||||
|
shippingAddress.firstName &&
|
||||||
|
shippingAddress.lastName &&
|
||||||
|
shippingAddress.streetAddress1 &&
|
||||||
|
shippingAddress.city &&
|
||||||
|
shippingAddress.postalCode &&
|
||||||
|
shippingAddress.country;
|
||||||
|
|
||||||
|
if (!isAddressComplete) {
|
||||||
|
setShippingMethods([]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const timer = setTimeout(async () => {
|
||||||
|
setIsLoadingShipping(true);
|
||||||
|
try {
|
||||||
|
console.log("Fetching shipping methods...");
|
||||||
|
|
||||||
|
// First update the shipping address
|
||||||
|
await saleorClient.mutate<ShippingAddressUpdateResponse>({
|
||||||
|
mutation: CHECKOUT_SHIPPING_ADDRESS_UPDATE,
|
||||||
|
variables: {
|
||||||
|
checkoutId: checkout.id,
|
||||||
|
shippingAddress: {
|
||||||
|
firstName: shippingAddress.firstName,
|
||||||
|
lastName: shippingAddress.lastName,
|
||||||
|
streetAddress1: shippingAddress.streetAddress1,
|
||||||
|
streetAddress2: shippingAddress.streetAddress2,
|
||||||
|
city: shippingAddress.city,
|
||||||
|
postalCode: shippingAddress.postalCode,
|
||||||
|
country: shippingAddress.country,
|
||||||
|
phone: shippingAddress.phone,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// Then query for shipping methods
|
||||||
|
const checkoutQueryResult = await saleorClient.query<CheckoutQueryResponse>({
|
||||||
|
query: GET_CHECKOUT_BY_ID,
|
||||||
|
variables: { id: checkout.id },
|
||||||
|
fetchPolicy: "network-only",
|
||||||
|
});
|
||||||
|
|
||||||
|
const availableMethods = checkoutQueryResult.data?.checkout?.shippingMethods || [];
|
||||||
|
console.log("Available shipping methods:", availableMethods);
|
||||||
|
|
||||||
|
setShippingMethods(availableMethods);
|
||||||
|
|
||||||
|
// Auto-select first method if none selected
|
||||||
|
if (availableMethods.length > 0 && !selectedShippingMethod) {
|
||||||
|
setSelectedShippingMethod(availableMethods[0].id);
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
console.error("Error fetching shipping methods:", err);
|
||||||
|
} finally {
|
||||||
|
setIsLoadingShipping(false);
|
||||||
|
}
|
||||||
|
}, 500); // 500ms debounce
|
||||||
|
|
||||||
|
return () => clearTimeout(timer);
|
||||||
|
}, [checkout, shippingAddress]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!checkout) {
|
if (!checkout) {
|
||||||
refreshCheckout();
|
refreshCheckout();
|
||||||
@@ -189,25 +256,56 @@ export default function CheckoutPage() {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Validate all required fields
|
||||||
if (!shippingAddress.email || !shippingAddress.email.includes("@")) {
|
if (!shippingAddress.email || !shippingAddress.email.includes("@")) {
|
||||||
setError(t("errorEmailRequired"));
|
setError(t("errorEmailRequired"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!shippingAddress.firstName || !shippingAddress.lastName || !shippingAddress.streetAddress1 || !shippingAddress.city || !shippingAddress.postalCode || !shippingAddress.phone) {
|
if (!shippingAddress.phone || shippingAddress.phone.length < 8) {
|
||||||
|
setError(t("errorPhoneRequired"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!shippingAddress.firstName || !shippingAddress.lastName || !shippingAddress.streetAddress1 || !shippingAddress.city || !shippingAddress.postalCode) {
|
||||||
setError(t("errorFieldsRequired"));
|
setError(t("errorFieldsRequired"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!selectedShippingMethod) {
|
||||||
|
setError(t("errorSelectShipping"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
setError(null);
|
setError(null);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// If we're showing shipping methods and one is selected, complete the order
|
console.log("Completing order...");
|
||||||
if (showShippingMethods && selectedShippingMethod) {
|
|
||||||
console.log("Phase 2: Completing order with shipping method...");
|
|
||||||
|
|
||||||
console.log("Step 1: Updating billing address...");
|
console.log("Step 1: Updating email...");
|
||||||
|
const emailResult = await saleorClient.mutate<EmailUpdateResponse>({
|
||||||
|
mutation: CHECKOUT_EMAIL_UPDATE,
|
||||||
|
variables: {
|
||||||
|
checkoutId: checkout.id,
|
||||||
|
email: shippingAddress.email,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (emailResult.data?.checkoutEmailUpdate?.errors && emailResult.data.checkoutEmailUpdate.errors.length > 0) {
|
||||||
|
const errorMessage = emailResult.data.checkoutEmailUpdate.errors[0].message;
|
||||||
|
if (errorMessage.includes("Couldn't resolve to a node")) {
|
||||||
|
console.error("Checkout not found, clearing cart...");
|
||||||
|
localStorage.removeItem('cart');
|
||||||
|
localStorage.removeItem('checkoutId');
|
||||||
|
window.location.href = `/${locale}/products`;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
throw new Error(`Email update failed: ${errorMessage}`);
|
||||||
|
}
|
||||||
|
console.log("Step 1: Email updated successfully");
|
||||||
|
|
||||||
|
console.log("Step 2: Updating billing address...");
|
||||||
const billingResult = await saleorClient.mutate<BillingAddressUpdateResponse>({
|
const billingResult = await saleorClient.mutate<BillingAddressUpdateResponse>({
|
||||||
mutation: CHECKOUT_BILLING_ADDRESS_UPDATE,
|
mutation: CHECKOUT_BILLING_ADDRESS_UPDATE,
|
||||||
variables: {
|
variables: {
|
||||||
@@ -228,9 +326,9 @@ export default function CheckoutPage() {
|
|||||||
if (billingResult.data?.checkoutBillingAddressUpdate?.errors && billingResult.data.checkoutBillingAddressUpdate.errors.length > 0) {
|
if (billingResult.data?.checkoutBillingAddressUpdate?.errors && billingResult.data.checkoutBillingAddressUpdate.errors.length > 0) {
|
||||||
throw new Error(`Billing address update failed: ${billingResult.data.checkoutBillingAddressUpdate.errors[0].message}`);
|
throw new Error(`Billing address update failed: ${billingResult.data.checkoutBillingAddressUpdate.errors[0].message}`);
|
||||||
}
|
}
|
||||||
console.log("Step 1: Billing address updated successfully");
|
console.log("Step 2: Billing address updated successfully");
|
||||||
|
|
||||||
console.log("Step 2: Setting shipping method...");
|
console.log("Step 3: Setting shipping method...");
|
||||||
const shippingMethodResult = await saleorClient.mutate<ShippingMethodUpdateResponse>({
|
const shippingMethodResult = await saleorClient.mutate<ShippingMethodUpdateResponse>({
|
||||||
mutation: CHECKOUT_SHIPPING_METHOD_UPDATE,
|
mutation: CHECKOUT_SHIPPING_METHOD_UPDATE,
|
||||||
variables: {
|
variables: {
|
||||||
@@ -242,9 +340,9 @@ export default function CheckoutPage() {
|
|||||||
if (shippingMethodResult.data?.checkoutShippingMethodUpdate?.errors && shippingMethodResult.data.checkoutShippingMethodUpdate.errors.length > 0) {
|
if (shippingMethodResult.data?.checkoutShippingMethodUpdate?.errors && shippingMethodResult.data.checkoutShippingMethodUpdate.errors.length > 0) {
|
||||||
throw new Error(`Shipping method update failed: ${shippingMethodResult.data.checkoutShippingMethodUpdate.errors[0].message}`);
|
throw new Error(`Shipping method update failed: ${shippingMethodResult.data.checkoutShippingMethodUpdate.errors[0].message}`);
|
||||||
}
|
}
|
||||||
console.log("Step 2: Shipping method set successfully");
|
console.log("Step 3: Shipping method set successfully");
|
||||||
|
|
||||||
console.log("Step 3: Saving metadata...");
|
console.log("Step 4: Saving metadata...");
|
||||||
const metadataResult = await saleorClient.mutate<MetadataUpdateResponse>({
|
const metadataResult = await saleorClient.mutate<MetadataUpdateResponse>({
|
||||||
mutation: CHECKOUT_METADATA_UPDATE,
|
mutation: CHECKOUT_METADATA_UPDATE,
|
||||||
variables: {
|
variables: {
|
||||||
@@ -261,10 +359,10 @@ export default function CheckoutPage() {
|
|||||||
if (metadataResult.data?.updateMetadata?.errors && metadataResult.data.updateMetadata.errors.length > 0) {
|
if (metadataResult.data?.updateMetadata?.errors && metadataResult.data.updateMetadata.errors.length > 0) {
|
||||||
console.warn("Failed to save phone metadata:", metadataResult.data.updateMetadata.errors);
|
console.warn("Failed to save phone metadata:", metadataResult.data.updateMetadata.errors);
|
||||||
} else {
|
} else {
|
||||||
console.log("Step 3: Phone number saved successfully");
|
console.log("Step 4: Phone number saved successfully");
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log("Step 4: Completing checkout...");
|
console.log("Step 5: Completing checkout...");
|
||||||
const completeResult = await saleorClient.mutate<CheckoutCompleteResponse>({
|
const completeResult = await saleorClient.mutate<CheckoutCompleteResponse>({
|
||||||
mutation: CHECKOUT_COMPLETE,
|
mutation: CHECKOUT_COMPLETE,
|
||||||
variables: {
|
variables: {
|
||||||
@@ -304,93 +402,6 @@ export default function CheckoutPage() {
|
|||||||
} else {
|
} else {
|
||||||
throw new Error(t("errorCreatingOrder"));
|
throw new Error(t("errorCreatingOrder"));
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
// Phase 1: Update email and address, then fetch shipping methods
|
|
||||||
console.log("Phase 1: Updating email and address...");
|
|
||||||
|
|
||||||
console.log("Step 1: Updating email...");
|
|
||||||
const emailResult = await saleorClient.mutate<EmailUpdateResponse>({
|
|
||||||
mutation: CHECKOUT_EMAIL_UPDATE,
|
|
||||||
variables: {
|
|
||||||
checkoutId: checkout.id,
|
|
||||||
email: shippingAddress.email,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (emailResult.data?.checkoutEmailUpdate?.errors && emailResult.data.checkoutEmailUpdate.errors.length > 0) {
|
|
||||||
const errorMessage = emailResult.data.checkoutEmailUpdate.errors[0].message;
|
|
||||||
// Check if checkout no longer exists
|
|
||||||
if (errorMessage.includes("Couldn't resolve to a node")) {
|
|
||||||
console.error("Checkout not found, clearing cart...");
|
|
||||||
localStorage.removeItem('cart');
|
|
||||||
localStorage.removeItem('checkoutId');
|
|
||||||
window.location.href = `/${locale}/products`;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
throw new Error(`Email update failed: ${errorMessage}`);
|
|
||||||
}
|
|
||||||
console.log("Step 1: Email updated successfully");
|
|
||||||
|
|
||||||
console.log("Step 2: Updating shipping address...");
|
|
||||||
console.log("Shipping address data:", {
|
|
||||||
firstName: shippingAddress.firstName,
|
|
||||||
lastName: shippingAddress.lastName,
|
|
||||||
streetAddress1: shippingAddress.streetAddress1,
|
|
||||||
city: shippingAddress.city,
|
|
||||||
postalCode: shippingAddress.postalCode,
|
|
||||||
country: shippingAddress.country,
|
|
||||||
phone: shippingAddress.phone,
|
|
||||||
});
|
|
||||||
const shippingResult = await saleorClient.mutate<ShippingAddressUpdateResponse>({
|
|
||||||
mutation: CHECKOUT_SHIPPING_ADDRESS_UPDATE,
|
|
||||||
variables: {
|
|
||||||
checkoutId: checkout.id,
|
|
||||||
shippingAddress: {
|
|
||||||
firstName: shippingAddress.firstName,
|
|
||||||
lastName: shippingAddress.lastName,
|
|
||||||
streetAddress1: shippingAddress.streetAddress1,
|
|
||||||
streetAddress2: shippingAddress.streetAddress2,
|
|
||||||
city: shippingAddress.city,
|
|
||||||
postalCode: shippingAddress.postalCode,
|
|
||||||
country: shippingAddress.country,
|
|
||||||
phone: shippingAddress.phone,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (shippingResult.data?.checkoutShippingAddressUpdate?.errors && shippingResult.data.checkoutShippingAddressUpdate.errors.length > 0) {
|
|
||||||
throw new Error(`Shipping address update failed: ${shippingResult.data.checkoutShippingAddressUpdate.errors[0].message}`);
|
|
||||||
}
|
|
||||||
console.log("Step 2: Shipping address updated successfully");
|
|
||||||
|
|
||||||
// Query for checkout to get available shipping methods
|
|
||||||
console.log("Step 3: Fetching shipping methods...");
|
|
||||||
const checkoutQueryResult = await saleorClient.query<CheckoutQueryResponse>({
|
|
||||||
query: GET_CHECKOUT_BY_ID,
|
|
||||||
variables: {
|
|
||||||
id: checkout.id,
|
|
||||||
},
|
|
||||||
fetchPolicy: "network-only",
|
|
||||||
});
|
|
||||||
|
|
||||||
const availableMethods = checkoutQueryResult.data?.checkout?.shippingMethods || [];
|
|
||||||
console.log("Available shipping methods:", availableMethods);
|
|
||||||
|
|
||||||
if (availableMethods.length === 0) {
|
|
||||||
throw new Error(t("errorNoShippingMethods"));
|
|
||||||
}
|
|
||||||
|
|
||||||
setShippingMethods(availableMethods);
|
|
||||||
setShowShippingMethods(true);
|
|
||||||
|
|
||||||
// Track shipping step
|
|
||||||
trackCheckoutStep("shipping_method_selection", {
|
|
||||||
available_methods_count: availableMethods.length,
|
|
||||||
});
|
|
||||||
|
|
||||||
// Don't complete yet - show shipping method selection
|
|
||||||
console.log("Phase 1 complete. Waiting for shipping method selection...");
|
|
||||||
}
|
|
||||||
} catch (err: unknown) {
|
} catch (err: unknown) {
|
||||||
console.error("Checkout error:", err);
|
console.error("Checkout error:", err);
|
||||||
|
|
||||||
@@ -606,9 +617,17 @@ export default function CheckoutPage() {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Shipping Method Selection */}
|
{/* Shipping Method Selection */}
|
||||||
{showShippingMethods && shippingMethods.length > 0 && (
|
<div className="border-b border-border pb-6">
|
||||||
<div className="border-b border-border pb-6">
|
<h2 className="text-xl font-serif mb-4">{t("shippingMethod")}</h2>
|
||||||
<h2 className="text-xl font-serif mb-4">{t("shippingMethod")}</h2>
|
{isLoadingShipping ? (
|
||||||
|
<div className="flex items-center gap-2 text-foreground-muted">
|
||||||
|
<svg className="animate-spin h-5 w-5" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24">
|
||||||
|
<circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4"></circle>
|
||||||
|
<path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
||||||
|
</svg>
|
||||||
|
<span>{t("loadingShippingMethods")}</span>
|
||||||
|
</div>
|
||||||
|
) : shippingMethods.length > 0 ? (
|
||||||
<div className="space-y-3">
|
<div className="space-y-3">
|
||||||
{shippingMethods.map((method) => (
|
{shippingMethods.map((method) => (
|
||||||
<label
|
<label
|
||||||
@@ -636,18 +655,17 @@ export default function CheckoutPage() {
|
|||||||
</label>
|
</label>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
{!selectedShippingMethod && (
|
) : (
|
||||||
<p className="text-red-500 text-sm mt-2">{t("errorSelectShipping")}</p>
|
<p className="text-foreground-muted">{t("enterAddressForShipping")}</p>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
)}
|
|
||||||
|
|
||||||
<button
|
<button
|
||||||
type="submit"
|
type="submit"
|
||||||
disabled={isLoading || lines.length === 0 || (showShippingMethods && !selectedShippingMethod)}
|
disabled={isLoading || lines.length === 0 || !selectedShippingMethod}
|
||||||
className="w-full py-4 bg-foreground text-white font-medium hover:bg-accent-dark transition-colors disabled:opacity-50"
|
className="w-full py-4 bg-foreground text-white font-medium hover:bg-accent-dark transition-colors disabled:opacity-50"
|
||||||
>
|
>
|
||||||
{isLoading ? t("processing") : showShippingMethods ? t("completeOrder", { total: formatPrice(total) }) : t("continueToShipping")}
|
{isLoading ? t("processing") : t("completeOrder", { total: formatPrice(total) })}
|
||||||
</button>
|
</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
@@ -691,6 +709,12 @@ export default function CheckoutPage() {
|
|||||||
<span className="text-foreground-muted">{t("subtotal")}</span>
|
<span className="text-foreground-muted">{t("subtotal")}</span>
|
||||||
<span>{formatPrice(checkout?.subtotalPrice?.gross?.amount || 0)}</span>
|
<span>{formatPrice(checkout?.subtotalPrice?.gross?.amount || 0)}</span>
|
||||||
</div>
|
</div>
|
||||||
|
{selectedShippingMethod && (
|
||||||
|
<div className="flex justify-between">
|
||||||
|
<span className="text-foreground-muted">{t("shipping")}</span>
|
||||||
|
<span>{formatPrice(shippingMethods.find(m => m.id === selectedShippingMethod)?.price.amount || 0)}</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
<div className="flex justify-between font-medium text-lg pt-2 border-t border-border">
|
<div className="flex justify-between font-medium text-lg pt-2 border-t border-border">
|
||||||
<span>{t("total")}</span>
|
<span>{t("total")}</span>
|
||||||
<span>{formatPrice(total)}</span>
|
<span>{formatPrice(total)}</span>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState, useRef } from "react";
|
||||||
import { motion, AnimatePresence } from "framer-motion";
|
import { motion, AnimatePresence } from "framer-motion";
|
||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
@@ -30,14 +30,16 @@ export default function CartDrawer() {
|
|||||||
const lines = getLines();
|
const lines = getLines();
|
||||||
const total = getTotal();
|
const total = getTotal();
|
||||||
const lineCount = getLineCount();
|
const lineCount = getLineCount();
|
||||||
const [initialized, setInitialized] = useState(false);
|
const initializedRef = useRef(false);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!initialized) {
|
if (!initializedRef.current && locale) {
|
||||||
|
// Set language code before initializing checkout
|
||||||
|
useSaleorCheckoutStore.getState().setLanguageCode(locale);
|
||||||
initCheckout();
|
initCheckout();
|
||||||
setInitialized(true);
|
initializedRef.current = true;
|
||||||
}
|
}
|
||||||
}, [initialized]);
|
}, [locale]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (isOpen) {
|
if (isOpen) {
|
||||||
|
|||||||
@@ -15,9 +15,13 @@ interface NewHeroProps {
|
|||||||
|
|
||||||
export default function NewHero({ featuredProduct }: NewHeroProps) {
|
export default function NewHero({ featuredProduct }: NewHeroProps) {
|
||||||
const locale = useLocale();
|
const locale = useLocale();
|
||||||
const { addLine, openCart } = useSaleorCheckoutStore();
|
const { addLine, openCart, setLanguageCode } = useSaleorCheckoutStore();
|
||||||
|
|
||||||
const handleAddToCart = async () => {
|
const handleAddToCart = async () => {
|
||||||
|
// Set language code before adding to cart
|
||||||
|
if (locale) {
|
||||||
|
setLanguageCode(locale);
|
||||||
|
}
|
||||||
const variant = featuredProduct?.variants?.[0];
|
const variant = featuredProduct?.variants?.[0];
|
||||||
if (variant?.id) {
|
if (variant?.id) {
|
||||||
await addLine(variant.id, 1);
|
await addLine(variant.id, 1);
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import Link from "next/link";
|
|||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
import { usePathname } from "next/navigation";
|
import { usePathname } from "next/navigation";
|
||||||
import { AnimatePresence, motion } from "framer-motion";
|
import { AnimatePresence, motion } from "framer-motion";
|
||||||
import { useTranslations } from "next-intl";
|
import { useTranslations, useLocale } from "next-intl";
|
||||||
import { useSaleorCheckoutStore } from "@/stores/saleorCheckoutStore";
|
import { useSaleorCheckoutStore } from "@/stores/saleorCheckoutStore";
|
||||||
import { User, ShoppingBag, Menu, X, Globe } from "lucide-react";
|
import { User, ShoppingBag, Menu, X, Globe } from "lucide-react";
|
||||||
import CartDrawer from "@/components/cart/CartDrawer";
|
import CartDrawer from "@/components/cart/CartDrawer";
|
||||||
@@ -16,14 +16,15 @@ interface HeaderProps {
|
|||||||
locale?: string;
|
locale?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Header({ locale = "sr" }: HeaderProps) {
|
export default function Header({ locale: propLocale = "sr" }: HeaderProps) {
|
||||||
const t = useTranslations("Header");
|
const t = useTranslations("Header");
|
||||||
const pathname = usePathname();
|
const pathname = usePathname();
|
||||||
const dropdownRef = useRef<HTMLDivElement>(null);
|
const dropdownRef = useRef<HTMLDivElement>(null);
|
||||||
const [mobileMenuOpen, setMobileMenuOpen] = useState(false);
|
const [mobileMenuOpen, setMobileMenuOpen] = useState(false);
|
||||||
const [scrolled, setScrolled] = useState(false);
|
const [scrolled, setScrolled] = useState(false);
|
||||||
const [langDropdownOpen, setLangDropdownOpen] = useState(false);
|
const [langDropdownOpen, setLangDropdownOpen] = useState(false);
|
||||||
const { getLineCount, toggleCart, initCheckout } = useSaleorCheckoutStore();
|
const { getLineCount, toggleCart, initCheckout, setLanguageCode } = useSaleorCheckoutStore();
|
||||||
|
const locale = useLocale();
|
||||||
|
|
||||||
const itemCount = getLineCount();
|
const itemCount = getLineCount();
|
||||||
const currentLocale = isValidLocale(locale) ? LOCALE_CONFIG[locale] : LOCALE_CONFIG.sr;
|
const currentLocale = isValidLocale(locale) ? LOCALE_CONFIG[locale] : LOCALE_CONFIG.sr;
|
||||||
@@ -54,9 +55,14 @@ export default function Header({ locale = "sr" }: HeaderProps) {
|
|||||||
setLangDropdownOpen(false);
|
setLangDropdownOpen(false);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Set language code first, then initialize checkout
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
initCheckout();
|
if (locale) {
|
||||||
}, [initCheckout]);
|
setLanguageCode(locale);
|
||||||
|
// Initialize checkout after language code is set
|
||||||
|
initCheckout();
|
||||||
|
}
|
||||||
|
}, [locale, setLanguageCode, initCheckout]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const handleScroll = () => {
|
const handleScroll = () => {
|
||||||
|
|||||||
@@ -99,7 +99,7 @@ export default function ProductDetail({ product, relatedProducts, bundleProducts
|
|||||||
const [isAdding, setIsAdding] = useState(false);
|
const [isAdding, setIsAdding] = useState(false);
|
||||||
const [urgencyIndex, setUrgencyIndex] = useState(0);
|
const [urgencyIndex, setUrgencyIndex] = useState(0);
|
||||||
const [selectedBundleVariantId, setSelectedBundleVariantId] = useState<string | null>(null);
|
const [selectedBundleVariantId, setSelectedBundleVariantId] = useState<string | null>(null);
|
||||||
const { addLine, openCart } = useSaleorCheckoutStore();
|
const { addLine, openCart, setLanguageCode } = useSaleorCheckoutStore();
|
||||||
const { trackProductView, trackAddToCart } = useAnalytics();
|
const { trackProductView, trackAddToCart } = useAnalytics();
|
||||||
const validLocale = isValidLocale(locale) ? locale : "sr";
|
const validLocale = isValidLocale(locale) ? locale : "sr";
|
||||||
|
|
||||||
@@ -147,6 +147,11 @@ export default function ProductDetail({ product, relatedProducts, bundleProducts
|
|||||||
const handleAddToCart = async () => {
|
const handleAddToCart = async () => {
|
||||||
if (!selectedVariantId) return;
|
if (!selectedVariantId) return;
|
||||||
|
|
||||||
|
// Set language code before adding to cart
|
||||||
|
if (validLocale) {
|
||||||
|
setLanguageCode(validLocale);
|
||||||
|
}
|
||||||
|
|
||||||
setIsAdding(true);
|
setIsAdding(true);
|
||||||
try {
|
try {
|
||||||
await addLine(selectedVariantId, 1);
|
await addLine(selectedVariantId, 1);
|
||||||
|
|||||||
@@ -372,6 +372,11 @@
|
|||||||
"errorNoCheckout": "Keine aktive Kasse. Bitte versuchen Sie es erneut.",
|
"errorNoCheckout": "Keine aktive Kasse. Bitte versuchen Sie es erneut.",
|
||||||
"errorEmailRequired": "Bitte geben Sie eine gültige E-Mail-Adresse ein.",
|
"errorEmailRequired": "Bitte geben Sie eine gültige E-Mail-Adresse ein.",
|
||||||
"errorFieldsRequired": "Bitte füllen Sie alle erforderlichen Felder aus.",
|
"errorFieldsRequired": "Bitte füllen Sie alle erforderlichen Felder aus.",
|
||||||
|
"errorNoShippingMethods": "Keine Versandmethoden für diese Adresse verfügbar. Bitte überprüfen Sie Ihre Adresse oder kontaktieren Sie den Support.",
|
||||||
|
"errorSelectShipping": "Bitte wählen Sie eine Versandmethode.",
|
||||||
|
"errorPhoneRequired": "Bitte geben Sie eine gültige Telefonnummer ein.",
|
||||||
|
"loadingShippingMethods": "Versandoptionen werden geladen...",
|
||||||
|
"enterAddressForShipping": "Geben Sie Ihre Adresse ein, um Versandoptionen zu sehen.",
|
||||||
"errorOccurred": "Ein Fehler ist during des Checkouts aufgetreten.",
|
"errorOccurred": "Ein Fehler ist during des Checkouts aufgetreten.",
|
||||||
"errorCreatingOrder": "Bestellung konnte nicht erstellt werden.",
|
"errorCreatingOrder": "Bestellung konnte nicht erstellt werden.",
|
||||||
"orderConfirmed": "Bestellung bestätigt!",
|
"orderConfirmed": "Bestellung bestätigt!",
|
||||||
|
|||||||
@@ -420,6 +420,9 @@
|
|||||||
"errorFieldsRequired": "Please fill in all required fields.",
|
"errorFieldsRequired": "Please fill in all required fields.",
|
||||||
"errorNoShippingMethods": "No shipping methods available for this address. Please check your address or contact support.",
|
"errorNoShippingMethods": "No shipping methods available for this address. Please check your address or contact support.",
|
||||||
"errorSelectShipping": "Please select a shipping method.",
|
"errorSelectShipping": "Please select a shipping method.",
|
||||||
|
"errorPhoneRequired": "Please enter a valid phone number.",
|
||||||
|
"loadingShippingMethods": "Loading shipping options...",
|
||||||
|
"enterAddressForShipping": "Enter your address to see shipping options.",
|
||||||
"errorOccurred": "An error occurred during checkout.",
|
"errorOccurred": "An error occurred during checkout.",
|
||||||
"errorCreatingOrder": "Failed to create order.",
|
"errorCreatingOrder": "Failed to create order.",
|
||||||
"continueToShipping": "Continue to Shipping",
|
"continueToShipping": "Continue to Shipping",
|
||||||
|
|||||||
@@ -372,6 +372,11 @@
|
|||||||
"errorNoCheckout": "Pas de paiement actif. Veuillez réessayer.",
|
"errorNoCheckout": "Pas de paiement actif. Veuillez réessayer.",
|
||||||
"errorEmailRequired": "Veuillez entrer une adresse e-mail valide.",
|
"errorEmailRequired": "Veuillez entrer une adresse e-mail valide.",
|
||||||
"errorFieldsRequired": "Veuillez remplir tous les champs obligatoires.",
|
"errorFieldsRequired": "Veuillez remplir tous les champs obligatoires.",
|
||||||
|
"errorNoShippingMethods": "Aucune méthode de livraison disponible pour cette adresse. Veuillez vérifier votre adresse ou contacter le support.",
|
||||||
|
"errorSelectShipping": "Veuillez sélectionner une méthode de livraison.",
|
||||||
|
"errorPhoneRequired": "Veuillez entrer un numéro de téléphone valide.",
|
||||||
|
"loadingShippingMethods": "Chargement des options de livraison...",
|
||||||
|
"enterAddressForShipping": "Entrez votre adresse pour voir les options de livraison.",
|
||||||
"errorOccurred": "Une erreur s'est produite lors du paiement.",
|
"errorOccurred": "Une erreur s'est produite lors du paiement.",
|
||||||
"errorCreatingOrder": "Échec de la création de la commande.",
|
"errorCreatingOrder": "Échec de la création de la commande.",
|
||||||
"orderConfirmed": "Commande Confirmée!",
|
"orderConfirmed": "Commande Confirmée!",
|
||||||
|
|||||||
@@ -418,6 +418,11 @@
|
|||||||
"errorNoCheckout": "Nema aktivne korpe. Molimo pokušajte ponovo.",
|
"errorNoCheckout": "Nema aktivne korpe. Molimo pokušajte ponovo.",
|
||||||
"errorEmailRequired": "Molimo unesite validnu email adresu.",
|
"errorEmailRequired": "Molimo unesite validnu email adresu.",
|
||||||
"errorFieldsRequired": "Molimo popunite sva obavezna polja.",
|
"errorFieldsRequired": "Molimo popunite sva obavezna polja.",
|
||||||
|
"errorNoShippingMethods": "Nema dostupnih načina dostave za ovu adresu. Molimo proverite adresu ili kontaktirajte podršku.",
|
||||||
|
"errorSelectShipping": "Molimo izaberite način dostave.",
|
||||||
|
"errorPhoneRequired": "Molimo unesite validan broj telefona.",
|
||||||
|
"loadingShippingMethods": "Učitavanje opcija dostave...",
|
||||||
|
"enterAddressForShipping": "Unesite adresu da vidite opcije dostave.",
|
||||||
"errorOccurred": "Došlo je do greške prilikom kupovine.",
|
"errorOccurred": "Došlo je do greške prilikom kupovine.",
|
||||||
"errorCreatingOrder": "Neuspešno kreiranje narudžbine.",
|
"errorCreatingOrder": "Neuspešno kreiranje narudžbine.",
|
||||||
"orderConfirmed": "Narudžbina potvrđena!",
|
"orderConfirmed": "Narudžbina potvrđena!",
|
||||||
|
|||||||
@@ -58,6 +58,7 @@ interface GetCheckoutResponse {
|
|||||||
interface SaleorCheckoutStore {
|
interface SaleorCheckoutStore {
|
||||||
checkout: Checkout | null;
|
checkout: Checkout | null;
|
||||||
checkoutToken: string | null;
|
checkoutToken: string | null;
|
||||||
|
languageCode: string | null;
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
isLoading: boolean;
|
isLoading: boolean;
|
||||||
error: string | null;
|
error: string | null;
|
||||||
@@ -68,6 +69,7 @@ interface SaleorCheckoutStore {
|
|||||||
updateLine: (lineId: string, quantity: number) => Promise<void>;
|
updateLine: (lineId: string, quantity: number) => Promise<void>;
|
||||||
removeLine: (lineId: string) => Promise<void>;
|
removeLine: (lineId: string) => Promise<void>;
|
||||||
setEmail: (email: string) => Promise<void>;
|
setEmail: (email: string) => Promise<void>;
|
||||||
|
setLanguageCode: (languageCode: string) => void;
|
||||||
refreshCheckout: () => Promise<void>;
|
refreshCheckout: () => Promise<void>;
|
||||||
toggleCart: () => void;
|
toggleCart: () => void;
|
||||||
openCart: () => void;
|
openCart: () => void;
|
||||||
@@ -85,12 +87,13 @@ export const useSaleorCheckoutStore = create<SaleorCheckoutStore>()(
|
|||||||
(set, get) => ({
|
(set, get) => ({
|
||||||
checkout: null,
|
checkout: null,
|
||||||
checkoutToken: null,
|
checkoutToken: null,
|
||||||
|
languageCode: null,
|
||||||
isOpen: false,
|
isOpen: false,
|
||||||
isLoading: false,
|
isLoading: false,
|
||||||
error: null,
|
error: null,
|
||||||
|
|
||||||
initCheckout: async () => {
|
initCheckout: async () => {
|
||||||
const { checkoutToken } = get();
|
const { checkoutToken, languageCode } = get();
|
||||||
|
|
||||||
if (checkoutToken) {
|
if (checkoutToken) {
|
||||||
// Try to fetch existing checkout
|
// Try to fetch existing checkout
|
||||||
@@ -109,7 +112,7 @@ export const useSaleorCheckoutStore = create<SaleorCheckoutStore>()(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create new checkout
|
// Create new checkout with language code
|
||||||
try {
|
try {
|
||||||
const { data } = await saleorClient.mutate<CheckoutCreateResponse>({
|
const { data } = await saleorClient.mutate<CheckoutCreateResponse>({
|
||||||
mutation: CHECKOUT_CREATE,
|
mutation: CHECKOUT_CREATE,
|
||||||
@@ -117,6 +120,7 @@ export const useSaleorCheckoutStore = create<SaleorCheckoutStore>()(
|
|||||||
input: {
|
input: {
|
||||||
channel: CHANNEL,
|
channel: CHANNEL,
|
||||||
lines: [],
|
lines: [],
|
||||||
|
languageCode: languageCode ? languageCode.toUpperCase() : undefined,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
@@ -294,6 +298,7 @@ export const useSaleorCheckoutStore = create<SaleorCheckoutStore>()(
|
|||||||
openCart: () => set({ isOpen: true }),
|
openCart: () => set({ isOpen: true }),
|
||||||
closeCart: () => set({ isOpen: false }),
|
closeCart: () => set({ isOpen: false }),
|
||||||
clearError: () => set({ error: null }),
|
clearError: () => set({ error: null }),
|
||||||
|
setLanguageCode: (languageCode: string) => set({ languageCode }),
|
||||||
|
|
||||||
getLineCount: () => {
|
getLineCount: () => {
|
||||||
const { checkout } = get();
|
const { checkout } = get();
|
||||||
|
|||||||
Reference in New Issue
Block a user