@medusajs/draft-order 2.10.4-preview-20251001180205 → 2.10.4-preview-20251001210152

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -9852,127 +9852,22 @@ const EmailForm = ({ order }) => {
9852
9852
  const schema$3 = objectType({
9853
9853
  email: stringType().email()
9854
9854
  });
9855
- const NumberInput = React.forwardRef(
9856
- ({
9857
- value,
9858
- onChange,
9859
- size = "base",
9860
- min = 0,
9861
- max = 100,
9862
- step = 1,
9863
- className,
9864
- disabled,
9865
- ...props
9866
- }, ref) => {
9867
- const handleChange = (event) => {
9868
- const newValue = event.target.value === "" ? min : Number(event.target.value);
9869
- if (!isNaN(newValue) && (max === void 0 || newValue <= max) && (min === void 0 || newValue >= min)) {
9870
- onChange(newValue);
9871
- }
9872
- };
9873
- const handleIncrement = () => {
9874
- const newValue = value + step;
9875
- if (max === void 0 || newValue <= max) {
9876
- onChange(newValue);
9877
- }
9878
- };
9879
- const handleDecrement = () => {
9880
- const newValue = value - step;
9881
- if (min === void 0 || newValue >= min) {
9882
- onChange(newValue);
9883
- }
9884
- };
9885
- return /* @__PURE__ */ jsxRuntime.jsxs(
9886
- "div",
9887
- {
9888
- className: ui.clx(
9889
- "inline-flex rounded-md bg-ui-bg-field shadow-borders-base overflow-hidden divide-x transition-fg",
9890
- "[&:has(input:focus)]:shadow-borders-interactive-with-active",
9891
- {
9892
- "h-7": size === "small",
9893
- "h-8": size === "base"
9894
- },
9895
- className
9896
- ),
9897
- children: [
9898
- /* @__PURE__ */ jsxRuntime.jsx(
9899
- "input",
9900
- {
9901
- ref,
9902
- type: "number",
9903
- value,
9904
- onChange: handleChange,
9905
- min,
9906
- max,
9907
- step,
9908
- className: ui.clx(
9909
- "flex-1 px-2 py-1 bg-transparent txt-compact-small text-ui-fg-base outline-none [appearance:textfield]",
9910
- "[&::-webkit-outer-spin-button]:appearance-none [&::-webkit-inner-spin-button]:appearance-none",
9911
- "placeholder:text-ui-fg-muted"
9912
- ),
9913
- ...props
9914
- }
9915
- ),
9916
- /* @__PURE__ */ jsxRuntime.jsxs(
9917
- "button",
9918
- {
9919
- className: ui.clx(
9920
- "flex items-center justify-center outline-none transition-fg",
9921
- "disabled:cursor-not-allowed disabled:text-ui-fg-muted",
9922
- "focus:bg-ui-bg-field-component-hover",
9923
- "hover:bg-ui-bg-field-component-hover",
9924
- {
9925
- "size-7": size === "small",
9926
- "size-8": size === "base"
9927
- }
9928
- ),
9929
- type: "button",
9930
- onClick: handleDecrement,
9931
- disabled: min !== void 0 && value <= min || disabled,
9932
- children: [
9933
- /* @__PURE__ */ jsxRuntime.jsx(icons.Minus, {}),
9934
- /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: `Decrease by ${step}` })
9935
- ]
9936
- }
9937
- ),
9938
- /* @__PURE__ */ jsxRuntime.jsxs(
9939
- "button",
9940
- {
9941
- className: ui.clx(
9942
- "flex items-center justify-center outline-none transition-fg",
9943
- "disabled:cursor-not-allowed disabled:text-ui-fg-muted",
9944
- "focus:bg-ui-bg-field-hover",
9945
- "hover:bg-ui-bg-field-hover",
9946
- {
9947
- "size-7": size === "small",
9948
- "size-8": size === "base"
9949
- }
9950
- ),
9951
- type: "button",
9952
- onClick: handleIncrement,
9953
- disabled: max !== void 0 && value >= max || disabled,
9954
- children: [
9955
- /* @__PURE__ */ jsxRuntime.jsx(icons.Plus, {}),
9956
- /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: `Increase by ${step}` })
9957
- ]
9958
- }
9959
- )
9960
- ]
9961
- }
9962
- );
9963
- }
9964
- );
9965
- const PRODUCT_VARIANTS_QUERY_KEY = "product-variants";
9966
- const productVariantsQueryKeys = {
9855
+ const PROMOTION_QUERY_KEY = "promotions";
9856
+ const promotionsQueryKeys = {
9967
9857
  list: (query2) => [
9968
- PRODUCT_VARIANTS_QUERY_KEY,
9858
+ PROMOTION_QUERY_KEY,
9859
+ query2 ? query2 : void 0
9860
+ ],
9861
+ detail: (id, query2) => [
9862
+ PROMOTION_QUERY_KEY,
9863
+ id,
9969
9864
  query2 ? query2 : void 0
9970
9865
  ]
9971
9866
  };
9972
- const useProductVariants = (query2, options) => {
9867
+ const usePromotions = (query2, options) => {
9973
9868
  const { data, ...rest } = reactQuery.useQuery({
9974
- queryKey: productVariantsQueryKeys.list(query2),
9975
- queryFn: async () => await sdk.admin.productVariant.list(query2),
9869
+ queryKey: promotionsQueryKeys.list(query2),
9870
+ queryFn: async () => sdk.admin.promotion.list(query2),
9976
9871
  ...options
9977
9872
  });
9978
9873
  return { ...data, ...rest };
@@ -10023,65 +9918,85 @@ const useInitiateOrderEdit = ({
10023
9918
  run();
10024
9919
  }, [preview, navigate, mutateAsync]);
10025
9920
  };
10026
- function convertNumber(value) {
10027
- return typeof value === "string" ? Number(value.replace(",", ".")) : value;
10028
- }
10029
- const STACKED_MODAL_ID = "items_stacked_modal";
10030
- const Items = () => {
9921
+ const Promotions = () => {
10031
9922
  const { id } = reactRouterDom.useParams();
10032
9923
  const {
10033
9924
  order: preview,
10034
- isPending: isPreviewPending,
10035
9925
  isError: isPreviewError,
10036
9926
  error: previewError
10037
- } = useOrderPreview(id, void 0, {
10038
- placeholderData: reactQuery.keepPreviousData
10039
- });
9927
+ } = useOrderPreview(id, void 0);
10040
9928
  useInitiateOrderEdit({ preview });
10041
- const { draft_order, isPending, isError, error } = useDraftOrder(
10042
- id,
10043
- {
10044
- fields: "currency_code"
10045
- },
10046
- {
10047
- enabled: !!id
10048
- }
10049
- );
10050
9929
  const { onCancel } = useCancelOrderEdit({ preview });
10051
- if (isError) {
10052
- throw error;
10053
- }
10054
9930
  if (isPreviewError) {
10055
9931
  throw previewError;
10056
9932
  }
10057
- const ready = !!preview && !isPreviewPending && !!draft_order && !isPending;
10058
- return /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal, { onClose: onCancel, children: ready ? /* @__PURE__ */ jsxRuntime.jsx(ItemsForm, { preview, currencyCode: draft_order.currency_code }) : /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
10059
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Edit Items" }) }),
10060
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Loading data for the draft order, please wait..." }) })
10061
- ] }) });
9933
+ const isReady = !!preview;
9934
+ return /* @__PURE__ */ jsxRuntime.jsxs(RouteDrawer, { onClose: onCancel, children: [
9935
+ /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Header, { children: /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Heading, { children: "Edit Promotions" }) }) }),
9936
+ isReady && /* @__PURE__ */ jsxRuntime.jsx(PromotionForm, { preview })
9937
+ ] });
10062
9938
  };
10063
- const ItemsForm = ({ preview, currencyCode }) => {
10064
- var _a;
9939
+ const PromotionForm = ({ preview }) => {
9940
+ const { items, shipping_methods } = preview;
10065
9941
  const [isSubmitting, setIsSubmitting] = React.useState(false);
10066
- const [modalContent, setModalContent] = React.useState(
10067
- null
10068
- );
9942
+ const [comboboxValue, setComboboxValue] = React.useState("");
10069
9943
  const { handleSuccess } = useRouteModal();
10070
- const { searchValue, onSearchValueChange, query: query2 } = useDebouncedSearch();
9944
+ const { mutateAsync: addPromotions, isPending: isAddingPromotions } = useDraftOrderAddPromotions(preview.id);
9945
+ const promoIds = getPromotionIds(items, shipping_methods);
9946
+ const { promotions, isPending, isError, error } = usePromotions(
9947
+ {
9948
+ id: promoIds
9949
+ },
9950
+ {
9951
+ enabled: !!promoIds.length
9952
+ }
9953
+ );
9954
+ const comboboxData = useComboboxData({
9955
+ queryKey: ["promotions", "combobox", promoIds],
9956
+ queryFn: async (params) => {
9957
+ return await sdk.admin.promotion.list({
9958
+ ...params,
9959
+ id: {
9960
+ $nin: promoIds
9961
+ }
9962
+ });
9963
+ },
9964
+ getOptions: (data) => {
9965
+ return data.promotions.map((promotion) => ({
9966
+ label: promotion.code,
9967
+ value: promotion.code
9968
+ }));
9969
+ }
9970
+ });
9971
+ const add = async (value) => {
9972
+ if (!value) {
9973
+ return;
9974
+ }
9975
+ addPromotions(
9976
+ {
9977
+ promo_codes: [value]
9978
+ },
9979
+ {
9980
+ onError: (e) => {
9981
+ ui.toast.error(e.message);
9982
+ comboboxData.onSearchValueChange("");
9983
+ setComboboxValue("");
9984
+ },
9985
+ onSuccess: () => {
9986
+ comboboxData.onSearchValueChange("");
9987
+ setComboboxValue("");
9988
+ }
9989
+ }
9990
+ );
9991
+ };
10071
9992
  const { mutateAsync: confirmOrderEdit } = useDraftOrderConfirmEdit(preview.id);
10072
- const { mutateAsync: requestOrderEdit } = useDraftOrderRequestEdit(preview.id);
10073
- const itemCount = ((_a = preview.items) == null ? void 0 : _a.reduce((acc, item) => acc + item.quantity, 0)) || 0;
10074
- const matches = React.useMemo(() => {
10075
- return matchSorter.matchSorter(preview.items, query2, {
10076
- keys: ["product_title", "variant_title", "variant_sku", "title"]
10077
- });
10078
- }, [preview.items, query2]);
9993
+ const { mutateAsync: requestOrderEdit } = useOrderEditRequest(preview.id);
10079
9994
  const onSubmit = async () => {
10080
9995
  setIsSubmitting(true);
10081
9996
  let requestSucceeded = false;
10082
9997
  await requestOrderEdit(void 0, {
10083
9998
  onError: (e) => {
10084
- ui.toast.error(`Failed to request order edit: ${e.message}`);
9999
+ ui.toast.error(e.message);
10085
10000
  },
10086
10001
  onSuccess: () => {
10087
10002
  requestSucceeded = true;
@@ -10093,7 +10008,7 @@ const ItemsForm = ({ preview, currencyCode }) => {
10093
10008
  }
10094
10009
  await confirmOrderEdit(void 0, {
10095
10010
  onError: (e) => {
10096
- ui.toast.error(`Failed to confirm order edit: ${e.message}`);
10011
+ ui.toast.error(e.message);
10097
10012
  },
10098
10013
  onSuccess: () => {
10099
10014
  handleSuccess();
@@ -10103,860 +10018,294 @@ const ItemsForm = ({ preview, currencyCode }) => {
10103
10018
  }
10104
10019
  });
10105
10020
  };
10106
- const onKeyDown = React.useCallback(
10107
- (e) => {
10108
- if (e.key === "Enter" && (e.ctrlKey || e.metaKey)) {
10109
- if (modalContent || isSubmitting) {
10110
- return;
10111
- }
10112
- onSubmit();
10113
- }
10114
- },
10115
- [modalContent, isSubmitting, onSubmit]
10116
- );
10117
- React.useEffect(() => {
10118
- document.addEventListener("keydown", onKeyDown);
10119
- return () => {
10120
- document.removeEventListener("keydown", onKeyDown);
10121
- };
10122
- }, [onKeyDown]);
10123
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex h-full flex-col overflow-hidden", children: [
10124
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Header, {}),
10125
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Body, { className: "flex flex-1 flex-col overflow-hidden", children: /* @__PURE__ */ jsxRuntime.jsxs(
10126
- StackedFocusModal,
10127
- {
10128
- id: STACKED_MODAL_ID,
10129
- onOpenChangeCallback: (open) => {
10130
- if (!open) {
10131
- setModalContent(null);
10132
- }
10021
+ if (isError) {
10022
+ throw error;
10023
+ }
10024
+ return /* @__PURE__ */ jsxRuntime.jsxs(KeyboundForm, { className: "flex flex-1 flex-col", onSubmit, children: [
10025
+ /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Body, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col gap-4", children: [
10026
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col gap-3", children: [
10027
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col", children: [
10028
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Label, { size: "small", weight: "plus", htmlFor: "promotion-combobox", children: "Apply promotions" }),
10029
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Hint, { id: "promotion-combobox-hint", children: "Manage promotions that should be applied to the order." })
10030
+ ] }),
10031
+ /* @__PURE__ */ jsxRuntime.jsx(
10032
+ Combobox,
10033
+ {
10034
+ id: "promotion-combobox",
10035
+ "aria-describedby": "promotion-combobox-hint",
10036
+ isFetchingNextPage: comboboxData.isFetchingNextPage,
10037
+ fetchNextPage: comboboxData.fetchNextPage,
10038
+ options: comboboxData.options,
10039
+ onSearchValueChange: comboboxData.onSearchValueChange,
10040
+ searchValue: comboboxData.searchValue,
10041
+ disabled: comboboxData.disabled || isAddingPromotions,
10042
+ onChange: add,
10043
+ value: comboboxValue
10044
+ }
10045
+ )
10046
+ ] }),
10047
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Divider, { variant: "dashed" }),
10048
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex flex-col gap-2", children: promotions == null ? void 0 : promotions.map((promotion) => /* @__PURE__ */ jsxRuntime.jsx(
10049
+ PromotionItem,
10050
+ {
10051
+ promotion,
10052
+ orderId: preview.id,
10053
+ isLoading: isPending
10133
10054
  },
10134
- children: [
10135
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex flex-1 flex-col items-center overflow-y-auto", children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex w-full max-w-[720px] flex-col gap-y-6 px-6 py-16", children: [
10136
- /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
10137
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Heading, { children: "Edit Items" }) }),
10138
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", className: "text-ui-fg-subtle", children: "Edit the items in the draft order" }) })
10139
- ] }),
10140
- /* @__PURE__ */ jsxRuntime.jsx(ui.Divider, { variant: "dashed" }),
10141
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col gap-y-6", children: [
10142
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "grid grid-cols-2 items-center gap-3", children: [
10143
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col", children: [
10144
- /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: "Items" }),
10145
- /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", className: "text-ui-fg-subtle", children: "Choose items from the product catalog." })
10146
- ] }),
10147
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-2", children: [
10148
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex-1", children: /* @__PURE__ */ jsxRuntime.jsx(
10149
- ui.Input,
10150
- {
10151
- type: "search",
10152
- placeholder: "Search items",
10153
- value: searchValue,
10154
- onChange: (e) => onSearchValueChange(e.target.value)
10155
- }
10156
- ) }),
10157
- /* @__PURE__ */ jsxRuntime.jsxs(ui.DropdownMenu, { children: [
10158
- /* @__PURE__ */ jsxRuntime.jsx(ui.DropdownMenu.Trigger, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.IconButton, { type: "button", children: /* @__PURE__ */ jsxRuntime.jsx(icons.Plus, {}) }) }),
10159
- /* @__PURE__ */ jsxRuntime.jsxs(ui.DropdownMenu.Content, { children: [
10160
- /* @__PURE__ */ jsxRuntime.jsx(
10161
- StackedModalTrigger$1,
10162
- {
10163
- type: "add-items",
10164
- setModalContent
10165
- }
10166
- ),
10167
- /* @__PURE__ */ jsxRuntime.jsx(
10168
- StackedModalTrigger$1,
10169
- {
10170
- type: "add-custom-item",
10171
- setModalContent
10172
- }
10173
- )
10174
- ] })
10175
- ] })
10176
- ] })
10177
- ] }),
10178
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-subtle shadow-elevation-card-rest rounded-xl", children: [
10179
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "px-[5px]", children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "text-ui-fg-muted grid grid-cols-[2fr_1fr_2fr_28px] gap-3 px-4 py-2", children: [
10180
- /* @__PURE__ */ jsxRuntime.jsx("div", { children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: "Item" }) }),
10181
- /* @__PURE__ */ jsxRuntime.jsx("div", { children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: "Quantity" }) }),
10182
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "text-right", children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: "Price" }) }),
10183
- /* @__PURE__ */ jsxRuntime.jsx("div", {})
10184
- ] }) }),
10185
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex flex-col gap-y-1.5 px-[5px] pb-[5px]", children: itemCount <= 0 ? /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-base shadow-elevation-card-rest flex flex-col items-center justify-center gap-1 gap-x-3 rounded-lg p-4", children: [
10186
- /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: "There are no items in this order" }),
10187
- /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", className: "text-ui-fg-subtle", children: "Add items to the order to get started." })
10188
- ] }) : matches.length > 0 ? matches == null ? void 0 : matches.map((item) => /* @__PURE__ */ jsxRuntime.jsx(
10189
- Item,
10190
- {
10191
- item,
10192
- preview,
10193
- currencyCode
10194
- },
10195
- item.id
10196
- )) : /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-base shadow-elevation-card-rest flex flex-col items-center justify-center gap-1 gap-x-3 rounded-lg p-4", children: [
10197
- /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: "No items found" }),
10198
- /* @__PURE__ */ jsxRuntime.jsxs(ui.Text, { size: "small", className: "text-ui-fg-subtle", children: [
10199
- 'No items found for "',
10200
- query2,
10201
- '".'
10202
- ] })
10203
- ] }) })
10204
- ] })
10205
- ] }),
10206
- /* @__PURE__ */ jsxRuntime.jsx(ui.Divider, { variant: "dashed" }),
10207
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "grid grid-cols-[1fr_0.5fr_0.5fr] gap-3", children: [
10208
- /* @__PURE__ */ jsxRuntime.jsx("div", { children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: "Subtotal" }) }),
10209
- /* @__PURE__ */ jsxRuntime.jsx("div", { children: /* @__PURE__ */ jsxRuntime.jsxs(
10210
- ui.Text,
10211
- {
10212
- size: "small",
10213
- leading: "compact",
10214
- className: "text-ui-fg-subtle",
10215
- children: [
10216
- itemCount,
10217
- " ",
10218
- itemCount === 1 ? "item" : "items"
10219
- ]
10220
- }
10221
- ) }),
10222
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "text-right", children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: getStylizedAmount(preview.item_subtotal, currencyCode) }) })
10223
- ] })
10224
- ] }) }),
10225
- modalContent && (modalContent === "add-items" ? /* @__PURE__ */ jsxRuntime.jsx(ExistingItemsForm, { orderId: preview.id, items: preview.items }) : modalContent === "add-custom-item" ? /* @__PURE__ */ jsxRuntime.jsx(
10226
- CustomItemForm,
10227
- {
10228
- orderId: preview.id,
10229
- currencyCode
10230
- }
10231
- ) : null)
10232
- ]
10233
- }
10234
- ) }),
10235
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Footer, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center justify-end gap-x-2", children: [
10236
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Close, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", variant: "secondary", type: "button", children: "Cancel" }) }),
10055
+ promotion.id
10056
+ )) })
10057
+ ] }) }),
10058
+ /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Footer, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex justify-end gap-2", children: [
10059
+ /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Close, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", variant: "secondary", children: "Cancel" }) }),
10237
10060
  /* @__PURE__ */ jsxRuntime.jsx(
10238
10061
  ui.Button,
10239
10062
  {
10240
10063
  size: "small",
10241
- type: "button",
10242
- onClick: onSubmit,
10243
- isLoading: isSubmitting,
10064
+ type: "submit",
10065
+ isLoading: isSubmitting || isAddingPromotions,
10244
10066
  children: "Save"
10245
10067
  }
10246
10068
  )
