spoko-design-system 0.9.5 → 1.0.0

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.
Files changed (179) hide show
  1. package/.astro/content.d.ts +1 -1
  2. package/.astro/settings.json +4 -4
  3. package/.astro/types.d.ts +2 -1
  4. package/.github/dependabot.yml +11 -11
  5. package/.github/todo.yml +3 -3
  6. package/.github/workflows/deploy.yml +39 -39
  7. package/.github/workflows/release.yml +64 -0
  8. package/.releaserc.json +93 -0
  9. package/.stackblitzrc +5 -5
  10. package/.vscode/extensions.json +5 -5
  11. package/.vscode/launch.json +11 -11
  12. package/.vscode/settings.json +5 -5
  13. package/CHANGELOG.md +12 -0
  14. package/CONTRIBUTING.md +183 -0
  15. package/LICENSE +21 -21
  16. package/README.md +116 -116
  17. package/astro-i18next.config.mjs +17 -17
  18. package/astro-i18next.config.ts +10 -10
  19. package/astro.config.mjs +86 -86
  20. package/dev-dist/sw.js +91 -91
  21. package/dev-dist/workbox-c676b6d3.js +3391 -3391
  22. package/icon.config.ts +310 -310
  23. package/index.ts +70 -70
  24. package/package.json +22 -12
  25. package/public/arrow-bottom.svg +7 -7
  26. package/public/fonts/lg.svg +53 -53
  27. package/public/fonts/vwhead-bold-demo.html +549 -549
  28. package/public/fonts/vwhead-regular-demo.html +549 -549
  29. package/public/fonts/vwtext-bold-demo.html +549 -549
  30. package/public/fonts/vwtext-regular-demo.html +549 -549
  31. package/public/github.svg +3 -3
  32. package/public/grid_dot.svg +4 -4
  33. package/public/linkedin.svg +44 -44
  34. package/public/locales/en/translation.json +12 -12
  35. package/public/locales/pl/translation.json +12 -12
  36. package/public/make-scrollable-code-focusable.js +3 -3
  37. package/public/pagefind.yml +3 -3
  38. package/public/polo.blue.svg +29 -29
  39. package/public/spoko.space.svg +71 -71
  40. package/public/twitter.svg +46 -46
  41. package/renovate.json +6 -6
  42. package/sandbox.config.json +11 -11
  43. package/src/MyComponent.astro +8 -8
  44. package/src/components/Badge.vue +19 -19
  45. package/src/components/Badges.vue +21 -21
  46. package/src/components/Breadcrumbs.vue +94 -94
  47. package/src/components/Button.vue +101 -101
  48. package/src/components/ButtonCopy.astro +183 -183
  49. package/src/components/ButtonCopy.vue +36 -36
  50. package/src/components/Card.astro +27 -27
  51. package/src/components/Carousel.astro +26 -26
  52. package/src/components/Category/CategoriesCarousel.astro +101 -101
  53. package/src/components/Category/CategoryDetails.astro +169 -169
  54. package/src/components/Category/CategoryLink.vue +28 -28
  55. package/src/components/Category/CategorySidebarToggler.vue +9 -9
  56. package/src/components/Category/CategoryTile.astro +37 -37
  57. package/src/components/Category/CategoryViewToggler.astro +89 -89
  58. package/src/components/Category/SubCategoryLink.vue +19 -19
  59. package/src/components/Copyright.astro +12 -12
  60. package/src/components/Date.astro +7 -7
  61. package/src/components/Faq.astro +33 -33
  62. package/src/components/FaqItem.astro +80 -80
  63. package/src/components/FeaturesList.vue +37 -37
  64. package/src/components/FuckRussia.vue +62 -62
  65. package/src/components/HandDrive.astro +55 -29
  66. package/src/components/Header/Header.astro +210 -210
  67. package/src/components/Header/SkipToContent.astro +1 -1
  68. package/src/components/Headline.vue +87 -87
  69. package/src/components/Image.astro +30 -30
  70. package/src/components/LeftSidebar.astro +53 -53
  71. package/src/components/MainColors.vue +22 -22
  72. package/src/components/MainInput.vue +15 -15
  73. package/src/components/Modal.astro +27 -27
  74. package/src/components/PageContent.astro +5 -5
  75. package/src/components/PartNumber.vue +27 -27
  76. package/src/components/Post/PostCategories.astro +41 -41
  77. package/src/components/Post/PostCategories.vue +30 -30
  78. package/src/components/PostHeader.astro +103 -103
  79. package/src/components/PrCode.vue +141 -141
  80. package/src/components/Product/ProductButton.vue +18 -18
  81. package/src/components/Product/ProductCarousel.astro +35 -35
  82. package/src/components/Product/ProductColors.vue +1 -1
  83. package/src/components/Product/ProductEngineType.vue +42 -42
  84. package/src/components/Product/ProductImage.astro +40 -40
  85. package/src/components/Product/ProductLink.astro +101 -101
  86. package/src/components/Product/ProductLink.vue +59 -59
  87. package/src/components/Product/ProductLinkInfo.astro +37 -37
  88. package/src/components/Product/ProductModels.vue +4 -2
  89. package/src/components/Product/ProductNumber.astro +60 -60
  90. package/src/components/Product/ProductPositions.vue +1 -1
  91. package/src/components/ProductCarousel.astro +38 -38
  92. package/src/components/ProductCodes.vue +39 -39
  93. package/src/components/ProductDetailName.vue +52 -52
  94. package/src/components/ProductDetailsList.vue +216 -216
  95. package/src/components/ProductTile.astro +48 -48
  96. package/src/components/Quote.vue +23 -23
  97. package/src/components/ReloadPrompt.astro +50 -50
  98. package/src/components/SlimBanner.vue +72 -72
  99. package/src/components/Table.vue +32 -32
  100. package/src/components/TableOfContents.astro +15 -15
  101. package/src/components/Translations.vue +23 -23
  102. package/src/components/flags/FlagPL.vue +3 -3
  103. package/src/components/flags/FlagUA.vue +2 -2
  104. package/src/components/{Layout → layout}/CallToAction.astro +52 -52
  105. package/src/components/{Layout → layout}/Container.astro +7 -7
  106. package/src/components/{Layout → layout}/Header.astro +80 -80
  107. package/src/config.ts +56 -56
  108. package/src/design.config.ts +98 -98
  109. package/src/env.d.ts +6 -6
  110. package/src/layouts/Layout.astro +61 -61
  111. package/src/layouts/MainLayout.astro +81 -81
  112. package/src/layouts/partials/FooterCommon.astro +4 -4
  113. package/src/layouts/partials/HeadCommon.astro +44 -44
  114. package/src/layouts/partials/HeadSEO.astro +41 -41
  115. package/src/pages/components/badges.mdx +57 -57
  116. package/src/pages/components/breadcrumbs.mdx +139 -139
  117. package/src/pages/components/buttons.mdx +360 -360
  118. package/src/pages/components/card.mdx +294 -294
  119. package/src/pages/components/carousel.mdx +62 -62
  120. package/src/pages/components/copyright.mdx +42 -42
  121. package/src/pages/components/details-list.mdx +115 -115
  122. package/src/pages/components/features-list.mdx +37 -37
  123. package/src/pages/components/flags.mdx +49 -49
  124. package/src/pages/components/fuck-russia.mdx +39 -39
  125. package/src/pages/components/hand-drive.mdx +105 -38
  126. package/src/pages/components/headline.mdx +137 -137
  127. package/src/pages/components/icons.astro +135 -135
  128. package/src/pages/components/image.mdx +513 -513
  129. package/src/pages/components/input.mdx +367 -367
  130. package/src/pages/components/jumbotron.mdx +359 -359
  131. package/src/pages/components/modal.mdx +64 -64
  132. package/src/pages/components/post-header.mdx +64 -64
  133. package/src/pages/components/pr-code.mdx +65 -65
  134. package/src/pages/components/product-number.mdx +58 -58
  135. package/src/pages/components/product-tile.mdx +51 -51
  136. package/src/pages/components/quote.mdx +33 -33
  137. package/src/pages/components/slimbanner.mdx +35 -35
  138. package/src/pages/components/table.mdx +108 -108
  139. package/src/pages/core/colors.mdx +10 -10
  140. package/src/pages/core/grid.mdx +89 -89
  141. package/src/pages/core/introduction.mdx +77 -77
  142. package/src/pages/core/shadows.astro +20 -20
  143. package/src/pages/core/typography.astro +49 -49
  144. package/src/pages/index.astro +133 -133
  145. package/src/pages/patterns/introduction.mdx +60 -60
  146. package/src/pwa.ts +12 -12
  147. package/src/styles/_variables.scss +70 -70
  148. package/src/styles/base/base.css +184 -184
  149. package/src/styles/base/grid.css +92 -92
  150. package/src/styles/base/typography.css +70 -70
  151. package/src/styles/content.css +73 -73
  152. package/src/styles/main.css +7 -7
  153. package/src/types/Product.ts +31 -31
  154. package/src/types/astro.d.ts +3 -3
  155. package/src/utils/product/getPriceFormatted.ts +15 -15
  156. package/src/utils/product/getProductChecklist.ts +17 -17
  157. package/src/utils/product/useFormatProductNumber.ts +41 -41
  158. package/src/utils/seo/getShorterDescription.ts +14 -14
  159. package/src/utils/text/formatDate.ts +5 -5
  160. package/src/utils/text/formatLocaleNumber.ts +6 -6
  161. package/src/utils/text/formatPad.ts +12 -12
  162. package/src/utils/text/getNumberFormatted.ts +33 -33
  163. package/src/utils/text/getTranslatedLink.ts +5 -5
  164. package/src/utils/text.ts +19 -19
  165. package/tailwind.config.cjs +8 -8
  166. package/tsconfig.json +28 -28
  167. package/uno-config/index.ts +268 -268
  168. package/uno-config/theme/breakpoints.ts +9 -9
  169. package/uno-config/theme/colors.ts +64 -64
  170. package/uno-config/theme/dimensions.ts +17 -17
  171. package/uno-config/theme/effects.ts +14 -14
  172. package/uno-config/theme/grid.ts +10 -10
  173. package/uno-config/theme/index.ts +28 -28
  174. package/uno-config/theme/shortcuts/buttons.ts +53 -53
  175. package/uno-config/theme/shortcuts/components.ts +123 -123
  176. package/uno-config/theme/shortcuts/index.ts +20 -20
  177. package/uno-config/theme/shortcuts/layout.ts +74 -74
  178. package/uno-config/theme/typography.ts +29 -29
  179. package/uno.config.ts +2 -2
