3.5.7.9 自定义多Tab

在业务中,可能会遇到需要对多tab的交互或UI做全新设计的需求,这个时候可以自定义多tab组件支持。

首先继承平台的MultiTabsWidget组件,将自己vue文件设置到component处

import { MultiTabsWidget, SPI, ViewWidget } from '@kunlun/dependencies';
import Component from './CustomMultiTabs.vue';

@SPI.ClassFactory(
  ViewWidget.Token({
    // 这里的tagName跟平台的组件一样,这样才能覆盖平台的组件
    tagName: ['multi-tabs', 'MultiTabs']
  })
)
export class CustomMultiTabsWidget extends MultiTabsWidget{

  public initialize(props) {
    super.initialize(props);
    // 设置自定义的vue组件
    this.setComponent(Component);
    return this;
  }
}

vue文件中继承平台的props,编写自定义页面代码

export const MultiTabsProps = {
  /**
   * 组件是否可见
   */
  invisible: {
    type: Boolean
  },
  /**
   * tab列表数据
   */
  tabs: {
    type: Array as PropType<MultiTab[]>
  },
  /**
   * 当前激活的tab
   */
  activeTab: {
    type: Object as PropType<MultiTab>
  },
  /**
   * 鼠标悬浮所在的tab
   */
  hoverTab: {
    type: Object as PropType<MultiTab>
  },
  /**
   * 鼠标经过tab事件回调
   */
  onMouseenterTab: {
    type: Function as PropType<(tab: MultiTab) => void>
  },
  /**
   * 鼠标离开tab事件回调
   */
  onMouseleaveTab: {
    type: Function as PropType<(tab: MultiTab) => void>
  },
  /**
   * 点击tab
   */
  onClickTab: {
    type: Function as PropType<(tab: MultiTab) => void>
  },
  /**
   * 刷新当前tab
   */
  onRefreshCurrentTab: {
    type: Function as PropType<(tab: MultiTab) => void>
  },
  /**
   * 关闭当前tab
   */
  onCloseCurrentTab: {
    type: Function as PropType<(tab: MultiTab) => void>
  },
  /**
   * 关闭除当前tab外的其他所有tab
   */
  onCloseOtherTabs: {
    type: Function as PropType<(tab: MultiTab) => void>
  },
  /**
   * 关闭当前tab左侧的所有tab
   */
  onCloseLeftTabs: {
    type: Function as PropType<(tab: MultiTab) => void>
  },
  /**
   * 关闭当前tab右侧的所有tab
   */
  onCloseRightTabs: {
    type: Function as PropType<(tab: MultiTab) => void>
  },
  /**
   * 新窗口打开当前tab
   */
  onOpenCurrentTabOnNewWindow: {
    type: Function as PropType<(tab: MultiTab) => void>
  }
};
<template>
  <div class="k-layout-multi-tabs-main-area" v-if="tabs && tabs.length && !invisible">
    <div id="k-layout-multi-tabs" class="k-layout-multi-tabs oio-scrollbar">
      <div
        v-for="(tab, index) in tabs"
        :key="tab.key"
        :id="`k-layout-multi-tabs-tab-${index}`"
        class="k-layout-multi-tabs-tab"
        :class="{
          'k-layout-multi-tabs-tab-active': activeTab && tab.key === activeTab.key,
          'active-left-hover': activeTabLeftHoverTab,
          'k-layout-multi-tabs-tab-closable': allowClosable
        }"
        @click="onClickTab(tab)"
        @mouseenter="onMouseenterTab(tab)"
        @mouseleave="onMouseleaveTab(tab)"
      >
        <div class="first-tab-icon" v-if="index === 0">
          <oio-icon icon="oinone-shouye1" />
        </div>
        <div class="k-layout-multi-tabs-tab-title" :title="tab.title">{{ tab.title }}</div>
        <div class="close-button-area" :class="[index === 0 && 'k-layout-multi-tabs-tab-close-btn']">
          <oio-button type="link" @click.stop.prevent="onCloseCurrentTab">
            <oio-icon icon="oinone-guanbi" size="16px" />
          </oio-button>
        </div>
        <div class="split-area" v-if="!isInvisibleSplit(index)"></div>
      </div>
    </div>
    <div class="k-layout-multi-tabs-tools">
      <oio-dropdown trigger="click" overlay-class-name="top-bar-common-dropdown">
        <oio-icon icon="oinone-xiala2" size="16px" />
        <template #overlay>
          <div class="multi-tab-tool-overlay">
            <span
              v-for="item in TabToolItems"
              :key="item.key"
              :class="['multi-tab-tool-overlay-item', isTabToolItemDisable(item) && 'disabled']"
              @click="onTabToolItemClick(item)"
            >
              {{ item.title }}
            </span>
          </div>
        </template>
      </oio-dropdown>
    </div>
  </div>
</template>
<script lang="ts">
import { computed, defineComponent, watch } from 'vue';
import { MultiTabsProps, OioIcon, TabToolItems } from '@kunlun/dependencies';
import { OioButton, OioDropdown } from '@kunlun/vue-ui-antd';

