@eturnity/eturnity_reusable_components 8.13.13-epic-shading.0 → 8.13.13-qa-16-03-26.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.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@eturnity/eturnity_reusable_components",
3
- "version": "8.13.13-epic-shading.0",
3
+ "version": "8.13.13-qa-16-03-26.0",
4
4
  "files": [
5
5
  "dist",
6
6
  "src"
@@ -71,7 +71,6 @@ const theme = (() => {
71
71
  transparentWhite2: '#ffffff32',
72
72
  transparentWhite1: '#ffffff16',
73
73
  transparentBlack1: '#263238e6',
74
- transparentBlack2: '#263238e5',
75
74
  transparentBlue1: '#20a4cae6',
76
75
  blueElectric: '#66dffa',
77
76
  eturnityGrey: '#263238',
@@ -44,7 +44,7 @@
44
44
  import styled from 'vue3-styled-components'
45
45
 
46
46
  const ComponentWrapper = styled.div`
47
- min-height: 16px;
47
+ min-height: 18px;
48
48
  `
49
49
 
50
50
  const CheckWrapper = styled('div', { hasLabel: Boolean })`
@@ -179,7 +179,7 @@
179
179
  font-size: 13px;
180
180
  display: grid;
181
181
  align-items: center;
182
- min-height: 16px;
182
+ min-height: 18px;
183
183
  color: ${(props) =>
184
184
  props.isDisabled ? props.theme.colors.grey2 : 'unset'};
185
185
  `
@@ -60,7 +60,7 @@
60
60
  :show-linear-unit-name="showLinearUnitName"
61
61
  :slot-size="slotSize"
62
62
  :text-align="textAlign"
63
- :value="formatWithCurrency(value)"
63
+ :value="formattedValue"
64
64
  @blur="onInputBlur($event)"
65
65
  @focus="focusInput()"
66
66
  @input="onInput($event)"
@@ -74,7 +74,7 @@
74
74
 
75
75
  <UnitContainer
76
76
  v-if="unitName && showLinearUnitName && !hasSlot"
77
- :has-length="!!textInput.length"
77
+ :has-length="hasLength"
78
78
  :is-error="isError"
79
79
  >{{ unitName }}</UnitContainer
80
80
  >
@@ -91,7 +91,7 @@
91
91
  :disabled="isSelectDisabled"
92
92
  :select-width="`${selectWidth}px`"
93
93
  :show-border="false"
94
- @input-change="$emit('select-change', $event)"
94
+ @input-change="handleSelectChange"
95
95
  >
96
96
  <template #selector>
97
97
  <SelectText>{{ getSelectValue }}</SelectText>
@@ -449,8 +449,18 @@
449
449
  background-color: ${({ theme }) => theme.colors.grey4};
450
450
  `
451
451
 
452
+ const EVENT_TYPES = {
453
+ INPUT_FOCUS: 'input-focus',
454
+ INPUT_CHANGE: 'input-change',
455
+ INPUT_BLUR: 'input-blur',
456
+ PRESS_ENTER: 'on-enter-click',
457
+ INPUT_DRAG: 'on-input-drag',
458
+ SELECT_CHANGE: 'select-change',
459
+ }
460
+
452
461
  export default {
453
462
  name: 'InputNumber',
463
+ emits: [...Object.values(EVENT_TYPES)],
454
464
  components: {
455
465
  Container,
456
466
  InputContainer,
@@ -689,9 +699,10 @@
689
699
  },
690
700
  data() {
691
701
  return {
692
- textInput: '',
693
702
  isFocused: false,
694
703
  warningIcon: warningIcon,
704
+ inputValue: null,
705
+ enteredValue: null,
695
706
  }
696
707
  },
697
708
  computed: {
@@ -714,6 +725,14 @@
714
725
 
715
726
  return item ? item.label : '-'
716
727
  },
728
+ formattedValue() {
729
+ return this.isFocused
730
+ ? this.enteredValue
731
+ : this.formatWithCurrency(this.value)
732
+ },
733
+ hasLength() {
734
+ return this.formattedValue !== null && this.formattedValue.length > 0
735
+ },
717
736
  },
718
737
  watch: {
719
738
  focus(value) {
@@ -724,30 +743,19 @@
724
743
  clearInput: function (value) {
725
744
  if (value) {
726
745
  // If the value is typed, then we should clear the textInput on Continue
727
- this.textInput = ''
746
+ this.inputValue = ''
747
+ this.enteredValue = ''
728
748
  }
729
749
  },
730
- },
731
- created() {
732
- if (this.value) {
733
- this.textInput = numberToString({
734
- value: this.value,
735
- numberPrecision: this.numberPrecision,
736
- minDecimals: this.minDecimals,
737
- })
738
- } else if (this.defaultNumber !== null) {
739
- this.textInput = numberToString({
740
- value: this.defaultNumber,
741
- numberPrecision: this.numberPrecision,
742
- minDecimals: this.minDecimals,
743
- })
744
- } else if (this.minNumber !== null) {
745
- this.textInput = numberToString({
746
- value: this.minNumber,
747
- numberPrecision: this.numberPrecision,
748
- minDecimals: this.minDecimals,
749
- })
750
- }
750
+ value: {
751
+ immediate: true,
752
+ handler(val) {
753
+ if (this.value !== this.inputValue && !Number.isNaN(this.value)) {
754
+ this.inputValue = this.value
755
+ this.enteredValue = this.value
756
+ }
757
+ },
758
+ },
751
759
  },
752
760
  mounted() {
753
761
  if (this.focus) {
@@ -786,29 +794,28 @@
786
794
  }
787
795
  },
788
796
  onEnterPress() {
789
- this.$emit('on-enter-click')
797
+ this.$emit(EVENT_TYPES.PRESS_ENTER)
790
798
  this.$refs.inputField1.$el.blur()
791
799
  },
792
- onChangeHandler(event) {
793
- if (isNaN(event) || event === '') {
794
- event = this.defaultNumber
800
+ onChangeHandler(value) {
801
+ if (isNaN(value) || value === '') {
802
+ value = this.defaultNumber
795
803
  ? this.defaultNumber
796
804
  : this.minNumber || this.minNumber === 0
797
805
  ? this.minNumber
798
- : event
806
+ : value
799
807
  }
800
808
  if (!this.allowNegative) {
801
- event = Math.abs(event)
809
+ value = Math.abs(value)
802
810
  }
803
- event = parseFloat(event)
811
+ value = parseFloat(value)
804
812
  // Need to return an integer rather than a string
805
- this.$emit('input-change', event)
813
+ return parseFloat(value)
806
814
  },
807
- onEvaluateCode(event) {
815
+ onEvaluateCode(value) {
808
816
  // function to perform math on the code
809
817
  // filter the string in case of any malicious content
810
- const val = event.target.value
811
- let filtered = val.replace('(auto)', '').replace(/[^-()\d/*+.,]/g, '')
818
+ let filtered = value.replace('(auto)', '').replace(/[^-()\d/*+.,]/g, '')
812
819
  filtered = filtered.split(/([-+*/()])/)
813
820
  let formatted = filtered.map((item) => {
814
821
  if (
@@ -866,48 +873,29 @@
866
873
  return array
867
874
  },
868
875
  onInput(event) {
869
- if (!this.isFocused) {
876
+ this.enteredValue = event.target.value
877
+ if (!this.isFocused || this.enteredValue === this.inputValue) {
870
878
  return
871
879
  }
872
- if (event.target.value === '') {
873
- this.$emit('on-input', '')
874
- }
875
880
  let evaluatedVal
876
881
  try {
877
- evaluatedVal = this.onEvaluateCode(event)
882
+ evaluatedVal = this.onEvaluateCode(String(this.enteredValue))
878
883
  } finally {
879
- if (evaluatedVal && this.value != evaluatedVal) {
880
- this.$emit('on-input', evaluatedVal)
884
+ this.inputValue = this.onChangeHandler(evaluatedVal)
885
+
886
+ if (this.isFocused && typeof this.enteredValue !== 'number') {
887
+ this.$emit(EVENT_TYPES.INPUT_CHANGE, this.inputValue)
881
888
  }
882
889
  }
883
890
  this.textInput = evaluatedVal
884
891
  },
885
892
  onInputBlur(e) {
886
893
  this.isFocused = false
887
- let value = e.target.value
888
- let evaluatedInput = this.onEvaluateCode(e)
889
- this.onChangeHandler(evaluatedInput ? evaluatedInput : value)
890
- if ((evaluatedInput && value.length) || this.minNumber !== null) {
891
- this.textInput = numberToString({
892
- value:
893
- evaluatedInput && value.length
894
- ? evaluatedInput
895
- : this.defaultNumber
896
- ? this.defaultNumber
897
- : this.minNumber,
898
- numberPrecision: this.numberPrecision,
899
- minDecimals: this.minDecimals,
900
- })
901
- }
902
- let adjustedMinValue =
903
- evaluatedInput && evaluatedInput.length
904
- ? evaluatedInput
905
- : this.defaultNumber
906
- ? this.defaultNumber
907
- : this.minNumber || this.minNumber === 0
908
- ? this.minNumber
909
- : ''
910
- this.$emit('input-blur', adjustedMinValue)
894
+ this.enteredValue = this.inputValue
895
+ this.$emit(
896
+ EVENT_TYPES.INPUT_BLUR,
897
+ this.onEvaluateCode(String(this.inputValue))
898
+ )
911
899
  },
912
900
  focusInput() {
913
901
  if (this.disabled) {
@@ -917,7 +905,7 @@
917
905
  this.$nextTick(() => {
918
906
  this.$refs.inputField1.$el.select()
919
907
  })
920
- this.$emit('input-focus')
908
+ this.$emit(EVENT_TYPES.INPUT_FOCUS, this.inputValue)
921
909
  },
922
910
  blurInput() {
923
911
  if (this.disabled) {
@@ -937,7 +925,7 @@
937
925
  : this.minNumber || this.minNumber === 0
938
926
  ? this.minNumber
939
927
  : ''
940
- if ((adjustedMinValue || adjustedMinValue === 0) && !this.isFocused) {
928
+ if (adjustedMinValue || adjustedMinValue === 0) {
941
929
  let input = this.numberToStringEnabled
942
930
  ? numberToString({
943
931
  value: adjustedMinValue,
@@ -950,6 +938,8 @@
950
938
  return input + ' ' + unit
951
939
  } else if (!adjustedMinValue && adjustedMinValue !== 0) {
952
940
  return ''
941
+ } else if (this.isFocused) {
942
+ return value
953
943
  } else {
954
944
  return this.numberToStringEnabled
955
945
  ? numberToString({
@@ -970,14 +960,7 @@
970
960
  e.preventDefault()
971
961
  let value = parseFloat(this.value || 0)
972
962
  value += parseFloat(this.interactionStep) * parseInt(e.movementX)
973
- this.$emit('on-input-drag', value)
974
-
975
- this.textInput = numberToString({
976
- value: value && value.length ? value : this.minNumber,
977
- numberPrecision: this.numberPrecision,
978
- minDecimals: this.minDecimals,
979
- })
980
- //this.value=value
963
+ this.$emit(EVENT_TYPES.INPUT_DRAG, this.onChangeHandler(value))
981
964
  },
982
965
  stopInteract(e) {
983
966
  e.preventDefault()
@@ -985,6 +968,9 @@
985
968
  window.removeEventListener('mouseup', this.stopInteract, false)
986
969
  this.blurInput()
987
970
  },
971
+ handleSelectChange(value) {
972
+ this.$emit(EVENT_TYPES.SELECT_CHANGE, value)
973
+ },
988
974
  },
989
975
  }
990
976
  </script>
@@ -109,11 +109,7 @@
109
109
  >
110
110
  <slot name="selector" :selected-value="selectedValue"></slot>
111
111
  </Selector>
112
- <Caret
113
- class="caret_dropdown"
114
- :color-mode="colorMode"
115
- @click.stop="toggleCaretDropdown"
116
- >
112
+ <Caret class="caret_dropdown" :color-mode="colorMode">
117
113
  <Icon
118
114
  v-if="isDropdownOpen"
119
115
  :color="
@@ -142,12 +138,11 @@
142
138
  v-show="isSelectDropdownShown"
143
139
  ref="dropdown"
144
140
  :bg-color="
145
- dropdownBgColor ||
146
- colorMode == 'dark' ||
147
- colorMode == 'transparent'
141
+ colorMode == 'dark' || colorMode == 'transparent'
148
142
  ? 'black'
149
143
  : 'white'
150
144
  "
145
+ class="rc-select-dropdown"
151
146
  :dropdown-position="dropdownPosition"
152
147
  :font-color="
153
148
  dropdownFontColor ||
@@ -167,10 +162,17 @@
167
162
  :hovered-index="hoveredIndex"
168
163
  :hovered-value="hoveredValue"
169
164
  :is-active="isActive"
165
+ :is-fixed-dropdown-position="isFixedDropdownPosition"
166
+ :is-parent-modal="isParentModal"
170
167
  :min-width="minWidth"
171
168
  :no-relative="noRelative"
172
169
  :option-width="getOptionWidth"
173
170
  :selected-value="selectedValue"
171
+ :style="{
172
+ transform: `translate(${dropdownPosition?.left}px, ${
173
+ noRelative ? 'auto' : `${dropdownPosition?.top}px`
174
+ })`,
175
+ }"
174
176
  @mouseleave="optionLeave"
175
177
  @option-hovered="optionHovered"
176
178
  @option-selected="optionSelected"
@@ -208,7 +210,7 @@
208
210
  // </template>
209
211
  // </Select>
210
212
 
211
- import { Teleport } from 'vue'
213
+ import { Teleport, inject } from 'vue'
212
214
  import styled from 'vue3-styled-components'
213
215
  import InfoText from '../../infoText'
214
216
  import Icon from '../../icon'
@@ -392,14 +394,17 @@
392
394
  selectedValue: Number | String,
393
395
  noRelative: Boolean,
394
396
  minWidth: String,
397
+ isParentModal: Boolean,
398
+ isFixedDropdownPosition: Boolean,
395
399
  }
396
400
  const SelectDropdown = styled('div', selectDropdownAttrs)`
397
401
  box-sizing: border-box;
398
- z-index: ${(props) => (props.isActive ? '2' : '99999')};
399
- position: absolute;
400
- top: ${(props) =>
401
- props.noRelative ? 'auto' : props.dropdownPosition?.top + 'px'};
402
- left: ${(props) => props.dropdownPosition?.left}px;
402
+ z-index: ${(props) =>
403
+ props.isActive ? '2' : props.isParentModal ? '9999999' : '99999'};
404
+ position: ${(props) =>
405
+ props.isFixedDropdownPosition ? 'fixed' : 'absolute'};
406
+ top: 0px;
407
+ left: 0px;
403
408
  border: ${BORDER_WIDTH} solid ${(props) => props.theme.colors.grey4};
404
409
  border-radius: 4px;
405
410
  display: flex;
@@ -647,6 +652,11 @@
647
652
  type: String,
648
653
  required: false,
649
654
  },
655
+ isFixedDropdownPosition: {
656
+ type: Boolean,
657
+ required: false,
658
+ default: false,
659
+ },
650
660
  },
651
661
 
652
662
  data() {
@@ -664,6 +674,17 @@
664
674
  },
665
675
  dropdownWidth: null,
666
676
  hoveredValue: null,
677
+ isDisplayedAtBottom: true,
678
+ selectTopPosition: 0,
679
+ selectAndDropdownDistance: 0,
680
+ animationFrameId: null,
681
+ }
682
+ },
683
+ setup() {
684
+ const modalRef = inject('modalRef')
685
+
686
+ return {
687
+ modalRef,
667
688
  }
668
689
  },