10247
10069
  ] }) })
10248
10070
  ] });
10249
10071
  };
10250
- const Item = ({ item, preview, currencyCode }) => {
10251
- if (item.variant_id) {
10252
- return /* @__PURE__ */ jsxRuntime.jsx(VariantItem, { item, preview, currencyCode });
10253
- }
10254
- return /* @__PURE__ */ jsxRuntime.jsx(CustomItem, { item, preview, currencyCode });
10255
- };
10256
- const VariantItem = ({ item, preview, currencyCode }) => {
10257
- const [editing, setEditing] = React.useState(false);
10258
- const form = reactHookForm.useForm({
10259
- defaultValues: {
10260
- quantity: item.quantity,
10261
- unit_price: item.unit_price
10262
- },
10263
- resolver: zod.zodResolver(variantItemSchema)
10264
- });
10265
- const actionId = React.useMemo(() => {
10266
- var _a, _b;
10267
- return (_b = (_a = item.actions) == null ? void 0 : _a.find((a) => a.action === "ITEM_ADD")) == null ? void 0 : _b.id;
10268
- }, [item]);
10269
- const { mutateAsync: updateActionItem, isPending: isUpdatingActionItem } = useDraftOrderUpdateActionItem(preview.id);
10270
- const { mutateAsync: updateOriginalItem, isPending: isUpdatingOriginalItem } = useDraftOrderUpdateItem(preview.id);
10271
- const isPending = isUpdatingActionItem || isUpdatingOriginalItem;
10272
- const onSubmit = form.handleSubmit(async (data) => {
10273
- if (convertNumber(data.unit_price) === item.unit_price && data.quantity === item.quantity) {
10274
- setEditing(false);
10275
- return;
10276
- }
10277
- if (!actionId) {
10278
- await updateOriginalItem(
10279
- {
10280
- item_id: item.id,
10281
- quantity: data.quantity,
10282
- unit_price: convertNumber(data.unit_price)
10283
- },
10284
- {
10285
- onSuccess: () => {
10286
- setEditing(false);
10287
- },
10288
- onError: (e) => {
10289
- ui.toast.error(e.message);
10290
- }
10291
- }
10292
- );
10293
- return;
10294
- }
10295
- await updateActionItem(
10072
+ const PromotionItem = ({
10073
+ promotion,
10074
+ orderId,
10075
+ isLoading
10076
+ }) => {
10077
+ var _a;
10078
+ const { mutateAsync: removePromotions, isPending } = useDraftOrderRemovePromotions(orderId);
10079
+ const onRemove = async () => {
10080
+ removePromotions(
10296
10081
  {
10297
- action_id: actionId,
10298
- quantity: data.quantity,
10299
- unit_price: convertNumber(data.unit_price)
10082
+ promo_codes: [promotion.code]
10300
10083
  },
10301
10084
  {
10302
- onSuccess: () => {
10303
- setEditing(false);
10304
- },
10305
10085
  onError: (e) => {
10306
10086
  ui.toast.error(e.message);
10307
10087
  }
10308
10088
  }
10309
10089
  );
10310
- });
10311
- return /* @__PURE__ */ jsxRuntime.jsx(Form$2, { ...form, children: /* @__PURE__ */ jsxRuntime.jsx("form", { onSubmit, children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-base shadow-elevation-card-rest grid grid-cols-[minmax(0,2fr)_minmax(0,1fr)_minmax(0,2fr)_28px] items-center gap-3 rounded-lg px-4 py-2", children: [
10312
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex w-full items-center gap-x-3", children: [
10313
- /* @__PURE__ */ jsxRuntime.jsx(
10314
- Thumbnail,
10090
+ };
10091
+ const displayValue = getDisplayValue(promotion);
10092
+ return /* @__PURE__ */ jsxRuntime.jsxs(
10093
+ "div",
10094
+ {
10095
+ className: ui.clx(
10096
+ "bg-ui-bg-component shadow-elevation-card-rest flex items-center justify-between rounded-lg px-3 py-2",
10315
10097
  {
10316
- thumbnail: item.thumbnail,
10317
- alt: item.product_title ?? void 0
10098
+ "animate-pulse": isLoading
10318
10099
  }
10319
10100
  ),
10320
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col", children: [
10321
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-x-1", children: [
10322
- /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: item.product_title }),
10323
- /* @__PURE__ */ jsxRuntime.jsxs(
10324
- ui.Text,
10325
- {
10326
- size: "small",
10327
- leading: "compact",
10328
- className: "text-ui-fg-subtle",
10329
- children: [
10330
- "(",
10331
- item.variant_title,
10332
- ")"
10333
- ]
10334
- }
10335
- )
10101
+ children: [
10102
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
10103
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: promotion.code }),
10104
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "text-ui-fg-subtle flex items-center gap-1.5", children: [
10105
+ displayValue && /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-1.5", children: [
10106
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", leading: "compact", children: displayValue }),
10107
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", leading: "compact", children: "·" })
10108
+ ] }),
10109
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", leading: "compact", className: "capitalize", children: (_a = promotion.application_method) == null ? void 0 : _a.allocation })
10110
+ ] })
10336
10111
  ] }),
10337
10112
  /* @__PURE__ */ jsxRuntime.jsx(
10338
- ui.Text,
10113
+ ui.IconButton,
10339
10114
  {
10340
10115
  size: "small",
10341
- leading: "compact",
10342
- className: "text-ui-fg-subtle",
10343
- children: item.variant_sku
10344
- }
10345
- )
10346
- ] })
10347
- ] }),
10348
- editing ? /* @__PURE__ */ jsxRuntime.jsx("div", { className: "w-full flex-1", children: /* @__PURE__ */ jsxRuntime.jsx(
10349
- Form$2.Field,
10350
- {
10351
- control: form.control,
10352
- name: "quantity",
10353
- render: ({ field }) => {
10354
- return /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(NumberInput, { ...field }) }) });
10116
+ type: "button",
10117
+ variant: "transparent",
10118
+ onClick: onRemove,
10119
+ isLoading: isPending || isLoading,
10120
+ children: /* @__PURE__ */ jsxRuntime.jsx(icons.XMark, {})
10121
+ }
10122
+ )
10123
+ ]
10124
+ },
10125
+ promotion.id
10126
+ );
10127
+ };
10128
+ function getDisplayValue(promotion) {
10129
+ var _a, _b, _c, _d;
10130
+ const value = (_a = promotion.application_method) == null ? void 0 : _a.value;
10131
+ if (!value) {
10132
+ return null;
10133
+ }
10134
+ if (((_b = promotion.application_method) == null ? void 0 : _b.type) === "fixed") {
10135
+ const currency = (_c = promotion.application_method) == null ? void 0 : _c.currency_code;
10136
+ if (!currency) {
10137
+ return null;
10138
+ }
10139
+ return getLocaleAmount(value, currency);
10140
+ } else if (((_d = promotion.application_method) == null ? void 0 : _d.type) === "percentage") {
10141
+ return formatPercentage(value);
10142
+ }
10143
+ return null;
10144
+ }
10145
+ const formatter = new Intl.NumberFormat([], {
10146
+ style: "percent",
10147
+ minimumFractionDigits: 2
10148
+ });
10149
+ const formatPercentage = (value, isPercentageValue = false) => {
10150
+ let val = value || 0;
10151
+ if (!isPercentageValue) {
10152
+ val = val / 100;
10153
+ }
10154
+ return formatter.format(val);
10155
+ };
10156
+ function getPromotionIds(items, shippingMethods) {
10157
+ const promotionIds = /* @__PURE__ */ new Set();
10158
+ for (const item of items) {
10159
+ if (item.adjustments) {
10160
+ for (const adjustment of item.adjustments) {
10161
+ if (adjustment.promotion_id) {
10162
+ promotionIds.add(adjustment.promotion_id);
10355
10163
  }
10356
10164
  }
10357
- ) }) : /* @__PURE__ */ jsxRuntime.jsx("div", { className: "w-full flex-1", children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: item.quantity }) }),
10358
- editing ? /* @__PURE__ */ jsxRuntime.jsx("div", { className: "w-full flex-1", children: /* @__PURE__ */ jsxRuntime.jsx(
10359
- Form$2.Field,
10360
- {
10361
- control: form.control,
10362
- name: "unit_price",
10363
- render: ({ field: { onChange, ...field } }) => {
10364
- return /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(
10365
- ui.CurrencyInput,
10366
- {
10367
- ...field,
10368
- symbol: getNativeSymbol(currencyCode),
10369
- code: currencyCode,
10370
- onValueChange: (_value, _name, values) => onChange(values == null ? void 0 : values.value)
10371
- }
10372
- ) }) });
10165
+ }
10166
+ }
10167
+ for (const shippingMethod of shippingMethods) {
10168
+ if (shippingMethod.adjustments) {
10169
+ for (const adjustment of shippingMethod.adjustments) {
10170
+ if (adjustment.promotion_id) {
10171
+ promotionIds.add(adjustment.promotion_id);
10373
10172
  }
10374
10173
  }
10375
- ) }) : /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex w-full flex-1 items-center justify-end", children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: getLocaleAmount(item.unit_price, currencyCode) }) }),
10376
- /* @__PURE__ */ jsxRuntime.jsx(
10377
- ui.IconButton,
10174
+ }
10175
+ }
10176
+ return Array.from(promotionIds);
10177
+ }
10178
+ const InlineTip = React.forwardRef(
10179
+ ({ variant = "tip", label, className, children, ...props }, ref) => {
10180
+ const labelValue = label || (variant === "warning" ? "Warning" : "Tip");
10181
+ return /* @__PURE__ */ jsxRuntime.jsxs(
10182
+ "div",
10378
10183
  {
10379
- type: "button",
10380
- size: "small",
10381
- onClick: editing ? onSubmit : () => {
10382
- setEditing(true);
10383
- },
10384
- disabled: isPending,
10385
- children: editing ? /* @__PURE__ */ jsxRuntime.jsx(icons.Check, {}) : /* @__PURE__ */ jsxRuntime.jsx(icons.PencilSquare, {})
10184
+ ref,
10185
+ className: ui.clx(
10186
+ "bg-ui-bg-component txt-small text-ui-fg-subtle grid grid-cols-[4px_1fr] items-start gap-3 rounded-lg border p-3",
10187
+ className
10188
+ ),
10189
+ ...props,
10190
+ children: [
10191
+ /* @__PURE__ */ jsxRuntime.jsx(
10192
+ "div",
10193
+ {
10194
+ role: "presentation",
10195
+ className: ui.clx("w-4px bg-ui-tag-neutral-icon h-full rounded-full", {
10196
+ "bg-ui-tag-orange-icon": variant === "warning"
10197
+ })
10198
+ }
10199
+ ),
10200
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "text-pretty", children: [
10201
+ /* @__PURE__ */ jsxRuntime.jsxs("strong", { className: "txt-small-plus text-ui-fg-base", children: [
10202
+ labelValue,
10203
+ ":"
10204
+ ] }),
10205
+ " ",
10206
+ children
10207
+ ] })
10208
+ ]
10386
10209
  }
10387
- )
10388
- ] }) }) });
10389
- };
10390
- const variantItemSchema = objectType({
10391
- quantity: numberType(),
10392
- unit_price: unionType([numberType(), stringType()])
10210
+ );
10211
+ }
10212
+ );
10213
+ InlineTip.displayName = "InlineTip";
10214
+ const MetadataFieldSchema = objectType({
10215
+ key: stringType(),
10216
+ disabled: booleanType().optional(),
10217
+ value: anyType()
10393
10218
  });
