Browse Source

add stay-on-click prop to solve case of clicking user avatar in status popover

Henry Jameson 2 years ago
parent
commit
d5bc825616

+ 14 - 4
src/components/popover/popover.js

@@ -40,7 +40,10 @@ const Popover = {
     overlayCenters: Boolean,
     overlayCenters: Boolean,
 
 
     // What selector (witin popover!) to use for determining center of popover
     // What selector (witin popover!) to use for determining center of popover
-    overlayCentersSelector: String
+    overlayCentersSelector: String,
+
+    // Lets hover popover stay when clicking inside of it
+    stayOnClick: Boolean
   },
   },
   inject: ['popoversZLayer'], // override popover z layer
   inject: ['popoversZLayer'], // override popover z layer
   data () {
   data () {
@@ -50,6 +53,7 @@ const Popover = {
       // with popovers refusing to be hidden when user wants to interact with something in below popover
       // with popovers refusing to be hidden when user wants to interact with something in below popover
       lockReEntry: false,
       lockReEntry: false,
       hidden: true,
       hidden: true,
+      pinned: false,
       styles: {},
       styles: {},
       oldSize: { width: 0, height: 0 },
       oldSize: { width: 0, height: 0 },
       scrollable: null,
       scrollable: null,
@@ -196,9 +200,10 @@ const Popover = {
     },
     },
     showPopover () {
     showPopover () {
       if (this.disabled) return
       if (this.disabled) return
+      this.pinned = false
       const wasHidden = this.hidden
       const wasHidden = this.hidden
       this.hidden = false
       this.hidden = false
-      if (this.trigger === 'click') {
+      if (this.trigger === 'click' || this.stayOnClick) {
         document.addEventListener('click', this.onClickOutside)
         document.addEventListener('click', this.onClickOutside)
       }
       }
       this.scrollable.addEventListener('scroll', this.onScroll)
       this.scrollable.addEventListener('scroll', this.onScroll)
@@ -227,7 +232,7 @@ const Popover = {
       }
       }
     },
     },
     onMouseleave (e) {
     onMouseleave (e) {
-      if (this.trigger === 'hover') {
+      if (this.trigger === 'hover' && !this.pinned) {
         this.graceTimeout = setTimeout(() => this.hidePopover(), 1)
         this.graceTimeout = setTimeout(() => this.hidePopover(), 1)
       }
       }
     },
     },
@@ -240,7 +245,7 @@ const Popover = {
       }
       }
     },
     },
     onMouseleaveContent (e) {
     onMouseleaveContent (e) {
-      if (this.trigger === 'hover') {
+      if (this.trigger === 'hover' && !this.pinned) {
         this.graceTimeout = setTimeout(() => this.hidePopover(), 1)
         this.graceTimeout = setTimeout(() => this.hidePopover(), 1)
       }
       }
     },
     },
@@ -259,6 +264,11 @@ const Popover = {
       if (this.$el.contains(e.target)) return
       if (this.$el.contains(e.target)) return
       this.hidePopover()
       this.hidePopover()
     },
     },
+    onClickContent (e) {
+      if (this.trigger === 'hover' && this.stayOnClick) {
+        this.pinned = true
+      }
+    },
     onScroll (e) {
     onScroll (e) {
       this.updateStyles()
       this.updateStyles()
     },
     },

+ 21 - 0
src/components/popover/popover.vue

@@ -21,12 +21,22 @@
           :class="popoverClass || 'popover-default'"
           :class="popoverClass || 'popover-default'"
           @mouseenter="onMouseenterContent"
           @mouseenter="onMouseenterContent"
           @mouseleave="onMouseleaveContent"
           @mouseleave="onMouseleaveContent"
+          @click="onClickContent"
         >
         >
           <slot
           <slot
             name="content"
             name="content"
             class="popover-inner"
             class="popover-inner"
             :close="hidePopover"
             :close="hidePopover"
           />
           />
+          <div
+            v-if="stayOnClick && pinned"
+            class="pinned-tooltip-icon popover popover-default"
+          >
+            <FAIcon
+              icon="thumbtack"
+              class="faint"
+            />
+          </div>
         </div>
         </div>
       </transition>
       </transition>
     </teleport>
     </teleport>
@@ -51,6 +61,17 @@
   box-shadow: var(--popupShadow);
   box-shadow: var(--popupShadow);
 }
 }
 
 
+.pinned-tooltip-icon {
+  position: absolute;
+  top: -1em;
+  left: -1em;
+  width: 2em;
+  height: 2em;
+  display: flex;
+  justify-content: center;
+  align-items: center;
+}
+
 .popover-default {
 .popover-default {
   &:after {
   &:after {
     content: '';
     content: '';

+ 1 - 0
src/components/status_popover/status_popover.vue

@@ -1,6 +1,7 @@
 <template>
 <template>
   <Popover
   <Popover
     trigger="hover"
     trigger="hover"
+    :stay-on-click="true"
     popover-class="popover-default status-popover"
     popover-class="popover-default status-popover"
     :bound-to="{ x: 'container' }"
     :bound-to="{ x: 'container' }"
     @show="enter"
     @show="enter"