@veritree/ui 0.22.1 → 0.22.2-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 (75) hide show
  1. package/index.js +64 -72
  2. package/mixins/floating-ui-content.js +1 -22
  3. package/mixins/floating-ui-item.js +101 -51
  4. package/mixins/floating-ui.js +7 -9
  5. package/mixins/form-control-icon.js +5 -5
  6. package/mixins/form-control.js +17 -18
  7. package/nuxt.js +30 -23
  8. package/package.json +14 -6
  9. package/src/components/Alert/VTAlert.vue +55 -14
  10. package/src/components/Button/VTButton.vue +20 -12
  11. package/src/components/Checkbox/VTCheckbox.vue +134 -0
  12. package/src/components/Checkbox/VTCheckboxLabel.vue +3 -0
  13. package/src/components/Checkbox/VTCheckboxText.vue +20 -0
  14. package/src/components/Dialog/VTDialog.vue +22 -32
  15. package/src/components/Dialog/VTDialogClose.vue +19 -25
  16. package/src/components/Dialog/VTDialogContent.vue +24 -19
  17. package/src/components/Dialog/VTDialogFooter.vue +11 -16
  18. package/src/components/Dialog/VTDialogHeader.vue +16 -18
  19. package/src/components/Dialog/VTDialogMain.vue +11 -18
  20. package/src/components/Dialog/VTDialogOverlay.vue +14 -18
  21. package/src/components/Dialog/VTDialogTitle.vue +10 -7
  22. package/src/components/Disclosure/VTDisclosureContent.vue +1 -1
  23. package/src/components/Disclosure/VTDisclosureDetails.vue +1 -1
  24. package/src/components/Disclosure/VTDisclosureHeader.vue +2 -2
  25. package/src/components/Disclosure/VTDisclosureIcon.vue +1 -1
  26. package/src/components/Drawer/VTDrawer.vue +6 -15
  27. package/src/components/Drawer/VTDrawerClose.vue +5 -5
  28. package/src/components/Drawer/VTDrawerContent.vue +9 -9
  29. package/src/components/Drawer/VTDrawerFooter.vue +3 -3
  30. package/src/components/Drawer/VTDrawerHeader.vue +4 -4
  31. package/src/components/Drawer/VTDrawerMain.vue +5 -5
  32. package/src/components/Drawer/VTDrawerOverlay.vue +6 -6
  33. package/src/components/Drawer/VTDrawerTitle.vue +5 -5
  34. package/src/components/DropdownMenu/VTDropdownMenu.vue +19 -6
  35. package/src/components/DropdownMenu/VTDropdownMenuContent.vue +14 -6
  36. package/src/components/DropdownMenu/VTDropdownMenuDivider.vue +7 -16
  37. package/src/components/DropdownMenu/VTDropdownMenuItem.vue +7 -3
  38. package/src/components/DropdownMenu/VTDropdownMenuLabel.vue +1 -10
  39. package/src/components/DropdownMenu/VTDropdownMenuTrigger.vue +2 -4
  40. package/src/components/Form/VTFormFeedback.vue +11 -5
  41. package/src/components/Form/VTFormGroup.vue +5 -7
  42. package/src/components/Form/VTFormLabel.vue +22 -0
  43. package/src/components/Form/VTFormRow.vue +5 -0
  44. package/src/components/Form/VTInput.vue +7 -7
  45. package/src/components/Form/VTInputIcon.vue +1 -2
  46. package/src/components/Form/VTInputPassword.vue +14 -5
  47. package/src/components/Form/VTTextarea.vue +1 -1
  48. package/src/components/Image/VTImage.vue +33 -4
  49. package/src/components/Listbox/VTListbox.vue +105 -14
  50. package/src/components/Listbox/VTListboxContent.vue +3 -7
  51. package/src/components/Listbox/VTListboxItem.vue +127 -6
  52. package/src/components/Listbox/VTListboxLabel.vue +3 -4
  53. package/src/components/Listbox/VTListboxList.vue +3 -24
  54. package/src/components/Listbox/VTListboxSearch.vue +64 -55
  55. package/src/components/Listbox/VTListboxTrigger.vue +10 -4
  56. package/src/components/Popover/VTPopover.vue +19 -0
  57. package/src/components/Popover/VTPopoverItem.vue +6 -2
  58. package/src/components/ProgressBar/VTProgressBar.vue +21 -3
  59. package/src/components/Skeleton/VTSkeleton.vue +11 -0
  60. package/src/components/Skeleton/VTSkeletonItem.vue +9 -0
  61. package/src/components/Tabs/VTTab.vue +6 -5
  62. package/src/components/Tabs/VTTabGroup.vue +9 -7
  63. package/src/components/Tabs/VTTabPanel.vue +4 -5
  64. package/src/components/Tooltip/VTTooltip.vue +65 -0
  65. package/src/components/Tooltip/VTTooltipContent.vue +59 -0
  66. package/src/components/Tooltip/VTTooltipTrigger.vue +98 -0
  67. package/src/components/Transitions/FadeInOut.vue +2 -2
  68. package/src/components/Utils/FloatingUi.vue +56 -24
  69. package/src/utils/components.js +18 -0
  70. package/src/components/Input/VTInput.vue +0 -82
  71. package/src/components/Input/VTInputDate.vue +0 -36
  72. package/src/components/Input/VTInputFile.vue +0 -60
  73. package/src/components/Input/VTInputUpload.vue +0 -54
  74. package/src/components/Modal/VTModal.vue +0 -69
  75. package/src/utils/genId.js +0 -13