10394
- const CustomItem = ({ item, preview, currencyCode }) => {
10395
- const [editing, setEditing] = React.useState(false);
10396
- const { quantity, unit_price, title } = item;
10219
+ const MetadataSchema = objectType({
10220
+ metadata: arrayType(MetadataFieldSchema)
10221
+ });
10222
+ const Metadata = () => {
10223
+ const { id } = reactRouterDom.useParams();
10224
+ const { order, isPending, isError, error } = useOrder(id, {
10225
+ fields: "metadata"
10226
+ });
10227
+ if (isError) {
10228
+ throw error;
10229
+ }
10230
+ const isReady = !isPending && !!order;
10231
+ return /* @__PURE__ */ jsxRuntime.jsxs(RouteDrawer, { children: [
10232
+ /* @__PURE__ */ jsxRuntime.jsxs(RouteDrawer.Header, { children: [
10233
+ /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Heading, { children: "Metadata" }) }),
10234
+ /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Add metadata to the draft order." }) })
10235
+ ] }),
10236
+ !isReady ? /* @__PURE__ */ jsxRuntime.jsx(PlaceholderInner, {}) : /* @__PURE__ */ jsxRuntime.jsx(MetadataForm, { orderId: id, metadata: order == null ? void 0 : order.metadata })
10237
+ ] });
10238
+ };
10239
+ const METADATA_KEY_LABEL_ID = "metadata-form-key-label";
10240
+ const METADATA_VALUE_LABEL_ID = "metadata-form-value-label";
10241
+ const MetadataForm = ({ orderId, metadata }) => {
10242
+ const { handleSuccess } = useRouteModal();
10243
+ const hasUneditableRows = getHasUneditableRows(metadata);
10244
+ const { mutateAsync, isPending } = useUpdateDraftOrder(orderId);
10397
10245
  const form = reactHookForm.useForm({
10398
10246
  defaultValues: {
10399
- title,
10400
- quantity,
10401
- unit_price
10247
+ metadata: getDefaultValues(metadata)
10402
10248
  },
10403
- resolver: zod.zodResolver(customItemSchema)
10249
+ resolver: zod.zodResolver(MetadataSchema)
10404
10250
  });
10405
- React.useEffect(() => {
10406
- form.reset({
10407
- title,
10408
- quantity,
10409
- unit_price
10410
- });
10411
- }, [form, title, quantity, unit_price]);
10412
- const actionId = React.useMemo(() => {
10413
- var _a, _b;
10414
- return (_b = (_a = item.actions) == null ? void 0 : _a.find((a) => a.action === "ITEM_ADD")) == null ? void 0 : _b.id;
10415
- }, [item]);
10416
- const { mutateAsync: updateActionItem, isPending: isUpdatingActionItem } = useDraftOrderUpdateActionItem(preview.id);
10417
- const { mutateAsync: removeActionItem, isPending: isRemovingActionItem } = useDraftOrderRemoveActionItem(preview.id);
10418
- const { mutateAsync: updateOriginalItem, isPending: isUpdatingOriginalItem } = useDraftOrderUpdateItem(preview.id);
10419
- const isPending = isUpdatingActionItem || isUpdatingOriginalItem;
10420
- const onSubmit = form.handleSubmit(async (data) => {
10421
- if (convertNumber(data.unit_price) === item.unit_price && data.quantity === item.quantity && data.title === item.title) {
10422
- setEditing(false);
10423
- return;
10424
- }
10425
- if (!actionId) {
10426
- await updateOriginalItem(
10427
- {
10428
- item_id: item.id,
10429
- quantity: data.quantity,
10430
- unit_price: convertNumber(data.unit_price)
10431
- },
10432
- {
10433
- onSuccess: () => {
10434
- setEditing(false);
10435
- },
10436
- onError: (e) => {
10437
- ui.toast.error(e.message);
10438
- }
10439
- }
10440
- );
10441
- return;
10442
- }
10443
- if (data.quantity === 0) {
10444
- await removeActionItem(actionId, {
10445
- onSuccess: () => {
10446
- setEditing(false);
10447
- },
10448
- onError: (e) => {
10449
- ui.toast.error(e.message);
10450
- }
10451
- });
10452
- return;
10453
- }
10454
- await updateActionItem(
10251
+ const handleSubmit = form.handleSubmit(async (data) => {
10252
+ const parsedData = parseValues(data);
10253
+ await mutateAsync(
10455
10254
  {
10456
- action_id: actionId,
10457
- quantity: data.quantity,
10458
- unit_price: convertNumber(data.unit_price)
10255
+ metadata: parsedData
10459
10256
  },
10460
10257
  {
10461
10258
  onSuccess: () => {
10462
- setEditing(false);
10259
+ ui.toast.success("Metadata updated");
10260
+ handleSuccess();
10463
10261
  },
10464
- onError: (e) => {
10465
- ui.toast.error(e.message);
10262
+ onError: (error) => {
10263
+ ui.toast.error(error.message);
10466
10264
  }
10467
10265
  }
10468
10266
  );
10469
10267
  });
10470
- return /* @__PURE__ */ jsxRuntime.jsx(Form$2, { ...form, children: /* @__PURE__ */ jsxRuntime.jsx("form", { onSubmit, children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-base shadow-elevation-card-rest grid grid-cols-[minmax(0,2fr)_minmax(0,1fr)_minmax(0,2fr)_28px] items-center gap-3 rounded-lg px-4 py-2", children: [
10471
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-x-3", children: [
10472
- /* @__PURE__ */ jsxRuntime.jsx(
10473
- Thumbnail,
10474
- {
10475
- thumbnail: item.thumbnail,
10476
- alt: item.title ?? void 0
10477
- }
10478
- ),
10479
- editing ? /* @__PURE__ */ jsxRuntime.jsx(
10480
- Form$2.Field,
10481
- {
10482
- control: form.control,
10483
- name: "title",
10484
- render: ({ field }) => {
10485
- return /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(ui.Input, { ...field }) }) });
10486
- }
10487
- }
10488
- ) : /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: item.title })
10489
- ] }),
10490
- editing ? /* @__PURE__ */ jsxRuntime.jsx(
10491
- Form$2.Field,
10492
- {
10493
- control: form.control,
10494
- name: "quantity",
10495
- render: ({ field }) => {
10496
- return /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(NumberInput, { ...field }) }) });
10497
- }
10498
- }
10499
- ) : /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: item.quantity }),
10500
- editing ? /* @__PURE__ */ jsxRuntime.jsx(
10501
- Form$2.Field,
10502
- {
10503
- control: form.control,
10504
- name: "unit_price",
10505
- render: ({ field: { onChange, ...field } }) => {
10506
- return /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(
10507
- ui.CurrencyInput,
10508
- {
10509
- ...field,
10510
- symbol: getNativeSymbol(currencyCode),
10511
- code: currencyCode,
10512
- onValueChange: (_value, _name, values) => onChange(values == null ? void 0 : values.value)
10513
- }
10514
- ) }) });
10515
- }
10516
- }
10517
- ) : /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex flex-1 items-center justify-end", children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: getLocaleAmount(item.unit_price, currencyCode) }) }),
10518
- /* @__PURE__ */ jsxRuntime.jsx(
10519
- ui.IconButton,
10520
- {
10521
- type: "button",
10522
- size: "small",
10523
- onClick: editing ? onSubmit : () => {
10524
- setEditing(true);
10525
- },
10526
- disabled: isPending,
10527
- children: editing ? /* @__PURE__ */ jsxRuntime.jsx(icons.Check, {}) : /* @__PURE__ */ jsxRuntime.jsx(icons.PencilSquare, {})
10528
- }
10529
- )
10530
- ] }) }) });
10531
- };
10532
- const StackedModalTrigger$1 = ({
10533
- type,
10534
- setModalContent
10535
- }) => {
10536
- const { setIsOpen } = useStackedModal();
10537
- const onClick = React.useCallback(() => {
10538
- setModalContent(type);
10539
- setIsOpen(STACKED_MODAL_ID, true);
10540
- }, [setModalContent, setIsOpen, type]);
10541
- return /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Trigger, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.DropdownMenu.Item, { onClick, children: type === "add-items" ? "Add items" : "Add custom item" }) });
10542
- };
10543
- const VARIANT_PREFIX = "items";
10544
- const LIMIT = 50;
10545
- const ExistingItemsForm = ({ orderId, items }) => {
10546
- const { setIsOpen } = useStackedModal();
10547
- const [rowSelection, setRowSelection] = React.useState(
10548
- items.reduce((acc, item) => {
10549
- acc[item.variant_id] = true;
10550
- return acc;
10551
- }, {})
10552
- );
10553
- React.useEffect(() => {
10554
- setRowSelection(
10555
- items.reduce((acc, item) => {
10556
- if (item.variant_id) {
10557
- acc[item.variant_id] = true;
10558
- }
10559
- return acc;
10560
- }, {})
10561
- );
10562
- }, [items]);
10563
- const { q, order, offset } = useQueryParams(
10564
- ["q", "order", "offset"],
10565
- VARIANT_PREFIX
10566
- );
10567
- const { variants, count, isPending, isError, error } = useProductVariants(
10568
- {
10569
- q,
10570
- order,
10571
- offset: offset ? parseInt(offset) : void 0,
10572
- limit: LIMIT
10573
- },
10574
- {
10575
- placeholderData: reactQuery.keepPreviousData
10268
+ const { fields, insert, remove } = reactHookForm.useFieldArray({
10269
+ control: form.control,
10270
+ name: "metadata"
10271
+ });
10272
+ function deleteRow(index) {
10273
+ remove(index);
10274
+ if (fields.length === 1) {
10275
+ insert(0, {
10276
+ key: "",
10277
+ value: "",
10278
+ disabled: false
10279
+ });
10576
10280
  }
10577
- );
10578
- const columns = useColumns();
10579
- const { mutateAsync } = useDraftOrderAddItems(orderId);
10580
- const onSubmit = async () => {
10581
- const ids = Object.keys(rowSelection).filter(
10582
- (id) => !items.find((i) => i.variant_id === id)
10583
- );
10584
- await mutateAsync(
10585
- {
10586
- items: ids.map((id) => ({
10587
- variant_id: id,
10588
- quantity: 1
10589
- }))
10590
- },
10591
- {
10592
- onSuccess: () => {
10593
- setRowSelection({});
10594
- setIsOpen(STACKED_MODAL_ID, false);
10595
- },
10596
- onError: (e) => {
10597
- ui.toast.error(e.message);
10598
- }
10599
- }
10600
- );
10601
- };
10602
- if (isError) {
10603
- throw error;
10604
10281
  }
10605
- return /* @__PURE__ */ jsxRuntime.jsxs(
10606
- StackedFocusModal.Content,
10282
+ function insertRow(index, position) {
10283
+ insert(index + (position === "above" ? 0 : 1), {
10284
+ key: "",
10285
+ value: "",
10286
+ disabled: false
10287
+ });
10288
+ }
10289
+ return /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Form, { form, children: /* @__PURE__ */ jsxRuntime.jsxs(
10290
+ KeyboundForm,
10607
10291
  {
10608
- onOpenAutoFocus: (e) => {
10609
- e.preventDefault();
10610
- const searchInput = document.querySelector(
10611
- "[data-modal-id='modal-search-input']"
10612
- );
10613
- if (searchInput) {
10614
- searchInput.focus();
10615
- }
10616
- },
10292
+ onSubmit: handleSubmit,
10293
+ className: "flex flex-1 flex-col overflow-hidden",
10617
10294
  children: [
10618
- /* @__PURE__ */ jsxRuntime.jsxs(StackedFocusModal.Header, { children: [
10619
- /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Product Variants" }) }),
10620
- /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Choose product variants to add to the order." }) })
10621
- ] }),
10622
- /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Body, { className: "flex-1 overflow-hidden", children: /* @__PURE__ */ jsxRuntime.jsx(
10623
- DataTable,
10624
- {
10625
- data: variants,
10626
- columns,
10627
- isLoading: isPending,
10628
- getRowId: (row) => row.id,
10629
- rowCount: count,
10630
- prefix: VARIANT_PREFIX,
10631
- layout: "fill",
10632
- rowSelection: {
10633
- state: rowSelection,
10634
- onRowSelectionChange: setRowSelection,
10635
- enableRowSelection: (row) => {
10636
- return !items.find((i) => i.variant_id === row.original.id);
10295
+ /* @__PURE__ */ jsxRuntime.jsxs(RouteDrawer.Body, { className: "flex flex-1 flex-col gap-y-8 overflow-y-auto", children: [
10296
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-base shadow-elevation-card-rest grid grid-cols-1 divide-y rounded-lg", children: [
10297
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-subtle grid grid-cols-2 divide-x rounded-t-lg", children: [
10298
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "txt-compact-small-plus text-ui-fg-subtle px-2 py-1.5", children: /* @__PURE__ */ jsxRuntime.jsx("label", { id: METADATA_KEY_LABEL_ID, children: "Key" }) }),
10299
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "txt-compact-small-plus text-ui-fg-subtle px-2 py-1.5", children: /* @__PURE__ */ jsxRuntime.jsx("label", { id: METADATA_VALUE_LABEL_ID, children: "Value" }) })
10300
+ ] }),
10301
+ fields.map((field, index) => {
10302
+ const isDisabled = field.disabled || false;
10303
+ let placeholder = "-";
10304
+ if (typeof field.value === "object") {
10305
+ placeholder = "{ ... }";
10637
10306
  }
10638
- },
10639
- autoFocusSearch: true
10640
- }
10641
- ) }),
10642
- /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Footer, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center justify-end gap-x-2", children: [
10643
- /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Close, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", variant: "secondary", type: "button", children: "Cancel" }) }),
10644
- /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", type: "button", onClick: onSubmit, children: "Update items" })
10645
- ] }) })
10646
- ]
10647
- }
10648
- );
10649
- };
10650
- const columnHelper = ui.createDataTableColumnHelper();
10651
- const useColumns = () => {
10652
- return React.useMemo(() => {
10653
- return [
10654
- columnHelper.select(),
10655
- columnHelper.accessor("product.title", {
10656
- header: "Product",
10657
- cell: ({ row }) => {
10658
- var _a, _b, _c;
10659
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-x-2", children: [
10660
- /* @__PURE__ */ jsxRuntime.jsx(
10661
- Thumbnail,
10662
- {
10663
- thumbnail: (_a = row.original.product) == null ? void 0 : _a.thumbnail,
10664
- alt: (_b = row.original.product) == null ? void 0 : _b.title
10665
- }
10666
- ),
10667
- /* @__PURE__ */ jsxRuntime.jsx("span", { children: (_c = row.original.product) == null ? void 0 : _c.title })
10668
- ] });
10669
- },
10670
- enableSorting: true
10671
- }),
10672
- columnHelper.accessor("title", {
10673
- header: "Variant",
10674
- enableSorting: true
10675
- }),
10676
- columnHelper.accessor("sku", {
10677
- header: "SKU",
10678
- cell: ({ getValue }) => {
10679
- return getValue() ?? "-";
10680
- },
10681
- enableSorting: true
10682
- }),
10683
- columnHelper.accessor("updated_at", {
10684
- header: "Updated",
10685
- cell: ({ getValue }) => {
10686
- return /* @__PURE__ */ jsxRuntime.jsx(
10687
- ui.Tooltip,
10688
- {
10689
- content: getFullDate({ date: getValue(), includeTime: true }),
10690
- children: /* @__PURE__ */ jsxRuntime.jsx("span", { children: getFullDate({ date: getValue() }) })
10691
- }
10692
- );
10693
- },
10694
- enableSorting: true,
10695
- sortAscLabel: "Oldest first",
10696
- sortDescLabel: "Newest first"
10697
- }),
10698
- columnHelper.accessor("created_at", {
10699
- header: "Created",
10700
- cell: ({ getValue }) => {
10701
- return /* @__PURE__ */ jsxRuntime.jsx(
10702
- ui.Tooltip,
10703
- {
10704
- content: getFullDate({ date: getValue(), includeTime: true }),
10705
- children: /* @__PURE__ */ jsxRuntime.jsx("span", { children: getFullDate({ date: getValue() }) })
10706
- }
10707
- );
10708
- },
10709
- enableSorting: true,
10710
- sortAscLabel: "Oldest first",
10711
- sortDescLabel: "Newest first"
10712
- })
10713
- ];
10714
- }, []);
10715
- };
10716
- const CustomItemForm = ({ orderId, currencyCode }) => {
10717
- const { setIsOpen } = useStackedModal();
10718
- const { mutateAsync: addItems } = useDraftOrderAddItems(orderId);
10719
- const form = reactHookForm.useForm({
10720
- defaultValues: {
10721
- title: "",
10722
- quantity: 1,
10723
- unit_price: ""
10724
- },
10725
- resolver: zod.zodResolver(customItemSchema)
10726
- });
10727
- const onSubmit = form.handleSubmit(async (data) => {
10728
- await addItems(
10729
- {
10730
- items: [
10731
- {
10732
- title: data.title,
10733
- quantity: data.quantity,
10734
- unit_price: convertNumber(data.unit_price)
10735
- }
10736
- ]
10737
- },
10738
- {
10739
- onSuccess: () => {
10740
- setIsOpen(STACKED_MODAL_ID, false);
10741
- },
10742
- onError: (e) => {
10743
- ui.toast.error(e.message);
10744
- }
10745
- }
10746
- );
10747
- });
10748
- return /* @__PURE__ */ jsxRuntime.jsx(Form$2, { ...form, children: /* @__PURE__ */ jsxRuntime.jsx(KeyboundForm, { onSubmit, children: /* @__PURE__ */ jsxRuntime.jsxs(StackedFocusModal.Content, { children: [
10749
- /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Header, {}),
10750
- /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Body, { className: "flex flex-1 flex-col overflow-hidden", children: /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex flex-1 flex-col items-center overflow-y-auto", children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex w-full max-w-[720px] flex-col gap-y-6 px-2 py-16", children: [
10751
- /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
10752
- /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Heading, { children: "Add custom item" }) }),
10753
- /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", className: "text-ui-fg-subtle", children: "Add a custom item to the order. This will add a new line item that is not associated with an existing product." }) })
10754
- ] }),
10755
- /* @__PURE__ */ jsxRuntime.jsx(ui.Divider, { variant: "dashed" }),
10756
- /* @__PURE__ */ jsxRuntime.jsx(
10757
- Form$2.Field,
10758
- {
10759
- control: form.control,
10760
- name: "title",
10761
- render: ({ field }) => /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "grid grid-cols-2 gap-x-3", children: [
10762
- /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
10763
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.Label, { children: "Title" }),
10764
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.Hint, { children: "Enter the title of the item" })
10765
- ] }),
10766
- /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
10767
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(ui.Input, { ...field }) }),
10768
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.ErrorMessage, {})
10769
- ] })
10770
- ] }) })
10771
- }
10772
- ),
10773
- /* @__PURE__ */ jsxRuntime.jsx(ui.Divider, { variant: "dashed" }),
10774
- /* @__PURE__ */ jsxRuntime.jsx(
10775
- Form$2.Field,
10776
- {
10777
- control: form.control,
10778
- name: "unit_price",
10779
- render: ({ field: { onChange, ...field } }) => /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "grid grid-cols-2 gap-x-3", children: [
10780
- /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
10781
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.Label, { children: "Unit price" }),
10782
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.Hint, { children: "Enter the unit price of the item" })
10783
- ] }),
10784
- /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
10785
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(
10786
- ui.CurrencyInput,
10787
- {
10788
- symbol: getNativeSymbol(currencyCode),
10789
- code: currencyCode,
10790
- onValueChange: (_value, _name, values) => onChange(values == null ? void 0 : values.value),
10791
- ...field
10792
- }
10793
- ) }),
10794
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.ErrorMessage, {})
10795
- ] })
10796
- ] }) })
10797
- }
10798
- ),
10799
- /* @__PURE__ */ jsxRuntime.jsx(ui.Divider, { variant: "dashed" }),
10800
- /* @__PURE__ */ jsxRuntime.jsx(
10801
- Form$2.Field,
10802
- {
10803
- control: form.control,
10804
- name: "quantity",
10805
- render: ({ field }) => /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "grid grid-cols-2 gap-x-3", children: [
10806
- /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
10807
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.Label, { children: "Quantity" }),
10808
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.Hint, { children: "Enter the quantity of the item" })
10809
- ] }),
10810
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "w-full flex-1", children: [
10811
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx("div", { className: "w-full flex-1", children: /* @__PURE__ */ jsxRuntime.jsx(NumberInput, { ...field, className: "w-full" }) }) }),
10812
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.ErrorMessage, {})
10813
- ] })
10814
- ] }) })
10815
- }
10816
- )
10817
- ] }) }) }),
10818
- /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Footer, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center justify-end gap-x-2", children: [
10819
- /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Close, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", variant: "secondary", type: "button", children: "Cancel" }) }),
10820
- /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", type: "button", onClick: onSubmit, children: "Add item" })
10821
- ] }) })
10822
- ] }) }) });
10823
- };
10824
- const customItemSchema = objectType({
10825
- title: stringType().min(1),
10826
- quantity: numberType(),
10827
- unit_price: unionType([numberType(), stringType()])
10828
- });
10829
- const InlineTip = React.forwardRef(
10830
- ({ variant = "tip", label, className, children, ...props }, ref) => {
10831
- const labelValue = label || (variant === "warning" ? "Warning" : "Tip");
10832
- return /* @__PURE__ */ jsxRuntime.jsxs(
10833
- "div",
10834
- {
10835
- ref,
10836
- className: ui.clx(
10837
- "bg-ui-bg-component txt-small text-ui-fg-subtle grid grid-cols-[4px_1fr] items-start gap-3 rounded-lg border p-3",
10838
- className
10839
- ),
10840
- ...props,
10841
- children: [
10842
- /* @__PURE__ */ jsxRuntime.jsx(
10843
- "div",
10844
- {
10845
- role: "presentation",
10846
- className: ui.clx("w-4px bg-ui-tag-neutral-icon h-full rounded-full", {
10847
- "bg-ui-tag-orange-icon": variant === "warning"
10848
- })
10849
- }
10850
- ),
10851
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "text-pretty", children: [
10852
- /* @__PURE__ */ jsxRuntime.jsxs("strong", { className: "txt-small-plus text-ui-fg-base", children: [
10853
- labelValue,
10854
- ":"
10855
- ] }),
10856
- " ",
10857
- children
10858
- ] })
10859
- ]
10860
- }
10861
- );
10862
- }
10863
- );
10864
- InlineTip.displayName = "InlineTip";
10865
- const MetadataFieldSchema = objectType({
10866
- key: stringType(),
10867
- disabled: booleanType().optional(),
10868
- value: anyType()
10869
- });
10870
- const MetadataSchema = objectType({
10871
- metadata: arrayType(MetadataFieldSchema)
10872
- });
10873
- const Metadata = () => {
10874
- const { id } = reactRouterDom.useParams();
10875
- const { order, isPending, isError, error } = useOrder(id, {
10876
- fields: "metadata"
10877
- });
10878
- if (isError) {
10879
- throw error;
10880
- }
10881
- const isReady = !isPending && !!order;
10882
- return /* @__PURE__ */ jsxRuntime.jsxs(RouteDrawer, { children: [
10883
- /* @__PURE__ */ jsxRuntime.jsxs(RouteDrawer.Header, { children: [
10884
- /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Heading, { children: "Metadata" }) }),
10885
- /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Add metadata to the draft order." }) })
10886
- ] }),
10887
- !isReady ? /* @__PURE__ */ jsxRuntime.jsx(PlaceholderInner, {}) : /* @__PURE__ */ jsxRuntime.jsx(MetadataForm, { orderId: id, metadata: order == null ? void 0 : order.metadata })
10888
- ] });
10889
- };
10890
- const METADATA_KEY_LABEL_ID = "metadata-form-key-label";
10891
- const METADATA_VALUE_LABEL_ID = "metadata-form-value-label";
10892
- const MetadataForm = ({ orderId, metadata }) => {
10893
- const { handleSuccess } = useRouteModal();
10894
- const hasUneditableRows = getHasUneditableRows(metadata);
10895
- const { mutateAsync, isPending } = useUpdateDraftOrder(orderId);
10896
- const form = reactHookForm.useForm({
10897
- defaultValues: {
10898
- metadata: getDefaultValues(metadata)
10899
- },
10900
- resolver: zod.zodResolver(MetadataSchema)
10901
- });
10902
- const handleSubmit = form.handleSubmit(async (data) => {
10903
- const parsedData = parseValues(data);
10904
- await mutateAsync(
10905
- {
10906
- metadata: parsedData
10907
- },
10908
- {
10909
- onSuccess: () => {
10910
- ui.toast.success("Metadata updated");
10911
- handleSuccess();
10912
- },
10913
- onError: (error) => {
10914
- ui.toast.error(error.message);
10915
- }
10916
- }
10917
- );
10918
- });
10919
- const { fields, insert, remove } = reactHookForm.useFieldArray({
10920
- control: form.control,
10921
- name: "metadata"
10922
- });
10923
- function deleteRow(index) {
10924
- remove(index);
10925
- if (fields.length === 1) {
10926
- insert(0, {
10927
- key: "",
10928
- value: "",
10929
- disabled: false
10930
- });
10931
- }
10932
- }
10933
- function insertRow(index, position) {
10934
- insert(index + (position === "above" ? 0 : 1), {
10935
- key: "",
10936
- value: "",
10937
- disabled: false
10938
- });
10939
- }
10940
- return /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Form, { form, children: /* @__PURE__ */ jsxRuntime.jsxs(
10941
- KeyboundForm,
10942
- {
10943
- onSubmit: handleSubmit,
10944
- className: "flex flex-1 flex-col overflow-hidden",
10945
- children: [
10946
- /* @__PURE__ */ jsxRuntime.jsxs(RouteDrawer.Body, { className: "flex flex-1 flex-col gap-y-8 overflow-y-auto", children: [
10947
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-base shadow-elevation-card-rest grid grid-cols-1 divide-y rounded-lg", children: [
10948
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-subtle grid grid-cols-2 divide-x rounded-t-lg", children: [
10949
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "txt-compact-small-plus text-ui-fg-subtle px-2 py-1.5", children: /* @__PURE__ */ jsxRuntime.jsx("label", { id: METADATA_KEY_LABEL_ID, children: "Key" }) }),
10950
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "txt-compact-small-plus text-ui-fg-subtle px-2 py-1.5", children: /* @__PURE__ */ jsxRuntime.jsx("label", { id: METADATA_VALUE_LABEL_ID, children: "Value" }) })
10951
- ] }),
10952
- fields.map((field, index) => {
10953
- const isDisabled = field.disabled || false;
10954
- let placeholder = "-";
10955
- if (typeof field.value === "object") {
10956
- placeholder = "{ ... }";
10957
- }
10958
- if (Array.isArray(field.value)) {
10959
- placeholder = "[ ... ]";
10307
+ if (Array.isArray(field.value)) {
10308
+ placeholder = "[ ... ]";
10960
10309
  }
10961
10310
  return /* @__PURE__ */ jsxRuntime.jsx(
10962
10311
  ConditionalTooltip,
@@ -11176,105 +10525,80 @@ function getHasUneditableRows(metadata) {
11176
10525
  (value) => !EDITABLE_TYPES.includes(typeof value)
11177
10526
  );
11178
10527
  }
11179
- const PROMOTION_QUERY_KEY = "promotions";
11180
- const promotionsQueryKeys = {
11181
- list: (query2) => [
11182
- PROMOTION_QUERY_KEY,
11183
- query2 ? query2 : void 0
11184
- ],
11185
- detail: (id, query2) => [
11186
- PROMOTION_QUERY_KEY,
11187
- id,
11188
- query2 ? query2 : void 0
11189
- ]
11190
- };
11191
- const usePromotions = (query2, options) => {
11192
- const { data, ...rest } = reactQuery.useQuery({
11193
- queryKey: promotionsQueryKeys.list(query2),
11194
- queryFn: async () => sdk.admin.promotion.list(query2),
11195
- ...options
11196
- });
11197
- return { ...data, ...rest };
11198
- };
11199
- const Promotions = () => {
10528
+ function convertNumber(value) {
10529
+ return typeof value === "string" ? Number(value.replace(",", ".")) : value;
10530
+ }
10531
+ const STACKED_FOCUS_MODAL_ID = "shipping-form";
10532
+ const Shipping = () => {
10533
+ var _a;
11200
10534
  const { id } = reactRouterDom.useParams();
10535
+ const { order, isPending, isError, error } = useOrder(id, {
10536
+ fields: "+items.*,+items.variant.*,+items.variant.product.*,+items.variant.product.shipping_profile.*,+currency_code"
10537
+ });
11201
10538
  const {
11202
10539
  order: preview,
10540
+ isPending: isPreviewPending,
11203
10541
  isError: isPreviewError,
11204
10542
  error: previewError
11205
- } = useOrderPreview(id, void 0);
10543
+ } = useOrderPreview(id);
11206
10544
  useInitiateOrderEdit({ preview });
11207
10545
  const { onCancel } = useCancelOrderEdit({ preview });
10546
+ if (isError) {
10547
+ throw error;
10548
+ }
11208
10549
  if (isPreviewError) {
11209
10550
  throw previewError;
11210
10551
  }
11211
- const isReady = !!preview;
11212
- return /* @__PURE__ */ jsxRuntime.jsxs(RouteDrawer, { onClose: onCancel, children: [
11213
- /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Header, { children: /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Heading, { children: "Edit Promotions" }) }) }),
11214
- isReady && /* @__PURE__ */ jsxRuntime.jsx(PromotionForm, { preview })
11215
- ] });
10552
+ const orderHasItems = (((_a = order == null ? void 0 : order.items) == null ? void 0 : _a.length) || 0) > 0;
10553
+ const isReady = preview && !isPreviewPending && order && !isPending;
10554
+ return /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal, { onClose: onCancel, children: !orderHasItems ? /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex h-full flex-col overflow-hidden ", children: [
10555
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Header, {}),
10556
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Body, { className: "flex flex-1 flex-col overflow-hidden", children: /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex flex-1 flex-col items-center overflow-y-auto", children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex w-full max-w-[720px] flex-col gap-y-6 py-16 px-6", children: [
10557
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Heading, { children: "Shipping" }) }),
10558
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", className: "text-ui-fg-subtle", children: "This draft order currently has no items. Add items to the order before adding shipping." }) })
10559
+ ] }) }) }),
10560
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Footer, { children: /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Close, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", variant: "secondary", type: "button", children: "Cancel" }) }) })
10561
+ ] }) : isReady ? /* @__PURE__ */ jsxRuntime.jsx(ShippingForm, { preview, order }) : /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
10562
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Edit Shipping" }) }),
10563
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Loading data for the draft order, please wait..." }) })
10564
+ ] }) });
11216
10565
  };
