quasar
Version:
Build high-performance VueJS user interfaces (SPA, PWA, SSR, Mobile and Desktop) in record time
145 lines (123 loc) • 4.07 kB
JavaScript
import { css } from '../utils/dom.js'
import { position, stop } from '../utils/event.js'
import { addEvt, cleanEvt } from '../utils/touch.js'
import { isKeyCode } from '../utils/key-composition.js'
import { client } from '../plugins/Platform.js'
import throttle from '../utils/throttle.js'
import { $q } from '../install.js'
function showRipple (evt, el, ctx, forceCenter) {
ctx.modifiers.stop === true && stop(evt)
let { center, color } = ctx.modifiers
center = center === true || forceCenter === true
const
node = document.createElement('span'),
innerNode = document.createElement('span'),
pos = position(evt),
{ left, top, width, height } = el.getBoundingClientRect(),
diameter = Math.sqrt(width * width + height * height),
radius = diameter / 2,
centerX = `${(width - diameter) / 2}px`,
x = center ? centerX : `${pos.left - left - radius}px`,
centerY = `${(height - diameter) / 2}px`,
y = center ? centerY : `${pos.top - top - radius}px`
innerNode.className = 'q-ripple__inner'
css(innerNode, {
height: `${diameter}px`,
width: `${diameter}px`,
transform: `translate3d(${x},${y},0) scale3d(.2,.2,1)`,
opacity: 0
})
node.className = `q-ripple${color ? ' text-' + color : ''}`
node.setAttribute('dir', 'ltr')
node.appendChild(innerNode)
el.appendChild(node)
const abort = () => {
node.remove()
clearTimeout(timer)
}
ctx.abort.push(abort)
let timer = setTimeout(() => {
innerNode.classList.add('q-ripple__inner--enter')
innerNode.style.transform = `translate3d(${centerX},${centerY},0) scale3d(1,1,1)`
innerNode.style.opacity = 0.2
timer = setTimeout(() => {
innerNode.classList.remove('q-ripple__inner--enter')
innerNode.classList.add('q-ripple__inner--leave')
innerNode.style.opacity = 0
timer = setTimeout(() => {
node.remove()
ctx.abort.splice(ctx.abort.indexOf(abort), 1)
}, 275)
}, 250)
}, 50)
}
function updateCtx (ctx, { value, modifiers, arg }) {
ctx.enabled = value !== false
if (ctx.enabled === true) {
const cfg = Object.assign({}, $q.config.ripple, modifiers, value)
ctx.modifiers = {
early: cfg.early === true,
stop: cfg.stop === true,
center: cfg.center === true,
color: cfg.color || arg,
keyCodes: [].concat(cfg.keyCodes || 13)
}
}
}
export default {
name: 'ripple',
inserted (el, binding) {
const ctx = {
modifiers: {},
abort: [],
start (evt) {
if (
ctx.enabled === true &&
evt.qSkipRipple !== true &&
// on ENTER in form IE emits a PointerEvent with negative client cordinates
(client.is.ie !== true || evt.clientX >= 0) &&
(
ctx.modifiers.early === true
? ['mousedown', 'touchstart'].includes(evt.type) === true
: evt.type === 'click'
)
) {
showRipple(evt, el, ctx, evt.qKeyEvent === true)
}
},
keystart: throttle(evt => {
if (
ctx.enabled === true &&
evt.qSkipRipple !== true &&
isKeyCode(evt, ctx.modifiers.keyCodes) === true &&
evt.type === `key${ctx.modifiers.early === true ? 'down' : 'up'}`
) {
showRipple(evt, el, ctx, true)
}
}, 300)
}
updateCtx(ctx, binding)
if (el.__qripple) {
el.__qripple_old = el.__qripple
}
el.__qripple = ctx
addEvt(ctx, 'main', [
[ el, 'mousedown', 'start', 'passive' ],
[ el, 'touchstart', 'start', 'passive' ],
[ el, 'click', 'start', 'passive' ],
[ el, 'keydown', 'keystart', 'passive' ],
[ el, 'keyup', 'keystart', 'passive' ]
])
},
update (el, binding) {
el.__qripple !== void 0 && updateCtx(el.__qripple, binding)
},
unbind (el) {
const ctx = el.__qripple_old || el.__qripple
if (ctx !== void 0) {
ctx.abort.forEach(fn => { fn() })
cleanEvt(ctx, 'main')
delete el[el.__qripple_old ? '__qripple_old' : '__qripple']
}
}
}