vant/src/tabbar-item/index.js
2020-09-26 11:13:49 +08:00

88 lines
2.2 KiB
JavaScript

import { computed, getCurrentInstance } from 'vue';
import { TABBAR_KEY } from '../tabbar';
// Utils
import { createNamespace, isObject, isDef } from '../utils';
// Composition
import { useParent } from '../composition/use-relation';
import { routeProps, useRoute } from '../composition/use-route';
// Components
import Icon from '../icon';
import Badge from '../badge';
const [createComponent, bem] = createNamespace('tabbar-item');
export default createComponent({
props: {
...routeProps,
dot: Boolean,
icon: String,
name: [Number, String],
badge: [Number, String],
iconPrefix: String,
},
emits: ['click'],
setup(props, { emit, slots }) {
const route = useRoute();
const vm = getCurrentInstance().proxy;
const { parent, index } = useParent(TABBAR_KEY);
const active = computed(() => {
const { $route } = vm;
const { route, modelValue } = parent.props;
if (route && $route) {
const { to } = props;
const config = isObject(to) ? to : { path: to };
const pathMatched = config.path === $route.path;
const nameMatched = isDef(config.name) && config.name === $route.name;
return pathMatched || nameMatched;
}
return (props.name || index.value) === modelValue;
});
const onClick = (event) => {
parent.setActive(props.name || index.value);
emit('click', event);
route();
};
const renderIcon = () => {
if (slots.icon) {
return slots.icon({ active: active.value });
}
if (props.icon) {
return <Icon name={props.icon} classPrefix={props.iconPrefix} />;
}
};
return () => {
const { dot, badge } = props;
const { activeColor, inactiveColor } = parent.props;
const color = active.value ? activeColor : inactiveColor;
return (
<div
class={bem({ active: active.value })}
style={{ color }}
onClick={onClick}
>
<div class={bem('icon')}>
{renderIcon()}
<Badge dot={dot} badge={badge} />
</div>
<div class={bem('text')}>
{slots.default?.({ active: active.value })}
</div>
</div>
);
};
},
});