@@ -1,43 +1,43 @@
1
- <script setup lang="ts">
2
- interface Engine {
3
- id: number | string;
4
- name: string;
5
- }
6
-
7
- const props = withDefaults(defineProps<{
8
- engines: Engine[];
9
- }>(), {
10
- engines: () => []
11
- });
12
- </script>
13
-
14
- <template>
15
- <ul v-if="engines.length" class="engine-types">
16
- <li
17
- v-for="(engine, index) in engines"
18
- :key="engine.id"
19
- class="engine-type"
20
- :class="{ 'with-comma': index !== engines.length - 1 }"
21
- >
22
- {{ engine.name }}
23
- </li>
24
- </ul>
25
- </template>
26
-
27
- <style >
28
- .engine-types {
29
- @apply flex flex-wrap items-center;
30
- }
31
-
32
- .engine-type {
33
- @apply inline-block;
34
-
35
- &.with-comma {
36
- @apply mr-1;
37
-
38
- &:after {
39
- content: ',';
40
- }
41
- }
42
- }
1
+ <script setup lang="ts">
2
+ interface Engine {
3
+ id: number | string;
4
+ name: string;
5
+ }
6
+
7
+ const props = withDefaults(defineProps<{
8
+ engines: Engine[];
9
+ }>(), {
10
+ engines: () => []
11
+ });
12
+ </script>
13
+
14
+ <template>
15
+ <ul v-if="engines.length" class="engine-types">
16
+ <li
17
+ v-for="(engine, index) in engines"
18
+ :key="engine.id"
19
+ class="engine-type"
20
+ :class="{ 'with-comma': index !== engines.length - 1 }"
21
+ >
22
+ {{ engine.name }}
23
+ </li>
24
+ </ul>
25
+ </template>
26
+
27
+ <style >
28
+ .engine-types {
29
+ @apply flex flex-wrap items-center;
30
+ }
31
+
32
+ .engine-type {
33
+ @apply inline-block;
34
+
35
+ &.with-comma {
36
+ @apply mr-1;
37
+
38
+ &:after {
39
+ content: ',';
40
+ }
41
+ }
42
+ }
43
43
  </style>