669
690
  computed: {
@@ -721,6 +742,9 @@
721
742
  /windows phone/i.test(userAgent)
722
743
  )
723
744
  },
745
+ isParentModal() {
746
+ return !!this.modalRef
747
+ },
724
748
  },
725
749
  watch: {
726
750
  value(val) {
@@ -734,8 +758,13 @@
734
758
  }, 10)
735
759
  await this.$nextTick()
736
760
  this.handleSetDropdownOffet()
761
+ if (!this.isFixedDropdownPosition) this.calculateSelectTopPosition()
737
762
  } else {
738
763
  this.dropdownPosition.left = null
764
+ if (this.animationFrameId) {
765
+ cancelAnimationFrame(this.animationFrameId)
766
+ this.animationFrameId = null
767
+ }
739
768
  setTimeout(() => {
740
769
  this.isClickOutsideActive = false
741
770
  }, 10)
@@ -748,11 +777,30 @@
748
777
  })
749
778
  }
750
779
  },
780
+ isSelectDropdownShown(isShown) {
781
+ if (!isShown) return
782
+ // Need to wait for 1ms to make sure the dropdown menu is shown in the DOM
783
+ // before getting the distance between the select and the dropdown menu
784
+ setTimeout(() => {
785
+ this.getDistanceBetweenSelectAndDropdownMenu()
786
+ }, 100)
787
+ },
788
+ selectTopPosition() {
789
+ this.dropdownPosition.top =
790
+ this.selectTopPosition +
791
+ this.$refs.select.$el.clientHeight +
792
+ this.selectAndDropdownDistance
793
+ },
751
794
  },