11217
- const PromotionForm = ({ preview }) => {
11218
- const { items, shipping_methods } = preview;
10566
+ const ShippingForm = ({ preview, order }) => {
10567
+ var _a;
10568
+ const { setIsOpen } = useStackedModal();
11219
10569
  const [isSubmitting, setIsSubmitting] = React.useState(false);
11220
- const [comboboxValue, setComboboxValue] = React.useState("");
11221
- const { handleSuccess } = useRouteModal();
11222
- const { mutateAsync: addPromotions, isPending: isAddingPromotions } = useDraftOrderAddPromotions(preview.id);
11223
- const promoIds = getPromotionIds(items, shipping_methods);
11224
- const { promotions, isPending, isError, error } = usePromotions(
10570
+ const [data, setData] = React.useState(null);
10571
+ const appliedShippingOptionIds = (_a = preview.shipping_methods) == null ? void 0 : _a.map((method) => method.shipping_option_id).filter(Boolean);
10572
+ const { shipping_options } = useShippingOptions(
11225
10573
  {
11226
- id: promoIds
10574
+ id: appliedShippingOptionIds,
10575
+ fields: "+service_zone.*,+service_zone.fulfillment_set.*,+service_zone.fulfillment_set.location.*"
11227
10576
  },
11228
10577
  {
11229
- enabled: !!promoIds.length
10578
+ enabled: appliedShippingOptionIds.length > 0
11230
10579
  }
11231
10580
  );
11232
- const comboboxData = useComboboxData({
11233
- queryKey: ["promotions", "combobox", promoIds],
11234
- queryFn: async (params) => {
11235
- return await sdk.admin.promotion.list({
11236
- ...params,
11237
- id: {
11238
- $nin: promoIds
11239
- }
11240
- });
11241
- },
11242
- getOptions: (data) => {
11243
- return data.promotions.map((promotion) => ({
11244
- label: promotion.code,
11245
- value: promotion.code
11246
- }));
11247
- }
11248
- });
11249
- const add = async (value) => {
11250
- if (!value) {
11251
- return;
11252
- }
11253
- addPromotions(
11254
- {
11255
- promo_codes: [value]
11256
- },
11257
- {
11258
- onError: (e) => {
11259
- ui.toast.error(e.message);
11260
- comboboxData.onSearchValueChange("");
11261
- setComboboxValue("");
11262
- },
11263
- onSuccess: () => {
11264
- comboboxData.onSearchValueChange("");
11265
- setComboboxValue("");
11266
- }
11267
- }
11268
- );
11269
- };
10581
+ const uniqueShippingProfiles = React.useMemo(() => {
10582
+ const profiles = /* @__PURE__ */ new Map();
10583
+ getUniqueShippingProfiles(order.items).forEach((profile) => {
10584
+ profiles.set(profile.id, profile);
10585
+ });
10586
+ shipping_options == null ? void 0 : shipping_options.forEach((option) => {
10587
+ profiles.set(option.shipping_profile_id, option.shipping_profile);
10588
+ });
10589
+ return Array.from(profiles.values());
10590
+ }, [order.items, shipping_options]);
10591
+ const { handleSuccess } = useRouteModal();
11270
10592
  const { mutateAsync: confirmOrderEdit } = useDraftOrderConfirmEdit(preview.id);
11271
- const { mutateAsync: requestOrderEdit } = useOrderEditRequest(preview.id);
10593
+ const { mutateAsync: requestOrderEdit } = useDraftOrderRequestEdit(preview.id);
10594
+ const { mutateAsync: removeShippingMethod } = useDraftOrderRemoveShippingMethod(preview.id);
10595
+ const { mutateAsync: removeActionShippingMethod } = useDraftOrderRemoveActionShippingMethod(preview.id);
11272
10596
  const onSubmit = async () => {
11273
10597
  setIsSubmitting(true);
11274
10598
  let requestSucceeded = false;
11275
10599
  await requestOrderEdit(void 0, {
11276
10600
  onError: (e) => {
11277
- ui.toast.error(e.message);
10601
+ ui.toast.error(`Failed to request order edit: ${e.message}`);
11278
10602
  },
11279
10603
  onSuccess: () => {
11280
10604
  requestSucceeded = true;
@@ -11286,7 +10610,7 @@ const PromotionForm = ({ preview }) => {
11286
10610
  }
11287
10611
  await confirmOrderEdit(void 0, {
11288
10612
  onError: (e) => {
11289
- ui.toast.error(e.message);
10613
+ ui.toast.error(`Failed to confirm order edit: ${e.message}`);
11290
10614
  },
11291
10615
  onSuccess: () => {
11292
10616
  handleSuccess();
@@ -11296,371 +10620,16 @@ const PromotionForm = ({ preview }) => {
11296
10620
  }
11297
10621
  });
11298
10622
  };
11299
- if (isError) {
11300
- throw error;
11301
- }
11302
- return /* @__PURE__ */ jsxRuntime.jsxs(KeyboundForm, { className: "flex flex-1 flex-col", onSubmit, children: [
11303
- /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Body, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col gap-4", children: [
11304
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col gap-3", children: [
11305
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col", children: [
11306
- /* @__PURE__ */ jsxRuntime.jsx(ui.Label, { size: "small", weight: "plus", htmlFor: "promotion-combobox", children: "Apply promotions" }),
11307
- /* @__PURE__ */ jsxRuntime.jsx(ui.Hint, { id: "promotion-combobox-hint", children: "Manage promotions that should be applied to the order." })
11308
- ] }),
11309
- /* @__PURE__ */ jsxRuntime.jsx(
11310
- Combobox,
11311
- {
11312
- id: "promotion-combobox",
11313
- "aria-describedby": "promotion-combobox-hint",
11314
- isFetchingNextPage: comboboxData.isFetchingNextPage,
11315
- fetchNextPage: comboboxData.fetchNextPage,
11316
- options: comboboxData.options,
11317
- onSearchValueChange: comboboxData.onSearchValueChange,
11318
- searchValue: comboboxData.searchValue,
11319
- disabled: comboboxData.disabled || isAddingPromotions,
11320
- onChange: add,
11321
- value: comboboxValue
11322
- }
11323
- )
11324
- ] }),
11325
- /* @__PURE__ */ jsxRuntime.jsx(ui.Divider, { variant: "dashed" }),
11326
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex flex-col gap-2", children: promotions == null ? void 0 : promotions.map((promotion) => /* @__PURE__ */ jsxRuntime.jsx(
11327
- PromotionItem,
11328
- {
11329
- promotion,
11330
- orderId: preview.id,
11331
- isLoading: isPending
11332
- },
11333
- promotion.id
11334
- )) })
11335
- ] }) }),
11336
- /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Footer, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex justify-end gap-2", children: [
11337
- /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Close, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", variant: "secondary", children: "Cancel" }) }),
11338
- /* @__PURE__ */ jsxRuntime.jsx(
11339
- ui.Button,
11340
- {
11341
- size: "small",
11342
- type: "submit",
11343
- isLoading: isSubmitting || isAddingPromotions,
11344
- children: "Save"
11345
- }
11346
- )
11347
- ] }) })
11348
- ] });
11349
- };
11350
- const PromotionItem = ({
11351
- promotion,
11352
- orderId,
11353
- isLoading
11354
- }) => {
11355
- var _a;
11356
- const { mutateAsync: removePromotions, isPending } = useDraftOrderRemovePromotions(orderId);
11357
- const onRemove = async () => {
11358
- removePromotions(
11359
- {
11360
- promo_codes: [promotion.code]
11361
- },
11362
- {
11363
- onError: (e) => {
11364
- ui.toast.error(e.message);
10623
+ const onKeydown = React.useCallback(
10624
+ (e) => {
10625
+ if (e.key === "Enter" && (e.ctrlKey || e.metaKey)) {
10626
+ if (data || isSubmitting) {
10627
+ return;
11365
10628
  }
10629
+ onSubmit();
11366
10630
  }
11367
- );
11368
- };
11369
- const displayValue = getDisplayValue(promotion);
11370
- return /* @__PURE__ */ jsxRuntime.jsxs(
11371
- "div",
11372
- {
11373
- className: ui.clx(
11374
- "bg-ui-bg-component shadow-elevation-card-rest flex items-center justify-between rounded-lg px-3 py-2",
11375
- {
11376
- "animate-pulse": isLoading
11377
- }
11378
- ),
11379
- children: [
11380
- /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
11381
- /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: promotion.code }),
11382
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "text-ui-fg-subtle flex items-center gap-1.5", children: [
11383
- displayValue && /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-1.5", children: [
11384
- /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", leading: "compact", children: displayValue }),
11385
- /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", leading: "compact", children: "·" })
11386
- ] }),
11387
- /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", leading: "compact", className: "capitalize", children: (_a = promotion.application_method) == null ? void 0 : _a.allocation })
11388
- ] })
11389
- ] }),
11390
- /* @__PURE__ */ jsxRuntime.jsx(
11391
- ui.IconButton,
11392
- {
11393
- size: "small",
11394
- type: "button",
11395
- variant: "transparent",
11396
- onClick: onRemove,
11397
- isLoading: isPending || isLoading,
11398
- children: /* @__PURE__ */ jsxRuntime.jsx(icons.XMark, {})
11399
- }
11400
- )
11401
- ]
11402
10631
  },
