mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-08-12 07:29:46 +08:00
97 lines
2.5 KiB
TypeScript
97 lines
2.5 KiB
TypeScript
import { computed, getCurrentInstance, defineComponent } from 'vue';
|
|
|
|
// Utils
|
|
import { createNamespace, extend, isObject } from '../utils';
|
|
import { TABBAR_KEY, TabbarProvide } from '../tabbar/Tabbar';
|
|
|
|
// Composables
|
|
import { useParent } from '@vant/use';
|
|
import { routeProps, useRoute } from '../composables/use-route';
|
|
|
|
// Components
|
|
import { Icon } from '../icon';
|
|
import { Badge } from '../badge';
|
|
|
|
const [name, bem] = createNamespace('tabbar-item');
|
|
|
|
export default defineComponent({
|
|
name,
|
|
|
|
props: extend({}, 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<TabbarProvide>(TABBAR_KEY);
|
|
|
|
if (!parent) {
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
console.error(
|
|
'[Vant] <TabbarItem> must be a child component of <Tabbar>.'
|
|
);
|
|
}
|
|
return;
|
|
}
|
|
|
|
const active = computed(() => {
|
|
const { route, modelValue } = parent.props;
|
|
|
|
if (route && '$route' in vm) {
|
|
const { $route } = vm;
|
|
const { to } = props;
|
|
const config = isObject(to) ? to : { path: to };
|
|
const pathMatched = 'path' in config && config.path === $route.path;
|
|
const nameMatched = 'name' in config && config.name === $route.name;
|
|
|
|
return pathMatched || nameMatched;
|
|
}
|
|
|
|
return (props.name || index.value) === modelValue;
|
|
});
|
|
|
|
const onClick = (event: MouseEvent) => {
|
|
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}
|
|
>
|
|
<Badge dot={dot} content={badge} class={bem('icon')}>
|
|
{renderIcon()}
|
|
</Badge>
|
|
<div class={bem('text')}>
|
|
{slots.default?.({ active: active.value })}
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
},
|
|
});
|