refactor: backtop
parent
5bcca46fff
commit
b96fc440e9
|
@ -1,17 +0,0 @@
|
||||||
export default function getScroll(target, top) {
|
|
||||||
if (typeof window === 'undefined') {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
const prop = top ? 'pageYOffset' : 'pageXOffset';
|
|
||||||
const method = top ? 'scrollTop' : 'scrollLeft';
|
|
||||||
const isWindow = target === window;
|
|
||||||
|
|
||||||
let ret = isWindow ? target[prop] : target[method];
|
|
||||||
// ie6,7,8 standard mode
|
|
||||||
if (isWindow && typeof ret !== 'number') {
|
|
||||||
ret = window.document.documentElement[method];
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
|
@ -0,0 +1,27 @@
|
||||||
|
export function isWindow(obj: any) {
|
||||||
|
return obj !== null && obj !== undefined && obj === obj.window;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function getScroll(
|
||||||
|
target: HTMLElement | Window | Document | null,
|
||||||
|
top: boolean,
|
||||||
|
): number {
|
||||||
|
if (typeof window === 'undefined') {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
const method = top ? 'scrollTop' : 'scrollLeft';
|
||||||
|
let result = 0;
|
||||||
|
if (isWindow(target)) {
|
||||||
|
result = (target as Window)[top ? 'pageYOffset' : 'pageXOffset'];
|
||||||
|
} else if (target instanceof Document) {
|
||||||
|
result = target.documentElement[method];
|
||||||
|
} else if (target) {
|
||||||
|
result = (target as HTMLElement)[method];
|
||||||
|
}
|
||||||
|
if (target && !isWindow(target) && typeof result !== 'number') {
|
||||||
|
result = ((target as HTMLElement).ownerDocument || (target as Document)).documentElement?.[
|
||||||
|
method
|
||||||
|
];
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
|
@ -1,9 +1,10 @@
|
||||||
import getScroll from './getScroll';
|
import raf from './raf';
|
||||||
|
import getScroll, { isWindow } from './getScroll';
|
||||||
import { easeInOutCubic } from './easings';
|
import { easeInOutCubic } from './easings';
|
||||||
|
|
||||||
interface ScrollToOptions {
|
interface ScrollToOptions {
|
||||||
/** Scroll container, default as window */
|
/** Scroll container, default as window */
|
||||||
getContainer?: () => HTMLElement | Window;
|
getContainer?: () => HTMLElement | Window | Document;
|
||||||
/** Scroll end callback */
|
/** Scroll end callback */
|
||||||
callback?: () => any;
|
callback?: () => any;
|
||||||
/** Animation duration, default as 450 */
|
/** Animation duration, default as 450 */
|
||||||
|
@ -12,7 +13,6 @@ interface ScrollToOptions {
|
||||||
|
|
||||||
export default function scrollTo(y: number, options: ScrollToOptions = {}) {
|
export default function scrollTo(y: number, options: ScrollToOptions = {}) {
|
||||||
const { getContainer = () => window, callback, duration = 450 } = options;
|
const { getContainer = () => window, callback, duration = 450 } = options;
|
||||||
|
|
||||||
const container = getContainer();
|
const container = getContainer();
|
||||||
const scrollTop = getScroll(container, true);
|
const scrollTop = getScroll(container, true);
|
||||||
const startTime = Date.now();
|
const startTime = Date.now();
|
||||||
|
@ -21,16 +21,18 @@ export default function scrollTo(y: number, options: ScrollToOptions = {}) {
|
||||||
const timestamp = Date.now();
|
const timestamp = Date.now();
|
||||||
const time = timestamp - startTime;
|
const time = timestamp - startTime;
|
||||||
const nextScrollTop = easeInOutCubic(time > duration ? duration : time, scrollTop, y, duration);
|
const nextScrollTop = easeInOutCubic(time > duration ? duration : time, scrollTop, y, duration);
|
||||||
if (container === window) {
|
if (isWindow(container)) {
|
||||||
window.scrollTo(window.pageXOffset, nextScrollTop);
|
(container as Window).scrollTo(window.pageXOffset, nextScrollTop);
|
||||||
|
} else if (container instanceof HTMLDocument || container.constructor.name === 'HTMLDocument') {
|
||||||
|
(container as HTMLDocument).documentElement.scrollTop = nextScrollTop;
|
||||||
} else {
|
} else {
|
||||||
(container as HTMLElement).scrollTop = nextScrollTop;
|
(container as HTMLElement).scrollTop = nextScrollTop;
|
||||||
}
|
}
|
||||||
if (time < duration) {
|
if (time < duration) {
|
||||||
requestAnimationFrame(frameFunc);
|
raf(frameFunc);
|
||||||
} else if (typeof callback === 'function') {
|
} else if (typeof callback === 'function') {
|
||||||
callback();
|
callback();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
requestAnimationFrame(frameFunc);
|
raf(frameFunc);
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,8 +7,13 @@ import {
|
||||||
onBeforeUnmount,
|
onBeforeUnmount,
|
||||||
onMounted,
|
onMounted,
|
||||||
reactive,
|
reactive,
|
||||||
|
PropType,
|
||||||
|
ref,
|
||||||
|
watch,
|
||||||
|
onDeactivated,
|
||||||
|
computed,
|
||||||
} from 'vue';
|
} from 'vue';
|
||||||
import classNames from '../_util/classNames';
|
import VerticalAlignTopOutlined from '@ant-design/icons-vue/VerticalAlignTopOutlined';
|
||||||
import PropTypes from '../_util/vue-types';
|
import PropTypes from '../_util/vue-types';
|
||||||
import addEventListener from '../vc-util/Dom/addEventListener';
|
import addEventListener from '../vc-util/Dom/addEventListener';
|
||||||
import getScroll from '../_util/getScroll';
|
import getScroll from '../_util/getScroll';
|
||||||
|
@ -16,18 +21,12 @@ import { getTransitionProps, Transition } from '../_util/transition';
|
||||||
import { defaultConfigProvider } from '../config-provider';
|
import { defaultConfigProvider } from '../config-provider';
|
||||||
import scrollTo from '../_util/scrollTo';
|
import scrollTo from '../_util/scrollTo';
|
||||||
import { withInstall } from '../_util/type';
|
import { withInstall } from '../_util/type';
|
||||||
|
import throttleByAnimationFrame from '../_util/throttleByAnimationFrame';
|
||||||
function getDefaultTarget() {
|
|
||||||
return window;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const backTopProps = {
|
export const backTopProps = {
|
||||||
// 滚动高度达到此参数值才出现 BackTop
|
|
||||||
visibilityHeight: PropTypes.number.def(400),
|
visibilityHeight: PropTypes.number.def(400),
|
||||||
// 回到顶部所需时间(ms) @4.4.0
|
|
||||||
duration: PropTypes.number.def(450),
|
duration: PropTypes.number.def(450),
|
||||||
// 设置需要监听其滚动事件的元素,值为一个返回对应 DOM 元素的函数
|
target: Function as PropType<() => HTMLElement | Window | Document>,
|
||||||
target: PropTypes.func,
|
|
||||||
prefixCls: PropTypes.string,
|
prefixCls: PropTypes.string,
|
||||||
onClick: PropTypes.func,
|
onClick: PropTypes.func,
|
||||||
// visible: PropTypes.looseBool, // Only for test. Don't use it.
|
// visible: PropTypes.looseBool, // Only for test. Don't use it.
|
||||||
|
@ -42,12 +41,15 @@ const BackTop = defineComponent({
|
||||||
emits: ['click'],
|
emits: ['click'],
|
||||||
setup(props, { slots, attrs, emit }) {
|
setup(props, { slots, attrs, emit }) {
|
||||||
const configProvider = inject('configProvider', defaultConfigProvider);
|
const configProvider = inject('configProvider', defaultConfigProvider);
|
||||||
|
const domRef = ref();
|
||||||
const state = reactive({
|
const state = reactive({
|
||||||
visible: false,
|
visible: false,
|
||||||
scrollEvent: null,
|
scrollEvent: null,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const getDefaultTarget = () =>
|
||||||
|
domRef.value && domRef.value.ownerDocument ? domRef.value.ownerDocument : window;
|
||||||
|
|
||||||
const scrollToTop = (e: Event) => {
|
const scrollToTop = (e: Event) => {
|
||||||
const { target = getDefaultTarget, duration } = props;
|
const { target = getDefaultTarget, duration } = props;
|
||||||
scrollTo(0, {
|
scrollTo(0, {
|
||||||
|
@ -57,51 +59,85 @@ const BackTop = defineComponent({
|
||||||
emit('click', e);
|
emit('click', e);
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleScroll = () => {
|
const handleScroll = throttleByAnimationFrame((e: Event | { target: any }) => {
|
||||||
const { visibilityHeight, target = getDefaultTarget } = props;
|
const { visibilityHeight } = props;
|
||||||
const scrollTop = getScroll(target(), true);
|
const scrollTop = getScroll(e.target, true);
|
||||||
state.visible = scrollTop > visibilityHeight;
|
state.visible = scrollTop > visibilityHeight;
|
||||||
|
});
|
||||||
|
|
||||||
|
const bindScrollEvent = () => {
|
||||||
|
const { target } = props;
|
||||||
|
const getTarget = target || getDefaultTarget;
|
||||||
|
const container = getTarget();
|
||||||
|
state.scrollEvent = addEventListener(container, 'scroll', (e: Event) => {
|
||||||
|
handleScroll(e);
|
||||||
|
});
|
||||||
|
handleScroll({
|
||||||
|
target: container,
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const scrollRemove = () => {
|
||||||
|
if (state.scrollEvent) {
|
||||||
|
state.scrollEvent.remove();
|
||||||
|
}
|
||||||
|
(handleScroll as any).cancel();
|
||||||
|
};
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => props.target,
|
||||||
|
() => {
|
||||||
|
scrollRemove();
|
||||||
|
nextTick(() => {
|
||||||
|
bindScrollEvent();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
nextTick(() => {
|
nextTick(() => {
|
||||||
const getTarget = props.target || getDefaultTarget;
|
bindScrollEvent();
|
||||||
state.scrollEvent = addEventListener(getTarget(), 'scroll', handleScroll);
|
|
||||||
handleScroll();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
onActivated(() => {
|
onActivated(() => {
|
||||||
nextTick(() => {
|
nextTick(() => {
|
||||||
handleScroll();
|
bindScrollEvent();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
onBeforeUnmount(() => {
|
onDeactivated(() => {
|
||||||
if (state.scrollEvent) {
|
scrollRemove();
|
||||||
state.scrollEvent.remove();
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
return () => {
|
onBeforeUnmount(() => {
|
||||||
const { prefixCls: customizePrefixCls } = props;
|
scrollRemove();
|
||||||
|
});
|
||||||
|
|
||||||
const getPrefixCls = configProvider.getPrefixCls;
|
const prefixCls = computed(() => configProvider.getPrefixCls('back-top', props.prefixCls));
|
||||||
const prefixCls = getPrefixCls('back-top', customizePrefixCls);
|
|
||||||
const classString = classNames(prefixCls, attrs.class);
|
return () => {
|
||||||
const defaultElement = (
|
const defaultElement = (
|
||||||
<div class={`${prefixCls}-content`}>
|
<div class={`${prefixCls.value}-content`}>
|
||||||
<div class={`${prefixCls}-icon`} />
|
<div class={`${prefixCls.value}-icon`}>
|
||||||
|
<VerticalAlignTopOutlined />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
const divProps = {
|
const divProps = {
|
||||||
...attrs,
|
...attrs,
|
||||||
onClick: scrollToTop,
|
onClick: scrollToTop,
|
||||||
class: classString,
|
class: {
|
||||||
|
[`${prefixCls.value}`]: true,
|
||||||
|
[`${attrs.class}`]: attrs.class,
|
||||||
|
[`${prefixCls}-rtl`]: configProvider.direction === 'rtl',
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const backTopBtn = state.visible ? (
|
const backTopBtn = state.visible ? (
|
||||||
<div {...divProps}>{slots.default?.() || defaultElement}</div>
|
<div {...divProps} ref={domRef}>
|
||||||
|
{slots.default?.() || defaultElement}
|
||||||
|
</div>
|
||||||
) : null;
|
) : null;
|
||||||
const transitionProps = getTransitionProps('fade');
|
const transitionProps = getTransitionProps('fade');
|
||||||
return <Transition {...transitionProps}>{backTopBtn}</Transition>;
|
return <Transition {...transitionProps}>{backTopBtn}</Transition>;
|
||||||
|
|
|
@ -18,6 +18,12 @@
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&-rtl {
|
||||||
|
right: auto;
|
||||||
|
left: 100px;
|
||||||
|
direction: rtl;
|
||||||
|
}
|
||||||
|
|
||||||
&-content {
|
&-content {
|
||||||
width: 40px;
|
width: 40px;
|
||||||
height: 40px;
|
height: 40px;
|
||||||
|
@ -26,20 +32,17 @@
|
||||||
text-align: center;
|
text-align: center;
|
||||||
background-color: @back-top-bg;
|
background-color: @back-top-bg;
|
||||||
border-radius: 20px;
|
border-radius: 20px;
|
||||||
transition: all 0.3s @ease-in-out;
|
transition: all 0.3s;
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
background-color: @back-top-hover-bg;
|
background-color: @back-top-hover-bg;
|
||||||
transition: all 0.3s @ease-in-out;
|
transition: all 0.3s;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&-icon {
|
&-icon {
|
||||||
width: 14px;
|
font-size: 24px;
|
||||||
height: 16px;
|
line-height: 40px;
|
||||||
margin: 12px auto;
|
|
||||||
background: url()
|
|
||||||
~'100%/100%' no-repeat;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
2
v2-doc
2
v2-doc
|
@ -1 +1 @@
|
||||||
Subproject commit 4f1ece5073f736e79c6eb22527a9a83e8c6182b3
|
Subproject commit d197053285b81e77718621c0b5b94cb3b21831a2
|
Loading…
Reference in New Issue