Blame view
src/directive/sticky.js
2.44 KB
d7d9c38c2 auto commit the c... |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 |
const vueSticky = {} let listenAction vueSticky.install = Vue => { Vue.directive('sticky', { inserted(el, binding) { const params = binding.value || {} const stickyTop = params.stickyTop || 0 const zIndex = params.zIndex || 1000 const elStyle = el.style elStyle.position = '-webkit-sticky' elStyle.position = 'sticky' // if the browser support css sticky(Currently Safari, Firefox and Chrome Canary) // if (~elStyle.position.indexOf('sticky')) { // elStyle.top = `${stickyTop}px`; // elStyle.zIndex = zIndex; // return // } const elHeight = el.getBoundingClientRect().height const elWidth = el.getBoundingClientRect().width elStyle.cssText = `top: ${stickyTop}px; z-index: ${zIndex}` const parentElm = el.parentNode || document.documentElement const placeholder = document.createElement('div') placeholder.style.display = 'none' placeholder.style.width = `${elWidth}px` placeholder.style.height = `${elHeight}px` parentElm.insertBefore(placeholder, el) let active = false const getScroll = (target, top) => { const prop = top ? 'pageYOffset' : 'pageXOffset' const method = top ? 'scrollTop' : 'scrollLeft' let ret = target[prop] if (typeof ret !== 'number') { ret = window.document.documentElement[method] } return ret } const sticky = () => { if (active) { return } if (!elStyle.height) { elStyle.height = `${el.offsetHeight}px` } elStyle.position = 'fixed' elStyle.width = `${elWidth}px` placeholder.style.display = 'inline-block' active = true } const reset = () => { if (!active) { return } elStyle.position = '' placeholder.style.display = 'none' active = false } const check = () => { const scrollTop = getScroll(window, true) const offsetTop = el.getBoundingClientRect().top if (offsetTop < stickyTop) { sticky() } else { if (scrollTop < elHeight + stickyTop) { reset() } } } listenAction = () => { check() } window.addEventListener('scroll', listenAction) }, unbind() { window.removeEventListener('scroll', listenAction) } }) } export default vueSticky |