752
795
  mounted() {
753
796
  this.observeDropdownHeight()
754
797
  this.observeSelectWidth()
755
798
  window.addEventListener('resize', this.handleSetDropdownOffet)
799
+ if (!this.isFixedDropdownPosition)
800
+ document.body.addEventListener(
801
+ 'scroll',
802
+ this.calculateSelectTopPosition
803
+ )
756
804
  },
757
805
  beforeMount() {
758
806
  this.selectedValue = this.value
@@ -761,6 +809,12 @@
761
809
  window.removeEventListener('resize', this.handleSetDropdownOffet)
762
810
  if (this.dropdownResizeObserver) this.dropdownResizeObserver.disconnect()
763
811
  if (this.selectResizeObserver) this.selectResizeObserver.disconnect()
812
+ if (!this.isFixedDropdownPosition) {
813
+ document.body.removeEventListener(
814
+ 'scroll',
815
+ this.calculateSelectTopPosition
816
+ )
817
+ }
764
818
  },
765
819
  unmounted() {
766
820
  document.removeEventListener('click', this.clickOutside)
@@ -866,11 +920,11 @@
866
920
  return
867
921
  }
868
922
  await this.$nextTick()
869
- const isDisplayedAtBottom = await this.generateDropdownPosition()
923
+ this.isDisplayedAtBottom = await this.generateDropdownPosition()
870
924
  // If the dropdown menu is going to be displayed at the bottom,