@@ -48,11 +48,30 @@ export default {
48
48
  type: Boolean,
49
49
  default: false,
50
50
  },
51
+ placement: {
52
+ type: String,
53
+ default: 'bottom-start',
54
+ },
51
55
  },
52
56
 
53
57
  data() {
54
58
  return {
55
59
  componentId: genId(),
60
+ /**
61
+ * Explaining the need for the floatingUiMinWidth data
62
+ *
63
+ * The floating ui is a result of two items:
64
+ *
65
+ * 1. Trigger: the action button
66
+ * 2. Content: the popper/wrapper that appears after triggering the action button
67
+ *
68
+ * By default, the content will match the triggers width.
69
+ * The problem with this is that the trigger width
70
+ * might be too small causing the content to not fit
71
+ * what is inside it properly. So, to avoid this,
72
+ * a min width is needed.
73
+ */
74
+ floatingUiMinWidth: 200,
56
75
  };
57
76
  },
58
77
 
@@ -5,7 +5,7 @@
5
5
  // default styles
6
6
  headless
7
7
  ? 'popover-item'
8
- : 'relative z-10 -mx-3 flex items-center gap-2 px-3 py-2 text-inherit no-underline hover:bg-secondary-200/10',
8
+ : 'hover:bg-secondary-200/10 relative z-10 -mx-3 flex items-center gap-2 px-3 py-2 text-inherit no-underline',
9
9
  ]"
10
10
  @click="onClick"
11
11
  >
@@ -44,7 +44,11 @@ export default {
44
44
  },
45
45
 
46
46
  as() {
47
- return this.href ? 'a' : this.to ? 'NuxtLink' : 'button';
47
+ return this.href
48
+ ? 'a'
49
+ : this.to
50
+ ? resolveComponent('NuxtLink')
51
+ : 'button';
48
52
  },
49
53
  },
50
54
 
@@ -7,7 +7,7 @@
7
7
  ]"
8
8
  role="progressbar"
9
9
  aria-valuemin="0"
10
- aria-valuemax="100"
10
+ :aria-valuemax="max"
11
11
  :aria-valuenow="value"
12
12
  :aria-label="label"
13
13
  >
@@ -15,9 +15,9 @@
15
15
  :class="[
16
16
  headless
17
17
  ? 'progress-bar__indicator'
18
- : 'absolute left-0 h-full bg-secondary-300 transition-all',
18
+ : 'bg-secondary-200 absolute left-0 h-full transition-all',
19
19
  ]"
20
- :style="{ width: `${value}%` }"
20
+ :style="{ width: `${percentageComputed}%` }"
21
21
  ></div>
22
22
  </div>
23
23
  </template>
@@ -37,6 +37,24 @@ export default {
37
37
  type: [String, Number],
38
38
  default: 0,
39
39
  },
