@kiva/kv-components 3.107.0 → 3.107.2

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 (177) hide show
  1. package/CHANGELOG.md +22 -0
  2. package/dist/components/.storybook/main.js +85 -0
  3. package/dist/components/.storybook/package.json +3 -0
  4. package/dist/components/.storybook/preview.js +61 -0
  5. package/dist/components/.storybook/tailwind.css +5 -0
  6. package/dist/components/KvAccordionItem.vue +130 -0
  7. package/dist/components/KvActivityRow.vue +33 -0
  8. package/dist/components/KvBorrowerImage.vue +179 -0
  9. package/dist/components/KvButton.vue +287 -0
  10. package/dist/components/KvCarousel.vue +297 -0
  11. package/dist/components/KvCartModal.vue +365 -0
  12. package/dist/components/KvCheckbox.vue +203 -0
  13. package/dist/components/KvChip.vue +54 -0
  14. package/dist/components/KvClassicLoanCard.vue +527 -0
  15. package/dist/components/KvCommentsAdd.vue +135 -0
  16. package/dist/components/KvCommentsContainer.vue +84 -0
  17. package/dist/components/KvCommentsHeartButton.vue +70 -0
  18. package/dist/components/KvCommentsList.vue +68 -0
  19. package/dist/components/KvCommentsListItem.vue +241 -0
  20. package/dist/components/KvCommentsReplyButton.vue +52 -0
  21. package/dist/components/KvContentfulImg.vue +273 -0
  22. package/dist/components/KvCountdownTimer.vue +59 -0
  23. package/dist/components/KvExpandable.vue +84 -0
  24. package/dist/components/KvExpandableQuestion.vue +120 -0
  25. package/dist/components/KvFlag.vue +120 -0
  26. package/dist/components/KvGrid.vue +28 -0
  27. package/dist/components/KvImpactDashboardHeader.vue +40 -0
  28. package/dist/components/KvInlineActivityCard.vue +55 -0
  29. package/dist/components/KvInlineActivityFeed.vue +38 -0
  30. package/dist/components/KvIntroductionLoanCard.vue +446 -0
  31. package/dist/components/KvLendAmountButton.vue +65 -0
  32. package/dist/components/KvLendCta.vue +451 -0
  33. package/dist/components/KvLightbox.vue +334 -0
  34. package/dist/components/KvLineGraph.vue +128 -0
  35. package/dist/components/KvLoadingPlaceholder.vue +38 -0
  36. package/dist/components/KvLoadingSpinner.vue +81 -0
  37. package/dist/components/KvLoanActivities.vue +268 -0
  38. package/dist/components/KvLoanBookmark.vue +39 -0
  39. package/dist/components/KvLoanCallouts.vue +53 -0
  40. package/dist/components/KvLoanProgressGroup.vue +76 -0
  41. package/dist/components/KvLoanTag.vue +88 -0
  42. package/dist/components/KvLoanTeamPick.vue +44 -0
  43. package/dist/components/KvLoanUse.vue +92 -0
  44. package/dist/components/KvMap.vue +599 -0
  45. package/dist/components/KvMaterialIcon.vue +47 -0
  46. package/dist/components/KvPageContainer.vue +15 -0
  47. package/dist/components/KvPagination.vue +198 -0
  48. package/dist/components/KvPieChart.vue +257 -0
  49. package/dist/components/KvPopper.vue +178 -0
  50. package/dist/components/KvProgressBar.vue +149 -0
  51. package/dist/components/KvRadio.vue +198 -0
  52. package/dist/components/KvSelect.vue +114 -0
  53. package/dist/components/KvSideSheet.vue +134 -0
  54. package/dist/components/KvSwitch.vue +143 -0
  55. package/dist/components/KvTab.vue +90 -0
  56. package/dist/components/KvTabPanel.vue +64 -0
  57. package/dist/components/KvTabs.vue +182 -0
  58. package/dist/components/KvTextInput.vue +247 -0
  59. package/dist/components/KvTextLink.vue +138 -0
  60. package/dist/components/KvThemeProvider.vue +122 -0
  61. package/dist/components/KvToast.vue +221 -0
  62. package/dist/components/KvTooltip.vue +168 -0
  63. package/dist/components/KvTreeMapChart.vue +229 -0
  64. package/dist/components/KvUserAvatar.vue +132 -0
  65. package/dist/components/KvVerticalCarousel.vue +156 -0
  66. package/dist/components/KvVotingCard.vue +160 -0
  67. package/dist/components/KvVotingCardV2.vue +154 -0
  68. package/dist/components/KvWideLoanCard.vue +432 -0
  69. package/dist/components/stories/Forms.stories.js +62 -0
  70. package/dist/components/stories/KvAccordionItem.stories.js +24 -0
  71. package/dist/components/stories/KvActivityRow.stories.js +25 -0
  72. package/dist/components/stories/KvBorrowerImage.stories.js +68 -0
  73. package/dist/components/stories/KvButton.stories.js +144 -0
  74. package/dist/components/stories/KvCarousel.stories.js +426 -0
  75. package/dist/components/stories/KvCartModal.stories.js +54 -0
  76. package/dist/components/stories/KvCheckbox.stories.js +163 -0
  77. package/dist/components/stories/KvChip.stories.js +43 -0
  78. package/dist/components/stories/KvClassicLoanCard.stories.js +480 -0
  79. package/dist/components/stories/KvCommentsAdd.stories.js +32 -0
  80. package/dist/components/stories/KvCommentsContainer.stories.js +42 -0
  81. package/dist/components/stories/KvCommentsHeartButton.stories.js +25 -0
  82. package/dist/components/stories/KvCommentsList.stories.js +39 -0
  83. package/dist/components/stories/KvCommentsListItem.stories.js +45 -0
  84. package/dist/components/stories/KvCommentsReplyButton.stories.js +21 -0
  85. package/dist/components/stories/KvContentfulImg.stories.js +196 -0
  86. package/dist/components/stories/KvCountdownTimer.stories.js +30 -0
  87. package/dist/components/stories/KvExpandableQuestion.stories.js +129 -0
  88. package/dist/components/stories/KvFlag.stories.js +36 -0
  89. package/dist/components/stories/KvGrid.stories.js +97 -0
  90. package/dist/components/stories/KvImpactDashboardHeader.stories.js +22 -0
  91. package/dist/components/stories/KvInlineActivityCard.stories.js +69 -0
  92. package/dist/components/stories/KvInlineActivityFeed.stories.js +76 -0
  93. package/dist/components/stories/KvIntroductionLoanCard.stories.js +208 -0
  94. package/dist/components/stories/KvLendAmountButton.stories.js +31 -0
  95. package/dist/components/stories/KvLendCta.stories.js +177 -0
  96. package/dist/components/stories/KvLightbox.stories.js +304 -0
  97. package/dist/components/stories/KvLineGraph.stories.js +52 -0
  98. package/dist/components/stories/KvLoadingPlaceholder.stories.js +17 -0
  99. package/dist/components/stories/KvLoadingSpinner.stories.js +52 -0
  100. package/dist/components/stories/KvLoanActivities.stories.js +104 -0
  101. package/dist/components/stories/KvLoanBookmark.stories.js +22 -0
  102. package/dist/components/stories/KvLoanCallouts.stories.js +22 -0
  103. package/dist/components/stories/KvLoanProgressGroup.stories.js +29 -0
  104. package/dist/components/stories/KvLoanTag.stories.js +61 -0
  105. package/dist/components/stories/KvLoanTeamPick.stories.js +20 -0
  106. package/dist/components/stories/KvLoanUse.stories.js +60 -0
  107. package/dist/components/stories/KvMap.stories.js +121 -0
  108. package/dist/components/stories/KvMaterialIcon.stories.js +201 -0
  109. package/dist/components/stories/KvPageContainer.stories.js +50 -0
  110. package/dist/components/stories/KvPagination.stories.js +70 -0
  111. package/dist/components/stories/KvPieChart.stories.js +47 -0
  112. package/dist/components/stories/KvProgressBar.stories.js +53 -0
  113. package/dist/components/stories/KvRadio.stories.js +140 -0
  114. package/dist/components/stories/KvSelect.stories.js +125 -0
  115. package/dist/components/stories/KvSideSheet.stories.js +50 -0
  116. package/dist/components/stories/KvSwitch.stories.js +66 -0
  117. package/dist/components/stories/KvTabs.stories.js +106 -0
  118. package/dist/components/stories/KvTextInput.stories.js +194 -0
  119. package/dist/components/stories/KvTextLink.stories.js +55 -0
  120. package/dist/components/stories/KvThemeProvider.stories.js +178 -0
  121. package/dist/components/stories/KvToast.stories.js +117 -0
  122. package/dist/components/stories/KvTooltip.stories.js +26 -0
  123. package/dist/components/stories/KvTreeMapChart.stories.js +42 -0
  124. package/dist/components/stories/KvUserAvatar.stories.js +47 -0
  125. package/dist/components/stories/KvVerticalCarousel.stories.js +168 -0
  126. package/dist/components/stories/KvVotingCard.stories.js +33 -0
  127. package/dist/components/stories/KvVotingCardV2.stories.js +89 -0
  128. package/dist/components/stories/KvWideLoanCard.stories.js +292 -0
  129. package/dist/components/stories/StyleguidePrimitives.stories.js +499 -0
  130. package/dist/components/stories/StyleguideProse.stories.js +215 -0
  131. package/dist/data/countries-borders.json +1 -0
  132. package/dist/data/ne_110m_admin_0_countries.json +1 -0
  133. package/dist/utils/Alea.js +9 -0
  134. package/dist/utils/attrs.js +7 -0
  135. package/dist/utils/carousels.js +8 -0
  136. package/dist/{attrs.js → utils/chunk-3HK4G4NT.js} +1 -0
  137. package/dist/{loanCard.js → utils/chunk-55HF2ORX.js} +1 -0
  138. package/dist/{expander.js → utils/chunk-AY3PR5S4.js} +3 -2
  139. package/dist/{carousels.js → utils/chunk-AZPWOFD5.js} +1 -0
  140. package/dist/{printing.js → utils/chunk-B5J5WLAH.js} +1 -0
  141. package/dist/{Alea.js → utils/chunk-GPSH6OPA.js} +2 -1
  142. package/dist/{scrollLock.js → utils/chunk-HIY5IW65.js} +2 -1
  143. package/dist/{treemap.js → utils/chunk-MSMZIN54.js} +1 -0
  144. package/dist/{imageUtils.js → utils/chunk-OXJCCNNW.js} +1 -0
  145. package/dist/{touchEvents.js → utils/chunk-S3MABILA.js} +3 -2
  146. package/dist/{mapUtils.js → utils/chunk-VIGEMAKO.js} +5 -4
  147. package/dist/utils/chunk-YCNMJ4YV.js +37 -0
  148. package/dist/{loanUtils.js → utils/chunk-YFEC5ODJ.js} +7 -6
  149. package/dist/utils/expander.js +9 -0
  150. package/dist/utils/imageUtils.js +9 -0
  151. package/dist/utils/index.cjs +1118 -0
  152. package/dist/utils/index.js +166 -0
  153. package/dist/utils/loanCard.js +9 -0
  154. package/dist/utils/loanUtils.js +23 -0
  155. package/dist/utils/mapUtils.js +15 -0
  156. package/dist/utils/printing.js +9 -0
  157. package/dist/utils/scrollLock.js +13 -0
  158. package/dist/{throttle.js → utils/throttle.js} +1 -0
  159. package/dist/utils/touchEvents.js +11 -0
  160. package/dist/utils/treemap.js +7 -0
  161. package/package.json +11 -7
  162. package/utils/index.js +14 -0
  163. package/index.js +0 -3
  164. /package/dist/{Alea.cjs → utils/Alea.cjs} +0 -0
  165. /package/dist/{attrs.cjs → utils/attrs.cjs} +0 -0
  166. /package/dist/{carousels.cjs → utils/carousels.cjs} +0 -0
  167. /package/dist/{chunk-HV3AUBFT.js → utils/chunk-HV3AUBFT.js} +0 -0
  168. /package/dist/{expander.cjs → utils/expander.cjs} +0 -0
  169. /package/dist/{imageUtils.cjs → utils/imageUtils.cjs} +0 -0
  170. /package/dist/{loanCard.cjs → utils/loanCard.cjs} +0 -0
  171. /package/dist/{loanUtils.cjs → utils/loanUtils.cjs} +0 -0
  172. /package/dist/{mapUtils.cjs → utils/mapUtils.cjs} +0 -0
  173. /package/dist/{printing.cjs → utils/printing.cjs} +0 -0
  174. /package/dist/{scrollLock.cjs → utils/scrollLock.cjs} +0 -0
  175. /package/dist/{throttle.cjs → utils/throttle.cjs} +0 -0
  176. /package/dist/{touchEvents.cjs → utils/touchEvents.cjs} +0 -0
  177. /package/dist/{treemap.cjs → utils/treemap.cjs} +0 -0