11403
- promotion.id
11404
- );
11405
- };
11406
- function getDisplayValue(promotion) {
11407
- var _a, _b, _c, _d;
11408
- const value = (_a = promotion.application_method) == null ? void 0 : _a.value;
11409
- if (!value) {
11410
- return null;
11411
- }
11412
- if (((_b = promotion.application_method) == null ? void 0 : _b.type) === "fixed") {
11413
- const currency = (_c = promotion.application_method) == null ? void 0 : _c.currency_code;
11414
- if (!currency) {
11415
- return null;
11416
- }
11417
- return getLocaleAmount(value, currency);
11418
- } else if (((_d = promotion.application_method) == null ? void 0 : _d.type) === "percentage") {
11419
- return formatPercentage(value);
11420
- }
11421
- return null;
11422
- }
11423
- const formatter = new Intl.NumberFormat([], {
11424
- style: "percent",
11425
- minimumFractionDigits: 2
11426
- });
11427
- const formatPercentage = (value, isPercentageValue = false) => {
11428
- let val = value || 0;
11429
- if (!isPercentageValue) {
11430
- val = val / 100;
11431
- }
11432
- return formatter.format(val);
11433
- };
11434
- function getPromotionIds(items, shippingMethods) {
11435
- const promotionIds = /* @__PURE__ */ new Set();
11436
- for (const item of items) {
11437
- if (item.adjustments) {
11438
- for (const adjustment of item.adjustments) {
11439
- if (adjustment.promotion_id) {
11440
- promotionIds.add(adjustment.promotion_id);
11441
- }
11442
- }
11443
- }
11444
- }
11445
- for (const shippingMethod of shippingMethods) {
11446
- if (shippingMethod.adjustments) {
11447
- for (const adjustment of shippingMethod.adjustments) {
11448
- if (adjustment.promotion_id) {
11449
- promotionIds.add(adjustment.promotion_id);
11450
- }
11451
- }
11452
- }
11453
- }
11454
- return Array.from(promotionIds);
11455
- }
11456
- const SalesChannel = () => {
11457
- const { id } = reactRouterDom.useParams();
11458
- const { draft_order, isPending, isError, error } = useDraftOrder(
11459
- id,
11460
- {
11461
- fields: "+sales_channel_id"
11462
- },
11463
- {
11464
- enabled: !!id
11465
- }
11466
- );
11467
- if (isError) {
11468
- throw error;
11469
- }
11470
- const ISrEADY = !!draft_order && !isPending;
11471
- return /* @__PURE__ */ jsxRuntime.jsxs(RouteDrawer, { children: [
11472
- /* @__PURE__ */ jsxRuntime.jsxs(RouteDrawer.Header, { children: [
11473
- /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Heading, { children: "Edit Sales Channel" }) }),
11474
- /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Update which sales channel the draft order is associated with" }) })
11475
- ] }),
11476
- ISrEADY && /* @__PURE__ */ jsxRuntime.jsx(SalesChannelForm, { order: draft_order })
11477
- ] });
11478
- };
11479
- const SalesChannelForm = ({ order }) => {
11480
- const form = reactHookForm.useForm({
11481
- defaultValues: {
11482
- sales_channel_id: order.sales_channel_id || ""
11483
- },
11484
- resolver: zod.zodResolver(schema$2)
11485
- });
11486
- const { mutateAsync, isPending } = useUpdateDraftOrder(order.id);
11487
- const { handleSuccess } = useRouteModal();
11488
- const onSubmit = form.handleSubmit(async (data) => {
11489
- await mutateAsync(
11490
- {
11491
- sales_channel_id: data.sales_channel_id
11492
- },
11493
- {
11494
- onSuccess: () => {
11495
- ui.toast.success("Sales channel updated");
11496
- handleSuccess();
11497
- },
11498
- onError: (error) => {
11499
- ui.toast.error(error.message);
11500
- }
11501
- }
11502
- );
11503
- });
11504
- return /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Form, { form, children: /* @__PURE__ */ jsxRuntime.jsxs(
11505
- KeyboundForm,
11506
- {
11507
- className: "flex flex-1 flex-col overflow-hidden",
11508
- onSubmit,
11509
- children: [
11510
- /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Body, { className: "flex flex-col gap-y-6 overflow-y-auto", children: /* @__PURE__ */ jsxRuntime.jsx(SalesChannelField, { control: form.control, order }) }),
11511
- /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Footer, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex justify-end gap-2", children: [
11512
- /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Close, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", variant: "secondary", children: "Cancel" }) }),
11513
- /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", type: "submit", isLoading: isPending, children: "Save" })
11514
- ] }) })
11515
- ]
11516
- }
11517
- ) });
11518
- };
11519
- const SalesChannelField = ({ control, order }) => {
11520
- const salesChannels = useComboboxData({
11521
- queryFn: async (params) => {
11522
- return await sdk.admin.salesChannel.list(params);
11523
- },
11524
- queryKey: ["sales-channels"],
11525
- getOptions: (data) => {
11526
- return data.sales_channels.map((salesChannel) => ({
11527
- label: salesChannel.name,
11528
- value: salesChannel.id
11529
- }));
11530
- },
11531
- defaultValue: order.sales_channel_id || void 0
11532
- });
11533
- return /* @__PURE__ */ jsxRuntime.jsx(
11534
- Form$2.Field,
11535
- {
11536
- control,
11537
- name: "sales_channel_id",
11538
- render: ({ field }) => {
11539
- return /* @__PURE__ */ jsxRuntime.jsxs(Form$2.Item, { children: [
11540
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.Label, { children: "Sales Channel" }),
11541
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(
11542
- Combobox,
11543
- {
11544
- options: salesChannels.options,
11545
- fetchNextPage: salesChannels.fetchNextPage,
11546
- isFetchingNextPage: salesChannels.isFetchingNextPage,
11547
- searchValue: salesChannels.searchValue,
11548
- onSearchValueChange: salesChannels.onSearchValueChange,
11549
- placeholder: "Select sales channel",
11550
- ...field
11551
- }
11552
- ) }),
11553
- /* @__PURE__ */ jsxRuntime.jsx(Form$2.ErrorMessage, {})
11554
- ] });
11555
- }
11556
- }
11557
- );
11558
- };
11559
- const schema$2 = objectType({
11560
- sales_channel_id: stringType().min(1)
11561
- });
11562
- const STACKED_FOCUS_MODAL_ID = "shipping-form";
11563
- const Shipping = () => {
11564
- var _a;
11565
- const { id } = reactRouterDom.useParams();
11566
- const { order, isPending, isError, error } = useOrder(id, {
11567
- fields: "+items.*,+items.variant.*,+items.variant.product.*,+items.variant.product.shipping_profile.*,+currency_code"
11568
- });
11569
- const {
11570
- order: preview,
11571
- isPending: isPreviewPending,
11572
- isError: isPreviewError,
11573
- error: previewError
11574
- } = useOrderPreview(id);
11575
- useInitiateOrderEdit({ preview });
11576
- const { onCancel } = useCancelOrderEdit({ preview });
11577
- if (isError) {
11578
- throw error;
11579
- }
11580
- if (isPreviewError) {
11581
- throw previewError;
11582
- }
11583
- const orderHasItems = (((_a = order == null ? void 0 : order.items) == null ? void 0 : _a.length) || 0) > 0;
11584
- const isReady = preview && !isPreviewPending && order && !isPending;
11585
- return /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal, { onClose: onCancel, children: !orderHasItems ? /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex h-full flex-col overflow-hidden ", children: [
11586
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Header, {}),
11587
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Body, { className: "flex flex-1 flex-col overflow-hidden", children: /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex flex-1 flex-col items-center overflow-y-auto", children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex w-full max-w-[720px] flex-col gap-y-6 py-16 px-6", children: [
11588
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Heading, { children: "Shipping" }) }),
11589
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", className: "text-ui-fg-subtle", children: "This draft order currently has no items. Add items to the order before adding shipping." }) })
11590
- ] }) }) }),
11591
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Footer, { children: /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Close, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", variant: "secondary", type: "button", children: "Cancel" }) }) })
11592
- ] }) : isReady ? /* @__PURE__ */ jsxRuntime.jsx(ShippingForm, { preview, order }) : /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
11593
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Edit Shipping" }) }),
11594
- /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Loading data for the draft order, please wait..." }) })
11595
- ] }) });
11596
- };
11597
- const ShippingForm = ({ preview, order }) => {
11598
- var _a;
11599
- const { setIsOpen } = useStackedModal();
11600
- const [isSubmitting, setIsSubmitting] = React.useState(false);
11601
- const [data, setData] = React.useState(null);
11602
- const appliedShippingOptionIds = (_a = preview.shipping_methods) == null ? void 0 : _a.map((method) => method.shipping_option_id).filter(Boolean);
11603
- const { shipping_options } = useShippingOptions(
11604
- {
11605
- id: appliedShippingOptionIds,
11606
- fields: "+service_zone.*,+service_zone.fulfillment_set.*,+service_zone.fulfillment_set.location.*"
11607
- },
11608
- {
11609
- enabled: appliedShippingOptionIds.length > 0
11610
- }
11611
- );
11612
- const uniqueShippingProfiles = React.useMemo(() => {
11613
- const profiles = /* @__PURE__ */ new Map();
11614
- getUniqueShippingProfiles(order.items).forEach((profile) => {
11615
- profiles.set(profile.id, profile);
11616
- });
11617
- shipping_options == null ? void 0 : shipping_options.forEach((option) => {
11618
- profiles.set(option.shipping_profile_id, option.shipping_profile);
11619
- });
11620
- return Array.from(profiles.values());
11621
- }, [order.items, shipping_options]);
11622
- const { handleSuccess } = useRouteModal();
11623
- const { mutateAsync: confirmOrderEdit } = useDraftOrderConfirmEdit(preview.id);
11624
- const { mutateAsync: requestOrderEdit } = useDraftOrderRequestEdit(preview.id);
11625
- const { mutateAsync: removeShippingMethod } = useDraftOrderRemoveShippingMethod(preview.id);
11626
- const { mutateAsync: removeActionShippingMethod } = useDraftOrderRemoveActionShippingMethod(preview.id);
11627
- const onSubmit = async () => {
11628
- setIsSubmitting(true);
11629
- let requestSucceeded = false;
11630
- await requestOrderEdit(void 0, {
11631
- onError: (e) => {
11632
- ui.toast.error(`Failed to request order edit: ${e.message}`);
11633
- },
11634
- onSuccess: () => {
11635
- requestSucceeded = true;
11636
- }
11637
- });
11638
- if (!requestSucceeded) {
11639
- setIsSubmitting(false);
11640
- return;
11641
- }
11642
- await confirmOrderEdit(void 0, {
11643
- onError: (e) => {
11644
- ui.toast.error(`Failed to confirm order edit: ${e.message}`);
11645
- },
11646
- onSuccess: () => {
11647
- handleSuccess();
11648
- },
11649
- onSettled: () => {
11650
- setIsSubmitting(false);
11651
- }
11652
- });
11653
- };
11654
- const onKeydown = React.useCallback(
11655
- (e) => {
11656
- if (e.key === "Enter" && (e.ctrlKey || e.metaKey)) {
11657
- if (data || isSubmitting) {
11658
- return;
11659
- }
11660
- onSubmit();
11661
- }
11662
- },
11663
- [data, isSubmitting, onSubmit]
10632
+ [data, isSubmitting, onSubmit]
11664
10633
  );
11665
10634
  React.useEffect(() => {
11666
10635
  document.addEventListener("keydown", onKeydown);
@@ -11866,7 +10835,7 @@ const ShippingForm = ({ preview, order }) => {
11866
10835
  ]
11867
10836
  }
11868
10837
  ) : /* @__PURE__ */ jsxRuntime.jsx(
11869
- StackedModalTrigger,
10838
+ StackedModalTrigger$1,
11870
10839
  {
11871
10840
  shippingProfileId: profile.id,
11872
10841
  shippingOption,
@@ -11977,7 +10946,7 @@ const ShippingForm = ({ preview, order }) => {
11977
10946
  ] }) })
