Exception.vue 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. <script lang="tsx">
  2. import type { PropType } from 'vue';
  3. import { Result, Button } from 'ant-design-vue';
  4. import { defineComponent, ref, computed, unref } from 'vue';
  5. import { ExceptionEnum } from '/@/enums/exceptionEnum';
  6. import notDataSvg from '/@/assets/svg/no-data.svg';
  7. import netWorkSvg from '/@/assets/svg/net-error.svg';
  8. import { useRoute } from 'vue-router';
  9. import { useDesign } from '/@/hooks/web/useDesign';
  10. import { useI18n } from '/@/hooks/web/useI18n';
  11. import { useGo, useRedo } from '/@/hooks/web/usePage';
  12. import { PageEnum } from '/@/enums/pageEnum';
  13. interface MapValue {
  14. title: string;
  15. subTitle: string;
  16. btnText?: string;
  17. icon?: string;
  18. handler?: Fn;
  19. status?: string;
  20. }
  21. export default defineComponent({
  22. name: 'ErrorPage',
  23. props: {
  24. // 状态码
  25. status: {
  26. type: Number as PropType<number>,
  27. default: ExceptionEnum.PAGE_NOT_FOUND,
  28. },
  29. title: {
  30. type: String as PropType<string>,
  31. default: '',
  32. },
  33. subTitle: {
  34. type: String as PropType<string>,
  35. default: '',
  36. },
  37. full: {
  38. type: Boolean as PropType<boolean>,
  39. default: false,
  40. },
  41. },
  42. setup(props) {
  43. const statusMapRef = ref(new Map<string | number, MapValue>());
  44. const { query } = useRoute();
  45. const go = useGo();
  46. const redo = useRedo();
  47. const { t } = useI18n();
  48. const { prefixCls } = useDesign('app-exception-page');
  49. const getStatus = computed(() => {
  50. const { status: routeStatus } = query;
  51. const { status } = props;
  52. return Number(routeStatus) || status;
  53. });
  54. const getMapValue = computed(
  55. (): MapValue => {
  56. return unref(statusMapRef).get(unref(getStatus)) as MapValue;
  57. }
  58. );
  59. const backLoginI18n = t('sys.exception.backLogin');
  60. const backHomeI18n = t('sys.exception.backHome');
  61. unref(statusMapRef).set(ExceptionEnum.PAGE_NOT_ACCESS, {
  62. title: '403',
  63. status: `${ExceptionEnum.PAGE_NOT_ACCESS}`,
  64. subTitle: t('sys.exception.subTitle403'),
  65. btnText: props.full ? backLoginI18n : backHomeI18n,
  66. handler: () => (props.full ? go(PageEnum.BASE_LOGIN) : go()),
  67. });
  68. unref(statusMapRef).set(ExceptionEnum.PAGE_NOT_FOUND, {
  69. title: '404',
  70. status: `${ExceptionEnum.PAGE_NOT_FOUND}`,
  71. subTitle: t('sys.exception.subTitle404'),
  72. btnText: props.full ? backLoginI18n : backHomeI18n,
  73. handler: () => (props.full ? go(PageEnum.BASE_LOGIN) : go()),
  74. });
  75. unref(statusMapRef).set(ExceptionEnum.ERROR, {
  76. title: '500',
  77. status: `${ExceptionEnum.ERROR}`,
  78. subTitle: t('sys.exception.subTitle500'),
  79. btnText: backHomeI18n,
  80. handler: () => go(),
  81. });
  82. unref(statusMapRef).set(ExceptionEnum.PAGE_NOT_DATA, {
  83. title: t('sys.exception.noDataTitle'),
  84. subTitle: '',
  85. btnText: t('common.redo'),
  86. handler: () => redo(),
  87. icon: notDataSvg,
  88. });
  89. unref(statusMapRef).set(ExceptionEnum.NET_WORK_ERROR, {
  90. title: t('sys.exception.networkErrorTitle'),
  91. subTitle: t('sys.exception.networkErrorSubTitle'),
  92. btnText: t('common.redo'),
  93. handler: () => redo(),
  94. icon: netWorkSvg,
  95. });
  96. return () => {
  97. const { title, subTitle, btnText, icon, handler, status } = unref(getMapValue) || {};
  98. return (
  99. <Result
  100. class={prefixCls}
  101. status={status as any}
  102. title={props.title || title}
  103. sub-title={props.subTitle || subTitle}
  104. >
  105. {{
  106. extra: () =>
  107. btnText && (
  108. <Button type="primary" onClick={handler}>
  109. {() => btnText}
  110. </Button>
  111. ),
  112. icon: () => (icon ? <img src={icon} /> : null),
  113. }}
  114. </Result>
  115. );
  116. };
  117. },
  118. });
  119. </script>
  120. <style lang="less">
  121. @prefix-cls: ~'@{namespace}-app-exception-page';
  122. .@{prefix-cls} {
  123. display: flex;
  124. align-items: center;
  125. flex-direction: column;
  126. .ant-result-icon {
  127. img {
  128. max-width: 400px;
  129. max-height: 300px;
  130. }
  131. }
  132. }
  133. </style>