feat: tabbar support mouse wheel vertical (#5129)

* feat: tabbar support mouse wheel

* docs: add tabbar wheelable tips

* chore: resolve vitest test
dev-v5
Netfan 2024-12-13 14:45:06 +08:00 committed by GitHub
parent be208fe915
commit 7fbf7b189a
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
12 changed files with 54 additions and 5 deletions

View File

@ -83,6 +83,7 @@ exports[`defaultPreferences immutability test > should not modify the config obj
"showMaximize": true, "showMaximize": true,
"showMore": true, "showMore": true,
"styleType": "chrome", "styleType": "chrome",
"wheelable": true,
}, },
"theme": { "theme": {
"builtinType": "default", "builtinType": "default",

View File

@ -83,6 +83,7 @@ const defaultPreferences: Preferences = {
showMaximize: true, showMaximize: true,
showMore: true, showMore: true,
styleType: 'chrome', styleType: 'chrome',
wheelable: true,
}, },
theme: { theme: {
builtinType: 'default', builtinType: 'default',

View File

@ -173,6 +173,8 @@ interface TabbarPreferences {
showMore: boolean; showMore: boolean;
/** 标签页风格 */ /** 标签页风格 */
styleType: TabsStyleType; styleType: TabsStyleType;
/** 是否开启鼠标滚轮响应 */
wheelable: boolean;
} }
interface ThemePreferences { interface ThemePreferences {

View File

@ -19,6 +19,7 @@ const props = withDefaults(defineProps<Props>(), {
contentClass: 'vben-tabs-content', contentClass: 'vben-tabs-content',
draggable: true, draggable: true,
styleType: 'chrome', styleType: 'chrome',
wheelable: true,
}); });
const emit = defineEmits<TabsEmits>(); const emit = defineEmits<TabsEmits>();
@ -27,6 +28,7 @@ const forward = useForwardPropsEmits(props, emit);
const { const {
handleScrollAt, handleScrollAt,
handleWheel,
scrollbarRef, scrollbarRef,
scrollDirection, scrollDirection,
scrollIsAtLeft, scrollIsAtLeft,
@ -34,6 +36,14 @@ const {
showScrollButton, showScrollButton,
} = useTabsViewScroll(props); } = useTabsViewScroll(props);
function onWheel(e: WheelEvent) {
if (props.wheelable) {
handleWheel(e);
e.stopPropagation();
e.preventDefault();
}
}
useTabsDrag(props, emit); useTabsDrag(props, emit);
</script> </script>
@ -69,6 +79,7 @@ useTabsDrag(props, emit);
shadow-left shadow-left
shadow-right shadow-right
@scroll-at="handleScrollAt" @scroll-at="handleScrollAt"
@wheel="onWheel"
> >
<TabsChrome <TabsChrome
v-if="styleType === 'chrome'" v-if="styleType === 'chrome'"

View File

@ -33,7 +33,6 @@ export interface TabsProps {
* tabs-chrome * tabs-chrome
*/ */
maxWidth?: number; maxWidth?: number;
/** /**
* @zh_CN tab * @zh_CN tab
* tabs-chrome * tabs-chrome
@ -44,15 +43,20 @@ export interface TabsProps {
* @zh_CN * @zh_CN
*/ */
showIcon?: boolean; showIcon?: boolean;
/** /**
* @zh_CN * @zh_CN
*/ */
styleType?: TabsStyleType; styleType?: TabsStyleType;
/** /**
* @zh_CN * @zh_CN
*/ */
tabs?: TabDefinition[]; tabs?: TabDefinition[];
/**
* @zh_CN
*/
wheelable?: boolean;
} }
export interface TabConfig extends TabDefinition { export interface TabConfig extends TabDefinition {

View File

@ -142,6 +142,13 @@ export function useTabsViewScroll(props: TabsProps) {
scrollIsAtRight.value = right; scrollIsAtRight.value = right;
}, 100); }, 100);
function handleWheel({ deltaY }: WheelEvent) {
scrollViewportEl.value?.scrollBy({
behavior: 'smooth',
left: deltaY * 3,
});
}
watch( watch(
() => props.active, () => props.active,
async () => { async () => {
@ -184,6 +191,7 @@ export function useTabsViewScroll(props: TabsProps) {
return { return {
handleScrollAt, handleScrollAt,
handleWheel,
initScrollbar, initScrollbar,
scrollbarRef, scrollbarRef,
scrollDirection, scrollDirection,

View File

@ -55,6 +55,7 @@ if (!preferences.tabbar.persist) {
:show-icon="showIcon" :show-icon="showIcon"
:style-type="preferences.tabbar.styleType" :style-type="preferences.tabbar.styleType"
:tabs="currentTabs" :tabs="currentTabs"
:wheelable="preferences.tabbar.wheelable"
@close="handleClose" @close="handleClose"
@sort-tabs="tabbarStore.sortTabs" @sort-tabs="tabbarStore.sortTabs"
@unpin="unpinTab" @unpin="unpinTab"

View File

@ -18,6 +18,7 @@ const tabbarEnable = defineModel<boolean>('tabbarEnable');
const tabbarShowIcon = defineModel<boolean>('tabbarShowIcon'); const tabbarShowIcon = defineModel<boolean>('tabbarShowIcon');
const tabbarPersist = defineModel<boolean>('tabbarPersist'); const tabbarPersist = defineModel<boolean>('tabbarPersist');
const tabbarDraggable = defineModel<boolean>('tabbarDraggable'); const tabbarDraggable = defineModel<boolean>('tabbarDraggable');
const tabbarWheelable = defineModel<boolean>('tabbarWheelable');
const tabbarStyleType = defineModel<string>('tabbarStyleType'); const tabbarStyleType = defineModel<string>('tabbarStyleType');
const tabbarShowMore = defineModel<boolean>('tabbarShowMore'); const tabbarShowMore = defineModel<boolean>('tabbarShowMore');
const tabbarShowMaximize = defineModel<boolean>('tabbarShowMaximize'); const tabbarShowMaximize = defineModel<boolean>('tabbarShowMaximize');
@ -53,6 +54,13 @@ const styleItems = computed((): SelectOption[] => [
<SwitchItem v-model="tabbarDraggable" :disabled="!tabbarEnable"> <SwitchItem v-model="tabbarDraggable" :disabled="!tabbarEnable">
{{ $t('preferences.tabbar.draggable') }} {{ $t('preferences.tabbar.draggable') }}
</SwitchItem> </SwitchItem>
<SwitchItem
v-model="tabbarWheelable"
:disabled="!tabbarEnable"
:tip="$t('preferences.tabbar.wheelableTip')"
>
{{ $t('preferences.tabbar.wheelable') }}
</SwitchItem>
<SwitchItem v-model="tabbarShowIcon" :disabled="!tabbarEnable"> <SwitchItem v-model="tabbarShowIcon" :disabled="!tabbarEnable">
{{ $t('preferences.tabbar.icon') }} {{ $t('preferences.tabbar.icon') }}
</SwitchItem> </SwitchItem>

View File

@ -8,8 +8,9 @@ defineOptions({
name: 'PreferenceSwitchItem', name: 'PreferenceSwitchItem',
}); });
withDefaults(defineProps<{ disabled?: boolean }>(), { withDefaults(defineProps<{ disabled?: boolean; tip?: string }>(), {
disabled: false, disabled: false,
tip: '',
}); });
const checked = defineModel<boolean>(); const checked = defineModel<boolean>();
@ -32,11 +33,17 @@ function handleClick() {
<span class="flex items-center text-sm"> <span class="flex items-center text-sm">
<slot></slot> <slot></slot>
<VbenTooltip v-if="slots.tip" side="bottom"> <VbenTooltip v-if="slots.tip || tip" side="bottom">
<template #trigger> <template #trigger>
<CircleHelp class="ml-1 size-3 cursor-help" /> <CircleHelp class="ml-1 size-3 cursor-help" />
</template> </template>
<slot name="tip"></slot> <slot name="tip">
<template v-if="tip">
<p v-for="(line, index) in tip.split('\n')" :key="index">
{{ line }}
</p>
</template>
</slot>
</VbenTooltip> </VbenTooltip>
</span> </span>
<span v-if="$slots.shortcut" class="ml-auto mr-2 text-xs opacity-60"> <span v-if="$slots.shortcut" class="ml-auto mr-2 text-xs opacity-60">

View File

@ -105,6 +105,7 @@ const tabbarShowMore = defineModel<boolean>('tabbarShowMore');
const tabbarShowMaximize = defineModel<boolean>('tabbarShowMaximize'); const tabbarShowMaximize = defineModel<boolean>('tabbarShowMaximize');
const tabbarPersist = defineModel<boolean>('tabbarPersist'); const tabbarPersist = defineModel<boolean>('tabbarPersist');
const tabbarDraggable = defineModel<boolean>('tabbarDraggable'); const tabbarDraggable = defineModel<boolean>('tabbarDraggable');
const tabbarWheelable = defineModel<boolean>('tabbarWheelable');
const tabbarStyleType = defineModel<string>('tabbarStyleType'); const tabbarStyleType = defineModel<string>('tabbarStyleType');
const navigationStyleType = defineModel<NavigationStyleType>( const navigationStyleType = defineModel<NavigationStyleType>(
@ -345,6 +346,7 @@ async function handleReset() {
v-model:tabbar-show-maximize="tabbarShowMaximize" v-model:tabbar-show-maximize="tabbarShowMaximize"
v-model:tabbar-show-more="tabbarShowMore" v-model:tabbar-show-more="tabbarShowMore"
v-model:tabbar-style-type="tabbarStyleType" v-model:tabbar-style-type="tabbarStyleType"
v-model:tabbar-wheelable="tabbarWheelable"
/> />
</Block> </Block>
<Block :title="$t('preferences.widget.title')"> <Block :title="$t('preferences.widget.title')">

View File

@ -55,6 +55,8 @@
"showMaximize": "Show Maximize Button", "showMaximize": "Show Maximize Button",
"persist": "Persist Tabs", "persist": "Persist Tabs",
"draggable": "Enable Draggable Sort", "draggable": "Enable Draggable Sort",
"wheelable": "Support Mouse Wheel",
"wheelableTip": "When enabled, the Tabbar area responds to vertical scrolling events of the scroll wheel.",
"styleType": { "styleType": {
"title": "Tabs Style", "title": "Tabs Style",
"chrome": "Chrome", "chrome": "Chrome",

View File

@ -55,6 +55,8 @@
"showMaximize": "显示最大化按钮", "showMaximize": "显示最大化按钮",
"persist": "持久化标签页", "persist": "持久化标签页",
"draggable": "启动拖拽排序", "draggable": "启动拖拽排序",
"wheelable": "启用纵向滚轮响应",
"wheelableTip": "开启后,标签栏区域可以响应滚轮的纵向滚动事件。\n关闭时只能响应系统的横向滚动事件需要按下Shift再滚动滚轮",
"styleType": { "styleType": {
"title": "标签页风格", "title": "标签页风格",
"chrome": "谷歌", "chrome": "谷歌",