40
+ max: {
41
+ type: [String, Number],
42
+ default: 100,
43
+ },
44
+ percentage: {
45
+ type: [String, Number],
46
+ default: null,
47
+ },
48
+ },
49
+
50
+ computed: {
51
+ percentageComputed() {
52
+ if (typeof this.percentage !== 'undefined' && this.percentage !== null) {
53
+ return this.percentage;
54
+ }
55
+
56
+ return (this.value / this.max) * 100;
57
+ },
40
58
  },
41
59
  };
42
60
  </script>
@@ -0,0 +1,11 @@
1
+ <template>
2
+ <div class="animate-pulse">
3
+ <slot />
4
+ </div>
5
+ </template>
6
+
7
+ <script>
8
+ export default {
9
+ name: 'VTSkeleton',
10
+ };
11
+ </script>
@@ -0,0 +1,9 @@
1
+ <template>
2
+ <div class="bg-gray-300" />
3
+ </template>
4
+
5
+ <script>
6
+ export default {
7
+ name: 'VTSkeletonItem',
8
+ };
9
+ </script>
@@ -18,11 +18,11 @@
18
18
  ]"
19
19
  role="tab"
20
20
  type="button"
21
- @click.stop="onClick"
21
+ @click.prevent.stop="onClick"
22
22
  @keydown="onKeyDown"
23
23
  @blur="onBlur"
24
24
  >
25
- <slot></slot>
25
+ <slot :selected="selected"></slot>
26
26
  </button>
27
27
  </template>
28
28
 