11978
10947
  ] });
11979
10948
  };
11980
- const StackedModalTrigger = ({
10949
+ const StackedModalTrigger$1 = ({
11981
10950
  shippingProfileId,
11982
10951
  shippingOption,
11983
10952
  shippingMethod,
@@ -12398,7 +11367,7 @@ const ShippingAddressForm = ({ order }) => {
12398
11367
  postal_code: ((_i = order.shipping_address) == null ? void 0 : _i.postal_code) ?? "",
12399
11368
  phone: ((_j = order.shipping_address) == null ? void 0 : _j.phone) ?? ""
12400
11369
  },
12401
- resolver: zod.zodResolver(schema$1)
11370
+ resolver: zod.zodResolver(schema$2)
12402
11371
  });
12403
11372
  const { mutateAsync, isPending } = useUpdateDraftOrder(order.id);
12404
11373
  const { handleSuccess } = useRouteModal();
@@ -12568,7 +11537,7 @@ const ShippingAddressForm = ({ order }) => {
12568
11537
  }
12569
11538
  ) });
12570
11539
  };
12571
- const schema$1 = addressSchema;
11540
+ const schema$2 = addressSchema;
12572
11541
  const TransferOwnership = () => {
12573
11542
  const { id } = reactRouterDom.useParams();
12574
11543
  const { draft_order, isPending, isError, error } = useDraftOrder(id, {
@@ -12592,7 +11561,7 @@ const TransferOwnershipForm = ({ order }) => {
12592
11561
  defaultValues: {
12593
11562
  customer_id: order.customer_id || ""
12594
11563
  },
12595
- resolver: zod.zodResolver(schema)
11564
+ resolver: zod.zodResolver(schema$1)
12596
11565
  });
12597
11566
  const { mutateAsync, isPending } = useUpdateDraftOrder(order.id);
12598
11567
  const { handleSuccess } = useRouteModal();
@@ -13022,28 +11991,1059 @@ const Illustration = () => {
13022
11991
  /* @__PURE__ */ jsxRuntime.jsx("clipPath", { id: "clip4_20915_38670", children: /* @__PURE__ */ jsxRuntime.jsx(
13023
11992
  "rect",
13024
11993
  {
13025
- width: "12",
13026
- height: "12",
13027
- fill: "white",
13028
- transform: "matrix(0.865865 0.500278 -0.871576 0.490261 131.318 90.4594)"
11994
+ width: "12",
11995
+ height: "12",
11996
+ fill: "white",
11997
+ transform: "matrix(0.865865 0.500278 -0.871576 0.490261 131.318 90.4594)"
11998
+ }
11999
+ ) }),
12000
+ /* @__PURE__ */ jsxRuntime.jsx("clipPath", { id: "clip5_20915_38670", children: /* @__PURE__ */ jsxRuntime.jsx(
12001
+ "rect",
12002
+ {
12003
+ width: "12",
12004
+ height: "12",
12005
+ fill: "white",
12006
+ transform: "matrix(0.865865 0.500278 -0.871576 0.490261 141.709 96.4627)"
12007
+ }
12008
+ ) })
12009
+ ] })
12010
+ ]
12011
+ }
12012
+ );
12013
+ };
12014
+ const schema$1 = objectType({
12015
+ customer_id: stringType().min(1)
12016
+ });
12017
+ const SalesChannel = () => {
12018
+ const { id } = reactRouterDom.useParams();
12019
+ const { draft_order, isPending, isError, error } = useDraftOrder(
12020
+ id,
12021
+ {
12022
+ fields: "+sales_channel_id"
12023
+ },
12024
+ {
12025
+ enabled: !!id
12026
+ }
12027
+ );
12028
+ if (isError) {
12029
+ throw error;
12030
+ }
12031
+ const ISrEADY = !!draft_order && !isPending;
12032
+ return /* @__PURE__ */ jsxRuntime.jsxs(RouteDrawer, { children: [
12033
+ /* @__PURE__ */ jsxRuntime.jsxs(RouteDrawer.Header, { children: [
12034
+ /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Heading, { children: "Edit Sales Channel" }) }),
12035
+ /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Update which sales channel the draft order is associated with" }) })
12036
+ ] }),
12037
+ ISrEADY && /* @__PURE__ */ jsxRuntime.jsx(SalesChannelForm, { order: draft_order })
12038
+ ] });
12039
+ };
12040
+ const SalesChannelForm = ({ order }) => {
12041
+ const form = reactHookForm.useForm({
12042
+ defaultValues: {
12043
+ sales_channel_id: order.sales_channel_id || ""
12044
+ },
12045
+ resolver: zod.zodResolver(schema)
12046
+ });
12047
+ const { mutateAsync, isPending } = useUpdateDraftOrder(order.id);
12048
+ const { handleSuccess } = useRouteModal();
12049
+ const onSubmit = form.handleSubmit(async (data) => {
12050
+ await mutateAsync(
12051
+ {
12052
+ sales_channel_id: data.sales_channel_id
12053
+ },
12054
+ {
12055
+ onSuccess: () => {
12056
+ ui.toast.success("Sales channel updated");
12057
+ handleSuccess();
12058
+ },
12059
+ onError: (error) => {
12060
+ ui.toast.error(error.message);
12061
+ }
12062
+ }
12063
+ );
12064
+ });
12065
+ return /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Form, { form, children: /* @__PURE__ */ jsxRuntime.jsxs(
12066
+ KeyboundForm,
12067
+ {
12068
+ className: "flex flex-1 flex-col overflow-hidden",
12069
+ onSubmit,
12070
+ children: [
12071
+ /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Body, { className: "flex flex-col gap-y-6 overflow-y-auto", children: /* @__PURE__ */ jsxRuntime.jsx(SalesChannelField, { control: form.control, order }) }),
12072
+ /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Footer, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex justify-end gap-2", children: [
12073
+ /* @__PURE__ */ jsxRuntime.jsx(RouteDrawer.Close, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", variant: "secondary", children: "Cancel" }) }),
12074
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", type: "submit", isLoading: isPending, children: "Save" })
12075
+ ] }) })
12076
+ ]
12077
+ }
12078
+ ) });
12079
+ };
12080
+ const SalesChannelField = ({ control, order }) => {
12081
+ const salesChannels = useComboboxData({
12082
+ queryFn: async (params) => {
12083
+ return await sdk.admin.salesChannel.list(params);
12084
+ },
12085
+ queryKey: ["sales-channels"],
12086
+ getOptions: (data) => {
12087
+ return data.sales_channels.map((salesChannel) => ({
12088
+ label: salesChannel.name,
12089
+ value: salesChannel.id
12090
+ }));
12091
+ },
12092
+ defaultValue: order.sales_channel_id || void 0
12093
+ });
12094
+ return /* @__PURE__ */ jsxRuntime.jsx(
12095
+ Form$2.Field,
12096
+ {
12097
+ control,
12098
+ name: "sales_channel_id",
12099
+ render: ({ field }) => {
12100
+ return /* @__PURE__ */ jsxRuntime.jsxs(Form$2.Item, { children: [
12101
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.Label, { children: "Sales Channel" }),
12102
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(
12103
+ Combobox,
12104
+ {
12105
+ options: salesChannels.options,
12106
+ fetchNextPage: salesChannels.fetchNextPage,
12107
+ isFetchingNextPage: salesChannels.isFetchingNextPage,
12108
+ searchValue: salesChannels.searchValue,
12109
+ onSearchValueChange: salesChannels.onSearchValueChange,
12110
+ placeholder: "Select sales channel",
12111
+ ...field
12112
+ }
12113
+ ) }),
12114
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.ErrorMessage, {})
12115
+ ] });
12116
+ }
12117
+ }
12118
+ );
12119
+ };
12120
+ const schema = objectType({
12121
+ sales_channel_id: stringType().min(1)
12122
+ });
12123
+ const NumberInput = React.forwardRef(
12124
+ ({
12125
+ value,
12126
+ onChange,
12127
+ size = "base",
12128
+ min = 0,
12129
+ max = 100,
12130
+ step = 1,
12131
+ className,
12132
+ disabled,
12133
+ ...props
12134
+ }, ref) => {
12135
+ const handleChange = (event) => {
12136
+ const newValue = event.target.value === "" ? min : Number(event.target.value);
12137
+ if (!isNaN(newValue) && (max === void 0 || newValue <= max) && (min === void 0 || newValue >= min)) {
12138
+ onChange(newValue);
12139
+ }
12140
+ };
12141
+ const handleIncrement = () => {
12142
+ const newValue = value + step;
12143
+ if (max === void 0 || newValue <= max) {
12144
+ onChange(newValue);
12145
+ }
12146
+ };
12147
+ const handleDecrement = () => {
12148
+ const newValue = value - step;
12149
+ if (min === void 0 || newValue >= min) {
12150
+ onChange(newValue);
12151
+ }
12152
+ };
12153
+ return /* @__PURE__ */ jsxRuntime.jsxs(
12154
+ "div",
12155
+ {
12156
+ className: ui.clx(
12157
+ "inline-flex rounded-md bg-ui-bg-field shadow-borders-base overflow-hidden divide-x transition-fg",
12158
+ "[&:has(input:focus)]:shadow-borders-interactive-with-active",
12159
+ {
12160
+ "h-7": size === "small",
12161
+ "h-8": size === "base"
12162
+ },
12163
+ className
12164
+ ),
12165
+ children: [
12166
+ /* @__PURE__ */ jsxRuntime.jsx(
12167
+ "input",
12168
+ {
12169
+ ref,
12170
+ type: "number",
12171
+ value,
12172
+ onChange: handleChange,
12173
+ min,
12174
+ max,
12175
+ step,
12176
+ className: ui.clx(
12177
+ "flex-1 px-2 py-1 bg-transparent txt-compact-small text-ui-fg-base outline-none [appearance:textfield]",
12178
+ "[&::-webkit-outer-spin-button]:appearance-none [&::-webkit-inner-spin-button]:appearance-none",
12179
+ "placeholder:text-ui-fg-muted"
12180
+ ),
12181
+ ...props
12182
+ }
12183
+ ),
12184
+ /* @__PURE__ */ jsxRuntime.jsxs(
12185
+ "button",
12186
+ {
12187
+ className: ui.clx(
12188
+ "flex items-center justify-center outline-none transition-fg",
12189
+ "disabled:cursor-not-allowed disabled:text-ui-fg-muted",
12190
+ "focus:bg-ui-bg-field-component-hover",
12191
+ "hover:bg-ui-bg-field-component-hover",
12192
+ {
12193
+ "size-7": size === "small",
12194
+ "size-8": size === "base"
12195
+ }
12196
+ ),
12197
+ type: "button",
12198
+ onClick: handleDecrement,
12199
+ disabled: min !== void 0 && value <= min || disabled,
12200
+ children: [
12201
+ /* @__PURE__ */ jsxRuntime.jsx(icons.Minus, {}),
12202
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: `Decrease by ${step}` })
12203
+ ]
12204
+ }
12205
+ ),
12206
+ /* @__PURE__ */ jsxRuntime.jsxs(
12207
+ "button",
12208
+ {
12209
+ className: ui.clx(
12210
+ "flex items-center justify-center outline-none transition-fg",
12211
+ "disabled:cursor-not-allowed disabled:text-ui-fg-muted",
12212
+ "focus:bg-ui-bg-field-hover",
12213
+ "hover:bg-ui-bg-field-hover",
12214
+ {
12215
+ "size-7": size === "small",
12216
+ "size-8": size === "base"
12217
+ }
12218
+ ),
12219
+ type: "button",
12220
+ onClick: handleIncrement,
12221
+ disabled: max !== void 0 && value >= max || disabled,
12222
+ children: [
12223
+ /* @__PURE__ */ jsxRuntime.jsx(icons.Plus, {}),
12224
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: `Increase by ${step}` })
12225
+ ]
12226
+ }
12227
+ )
12228
+ ]
12229
+ }
12230
+ );
12231
+ }
12232
+ );
12233
+ const PRODUCT_VARIANTS_QUERY_KEY = "product-variants";
12234
+ const productVariantsQueryKeys = {
12235
+ list: (query2) => [
12236
+ PRODUCT_VARIANTS_QUERY_KEY,
12237
+ query2 ? query2 : void 0
12238
+ ]
12239
+ };
12240
+ const useProductVariants = (query2, options) => {
12241
+ const { data, ...rest } = reactQuery.useQuery({
12242
+ queryKey: productVariantsQueryKeys.list(query2),
12243
+ queryFn: async () => await sdk.admin.productVariant.list(query2),
12244
+ ...options
12245
+ });
12246
+ return { ...data, ...rest };
12247
+ };
12248
+ const STACKED_MODAL_ID = "items_stacked_modal";
12249
+ const Items = () => {
12250
+ const { id } = reactRouterDom.useParams();
12251
+ const {
12252
+ order: preview,
12253
+ isPending: isPreviewPending,
12254
+ isError: isPreviewError,
12255
+ error: previewError
12256
+ } = useOrderPreview(id, void 0, {
12257
+ placeholderData: reactQuery.keepPreviousData
12258
+ });
12259
+ useInitiateOrderEdit({ preview });
12260
+ const { draft_order, isPending, isError, error } = useDraftOrder(
12261
+ id,
12262
+ {
12263
+ fields: "currency_code"
12264
+ },
12265
+ {
12266
+ enabled: !!id
12267
+ }
12268
+ );
12269
+ const { onCancel } = useCancelOrderEdit({ preview });
12270
+ if (isError) {
12271
+ throw error;
12272
+ }
12273
+ if (isPreviewError) {
12274
+ throw previewError;
12275
+ }
12276
+ const ready = !!preview && !isPreviewPending && !!draft_order && !isPending;
12277
+ return /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal, { onClose: onCancel, children: ready ? /* @__PURE__ */ jsxRuntime.jsx(ItemsForm, { preview, currencyCode: draft_order.currency_code }) : /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
12278
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Edit Items" }) }),
12279
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Loading data for the draft order, please wait..." }) })
12280
+ ] }) });
12281
+ };
12282
+ const ItemsForm = ({ preview, currencyCode }) => {
12283
+ var _a;
12284
+ const [isSubmitting, setIsSubmitting] = React.useState(false);
12285
+ const [modalContent, setModalContent] = React.useState(
12286
+ null
12287
+ );
12288
+ const { handleSuccess } = useRouteModal();
12289
+ const { searchValue, onSearchValueChange, query: query2 } = useDebouncedSearch();
12290
+ const { mutateAsync: confirmOrderEdit } = useDraftOrderConfirmEdit(preview.id);
12291
+ const { mutateAsync: requestOrderEdit } = useDraftOrderRequestEdit(preview.id);
12292
+ const itemCount = ((_a = preview.items) == null ? void 0 : _a.reduce((acc, item) => acc + item.quantity, 0)) || 0;
12293
+ const matches = React.useMemo(() => {
12294
+ return matchSorter.matchSorter(preview.items, query2, {
12295
+ keys: ["product_title", "variant_title", "variant_sku", "title"]
12296
+ });
12297
+ }, [preview.items, query2]);
12298
+ const onSubmit = async () => {
12299
+ setIsSubmitting(true);
12300
+ let requestSucceeded = false;
12301
+ await requestOrderEdit(void 0, {
12302
+ onError: (e) => {
12303
+ ui.toast.error(`Failed to request order edit: ${e.message}`);
12304
+ },
12305
+ onSuccess: () => {
12306
+ requestSucceeded = true;
12307
+ }
12308
+ });
12309
+ if (!requestSucceeded) {
12310
+ setIsSubmitting(false);
12311
+ return;
12312
+ }
12313
+ await confirmOrderEdit(void 0, {
12314
+ onError: (e) => {
12315
+ ui.toast.error(`Failed to confirm order edit: ${e.message}`);
12316
+ },
12317
+ onSuccess: () => {
12318
+ handleSuccess();
12319
+ },
12320
+ onSettled: () => {
12321
+ setIsSubmitting(false);
12322
+ }
12323
+ });
12324
+ };
12325
+ const onKeyDown = React.useCallback(
12326
+ (e) => {
12327
+ if (e.key === "Enter" && (e.ctrlKey || e.metaKey)) {
12328
+ if (modalContent || isSubmitting) {
12329
+ return;
12330
+ }
12331
+ onSubmit();
12332
+ }
12333
+ },
12334
+ [modalContent, isSubmitting, onSubmit]
12335
+ );
12336
+ React.useEffect(() => {
12337
+ document.addEventListener("keydown", onKeyDown);
12338
+ return () => {
12339
+ document.removeEventListener("keydown", onKeyDown);
12340
+ };
12341
+ }, [onKeyDown]);
12342
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex h-full flex-col overflow-hidden", children: [
12343
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Header, {}),
12344
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Body, { className: "flex flex-1 flex-col overflow-hidden", children: /* @__PURE__ */ jsxRuntime.jsxs(
12345
+ StackedFocusModal,
12346
+ {
12347
+ id: STACKED_MODAL_ID,
12348
+ onOpenChangeCallback: (open) => {
12349
+ if (!open) {
12350
+ setModalContent(null);
12351
+ }
12352
+ },
12353
+ children: [
12354
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex flex-1 flex-col items-center overflow-y-auto", children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex w-full max-w-[720px] flex-col gap-y-6 px-6 py-16", children: [
12355
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
12356
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Heading, { children: "Edit Items" }) }),
12357
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", className: "text-ui-fg-subtle", children: "Edit the items in the draft order" }) })
12358
+ ] }),
12359
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Divider, { variant: "dashed" }),
12360
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col gap-y-6", children: [
12361
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "grid grid-cols-2 items-center gap-3", children: [
12362
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col", children: [
12363
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: "Items" }),
12364
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", className: "text-ui-fg-subtle", children: "Choose items from the product catalog." })
12365
+ ] }),
12366
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-2", children: [
12367
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex-1", children: /* @__PURE__ */ jsxRuntime.jsx(
12368
+ ui.Input,
12369
+ {
12370
+ type: "search",
12371
+ placeholder: "Search items",
12372
+ value: searchValue,
12373
+ onChange: (e) => onSearchValueChange(e.target.value)
12374
+ }
12375
+ ) }),
12376
+ /* @__PURE__ */ jsxRuntime.jsxs(ui.DropdownMenu, { children: [
12377
+ /* @__PURE__ */ jsxRuntime.jsx(ui.DropdownMenu.Trigger, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.IconButton, { type: "button", children: /* @__PURE__ */ jsxRuntime.jsx(icons.Plus, {}) }) }),
12378
+ /* @__PURE__ */ jsxRuntime.jsxs(ui.DropdownMenu.Content, { children: [
12379
+ /* @__PURE__ */ jsxRuntime.jsx(
12380
+ StackedModalTrigger,
12381
+ {
12382
+ type: "add-items",
12383
+ setModalContent
12384
+ }
12385
+ ),
12386
+ /* @__PURE__ */ jsxRuntime.jsx(
12387
+ StackedModalTrigger,
12388
+ {
12389
+ type: "add-custom-item",
12390
+ setModalContent
12391
+ }
12392
+ )
12393
+ ] })
12394
+ ] })
12395
+ ] })
12396
+ ] }),
12397
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-subtle shadow-elevation-card-rest rounded-xl", children: [
12398
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "px-[5px]", children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "text-ui-fg-muted grid grid-cols-[2fr_1fr_2fr_28px] gap-3 px-4 py-2", children: [
12399
+ /* @__PURE__ */ jsxRuntime.jsx("div", { children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: "Item" }) }),
12400
+ /* @__PURE__ */ jsxRuntime.jsx("div", { children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: "Quantity" }) }),
12401
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "text-right", children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: "Price" }) }),
12402
+ /* @__PURE__ */ jsxRuntime.jsx("div", {})
12403
+ ] }) }),
12404
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex flex-col gap-y-1.5 px-[5px] pb-[5px]", children: itemCount <= 0 ? /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-base shadow-elevation-card-rest flex flex-col items-center justify-center gap-1 gap-x-3 rounded-lg p-4", children: [
12405
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: "There are no items in this order" }),
12406
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", className: "text-ui-fg-subtle", children: "Add items to the order to get started." })
12407
+ ] }) : matches.length > 0 ? matches == null ? void 0 : matches.map((item) => /* @__PURE__ */ jsxRuntime.jsx(
12408
+ Item,
12409
+ {
12410
+ item,
12411
+ preview,
12412
+ currencyCode
12413
+ },
12414
+ item.id
12415
+ )) : /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-base shadow-elevation-card-rest flex flex-col items-center justify-center gap-1 gap-x-3 rounded-lg p-4", children: [
12416
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: "No items found" }),
12417
+ /* @__PURE__ */ jsxRuntime.jsxs(ui.Text, { size: "small", className: "text-ui-fg-subtle", children: [
12418
+ 'No items found for "',
12419
+ query2,
12420
+ '".'
12421
+ ] })
12422
+ ] }) })
12423
+ ] })
12424
+ ] }),
12425
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Divider, { variant: "dashed" }),
12426
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "grid grid-cols-[1fr_0.5fr_0.5fr] gap-3", children: [
12427
+ /* @__PURE__ */ jsxRuntime.jsx("div", { children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: "Subtotal" }) }),
12428
+ /* @__PURE__ */ jsxRuntime.jsx("div", { children: /* @__PURE__ */ jsxRuntime.jsxs(
12429
+ ui.Text,
12430
+ {
12431
+ size: "small",
12432
+ leading: "compact",
12433
+ className: "text-ui-fg-subtle",
12434
+ children: [
12435
+ itemCount,
12436
+ " ",
12437
+ itemCount === 1 ? "item" : "items"
12438
+ ]
12439
+ }
12440
+ ) }),
12441
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "text-right", children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: getStylizedAmount(preview.item_subtotal, currencyCode) }) })
12442
+ ] })
12443
+ ] }) }),
12444
+ modalContent && (modalContent === "add-items" ? /* @__PURE__ */ jsxRuntime.jsx(ExistingItemsForm, { orderId: preview.id, items: preview.items }) : modalContent === "add-custom-item" ? /* @__PURE__ */ jsxRuntime.jsx(
12445
+ CustomItemForm,
12446
+ {
12447
+ orderId: preview.id,
12448
+ currencyCode
12449
+ }
12450
+ ) : null)
12451
+ ]
12452
+ }
12453
+ ) }),
12454
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Footer, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center justify-end gap-x-2", children: [
12455
+ /* @__PURE__ */ jsxRuntime.jsx(RouteFocusModal.Close, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", variant: "secondary", type: "button", children: "Cancel" }) }),
12456
+ /* @__PURE__ */ jsxRuntime.jsx(
12457
+ ui.Button,
12458
+ {
12459
+ size: "small",
12460
+ type: "button",
12461
+ onClick: onSubmit,
12462
+ isLoading: isSubmitting,
12463
+ children: "Save"
12464
+ }
12465
+ )
12466
+ ] }) })
12467
+ ] });
12468
+ };
12469
+ const Item = ({ item, preview, currencyCode }) => {
12470
+ if (item.variant_id) {
12471
+ return /* @__PURE__ */ jsxRuntime.jsx(VariantItem, { item, preview, currencyCode });
12472
+ }
12473
+ return /* @__PURE__ */ jsxRuntime.jsx(CustomItem, { item, preview, currencyCode });
12474
+ };
12475
+ const VariantItem = ({ item, preview, currencyCode }) => {
12476
+ const [editing, setEditing] = React.useState(false);
12477
+ const form = reactHookForm.useForm({
12478
+ defaultValues: {
12479
+ quantity: item.quantity,
12480
+ unit_price: item.unit_price
12481
+ },
12482
+ resolver: zod.zodResolver(variantItemSchema)
12483
+ });
12484
+ const actionId = React.useMemo(() => {
12485
+ var _a, _b;
12486
+ return (_b = (_a = item.actions) == null ? void 0 : _a.find((a) => a.action === "ITEM_ADD")) == null ? void 0 : _b.id;
12487
+ }, [item]);
12488
+ const { mutateAsync: updateActionItem, isPending: isUpdatingActionItem } = useDraftOrderUpdateActionItem(preview.id);
12489
+ const { mutateAsync: updateOriginalItem, isPending: isUpdatingOriginalItem } = useDraftOrderUpdateItem(preview.id);
12490
+ const isPending = isUpdatingActionItem || isUpdatingOriginalItem;
12491
+ const onSubmit = form.handleSubmit(async (data) => {
12492
+ if (convertNumber(data.unit_price) === item.unit_price && data.quantity === item.quantity) {
12493
+ setEditing(false);
12494
+ return;
12495
+ }
12496
+ if (!actionId) {
12497
+ await updateOriginalItem(
12498
+ {
12499
+ item_id: item.id,
12500
+ quantity: data.quantity,
12501
+ unit_price: convertNumber(data.unit_price)
12502
+ },
12503
+ {
12504
+ onSuccess: () => {
12505
+ setEditing(false);
12506
+ },
12507
+ onError: (e) => {
12508
+ ui.toast.error(e.message);
12509
+ }
12510
+ }
12511
+ );
12512
+ return;
12513
+ }
12514
+ await updateActionItem(
12515
+ {
12516
+ action_id: actionId,
12517
+ quantity: data.quantity,
12518
+ unit_price: convertNumber(data.unit_price)
12519
+ },
12520
+ {
12521
+ onSuccess: () => {
12522
+ setEditing(false);
12523
+ },
12524
+ onError: (e) => {
12525
+ ui.toast.error(e.message);
12526
+ }
12527
+ }
12528
+ );
12529
+ });
12530
+ return /* @__PURE__ */ jsxRuntime.jsx(Form$2, { ...form, children: /* @__PURE__ */ jsxRuntime.jsx("form", { onSubmit, children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-base shadow-elevation-card-rest grid grid-cols-[minmax(0,2fr)_minmax(0,1fr)_minmax(0,2fr)_28px] items-center gap-3 rounded-lg px-4 py-2", children: [
12531
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex w-full items-center gap-x-3", children: [
12532
+ /* @__PURE__ */ jsxRuntime.jsx(
12533
+ Thumbnail,
12534
+ {
12535
+ thumbnail: item.thumbnail,
12536
+ alt: item.product_title ?? void 0
12537
+ }
12538
+ ),
12539
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col", children: [
12540
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-x-1", children: [
12541
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", leading: "compact", children: item.product_title }),
12542
+ /* @__PURE__ */ jsxRuntime.jsxs(
12543
+ ui.Text,
12544
+ {
12545
+ size: "small",
12546
+ leading: "compact",
12547
+ className: "text-ui-fg-subtle",
12548
+ children: [
12549
+ "(",
12550
+ item.variant_title,
12551
+ ")"
12552
+ ]
12553
+ }
12554
+ )
12555
+ ] }),
12556
+ /* @__PURE__ */ jsxRuntime.jsx(
12557
+ ui.Text,
12558
+ {
12559
+ size: "small",
12560
+ leading: "compact",
12561
+ className: "text-ui-fg-subtle",
12562
+ children: item.variant_sku
12563
+ }
12564
+ )
12565
+ ] })
12566
+ ] }),
12567
+ editing ? /* @__PURE__ */ jsxRuntime.jsx("div", { className: "w-full flex-1", children: /* @__PURE__ */ jsxRuntime.jsx(
12568
+ Form$2.Field,
12569
+ {
12570
+ control: form.control,
12571
+ name: "quantity",
12572
+ render: ({ field }) => {
12573
+ return /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(NumberInput, { ...field }) }) });
12574
+ }
12575
+ }
12576
+ ) }) : /* @__PURE__ */ jsxRuntime.jsx("div", { className: "w-full flex-1", children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: item.quantity }) }),
12577
+ editing ? /* @__PURE__ */ jsxRuntime.jsx("div", { className: "w-full flex-1", children: /* @__PURE__ */ jsxRuntime.jsx(
12578
+ Form$2.Field,
12579
+ {
12580
+ control: form.control,
12581
+ name: "unit_price",
12582
+ render: ({ field: { onChange, ...field } }) => {
12583
+ return /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(
12584
+ ui.CurrencyInput,
12585
+ {
12586
+ ...field,
12587
+ symbol: getNativeSymbol(currencyCode),
12588
+ code: currencyCode,
12589
+ onValueChange: (_value, _name, values) => onChange(values == null ? void 0 : values.value)
12590
+ }
12591
+ ) }) });
12592
+ }
12593
+ }
12594
+ ) }) : /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex w-full flex-1 items-center justify-end", children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: getLocaleAmount(item.unit_price, currencyCode) }) }),
12595
+ /* @__PURE__ */ jsxRuntime.jsx(
12596
+ ui.IconButton,
12597
+ {
12598
+ type: "button",
12599
+ size: "small",
12600
+ onClick: editing ? onSubmit : () => {
12601
+ setEditing(true);
12602
+ },
12603
+ disabled: isPending,
12604
+ children: editing ? /* @__PURE__ */ jsxRuntime.jsx(icons.Check, {}) : /* @__PURE__ */ jsxRuntime.jsx(icons.PencilSquare, {})
12605
+ }
12606
+ )
12607
+ ] }) }) });
12608
+ };
12609
+ const variantItemSchema = objectType({
12610
+ quantity: numberType(),
12611
+ unit_price: unionType([numberType(), stringType()])
12612
+ });
12613
+ const CustomItem = ({ item, preview, currencyCode }) => {
12614
+ const [editing, setEditing] = React.useState(false);
12615
+ const { quantity, unit_price, title } = item;
12616
+ const form = reactHookForm.useForm({
12617
+ defaultValues: {
12618
+ title,
12619
+ quantity,
12620
+ unit_price
12621
+ },
12622
+ resolver: zod.zodResolver(customItemSchema)
12623
+ });
12624
+ React.useEffect(() => {
12625
+ form.reset({
12626
+ title,
12627
+ quantity,
12628
+ unit_price
12629
+ });
12630
+ }, [form, title, quantity, unit_price]);
12631
+ const actionId = React.useMemo(() => {
12632
+ var _a, _b;
12633
+ return (_b = (_a = item.actions) == null ? void 0 : _a.find((a) => a.action === "ITEM_ADD")) == null ? void 0 : _b.id;
12634
+ }, [item]);
12635
+ const { mutateAsync: updateActionItem, isPending: isUpdatingActionItem } = useDraftOrderUpdateActionItem(preview.id);
12636
+ const { mutateAsync: removeActionItem, isPending: isRemovingActionItem } = useDraftOrderRemoveActionItem(preview.id);
12637
+ const { mutateAsync: updateOriginalItem, isPending: isUpdatingOriginalItem } = useDraftOrderUpdateItem(preview.id);
12638
+ const isPending = isUpdatingActionItem || isUpdatingOriginalItem;
12639
+ const onSubmit = form.handleSubmit(async (data) => {
12640
+ if (convertNumber(data.unit_price) === item.unit_price && data.quantity === item.quantity && data.title === item.title) {
12641
+ setEditing(false);
12642
+ return;
12643
+ }
12644
+ if (!actionId) {
12645
+ await updateOriginalItem(
12646
+ {
12647
+ item_id: item.id,
12648
+ quantity: data.quantity,
12649
+ unit_price: convertNumber(data.unit_price)
12650
+ },
12651
+ {
12652
+ onSuccess: () => {
12653
+ setEditing(false);
12654
+ },
12655
+ onError: (e) => {
12656
+ ui.toast.error(e.message);
12657
+ }
12658
+ }
12659
+ );
12660
+ return;
12661
+ }
12662
+ if (data.quantity === 0) {
12663
+ await removeActionItem(actionId, {
12664
+ onSuccess: () => {
12665
+ setEditing(false);
12666
+ },
12667
+ onError: (e) => {
12668
+ ui.toast.error(e.message);
12669
+ }
12670
+ });
12671
+ return;
12672
+ }
12673
+ await updateActionItem(
12674
+ {
12675
+ action_id: actionId,
12676
+ quantity: data.quantity,
12677
+ unit_price: convertNumber(data.unit_price)
12678
+ },
12679
+ {
12680
+ onSuccess: () => {
12681
+ setEditing(false);
12682
+ },
12683
+ onError: (e) => {
12684
+ ui.toast.error(e.message);
12685
+ }
12686
+ }
12687
+ );
12688
+ });
12689
+ return /* @__PURE__ */ jsxRuntime.jsx(Form$2, { ...form, children: /* @__PURE__ */ jsxRuntime.jsx("form", { onSubmit, children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "bg-ui-bg-base shadow-elevation-card-rest grid grid-cols-[minmax(0,2fr)_minmax(0,1fr)_minmax(0,2fr)_28px] items-center gap-3 rounded-lg px-4 py-2", children: [
12690
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-x-3", children: [
12691
+ /* @__PURE__ */ jsxRuntime.jsx(
12692
+ Thumbnail,
12693
+ {
12694
+ thumbnail: item.thumbnail,
12695
+ alt: item.title ?? void 0
12696
+ }
12697
+ ),
12698
+ editing ? /* @__PURE__ */ jsxRuntime.jsx(
12699
+ Form$2.Field,
12700
+ {
12701
+ control: form.control,
12702
+ name: "title",
12703
+ render: ({ field }) => {
12704
+ return /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(ui.Input, { ...field }) }) });
12705
+ }
12706
+ }
12707
+ ) : /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: item.title })
12708
+ ] }),
12709
+ editing ? /* @__PURE__ */ jsxRuntime.jsx(
12710
+ Form$2.Field,
12711
+ {
12712
+ control: form.control,
12713
+ name: "quantity",
12714
+ render: ({ field }) => {
12715
+ return /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(NumberInput, { ...field }) }) });
12716
+ }
12717
+ }
12718
+ ) : /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: item.quantity }),
12719
+ editing ? /* @__PURE__ */ jsxRuntime.jsx(
12720
+ Form$2.Field,
12721
+ {
12722
+ control: form.control,
12723
+ name: "unit_price",
12724
+ render: ({ field: { onChange, ...field } }) => {
12725
+ return /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(
12726
+ ui.CurrencyInput,
12727
+ {
12728
+ ...field,
12729
+ symbol: getNativeSymbol(currencyCode),
12730
+ code: currencyCode,
12731
+ onValueChange: (_value, _name, values) => onChange(values == null ? void 0 : values.value)
12732
+ }
12733
+ ) }) });
12734
+ }
12735
+ }
12736
+ ) : /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex flex-1 items-center justify-end", children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", weight: "plus", children: getLocaleAmount(item.unit_price, currencyCode) }) }),
12737
+ /* @__PURE__ */ jsxRuntime.jsx(
12738
+ ui.IconButton,
12739
+ {
12740
+ type: "button",
12741
+ size: "small",
12742
+ onClick: editing ? onSubmit : () => {
12743
+ setEditing(true);
12744
+ },
12745
+ disabled: isPending,
12746
+ children: editing ? /* @__PURE__ */ jsxRuntime.jsx(icons.Check, {}) : /* @__PURE__ */ jsxRuntime.jsx(icons.PencilSquare, {})
12747
+ }
12748
+ )
12749
+ ] }) }) });
12750
+ };
12751
+ const StackedModalTrigger = ({
12752
+ type,
12753
+ setModalContent
12754
+ }) => {
12755
+ const { setIsOpen } = useStackedModal();
12756
+ const onClick = React.useCallback(() => {
12757
+ setModalContent(type);
12758
+ setIsOpen(STACKED_MODAL_ID, true);
12759
+ }, [setModalContent, setIsOpen, type]);
12760
+ return /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Trigger, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.DropdownMenu.Item, { onClick, children: type === "add-items" ? "Add items" : "Add custom item" }) });
12761
+ };
12762
+ const VARIANT_PREFIX = "items";
12763
+ const LIMIT = 50;
12764
+ const ExistingItemsForm = ({ orderId, items }) => {
12765
+ const { setIsOpen } = useStackedModal();
12766
+ const [rowSelection, setRowSelection] = React.useState(
12767
+ items.reduce((acc, item) => {
12768
+ acc[item.variant_id] = true;
12769
+ return acc;
12770
+ }, {})
12771
+ );
12772
+ React.useEffect(() => {
12773
+ setRowSelection(
12774
+ items.reduce((acc, item) => {
12775
+ if (item.variant_id) {
12776
+ acc[item.variant_id] = true;
12777
+ }
12778
+ return acc;
12779
+ }, {})
12780
+ );
12781
+ }, [items]);
12782
+ const { q, order, offset } = useQueryParams(
12783
+ ["q", "order", "offset"],
12784
+ VARIANT_PREFIX
12785
+ );
12786
+ const { variants, count, isPending, isError, error } = useProductVariants(
12787
+ {
12788
+ q,
12789
+ order,
12790
+ offset: offset ? parseInt(offset) : void 0,
12791
+ limit: LIMIT
12792
+ },
12793
+ {
12794
+ placeholderData: reactQuery.keepPreviousData
12795
+ }
12796
+ );
12797
+ const columns = useColumns();
12798
+ const { mutateAsync } = useDraftOrderAddItems(orderId);
12799
+ const onSubmit = async () => {
12800
+ const ids = Object.keys(rowSelection).filter(
12801
+ (id) => !items.find((i) => i.variant_id === id)
12802
+ );
12803
+ await mutateAsync(
12804
+ {
12805
+ items: ids.map((id) => ({
12806
+ variant_id: id,
12807
+ quantity: 1
12808
+ }))
12809
+ },
12810
+ {
12811
+ onSuccess: () => {
12812
+ setRowSelection({});
12813
+ setIsOpen(STACKED_MODAL_ID, false);
12814
+ },
12815
+ onError: (e) => {
12816
+ ui.toast.error(e.message);
12817
+ }
12818
+ }
12819
+ );
12820
+ };
12821
+ if (isError) {
12822
+ throw error;
12823
+ }
12824
+ return /* @__PURE__ */ jsxRuntime.jsxs(
12825
+ StackedFocusModal.Content,
12826
+ {
12827
+ onOpenAutoFocus: (e) => {
12828
+ e.preventDefault();
12829
+ const searchInput = document.querySelector(
12830
+ "[data-modal-id='modal-search-input']"
12831
+ );
12832
+ if (searchInput) {
12833
+ searchInput.focus();
12834
+ }
12835
+ },
12836
+ children: [
12837
+ /* @__PURE__ */ jsxRuntime.jsxs(StackedFocusModal.Header, { children: [
12838
+ /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Product Variants" }) }),
12839
+ /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "sr-only", children: "Choose product variants to add to the order." }) })
12840
+ ] }),
12841
+ /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Body, { className: "flex-1 overflow-hidden", children: /* @__PURE__ */ jsxRuntime.jsx(
12842
+ DataTable,
12843
+ {
12844
+ data: variants,
12845
+ columns,
12846
+ isLoading: isPending,
12847
+ getRowId: (row) => row.id,
12848
+ rowCount: count,
12849
+ prefix: VARIANT_PREFIX,
12850
+ layout: "fill",
12851
+ rowSelection: {
12852
+ state: rowSelection,
12853
+ onRowSelectionChange: setRowSelection,
12854
+ enableRowSelection: (row) => {
12855
+ return !items.find((i) => i.variant_id === row.original.id);
12856
+ }
12857
+ },
12858
+ autoFocusSearch: true
12859
+ }
12860
+ ) }),
12861
+ /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Footer, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center justify-end gap-x-2", children: [
12862
+ /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Close, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", variant: "secondary", type: "button", children: "Cancel" }) }),
12863
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", type: "button", onClick: onSubmit, children: "Update items" })
12864
+ ] }) })
12865
+ ]
12866
+ }
12867
+ );
12868
+ };
12869
+ const columnHelper = ui.createDataTableColumnHelper();
12870
+ const useColumns = () => {
12871
+ return React.useMemo(() => {
12872
+ return [
12873
+ columnHelper.select(),
12874
+ columnHelper.accessor("product.title", {
12875
+ header: "Product",
12876
+ cell: ({ row }) => {
12877
+ var _a, _b, _c;
12878
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-x-2", children: [
12879
+ /* @__PURE__ */ jsxRuntime.jsx(
12880
+ Thumbnail,
12881
+ {
12882
+ thumbnail: (_a = row.original.product) == null ? void 0 : _a.thumbnail,
12883
+ alt: (_b = row.original.product) == null ? void 0 : _b.title
12884
+ }
12885
+ ),
12886
+ /* @__PURE__ */ jsxRuntime.jsx("span", { children: (_c = row.original.product) == null ? void 0 : _c.title })
12887
+ ] });
12888
+ },
12889
+ enableSorting: true
12890
+ }),
12891
+ columnHelper.accessor("title", {
12892
+ header: "Variant",
12893
+ enableSorting: true
12894
+ }),
12895
+ columnHelper.accessor("sku", {
12896
+ header: "SKU",
12897
+ cell: ({ getValue }) => {
12898
+ return getValue() ?? "-";
12899
+ },
12900
+ enableSorting: true
12901
+ }),
12902
+ columnHelper.accessor("updated_at", {
12903
+ header: "Updated",
12904
+ cell: ({ getValue }) => {
12905
+ return /* @__PURE__ */ jsxRuntime.jsx(
12906
+ ui.Tooltip,
12907
+ {
12908
+ content: getFullDate({ date: getValue(), includeTime: true }),
12909
+ children: /* @__PURE__ */ jsxRuntime.jsx("span", { children: getFullDate({ date: getValue() }) })
13029
12910
  }
13030
- ) }),
13031
- /* @__PURE__ */ jsxRuntime.jsx("clipPath", { id: "clip5_20915_38670", children: /* @__PURE__ */ jsxRuntime.jsx(
13032
- "rect",
12911
+ );
12912
+ },
12913
+ enableSorting: true,
12914
+ sortAscLabel: "Oldest first",
12915
+ sortDescLabel: "Newest first"
12916
+ }),
12917
+ columnHelper.accessor("created_at", {
12918
+ header: "Created",
12919
+ cell: ({ getValue }) => {
12920
+ return /* @__PURE__ */ jsxRuntime.jsx(
12921
+ ui.Tooltip,
13033
12922
  {
13034
- width: "12",
13035
- height: "12",
13036
- fill: "white",
13037
- transform: "matrix(0.865865 0.500278 -0.871576 0.490261 141.709 96.4627)"
12923
+ content: getFullDate({ date: getValue(), includeTime: true }),
12924
+ children: /* @__PURE__ */ jsxRuntime.jsx("span", { children: getFullDate({ date: getValue() }) })
13038
12925
  }
13039
- ) })
13040
- ] })
13041
- ]
13042
- }
13043
- );
12926
+ );
12927
+ },
12928
+ enableSorting: true,
12929
+ sortAscLabel: "Oldest first",
12930
+ sortDescLabel: "Newest first"
12931
+ })
12932
+ ];
12933
+ }, []);
13044
12934
  };