export default defineComponent({
  name: 'MyMultiTabs',
  inheritAttrs: false,
  components: { OioButton, OioDropdown, OioIcon },
  props: {
    ...MultiTabsProps
  },
  setup(props) {
    const activeTabLeftHoverTab = computed(() => {
      return (
        props.activeTab &&
        props.hoverTab &&
        props.activeTab.currentIndex !== undefined &&
        props.hoverTab.currentIndex !== undefined &&
        props.activeTab.currentIndex + 1 === props.hoverTab.currentIndex
      );
    });

    watch(
      () => props.activeTab,
      () => {
        setTimeout(() => {
          if (props.activeTab && props.activeTab.currentIndex) {
            const tabDoc = document.getElementById(`k-layout-multi-tabs-tab-${props.activeTab.currentIndex}`);
            if (tabDoc) {
              tabDoc.scrollIntoView(false);
            }
          }
        }, 10);
      }
    );

    const allowClosable = computed(() => (props.tabs?.length || 0) >= 2);

    function isInvisibleSplit(index: number) {
      if (
        props.hoverTab &&
        props.hoverTab.currentIndex !== undefined &&
        (props.hoverTab.currentIndex === index || props.hoverTab.currentIndex - 1 === index)
      ) {
        return true;
      }
      if (props.activeTab && props.activeTab.currentIndex !== undefined && props.activeTab.currentIndex >= 0) {
        return props.activeTab.currentIndex === index || props.activeTab.currentIndex - 1 === index;
      }
      return false;
    }

    function isTabToolItemDisable(tabToolItem) {
      return tabToolItem.disabled(props.tabs?.length || 0, props.activeTab ? props.activeTab.currentIndex : -1);
    }

    function onTabToolItemClick(tabToolItem) {
      if (isTabToolItemDisable(tabToolItem)) {
        return null;
      }
      props.activeTab && props[tabToolItem.key]?.(props.activeTab);
    }

    return {
      activeTabLeftHoverTab,
      allowClosable,
      TabToolItems,
      isInvisibleSplit,
      isTabToolItemDisable,
      onTabToolItemClick
    };
  }
});
</script>

文件目录结构如下图

image.png

最后再到运行路径中导入该组件

这里以启动工程的 main.ts 举例,如果运行时未看到该组件的效果,请检查是否正确导入到运行时的路径中

image.png

Oinone社区 作者:史, 昂原创文章,如若转载,请注明出处:https://doc.oinone.top/oio4/9272.html

访问Oinone官网:https://www.oinone.top获取数式Oinone低代码应用平台体验

(0)
史, 昂的头像史, 昂数式管理员
上一篇 2024年5月23日 am9:06
下一篇 2024年5月23日 am9:08

相关推荐

  • 5.6 商业支撑之商品域

    一、基础介绍 当业务在线化后,用于内部管理的产品主数据,叠加一堆销售属性变成了商品被推倒了前台,成为导购链路中最最重要的信息载体。看似最基础和最简单的商品模块也有很多门道。主要集中在以下几个方面: 商品的属性如何管理、呈现、参与导购(类目、搜索的过滤条件) 如何解决固定不变的内部管理需求与基于销售特性长期变化的运营需求之间的矛盾 在多渠道情况渠道商品,如何映射到实际sku进行履约 二、模型介绍 图5-6-1 模型介绍 类目属性,解决“商品的属性如何管理、呈现、参与导购(类目、搜索的过滤条件)” 前后台类目设计,解决“如何解决固定不变的内部管理需求与基于销售特性长期变化的运营需求之间的矛盾” 销售Sku和库存Sku设计,解决“在多渠道情况渠道商品,如何映射到实际sku进行履约” 要把这些问题搞清楚,得先把名词统一下: 领域 名称 oinone的定义 说明 举例 平台运营视角 Spu Product –>Spu2.1.9 –> 3.0.0 SPU(Standard Product Unit):标准化产品单元。SPU是商品信息聚合的最小单位,是一组可复用、易检索的标准化信息的集合,该集合描述了一个产品的特性 iPhone X可以确定一个产品 后台类目 后台类目(Category) 商品分类分级管理,以及规范该类目下公共属性可以分为普通属性、销售属性 比如类目:3c数码/手机销售属性:内存大小、颜色等普通属性:分辨率 前台类目 前台类目(FrontCategory) 平台导购类目 通过前台类目关联后台类目或后台类目属性,用于满足运营需求 大体上SPU处于最上层、Item属于下一级,而SKU属于最低一层。SPU是平台层面,Item是商家层面,SKU是商家的Item确定销售属性SPU非必须,在平台类交易中,平台方为了规范商家发布商品信息,进行统一运营时需要 商家销售视角 Item 渠道商品(Item) 简单来说是:SPU加上归属商家、以及商家自有的价格与描述 商家A的iPhone X Sku 销售Sku(SaleSku) SKU=Stock Keeping Unit(库存保有单位)。是对每一个产品和服务的唯一标示符,该系统的使用SKU的值根于数据管理,使公司能够跟踪系统,如仓库和零售商店或产品的库存情况。 iPhone X 64G 银色 则是一个SKU。 店铺类目 ShopCategory 商家店铺导购类目 在平台类电商,商家都会有自己独立的店铺主页,商家类目跟前台类目作用类似,只是局限影响范围为商家店铺内 销售SKU中会有一个InvSkuCode来关联InventorySku,比如:品牌上在不同渠道(淘宝、京东、自建电商)中会有不同的销售SKU,在从渠道同步销售SKU会根据外部code 商家管理视角 产品或库存Sku InventorySku 跟销售领域的sku的定义类似,但销售领域是为了规范购买行为,这里规范企业内部管理。 iPhone X 64G 银色 组合Sku InventorySkuComposition 空调有内外机组合而成,这就是一个组合sku 产品分类 ProductKind 企业内部管理划分 商品系列 ItemSeries 指互相关联或相似的产品,是按照一定的分类标准对企业生产经营的全部产品进行划分的结果。一个产品系列内往往包括多个产品项目。产品系列的划分标准有产品功能、消费上的连带性、面向的顾客群、分销渠道、价格范围等 存货类别 StorageKind 为了反映存货的组成内容,正确计算产品的生产成本以及销售成本,会计上必须对存货进行科学地分类,按存货的不同类别进行核算 表5-6-1 各领域名称说明

    2024年5月23日
    1.1K00

Leave a Reply

登录后才能评论