diff --git a/packages/components/components.ts b/packages/components/components.ts index 873f7fd..ca75aca 100644 --- a/packages/components/components.ts +++ b/packages/components/components.ts @@ -4,3 +4,5 @@ export * from "../core/src/item/component"; export * from "../core/src/icon/component"; export * from "../core/src/switch/component"; export * from "../core/src/config-provider/component"; +export * from "../core/src/tabs/component"; +export * from "../core/src/tab-pane/component"; diff --git a/packages/core/src/config-provider/component.ts b/packages/core/src/config-provider/component.ts index 98f69b8..34dc00c 100644 --- a/packages/core/src/config-provider/component.ts +++ b/packages/core/src/config-provider/component.ts @@ -5,10 +5,10 @@ import { provideSharedConfig } from "@hoci/core"; export const HiConfigProvider = defineComponent({ props: { icon: { - type: Object as PropType + type: Object as PropType> }, activateEvent: { - type: String as PropType + type: String as PropType> } }, setup(props, context) { diff --git a/packages/core/src/item/component.ts b/packages/core/src/item/component.ts index 4435798..db505ae 100644 --- a/packages/core/src/item/component.ts +++ b/packages/core/src/item/component.ts @@ -1,5 +1,6 @@ -import { capitalize, defineComponent, h } from "vue"; +import { defineComponent, h } from "vue"; import { itemProps, useSelectionItem } from "@hoci/core"; +import { capitalize } from "tslx"; export const HiItem = defineComponent({ name: "HiItem", diff --git a/packages/core/src/item/index.ts b/packages/core/src/item/index.ts index ebde7c2..263a9df 100644 --- a/packages/core/src/item/index.ts +++ b/packages/core/src/item/index.ts @@ -47,16 +47,20 @@ export const useSelectionItem = defineHookComponent({ context.changeActive(props.value); }; + const label = computed(() => { + let label = props.label ?? context.label; + if (label && typeof label == "function") { + label = label(props.value)!; + } + return Array.isArray(label) ? label : [label]; + }); + function render() { return renderSlot(slots, "default", { active: context.isActive(props.value), activate }, () => { - let label = props.label ?? context.label; - if (label && typeof label == "function") { - label = label(props.value)!; - } - return Array.isArray(label) ? label : [label]; + return label.value; }); } @@ -104,7 +108,8 @@ export const useSelectionItem = defineHookComponent({ isActive, isDisabled, className, - activateEvent + activateEvent, + label }; } }); diff --git a/packages/core/src/tab-pane/component.ts b/packages/core/src/tab-pane/component.ts new file mode 100644 index 0000000..426cc2e --- /dev/null +++ b/packages/core/src/tab-pane/component.ts @@ -0,0 +1,21 @@ +import { defineComponent, h } from "vue"; +import { capitalize } from "tslx"; +import { itemProps, useSelectionItem } from "../item"; + +export const HiTabPane = defineComponent({ + props: { + ...itemProps + }, + setup(props, context) { + const { className, activateEvent, activate, isDisabled, label } = useSelectionItem(props, context); + return () => { + return h("div", + { + class: className.value, + [`on${capitalize(activateEvent.value)}`]: activate, + disabled: isDisabled.value + } + , label.value); + }; + } +}); diff --git a/packages/core/src/tabs/component.ts b/packages/core/src/tabs/component.ts new file mode 100644 index 0000000..ef368af --- /dev/null +++ b/packages/core/src/tabs/component.ts @@ -0,0 +1,42 @@ +import { defineComponent, h, renderSlot } from "vue"; +import { classPropType } from "@hoci/shared"; +import { selectionProps, useSelectionList } from "../selection"; + +export const HiTabs = defineComponent({ + props: { + ...selectionProps, + headerClass: { + type: classPropType + }, + contentClass: { + type: classPropType + }, + as: { + type: String, + default: "div" + }, + headerAs: { + type: String, + default: "div" + }, + contentAs: { + type: String, + default: "div" + } + }, + setup(props, context) { + const selection = useSelectionList(props, context); + + return () => { + const content = selection.renderItem(); + return h(props.as, [ + h(props.headerAs, { + class: props.headerClass + }, renderSlot(context.slots, "default")), + h(props.contentAs, { + class: props.contentClass + }, content) + ]); + }; + } +}); diff --git a/playground/src/app.vue b/playground/src/app.vue index a5366bd..3aaae0b 100644 --- a/playground/src/app.vue +++ b/playground/src/app.vue @@ -1,8 +1,14 @@