@feedspring/framer-google-reviews
Version:
224 lines • 106 kB
JavaScript
import { p as h, k as L, b as U, f as w, c as S, d as Q, e as D, s as T, n as j, r as k, g as M, h as O, l as A, j as b, W as r, u as P, m as V, i as f } from "./Welcome-8909f8da.js";
import { ControlType as I } from "framer";
import { c as E, C as K } from "./Card-38a3ddfb.js";
import { C as q, a as _ } from "./ArrowRight-cc825733.js";
import "react";
function R() {
return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop).";
}
const Zg = {
feedId: h,
items: L,
skip: U,
fontSettings: {
type: I.Object,
title: "Font",
controls: w
},
containerSettings: {
type: I.Object,
title: "Container",
controls: S
},
cardSettings: {
type: I.Object,
title: "Card",
controls: E()
},
averageRatingSettings: {
type: I.Object,
title: "Average Rating",
controls: Q
},
avatarSettings: {
type: I.Object,
title: "Avatar",
controls: D()
},
starsSettings: {
type: I.Object,
title: "Stars",
controls: T
},
nameSettings: {
type: I.Object,
title: "Name",
controls: j()
},
reviewSettings: {
type: I.Object,
title: "Review",
controls: k
},
dateSettings: {
type: I.Object,
title: "Date",
controls: M()
},
separatorSettings: {
type: I.Object,
title: "Separator",
controls: O()
},
slider: {
type: I.Object,
title: "Slider",
controls: {
gap: {
type: I.Number,
defaultValue: 16,
min: 0,
max: 100,
step: 1,
displayStepper: !0
},
showLeftArrow: {
type: I.Boolean,
title: "Show Left Arrow",
defaultValue: !0
},
showRightArrow: {
type: I.Boolean,
title: "Show Right Arrow",
defaultValue: !0
},
arrowSize: {
type: I.Number,
defaultValue: 18,
min: 16,
max: 48,
step: 2,
displayStepper: !0
},
arrowColor: {
type: I.Color,
defaultValue: "#ffffff"
},
arrowBackground: {
type: I.Color,
defaultValue: "#000000"
},
arrowRadius: {
type: I.String,
defaultValue: "50%",
title: "Arrow Radius"
},
visibleItems: {
type: I.Number,
title: "Visible Slides",
defaultValue: 3,
min: 1,
max: 10,
step: 1,
displayStepper: !0
},
infiniteScroll: {
type: I.Boolean,
title: "Infinite Scroll",
defaultValue: !0
}
}
},
customArrows: {
type: I.Boolean,
title: "Custom Arrows",
defaultValue: !1
},
leftArrow: {
type: I.ComponentInstance,
title: "Left Arrow Component",
hidden: (g) => !g.customArrows
},
rightArrow: {
type: I.ComponentInstance,
title: "Right Arrow Component",
hidden: (g) => !g.customArrows
}
}, dg = (g) => {
if (!g.settings.feedId)
return /* @__PURE__ */ b(r, {});
const {
feed: c,
items: H
} = P(g.settings.feedId, g.settings.items, g.settings.skip);
if (!c)
return null;
const l = H || [], a = g.settings.averageRatingSettings.visible && c.extra && c.extra.total, {
gap: F = 16,
visibleItems: v = 3,
showLeftArrow: N,
showRightArrow: Y,
arrowSize: G,
arrowColor: X,
arrowBackground: x,
arrowRadius: y,
infiniteScroll: W
} = g.settings.slider, J = Math.max(0, F || 16), Z = Math.max(1, v || 3), d = g.settings.customArrows || !1, t = g.settings.leftArrow, p = g.settings.rightArrow, m = `slider-${g.settings.feedId}-${globalThis.Date.now()}`, s = () => {
const C = document.getElementById(m);
if (!C || l.length === 0)
return;
const B = C.scrollWidth / l.length * Z, i = C.scrollLeft <= 1;
W && i ? C.scrollTo({
left: C.scrollWidth - C.offsetWidth,
behavior: "smooth"
}) : C.scrollBy({
left: -B,
behavior: "smooth"
});
}, z = () => {
const C = document.getElementById(m);
if (!C || l.length === 0)
return;
const B = C.scrollWidth / l.length * Z, i = C.scrollWidth - C.offsetWidth, e = C.scrollLeft >= i - 1;
W && e ? C.scrollTo({
left: 0,
behavior: "smooth"
}) : C.scrollBy({
left: B,
behavior: "smooth"
});
}, o = l;
return /* @__PURE__ */ V($, { family: g.settings.fontSettings.family, children: [
/* @__PURE__ */ V(gg, { children: [
N && /* @__PURE__ */ b(n, { onClick: s, size: G, color: X, background: x, radius: y, isCustom: d, children: d ? t : /* @__PURE__ */ b(q, { settings: {
size: G,
color: X
} }) }),
/* @__PURE__ */ b(Ig, { children: /* @__PURE__ */ b(Cg, { id: m, visibleItems: Z, gap: J, children: o.map((C, u) => /* @__PURE__ */ b(bg, { visibleItems: Z, gap: J, children: /* @__PURE__ */ b(K, { review: C, extra: c.extra, cardSettings: g.settings.cardSettings, avatarSettings: g.settings.avatarSettings, starsSettings: g.settings.starsSettings, nameSettings: g.settings.nameSettings, reviewSettings: g.settings.reviewSettings, dateSettings: g.settings.dateSettings, separatorSettings: g.settings.separatorSettings }, C.id) }, u)) }) }),
Y && /* @__PURE__ */ b(n, { onClick: z, size: G, color: X, background: x, radius: y, isCustom: d, children: d ? p : /* @__PURE__ */ b(_, { settings: {
size: G,
color: X
} }) })
] }),
a && /* @__PURE__ */ b(f, { total: c.extra.total, averageRating: c.extra.averageRating, settings: g.settings.averageRatingSettings })
] });
}, $ = /* @__PURE__ */ A("section", process.env.NODE_ENV === "production" ? {
target: "epdbbyj5"
} : {
target: "epdbbyj5",
label: "SliderSection"
})("display:flex;flex-direction:column;align-items:center;width:100%;font-family:", (g) => g.family, ",sans-serif;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/feedspring-scripts/feedspring-scripts/apps/framer-google-reviews/src/lib/layouts/BasicSlider.tsx"],"names":[],"mappings":"AA2VwD","file":"/home/runner/work/feedspring-scripts/feedspring-scripts/apps/framer-google-reviews/src/lib/layouts/BasicSlider.tsx","sourcesContent":["import React from 'react'\nimport styled from '@emotion/styled'\nimport { useFeedState } from '../hooks/useFeedState'\n\nimport {\n  ControlType,\n  propertyControlFeedId,\n  propertyControlFeedItems,\n  propertyControlFeedSkip,\n  PropertyControls,\n} from '@feedspring/framer-helper'\n\nimport { Welcome } from '../components/Welcome'\nimport * as Card from '../components/basic-card/Card'\nimport * as AverageRating from '../components/AverageRating'\nimport * as Section from '../components/Section'\nimport * as Avatar from '../components/Avatar'\nimport * as Stars from '../components/Stars'\nimport * as Name from '../components/Name'\nimport * as Review from '../components/Review'\nimport * as Date from '../components/Date'\nimport * as Separator from '../components/Separator'\nimport * as ArrowLeft from '../components/arrows/ArrowLeft'\nimport * as ArrowRight from '../components/arrows/ArrowRight'\n\nexport interface Settings {\n  feedId: string\n  items: number\n  skip: number\n  fontSettings: Section.FontSettings\n  containerSettings: Section.ContainerSettings\n  cardSettings: Card.CardSettings\n  averageRatingSettings: AverageRating.AverageRatingSettings\n  avatarSettings: Avatar.AvatarSettings\n  starsSettings: Stars.StarsSettings\n  nameSettings: Name.NameSettings\n  reviewSettings: Review.ReviewSettings\n  dateSettings: Date.DateSettings\n  separatorSettings: Separator.SeparatorSettings\n  slider: SliderSettings\n  customArrows?: boolean\n  leftArrow?: React.ReactNode\n  rightArrow?: React.ReactNode\n}\n\nexport interface SliderSettings {\n  gap: number\n  showLeftArrow: boolean\n  showRightArrow: boolean\n  arrowSize: number\n  arrowColor: string\n  arrowBackground: string\n  arrowRadius: string\n  visibleItems: number\n  infiniteScroll: boolean\n}\n\nexport const propertyControls: PropertyControls<Settings> = {\n  feedId: propertyControlFeedId,\n  items: propertyControlFeedItems,\n  skip: propertyControlFeedSkip,\n  fontSettings: {\n    type: ControlType.Object,\n    title: 'Font',\n    controls: Section.fontPropertyControls,\n  },\n  containerSettings: {\n    type: ControlType.Object,\n    title: 'Container',\n    controls: Section.containerPropertyControls,\n  },\n  cardSettings: {\n    type: ControlType.Object,\n    title: 'Card',\n    controls: Card.cardPropertyControls(),\n  },\n  averageRatingSettings: {\n    type: ControlType.Object,\n    title: 'Average Rating',\n    controls: AverageRating.averageRatingPropertyControls,\n  },\n  avatarSettings: {\n    type: ControlType.Object,\n    title: 'Avatar',\n    controls: Avatar.avatarPropertyControls(),\n  },\n  starsSettings: {\n    type: ControlType.Object,\n    title: 'Stars',\n    controls: Stars.starsPropertyControls,\n  },\n  nameSettings: {\n    type: ControlType.Object,\n    title: 'Name',\n    controls: Name.namePropertyControls(),\n  },\n  reviewSettings: {\n    type: ControlType.Object,\n    title: 'Review',\n    controls: Review.reviewPropertyControls,\n  },\n  dateSettings: {\n    type: ControlType.Object,\n    title: 'Date',\n    controls: Date.datePropertyControls(),\n  },\n  separatorSettings: {\n    type: ControlType.Object,\n    title: 'Separator',\n    controls: Separator.separatorPropertyControls(),\n  },\n  slider: {\n    type: ControlType.Object,\n    title: 'Slider',\n    controls: {\n      gap: {\n        type: ControlType.Number,\n        defaultValue: 16,\n        min: 0,\n        max: 100,\n        step: 1,\n        displayStepper: true,\n      },\n      showLeftArrow: {\n        type: ControlType.Boolean,\n        title: 'Show Left Arrow',\n        defaultValue: true,\n      },\n      showRightArrow: {\n        type: ControlType.Boolean,\n        title: 'Show Right Arrow',\n        defaultValue: true,\n      },\n      arrowSize: {\n        type: ControlType.Number,\n        defaultValue: 18,\n        min: 16,\n        max: 48,\n        step: 2,\n        displayStepper: true,\n      },\n      arrowColor: {\n        type: ControlType.Color,\n        defaultValue: '#ffffff',\n      },\n      arrowBackground: {\n        type: ControlType.Color,\n        defaultValue: '#000000',\n      },\n      arrowRadius: {\n        type: ControlType.String,\n        defaultValue: '50%',\n        title: 'Arrow Radius',\n      },\n      visibleItems: {\n        type: ControlType.Number,\n        title: 'Visible Slides',\n        defaultValue: 3,\n        min: 1,\n        max: 10,\n        step: 1,\n        displayStepper: true,\n      },\n      infiniteScroll: {\n        type: ControlType.Boolean,\n        title: 'Infinite Scroll',\n        defaultValue: true,\n      },\n    },\n  },\n  customArrows: {\n    type: ControlType.Boolean,\n    title: 'Custom Arrows',\n    defaultValue: false,\n  },\n  leftArrow: {\n    type: ControlType.ComponentInstance,\n    title: 'Left Arrow Component',\n    hidden: (props: Settings) => !props.customArrows,\n  },\n  rightArrow: {\n    type: ControlType.ComponentInstance,\n    title: 'Right Arrow Component',\n    hidden: (props: Settings) => !props.customArrows,\n  },\n}\n\ninterface Props {\n  settings: Settings\n}\n\nexport const Component: React.FC<Props> = (props) => {\n  if (!props.settings.feedId) return <Welcome />\n\n  const { feed, items } = useFeedState(\n    props.settings.feedId,\n    props.settings.items,\n    props.settings.skip\n  )\n\n  if (!feed) return null\n\n  const itemsArray = items || []\n\n  const isAverageRatingVisible =\n    props.settings.averageRatingSettings.visible &&\n    feed.extra &&\n    feed.extra.total\n\n  const {\n    gap = 16,\n    visibleItems = 3,\n    showLeftArrow,\n    showRightArrow,\n    arrowSize,\n    arrowColor,\n    arrowBackground,\n    arrowRadius,\n    infiniteScroll,\n  } = props.settings.slider\n\n  const safeGap = Math.max(0, gap || 16)\n  const safeVisibleItems = Math.max(1, visibleItems || 3)\n\n  const isCustomArrows = props.settings.customArrows || false\n  const customLeftArrow = props.settings.leftArrow\n  const customRightArrow = props.settings.rightArrow\n\n  const sliderId = `slider-${props.settings.feedId}-${globalThis.Date.now()}`\n\n  const goToPreviousSlide = () => {\n    const container = document.getElementById(sliderId)\n    if (!container || itemsArray.length === 0) return\n\n    const slideWidth = container.scrollWidth / itemsArray.length\n    const scrollAmount = slideWidth * safeVisibleItems\n    const isAtStart = container.scrollLeft <= 1\n\n    if (infiniteScroll && isAtStart) {\n      container.scrollTo({\n        left: container.scrollWidth - container.offsetWidth,\n        behavior: 'smooth',\n      })\n    } else {\n      container.scrollBy({ left: -scrollAmount, behavior: 'smooth' })\n    }\n  }\n\n  const goToNextSlide = () => {\n    const container = document.getElementById(sliderId)\n    if (!container || itemsArray.length === 0) return\n\n    const slideWidth = container.scrollWidth / itemsArray.length\n    const scrollAmount = slideWidth * safeVisibleItems\n    const maxScroll = container.scrollWidth - container.offsetWidth\n    const isAtEnd = container.scrollLeft >= maxScroll - 1\n\n    if (infiniteScroll && isAtEnd) {\n      container.scrollTo({ left: 0, behavior: 'smooth' })\n    } else {\n      container.scrollBy({ left: scrollAmount, behavior: 'smooth' })\n    }\n  }\n\n  const allSlides = itemsArray\n\n  return (\n    <SliderSection family={props.settings.fontSettings.family}>\n      <SliderWrapper>\n        {showLeftArrow && (\n          <ArrowButton\n            onClick={goToPreviousSlide}\n            size={arrowSize}\n            color={arrowColor}\n            background={arrowBackground}\n            radius={arrowRadius}\n            isCustom={isCustomArrows}\n          >\n            {isCustomArrows ? (\n              customLeftArrow\n            ) : (\n              <ArrowLeft.Component\n                settings={{ size: arrowSize, color: arrowColor }}\n              />\n            )}\n          </ArrowButton>\n        )}\n\n        <Viewport>\n          <ScrollContainer\n            id={sliderId}\n            visibleItems={safeVisibleItems}\n            gap={safeGap}\n          >\n            {allSlides.map((item, idx) => (\n              <SlideItem\n                key={idx}\n                visibleItems={safeVisibleItems}\n                gap={safeGap}\n              >\n                <Card.Component\n                  key={item.id}\n                  review={item}\n                  extra={feed.extra!}\n                  cardSettings={props.settings.cardSettings}\n                  avatarSettings={props.settings.avatarSettings}\n                  starsSettings={props.settings.starsSettings}\n                  nameSettings={props.settings.nameSettings}\n                  reviewSettings={props.settings.reviewSettings}\n                  dateSettings={props.settings.dateSettings}\n                  separatorSettings={props.settings.separatorSettings}\n                />\n              </SlideItem>\n            ))}\n          </ScrollContainer>\n        </Viewport>\n\n        {showRightArrow && (\n          <ArrowButton\n            onClick={goToNextSlide}\n            size={arrowSize}\n            color={arrowColor}\n            background={arrowBackground}\n            radius={arrowRadius}\n            isCustom={isCustomArrows}\n          >\n            {isCustomArrows ? (\n              customRightArrow\n            ) : (\n              <ArrowRight.Component\n                settings={{ size: arrowSize, color: arrowColor }}\n              />\n            )}\n          </ArrowButton>\n        )}\n      </SliderWrapper>\n      {isAverageRatingVisible && (\n        <AverageRating.Component\n          total={feed.extra!.total}\n          averageRating={feed.extra!.averageRating}\n          settings={props.settings.averageRatingSettings}\n        />\n      )}\n    </SliderSection>\n  )\n}\n\nconst SliderSection = styled.section<{ family: string }>`\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  width: 100%;\n\n  font-family: ${(p) => p.family}, sans-serif;\n`\n\nconst SliderWrapper = styled.div`\n  position: relative;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  gap: 12px;\n`\n\nconst Viewport = styled.div`\n  overflow: hidden;\n  flex: 1;\n  width: 100%;\n  position: relative;\n`\n\nconst ScrollContainer = styled.div<{\n  visibleItems: number\n  gap: number\n}>`\n  display: flex;\n  gap: ${(p) => p.gap}px;\n  overflow-x: auto;\n  scroll-behavior: smooth;\n  scroll-snap-type: x mandatory;\n  scrollbar-width: none;\n  -ms-overflow-style: none;\n\n  &::-webkit-scrollbar {\n    display: none;\n  }\n`\n\nconst ArrowButton = styled.button<{\n  size: number\n  color: string\n  background: string\n  radius: string\n  isCustom?: boolean\n}>`\n  width: ${(p) => (p.isCustom ? 'auto' : p.size + 16 + 'px')};\n  height: ${(p) => (p.isCustom ? 'auto' : p.size + 16 + 'px')};\n  border-radius: ${(p) => (p.isCustom ? '0' : p.radius)};\n  border: none;\n  background: ${(p) => (p.isCustom ? 'transparent' : p.background)};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  cursor: pointer;\n  flex-shrink: 0;\n  transition: opacity 0.2s ease;\n  padding: ${(p) => (p.isCustom ? '0' : 'initial')};\n\n  &:hover:not(:disabled) {\n    opacity: 0.8;\n  }\n`\n\nconst SlideItem = styled.div<{\n  visibleItems: number\n  gap: number\n}>`\n  width: calc(\n    (100% - ${(p) => (p.visibleItems - 1) * p.gap}px) / ${(p) => p.visibleItems}\n  );\n  min-width: calc(\n    (100% - ${(p) => (p.visibleItems - 1) * p.gap}px) / ${(p) => p.visibleItems}\n  );\n  box-sizing: border-box;\n  flex-shrink: 0;\n  display: flex;\n  flex-direction: column;\n  scroll-snap-align: start;\n\n  > * {\n    flex: 1;\n    height: 100%;\n  }\n`\n"]} */")), gg = /* @__PURE__ */ A("div", process.env.NODE_ENV === "production" ? {
target: "epdbbyj4"
} : {
target: "epdbbyj4",
label: "SliderWrapper"
})(process.env.NODE_ENV === "production" ? {
name: "bbnbhy",
styles: "position:relative;display:flex;align-items:center;width:100%;gap:12px"
} : {
name: "bbnbhy",
styles: "position:relative;display:flex;align-items:center;width:100%;gap:12px",
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/feedspring-scripts/feedspring-scripts/apps/framer-google-reviews/src/lib/layouts/BasicSlider.tsx"],"names":[],"mappings":"AAoWgC","file":"/home/runner/work/feedspring-scripts/feedspring-scripts/apps/framer-google-reviews/src/lib/layouts/BasicSlider.tsx","sourcesContent":["import React from 'react'\nimport styled from '@emotion/styled'\nimport { useFeedState } from '../hooks/useFeedState'\n\nimport {\n  ControlType,\n  propertyControlFeedId,\n  propertyControlFeedItems,\n  propertyControlFeedSkip,\n  PropertyControls,\n} from '@feedspring/framer-helper'\n\nimport { Welcome } from '../components/Welcome'\nimport * as Card from '../components/basic-card/Card'\nimport * as AverageRating from '../components/AverageRating'\nimport * as Section from '../components/Section'\nimport * as Avatar from '../components/Avatar'\nimport * as Stars from '../components/Stars'\nimport * as Name from '../components/Name'\nimport * as Review from '../components/Review'\nimport * as Date from '../components/Date'\nimport * as Separator from '../components/Separator'\nimport * as ArrowLeft from '../components/arrows/ArrowLeft'\nimport * as ArrowRight from '../components/arrows/ArrowRight'\n\nexport interface Settings {\n  feedId: string\n  items: number\n  skip: number\n  fontSettings: Section.FontSettings\n  containerSettings: Section.ContainerSettings\n  cardSettings: Card.CardSettings\n  averageRatingSettings: AverageRating.AverageRatingSettings\n  avatarSettings: Avatar.AvatarSettings\n  starsSettings: Stars.StarsSettings\n  nameSettings: Name.NameSettings\n  reviewSettings: Review.ReviewSettings\n  dateSettings: Date.DateSettings\n  separatorSettings: Separator.SeparatorSettings\n  slider: SliderSettings\n  customArrows?: boolean\n  leftArrow?: React.ReactNode\n  rightArrow?: React.ReactNode\n}\n\nexport interface SliderSettings {\n  gap: number\n  showLeftArrow: boolean\n  showRightArrow: boolean\n  arrowSize: number\n  arrowColor: string\n  arrowBackground: string\n  arrowRadius: string\n  visibleItems: number\n  infiniteScroll: boolean\n}\n\nexport const propertyControls: PropertyControls<Settings> = {\n  feedId: propertyControlFeedId,\n  items: propertyControlFeedItems,\n  skip: propertyControlFeedSkip,\n  fontSettings: {\n    type: ControlType.Object,\n    title: 'Font',\n    controls: Section.fontPropertyControls,\n  },\n  containerSettings: {\n    type: ControlType.Object,\n    title: 'Container',\n    controls: Section.containerPropertyControls,\n  },\n  cardSettings: {\n    type: ControlType.Object,\n    title: 'Card',\n    controls: Card.cardPropertyControls(),\n  },\n  averageRatingSettings: {\n    type: ControlType.Object,\n    title: 'Average Rating',\n    controls: AverageRating.averageRatingPropertyControls,\n  },\n  avatarSettings: {\n    type: ControlType.Object,\n    title: 'Avatar',\n    controls: Avatar.avatarPropertyControls(),\n  },\n  starsSettings: {\n    type: ControlType.Object,\n    title: 'Stars',\n    controls: Stars.starsPropertyControls,\n  },\n  nameSettings: {\n    type: ControlType.Object,\n    title: 'Name',\n    controls: Name.namePropertyControls(),\n  },\n  reviewSettings: {\n    type: ControlType.Object,\n    title: 'Review',\n    controls: Review.reviewPropertyControls,\n  },\n  dateSettings: {\n    type: ControlType.Object,\n    title: 'Date',\n    controls: Date.datePropertyControls(),\n  },\n  separatorSettings: {\n    type: ControlType.Object,\n    title: 'Separator',\n    controls: Separator.separatorPropertyControls(),\n  },\n  slider: {\n    type: ControlType.Object,\n    title: 'Slider',\n    controls: {\n      gap: {\n        type: ControlType.Number,\n        defaultValue: 16,\n        min: 0,\n        max: 100,\n        step: 1,\n        displayStepper: true,\n      },\n      showLeftArrow: {\n        type: ControlType.Boolean,\n        title: 'Show Left Arrow',\n        defaultValue: true,\n      },\n      showRightArrow: {\n        type: ControlType.Boolean,\n        title: 'Show Right Arrow',\n        defaultValue: true,\n      },\n      arrowSize: {\n        type: ControlType.Number,\n        defaultValue: 18,\n        min: 16,\n        max: 48,\n        step: 2,\n        displayStepper: true,\n      },\n      arrowColor: {\n        type: ControlType.Color,\n        defaultValue: '#ffffff',\n      },\n      arrowBackground: {\n        type: ControlType.Color,\n        defaultValue: '#000000',\n      },\n      arrowRadius: {\n        type: ControlType.String,\n        defaultValue: '50%',\n        title: 'Arrow Radius',\n      },\n      visibleItems: {\n        type: ControlType.Number,\n        title: 'Visible Slides',\n        defaultValue: 3,\n        min: 1,\n        max: 10,\n        step: 1,\n        displayStepper: true,\n      },\n      infiniteScroll: {\n        type: ControlType.Boolean,\n        title: 'Infinite Scroll',\n        defaultValue: true,\n      },\n    },\n  },\n  customArrows: {\n    type: ControlType.Boolean,\n    title: 'Custom Arrows',\n    defaultValue: false,\n  },\n  leftArrow: {\n    type: ControlType.ComponentInstance,\n    title: 'Left Arrow Component',\n    hidden: (props: Settings) => !props.customArrows,\n  },\n  rightArrow: {\n    type: ControlType.ComponentInstance,\n    title: 'Right Arrow Component',\n    hidden: (props: Settings) => !props.customArrows,\n  },\n}\n\ninterface Props {\n  settings: Settings\n}\n\nexport const Component: React.FC<Props> = (props) => {\n  if (!props.settings.feedId) return <Welcome />\n\n  const { feed, items } = useFeedState(\n    props.settings.feedId,\n    props.settings.items,\n    props.settings.skip\n  )\n\n  if (!feed) return null\n\n  const itemsArray = items || []\n\n  const isAverageRatingVisible =\n    props.settings.averageRatingSettings.visible &&\n    feed.extra &&\n    feed.extra.total\n\n  const {\n    gap = 16,\n    visibleItems = 3,\n    showLeftArrow,\n    showRightArrow,\n    arrowSize,\n    arrowColor,\n    arrowBackground,\n    arrowRadius,\n    infiniteScroll,\n  } = props.settings.slider\n\n  const safeGap = Math.max(0, gap || 16)\n  const safeVisibleItems = Math.max(1, visibleItems || 3)\n\n  const isCustomArrows = props.settings.customArrows || false\n  const customLeftArrow = props.settings.leftArrow\n  const customRightArrow = props.settings.rightArrow\n\n  const sliderId = `slider-${props.settings.feedId}-${globalThis.Date.now()}`\n\n  const goToPreviousSlide = () => {\n    const container = document.getElementById(sliderId)\n    if (!container || itemsArray.length === 0) return\n\n    const slideWidth = container.scrollWidth / itemsArray.length\n    const scrollAmount = slideWidth * safeVisibleItems\n    const isAtStart = container.scrollLeft <= 1\n\n    if (infiniteScroll && isAtStart) {\n      container.scrollTo({\n        left: container.scrollWidth - container.offsetWidth,\n        behavior: 'smooth',\n      })\n    } else {\n      container.scrollBy({ left: -scrollAmount, behavior: 'smooth' })\n    }\n  }\n\n  const goToNextSlide = () => {\n    const container = document.getElementById(sliderId)\n    if (!container || itemsArray.length === 0) return\n\n    const slideWidth = container.scrollWidth / itemsArray.length\n    const scrollAmount = slideWidth * safeVisibleItems\n    const maxScroll = container.scrollWidth - container.offsetWidth\n    const isAtEnd = container.scrollLeft >= maxScroll - 1\n\n    if (infiniteScroll && isAtEnd) {\n      container.scrollTo({ left: 0, behavior: 'smooth' })\n    } else {\n      container.scrollBy({ left: scrollAmount, behavior: 'smooth' })\n    }\n  }\n\n  const allSlides = itemsArray\n\n  return (\n    <SliderSection family={props.settings.fontSettings.family}>\n      <SliderWrapper>\n        {showLeftArrow && (\n          <ArrowButton\n            onClick={goToPreviousSlide}\n            size={arrowSize}\n            color={arrowColor}\n            background={arrowBackground}\n            radius={arrowRadius}\n            isCustom={isCustomArrows}\n          >\n            {isCustomArrows ? (\n              customLeftArrow\n            ) : (\n              <ArrowLeft.Component\n                settings={{ size: arrowSize, color: arrowColor }}\n              />\n            )}\n          </ArrowButton>\n        )}\n\n        <Viewport>\n          <ScrollContainer\n            id={sliderId}\n            visibleItems={safeVisibleItems}\n            gap={safeGap}\n          >\n            {allSlides.map((item, idx) => (\n              <SlideItem\n                key={idx}\n                visibleItems={safeVisibleItems}\n                gap={safeGap}\n              >\n                <Card.Component\n                  key={item.id}\n                  review={item}\n                  extra={feed.extra!}\n                  cardSettings={props.settings.cardSettings}\n                  avatarSettings={props.settings.avatarSettings}\n                  starsSettings={props.settings.starsSettings}\n                  nameSettings={props.settings.nameSettings}\n                  reviewSettings={props.settings.reviewSettings}\n                  dateSettings={props.settings.dateSettings}\n                  separatorSettings={props.settings.separatorSettings}\n                />\n              </SlideItem>\n            ))}\n          </ScrollContainer>\n        </Viewport>\n\n        {showRightArrow && (\n          <ArrowButton\n            onClick={goToNextSlide}\n            size={arrowSize}\n            color={arrowColor}\n            background={arrowBackground}\n            radius={arrowRadius}\n            isCustom={isCustomArrows}\n          >\n            {isCustomArrows ? (\n              customRightArrow\n            ) : (\n              <ArrowRight.Component\n                settings={{ size: arrowSize, color: arrowColor }}\n              />\n            )}\n          </ArrowButton>\n        )}\n      </SliderWrapper>\n      {isAverageRatingVisible && (\n        <AverageRating.Component\n          total={feed.extra!.total}\n          averageRating={feed.extra!.averageRating}\n          settings={props.settings.averageRatingSettings}\n        />\n      )}\n    </SliderSection>\n  )\n}\n\nconst SliderSection = styled.section<{ family: string }>`\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  width: 100%;\n\n  font-family: ${(p) => p.family}, sans-serif;\n`\n\nconst SliderWrapper = styled.div`\n  position: relative;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  gap: 12px;\n`\n\nconst Viewport = styled.div`\n  overflow: hidden;\n  flex: 1;\n  width: 100%;\n  position: relative;\n`\n\nconst ScrollContainer = styled.div<{\n  visibleItems: number\n  gap: number\n}>`\n  display: flex;\n  gap: ${(p) => p.gap}px;\n  overflow-x: auto;\n  scroll-behavior: smooth;\n  scroll-snap-type: x mandatory;\n  scrollbar-width: none;\n  -ms-overflow-style: none;\n\n  &::-webkit-scrollbar {\n    display: none;\n  }\n`\n\nconst ArrowButton = styled.button<{\n  size: number\n  color: string\n  background: string\n  radius: string\n  isCustom?: boolean\n}>`\n  width: ${(p) => (p.isCustom ? 'auto' : p.size + 16 + 'px')};\n  height: ${(p) => (p.isCustom ? 'auto' : p.size + 16 + 'px')};\n  border-radius: ${(p) => (p.isCustom ? '0' : p.radius)};\n  border: none;\n  background: ${(p) => (p.isCustom ? 'transparent' : p.background)};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  cursor: pointer;\n  flex-shrink: 0;\n  transition: opacity 0.2s ease;\n  padding: ${(p) => (p.isCustom ? '0' : 'initial')};\n\n  &:hover:not(:disabled) {\n    opacity: 0.8;\n  }\n`\n\nconst SlideItem = styled.div<{\n  visibleItems: number\n  gap: number\n}>`\n  width: calc(\n    (100% - ${(p) => (p.visibleItems - 1) * p.gap}px) / ${(p) => p.visibleItems}\n  );\n  min-width: calc(\n    (100% - ${(p) => (p.visibleItems - 1) * p.gap}px) / ${(p) => p.visibleItems}\n  );\n  box-sizing: border-box;\n  flex-shrink: 0;\n  display: flex;\n  flex-direction: column;\n  scroll-snap-align: start;\n\n  > * {\n    flex: 1;\n    height: 100%;\n  }\n`\n"]} */",
toString: R
}), Ig = /* @__PURE__ */ A("div", process.env.NODE_ENV === "production" ? {
target: "epdbbyj3"
} : {
target: "epdbbyj3",
label: "Viewport"
})(process.env.NODE_ENV === "production" ? {
name: "fguyqj",
styles: "overflow:hidden;flex:1;width:100%;position:relative"
} : {
name: "fguyqj",
styles: "overflow:hidden;flex:1;width:100%;position:relative",
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL2ZlZWRzcHJpbmctc2NyaXB0cy9mZWVkc3ByaW5nLXNjcmlwdHMvYXBwcy9mcmFtZXItZ29vZ2xlLXJldmlld3Mvc3JjL2xpYi9sYXlvdXRzL0Jhc2ljU2xpZGVyLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUE0VzJCIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL2ZlZWRzcHJpbmctc2NyaXB0cy9mZWVkc3ByaW5nLXNjcmlwdHMvYXBwcy9mcmFtZXItZ29vZ2xlLXJldmlld3Mvc3JjL2xpYi9sYXlvdXRzL0Jhc2ljU2xpZGVyLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHsgdXNlRmVlZFN0YXRlIH0gZnJvbSAnLi4vaG9va3MvdXNlRmVlZFN0YXRlJ1xuXG5pbXBvcnQge1xuICBDb250cm9sVHlwZSxcbiAgcHJvcGVydHlDb250cm9sRmVlZElkLFxuICBwcm9wZXJ0eUNvbnRyb2xGZWVkSXRlbXMsXG4gIHByb3BlcnR5Q29udHJvbEZlZWRTa2lwLFxuICBQcm9wZXJ0eUNvbnRyb2xzLFxufSBmcm9tICdAZmVlZHNwcmluZy9mcmFtZXItaGVscGVyJ1xuXG5pbXBvcnQgeyBXZWxjb21lIH0gZnJvbSAnLi4vY29tcG9uZW50cy9XZWxjb21lJ1xuaW1wb3J0ICogYXMgQ2FyZCBmcm9tICcuLi9jb21wb25lbnRzL2Jhc2ljLWNhcmQvQ2FyZCdcbmltcG9ydCAqIGFzIEF2ZXJhZ2VSYXRpbmcgZnJvbSAnLi4vY29tcG9uZW50cy9BdmVyYWdlUmF0aW5nJ1xuaW1wb3J0ICogYXMgU2VjdGlvbiBmcm9tICcuLi9jb21wb25lbnRzL1NlY3Rpb24nXG5pbXBvcnQgKiBhcyBBdmF0YXIgZnJvbSAnLi4vY29tcG9uZW50cy9BdmF0YXInXG5pbXBvcnQgKiBhcyBTdGFycyBmcm9tICcuLi9jb21wb25lbnRzL1N0YXJzJ1xuaW1wb3J0ICogYXMgTmFtZSBmcm9tICcuLi9jb21wb25lbnRzL05hbWUnXG5pbXBvcnQgKiBhcyBSZXZpZXcgZnJvbSAnLi4vY29tcG9uZW50cy9SZXZpZXcnXG5pbXBvcnQgKiBhcyBEYXRlIGZyb20gJy4uL2NvbXBvbmVudHMvRGF0ZSdcbmltcG9ydCAqIGFzIFNlcGFyYXRvciBmcm9tICcuLi9jb21wb25lbnRzL1NlcGFyYXRvcidcbmltcG9ydCAqIGFzIEFycm93TGVmdCBmcm9tICcuLi9jb21wb25lbnRzL2Fycm93cy9BcnJvd0xlZnQnXG5pbXBvcnQgKiBhcyBBcnJvd1JpZ2h0IGZyb20gJy4uL2NvbXBvbmVudHMvYXJyb3dzL0Fycm93UmlnaHQnXG5cbmV4cG9ydCBpbnRlcmZhY2UgU2V0dGluZ3Mge1xuICBmZWVkSWQ6IHN0cmluZ1xuICBpdGVtczogbnVtYmVyXG4gIHNraXA6IG51bWJlclxuICBmb250U2V0dGluZ3M6IFNlY3Rpb24uRm9udFNldHRpbmdzXG4gIGNvbnRhaW5lclNldHRpbmdzOiBTZWN0aW9uLkNvbnRhaW5lclNldHRpbmdzXG4gIGNhcmRTZXR0aW5nczogQ2FyZC5DYXJkU2V0dGluZ3NcbiAgYXZlcmFnZVJhdGluZ1NldHRpbmdzOiBBdmVyYWdlUmF0aW5nLkF2ZXJhZ2VSYXRpbmdTZXR0aW5nc1xuICBhdmF0YXJTZXR0aW5nczogQXZhdGFyLkF2YXRhclNldHRpbmdzXG4gIHN0YXJzU2V0dGluZ3M6IFN0YXJzLlN0YXJzU2V0dGluZ3NcbiAgbmFtZVNldHRpbmdzOiBOYW1lLk5hbWVTZXR0aW5nc1xuICByZXZpZXdTZXR0aW5nczogUmV2aWV3LlJldmlld1NldHRpbmdzXG4gIGRhdGVTZXR0aW5nczogRGF0ZS5EYXRlU2V0dGluZ3NcbiAgc2VwYXJhdG9yU2V0dGluZ3M6IFNlcGFyYXRvci5TZXBhcmF0b3JTZXR0aW5nc1xuICBzbGlkZXI6IFNsaWRlclNldHRpbmdzXG4gIGN1c3RvbUFycm93cz86IGJvb2xlYW5cbiAgbGVmdEFycm93PzogUmVhY3QuUmVhY3ROb2RlXG4gIHJpZ2h0QXJyb3c/OiBSZWFjdC5SZWFjdE5vZGVcbn1cblxuZXhwb3J0IGludGVyZmFjZSBTbGlkZXJTZXR0aW5ncyB7XG4gIGdhcDogbnVtYmVyXG4gIHNob3dMZWZ0QXJyb3c6IGJvb2xlYW5cbiAgc2hvd1JpZ2h0QXJyb3c6IGJvb2xlYW5cbiAgYXJyb3dTaXplOiBudW1iZXJcbiAgYXJyb3dDb2xvcjogc3RyaW5nXG4gIGFycm93QmFja2dyb3VuZDogc3RyaW5nXG4gIGFycm93UmFkaXVzOiBzdHJpbmdcbiAgdmlzaWJsZUl0ZW1zOiBudW1iZXJcbiAgaW5maW5pdGVTY3JvbGw6IGJvb2xlYW5cbn1cblxuZXhwb3J0IGNvbnN0IHByb3BlcnR5Q29udHJvbHM6IFByb3BlcnR5Q29udHJvbHM8U2V0dGluZ3M+ID0ge1xuICBmZWVkSWQ6IHByb3BlcnR5Q29udHJvbEZlZWRJZCxcbiAgaXRlbXM6IHByb3BlcnR5Q29udHJvbEZlZWRJdGVtcyxcbiAgc2tpcDogcHJvcGVydHlDb250cm9sRmVlZFNraXAsXG4gIGZvbnRTZXR0aW5nczoge1xuICAgIHR5cGU6IENvbnRyb2xUeXBlLk9iamVjdCxcbiAgICB0aXRsZTogJ0ZvbnQnLFxuICAgIGNvbnRyb2xzOiBTZWN0aW9uLmZvbnRQcm9wZXJ0eUNvbnRyb2xzLFxuICB9LFxuICBjb250YWluZXJTZXR0aW5nczoge1xuICAgIHR5cGU6IENvbnRyb2xUeXBlLk9iamVjdCxcbiAgICB0aXRsZTogJ0NvbnRhaW5lcicsXG4gICAgY29udHJvbHM6IFNlY3Rpb24uY29udGFpbmVyUHJvcGVydHlDb250cm9scyxcbiAgfSxcbiAgY2FyZFNldHRpbmdzOiB7XG4gICAgdHlwZTogQ29udHJvbFR5cGUuT2JqZWN0LFxuICAgIHRpdGxlOiAnQ2FyZCcsXG4gICAgY29udHJvbHM6IENhcmQuY2FyZFByb3BlcnR5Q29udHJvbHMoKSxcbiAgfSxcbiAgYXZlcmFnZVJhdGluZ1NldHRpbmdzOiB7XG4gICAgdHlwZTogQ29udHJvbFR5cGUuT2JqZWN0LFxuICAgIHRpdGxlOiAnQXZlcmFnZSBSYXRpbmcnLFxuICAgIGNvbnRyb2xzOiBBdmVyYWdlUmF0aW5nLmF2ZXJhZ2VSYXRpbmdQcm9wZXJ0eUNvbnRyb2xzLFxuICB9LFxuICBhdmF0YXJTZXR0aW5nczoge1xuICAgIHR5cGU6IENvbnRyb2xUeXBlLk9iamVjdCxcbiAgICB0aXRsZTogJ0F2YXRhcicsXG4gICAgY29udHJvbHM6IEF2YXRhci5hdmF0YXJQcm9wZXJ0eUNvbnRyb2xzKCksXG4gIH0sXG4gIHN0YXJzU2V0dGluZ3M6IHtcbiAgICB0eXBlOiBDb250cm9sVHlwZS5PYmplY3QsXG4gICAgdGl0bGU6ICdTdGFycycsXG4gICAgY29udHJvbHM6IFN0YXJzLnN0YXJzUHJvcGVydHlDb250cm9scyxcbiAgfSxcbiAgbmFtZVNldHRpbmdzOiB7XG4gICAgdHlwZTogQ29udHJvbFR5cGUuT2JqZWN0LFxuICAgIHRpdGxlOiAnTmFtZScsXG4gICAgY29udHJvbHM6IE5hbWUubmFtZVByb3BlcnR5Q29udHJvbHMoKSxcbiAgfSxcbiAgcmV2aWV3U2V0dGluZ3M6IHtcbiAgICB0eXBlOiBDb250cm9sVHlwZS5PYmplY3QsXG4gICAgdGl0bGU6ICdSZXZpZXcnLFxuICAgIGNvbnRyb2xzOiBSZXZpZXcucmV2aWV3UHJvcGVydHlDb250cm9scyxcbiAgfSxcbiAgZGF0ZVNldHRpbmdzOiB7XG4gICAgdHlwZTogQ29udHJvbFR5cGUuT2JqZWN0LFxuICAgIHRpdGxlOiAnRGF0ZScsXG4gICAgY29udHJvbHM6IERhdGUuZGF0ZVByb3BlcnR5Q29udHJvbHMoKSxcbiAgfSxcbiAgc2VwYXJhdG9yU2V0dGluZ3M6IHtcbiAgICB0eXBlOiBDb250cm9sVHlwZS5PYmplY3QsXG4gICAgdGl0bGU6ICdTZXBhcmF0b3InLFxuICAgIGNvbnRyb2xzOiBTZXBhcmF0b3Iuc2VwYXJhdG9yUHJvcGVydHlDb250cm9scygpLFxuICB9LFxuICBzbGlkZXI6IHtcbiAgICB0eXBlOiBDb250cm9sVHlwZS5PYmplY3QsXG4gICAgdGl0bGU6ICdTbGlkZXInLFxuICAgIGNvbnRyb2xzOiB7XG4gICAgICBnYXA6IHtcbiAgICAgICAgdHlwZTogQ29udHJvbFR5cGUuTnVtYmVyLFxuICAgICAgICBkZWZhdWx0VmFsdWU6IDE2LFxuICAgICAgICBtaW46IDAsXG4gICAgICAgIG1heDogMTAwLFxuICAgICAgICBzdGVwOiAxLFxuICAgICAgICBkaXNwbGF5U3RlcHBlcjogdHJ1ZSxcbiAgICAgIH0sXG4gICAgICBzaG93TGVmdEFycm93OiB7XG4gICAgICAgIHR5cGU6IENvbnRyb2xUeXBlLkJvb2xlYW4sXG4gICAgICAgIHRpdGxlOiAnU2hvdyBMZWZ0IEFycm93JyxcbiAgICAgICAgZGVmYXVsdFZhbHVlOiB0cnVlLFxuICAgICAgfSxcbiAgICAgIHNob3dSaWdodEFycm93OiB7XG4gICAgICAgIHR5cGU6IENvbnRyb2xUeXBlLkJvb2xlYW4sXG4gICAgICAgIHRpdGxlOiAnU2hvdyBSaWdodCBBcnJvdycsXG4gICAgICAgIGRlZmF1bHRWYWx1ZTogdHJ1ZSxcbiAgICAgIH0sXG4gICAgICBhcnJvd1NpemU6IHtcbiAgICAgICAgdHlwZTogQ29udHJvbFR5cGUuTnVtYmVyLFxuICAgICAgICBkZWZhdWx0VmFsdWU6IDE4LFxuICAgICAgICBtaW46IDE2LFxuICAgICAgICBtYXg6IDQ4LFxuICAgICAgICBzdGVwOiAyLFxuICAgICAgICBkaXNwbGF5U3RlcHBlcjogdHJ1ZSxcbiAgICAgIH0sXG4gICAgICBhcnJvd0NvbG9yOiB7XG4gICAgICAgIHR5cGU6IENvbnRyb2xUeXBlLkNvbG9yLFxuICAgICAgICBkZWZhdWx0VmFsdWU6ICcjZmZmZmZmJyxcbiAgICAgIH0sXG4gICAgICBhcnJvd0JhY2tncm91bmQ6IHtcbiAgICAgICAgdHlwZTogQ29udHJvbFR5cGUuQ29sb3IsXG4gICAgICAgIGRlZmF1bHRWYWx1ZTogJyMwMDAwMDAnLFxuICAgICAgfSxcbiAgICAgIGFycm93UmFkaXVzOiB7XG4gICAgICAgIHR5cGU6IENvbnRyb2xUeXBlLlN0cmluZyxcbiAgICAgICAgZGVmYXVsdFZhbHVlOiAnNTAlJyxcbiAgICAgICAgdGl0bGU6ICdBcnJvdyBSYWRpdXMnLFxuICAgICAgfSxcbiAgICAgIHZpc2libGVJdGVtczoge1xuICAgICAgICB0eXBlOiBDb250cm9sVHlwZS5OdW1iZXIsXG4gICAgICAgIHRpdGxlOiAnVmlzaWJsZSBTbGlkZXMnLFxuICAgICAgICBkZWZhdWx0VmFsdWU6IDMsXG4gICAgICAgIG1pbjogMSxcbiAgICAgICAgbWF4OiAxMCxcbiAgICAgICAgc3RlcDogMSxcbiAgICAgICAgZGlzcGxheVN0ZXBwZXI6IHRydWUsXG4gICAgICB9LFxuICAgICAgaW5maW5pdGVTY3JvbGw6IHtcbiAgICAgICAgdHlwZTogQ29udHJvbFR5cGUuQm9vbGVhbixcbiAgICAgICAgdGl0bGU6ICdJbmZpbml0ZSBTY3JvbGwnLFxuICAgICAgICBkZWZhdWx0VmFsdWU6IHRydWUsXG4gICAgICB9LFxuICAgIH0sXG4gIH0sXG4gIGN1c3RvbUFycm93czoge1xuICAgIHR5cGU6IENvbnRyb2xUeXBlLkJvb2xlYW4sXG4gICAgdGl0bGU6ICdDdXN0b20gQXJyb3dzJyxcbiAgICBkZWZhdWx0VmFsdWU6IGZhbHNlLFxuICB9LFxuICBsZWZ0QXJyb3c6IHtcbiAgICB0eXBlOiBDb250cm9sVHlwZS5Db21wb25lbnRJbnN0YW5jZSxcbiAgICB0aXRsZTogJ0xlZnQgQXJyb3cgQ29tcG9uZW50JyxcbiAgICBoaWRkZW46IChwcm9wczogU2V0dGluZ3MpID0+ICFwcm9wcy5jdXN0b21BcnJvd3MsXG4gIH0sXG4gIHJpZ2h0QXJyb3c6IHtcbiAgICB0eXBlOiBDb250cm9sVHlwZS5Db21wb25lbnRJbnN0YW5jZSxcbiAgICB0aXRsZTogJ1JpZ2h0IEFycm93IENvbXBvbmVudCcsXG4gICAgaGlkZGVuOiAocHJvcHM6IFNldHRpbmdzKSA9PiAhcHJvcHMuY3VzdG9tQXJyb3dzLFxuICB9LFxufVxuXG5pbnRlcmZhY2UgUHJvcHMge1xuICBzZXR0aW5nczogU2V0dGluZ3Ncbn1cblxuZXhwb3J0IGNvbnN0IENvbXBvbmVudDogUmVhY3QuRkM8UHJvcHM+ID0gKHByb3BzKSA9PiB7XG4gIGlmICghcHJvcHMuc2V0dGluZ3MuZmVlZElkKSByZXR1cm4gPFdlbGNvbWUgLz5cblxuICBjb25zdCB7IGZlZWQsIGl0ZW1zIH0gPSB1c2VGZWVkU3RhdGUoXG4gICAgcHJvcHMuc2V0dGluZ3MuZmVlZElkLFxuICAgIHByb3BzLnNldHRpbmdzLml0ZW1zLFxuICAgIHByb3BzLnNldHRpbmdzLnNraXBcbiAgKVxuXG4gIGlmICghZmVlZCkgcmV0dXJuIG51bGxcblxuICBjb25zdCBpdGVtc0FycmF5ID0gaXRlbXMgfHwgW11cblxuICBjb25zdCBpc0F2ZXJhZ2VSYXRpbmdWaXNpYmxlID1cbiAgICBwcm9wcy5zZXR0aW5ncy5hdmVyYWdlUmF0aW5nU2V0dGluZ3MudmlzaWJsZSAmJlxuICAgIGZlZWQuZXh0cmEgJiZcbiAgICBmZWVkLmV4dHJhLnRvdGFsXG5cbiAgY29uc3Qge1xuICAgIGdhcCA9IDE2LFxuICAgIHZpc2libGVJdGVtcyA9IDMsXG4gICAgc2hvd0xlZnRBcnJvdyxcbiAgICBzaG93UmlnaHRBcnJvdyxcbiAgICBhcnJvd1NpemUsXG4gICAgYXJyb3dDb2xvcixcbiAgICBhcnJvd0JhY2tncm91bmQsXG4gICAgYXJyb3dSYWRpdXMsXG4gICAgaW5maW5pdGVTY3JvbGwsXG4gIH0gPSBwcm9wcy5zZXR0aW5ncy5zbGlkZXJcblxuICBjb25zdCBzYWZlR2FwID0gTWF0aC5tYXgoMCwgZ2FwIHx8IDE2KVxuICBjb25zdCBzYWZlVmlzaWJsZUl0ZW1zID0gTWF0aC5tYXgoMSwgdmlzaWJsZUl0ZW1zIHx8IDMpXG5cbiAgY29uc3QgaXNDdXN0b21BcnJvd3MgPSBwcm9wcy5zZXR0aW5ncy5jdXN0b21BcnJvd3MgfHwgZmFsc2VcbiAgY29uc3QgY3VzdG9tTGVmdEFycm93ID0gcHJvcHMuc2V0dGluZ3MubGVmdEFycm93XG4gIGNvbnN0IGN1c3RvbVJpZ2h0QXJyb3cgPSBwcm9wcy5zZXR0aW5ncy5yaWdodEFycm93XG5cbiAgY29uc3Qgc2xpZGVySWQgPSBgc2xpZGVyLSR7cHJvcHMuc2V0dGluZ3MuZmVlZElkfS0ke2dsb2JhbFRoaXMuRGF0ZS5ub3coKX1gXG5cbiAgY29uc3QgZ29Ub1ByZXZpb3VzU2xpZGUgPSAoKSA9PiB7XG4gICAgY29uc3QgY29udGFpbmVyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoc2xpZGVySWQpXG4gICAgaWYgKCFjb250YWluZXIgfHwgaXRlbXNBcnJheS5sZW5ndGggPT09IDApIHJldHVyblxuXG4gICAgY29uc3Qgc2xpZGVXaWR0aCA9IGNvbnRhaW5lci5zY3JvbGxXaWR0aCAvIGl0ZW1zQXJyYXkubGVuZ3RoXG4gICAgY29uc3Qgc2Nyb2xsQW1vdW50ID0gc2xpZGVXaWR0aCAqIHNhZmVWaXNpYmxlSXRlbXNcbiAgICBjb25zdCBpc0F0U3RhcnQgPSBjb250YWluZXIuc2Nyb2xsTGVmdCA8PSAxXG5cbiAgICBpZiAoaW5maW5pdGVTY3JvbGwgJiYgaXNBdFN0YXJ0KSB7XG4gICAgICBjb250YWluZXIuc2Nyb2xsVG8oe1xuICAgICAgICBsZWZ0OiBjb250YWluZXIuc2Nyb2xsV2lkdGggLSBjb250YWluZXIub2Zmc2V0V2lkdGgsXG4gICAgICAgIGJlaGF2aW9yOiAnc21vb3RoJyxcbiAgICAgIH0pXG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnRhaW5lci5zY3JvbGxCeSh7IGxlZnQ6IC1zY3JvbGxBbW91bnQsIGJlaGF2aW9yOiAnc21vb3RoJyB9KVxuICAgIH1cbiAgfVxuXG4gIGNvbnN0IGdvVG9OZXh0U2xpZGUgPSAoKSA9PiB7XG4gICAgY29uc3QgY29udGFpbmVyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoc2xpZGVySWQpXG4gICAgaWYgKCFjb250YWluZXIgfHwgaXRlbXNBcnJheS5sZW5ndGggPT09IDApIHJldHVyblxuXG4gICAgY29uc3Qgc2xpZGVXaWR0aCA9IGNvbnRhaW5lci5zY3JvbGxXaWR0aCAvIGl0ZW1zQXJyYXkubGVuZ3RoXG4gICAgY29uc3Qgc2Nyb2xsQW1vdW50ID0gc2xpZGVXaWR0aCAqIHNhZmVWaXNpYmxlSXRlbXNcbiAgICBjb25zdCBtYXhTY3JvbGwgPSBjb250YWluZXIuc2Nyb2xsV2lkdGggLSBjb250YWluZXIub2Zmc2V0V2lkdGhcbiAgICBjb25zdCBpc0F0RW5kID0gY29udGFpbmVyLnNjcm9sbExlZnQgPj0gbWF4U2Nyb2xsIC0gMVxuXG4gICAgaWYgKGluZmluaXRlU2Nyb2xsICYmIGlzQXRFbmQpIHtcbiAgICAgIGNvbnRhaW5lci5zY3JvbGxUbyh7IGxlZnQ6IDAsIGJlaGF2aW9yOiAnc21vb3RoJyB9KVxuICAgIH0gZWxzZSB7XG4gICAgICBjb250YWluZXIuc2Nyb2xsQnkoeyBsZWZ0OiBzY3JvbGxBbW91bnQsIGJlaGF2aW9yOiAnc21vb3RoJyB9KVxuICAgIH1cbiAgfVxuXG4gIGNvbnN0IGFsbFNsaWRlcyA9IGl0ZW1zQXJyYXlcblxuICByZXR1cm4gKFxuICAgIDxTbGlkZXJTZWN0aW9uIGZhbWlseT17cHJvcHMuc2V0dGluZ3MuZm9udFNldHRpbmdzLmZhbWlseX0+XG4gICAgICA8U2xpZGVyV3JhcHBlcj5cbiAgICAgICAge3Nob3dMZWZ0QXJyb3cgJiYgKFxuICAgICAgICAgIDxBcnJvd0J1dHRvblxuICAgICAgICAgICAgb25DbGljaz17Z29Ub1ByZXZpb3VzU2xpZGV9XG4gICAgICAgICAgICBzaXplPXthcnJvd1NpemV9XG4gICAgICAgICAgICBjb2xvcj17YXJyb3dDb2xvcn1cbiAgICAgICAgICAgIGJhY2tncm91bmQ9e2Fycm93QmFja2dyb3VuZH1cbiAgICAgICAgICAgIHJhZGl1cz17YXJyb3dSYWRpdXN9XG4gICAgICAgICAgICBpc0N1c3RvbT17aXNDdXN0b21BcnJvd3N9XG4gICAgICAgICAgPlxuICAgICAgICAgICAge2lzQ3VzdG9tQXJyb3dzID8gKFx