@fe-free/core 2.2.7 → 2.2.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,12 @@
1
1
  # @fe-free/core
2
2
 
3
+ ## 2.2.8
4
+
5
+ ### Patch Changes
6
+
7
+ - feat: global
8
+ - @fe-free/tool@2.2.8
9
+
3
10
  ## 2.2.7
4
11
 
5
12
  ### Patch Changes
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@fe-free/core",
3
- "version": "2.2.7",
3
+ "version": "2.2.8",
4
4
  "description": "",
5
5
  "main": "./src/index.ts",
6
6
  "author": "",
@@ -28,7 +28,9 @@
28
28
  "ahooks": "^3.7.8",
29
29
  "axios": "^1.6.5",
30
30
  "classnames": "^2.5.1",
31
+ "file-saver": "^2.0.5",
31
32
  "github-markdown-css": "^5.8.1",
33
+ "i": "^0.3.7",
32
34
  "localforage": "^1.10.0",
33
35
  "lodash-es": "^4.17.21",
34
36
  "react-ace": "^11.0.1",
@@ -39,7 +41,7 @@
39
41
  "remark-gfm": "^4.0.1",
40
42
  "vanilla-jsoneditor": "^0.23.1",
41
43
  "zustand": "^4.5.4",
42
- "@fe-free/tool": "2.2.7"
44
+ "@fe-free/tool": "2.2.8"
43
45
  },
44
46
  "peerDependencies": {
45
47
  "@ant-design/pro-components": "2.8.9",
@@ -0,0 +1,89 @@
1
+ import { message } from 'antd';
2
+ import type { AxiosRequestConfig, AxiosResponse } from 'axios';
3
+ import { AxiosError } from 'axios';
4
+
5
+ class RequestError extends Error {
6
+ silent: boolean | undefined;
7
+ status: string | undefined;
8
+ config: AxiosRequestConfig<any> | undefined;
9
+ request: XMLHttpRequest | undefined;
10
+ response: AxiosResponse<any, any> | undefined;
11
+
12
+ constructor(
13
+ message: string,
14
+ {
15
+ silent,
16
+ config,
17
+ request,
18
+ response,
19
+ }: {
20
+ silent?: boolean;
21
+ status?: number;
22
+ config?: AxiosRequestConfig;
23
+ request?: XMLHttpRequest;
24
+ response?: AxiosResponse;
25
+ },
26
+ ) {
27
+ super(message);
28
+ this.name = 'RequestError';
29
+
30
+ this.silent = silent;
31
+ this.status = status;
32
+ this.config = config;
33
+ this.request = request;
34
+ this.response = response;
35
+ }
36
+ }
37
+
38
+ function commonHandleError(event) {
39
+ console.log('handleError', event);
40
+
41
+ if (event.reason) {
42
+ if (event.reason instanceof AxiosError) {
43
+ if (event.reason.code === 'ERR_NETWORK') {
44
+ message.error('网络异常');
45
+ return;
46
+ }
47
+ if (event.reason.code === 'ECONNABORTED') {
48
+ message.error('请求超时');
49
+ return;
50
+ }
51
+
52
+ if (event.reason.code === 'ERR_CANCELED') {
53
+ // 正常逻辑
54
+ return;
55
+ }
56
+ }
57
+
58
+ if (event.reason.message || event.reason.reason) {
59
+ message.error(event.reason.message || event.reason.reason);
60
+ return;
61
+ }
62
+ }
63
+
64
+ if (event.error) {
65
+ if (event.error.message || event.error.reason) {
66
+ message.error(event.error.message || event.error.reason);
67
+ return;
68
+ }
69
+ }
70
+
71
+ // unknown error
72
+ }
73
+
74
+ function initErrorHandle(onError) {
75
+ const handleError = (event) => {
76
+ commonHandleError(event);
77
+ onError?.(event);
78
+ };
79
+
80
+ window.addEventListener('error', handleError);
81
+ window.addEventListener('unhandledrejection', handleError);
82
+
83
+ return () => {
84
+ window.removeEventListener('error', handleError);
85
+ window.removeEventListener('unhandledrejection', handleError);
86
+ };
87
+ }
88
+
89
+ export { initErrorHandle, RequestError };
@@ -0,0 +1,34 @@
1
+ import type { AxiosInstance } from 'axios';
2
+ import { saveAs } from 'file-saver';
3
+
4
+ function downloadInterceptor(instance: AxiosInstance) {
5
+ instance.interceptors.response.use(async function (response) {
6
+ const contentDisposition = response.headers['content-disposition'];
7
+
8
+ if (contentDisposition) {
9
+ let filename;
10
+
11
+ // 更加健壮且简洁的写法,优先处理 filename*=,否则处理 filename=
12
+ if (contentDisposition.includes('filename*=')) {
13
+ // RFC 5987 格式:filename*=utf-8''xxx
14
+ const match = contentDisposition.match(/filename\*\s*=\s*([^']*)''([^;]+)/i);
15
+ filename = match ? decodeURIComponent(match[2]) : undefined;
16
+ } else if (contentDisposition.includes('filename=')) {
17
+ // 普通格式:filename=xxx
18
+ const match = contentDisposition.match(/filename\s*=\s*("?)([^";]+)\1/i);
19
+ filename = match ? decodeURIComponent(match[2]) : undefined;
20
+ }
21
+
22
+ filename = filename || 'download';
23
+
24
+ // 处理文件下载 - 确保正确处理二进制数据
25
+ const blob = response.data;
26
+
27
+ saveAs(blob, filename);
28
+ }
29
+
30
+ return response;
31
+ });
32
+ }
33
+
34
+ export { downloadInterceptor };
package/src/index.ts CHANGED
@@ -30,6 +30,8 @@ export {
30
30
  ProFormUploadDragger,
31
31
  proFormSelectSearchProps,
32
32
  } from './form';
33
+ export { RequestError, initErrorHandle } from './global/error';
34
+ export { downloadInterceptor } from './global/interceptors';
33
35
  export { Markdown } from './markdown';
34
36
  export { PageLayout } from './page_layout';
35
37
  export { routeTool } from './route';