13045
- const schema = objectType({
13046
- customer_id: stringType().min(1)
12935
+ const CustomItemForm = ({ orderId, currencyCode }) => {
12936
+ const { setIsOpen } = useStackedModal();
12937
+ const { mutateAsync: addItems } = useDraftOrderAddItems(orderId);
12938
+ const form = reactHookForm.useForm({
12939
+ defaultValues: {
12940
+ title: "",
12941
+ quantity: 1,
12942
+ unit_price: ""
12943
+ },
12944
+ resolver: zod.zodResolver(customItemSchema)
12945
+ });
12946
+ const onSubmit = form.handleSubmit(async (data) => {
12947
+ await addItems(
12948
+ {
12949
+ items: [
12950
+ {
12951
+ title: data.title,
12952
+ quantity: data.quantity,
12953
+ unit_price: convertNumber(data.unit_price)
12954
+ }
12955
+ ]
12956
+ },
12957
+ {
12958
+ onSuccess: () => {
12959
+ setIsOpen(STACKED_MODAL_ID, false);
12960
+ },
12961
+ onError: (e) => {
12962
+ ui.toast.error(e.message);
12963
+ }
12964
+ }
12965
+ );
12966
+ });
12967
+ return /* @__PURE__ */ jsxRuntime.jsx(Form$2, { ...form, children: /* @__PURE__ */ jsxRuntime.jsx(KeyboundForm, { onSubmit, children: /* @__PURE__ */ jsxRuntime.jsxs(StackedFocusModal.Content, { children: [
12968
+ /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Header, {}),
12969
+ /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Body, { className: "flex flex-1 flex-col overflow-hidden", children: /* @__PURE__ */ jsxRuntime.jsx("div", { className: "flex flex-1 flex-col items-center overflow-y-auto", children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex w-full max-w-[720px] flex-col gap-y-6 px-2 py-16", children: [
12970
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
12971
+ /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Title, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Heading, { children: "Add custom item" }) }),
12972
+ /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Description, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { size: "small", className: "text-ui-fg-subtle", children: "Add a custom item to the order. This will add a new line item that is not associated with an existing product." }) })
12973
+ ] }),
12974
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Divider, { variant: "dashed" }),
12975
+ /* @__PURE__ */ jsxRuntime.jsx(
12976
+ Form$2.Field,
12977
+ {
12978
+ control: form.control,
12979
+ name: "title",
12980
+ render: ({ field }) => /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "grid grid-cols-2 gap-x-3", children: [
12981
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
12982
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.Label, { children: "Title" }),
12983
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.Hint, { children: "Enter the title of the item" })
12984
+ ] }),
12985
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
12986
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(ui.Input, { ...field }) }),
12987
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.ErrorMessage, {})
12988
+ ] })
12989
+ ] }) })
12990
+ }
12991
+ ),
12992
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Divider, { variant: "dashed" }),
12993
+ /* @__PURE__ */ jsxRuntime.jsx(
12994
+ Form$2.Field,
12995
+ {
12996
+ control: form.control,
12997
+ name: "unit_price",
12998
+ render: ({ field: { onChange, ...field } }) => /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "grid grid-cols-2 gap-x-3", children: [
12999
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
13000
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.Label, { children: "Unit price" }),
13001
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.Hint, { children: "Enter the unit price of the item" })
13002
+ ] }),
13003
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
13004
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx(
13005
+ ui.CurrencyInput,
13006
+ {
13007
+ symbol: getNativeSymbol(currencyCode),
13008
+ code: currencyCode,
13009
+ onValueChange: (_value, _name, values) => onChange(values == null ? void 0 : values.value),
13010
+ ...field
13011
+ }
13012
+ ) }),
13013
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.ErrorMessage, {})
13014
+ ] })
13015
+ ] }) })
13016
+ }
13017
+ ),
13018
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Divider, { variant: "dashed" }),
13019
+ /* @__PURE__ */ jsxRuntime.jsx(
13020
+ Form$2.Field,
13021
+ {
13022
+ control: form.control,
13023
+ name: "quantity",
13024
+ render: ({ field }) => /* @__PURE__ */ jsxRuntime.jsx(Form$2.Item, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "grid grid-cols-2 gap-x-3", children: [
13025
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
13026
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.Label, { children: "Quantity" }),
13027
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.Hint, { children: "Enter the quantity of the item" })
13028
+ ] }),
13029
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "w-full flex-1", children: [
13030
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.Control, { children: /* @__PURE__ */ jsxRuntime.jsx("div", { className: "w-full flex-1", children: /* @__PURE__ */ jsxRuntime.jsx(NumberInput, { ...field, className: "w-full" }) }) }),
13031
+ /* @__PURE__ */ jsxRuntime.jsx(Form$2.ErrorMessage, {})
13032
+ ] })
13033
+ ] }) })
13034
+ }
13035
+ )
13036
+ ] }) }) }),
13037
+ /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Footer, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center justify-end gap-x-2", children: [
13038
+ /* @__PURE__ */ jsxRuntime.jsx(StackedFocusModal.Close, { asChild: true, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", variant: "secondary", type: "button", children: "Cancel" }) }),
13039
+ /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { size: "small", type: "button", onClick: onSubmit, children: "Add item" })
13040
+ ] }) })
13041
+ ] }) }) });
13042
+ };
13043
+ const customItemSchema = objectType({
13044
+ title: stringType().min(1),
13045
+ quantity: numberType(),
13046
+ unit_price: unionType([numberType(), stringType()])
13047
13047
  });