@@ -32,7 +32,7 @@ import { keys } from '../../utils/keyboard';
32
32
  export default {
33
33
  name: 'VTTabItem',
34
34
 
35
- inject: ['api'],
35
+ inject: ['apiTabs'],
36
36
 
37
37
  props: {
38
38
  headless: {
@@ -43,7 +43,7 @@ export default {
43
43
 
44
44
  data() {
45
45
  return {
46
- api: this.api(),
46
+ api: this.apiTabs(),
47
47
  index: null,
48
48
  indexFocus: null,
49
49
  selected: false,
@@ -60,7 +60,7 @@ export default {
60
60
  this.api.registerTab(this);
61
61
  },
62
62
 
63
- beforeDestroy() {
63
+ beforeUnmount() {
64
64
  this.api.unregisterTab(this.index);
65
65
  },
66
66
 
@@ -106,6 +106,7 @@ export default {
106
106
  onClick() {
107
107
  this.api.selectTab(this.index);
108
108
  this.$emit('change');
109
+ this.$emit('click');
109
110
  },
110
111
 
111
112
  onKeyDown(event) {
@@ -5,10 +5,10 @@
5
5
  </template>
6
6
 
7
7
  <style scoped>
8
- .TabGroup.vertical{
8
+ .TabGroup.vertical {
9
9
  display: flex;
10
10
  }
11
- .TabGroup.vertical .TabList{
11
+ .TabGroup.vertical .TabList {
12
12
  display: flex;
13
13
  flex-direction: column;
14
14
  }
@@ -17,15 +17,17 @@
17
17
  <script>
18
18
  export default {
19
19
  name: 'VTTabGroup',
20
+
20
21
  props: {
21
- vertical: {
22
- type: Boolean,
23
- default: false
24
- }
22
+ vertical: {
23
+ type: Boolean,
24
+ default: false,
25
+ },
25
26
  },
27
+
26
28
  provide() {
27
29
  return {
28
- api: () => {
30
+ apiTabs: () => {
29
31
  const registerTab = (tab) => {
30
32
  _register(this.tabs, tab);
31
33
  if (tab.index === 0) tab.select();
@@ -14,11 +14,10 @@
14
14
  export default {
15
15
  name: 'VTTabPanel',
16
16
 
17
- inject: ['api'],
17
+ inject: ['apiTabs'],
18
18
 
19
19
  data() {
20
20
  return {
21
- api: this.api(),
22
21
  index: null,
23
22
  visible: false,
24
23
  };
@@ -31,11 +30,11 @@ export default {
31
30
  },
32
31
 
33
32
  mounted() {
34
- this.api.registerTabPanel(this);
33
+ this.apiTabs().registerTabPanel(this);
35
34
  },
36
35
 
37
- beforeDestroy() {
38
- this.api.unregisterTabPanel(this.id);
36
+ beforeUnmount() {
37
+ this.apiTabs().unregisterTabPanel(this.id);
39
38
  },
40
39
 
41
40
  methods: {
@@ -0,0 +1,65 @@
1
+ <template>
2
+ <div>
3
+ <slot />
4
+ </div>
5
+ </template>
6
+
7
+ <script>
8
+ import { floatingUiMixin } from '../../../mixins/floating-ui';
9
+ import { genId } from '../../utils/ids';
10
+
11
+ export default {
12
+ name: 'VTTooltip',
13
+
14
+ mixins: [floatingUiMixin],
15
+
16
+ props: {
17
+ delayDuration: {
18
+ type: [String, Number],
19
+ default: 500,
20
+ },
21
+ placement: {
22
+ type: String,
23
+ default: 'bottom',
24
+ },
25
+ },
26
+
27
+ data() {
28
+ return {
29
+ floatingUiMinWidth: 0,
30
+ };
31
+ },
32
+
33
+ provide() {
34
+ return {
35
+ apiTooltip: () => {
36
+ const registerTrigger = (trigger) => {
37
+ if (!trigger) return;
38
+ this.componentTrigger = trigger;
39
+ };
40
+
41
+ const registerContent = (content) => {
42
+ if (!content) return;
43
+ this.componentContent = content;
44
+ };
45
+
46
+ return {
47
+ id: this.componentId,
48
+ component: this.component,
49
+ componentTrigger: this.componentTrigger,
50
+ componentContent: this.componentContent,
51
+ delayDuration: this.delayDuration,
52
+ registerTrigger,
53
+ registerContent,
54
+ };
55
+ },
56
+ };
57
+ },
58
+
59
+ data() {
60
+ return {
61
+ componentId: genId(),
62
+ };
63
+ },
64
+ };
65
+ </script>
@@ -0,0 +1,59 @@
1
+ <template>
2
+ <FloatingUi
3
+ :id="id"
4
+ :visible="visible"
5
+ :headless="headless"
6
+ component="tooltip"
7
+ >
8
+ <slot></slot>
9
+ </FloatingUi>
10
+ </template>
11
+
12
+ <script>
13
+ import { floatingUiContentMixin } from '../../../mixins/floating-ui-content';
14
+
15
+ export default {
16
+ name: 'VTTooltipContent',
17
+
18
+ inheritAttrs: false,
19
+
20
+ mixins: [floatingUiContentMixin],
21
+
22
+ inject: ['apiTooltip'],
23
+
24
+ data() {
25
+ return {
26
+ visible: false,
27
+ };
28
+ },
29
+
30
+ computed: {
31
+ id() {
32
+ return `tooltip-content-${this.apiTooltip().id}`;
33
+ },
34
+
35
+ component() {
36
+ return this.apiTooltip().component;
37
+ },
38
+
39
+ componentTrigger() {
40
+ return this.apiTooltip().componentTrigger;
41
+ },
42
+
43
+ ariaLabelledby() {
44
+ if (!this.componentTrigger) return null;
45
+ return this.visible ? this.componentTrigger.id : null;
46
+ },
47
+ },
48
+
49
+ mounted() {
50
+ const content = {
51
+ id: this.id,
52
+ hide: this.hide,
53
+ show: this.show,
54
+ };
55
+
56
+ this.apiTooltip().registerContent(content);
57
+ },
58
+ };
59
+ </script>
@@ -0,0 +1,98 @@
1
+ <template>
2
+ <div
3
+ :id="id"
4
+ :aria-describedby="ariaDescribedBy"
5
+ @mouseenter="onMouseenter"
6
+ @mouseleave="onmouseout"
7
+ >
8
+ <slot />
9
+ </div>
10
+ </template>
11
+
12
+ <script>
13
+ let tooltipTriggerTimeout = null;
14
+
15
+ export default {
16
+ name: 'VTTooltipTrigger',
17
+
18
+ inject: ['apiTooltip'],
19
+
20
+ data() {
21
+ return {
22
+ expanded: false,
23
+ };
24
+ },
25
+
26
+ computed: {
27
+ id() {
28
+ return `tooltip-trigger-${this.apiTooltip().id}`;
29
+ },
30
+
31
+ componentContent() {
32
+ return this.apiTooltip().componentContent;
33
+ },
34
+
35
+ ariaDescribedBy() {
36
+ if (!this.componentContent) return null;
37
+ return this.expanded ? this.componentContent.id : null;
38
+ },
39
+ },
40
+
41
+ mounted() {
42
+ const trigger = {
43
+ cancel: this.cancel,
44
+ id: this.id,
45
+ };
46
+
47
+ this.apiTooltip().registerTrigger(trigger);
48
+ },
49
+
50
+ methods: {
51
+ onMouseenter(e) {
52
+ tooltipTriggerTimeout = setTimeout(() => {
53
+ this.init(e);
54
+ }, this.apiTooltip().delayDuration);
55
+ },
56
+
57
+ onmouseout() {
58
+ clearTimeout(tooltipTriggerTimeout);
59
+ this.cancel();
60
+ },
61
+
62
+ init(e) {
63
+ if (!this.componentContent) {
64
+ return;
65
+ }
66
+
67
+ if (this.expanded) {
68
+ this.cancel();
69
+ return;
70
+ }
71
+
72
+ this.expanded = true;
73
+
74
+ // delay stop propagation to close other visible
75
+ // dropdowns and delay click event to control
76
+ // this dropdown visibility
77
+ this.showComponentContent();
78
+ },
79
+
80
+ cancel() {
81
+ if (!this.componentContent) {
82
+ return;
83
+ }
84
+
85
+ this.expanded = false;
86
+ this.hideComponentContent();
87
+ },
88
+
89
+ showComponentContent() {
90
+ this.componentContent.show();
91
+ },
92
+
93
+ hideComponentContent() {
94
+ this.componentContent.hide();
95
+ },
96
+ },
97
+ };
98
+ </script>
@@ -1,10 +1,10 @@
1
1
  <template>
2
2
  <transition
3
3
  enter-active-class="duration-300 ease-out"
4
- enter-class="transform opacity-0"
4
+ enter-from-class="transform opacity-0"
5
5
  enter-to-class="opacity-100"
6
6
  leave-active-class="duration-300 ease-in"
7
- leave-class="opacity-100"
7
+ leave-from-class="opacity-100"
8
8
  leave-to-class="transform opacity-0"
9
9
  @after-enter="afterEnter"
10
10
  @after-leave="afterLeave"
@@ -1,60 +1,92 @@
1
1
  <template>
2
- <Portal>
2
+ <Teleport to="body">
3
3
  <transition
4
4
  enter-active-class="duration-200 ease-out"
5
- enter-class="translate-y-[15px] opacity-0"
6
- enter-to-class="translate-y-0 opacity-100"
5
+ :enter-from-class="transitionEnterClass"
6
+ :enter-to-class="transitionEnterToClass"
7
7
  leave-active-class="duration-200 ease-in"
8
- leave-class="translate-y-0 opacity-100"
9
- leave-to-class="translate-y-[15px] opacity-0"
10
- @after-leave="hide"
8
+ :leave-from-class="transitionLeaveClass"
9
+ :leave-to-class="transitionLeaveToClass"
10
+ @after-leave="hidden"
11
+ @after-enter="shown"
11
12
  >
12
13
  <div
13
14
  v-if="visible"
14
- :id="id"
15
15
  :class="[
16
16
  headless
17
- ? null
18
- : 'absolute z-50 grid overflow-x-hidden rounded-md py-2 px-3 border-gray-100 bg-white shadow-300',
19
- floatingUiClass ? floatingUiClass : null,
17
+ ? `${this.component}-content`
18
+ : `absolute z-50 grid overflow-hidden rounded-md shadow-md ${this.classes} ${this.portalClass}`,
20
19
  ]"
20
+ v-bind="$attrs"
21
21
  >
22
- <slot></slot>
22
+ <slot />
23
23
  </div>
24
24
  </transition>
25
- </Portal>
25
+ </Teleport>
26
26
  </template>
27
27
 
28
28
  <script>
29
- import { Portal } from '@linusborg/vue-simple-portal';
30
-
31
29
  export default {
32
- components: {
33
- Portal,
34
- },
30
+ inheritAttrs: false,
35
31
 
36
32
  props: {
33
+ component: {
34
+ type: String,
35
+ default: null,
36
+ },
37
37
  headless: {
38
38
  type: Boolean,
39
39
  default: false,
40
40
  },
41
- id: {
42
- type: [String, Number],
43
- default: null,
44
- },
45
41
  visible: {
46
42
  type: Boolean,
47
43
  default: false,
48
44
  },
49
- floatingUiClass: {
45
+ ariaActivedescendant: {
46
+ type: String,
47
+ default: null,
48
+ },
49
+ portalClass: {
50
50
  type: [String, Function],
51
51
  default: null,
52
52
  },
53
53
  },
54
54
 
55
+ computed: {
56
+ isTooltip() {
57
+ return this.component === 'tooltip';
58
+ },
59
+
60
+ classes() {
61
+ return this.isTooltip
62
+ ? 'bg-gray-800 text-sm text-white py-1 px-2'
63
+ : 'bg-white py-2 px-3';
64
+ },
65
+
66
+ transitionEnterClass() {
67
+ return this.isTooltip ? 'opacity-0' : 'translate-y-[15px] opacity-0';
68
+ },
69
+
70
+ transitionEnterToClass() {
71
+ return this.isTooltip ? 'opacity-100' : 'translate-y-0 opacity-100';
72
+ },
73
+
74
+ transitionLeaveClass() {
75
+ return this.transitionEnterToClass;
76
+ },
77
+
78
+ transitionLeaveToClass() {
79
+ return this.transitionEnterClass;
80
+ },
81
+ },
82
+
55
83
  methods: {
56
- hide() {
57
- this.$emit('hide');
84
+ shown() {
85
+ this.$emit('shown');
86
+ },
87
+
88
+ hidden() {
89
+ this.$emit('hidden');
58
90
  },
59
91
  },
60
92
  };
@@ -0,0 +1,18 @@
1
+ /**
2
+ *
3
+ * @param {HTMLElement} el
4
+ * @param {HTMLElement} parent
5
+ */
6
+ export const scrollElementIntoView = (el, parent) => {
7
+ // this works better than scrollIntoView
8
+ if (parent.scrollHeight <= parent.clientHeight) return;
9
+
10
+ const scrollBottom = parent.clientHeight + parent.scrollTop;
11
+ const elBottom = el.offsetTop + el.offsetHeight;
12
+
13
+ if (elBottom > scrollBottom) {
14
+ parent.scrollTop = elBottom - parent.clientHeight;
15
+ } else if (el.offsetTop < parent.scrollTop) {
16
+ parent.scrollTop = el.offsetTop;
17
+ }
18
+ };
@@ -1,82 +0,0 @@
1
- <template>
2
- <input
3
- :class="classes"
4
- class="form-control"
5
- :data-theme="theme"
6
- :type="type"
7
- :value="value"
8
- v-on="listeners"
9
- />
10
- </template>
11
-
12
- <script>
13
- export default {
14
- name: 'VTInput',
15
-
16
- props: {
17
- lazy: {
18
- type: Boolean,
19
- default: false,
20
- },
21
- type: {
22
- type: String,
23
- default: 'text',
24
- },
25
- theme: {
26
- type: String,
27
- default: null,
28
- validator(value) {
29
- return ['dark'].includes(value);
30
- },
31
- },
32
- variant: {
33
- type: [String, Object],
34
- default: '',
35
- validator(value) {
36
- if (value === '' || typeof value === 'object') {
37
- return true;
38
- }
39
-
40
- return ['success', 'warning', 'error'].includes(value);
41
- },
42
- },
43
- value: {
44
- type: [String, Number, Object, Array],
45
- default: null,
46
- },
47
- },
48
-
49
- computed: {
50
- classes() {
51
- const classes = {};
52
-
53
- if (this.variant) {
54
- classes[`form-control--${this.variant}`] = true;
55
- }
56
-
57
- return classes;
58
- },
59
-
60
- listeners() {
61
- // `Object.assign` merges objects together to form a new object
62
- return Object.assign(
63
- {},
64
- // We add all the listeners from the parent
65
- this.$listeners,
66
- // Then we can add custom listeners or override the
67
- // behavior of some listeners.
68
- {
69
- // This ensures that the component works with v-model
70
- input: (event) => {
71
- if (this.lazy) return;
72
- this.$emit('input', event.target.value);
73
- },
74
- blur: (event) => {
75
- this.$emit('blur', event);
76
- },
77
- }
78
- );
79
- },
80
- },
81
- };
82
- </script>