@@ -1,41 +1,41 @@
1
- ---
2
- import { Image } from 'astro:assets'
3
- interface ImageObject {
4
- src: string;
5
- alt?: string;
6
- height?: string | number;
7
- width?: string | number;
8
- loading?: 'lazy' | 'eager';
9
- srcset?: string[];
10
- class?: string;
11
- index?: number;
12
- }
13
-
14
- const { imageObject } = Astro.props as { imageObject: ImageObject; };
15
-
16
- let inputProps = {};
17
-
18
-
19
- if (imageObject.index && imageObject.index === 1) {
20
- inputProps['data-pagefind-meta'] = 'image[src], image_alt[alt]';
21
- imageObject.loading = 'eager'
22
- }
23
-
24
- if (imageObject.srcset && imageObject.srcset.length) {
25
- inputProps['widths'] = imageObject.srcset
26
- }
27
-
28
- ---
29
- <Image
30
- src={imageObject.src}
31
- alt={imageObject.alt}
32
- height={Number(imageObject.height)}
33
- width={Number(imageObject.width)}
34
- loading={imageObject.loading ? imageObject.loading : 'lazy'}
35
- itemprop="image"
36
- format="avif"
37
- data-pagefind-index-attrs={imageObject.alt}
38
- onerror="this.style.display='none';"
39
- class={`product-image ${imageObject.class || ''}`}
40
- { ...inputProps}
1
+ ---
2
+ import { Image } from 'astro:assets'
3
+ interface ImageObject {
4
+ src: string;
5
+ alt?: string;
6
+ height?: string | number;
7
+ width?: string | number;
8
+ loading?: 'lazy' | 'eager';
9
+ srcset?: string[];
10
+ class?: string;
11
+ index?: number;
12
+ }
13
+
14
+ const { imageObject } = Astro.props as { imageObject: ImageObject; };
15
+
16
+ let inputProps = {};
17
+
18
+
19
+ if (imageObject.index && imageObject.index === 1) {
20
+ inputProps['data-pagefind-meta'] = 'image[src], image_alt[alt]';
21
+ imageObject.loading = 'eager'
22
+ }
23
+
24
+ if (imageObject.srcset && imageObject.srcset.length) {
25
+ inputProps['widths'] = imageObject.srcset
26
+ }
27
+
28
+ ---
29
+ <Image
30
+ src={imageObject.src}
31
+ alt={imageObject.alt}
32
+ height={Number(imageObject.height)}
33
+ width={Number(imageObject.width)}
34
+ loading={imageObject.loading ? imageObject.loading : 'lazy'}
35
+ itemprop="image"
36
+ format="avif"
37
+ data-pagefind-index-attrs={imageObject.alt}
38
+ onerror="this.style.display='none';"
39
+ class={`product-image ${imageObject.class || ''}`}
40
+ { ...inputProps}
41
41
  />
