Compare commits

..

5 Commits

Author SHA1 Message Date
neverland
8d9ccfd8c7
fix(@vant/use): keep forward compatible (#9592) 2021-09-29 18:24:31 +08:00
neverland
95ffe9bc19
perf(@vant/lazyload): reuse some utils (#9591) 2021-09-29 18:11:04 +08:00
neverland
ec5067ce7d
perf: remove passive event polyfill (#9590) 2021-09-29 17:36:50 +08:00
neverland
7aba794d80
perf(@vant/lazyload): remove CustomEvent polyfill (#9589) 2021-09-29 17:31:05 +08:00
neverland
e8186795b7
perf: no longer need transform prefix (#9588)
* perf: no longer need transform prefix

* chore: upd

* chore: upd
2021-09-29 17:04:03 +08:00
15 changed files with 91 additions and 235 deletions

View File

@ -15,9 +15,6 @@ function findRootDir(dir: string): string {
return findRootDir(parentDir);
}
// Colors
export const GREEN = '#07c160';
// Root paths
export const CWD = process.cwd();
export const ROOT = findRootDir(CWD);
@ -29,7 +26,6 @@ export const SITE_DIST_DIR = join(ROOT, 'site-dist');
export const VANT_CONFIG_FILE = join(ROOT, 'vant.config.js');
export const PACKAGE_JSON_FILE = join(ROOT, 'package.json');
export const ROOT_POSTCSS_CONFIG_FILE = join(ROOT, 'postcss.config.js');
export const CACHE_DIR = join(ROOT, 'node_modules/.cache');
// Relative paths
export const DIST_DIR = join(__dirname, '../../dist');
@ -47,7 +43,6 @@ export const SITE_DESKTOP_SHARED_FILE = join(
export const STYLE_DEPS_JSON_FILE = join(DIST_DIR, 'style-deps.json');
// Config files
export const BABEL_CONFIG_FILE = join(CONFIG_DIR, 'babel.config.js');
export const POSTCSS_CONFIG_FILE = join(CONFIG_DIR, 'postcss.config.js');
export const JEST_SETUP_FILE = join(CONFIG_DIR, 'jest.setup.js');
export const JEST_CONFIG_FILE = join(CONFIG_DIR, 'jest.config.js');

View File

@ -25,6 +25,9 @@
"@babel/core": "^7.12.9",
"release-it": "^14.2.2"
},
"dependencies": {
"@vant/use": "^1.3.1"
},
"release-it": {
"git": {
"tag": false,

View File

@ -1,5 +1,5 @@
import { h } from 'vue';
import { inBrowser } from './util';
import { inBrowser } from '@vant/use';
export default (lazy) => ({
props: {

View File

@ -1,5 +1,6 @@
/* eslint-disable max-classes-per-file */
import { find, remove } from './util';
/* eslint-disable prefer-object-spread */
import { remove } from './util';
const defaultOptions = {
selector: 'img',
@ -74,13 +75,13 @@ export default class LazyContainerManager {
}
update(el, binding, vnode) {
const container = find(this._queue, (item) => item.el === el);
const container = this._queue.find((item) => item.el === el);
if (!container) return;
container.update({ el, binding, vnode });
}
unbind(el) {
const container = find(this._queue, (item) => item.el === el);
const container = this._queue.find((item) => item.el === el);
if (!container) return;
container.clear();
remove(this._queue, container);

View File

@ -1,4 +1,5 @@
import { inBrowser, loadImageAsync, noop } from './util';
import { inBrowser } from '@vant/use';
import { loadImageAsync, noop } from './util';
export default (lazyManager) => ({
props: {

View File

@ -1,14 +1,12 @@
import { nextTick } from 'vue';
import { inBrowser, getScrollParent } from '@vant/use';
import {
inBrowser,
CustomEvent,
remove,
find,
_,
on,
off,
throttle,
supportWebp,
getDPR,
scrollParent,
getBestSelectionFromSrcset,
isObject,
hasIntersectionObserver,
@ -148,7 +146,7 @@ export default function () {
}
if (!$parent) {
$parent = scrollParent(el);
$parent = getScrollParent(el);
}
const newListener = new ReactiveListener({
@ -187,7 +185,7 @@ export default function () {
let { src } = value;
src = getBestSelectionFromSrcset(el, this.options.scale) || src;
const exist = find(this.ListenerQueue, (item) => item.el === el);
const exist = this.ListenerQueue.find((item) => item.el === el);
if (!exist) {
this.add(el, binding, vnode);
} else {
@ -213,7 +211,7 @@ export default function () {
remove(el) {
if (!el) return;
this._observer && this._observer.unobserve(el);
const existItem = find(this.ListenerQueue, (item) => item.el === el);
const existItem = this.ListenerQueue.find((item) => item.el === el);
if (existItem) {
this._removeListenerTarget(existItem.$parent);
this._removeListenerTarget(window);
@ -274,7 +272,7 @@ export default function () {
*/
_addListenerTarget(el) {
if (!el) return;
let target = find(this.TargetQueue, (target) => target.el === el);
let target = this.TargetQueue.find((target) => target.el === el);
if (!target) {
target = {
el,
@ -316,7 +314,7 @@ export default function () {
*/
_initListen(el, start) {
this.options.ListenEvents.forEach((evt) =>
_[start ? 'on' : 'off'](el, evt, this.lazyLoadHandler)
(start ? on : off)(el, evt, this.lazyLoadHandler)
);
}

View File

@ -32,7 +32,6 @@ export default class ReactiveListener {
this.elRenderer = elRenderer;
this._imageCache = imageCache;
this.performanceData = {
init: Date.now(),
loadStart: 0,
loadEnd: 0,
};

View File

@ -1,4 +1,4 @@
const inBrowser = typeof window !== 'undefined';
import { inBrowser } from '@vant/use';
export const hasIntersectionObserver =
inBrowser &&
@ -11,32 +11,13 @@ export const modeType = {
observer: 'observer',
};
// CustomEvent polyfill
const CustomEvent = (function () {
if (!inBrowser) return;
if (typeof window.CustomEvent === 'function') return window.CustomEvent;
function CustomEvent(event, params) {
params = params || { bubbles: false, cancelable: false, detail: undefined };
const evt = document.createEvent('CustomEvent');
evt.initCustomEvent(
event,
params.bubbles,
params.cancelable,
params.detail
);
return evt;
}
CustomEvent.prototype = window.Event.prototype;
return CustomEvent;
})();
function remove(arr, item) {
export function remove(arr, item) {
if (!arr.length) return;
const index = arr.indexOf(item);
if (index > -1) return arr.splice(index, 1);
}
function getBestSelectionFromSrcset(el, scale) {
export function getBestSelectionFromSrcset(el, scale) {
if (el.tagName !== 'IMG' || !el.getAttribute('data-srcset')) return;
let options = el.getAttribute('data-srcset');
@ -101,21 +82,10 @@ function getBestSelectionFromSrcset(el, scale) {
return bestSelectedSrc;
}
function find(arr, fn) {
let item;
for (let i = 0, len = arr.length; i < len; i++) {
if (fn(arr[i])) {
item = arr[i];
break;
}
}
return item;
}
const getDPR = (scale = 1) =>
export const getDPR = (scale = 1) =>
inBrowser ? window.devicePixelRatio || scale : scale;
function supportWebp() {
export function supportWebp() {
if (!inBrowser) return false;
let support = true;
@ -133,7 +103,7 @@ function supportWebp() {
return support;
}
function throttle(action, delay) {
export function throttle(action, delay) {
let timeout = null;
let lastRun = 0;
return function (...args) {
@ -154,42 +124,18 @@ function throttle(action, delay) {
};
}
function testSupportsPassive() {
if (!inBrowser) return;
let support = false;
try {
const opts = Object.defineProperty({}, 'passive', {
// eslint-disable-next-line getter-return
get() {
support = true;
},
});
window.addEventListener('test', null, opts);
} catch (e) {
//
}
return support;
export function on(el, type, func) {
el.addEventListener(type, func, {
capture: false,
passive: true,
});
}
const supportsPassive = testSupportsPassive();
export function off(el, type, func) {
el.removeEventListener(type, func, false);
}
const _ = {
on(el, type, func, capture = false) {
if (supportsPassive) {
el.addEventListener(type, func, {
capture,
passive: true,
});
} else {
el.addEventListener(type, func, capture);
}
},
off(el, type, func, capture = false) {
el.removeEventListener(type, func, capture);
},
};
const loadImageAsync = (item, resolve, reject) => {
export const loadImageAsync = (item, resolve, reject) => {
const image = new Image();
if (!item || !item.src) {
@ -202,61 +148,23 @@ const loadImageAsync = (item, resolve, reject) => {
image.crossOrigin = item.cors;
}
image.onload = function () {
image.onload = () =>
resolve({
naturalHeight: image.naturalHeight,
naturalWidth: image.naturalWidth,
src: image.src,
});
};
image.onerror = function (e) {
reject(e);
};
image.onerror = (e) => reject(e);
};
const style = (el, prop) =>
typeof getComputedStyle !== 'undefined'
? getComputedStyle(el, null).getPropertyValue(prop)
: el.style[prop];
const overflow = (el) =>
style(el, 'overflow') + style(el, 'overflow-y') + style(el, 'overflow-x');
const scrollParent = (el) => {
if (!inBrowser) return;
if (!(el instanceof HTMLElement)) {
return window;
}
let parent = el;
while (parent) {
if (parent === document.body || parent === document.documentElement) {
break;
}
if (!parent.parentNode) {
break;
}
if (/(scroll|auto)/.test(overflow(parent))) {
return parent;
}
parent = parent.parentNode;
}
return window;
};
function isObject(obj) {
export function isObject(obj) {
return obj !== null && typeof obj === 'object';
}
function noop() {}
export function noop() {}
class ImageCache {
export class ImageCache {
constructor({ max }) {
this.options = {
max: max || 100,
@ -280,20 +188,3 @@ class ImageCache {
this._caches.shift();
}
}
export {
ImageCache,
inBrowser,
CustomEvent,
remove,
find,
noop,
_,
isObject,
throttle,
supportWebp,
getDPR,
scrollParent,
loadImageAsync,
getBestSelectionFromSrcset,
};

View File

@ -397,6 +397,11 @@
dependencies:
"@types/node" "*"
"@vant/use@^1.3.1":
version "1.3.1"
resolved "https://registry.nlark.com/@vant/use/download/@vant/use-1.3.1.tgz?cache=0&sync_timestamp=1632302722581&other_urls=https%3A%2F%2Fregistry.nlark.com%2F%40vant%2Fuse%2Fdownload%2F%40vant%2Fuse-1.3.1.tgz#8609fc7f86713df0c4ace7109eee6df2f3bd9a0f"
integrity sha1-hgn8f4ZxPfDErOcQnu5t8vO9mg8=
ansi-align@^3.0.0:
version "3.0.0"
resolved "https://registry.npm.taobao.org/ansi-align/download/ansi-align-3.0.0.tgz#b536b371cf687caaef236c18d3e21fe3797467cb"

View File

@ -2,22 +2,6 @@ import { Ref, watch, isRef, unref, onUnmounted, onDeactivated } from 'vue';
import { onMountedOrActivated } from '../onMountedOrActivated';
import { inBrowser } from '../utils';
// eslint-disable-next-line
export let supportsPassive = false;
if (inBrowser) {
try {
const opts = {};
Object.defineProperty(opts, 'passive', {
get() {
supportsPassive = true;
},
});
window.addEventListener('test-passive', null as any, opts);
// eslint-disable-next-line no-empty
} catch (e) {}
}
type TargetRef = EventTarget | Ref<EventTarget | undefined>;
export type UseEventListenerOptions = {
@ -43,11 +27,7 @@ export function useEventListener(
const element = unref(target);
if (element && !attached) {
element.addEventListener(
type,
listener,
supportsPassive ? { capture, passive } : capture
);
element.addEventListener(type, listener, { capture, passive });
attached = true;
}
};

View File

@ -1,5 +1,9 @@
export const inBrowser = typeof window !== 'undefined';
// Keep forward compatible
// should be removed in next major version
export const supportsPassive = true;
export function raf(fn: FrameRequestCallback): number {
return inBrowser ? requestAnimationFrame(fn) : -1;
}

View File

@ -415,8 +415,7 @@ export default defineComponent({
);
if (disabledDay) {
const lastAbledEndDay = getPrevDay(disabledDay);
select([startDay, lastAbledEndDay]);
select([startDay, getPrevDay(disabledDay)]);
} else {
select([startDay, date], true);
}
@ -528,9 +527,9 @@ export default defineComponent({
showTitle={props.showTitle}
showSubtitle={props.showSubtitle}
firstDayOfWeek={dayOffset.value}
onClick-subtitle={(event: MouseEvent) => {
emit('click-subtitle', event);
}}
onClick-subtitle={(event: MouseEvent) =>
emit('click-subtitle', event)
}
/>
<div ref={bodyRef} class={bem('body')} onScroll={onScroll}>
{months.value.map(renderMonth)}
@ -542,9 +541,7 @@ export default defineComponent({
watch(() => props.show, init);
watch(
() => [props.type, props.minDate, props.maxDate],
() => {
reset(getInitialDate(currentDate.value));
}
() => reset(getInitialDate(currentDate.value))
);
watch(
() => props.defaultDate,

View File

@ -24,7 +24,7 @@ export type CascaderOption = {
type CascaderTab = {
options: CascaderOption[];
selectedOption: CascaderOption | null;
selected: CascaderOption | null;
};
export type CascaderFieldNames = {
@ -111,7 +111,7 @@ export default defineComponent({
state.tabs = selectedOptions.map((option) => {
const tab = {
options: optionsCursor,
selectedOption: option,
selected: option,
};
const next = optionsCursor.find(
@ -127,7 +127,7 @@ export default defineComponent({
if (optionsCursor) {
state.tabs.push({
options: optionsCursor,
selectedOption: null,
selected: null,
});
}
@ -142,7 +142,7 @@ export default defineComponent({
state.tabs = [
{
options: props.options,
selectedOption: null,
selected: null,
},
];
};
@ -152,7 +152,7 @@ export default defineComponent({
return;
}
state.tabs[tabIndex].selectedOption = option;
state.tabs[tabIndex].selected = option;
if (state.tabs.length > tabIndex + 1) {
state.tabs = state.tabs.slice(0, tabIndex + 1);
@ -161,7 +161,7 @@ export default defineComponent({
if (option[childrenKey]) {
const nextTab = {
options: option[childrenKey],
selectedOption: null,
selected: null,
};
if (state.tabs[tabIndex + 1]) {
@ -176,7 +176,7 @@ export default defineComponent({
}
const selectedOptions = state.tabs
.map((tab) => tab.selectedOption)
.map((tab) => tab.selected)
.filter(Boolean);
const eventParams = {
@ -260,19 +260,19 @@ export default defineComponent({
);
const renderTab = (tab: CascaderTab, tabIndex: number) => {
const { options, selectedOption } = tab;
const title = selectedOption
? selectedOption[textKey]
const { options, selected } = tab;
const title = selected
? selected[textKey]
: props.placeholder || t('select');
return (
<Tab
title={title}
titleClass={bem('tab', {
unselected: !selectedOption,
unselected: !selected,
})}
>
{renderOptions(options, selectedOption, tabIndex)}
{renderOptions(options, selected, tabIndex)}
</Tab>
);
};
@ -298,9 +298,7 @@ export default defineComponent({
() => props.modelValue,
(value) => {
if (value || value === 0) {
const values = state.tabs.map(
(tab) => tab.selectedOption?.[valueKey]
);
const values = state.tabs.map((tab) => tab.selected?.[valueKey]);
if (values.includes(value)) {
return;
}

View File

@ -1,9 +1,5 @@
import { Ref, watch, onBeforeUnmount, onDeactivated } from 'vue';
import {
getScrollParent,
supportsPassive,
onMountedOrActivated,
} from '@vant/use';
import { getScrollParent, onMountedOrActivated } from '@vant/use';
import { useTouch } from './use-touch';
import { preventDefault } from '../utils';
@ -45,11 +41,7 @@ export function useLockScroll(
const lock = () => {
document.addEventListener('touchstart', touch.start);
document.addEventListener(
'touchmove',
onTouchMove,
supportsPassive ? { passive: false } : false
);
document.addEventListener('touchmove', onTouchMove, { passive: false });
if (!totalLockCount) {
document.body.classList.add(BODY_LOCK_CLASS);

View File

@ -29,10 +29,8 @@ const MOMENTUM_LIMIT_DISTANCE = 15;
const [name, bem] = createNamespace('picker-column');
function getElementTranslateY(element: Element) {
const style = window.getComputedStyle(element);
const transform = style.transform || style.webkitTransform;
const { transform } = window.getComputedStyle(element);
const translateY = transform.slice(7, transform.length - 1).split(', ')[5];
return Number(translateY);
}
@ -316,36 +314,30 @@ export default defineComponent({
watch(
() => props.defaultIndex,
(value) => {
setIndex(value);
}
(value) => setIndex(value)
);
return () => {
const wrapperStyle = {
transform: `translate3d(0, ${state.offset + baseOffset()}px, 0)`,
transitionDuration: `${state.duration}ms`,
transitionProperty: state.duration ? 'all' : 'none',
};
return (
<div
class={[bem(), props.className]}
onTouchstart={onTouchStart}
onTouchmove={onTouchMove}
onTouchend={onTouchEnd}
onTouchcancel={onTouchEnd}
return () => (
<div
class={[bem(), props.className]}
onTouchstart={onTouchStart}
onTouchmove={onTouchMove}
onTouchend={onTouchEnd}
onTouchcancel={onTouchEnd}
>
<ul
ref={wrapper}
style={{
transform: `translate3d(0, ${state.offset + baseOffset()}px, 0)`,
transitionDuration: `${state.duration}ms`,
transitionProperty: state.duration ? 'all' : 'none',
}}
class={bem('wrapper')}
onTransitionend={stopMomentum}
>
<ul
ref={wrapper}
style={wrapperStyle}
class={bem('wrapper')}
onTransitionend={stopMomentum}
>
{renderOptions()}
</ul>
</div>
);
};
{renderOptions()}
</ul>
</div>
);
},
});