13048
13048
  const widgetModule = { widgets: [] };
13049
13049
  const routeModule = {
@@ -13077,21 +13077,13 @@ const routeModule = {
13077
13077
  Component: Email,
13078
13078
  path: "/draft-orders/:id/email"
13079
13079
  },
13080
- {
13081
- Component: Items,
13082
- path: "/draft-orders/:id/items"
13083
- },
13084
- {
13085
- Component: Metadata,
13086
- path: "/draft-orders/:id/metadata"
13087
- },
13088
13080
  {
13089
13081
  Component: Promotions,
13090
13082
  path: "/draft-orders/:id/promotions"
13091
13083
  },
13092
13084
  {
13093
- Component: SalesChannel,
13094
- path: "/draft-orders/:id/sales-channel"
13085
+ Component: Metadata,
13086
+ path: "/draft-orders/:id/metadata"
13095
13087
  },
13096
13088
  {
13097
13089
  Component: Shipping,
@@ -13104,6 +13096,14 @@ const routeModule = {
13104
13096
  {
13105
13097
  Component: TransferOwnership,
13106
13098
  path: "/draft-orders/:id/transfer-ownership"
13099
+ },
13100
+ {
13101
+ Component: SalesChannel,
13102
+ path: "/draft-orders/:id/sales-channel"
13103
+ },
13104
+ {
13105
+ Component: Items,
13106
+ path: "/draft-orders/:id/items"
13107
13107
  }
13108
13108
  ]
13109
13109
  }