@@ -1,101 +1,101 @@
1
- ---
2
- import type { Product } from 'types/index';
3
-
4
- interface Props {
5
- productId?: number;
6
- productObject?: Product;
7
- locale: string;
8
- index: number;
9
- bigTile?: boolean;
10
- loading?: "eager" | "lazy";
11
- isShopProduct?: boolean;
12
- }
13
-
14
- const {
15
- productId,
16
- productObject,
17
- bigTile,
18
- locale,
19
- index,
20
- loading = "lazy",
21
- isShopProduct = false,
22
- } = Astro.props;
23
-
24
- import { getProductById } from '@utils/product/getProductById';
25
- import { getShopProductById } from '@utils/product/getShopProductById';
26
- import { getProductUrl } from "@utils/product/getProductUrl";
27
- import { getShopProductUrl } from "@utils/product/getShopProductUrl";
28
- import { getProductTranslation } from "@utils/product/getProductTranslation";
29
- import { getImageUrl } from '@utils/getImageUrl';
30
- import i18next, { t } from "i18next";
31
- import { ProductImage, ProductNumber, removeSemicolon, getPriceFormatted } from "spoko-design-system";
32
-
33
- // Użycie productObject jeśli przekazane, inaczej pobranie produktu na podstawie productId
34
- const product = productObject || (productId ? (isShopProduct ? await getShopProductById(productId) : await getProductById(productId)) : null);
35
-
36
- // Określenie URL miniatury produktu
37
- const thumb = product ? (
38
- isShopProduct
39
- ? await getImageUrl(product.images?.[0]?.path || "", "SHOP")
40
- : await getImageUrl(product.photo || "", `'ProductLink' ${product.number}`)
41
- ) : "";
42
-
43
- // Tłumaczenie nazwy produktu jeśli dostępne
44
- const productTranslation = i18next.exists(`parts.${product?.number}.name`)
45
- ? null
46
- : productId ? await getProductTranslation(productId, product?.number || "") : null;
47
-
48
- // const productName = product ? (
49
- // isShopProduct
50
- // ? (locale === "pl" ? product.name_pl : product.name_en)
51
- // : productTranslation?.name || t(`parts.${product.number}.name`)
52
- // ) : "";
53
-
54
- const productName = product.name || 'NO NAME';
55
-
56
- const nameFormatted = removeSemicolon(productName.toString());
57
- ---
58
-
59
- {product && (
60
- <div class={bigTile ? 'product-link--big-tile' : 'product-thumb--plp product-thumb--carousel'}>
61
- {product.photo !== null && thumb ? (
62
- <ProductImage
63
- imagesApiUrl="https://api.polo.blue"
64
- imageObject={{
65
- src: thumb,
66
- alt: productName,
67
- height: '180',
68
- width: '240',
69
- loading
70
- }}
71
- />
72
- ) : (
73
- <img src="/1x1.png" class="bg-neutral-lightest/70" alt={productName} />
74
- )}
75
- </div>
76
-
77
- <div class={bigTile ? '' : 'sm:pl-4'}>
78
- {product.price_pln && (
79
- <p class="block mb-2 font-600 font-headbold text-5">
80
- {getPriceFormatted(product)}
81
- </p>
82
- )}
83
-
84
- <a
85
- class="product-link--url"
86
- href={isShopProduct ? getShopProductUrl(product.slug, locale) : getProductUrl(product.number, locale)}
87
- itemprop="url"
88
- title={product.number}
89
- set:html={nameFormatted}
90
- />
91
-
92
- <ProductNumber productNumber={product.number} copyDisabled={true} />
93
-
94
- {index !== null && (
95
- <>
96
- <meta itemprop="position" content={index.toString()} />
97
- <meta itemprop="name" content={nameFormatted} />
98
- </>
99
- )}
100
- </div>
101
- )}
1
+ ---
2
+ import type { Product } from 'types/index';
3
+
4
+ interface Props {
5
+ productId?: number;
6
+ productObject?: Product;
7
+ locale: string;
8
+ index: number;
9
+ bigTile?: boolean;
10
+ loading?: "eager" | "lazy";
11
+ isShopProduct?: boolean;
12
+ }
13
+
14
+ const {
15
+ productId,
16
+ productObject,
17
+ bigTile,
18
+ locale,
19
+ index,
20
+ loading = "lazy",
21
+ isShopProduct = false,
22
+ } = Astro.props;
23
+
24
+ import { getProductById } from '@utils/product/getProductById';
25
+ import { getShopProductById } from '@utils/product/getShopProductById';
26
+ import { getProductUrl } from "@utils/product/getProductUrl";
27
+ import { getShopProductUrl } from "@utils/product/getShopProductUrl";
28
+ import { getProductTranslation } from "@utils/product/getProductTranslation";
29
+ import { getImageUrl } from '@utils/getImageUrl';
30
+ import i18next, { t } from "i18next";
31
+ import { ProductImage, ProductNumber, removeSemicolon, getPriceFormatted } from "spoko-design-system";
32
+
33
+ // Użycie productObject jeśli przekazane, inaczej pobranie produktu na podstawie productId
34
+ const product = productObject || (productId ? (isShopProduct ? await getShopProductById(productId) : await getProductById(productId)) : null);
35
+
36
+ // Określenie URL miniatury produktu
37
+ const thumb = product ? (
38
+ isShopProduct
39
+ ? await getImageUrl(product.images?.[0]?.path || "", "SHOP")
40
+ : await getImageUrl(product.photo || "", `'ProductLink' ${product.number}`)
41
+ ) : "";
42
+
43
+ // Tłumaczenie nazwy produktu jeśli dostępne
44
+ const productTranslation = i18next.exists(`parts.${product?.number}.name`)
45
+ ? null
46
+ : productId ? await getProductTranslation(productId, product?.number || "") : null;
47
+
48
+ // const productName = product ? (
49
+ // isShopProduct
50
+ // ? (locale === "pl" ? product.name_pl : product.name_en)
51
+ // : productTranslation?.name || t(`parts.${product.number}.name`)
52
+ // ) : "";
53
+
54
+ const productName = product.name || 'NO NAME';
55
+
56
+ const nameFormatted = removeSemicolon(productName.toString());
57
+ ---
58
+
59
+ {product && (
60
+ <div class={bigTile ? 'product-link--big-tile' : 'product-thumb--plp product-thumb--carousel'}>
61
+ {product.photo !== null && thumb ? (
62
+ <ProductImage
63
+ imagesApiUrl="https://api.polo.blue"
64
+ imageObject={{
65
+ src: thumb,
66
+ alt: productName,
67
+ height: '180',
68
+ width: '240',
69
+ loading
70
+ }}
71
+ />
72
+ ) : (
73
+ <img src="/1x1.png" class="bg-neutral-lightest/70" alt={productName} />
74
+ )}
75
+ </div>
76
+
77
+ <div class={bigTile ? '' : 'sm:pl-4'}>
78
+ {product.price_pln && (
79
+ <p class="block mb-2 font-600 font-headbold text-5">
80
+ {getPriceFormatted(product)}
81
+ </p>
82
+ )}
83
+
84
+ <a
85
+ class="product-link--url"
86
+ href={isShopProduct ? getShopProductUrl(product.slug, locale) : getProductUrl(product.number, locale)}
87
+ itemprop="url"
88
+ title={product.number}
89
+ set:html={nameFormatted}
90
+ />
91
+
92
+ <ProductNumber productNumber={product.number} copyDisabled={true} />
93
+
94
+ {index !== null && (
95
+ <>
96
+ <meta itemprop="position" content={index.toString()} />
97
+ <meta itemprop="name" content={nameFormatted} />
98
+ </>
99
+ )}
100
+ </div>
101
+ )}
@@ -1,60 +1,60 @@
1
- <template>
2
- <div class="product-link" itemscope itemtype="https://schema.org/Product">
3
- <div :class="[
4
- bigTile ? 'product-link--big-tile' : 'product-thumb--plp product-thumb--carousel'
5
- ]">
6
- <!-- Slot na ProductImage z Astro -->
7
- <slot name="image">
8
- <img src="/1x1.png" class="bg-neutral-lightest/70" :alt="productName" />
9
- </slot>
10
- </div>
11
-
12
- <div :class="[bigTile ? '' : 'sm:pl-4']">
13
- <p v-if="price" class="block mb-2 font-600 font-headbold text-5">
14
- {{ price }}
15
- </p>
16
-
17
- <a
18
- class="product-link--url"
19
- :href="url"
20
- itemprop="url"
21
- :title="productNumber"
22
- v-html="nameFormatted"
23
- />
24
-
25
- <ProductNumber
26
- :productNumber="productNumber"
27
- :copyDisabled="true"
28
- />
29
-
30
- <template v-if="index !== null">
31
- <meta itemprop="position" :content="index.toString()" />
32
- <meta itemprop="name" :content="nameFormatted" />
33
- </template>
34
- </div>
35
- </div>
36
- </template>
37
-
38
- <script setup lang="ts">
39
- import { computed } from 'vue'
40
- import { ProductNumber } from 'spoko-design-system'
41
- import { removeSemicolon } from 'spoko-design-system'
42
-
43
- interface Props {
44
- productName: string
45
- productNumber: string
46
- url: string
47
- price?: string
48
- bigTile?: boolean
49
- index?: number
50
- }
51
-
52
- const props = withDefaults(defineProps<Props>(), {
53
- bigTile: false,
54
- index: null,
55
- })
56
-
57
- const nameFormatted = computed(() => {
58
- return removeSemicolon(props.productName.toString())
59
- })
1
+ <template>
2
+ <div class="product-link" itemscope itemtype="https://schema.org/Product">
3
+ <div :class="[
4
+ bigTile ? 'product-link--big-tile' : 'product-thumb--plp product-thumb--carousel'
5
+ ]">
6
+ <!-- Slot na ProductImage z Astro -->
7
+ <slot name="image">
8
+ <img src="/1x1.png" class="bg-neutral-lightest/70" :alt="productName" />
9
+ </slot>
10
+ </div>
11
+
12
+ <div :class="[bigTile ? '' : 'sm:pl-4']">
13
+ <p v-if="price" class="block mb-2 font-600 font-headbold text-5">
14
+ {{ price }}
15
+ </p>
16
+
17
+ <a
18
+ class="product-link--url"
19
+ :href="url"
20
+ itemprop="url"
21
+ :title="productNumber"
22
+ v-html="nameFormatted"
23
+ />
24
+
25
+ <ProductNumber
26
+ :productNumber="productNumber"
27
+ :copyDisabled="true"
28
+ />
29
+
30
+ <template v-if="index !== null">
31
+ <meta itemprop="position" :content="index.toString()" />
32
+ <meta itemprop="name" :content="nameFormatted" />
33
+ </template>
34
+ </div>
35
+ </div>
36
+ </template>
37
+
38
+ <script setup lang="ts">
39
+ import { computed } from 'vue'
40
+ import { ProductNumber } from 'spoko-design-system'
41
+ import { removeSemicolon } from 'spoko-design-system'
42
+
43
+ interface Props {
44
+ productName: string
45
+ productNumber: string
46
+ url: string
47
+ price?: string
48
+ bigTile?: boolean
49
+ index?: number
50
+ }
51
+
52
+ const props = withDefaults(defineProps<Props>(), {
53
+ bigTile: false,
54
+ index: null,
55
+ })
56
+
57
+ const nameFormatted = computed(() => {
58
+ return removeSemicolon(props.productName.toString())
59
+ })
60
60
  </script>