871
925
  // we need reverify its position after a dom update (nextTick)
872
926
  await this.$nextTick()
873
- if (isDisplayedAtBottom) this.generateDropdownPosition()
927
+ if (this.isDisplayedAtBottom) this.generateDropdownPosition()
874
928
  },
875
929
  async generateDropdownPosition() {
876
930
  const isDropdownNotCompletelyVisible =
@@ -963,6 +1017,25 @@
963
1017
  }
964
1018
  }
965
1019
  },
1020
+ getDistanceBetweenSelectAndDropdownMenu() {
1021
+ const wholeSelectTopPosition =
1022
+ this.selectTopPosition + this.$refs.select.$el.clientHeight
1023
+ this.selectAndDropdownDistance =
1024
+ this.dropdownPosition.top - wholeSelectTopPosition
1025
+ },
1026
+ calculateSelectTopPosition() {
1027
+ const selectRef = this.$refs.select
1028
+ if (selectRef) {
1029
+ const currentTopPosition =
1030
+ selectRef.$el.getBoundingClientRect().top + window.scrollY
1031
+ if (this.selectTopPosition !== currentTopPosition) {
1032
+ this.selectTopPosition = currentTopPosition
1033
+ }
1034
+ }
1035
+ this.animationFrameId = requestAnimationFrame(
1036
+ this.calculateSelectTopPosition
1037
+ )
1038
+ },
966
1039
  },
967
1040
  }
968
1041
  </script>
@@ -1,6 +1,7 @@
1
1
  <template>
2
2
  <PageWrapper
3
3
  v-if="isOpen"
4
+ ref="modalRef"
4
5
  :add-padding-top="addPaddingTop"
5
6
  :backdrop="backdrop"
6
7
  :is-open="isOpen"
@@ -36,6 +37,7 @@
36
37
  // <div>Data....</div>
37
38
  // </modal>
38
39
 
40
+ import { ref, provide } from 'vue'
39
41
  import styled from 'vue3-styled-components'
40
42
  import CloseButton from '../../buttons/closeButton'
41
43
  import Spinner from '../../spinner'
@@ -58,14 +60,14 @@
58
60
  props.backdrop == 'dark'
59
61
  ? 'rgba(0, 0, 0, 0.4)'
60
62
  : 'rgba(255, 255, 255, 0.9)'};
61
- z-index: 99999;
63
+ z-index: 9999999;
62
64
  overflow: auto;
63
65
  padding-top: ${(props) => (props.addPaddingTop ? '80px' : '0')};
64
66
 
65
- @media (max-width: 425px) {
66
- background: white;
67
- }
68
- `
67
+ @media (max-width: 425px) {
68
+ background: white;
69
+ }
70
+ `
69
71
 
70
72
  const modalContainerAttrs = { overflow: String, isLoading: Boolean }
