乐趣区

关于前端:vuerouter源码十三RouterLink源码分析

前言

【vue-router 源码】系列文章将带你从 0 开始理解 vue-router 的具体实现。该系列文章源码参考 vue-router v4.0.15
源码地址:https://github.com/vuejs/router
浏览该文章的前提是你最好理解 vue-router 的根本应用,如果你没有应用过的话,可通过 vue-router 官网学习下。

该篇文章将剖析 RouterLink 组件的实现。

应用

<RouterLink
 to="/inex"
 reaplace
 custom
 activeClass="active"
 exactActiveClass="exact-active"
 ariaCurrentValue="page"
>To Index Page</RouterLink>

RouterLink

export const RouterLinkImpl = /*#__PURE__*/ defineComponent({
  name: 'RouterLink',
  props: {
    // 指标路由的链接
    to: {type: [String, Object] as PropType<RouteLocationRaw>,
      required: true,
    },
    // 决定是否调用 router.push()还是 router.replace()
    replace: Boolean,
    // 链接被激活时,用于渲染 a 标签的 class
    activeClass: String,
    // inactiveClass: String,
    // 链接精准激活时,用于渲染 a 标签的 class
    exactActiveClass: String,
    // 是否不应该将内容包裹在 <a/> 标签中
    custom: Boolean,
    // 传递给 aria-current 属性的值。https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/Attributes/aria-current
    ariaCurrentValue: {type: String as PropType<RouterLinkProps['ariaCurrentValue']>,
      default: 'page',
    },
  },
  useLink,

  setup(props, { slots}) {
    // 应用 useLink 创立 router-link 所需的一些属性和行为
    const link = reactive(useLink(props))
    // createRouter 时传入的 options
    const {options} = inject(routerKey)!

    // class 对象
    const elClass = computed(() => ({
      [getLinkClass(
        props.activeClass,
        options.linkActiveClass,
        'router-link-active'
      )]: link.isActive, // 被激活时的 class
      [getLinkClass(
        props.exactActiveClass,
        options.linkExactActiveClass,
        'router-link-exact-active'
      )]: link.isExactActive, // 被精准激活的 class
    }))

    return () => {
      // 默认插槽
      const children = slots.default && slots.default(link)
      // 如果设置了 props.custom,间接显示 chldren,反之须要应用 a 标签包裹
      return props.custom
        ? children
        : h(
            'a',
            {
              'aria-current': link.isExactActive
                ? props.ariaCurrentValue
                : null,
              href: link.href,
              onClick: link.navigate,
              class: elClass.value,
            },
            children
          )
    }
  },
})

export const RouterLink = RouterLinkImpl as unknown as {new (): {
    $props: AllowedComponentProps &
      ComponentCustomProps &
      VNodeProps &
      RouterLinkProps

    $slots: {default: (arg: UnwrapRef<ReturnType<typeof useLink>>) => VNode[]}
  }
  useLink: typeof useLink
}
退出移动版