@@ -0,0 +1,268 @@
1
+ <template>
2
+ <div
3
+ v-if="combinedActivities.length > 0"
4
+ class="loan-activity"
5
+ >
6
+ <div class="tw-flex tw-justify-center lg:tw-px-0">
7
+ <KvActivityRow
8
+ :activity="singleActivity"
9
+ />
10
+ </div>
11
+ <div class="tw-flex tw-justify-center tw-mt-1 tw-text-small">
12
+ <span v-if="lendersNumber && amountLent">
13
+ {{ lendersNumber }} lenders lent {{ amountLent }}. &nbsp;
14
+ </span>
15
+ <button
16
+ class="tw-text-action hover:tw-underline"
17
+ @click="showActivityModal"
18
+ >
19
+ See all activity
20
+ </button>
21
+ </div>
22
+ <KvLightbox
23
+ title="Activity Feed Modal"
24
+ :visible="lightboxOpen"
25
+ @lightbox-closed="closeLightbox"
26
+ >
27
+ <template #header>
28
+ <h2 class="tw-text-h3 tw-flex-1 data-hj-suppress">
29
+ {{ modalTitle }}
30
+ </h2>
31
+ </template>
32
+ <div class="tw-overflow-x-hidden tw-px-2 tw-pb-1">
33
+ <div
34
+ v-for="activity in combinedActivities"
35
+ :key="activity.key"
36
+ class="md:tw-px-8 lg:tw-px-14 tw-mt-4"
37
+ >
38
+ <h4 class="tw-text-center tw-text-h4">
39
+ {{ formattedDate(activity.key) }}
40
+ </h4>
41
+ <div class="tw-flex tw-flex-col tw-gap-y-1">
42
+ <KvActivityRow
43
+ v-for="(act, index) in activity.data"
44
+ :key="index"
45
+ :activity="act"
46
+ />
47
+ </div>
48
+ </div>
49
+ </div>
50
+ <div>
51
+ <p
52
+ v-if="errorMsg"
53
+ class="tw-mt-1 tw-text-small tw-text-danger"
54
+ >
55
+ {{ errorMsg }}
56
+ </p>
57
+ <div class="tw-flex tw-justify-end tw-mt-4 tw-pb-2.5 md:tw-pb-4">
58
+ <KvLendCta
59
+ :loan="loan"
60
+ :is-loading="false"
61
+ :kv-track-function="kvTrackFunction"
62
+ :get-cookie="getCookie"
63
+ :set-cookie="setCookie"
64
+ :user-balance="userBalance"
65
+ :basket-items="basketItems"
66
+ :route="route"
67
+ :is-adding="isAdding"
68
+ :enable-huge-amount="enableHugeAmount"
69
+ :is-visitor="isVisitor"
70
+ :enable-five-dollars-notes="enableFiveDollarsNotes"
71
+ @add-to-basket="$emit('add-to-basket', $event)"
72
+ />
73
+ </div>
74
+ </div>
75
+ </KvLightbox>
76
+ </div>
77
+ </template>
78
+
79
+ <script>
80
+ import { computed, ref, toRefs } from 'vue-demi';
81
+ import numeral from 'numeral';
82
+ import { format } from 'date-fns';
83
+ import KvActivityRow from './KvActivityRow.vue';
84
+ import KvLightbox from './KvLightbox.vue';
85
+ import KvLendCta from './KvLendCta.vue';
86
+
87
+ export default {
88
+ name: 'KvLoanActivities',
89
+ components: {
90
+ KvActivityRow,
91
+ KvLightbox,
92
+ KvLendCta,
93
+ },
94
+ props: {
95
+ /**
96
+ * loan object coming from parent component
97
+ */
98
+ loan: {
99
+ type: Object,
100
+ default: () => ({}),
101
+ },
102
+ /**
103
+ * loan combined activities coming from parent component
104
+ */
105
+ combinedActivities: {
106
+ type: Array,
107
+ default: () => ([]),
108
+ },
109
+ /**
110
+ * isAdding flag coming from parent add to basket method
111
+ */
112
+ isAdding: {
113
+ type: Boolean,
114
+ default: false,
115
+ },
116
+ /**
117
+ * basketItems array coming from parent component
118
+ */
119
+ basketItems: {
120
+ type: Array,
121
+ default: () => ([]),
122
+ },
123
+ /**
124
+ * userBalance string coming from parent component
125
+ */
126
+ userBalance: {
127
+ type: String,
128
+ default: undefined,
129
+ },
130
+ /**
131
+ * errorMsg string coming from parent add to basket method
132
+ */
133
+ errorMsg: {
134
+ type: String,
135
+ default: undefined,
136
+ },
137
+ /**
138
+ * kvTrackFunction function to track events from LendCta component
139
+ */
140
+ kvTrackFunction: {
141
+ type: Function,
142
+ required: true,
143
+ },
144
+ /**
145
+ * getCookie function for LendCta component
146
+ */
147
+ getCookie: {
148
+ type: Function,
149
+ default: undefined,
150
+ },
151
+ /**
152
+ * setCookie function for LendCta component
153
+ */
154
+ setCookie: {
155
+ type: Function,
156
+ default: undefined,
157
+ },
158
+ /**
159
+ * route object for LendCta component
160
+ */
161
+ route: {
162
+ type: Object,
163
+ default: undefined,
164
+ },
165
+ enableFiveDollarsNotes: {
166
+ type: Boolean,
167
+ default: false,
168
+ },
169
+ enableHugeAmount: {
170
+ type: Boolean,
171
+ default: false,
172
+ },
173
+ isVisitor: {
174
+ type: Boolean,
175
+ default: true,
176
+ },
177
+ },
178
+ emits: [
179
+ 'add-to-basket',
180
+ ],
181
+ setup(props) {
182
+ const {
183
+ loan,
184
+ combinedActivities,
185
+ kvTrackFunction,
186
+ } = toRefs(props);
187
+
188
+ const lightboxOpen = ref(false);
189
+
190
+ const modalTitle = computed(() => `Activity for ${loan.value?.name}`);
191
+
192
+ const lendersNumber = computed(() => loan.value?.lenders?.totalCount ?? 0);
193
+
194
+ const amountLent = computed(() => {
195
+ const amount = loan.value?.loanFundraisingInfo?.fundedAmount ?? 0;
196
+ return numeral(parseFloat(amount)).format('$0,0');
197
+ });
198
+
199
+ const singleActivity = computed(() => {
200
+ const singleAct = combinedActivities.value.find((activity) => {
201
+ return activity.data.some((element) => {
202
+ return element.type === 'LendingAction';
203
+ });
204
+ });
205
+
206
+ return singleAct?.data[0] ?? {};
207
+ });
208
+
209
+ const formattedDate = (date) => {
210
+ const dateObj = new Date(date);
211
+ return format(dateObj, 'MMM d, yyyy');
212
+ };
213
+ const showActivityModal = () => {
214
+ kvTrackFunction.value('loan-card', 'click', 'see-all-activity');
215
+ lightboxOpen.value = true;
216
+ };
217
+ const closeLightbox = () => {
218
+ lightboxOpen.value = false;
219
+ };
220
+
221
+ return {
222
+ lightboxOpen,
223
+ modalTitle,
224
+ formattedDate,
225
+ showActivityModal,
226
+ closeLightbox,
227
+ singleActivity,
228
+ lendersNumber,
229
+ amountLent,
230
+ };
231
+ },
232
+ };
233
+ </script>
234
+
235
+ <style scoped lang="postcss">
236
+ .loan-activity >>> #kvLightboxBody {
237
+ @apply tw-flex tw-flex-col tw-px-0 tw-pb-0;
238
+ }
239
+
240
+ .loan-activity >>> div > div > div > div > div:first-child {
241
+ box-shadow: var(--kiva-box-shadow);
242
+ }
243
+
244
+ .loan-activity >>> div > div > div > div > div:first-child > div,
245
+ .loan-activity >>> #kvLightboxBody div {
246
+ box-shadow: none;
247
+ }
248
+
249
+ .loan-activity >>> #kvLightboxBody > div:first-child {
250
+ @apply tw-px-4;
251
+ }
252
+
253
+ .loan-activity >>> [role=dialog] {
254
+ min-width: 840px;
255
+ max-width: 840px !important;
256
+
257
+ @media (max-width: calc(840px + 2rem)) {
258
+ min-width: 100%;
259
+ max-width: 100% !important;
260
+ }
261
+ }
262
+
263
+ .loan-activity >>> #kvLightboxBody > div:nth-child(2) {
264
+ @apply tw-px-4;
265
+
266
+ box-shadow: var(--kiva-negative-box-shadow);
267
+ }
268
+ </style>
@@ -0,0 +1,39 @@
1
+ <template>
2
+ <div>
3
+ <button
4
+ style="border-radius: 8px; padding: 3px;"
5
+ class="tw-cursor-pointer tw-font-medium tw-bg-white tw-flex tw-justify-center"
6
+ @click="$emit('toggle-bookmark')"
7
+ >
8
+ <kv-material-icon
9
+ class="tw-w-3"
10
+ :class="{ 'tw-text-black': !isBookmarked, 'tw-text-marigold': isBookmarked }"
11
+ :icon="`${!isBookmarked ? mdiBookmarkOutline : mdiBookmark}`"
12
+ />
13
+ </button>
14
+ </div>
15
+ </template>
16
+
17
+ <script>
18
+ import { mdiBookmarkOutline, mdiBookmark } from '@mdi/js';
19
+ import KvMaterialIcon from './KvMaterialIcon.vue';
20
+
21
+ export default {
22
+ name: 'KvLoanBookmark',
23
+ components: {
24
+ KvMaterialIcon,
25
+ },
26
+ props: {
27
+ isBookmarked: {
28
+ type: Boolean,
29
+ default: false,
30
+ },
31
+ },
32
+ data() {
33
+ return {
34
+ mdiBookmarkOutline,
35
+ mdiBookmark,
36
+ };
37
+ },
38
+ };
39
+ </script>
@@ -0,0 +1,53 @@
1
+ <template>
2
+ <div
3
+ class="tw-overflow-hidden"
4
+ style="width: inherit;"
5
+ >
6
+ <div
7
+ class="
8
+ tw-flex
9
+ tw-whitespace-nowrap
10
+ tw-w-full
11
+ tw-h-4
12
+ "
13
+ >
14
+ <span
15
+ v-for="tag in callouts"
16
+ :key="tag.label"
17
+ :title="tag.label"
18
+ class="
19
+ loan-callout
20
+ tw-text-ellipsis
21
+ tw-overflow-hidden
22
+ tw-rounded-full
23
+ tw-font-medium
24
+ tw-py-0.5
25
+ tw-px-1
26
+ tw-mr-0.5
27
+ tw-mb-0.5
28
+ tw-text-small
29
+ "
30
+ >
31
+ {{ tag.label }}
32
+ </span>
33
+ </div>
34
+ </div>
35
+ </template>
36
+
37
+ <script>
38
+ export default {
39
+ name: 'KvLoanCallouts',
40
+ props: {
41
+ callouts: {
42
+ type: Array,
43
+ required: true,
44
+ },
45
+ },
46
+ };
47
+ </script>
48
+
49
+ <style scoped>
50
+ .loan-callout {
51
+ background: #f1f1f1;
52
+ }
53
+ </style>
@@ -0,0 +1,76 @@
1
+ <template>
2
+ <figure>
3
+ <div :class="{'tw-flex tw-justify-between': amountGoal}">
4
+ <h4
5
+ class="tw-lowercase tw-mb-0.5"
6
+ :class="{ 'progress-group-amount-low': amountLow}"
7
+ >
8
+ {{ fundingText }}
9
+ </h4>
10
+ <h4
11
+ v-if="amountGoal"
12
+ class="tw-lowercase tw-mb-0.5"
13
+ >
14
+ {{ goalText }}
15
+ </h4>
16
+ </div>
17
+ <kv-progress-bar
18
+ aria-label="Percent the loan has funded"
19
+ :value="progressPercent * 100"
20
+ />
21
+ </figure>
22
+ </template>
23
+
24
+ <script>
25
+ import numeral from 'numeral';
26
+ import KvProgressBar from './KvProgressBar.vue';
27
+
28
+ export default {
29
+ name: 'KvLoanProgressGroup',
30
+ components: {
31
+ KvProgressBar,
32
+ },
33
+ props: {
34
+ moneyLeft: {
35
+ type: String,
36
+ default: '0.00',
37
+ },
38
+ progressPercent: {
39
+ type: Number,
40
+ default: 0,
41
+ },
42
+ amountGoal: {
43
+ type: String,
44
+ default: '',
45
+ },
46
+ },
47
+ computed: {
48
+ numeralLeft() {
49
+ return numeral(this.moneyLeft);
50
+ },
51
+ amountLow() {
52
+ return this.numeralLeft.value() < 100;
53
+ },
54
+ fundingText() {
55
+ if (!this.numeralLeft.value()) return 'funded!';
56
+ const formattedMoneyLeft = this.numeralLeft.format('$0,0[.]00');
57
+ const exclamationMark = this.amountLow ? '!' : '';
58
+ return `${formattedMoneyLeft} to go${exclamationMark}`;
59
+ },
60
+ numeralGoal() {
61
+ return numeral(this.amountGoal);
62
+ },
63
+ goalText() {
64
+ if (!this.numeralGoal.value()) return '';
65
+ const formattedGoal = this.numeralGoal.format('$0,0[.]00');
66
+ return `${formattedGoal} goal`;
67
+ },
68
+ },
69
+ };
70
+ </script>
71
+
72
+ <style scoped>
73
+ .progress-group-amount-low {
74
+ color: #CE4A00;
75
+ }
76
+ </style>
@@ -0,0 +1,88 @@
1
+ <template>
2
+ <div
3
+ v-if="!!variation"
4
+ class="tw-text-small tw-font-medium tw-pt-0.5 tw-line-clamp-1"
5
+ :style="{ color: tagColor }"
6
+ >
7
+ {{ tagText }}
8
+ <kv-countdown-timer
9
+ v-if="variation === 'ending-soon'"
10
+ :ms-left="msLeft"
11
+ />
12
+ </div>
13
+ </template>
14
+
15
+ <script>
16
+ import { differenceInDays, parseISO } from 'date-fns';
17
+ import numeral from 'numeral';
18
+ import KvCountdownTimer from './KvCountdownTimer.vue';
19
+
20
+ const LSE_LOAN_KEY = 'N/A';
21
+
22
+ export default {
23
+ name: 'KvLoanTag',
24
+ components: {
25
+ KvCountdownTimer,
26
+ },
27
+ props: {
28
+ loan: {
29
+ type: Object,
30
+ required: true,
31
+ },
32
+ useExpandedStyles: {
33
+ type: Boolean,
34
+ default: false,
35
+ },
36
+ },
37
+ data() {
38
+ return {
39
+ interval: null,
40
+ msLeft: parseISO(this.loan?.plannedExpirationDate).getTime() - new Date().getTime(),
41
+ };
42
+ },
43
+ computed: {
44
+ isLseLoan() {
45
+ return this.loan?.partnerName?.toUpperCase().includes(LSE_LOAN_KEY) ?? false;
46
+ },
47
+ amountLeft() {
48
+ const loanFundraisingInfo = this.loan?.loanFundraisingInfo ?? { fundedAmount: 0, reservedAmount: 0 };
49
+ const { fundedAmount, reservedAmount } = loanFundraisingInfo;
50
+ return numeral(this.loan?.loanAmount).subtract(fundedAmount).subtract(reservedAmount).value();
51
+ },
52
+ variation() {
53
+ if (this.loan?.matchingText) {
54
+ return 'matched-loan';
55
+ } if (this.isLseLoan) {
56
+ return 'lse-loan';
57
+ } if (differenceInDays(parseISO(this.loan?.plannedExpirationDate), Date.now()) <= 3) {
58
+ return 'ending-soon';
59
+ } if (this.amountLeft < 100 && this.amountLeft >= 0) {
60
+ return 'almost-funded';
61
+ }
62
+ return null;
63
+ },
64
+ tagText() {
65
+ switch (this.variation) {
66
+ case 'lse-loan': return `${this.useExpandedStyles ? '⚡ ' : ''}High community impact`;
67
+ case 'almost-funded': return `${this.useExpandedStyles ? '💸 ' : ''}Almost funded`;
68
+ // eslint-disable-next-line max-len
69
+ case 'matched-loan': return `${this.useExpandedStyles ? '🤝 ' : ''}${this.matchRatio + 1}x matching by ${this.loan?.matchingText}`;
70
+ default: return `${this.useExpandedStyles ? '⏰ ' : ''}Ending soon: `;
71
+ }
72
+ },
73
+ tagColor() {
74
+ if (this.useExpandedStyles) {
75
+ switch (this.variation) {
76
+ case 'almost-funded': return '#AF741C';
77
+ case 'ending-soon': return '#CE2626';
78
+ default: return '#2B7C5F';
79
+ }
80
+ }
81
+ return '#CE4A00';
82
+ },
83
+ matchRatio() {
84
+ return this.loan?.matchRatio;
85
+ },
86
+ },
87
+ };
88
+ </script>
@@ -0,0 +1,44 @@
1
+ <template>
2
+ <div
3
+ class="tw-bg-white tw-rounded tw-px-1 tw-right-6 tw-flex tw-items-center tw-gap-0.5 tw-py-0.5
4
+ tw-text-h5 tw-font-medium"
5
+ >
6
+ <svg
7
+ width="20"
8
+ height="20"
9
+ viewBox="0 0 20 20"
10
+ fill="none"
11
+ xmlns="http://www.w3.org/2000/svg"
12
+ xmlns:xlink="http://www.w3.org/1999/xlink"
13
+ >
14
+ <rect
15
+ width="20"
16
+ height="20"
17
+ fill="url(#pattern0)"
18
+ />
19
+ <defs>
20
+ <pattern
21
+ id="pattern0"
22
+ patternContentUnits="objectBoundingBox"
23
+ width="1"
24
+ height="1"
25
+ >
26
+ <use
27
+ xlink:href="#image0_3145_236"
28
+ transform="scale(0.00347222)"
29
+ />
30
+ </pattern>
31
+ <!-- eslint-disable -->
32
+ <image
33
+ id="image0_3145_236"
34
+ width="288"
35
+ height="288"
36
+ xlink:href=""
37
+ />
38
+ <!-- eslint-enable -->
39
+ </defs>
40
+ </svg>
41
+
42
+ Team pick
43
+ </div>
44
+ </template>