71
73
  const ModalContainer = styled('div', modalContainerAttrs)`
@@ -163,6 +165,14 @@
163
165
  default: false,
164
166
  },
165
167
  },
168
+ setup() {
169
+ const modalRef = ref(null)
170
+ provide('modalRef', modalRef)
171
+
172
+ return {
173
+ modalRef,
174
+ }
175
+ },
166
176
  watch: {
167
177
  isOpen: {
168
178
  immediate: true,
@@ -94,7 +94,7 @@ export const stringToNumber = ({
94
94
 
95
95
  export const numberToString = ({ value, numberPrecision, minDecimals }) => {
96
96
  const minimumRounding = minDecimals ? minDecimals : 0
97
- value = parseFloat(value)
97
+ value = !Number.isNaN(parseFloat(value)) ? parseFloat(value) : 0
98
98
  return value.toLocaleString(langForLocaleString(), {
99
99
  minimumFractionDigits: minimumRounding, // removing this for now. Why do we need this to be a minimum amount?
100
100
  maximumFractionDigits:
@@ -1,3 +0,0 @@
1
- <svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
2
- <path fill-rule="evenodd" clip-rule="evenodd" d="M8 0.5C8.27614 0.5 8.5 0.723858 8.5 1V2.86667C8.5 3.14281 8.27614 3.36667 8 3.36667C7.72386 3.36667 7.5 3.14281 7.5 2.86667V1C7.5 0.723858 7.72386 0.5 8 0.5ZM0.5 8C0.5 7.72386 0.723858 7.5 1 7.5H2.86667C3.14281 7.5 3.36667 7.72386 3.36667 8C3.36667 8.27614 3.14281 8.5 2.86667 8.5H1C0.723858 8.5 0.5 8.27614 0.5 8ZM12.6333 8C12.6333 7.72386 12.8572 7.5 13.1333 7.5H15C15.2761 7.5 15.5 7.72386 15.5 8C15.5 8.27614 15.2761 8.5 15 8.5H13.1333C12.8572 8.5 12.6333 8.27614 12.6333 8ZM8.99987 10.8145C9.09475 11.0816 9.39061 11.2244 9.64095 11.0915C10.059 10.8696 10.4293 10.5644 10.728 10.1927C11.1343 9.6872 11.3922 9.0787 11.4728 8.43514C11.5535 7.79158 11.4537 7.13827 11.1847 6.54812C10.9156 5.95796 10.4879 5.45417 9.94915 5.09297C9.41045 4.73177 8.78197 4.52736 8.13385 4.50256C7.48573 4.47776 6.84347 4.63353 6.27873 4.9525C5.71399 5.27148 5.24899 5.7411 4.93563 6.30896C4.70526 6.72642 4.56349 7.1849 4.51696 7.65586C4.48909 7.93792 4.73355 8.15738 5.01665 8.14374C5.29975 8.1301 5.51274 7.8874 5.55771 7.60756C5.60274 7.32731 5.69588 7.05561 5.83426 6.80486C6.05573 6.40352 6.38437 6.07162 6.7835 5.84619C7.18262 5.62076 7.63654 5.51066 8.0946 5.52819C8.55265 5.54572 8.99683 5.69019 9.37756 5.94546C9.75829 6.20074 10.0606 6.55679 10.2507 6.97388C10.4409 7.39098 10.5114 7.8527 10.4544 8.30754C10.3974 8.76237 10.2152 9.19242 9.928 9.54971C9.74858 9.77293 9.5326 9.96227 9.29044 10.1103C9.04864 10.2582 8.90499 10.5474 8.99987 10.8145ZM5.17261 9.95833C5.36097 9.79137 5.64409 9.79046 5.83352 9.95621L8.82925 12.5775C8.93776 12.6724 9 12.8096 9 12.9538V14.5C9 14.7761 9.22386 15 9.5 15C9.77614 15 10 14.7761 10 14.5V12.9538C10 12.5212 9.81328 12.1097 9.48776 11.8249L6.49202 9.20363C5.92373 8.70638 5.0744 8.7091 4.5093 9.20998L1.50504 11.8728C1.18385 12.1575 1 12.5662 1 12.9954V14.5C1 14.7761 1.22386 15 1.5 15C1.77614 15 2 14.7761 2 14.5V12.9954C2 12.8523 2.06128 12.7161 2.16835 12.6212L5.17261 9.95833ZM3.40391 2.69668C3.20865 2.50142 2.89206 2.50142 2.6968 2.69668C2.50154 2.89194 2.50154 3.20852 2.6968 3.40379L4.01673 4.72372C4.212 4.91898 4.52858 4.91898 4.72384 4.72372C4.9191 4.52846 4.9191 4.21187 4.72384 4.01661L3.40391 2.69668ZM11.2764 4.01661C11.0811 4.21187 11.0811 4.52846 11.2764 4.72372C11.4716 4.91898 11.7882 4.91898 11.9835 4.72372L13.3034 3.40379C13.4987 3.20852 13.4987 2.89194 13.3034 2.69668C13.1081 2.50142 12.7916 2.50142 12.5963 2.69668L11.2764 4.01661ZM11.9835 11.2762C11.7882 11.081 11.4716 11.081 11.2764 11.2762C11.0811 11.4715 11.0811 11.7881 11.2764 11.9833L12.5963 13.3033C12.7916 13.4985 13.1081 13.4985 13.3034 13.3033C13.4987 13.108 13.4987 12.7914 13.3034 12.5962L11.9835 11.2762Z" fill="black"/>
3
- </svg>
@@ -1,16 +0,0 @@
1
- const defaultProps = {
2
- title: 'Sample Title',
3
- infoText: 'Sample Info Text',
4
- initialPosition: { top: '0px', left: '0px' },
5
- minWidth: '284px',
6
- maxWidth: '284px',
7
- dragTargets: ['document'],
8
- dragBounds: {},
9
- sampleBody1:
10
- 'Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.',
11
- sampleBody2:
12
- 'Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.',
13
- sampleFooter: 'Sample footer',
14
- }
15
-
16
- export default defaultProps
@@ -1,99 +0,0 @@
1
- import { mount } from '@vue/test-utils'
2
- import DraggableCard from '@/components/draggableCard'
3
- import defaultProps from './defaultProps'
4
-
5
- jest.mock('@/components/icon/iconCache.mjs', () => ({
6
- // need to mock this due to how jest handles import.meta
7
- fetchIcon: jest.fn(() => Promise.resolve('mocked-icon-url.svg')),
8
- }))
9
-
10
- describe('DraggableCard.vue', () => {
11
- it('renders properly with required props', () => {
12
- const wrapper = mount(DraggableCard, { props: defaultProps })
13
- expect(
14
- wrapper.find('[data-test-id="draggable_card_header"]').text()
15
- ).toContain(defaultProps.title)
16
-
17
- // Expect no footer rendered if not provided
18
- const footerElement = wrapper.find('[data-test-id="draggable_card_footer"]')
19
- expect(footerElement.exists()).toBe(false)
20
- })
21
-
22
- it('renders body slot content correctly', () => {
23
- const wrapper = mount(DraggableCard, {
24
- props: defaultProps,
25
- slots: {
26
- body: `
27
- <div class="custom_body1">${defaultProps.sampleBody1}</div>
28
- <div class="custom_body2">${defaultProps.sampleBody2}</div>
29
- `,
30
- },
31
- })
32
- expect(wrapper.find('.custom_body1').exists()).toBe(true)
33
- expect(wrapper.text()).toContain(defaultProps.sampleBody1)
34
- expect(wrapper.find('.custom_body2').exists()).toBe(true)
35
- expect(wrapper.text()).toContain(defaultProps.sampleBody2)
36
- })
37
-
38
- it('renders footer slot content correctly', () => {
39
- const wrapper = mount(DraggableCard, {
40
- props: defaultProps,
41
- slots: {
42
- footer: `<div class="custom_footer">${defaultProps.sampleFooter}</div>`,
43
- },
44
- })
45
- expect(
46
- wrapper.find('[data-test-id="draggable_card_footer"]').exists()
47
- ).toBe(true)
48
- expect(wrapper.find('.custom_footer').exists()).toBe(true)
49
- expect(wrapper.text()).toContain(defaultProps.sampleFooter)
50
- })
51
-
52
- it('emits "on-close" when close button is clicked', async () => {
53
- const wrapper = mount(DraggableCard, { props: defaultProps })
54
- await wrapper
55
- .find('[data-test-id="draggable_card_close_button"]')
56
- .trigger('click')
57
- expect(wrapper.emitted('on-close')).toBeTruthy()
58
- })
59
-
60
- it('toggles collapse state when arrow icon is clicked', async () => {
61
- const wrapper = mount(DraggableCard, {
62
- props: { ...defaultProps, isCollapsible: true },
63
- })
64
- expect(wrapper.vm.isCollapsed).toBe(false)
65
- await wrapper
66
- .find('[data-test-id="draggable_card_collapse_button"]')
67
- .trigger('click')
68
- expect(wrapper.vm.isCollapsed).toBe(true)
69
- })
70
-
71
- it('handles drag start event', async () => {
72
- const wrapper = mount(DraggableCard, { props: defaultProps })
73
- await wrapper
74
- .find('[data-test-id="draggable_card_drag_icon"]')
75
- .trigger('mousedown', { clientX: 100, clientY: 100 })
76
- expect(wrapper.vm.isDragging).toBe(true)
77
- })
78
-
79
- it('handles drag move event', async () => {
80
- const wrapper = mount(DraggableCard, { props: defaultProps })
81
- wrapper.vm.onDragStart(
82
- { preventDefault: () => {}, clientX: 100, clientY: 100 },
83
- false
84
- )
85
- wrapper.vm.onDrag({ preventDefault: () => {}, clientX: 120, clientY: 120 })
86
- expect(wrapper.vm.eventCoordinates.x).toBe(120)
87
- expect(wrapper.vm.eventCoordinates.y).toBe(120)
88
- })
89
-
90
- it('handles drag end event', async () => {
91
- const wrapper = mount(DraggableCard, { props: defaultProps })
92
- wrapper.vm.onDragStart(
93
- { preventDefault: () => {}, clientX: 100, clientY: 100 },
94
- false
95
- )
96
- wrapper.vm.onDragEnd()
97
- expect(wrapper.vm.isDragging).toBe(false)
98
- })
99
- })
@@ -1,79 +0,0 @@
1
- import defaultProps from './defaultProps'
2
- import DraggableCard from './index.vue'
3
- import styled from 'vue3-styled-components'
4
- import theme from '@/assets/theme'
5
-
6
- export default {
7
- title: 'DraggableCard',
8
- component: DraggableCard,
9
- tags: ['autodocs'],
10
- }
11
-
12
- const TextContainer = styled.div`
13
- font-family: ${theme.fonts.mainFont};
14
- color: ${theme.colors.white};
15
- font-weight: 400;
16
- font-size: 14px;
17
- line-height: 21px;
18
- letter-spacing: 0%;
19
- `
20
-
21
- const CompleteTemplate = (args) => {
22
- return {
23
- components: { DraggableCard, TextContainer },
24
- setup() {
25
- return { args }
26
- },
27
- template: `
28
- <DraggableCard v-bind="args">
29
- <template #body>
30
- <TextContainer>{{ args.sampleBody1 }}</TextContainer>
31
- <TextContainer>{{ args.sampleBody2 }}</TextContainer>
32
- </template>
33
- <template #footer>
34
- <TextContainer>{{ args.sampleFooter }}</TextContainer>
35
- </template>
36
- </DraggableCard>
37
- `,
38
- }
39
- }
40
-
41
- export const Default = CompleteTemplate.bind({})
42
- Default.args = {
43
- ...defaultProps,
44
- }
45
-
46
- export const NotCollapsible = CompleteTemplate.bind({})
47
- NotCollapsible.args = {
48
- ...defaultProps,
49
- isCollapsible: false,
50
- }
51
-
52
- export const AdjustWidth = CompleteTemplate.bind({})
53
- AdjustWidth.args = {
54
- ...defaultProps,
55
- minWidth: '100px',
56
- maxWidth: '500px',
57
- }
58
-
59
- const NoFooterTemplate = (args) => {
60
- return {
61
- components: { DraggableCard, TextContainer },
62
- setup() {
63
- return { args }
64
- },
65
- template: `
66
- <DraggableCard v-bind="args">
67
- <template #body>
68
- <TextContainer>{{ args.sampleBody1 }}</TextContainer>
69
- <TextContainer>{{ args.sampleBody2 }}</TextContainer>
70
- </template>
71
- </DraggableCard>
72
- `,
73
- }
74
- }
75
-
76
- export const NoFooter = NoFooterTemplate.bind({})
77
- NoFooter.args = {
78
- ...defaultProps,
79
- }
@@ -1,354 +0,0 @@
1
- <template>
2
- <CardContainer
3
- data-test-id="draggable_card_container"
4
- :initial-position="initialPosition"
5
- :max-width="maxWidth"
6
- :min-width="minWidth"
7
- >
8
- <HeaderContainer data-test-id="draggable_card_header">
9
- <SubHeaderWrapper>
10
- <LeftIconsContainer>
11
- <DragHandleWrapper :class="{ dragging: isDragging }">
12
- <RCIcon
13
- :color="theme.colors.grey3"
14
- data-test-id="draggable_card_drag_icon"
15
- name="drag_icon"
16
- size="14px"
17
- :title="$gettext('drag_drop')"
18
- @mousedown="(event) => onDragStart(event, false)"
19
- @touchstart="
20
- (event) => {
21
- onDragStart(event, true)
22
- }
23
- "
24
- />
25
- </DragHandleWrapper>
26
- <ArrowIconWrapper
27
- v-if="isCollapsible"
28
- data-test-id="draggable_card_collapse_button"
29
- :is-collapsed="isCollapsed"
30
- @click="toggleCollapse"
31
- >
32
- <CollapseArrowIcon />
33
- </ArrowIconWrapper>
34
- </LeftIconsContainer>
35
- <TitleContainer :data-id="titleDataId">
36
- <TextContainer>{{ title }}</TextContainer>
37
- <InfoText v-if="infoText?.length" :text="infoText" />
38
- </TitleContainer>
39
- </SubHeaderWrapper>
40
- <CloseButton
41
- class="close"
42
- data-test-id="draggable_card_close_button"
43
- @click="$emit('on-close')"
44
- />
45
- </HeaderContainer>
46
- <BodyContainer v-if="!isCollapsed" data-test-id="draggable_card_body">
47
- <slot name="body"></slot>
48
- </BodyContainer>
49
- <FooterContainer v-if="$slots.footer" data-test-id="draggable_card_footer">
50
- <slot name="footer"></slot>
51
- </FooterContainer>
52
- </CardContainer>
53
- </template>
54
-
55
- <script>
56
- import styled from 'vue3-styled-components'
57
- import theme from '../../assets/theme.js'
58
- import RCIcon from '../icon'
59
- import CloseButton from '../buttons/closeButton'
60
- import InfoText from '../infoText'
61
- import CollapseArrowIcon from '../../assets/icons/collapse_arrow_icon.svg'
62
-
63
- const CardContainerAttr = {
64
- initialPosition: Object,
65
- minWidth: String,
66
- maxWidth: String,
67
- }
68
- const CardContainer = styled('div', CardContainerAttr)`
69
- position: absolute;
70
- display: flex;
71
- flex-direction: column;
72
- border-radius: 4px;
73
- min-width: ${(props) => props.minWidth};
74
- max-width: ${(props) => props.maxWidth};
75
- background: ${theme.colors.transparentBlack2};
76
- z-index: 5;
77
- ${(props) =>
78
- props.initialPosition?.top && `top: ${props.initialPosition.top};`}
79
- ${(props) =>
80
- props.initialPosition?.right && `right: ${props.initialPosition.right};`}
81
- ${(props) =>
82
- props.initialPosition?.left && `left: ${props.initialPosition.left};`}
83
- ${(props) =>
84
- props.initialPosition?.bottom &&
85
- `bottom: ${props.initialPosition.bottom};`}
86
- `
87
-
88
- const HeaderContainer = styled.div`
89
- display: flex;
90
- align-items: center;
91
- justify-content: space-between;
92
- padding: 8px;
93
- `
94
-
95
- const SubHeaderWrapper = styled.div`
96
- display: flex;
97
- min-width: 180px;
98
- align-items: center;
99
- justify-content: space-between;
100
- height: fit-content;
101
- margin-right: 14px;
102
- `
103
-
104
- const LeftIconsContainer = styled.div`
105
- display: flex;
106
- align-items: center;
107
- gap: 4px;
108
- margin-right: 14px;
109
- `
110
-
111
- const TitleContainer = styled.div`
112
- display: flex;
113
- align-items: center;
114
- gap: 8px;
115
- `
116
-
117
- const TextContainer = styled.div`
118
- font-family: ${theme.fonts.mainFont};
119
- font-weight: 600;
120
- font-size: 14px;
121
- line-height: 19.6px;
122
- letter-spacing: -1%;
123
- color: ${theme.colors.white};
124
- word-break: break-word;
125
- overflow-wrap: break-word;
126
- white-space: normal;
127
- `
128
-
129
- const DragHandleWrapper = styled.div`
130
- display: grid;
131
- align-items: center;
132
- justify-items: center;
133
- width: 26px;
134
- height: 26px;
135
- cursor: grab;
136
- &.dragging {
137
- cursor: grabbing;
138
- }
139
- `
140
-
141
- const collapsedAttrs = {
142
- isCollapsed: Boolean,
143
- }
144
- const ArrowIconWrapper = styled('div', collapsedAttrs)`
145
- display: flex;
146
- align-items: center;
147
- justify-content: center;
148
- width: 8px;
149
- height: 26px;
150
- cursor: pointer;
151
- transition: transform 0.3s ease;
152
- path {
153
- fill: ${theme.colors.white};
154
- }
155
-
156
- ${(props) => props.isCollapsed && 'transform: rotate(-180deg);'}
157
- `
158
-
159
- const BodyContainer = styled.div`
160
- display: flex;
161
- flex-direction: column;
162
- gap: 16px;
163
- padding-top: 8px;
164
- padding-right: 16px;
165
- padding-bottom: 8px;
166
- padding-left: 16px;
167
- word-break: break-word;
168
- overflow-wrap: break-word;
169
- white-space: normal;
170
- `
171
-
172
- const FooterContainer = styled.div`
173
- display: flex;
174
- border-radius: 4px;
175
- align-items: center;
176
- justify-content: space-between;
177
- padding: 8px;
178
- background: ${theme.colors.eturnityGrey};
179
- word-break: break-word;
180
- overflow-wrap: break-word;
181
- white-space: normal;
182
- `
183
-
184
- export default {
185
- name: 'DraggableCard',
186
- components: {
187
- CardContainer,
188
- HeaderContainer,
189
- SubHeaderWrapper,
190
- LeftIconsContainer,
191
- TextContainer,
192
- DragHandleWrapper,
193
- ArrowIconWrapper,
194
- CollapseArrowIcon,
195
- RCIcon,
196
- CloseButton,
197
- InfoText,
198
- TitleContainer,
199
- BodyContainer,
200
- FooterContainer,
201
- },
202
- props: {
203
- minWidth: {
204
- required: true,
205
- type: String,
206
- },
207
- maxWidth: {
208
- required: true,
209
- type: String,
210
- },
211
- initialPosition: {
212
- required: true,
213
- type: Object,
214
- },
215
- title: {
216
- required: true,
217
- type: String,
218
- },
219
- titleDataId: {
220
- required: false,
221
- type: String,
222
- default: '',
223
- },
224
- isCollapsible: {
225
- required: false,
226
- type: Boolean,
227
- default: true,
228
- },
229
- infoText: {
230
- required: false,
231
- type: String,
232
- default: '',
233
- },
234
- dragTargets: {
235
- required: false,
236
- type: Array,
237
- default: () => ['document'],
238
- },
239
- dragBounds: {
240
- required: false,
241
- type: Object,
242
- default: () => {},
243
- },
244
- },
245
- emits: ['on-close'],
246
- data() {
247
- return {
248
- isCollapsed: false,
249
- isDragging: false,
250
- isTouchStart: false,
251
- eventCoordinates: {
252
- x: 0,
253
- y: 0,
254
- },
255
- }
256
- },
257
- computed: {
258
- dragBoundsEnabled() {
259
- const minPosition = this.dragBounds?.min
260
- const maxPosition = this.dragBounds?.max
261
- return (
262
- minPosition?.top &&
263
- minPosition?.left &&
264
- maxPosition?.top &&
265
- maxPosition?.left
266
- )
267
- },
268
- theme() {
269
- return theme
270
- },
271
- },
272
- beforeUnmount() {
273
- this.removeEvents()
274
- },
275
- methods: {
276
- toggleCollapse() {
277
- this.isCollapsed = !this.isCollapsed
278
- },
279
- onDragStart(e, isTouchStart) {
280
- e.preventDefault()
281
- this.isDragging = true
282
- this.isTouchStart = isTouchStart
283
- this.eventCoordinates.x = isTouchStart
284
- ? e.touches[0].clientX
285
- : e.clientX
286
- this.eventCoordinates.y = isTouchStart
287
- ? e.touches[0].clientY
288
- : e.clientY
289
- this.dragTargets.forEach((id) => {
290
- const target =
291
- id === 'document' ? document : document.getElementById(id)
292
- if (target && target.addEventListener) {
293
- target.addEventListener(
294
- isTouchStart ? 'touchend' : 'mouseup',
295
- this.onDragEnd
296
- )
297
- target.addEventListener(
298
- isTouchStart ? 'touchmove' : 'mousemove',
299
- this.onDrag
300
- )
301
- }
302
- })
303
- document.addEventListener(
304
- isTouchStart ? 'touchend' : 'mouseup',
305
- this.onDragEnd
306
- )
307
- },
308
- onDrag(e) {
309
- e.preventDefault()
310
- const eventX = this.isTouchStart ? e.touches[0].clientX : e.clientX
311
- const eventY = this.isTouchStart ? e.touches[0].clientY : e.clientY
312
- const deltaX = this.eventCoordinates.x - eventX
313
- const deltaY = this.eventCoordinates.y - eventY
314
- this.eventCoordinates.x = eventX
315
- this.eventCoordinates.y = eventY
316
- const element = this.$el
317
- let positionY = element.offsetTop - deltaY
318
- let positionX = element.offsetLeft - deltaX
319
- if (this.dragBoundsEnabled) {
320
- const minPosition = this.dragBounds.min
321
- const maxPosition = this.dragBounds.max
322
- positionY = Math.min(
323
- Math.max(positionY, minPosition.top),
324
- maxPosition.top
325
- )
326
- positionX = Math.min(
327
- Math.max(positionX, minPosition.left),
328
- maxPosition.left
329
- )
330
- }
331
- element.style.top = positionY + 'px'
332
- element.style.left = positionX + 'px'
333
- },
334
- onDragEnd() {
335
- this.isDragging = false
336
- this.removeEvents()
337
- },
338
- removeEvents() {
339
- this.dragTargets.forEach((id) => {
340
- const target =
341
- id === 'document' ? document : document.getElementById(id)
342
- if (target && target.removeEventListener) {
343
- target.removeEventListener('mouseup', this.onDragEnd)
344
- target.removeEventListener('touchend', this.onDragEnd)
345
- target.removeEventListener('mousemove', this.onDrag)
346
- target.removeEventListener('touchmove', this.onDrag)
347
- }
348
- })
349
- document.removeEventListener('mouseup', this.onDragEnd)
350
- document.removeEventListener('touchend', this.onDragEnd)
351
- },
352
- },
353
- }
354
- </script>