2019-01-12 03:33:27 +00:00
|
|
|
import PropTypes from '../_util/vue-types';
|
|
|
|
import ScrollNumber from './ScrollNumber';
|
2020-08-31 08:53:19 +00:00
|
|
|
import classNames from '../_util/classNames';
|
2021-06-07 09:35:03 +00:00
|
|
|
import { getPropsSlot, flattenChildren } from '../_util/props-util';
|
2019-01-12 03:33:27 +00:00
|
|
|
import { cloneElement } from '../_util/vnode';
|
2020-10-22 14:45:21 +00:00
|
|
|
import { getTransitionProps, Transition } from '../_util/transition';
|
2022-03-26 14:52:54 +00:00
|
|
|
import type { ExtractPropTypes, CSSProperties, PropType } from 'vue';
|
2021-06-26 01:35:40 +00:00
|
|
|
import { defineComponent, computed, ref, watch } from 'vue';
|
2021-02-25 05:39:39 +00:00
|
|
|
import Ribbon from './Ribbon';
|
2021-04-11 14:09:41 +00:00
|
|
|
import { isPresetColor } from './utils';
|
2021-06-07 09:35:03 +00:00
|
|
|
import useConfigInject from '../_util/hooks/useConfigInject';
|
|
|
|
import isNumeric from '../_util/isNumeric';
|
2022-03-26 14:52:54 +00:00
|
|
|
import type { PresetStatusColorType } from '../_util/colors';
|
2018-03-09 09:50:33 +00:00
|
|
|
|
2022-03-26 14:52:54 +00:00
|
|
|
export const badgeProps = () => ({
|
2018-03-09 09:50:33 +00:00
|
|
|
/** Number to show in badge */
|
2021-06-07 09:35:03 +00:00
|
|
|
count: PropTypes.any,
|
2022-03-26 14:52:54 +00:00
|
|
|
showZero: { type: Boolean, default: undefined },
|
2018-03-09 09:50:33 +00:00
|
|
|
/** Max count to show */
|
2022-03-26 14:52:54 +00:00
|
|
|
overflowCount: { type: Number, default: 99 },
|
2018-03-09 09:50:33 +00:00
|
|
|
/** whether to show red dot without number */
|
2022-03-26 14:52:54 +00:00
|
|
|
dot: { type: Boolean, default: undefined },
|
|
|
|
prefixCls: String,
|
|
|
|
scrollNumberPrefixCls: String,
|
|
|
|
status: { type: String as PropType<PresetStatusColorType> },
|
|
|
|
size: { type: String as PropType<'default' | 'small'>, default: 'default' },
|
|
|
|
color: String,
|
2021-12-16 09:20:18 +00:00
|
|
|
text: PropTypes.any,
|
2022-03-26 14:52:54 +00:00
|
|
|
offset: Array as unknown as PropType<[number | string, number | string]>,
|
|
|
|
numberStyle: { type: Object as PropType<CSSProperties>, default: undefined as CSSProperties },
|
|
|
|
title: String,
|
|
|
|
});
|
2021-02-25 05:39:39 +00:00
|
|
|
|
2022-03-26 14:52:54 +00:00
|
|
|
export type BadgeProps = Partial<ExtractPropTypes<ReturnType<typeof badgeProps>>>;
|
2021-06-07 09:35:03 +00:00
|
|
|
|
|
|
|
export default defineComponent({
|
2018-04-08 13:17:20 +00:00
|
|
|
name: 'ABadge',
|
2021-02-25 07:41:43 +00:00
|
|
|
Ribbon,
|
2021-06-07 09:35:03 +00:00
|
|
|
inheritAttrs: false,
|
2022-03-26 14:52:54 +00:00
|
|
|
props: badgeProps(),
|
2021-06-07 09:35:03 +00:00
|
|
|
slots: ['text', 'count'],
|
|
|
|
setup(props, { slots, attrs }) {
|
|
|
|
const { prefixCls, direction } = useConfigInject('badge', props);
|
|
|
|
|
|
|
|
// ================================ Misc ================================
|
|
|
|
const numberedDisplayCount = computed(() => {
|
2021-06-23 15:08:16 +00:00
|
|
|
return (
|
|
|
|
(props.count as number) > (props.overflowCount as number)
|
|
|
|
? `${props.overflowCount}+`
|
|
|
|
: props.count
|
|
|
|
) as string | number | null;
|
2021-06-07 09:35:03 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
const hasStatus = computed(
|
|
|
|
() =>
|
|
|
|
(props.status !== null && props.status !== undefined) ||
|
|
|
|
(props.color !== null && props.color !== undefined),
|
|
|
|
);
|
|
|
|
|
|
|
|
const isZero = computed(
|
|
|
|
() => numberedDisplayCount.value === '0' || numberedDisplayCount.value === 0,
|
|
|
|
);
|
|
|
|
|
2022-03-12 01:56:32 +00:00
|
|
|
const showAsDot = computed(() => props.dot && !isZero.value);
|
2021-06-07 09:35:03 +00:00
|
|
|
|
|
|
|
const mergedCount = computed(() => (showAsDot.value ? '' : numberedDisplayCount.value));
|
|
|
|
|
|
|
|
const isHidden = computed(() => {
|
|
|
|
const isEmpty =
|
|
|
|
mergedCount.value === null || mergedCount.value === undefined || mergedCount.value === '';
|
|
|
|
return (isEmpty || (isZero.value && !props.showZero)) && !showAsDot.value;
|
|
|
|
});
|
|
|
|
|
|
|
|
// Count should be cache in case hidden change it
|
|
|
|
const livingCount = ref(props.count);
|
|
|
|
|
|
|
|
// We need cache count since remove motion should not change count display
|
|
|
|
const displayCount = ref(mergedCount.value);
|
|
|
|
|
|
|
|
// We will cache the dot status to avoid shaking on leaved motion
|
|
|
|
const isDotRef = ref(showAsDot.value);
|
|
|
|
|
|
|
|
watch(
|
|
|
|
[() => props.count, mergedCount, showAsDot],
|
|
|
|
() => {
|
|
|
|
if (!isHidden.value) {
|
|
|
|
livingCount.value = props.count;
|
|
|
|
displayCount.value = mergedCount.value;
|
|
|
|
isDotRef.value = showAsDot.value;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{ immediate: true },
|
|
|
|
);
|
|
|
|
|
|
|
|
// Shared styles
|
|
|
|
const statusCls = computed(() => ({
|
|
|
|
[`${prefixCls.value}-status-dot`]: hasStatus.value,
|
|
|
|
[`${prefixCls.value}-status-${props.status}`]: !!props.status,
|
|
|
|
[`${prefixCls.value}-status-${props.color}`]: isPresetColor(props.color),
|
|
|
|
}));
|
|
|
|
|
|
|
|
const statusStyle = computed(() => {
|
|
|
|
if (props.color && !isPresetColor(props.color)) {
|
|
|
|
return { background: props.color };
|
|
|
|
} else {
|
|
|
|
return {};
|
2019-01-03 12:46:53 +00:00
|
|
|
}
|
2021-06-07 09:35:03 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
const scrollNumberCls = computed(() => ({
|
|
|
|
[`${prefixCls.value}-dot`]: isDotRef.value,
|
|
|
|
[`${prefixCls.value}-count`]: !isDotRef.value,
|
|
|
|
[`${prefixCls.value}-count-sm`]: props.size === 'small',
|
|
|
|
[`${prefixCls.value}-multiple-words`]:
|
|
|
|
!isDotRef.value && displayCount.value && displayCount.value.toString().length > 1,
|
2021-07-19 01:58:22 +00:00
|
|
|
[`${prefixCls.value}-status-${props.status}`]: !!props.status,
|
2021-06-07 09:35:03 +00:00
|
|
|
[`${prefixCls.value}-status-${props.color}`]: isPresetColor(props.color),
|
|
|
|
}));
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
const { offset, title, color } = props;
|
|
|
|
const style = attrs.style as CSSProperties;
|
|
|
|
const text = getPropsSlot(slots, props, 'text');
|
|
|
|
const pre = prefixCls.value;
|
|
|
|
const count = livingCount.value;
|
|
|
|
let children = flattenChildren(slots.default?.());
|
|
|
|
children = children.length ? children : null;
|
|
|
|
|
|
|
|
const visible = !!(!isHidden.value || slots.count);
|
|
|
|
|
|
|
|
// =============================== Styles ===============================
|
|
|
|
const mergedStyle = (() => {
|
|
|
|
if (!offset) {
|
|
|
|
return { ...style };
|
|
|
|
}
|
|
|
|
|
|
|
|
const offsetStyle: CSSProperties = {
|
|
|
|
marginTop: isNumeric(offset[1]) ? `${offset[1]}px` : offset[1],
|
|
|
|
};
|
|
|
|
if (direction.value === 'rtl') {
|
|
|
|
offsetStyle.left = `${parseInt(offset[0] as string, 10)}px`;
|
|
|
|
} else {
|
|
|
|
offsetStyle.right = `${-parseInt(offset[0] as string, 10)}px`;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
...offsetStyle,
|
|
|
|
...style,
|
|
|
|
};
|
|
|
|
})();
|
|
|
|
|
|
|
|
// =============================== Render ===============================
|
|
|
|
// >>> Title
|
|
|
|
const titleNode =
|
|
|
|
title ?? (typeof count === 'string' || typeof count === 'number' ? count : undefined);
|
|
|
|
|
|
|
|
// >>> Status Text
|
|
|
|
const statusTextNode =
|
|
|
|
visible || !text ? null : <span class={`${pre}-status-text`}>{text}</span>;
|
|
|
|
|
|
|
|
// >>> Display Component
|
2021-06-20 14:53:42 +00:00
|
|
|
const displayNode =
|
|
|
|
typeof count === 'object' || (count === undefined && slots.count)
|
|
|
|
? cloneElement(
|
|
|
|
count ?? slots.count?.(),
|
|
|
|
{
|
|
|
|
style: mergedStyle,
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
)
|
|
|
|
: null;
|
2020-03-07 11:45:13 +00:00
|
|
|
|
2021-06-07 09:35:03 +00:00
|
|
|
const badgeClassName = classNames(
|
|
|
|
pre,
|
|
|
|
{
|
|
|
|
[`${pre}-status`]: hasStatus.value,
|
|
|
|
[`${pre}-not-a-wrapper`]: !children,
|
|
|
|
[`${pre}-rtl`]: direction.value === 'rtl',
|
|
|
|
},
|
|
|
|
attrs.class,
|
2019-01-12 03:33:27 +00:00
|
|
|
);
|
2018-03-09 09:50:33 +00:00
|
|
|
|
2021-06-07 09:35:03 +00:00
|
|
|
// <Badge status="success" />
|
|
|
|
if (!children && hasStatus.value) {
|
|
|
|
const statusTextColor = mergedStyle.color;
|
|
|
|
return (
|
|
|
|
<span {...attrs} class={badgeClassName} style={mergedStyle}>
|
|
|
|
<span class={statusCls.value} style={statusStyle.value} />
|
|
|
|
<span style={{ color: statusTextColor }} class={`${pre}-status-text`}>
|
|
|
|
{text}
|
|
|
|
</span>
|
2020-03-07 11:45:13 +00:00
|
|
|
</span>
|
2021-06-07 09:35:03 +00:00
|
|
|
);
|
|
|
|
}
|
2018-03-09 09:50:33 +00:00
|
|
|
|
2021-06-07 09:35:03 +00:00
|
|
|
const transitionProps = getTransitionProps(children ? `${pre}-zoom` : '', {
|
|
|
|
appear: false,
|
|
|
|
});
|
2022-03-26 14:52:54 +00:00
|
|
|
let scrollNumberStyle: CSSProperties = { ...mergedStyle, ...(props.numberStyle as object) };
|
2021-06-07 09:35:03 +00:00
|
|
|
if (color && !isPresetColor(color)) {
|
|
|
|
scrollNumberStyle = scrollNumberStyle || {};
|
|
|
|
scrollNumberStyle.background = color;
|
|
|
|
}
|
2018-03-09 09:50:33 +00:00
|
|
|
|
2021-06-07 09:35:03 +00:00
|
|
|
return (
|
|
|
|
<span {...attrs} class={badgeClassName}>
|
|
|
|
{children}
|
|
|
|
<Transition {...transitionProps}>
|
|
|
|
<ScrollNumber
|
|
|
|
v-show={visible}
|
|
|
|
prefixCls={props.scrollNumberPrefixCls}
|
|
|
|
show={visible}
|
|
|
|
class={scrollNumberCls.value}
|
|
|
|
count={displayCount.value}
|
|
|
|
title={titleNode}
|
|
|
|
style={scrollNumberStyle}
|
|
|
|
key="scrollNumber"
|
|
|
|
>
|
|
|
|
{displayNode}
|
|
|
|
</ScrollNumber>
|
|
|
|
</Transition>
|
|
|
|
{statusTextNode}
|
|
|
|
</span>
|
|
|
|
);
|
|
|
|
};
|
2017-11-16 10:29:02 +00:00
|
|
|
},
|
2020-10-13 07:02:35 +00:00
|
|
|
});
|