vant/src/cell/Cell.tsx
neverland bd609e1df0
perf: add truthProp util (#8522)
* perf: add TruthyProp util

* chore: rename

* chore: upd
2021-04-14 10:26:21 +08:00

151 lines
3.4 KiB
TypeScript

import { PropType, CSSProperties, defineComponent } from 'vue';
// Utils
import {
createNamespace,
extend,
isDef,
truthProp,
unknownProp,
} from '../utils';
// Composables
import { useRoute, routeProps } from '../composables/use-route';
// Components
import { Icon } from '../icon';
const [name, bem] = createNamespace('cell');
export type CellArrowDirection = 'up' | 'down' | 'left' | 'right';
export const cellProps = {
icon: String,
size: String as PropType<'large'>,
title: [Number, String],
value: [Number, String],
label: [Number, String],
center: Boolean,
isLink: Boolean,
border: truthProp,
required: Boolean,
iconPrefix: String,
valueClass: unknownProp,
labelClass: unknownProp,
titleClass: unknownProp,
titleStyle: (null as unknown) as PropType<string | CSSProperties>,
arrowDirection: String as PropType<CellArrowDirection>,
clickable: {
type: Boolean as PropType<boolean | null>,
default: null,
},
};
export default defineComponent({
name,
props: extend({}, cellProps, routeProps),
setup(props, { slots }) {
const route = useRoute();
const renderLabel = () => {
const showLabel = slots.label || isDef(props.label);
if (showLabel) {
return (
<div class={[bem('label'), props.labelClass]}>
{slots.label ? slots.label() : props.label}
</div>
);
}
};
const renderTitle = () => {
if (slots.title || isDef(props.title)) {
return (
<div
class={[bem('title'), props.titleClass]}
style={props.titleStyle}
>
{slots.title ? slots.title() : <span>{props.title}</span>}
{renderLabel()}
</div>
);
}
};
const renderValue = () => {
const hasValue = slots.default || isDef(props.value);
if (hasValue) {
const hasTitle = slots.title || isDef(props.title);
return (
<div class={[bem('value', { alone: !hasTitle }), props.valueClass]}>
{slots.default ? slots.default() : <span>{props.value}</span>}
</div>
);
}
};
const renderLeftIcon = () => {
if (slots.icon) {
return slots.icon();
}
if (props.icon) {
return (
<Icon
name={props.icon}
class={bem('left-icon')}
classPrefix={props.iconPrefix}
/>
);
}
};
const renderRightIcon = () => {
if (slots['right-icon']) {
return slots['right-icon']();
}
if (props.isLink) {
const name = props.arrowDirection
? `arrow-${props.arrowDirection}`
: 'arrow';
return <Icon name={name} class={bem('right-icon')} />;
}
};
return () => {
const { size, center, border, isLink, required } = props;
const clickable = props.clickable ?? isLink;
const classes: Record<string, boolean | undefined> = {
center,
required,
clickable,
borderless: !border,
};
if (size) {
classes[size] = !!size;
}
return (
<div
class={bem(classes)}
role={clickable ? 'button' : undefined}
tabindex={clickable ? 0 : undefined}
onClick={route}
>
{renderLeftIcon()}
{renderTitle()}
{renderValue()}
{renderRightIcon()}
{slots.extra?.()}
</div>
);
};
},
});