Merge branch 'main' of https://github.com/vbenjs/vue-vben-admin into dev
This commit is contained in:
commit
5e82866370
@ -4,7 +4,7 @@
|
|||||||
|
|
||||||
## 新增组件库应用
|
## 新增组件库应用
|
||||||
|
|
||||||
如果你想用其他别的组件库,你只需要按一下步骤进行操作:
|
如果你想用其他别的组件库,你只需要按以下步骤进行操作:
|
||||||
|
|
||||||
1. 在`apps`内创建一个新的文件夹,例如`apps/web-xxx`。
|
1. 在`apps`内创建一个新的文件夹,例如`apps/web-xxx`。
|
||||||
2. 更改`apps/web-xxx/package.json`的`name`字段为`web-xxx`。
|
2. 更改`apps/web-xxx/package.json`的`name`字段为`web-xxx`。
|
||||||
|
@ -4,7 +4,6 @@ import { interopDefault } from '../util';
|
|||||||
|
|
||||||
export async function vue(): Promise<Linter.Config[]> {
|
export async function vue(): Promise<Linter.Config[]> {
|
||||||
const [pluginVue, parserVue, parserTs] = await Promise.all([
|
const [pluginVue, parserVue, parserTs] = await Promise.all([
|
||||||
// @ts-expect-error missing types
|
|
||||||
interopDefault(import('eslint-plugin-vue')),
|
interopDefault(import('eslint-plugin-vue')),
|
||||||
interopDefault(import('vue-eslint-parser')),
|
interopDefault(import('vue-eslint-parser')),
|
||||||
// @ts-expect-error missing types
|
// @ts-expect-error missing types
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import type { UserConfig } from 'vite';
|
import type { CSSOptions, UserConfig } from 'vite';
|
||||||
|
|
||||||
import type { DefineApplicationOptions } from '../typing';
|
import type { DefineApplicationOptions } from '../typing';
|
||||||
|
|
||||||
@ -100,7 +100,7 @@ function defineApplicationConfig(userConfigPromise?: DefineApplicationOptions) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function createCssOptions(injectGlobalScss = true) {
|
function createCssOptions(injectGlobalScss = true): CSSOptions {
|
||||||
const root = findMonorepoRoot();
|
const root = findMonorepoRoot();
|
||||||
return {
|
return {
|
||||||
preprocessorOptions: injectGlobalScss
|
preprocessorOptions: injectGlobalScss
|
||||||
|
@ -28,6 +28,7 @@ export {
|
|||||||
Fullscreen,
|
Fullscreen,
|
||||||
Github,
|
Github,
|
||||||
Grip,
|
Grip,
|
||||||
|
GripVertical,
|
||||||
Info,
|
Info,
|
||||||
InspectionPanel,
|
InspectionPanel,
|
||||||
Languages,
|
Languages,
|
||||||
|
@ -166,7 +166,7 @@ const headerStyle = computed((): CSSProperties => {
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
...(isSidebarMixed ? { display: 'flex', justifyContent: 'center' } : {}),
|
...(isSidebarMixed ? { display: 'flex', justifyContent: 'center' } : {}),
|
||||||
height: `${headerHeight}px`,
|
height: `${headerHeight - 1}px`,
|
||||||
...contentWidthStyle.value,
|
...contentWidthStyle.value,
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
@ -16,6 +16,7 @@ export * from './pagination';
|
|||||||
export * from './pin-input';
|
export * from './pin-input';
|
||||||
export * from './popover';
|
export * from './popover';
|
||||||
export * from './radio-group';
|
export * from './radio-group';
|
||||||
|
export * from './resizable';
|
||||||
export * from './scroll-area';
|
export * from './scroll-area';
|
||||||
export * from './select';
|
export * from './select';
|
||||||
export * from './separator';
|
export * from './separator';
|
||||||
|
@ -0,0 +1,48 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
import { computed, type HTMLAttributes } from 'vue';
|
||||||
|
|
||||||
|
import { GripVertical } from '@vben-core/icons';
|
||||||
|
import { cn } from '@vben-core/shared/utils';
|
||||||
|
|
||||||
|
import {
|
||||||
|
SplitterResizeHandle,
|
||||||
|
type SplitterResizeHandleEmits,
|
||||||
|
type SplitterResizeHandleProps,
|
||||||
|
useForwardPropsEmits,
|
||||||
|
} from 'radix-vue';
|
||||||
|
|
||||||
|
const props = defineProps<
|
||||||
|
{
|
||||||
|
class?: HTMLAttributes['class'];
|
||||||
|
withHandle?: boolean;
|
||||||
|
} & SplitterResizeHandleProps
|
||||||
|
>();
|
||||||
|
const emits = defineEmits<SplitterResizeHandleEmits>();
|
||||||
|
|
||||||
|
const delegatedProps = computed(() => {
|
||||||
|
const { class: _, ...delegated } = props;
|
||||||
|
return delegated;
|
||||||
|
});
|
||||||
|
|
||||||
|
const forwarded = useForwardPropsEmits(delegatedProps, emits);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<SplitterResizeHandle
|
||||||
|
v-bind="forwarded"
|
||||||
|
:class="
|
||||||
|
cn(
|
||||||
|
'bg-border focus-visible:ring-ring relative flex w-px items-center justify-center after:absolute after:inset-y-0 after:left-1/2 after:w-1 after:-translate-x-1/2 focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-offset-1 [&[data-orientation=vertical]>div]:rotate-90 [&[data-orientation=vertical]]:h-px [&[data-orientation=vertical]]:w-full [&[data-orientation=vertical]]:after:left-0 [&[data-orientation=vertical]]:after:h-1 [&[data-orientation=vertical]]:after:w-full [&[data-orientation=vertical]]:after:-translate-y-1/2 [&[data-orientation=vertical]]:after:translate-x-0',
|
||||||
|
props.class,
|
||||||
|
)
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<template v-if="props.withHandle">
|
||||||
|
<div
|
||||||
|
class="bg-border z-10 flex h-4 w-3 items-center justify-center rounded-sm border"
|
||||||
|
>
|
||||||
|
<GripVertical class="h-2.5 w-2.5" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</SplitterResizeHandle>
|
||||||
|
</template>
|
@ -0,0 +1,38 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
import { computed, type HTMLAttributes } from 'vue';
|
||||||
|
|
||||||
|
import { cn } from '@vben-core/shared/utils';
|
||||||
|
|
||||||
|
import {
|
||||||
|
SplitterGroup,
|
||||||
|
type SplitterGroupEmits,
|
||||||
|
type SplitterGroupProps,
|
||||||
|
useForwardPropsEmits,
|
||||||
|
} from 'radix-vue';
|
||||||
|
|
||||||
|
const props = defineProps<
|
||||||
|
{ class?: HTMLAttributes['class'] } & SplitterGroupProps
|
||||||
|
>();
|
||||||
|
const emits = defineEmits<SplitterGroupEmits>();
|
||||||
|
|
||||||
|
const delegatedProps = computed(() => {
|
||||||
|
const { class: _, ...delegated } = props;
|
||||||
|
return delegated;
|
||||||
|
});
|
||||||
|
|
||||||
|
const forwarded = useForwardPropsEmits(delegatedProps, emits);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<SplitterGroup
|
||||||
|
v-bind="forwarded"
|
||||||
|
:class="
|
||||||
|
cn(
|
||||||
|
'flex h-full w-full data-[panel-group-direction=vertical]:flex-col',
|
||||||
|
props.class,
|
||||||
|
)
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<slot></slot>
|
||||||
|
</SplitterGroup>
|
||||||
|
</template>
|
@ -0,0 +1,3 @@
|
|||||||
|
export { default as ResizableHandle } from './ResizableHandle.vue';
|
||||||
|
export { default as ResizablePanelGroup } from './ResizablePanelGroup.vue';
|
||||||
|
export { SplitterPanel as ResizablePanel } from 'radix-vue';
|
107
packages/effects/common-ui/src/components/col-page/col-page.vue
Normal file
107
packages/effects/common-ui/src/components/col-page/col-page.vue
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
<script lang="ts" setup>
|
||||||
|
import type { ColPageProps } from './types';
|
||||||
|
|
||||||
|
import { computed, ref, useSlots } from 'vue';
|
||||||
|
|
||||||
|
import {
|
||||||
|
ResizableHandle,
|
||||||
|
ResizablePanel,
|
||||||
|
ResizablePanelGroup,
|
||||||
|
} from '@vben-core/shadcn-ui';
|
||||||
|
|
||||||
|
import Page from '../page/page.vue';
|
||||||
|
|
||||||
|
defineOptions({
|
||||||
|
name: 'ColPage',
|
||||||
|
inheritAttrs: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
const props = withDefaults(defineProps<ColPageProps>(), {
|
||||||
|
leftWidth: 30,
|
||||||
|
rightWidth: 70,
|
||||||
|
resizable: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
const delegatedProps = computed(() => {
|
||||||
|
const { leftWidth: _, ...delegated } = props;
|
||||||
|
return delegated;
|
||||||
|
});
|
||||||
|
|
||||||
|
const slots = useSlots();
|
||||||
|
|
||||||
|
const delegatedSlots = computed(() => {
|
||||||
|
const resultSlots: string[] = [];
|
||||||
|
|
||||||
|
for (const key of Object.keys(slots)) {
|
||||||
|
if (!['default', 'left'].includes(key)) {
|
||||||
|
resultSlots.push(key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return resultSlots;
|
||||||
|
});
|
||||||
|
|
||||||
|
const leftPanelRef = ref<InstanceType<typeof ResizablePanel>>();
|
||||||
|
|
||||||
|
function expandLeft() {
|
||||||
|
leftPanelRef.value?.expand();
|
||||||
|
}
|
||||||
|
|
||||||
|
function collapseLeft() {
|
||||||
|
leftPanelRef.value?.collapse();
|
||||||
|
}
|
||||||
|
|
||||||
|
defineExpose({
|
||||||
|
expandLeft,
|
||||||
|
collapseLeft,
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
<template>
|
||||||
|
<Page v-bind="delegatedProps">
|
||||||
|
<!-- 继承默认的slot -->
|
||||||
|
<template
|
||||||
|
v-for="slotName in delegatedSlots"
|
||||||
|
:key="slotName"
|
||||||
|
#[slotName]="slotProps"
|
||||||
|
>
|
||||||
|
<slot :name="slotName" v-bind="slotProps"></slot>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<ResizablePanelGroup class="w-full" direction="horizontal">
|
||||||
|
<ResizablePanel
|
||||||
|
ref="leftPanelRef"
|
||||||
|
:collapsed-size="leftCollapsedWidth"
|
||||||
|
:collapsible="leftCollapsible"
|
||||||
|
:default-size="leftWidth"
|
||||||
|
:max-size="leftMaxWidth"
|
||||||
|
:min-size="leftMinWidth"
|
||||||
|
>
|
||||||
|
<template #default="slotProps">
|
||||||
|
<slot
|
||||||
|
name="left"
|
||||||
|
v-bind="{
|
||||||
|
...slotProps,
|
||||||
|
expand: expandLeft,
|
||||||
|
collapse: collapseLeft,
|
||||||
|
}"
|
||||||
|
></slot>
|
||||||
|
</template>
|
||||||
|
</ResizablePanel>
|
||||||
|
<ResizableHandle
|
||||||
|
v-if="resizable"
|
||||||
|
:style="{ backgroundColor: splitLine ? undefined : 'transparent' }"
|
||||||
|
:with-handle="splitHandle"
|
||||||
|
/>
|
||||||
|
<ResizablePanel
|
||||||
|
:collapsed-size="rightCollapsedWidth"
|
||||||
|
:collapsible="rightCollapsible"
|
||||||
|
:default-size="rightWidth"
|
||||||
|
:max-size="rightMaxWidth"
|
||||||
|
:min-size="rightMinWidth"
|
||||||
|
>
|
||||||
|
<template #default>
|
||||||
|
<slot></slot>
|
||||||
|
</template>
|
||||||
|
</ResizablePanel>
|
||||||
|
</ResizablePanelGroup>
|
||||||
|
</Page>
|
||||||
|
</template>
|
@ -0,0 +1,2 @@
|
|||||||
|
export { default as ColPage } from './col-page.vue';
|
||||||
|
export * from './types';
|
26
packages/effects/common-ui/src/components/col-page/types.ts
Normal file
26
packages/effects/common-ui/src/components/col-page/types.ts
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import type { PageProps } from '../page/types';
|
||||||
|
|
||||||
|
export interface ColPageProps extends PageProps {
|
||||||
|
/**
|
||||||
|
* 左侧宽度
|
||||||
|
* @default 30
|
||||||
|
*/
|
||||||
|
leftWidth?: number;
|
||||||
|
leftMinWidth?: number;
|
||||||
|
leftMaxWidth?: number;
|
||||||
|
leftCollapsedWidth?: number;
|
||||||
|
leftCollapsible?: boolean;
|
||||||
|
/**
|
||||||
|
* 右侧宽度
|
||||||
|
* @default 70
|
||||||
|
*/
|
||||||
|
rightWidth?: number;
|
||||||
|
rightMinWidth?: number;
|
||||||
|
rightCollapsedWidth?: number;
|
||||||
|
rightMaxWidth?: number;
|
||||||
|
rightCollapsible?: boolean;
|
||||||
|
|
||||||
|
resizable?: boolean;
|
||||||
|
splitLine?: boolean;
|
||||||
|
splitHandle?: boolean;
|
||||||
|
}
|
@ -1,6 +1,7 @@
|
|||||||
export * from './api-component';
|
export * from './api-component';
|
||||||
export * from './captcha';
|
export * from './captcha';
|
||||||
export * from './code-mirror';
|
export * from './code-mirror';
|
||||||
|
export * from './col-page';
|
||||||
export * from './ellipsis-text';
|
export * from './ellipsis-text';
|
||||||
export * from './icon-picker';
|
export * from './icon-picker';
|
||||||
export * from './json-preview';
|
export * from './json-preview';
|
||||||
|
@ -1 +1,2 @@
|
|||||||
export { default as Page } from './page.vue';
|
export { default as Page } from './page.vue';
|
||||||
|
export * from './types';
|
||||||
|
@ -1,4 +1,6 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
|
import type { PageProps } from './types';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
computed,
|
computed,
|
||||||
nextTick,
|
nextTick,
|
||||||
@ -11,23 +13,11 @@ import {
|
|||||||
import { CSS_VARIABLE_LAYOUT_CONTENT_HEIGHT } from '@vben-core/shared/constants';
|
import { CSS_VARIABLE_LAYOUT_CONTENT_HEIGHT } from '@vben-core/shared/constants';
|
||||||
import { cn } from '@vben-core/shared/utils';
|
import { cn } from '@vben-core/shared/utils';
|
||||||
|
|
||||||
interface Props {
|
|
||||||
title?: string;
|
|
||||||
description?: string;
|
|
||||||
contentClass?: string;
|
|
||||||
/**
|
|
||||||
* 根据content可见高度自适应
|
|
||||||
*/
|
|
||||||
autoContentHeight?: boolean;
|
|
||||||
headerClass?: string;
|
|
||||||
footerClass?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
defineOptions({
|
defineOptions({
|
||||||
name: 'Page',
|
name: 'Page',
|
||||||
});
|
});
|
||||||
|
|
||||||
const { autoContentHeight = false } = defineProps<Props>();
|
const { autoContentHeight = false } = defineProps<PageProps>();
|
||||||
|
|
||||||
const headerHeight = ref(0);
|
const headerHeight = ref(0);
|
||||||
const footerHeight = ref(0);
|
const footerHeight = ref(0);
|
||||||
|
11
packages/effects/common-ui/src/components/page/types.ts
Normal file
11
packages/effects/common-ui/src/components/page/types.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
export interface PageProps {
|
||||||
|
title?: string;
|
||||||
|
description?: string;
|
||||||
|
contentClass?: string;
|
||||||
|
/**
|
||||||
|
* 根据content可见高度自适应
|
||||||
|
*/
|
||||||
|
autoContentHeight?: boolean;
|
||||||
|
headerClass?: string;
|
||||||
|
footerClass?: string;
|
||||||
|
}
|
@ -38,7 +38,7 @@ defineEmits(['click']);
|
|||||||
'border-b-0': index < 3,
|
'border-b-0': index < 3,
|
||||||
'pb-4': index > 2,
|
'pb-4': index > 2,
|
||||||
}"
|
}"
|
||||||
class="border-border group w-full cursor-pointer border-b border-r border-t p-4 transition-all hover:shadow-xl md:w-1/2 lg:w-1/3"
|
class="border-border group w-full cursor-pointer border-r border-t p-4 transition-all hover:shadow-xl md:w-1/2 lg:w-1/3"
|
||||||
>
|
>
|
||||||
<div class="flex items-center">
|
<div class="flex items-center">
|
||||||
<VbenIcon
|
<VbenIcon
|
||||||
|
@ -38,7 +38,7 @@ defineEmits(['click']);
|
|||||||
'pb-4': index > 2,
|
'pb-4': index > 2,
|
||||||
'border-b-0': index < 3,
|
'border-b-0': index < 3,
|
||||||
}"
|
}"
|
||||||
class="flex-col-center border-border group w-1/3 cursor-pointer border-b border-r border-t py-8 hover:shadow-xl"
|
class="flex-col-center border-border group w-1/3 cursor-pointer border-r border-t py-8 hover:shadow-xl"
|
||||||
@click="$emit('click', item)"
|
@click="$emit('click', item)"
|
||||||
>
|
>
|
||||||
<VbenIcon
|
<VbenIcon
|
||||||
|
@ -6,6 +6,7 @@ import type {
|
|||||||
VxeGridListeners,
|
VxeGridListeners,
|
||||||
VxeGridPropTypes,
|
VxeGridPropTypes,
|
||||||
VxeGridProps as VxeTableGridProps,
|
VxeGridProps as VxeTableGridProps,
|
||||||
|
VxeToolbarPropTypes,
|
||||||
} from 'vxe-table';
|
} from 'vxe-table';
|
||||||
|
|
||||||
import type { ExtendedVxeGridApi, VxeGridProps } from './types';
|
import type { ExtendedVxeGridApi, VxeGridProps } from './types';
|
||||||
@ -109,28 +110,28 @@ const showToolbar = computed(() => {
|
|||||||
const toolbarOptions = computed(() => {
|
const toolbarOptions = computed(() => {
|
||||||
const slotActions = slots[TOOLBAR_ACTIONS]?.();
|
const slotActions = slots[TOOLBAR_ACTIONS]?.();
|
||||||
const slotTools = slots[TOOLBAR_TOOLS]?.();
|
const slotTools = slots[TOOLBAR_TOOLS]?.();
|
||||||
|
const searchBtn: VxeToolbarPropTypes.ToolConfig = {
|
||||||
const toolbarConfig: VxeGridPropTypes.ToolbarConfig = {
|
|
||||||
tools:
|
|
||||||
gridOptions.value?.toolbarConfig?.search && !!formOptions.value
|
|
||||||
? [
|
|
||||||
{
|
|
||||||
code: 'search',
|
code: 'search',
|
||||||
icon: 'vxe-icon--search',
|
icon: 'vxe-icon--search',
|
||||||
circle: true,
|
circle: true,
|
||||||
status: showSearchForm.value ? 'primary' : undefined,
|
status: showSearchForm.value ? 'primary' : undefined,
|
||||||
title: $t('common.search'),
|
title: $t('common.search'),
|
||||||
},
|
|
||||||
]
|
|
||||||
: [],
|
|
||||||
};
|
};
|
||||||
|
// 将搜索按钮合并到用户配置的toolbarConfig.tools中
|
||||||
|
const toolbarConfig: VxeGridPropTypes.ToolbarConfig = {
|
||||||
|
tools: (gridOptions.value?.toolbarConfig?.tools ??
|
||||||
|
[]) as VxeToolbarPropTypes.ToolConfig[],
|
||||||
|
};
|
||||||
|
if (gridOptions.value?.toolbarConfig?.search && !!formOptions.value) {
|
||||||
|
toolbarConfig.tools = Array.isArray(toolbarConfig.tools)
|
||||||
|
? [...toolbarConfig.tools, searchBtn]
|
||||||
|
: [searchBtn];
|
||||||
|
}
|
||||||
|
|
||||||
if (!showToolbar.value) {
|
if (!showToolbar.value) {
|
||||||
return { toolbarConfig };
|
return { toolbarConfig };
|
||||||
}
|
}
|
||||||
|
|
||||||
// if (gridOptions.value?.toolbarConfig?.search) {
|
|
||||||
// }
|
|
||||||
// 强制使用固定的toolbar配置,不允许用户自定义
|
// 强制使用固定的toolbar配置,不允许用户自定义
|
||||||
// 减少配置的复杂度,以及后续维护的成本
|
// 减少配置的复杂度,以及后续维护的成本
|
||||||
toolbarConfig.slots = {
|
toolbarConfig.slots = {
|
||||||
@ -139,7 +140,6 @@ const toolbarOptions = computed(() => {
|
|||||||
: {}),
|
: {}),
|
||||||
...(slotTools ? { tools: TOOLBAR_TOOLS } : {}),
|
...(slotTools ? { tools: TOOLBAR_TOOLS } : {}),
|
||||||
};
|
};
|
||||||
|
|
||||||
return { toolbarConfig };
|
return { toolbarConfig };
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -149,7 +149,7 @@ const options = computed(() => {
|
|||||||
const mergedOptions: VxeTableGridProps = cloneDeep(
|
const mergedOptions: VxeTableGridProps = cloneDeep(
|
||||||
mergeWithArrayOverride(
|
mergeWithArrayOverride(
|
||||||
{},
|
{},
|
||||||
toolbarOptions.value,
|
toRaw(toolbarOptions.value),
|
||||||
toRaw(gridOptions.value),
|
toRaw(gridOptions.value),
|
||||||
globalGridConfig,
|
globalGridConfig,
|
||||||
),
|
),
|
||||||
|
@ -56,5 +56,11 @@
|
|||||||
"timestamp": "Timestamp:",
|
"timestamp": "Timestamp:",
|
||||||
"x": "x:",
|
"x": "x:",
|
||||||
"y": "y:"
|
"y": "y:"
|
||||||
|
},
|
||||||
|
"resize": {
|
||||||
|
"title": "Resize"
|
||||||
|
},
|
||||||
|
"layout": {
|
||||||
|
"col-page": "ColPage Layout"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -59,5 +59,8 @@
|
|||||||
"timestamp": "时间戳:",
|
"timestamp": "时间戳:",
|
||||||
"x": "x:",
|
"x": "x:",
|
||||||
"y": "y:"
|
"y": "y:"
|
||||||
|
},
|
||||||
|
"layout": {
|
||||||
|
"col-page": "双列布局"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -237,6 +237,17 @@ const routes: RouteRecordRaw[] = [
|
|||||||
title: $t('examples.resize.title'),
|
title: $t('examples.resize.title'),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
name: 'ColPageDemo',
|
||||||
|
path: '/examples/layout/col-page',
|
||||||
|
component: () => import('#/views/examples/layout/col-page.vue'),
|
||||||
|
meta: {
|
||||||
|
badge: 'Alpha',
|
||||||
|
badgeVariants: 'destructive',
|
||||||
|
icon: 'material-symbols:horizontal-distribute',
|
||||||
|
title: $t('examples.layout.col-page'),
|
||||||
|
},
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
106
playground/src/views/examples/layout/col-page.vue
Normal file
106
playground/src/views/examples/layout/col-page.vue
Normal file
@ -0,0 +1,106 @@
|
|||||||
|
<script lang="ts" setup>
|
||||||
|
import { reactive, ref } from 'vue';
|
||||||
|
|
||||||
|
import { ColPage } from '@vben/common-ui';
|
||||||
|
import { IconifyIcon } from '@vben/icons';
|
||||||
|
|
||||||
|
import {
|
||||||
|
Alert,
|
||||||
|
Button,
|
||||||
|
Card,
|
||||||
|
Checkbox,
|
||||||
|
Slider,
|
||||||
|
Tag,
|
||||||
|
Tooltip,
|
||||||
|
} from 'ant-design-vue';
|
||||||
|
|
||||||
|
const props = reactive({
|
||||||
|
leftCollapsedWidth: 5,
|
||||||
|
leftCollapsible: true,
|
||||||
|
leftMaxWidth: 50,
|
||||||
|
leftMinWidth: 20,
|
||||||
|
leftWidth: 30,
|
||||||
|
resizable: true,
|
||||||
|
rightWidth: 70,
|
||||||
|
splitHandle: false,
|
||||||
|
splitLine: false,
|
||||||
|
});
|
||||||
|
const leftMinWidth = ref(props.leftMinWidth || 1);
|
||||||
|
const leftMaxWidth = ref(props.leftMaxWidth || 100);
|
||||||
|
</script>
|
||||||
|
<template>
|
||||||
|
<ColPage
|
||||||
|
auto-content-height
|
||||||
|
description="ColPage 是一个双列布局组件,支持左侧折叠、拖拽调整宽度等功能。"
|
||||||
|
v-bind="props"
|
||||||
|
title="ColPage 双列布局组件"
|
||||||
|
>
|
||||||
|
<template #title>
|
||||||
|
<span class="mr-2 text-2xl font-bold">ColPage 双列布局组件</span>
|
||||||
|
<Tag color="hsl(var(--destructive))">Alpha</Tag>
|
||||||
|
</template>
|
||||||
|
<template #left="{ isCollapsed, expand }">
|
||||||
|
<div v-if="isCollapsed" @click="expand">
|
||||||
|
<Tooltip title="点击展开左侧">
|
||||||
|
<Button shape="circle" type="primary">
|
||||||
|
<template #icon>
|
||||||
|
<IconifyIcon class="text-2xl" icon="bi:arrow-right" />
|
||||||
|
</template>
|
||||||
|
</Button>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
v-else
|
||||||
|
:style="{ minWidth: '200px' }"
|
||||||
|
class="border-border bg-card mr-2 rounded-[var(--radius)] border p-2"
|
||||||
|
>
|
||||||
|
<p>这里是左侧内容</p>
|
||||||
|
<p>这里是左侧内容</p>
|
||||||
|
<p>这里是左侧内容</p>
|
||||||
|
<p>这里是左侧内容</p>
|
||||||
|
<p>这里是左侧内容</p>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<Card class="ml-2" title="基本使用">
|
||||||
|
<div class="flex flex-col gap-2">
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<Checkbox v-model:checked="props.resizable">可拖动调整宽度</Checkbox>
|
||||||
|
<Checkbox v-model:checked="props.splitLine">显示拖动分隔线</Checkbox>
|
||||||
|
<Checkbox v-model:checked="props.splitHandle">显示拖动手柄</Checkbox>
|
||||||
|
<Checkbox v-model:checked="props.leftCollapsible">
|
||||||
|
左侧可折叠
|
||||||
|
</Checkbox>
|
||||||
|
</div>
|
||||||
|
<div class="flex items-center gap-2">
|
||||||
|
<span>左侧最小宽度百分比:</span>
|
||||||
|
<Slider
|
||||||
|
v-model:value="leftMinWidth"
|
||||||
|
:max="props.leftMaxWidth - 1"
|
||||||
|
:min="1"
|
||||||
|
style="width: 100px"
|
||||||
|
@after-change="(value) => (props.leftMinWidth = value as number)"
|
||||||
|
/>
|
||||||
|
<span>左侧最大宽度百分比:</span>
|
||||||
|
<Slider
|
||||||
|
v-model:value="props.leftMaxWidth"
|
||||||
|
:max="100"
|
||||||
|
:min="leftMaxWidth + 1"
|
||||||
|
style="width: 100px"
|
||||||
|
@after-change="(value) => (props.leftMaxWidth = value as number)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<Alert message="实验性的组件" show-icon type="warning">
|
||||||
|
<template #description>
|
||||||
|
<p>
|
||||||
|
双列布局组件是一个在Page组件上扩展的相对基础的布局组件,支持左侧折叠(当拖拽导致左侧宽度比最小宽度还要小时,还可以进入折叠状态)、拖拽调整宽度等功能。
|
||||||
|
</p>
|
||||||
|
<p>以上宽度设置的数值是百分比,最小值为1,最大值为100。</p>
|
||||||
|
<p class="font-bold text-red-600">
|
||||||
|
这是一个实验性的组件,用法可能会发生变动,也可能最终不会被采用。在其用法正式出现在文档中之前,不建议在生产环境中使用。
|
||||||
|
</p>
|
||||||
|
</template>
|
||||||
|
</Alert>
|
||||||
|
</div>
|
||||||
|
</Card>
|
||||||
|
</ColPage>
|
||||||
|
</template>
|
Loading…
Reference in New Issue
Block a user