@@ -1,38 +1,38 @@
1
- ---
2
- import type { Product } from '../../types/Product';
3
- import ProductNumber from "./ProductNumber.astro"
4
-
5
- interface Props {
6
- product: Product;
7
- nameFormatted: string;
8
- price?: string;
9
- url: string;
10
- index?: number | null;
11
- bigTile?: boolean;
12
- }
13
-
14
- const { product, nameFormatted, price, url, index, bigTile } = Astro.props;
15
-
16
- ---
17
-
18
- <div class={`flex flex-col ${bigTile ? '' : 'sm:pl-4'}`}>
19
- {price && (
20
- <p class="block mb-2 font-600 font-headbold text-5">
21
- {price}
22
- </p>
23
- )}
24
- <a
25
- class="product-link--url"
26
- href={url}
27
- itemprop="url"
28
- title={product.number}
29
- set:html={nameFormatted}
30
- />
31
- <ProductNumber productNumber={product.number} copyDisabled={true} />
32
- {index !== null && (
33
- <>
34
- <meta itemprop="position" content={index} />
35
- <meta itemprop="name" content={nameFormatted} />
36
- </>
37
- )}
1
+ ---
2
+ import type { Product } from '../../types/Product';
3
+ import ProductNumber from "./ProductNumber.astro"
4
+
5
+ interface Props {
6
+ product: Product;
7
+ nameFormatted: string;
8
+ price?: string;
9
+ url: string;
10
+ index?: number | null;
11
+ bigTile?: boolean;
12
+ }
13
+
14
+ const { product, nameFormatted, price, url, index, bigTile } = Astro.props;
15
+
16
+ ---
17
+
18
+ <div class={`flex flex-col ${bigTile ? '' : 'sm:pl-4'}`}>
19
+ {price && (
20
+ <p class="block mb-2 font-600 font-headbold text-5">
21
+ {price}
22
+ </p>
23
+ )}
24
+ <a
25
+ class="product-link--url"
26
+ href={url}
27
+ itemprop="url"
28
+ title={product.number}
29
+ set:html={nameFormatted}
30
+ />
31
+ <ProductNumber productNumber={product.number} copyDisabled={true} />
32
+ {index !== null && (
33
+ <>
34
+ <meta itemprop="position" content={index} />
35
+ <meta itemprop="name" content={nameFormatted} />
36
+ </>
37
+ )}
38
38
  </div>
@@ -1,8 +1,10 @@
1
1
  <script lang="ts" setup>
2
+ import type { PropType } from 'vue'
3
+
2
4
  const props = defineProps({
3
5
  modelIds: {
4
- type: Array,
5
- default: null,
6
+ type: Array as PropType<(string | number)[]>,
7
+ default: () => [],
6
8
  required: true,
7
9
  },
8
10
  })