Commit a161bfa818cb63d9cc0b00ae062eb16b1efaf74f

Authored by vben
1 parent 78d4d41c

feat: added base64 file stream download

CHANGELOG.zh_CN.md
  1 +## Wip
  2 +
  3 +### ✨ Features
  4 +
  5 +- 新增 base64 文件流下载
  6 +
1 ## 2.0.0-rc.10 (2020-11-13) 7 ## 2.0.0-rc.10 (2020-11-13)
2 8
3 ### ✨ Refactor 9 ### ✨ Refactor
src/components/Menu/src/index.less
@@ -49,56 +49,10 @@ @@ -49,56 +49,10 @@
49 } 49 }
50 } 50 }
51 51
52 - // collapsed show title end  
53 - .ant-menu-item,  
54 - .ant-menu-submenu-title {  
55 - > .basic-menu__name {  
56 - width: 100%;  
57 -  
58 - .basic-menu__tag {  
59 - float: right;  
60 - margin-top: @app-menu-item-height / 2;  
61 - transform: translate(0%, -50%);  
62 - }  
63 - }  
64 - }  
65 -  
66 .ant-menu-item { 52 .ant-menu-item {
67 transition: unset; 53 transition: unset;
68 } 54 }
69 55
70 - &__tag {  
71 - display: inline-block;  
72 - padding: 2px 4px;  
73 - margin-right: 4px;  
74 - font-size: 12px;  
75 - line-height: 14px;  
76 - color: #fff;  
77 - border-radius: 2px;  
78 -  
79 - &.dot {  
80 - width: 8px;  
81 - height: 8px;  
82 - padding: 0;  
83 - border-radius: 50%;  
84 - }  
85 -  
86 - &.primary {  
87 - background: @primary-color;  
88 - }  
89 -  
90 - &.error {  
91 - background: @error-color;  
92 - }  
93 -  
94 - &.success {  
95 - background: @success-color;  
96 - }  
97 -  
98 - &.warn {  
99 - background: @warning-color;  
100 - }  
101 - }  
102 // scrollbar -s tart 56 // scrollbar -s tart
103 &__content { 57 &__content {
104 /* 滚动槽 */ 58 /* 滚动槽 */
@@ -245,6 +199,11 @@ @@ -245,6 +199,11 @@
245 margin: 0; 199 margin: 0;
246 line-height: @app-menu-item-height; 200 line-height: @app-menu-item-height;
247 } 201 }
  202 +
  203 + .ant-menu-item.basic-menu-item__level1 {
  204 + height: @app-menu-item-height;
  205 + line-height: @app-menu-item-height;
  206 + }
248 } 207 }
249 208
250 // 层级样式 209 // 层级样式
@@ -381,3 +340,51 @@ @@ -381,3 +340,51 @@
381 } 340 }
382 } 341 }
383 } 342 }
  343 +// collapsed show title end
  344 +.ant-menu-item,
  345 +.ant-menu-submenu-title {
  346 + > .basic-menu__name {
  347 + width: 100%;
  348 +
  349 + .basic-menu__tag {
  350 + float: right;
  351 + margin-top: @app-menu-item-height / 2;
  352 + transform: translate(0%, -50%);
  353 + }
  354 + }
  355 +}
  356 +
  357 +.basic-menu__tag {
  358 + display: inline-block;
  359 + padding: 2px 4px;
  360 + margin-right: 4px;
  361 + font-size: 12px;
  362 + line-height: 14px;
  363 + color: #fff;
  364 + border-radius: 2px;
  365 +
  366 + &.dot {
  367 + width: 8px;
  368 + height: 8px;
  369 + padding: 0;
  370 + margin-top: 21px !important;
  371 + margin-bottom: 2px;
  372 + border-radius: 50%;
  373 + }
  374 +
  375 + &.primary {
  376 + background: @primary-color;
  377 + }
  378 +
  379 + &.error {
  380 + background: @error-color;
  381 + }
  382 +
  383 + &.success {
  384 + background: @success-color;
  385 + }
  386 +
  387 + &.warn {
  388 + background: @warning-color;
  389 + }
  390 +}
src/design/var/index.less
@@ -6,7 +6,7 @@ @@ -6,7 +6,7 @@
6 @multiple-height: 30px; 6 @multiple-height: 30px;
7 7
8 // headers 8 // headers
9 -@header-height: 46px; 9 +@header-height: 48px;
10 10
11 // logo width 11 // logo width
12 @logo-width: 36px; 12 @logo-width: 36px;
src/layouts/logo/index.vue
@@ -93,5 +93,13 @@ @@ -93,5 +93,13 @@
93 opacity: 1; 93 opacity: 1;
94 }); 94 });
95 } 95 }
  96 +
  97 + // &.dark .logo-title {
  98 + // font-weight: 400;
  99 + // }
  100 +
  101 + &.light .logo-title {
  102 + color: @primary-color;
  103 + }
96 } 104 }
97 </style> 105 </style>
src/settings/colorSetting.ts
@@ -8,7 +8,7 @@ export const HEADER_PRESET_BG_COLOR_LIST: string[] = [ @@ -8,7 +8,7 @@ export const HEADER_PRESET_BG_COLOR_LIST: string[] = [
8 '#409eff', 8 '#409eff',
9 '#4e73df', 9 '#4e73df',
10 '#e74c3c', 10 '#e74c3c',
11 - '#f39c12', 11 + '#24292e',
12 '#394664', 12 '#394664',
13 '#001529', 13 '#001529',
14 ]; 14 ];
@@ -17,6 +17,7 @@ export const HEADER_PRESET_BG_COLOR_LIST: string[] = [ @@ -17,6 +17,7 @@ export const HEADER_PRESET_BG_COLOR_LIST: string[] = [
17 export const SIDE_BAR_BG_COLOR_LIST: string[] = [ 17 export const SIDE_BAR_BG_COLOR_LIST: string[] = [
18 '#273352', 18 '#273352',
19 '#ffffff', 19 '#ffffff',
  20 + '#191a23',
20 '#001529', 21 '#001529',
21 '#304156', 22 '#304156',
22 '#28333E', 23 '#28333E',
src/utils/file/FileDownload.ts
  1 +import { dataURLtoBlob } from './stream';
  2 +
  3 +export function downloadByBase64(buf: string, filename: string, mime?: string, bom?: BlobPart) {
  4 + const base64Buf = dataURLtoBlob(buf);
  5 + downloadByData(base64Buf, filename, mime, bom);
  6 +}
  7 +
1 /** 8 /**
2 - * 根据后台接口文件流下载 9 + * Download according to the background interface file stream
3 * @param {*} data 10 * @param {*} data
4 * @param {*} filename 11 * @param {*} filename
5 * @param {*} mime 12 * @param {*} mime
@@ -27,7 +34,7 @@ export function downloadByData(data: BlobPart, filename: string, mime?: string, @@ -27,7 +34,7 @@ export function downloadByData(data: BlobPart, filename: string, mime?: string,
27 } 34 }
28 35
29 /** 36 /**
30 - * 根据文件地址下载文件 37 + * Download file according to file address
31 * @param {*} sUrl 38 * @param {*} sUrl
32 */ 39 */
33 export function downloadByUrl({ 40 export function downloadByUrl({
@@ -43,7 +50,7 @@ export function downloadByUrl({ @@ -43,7 +50,7 @@ export function downloadByUrl({
43 const isSafari = window.navigator.userAgent.toLowerCase().indexOf('safari') > -1; 50 const isSafari = window.navigator.userAgent.toLowerCase().indexOf('safari') > -1;
44 51
45 if (/(iP)/g.test(window.navigator.userAgent)) { 52 if (/(iP)/g.test(window.navigator.userAgent)) {
46 - console.error('您的浏览器不支持下载!'); 53 + console.error('Your browser does not support download!');
47 return false; 54 return false;
48 } 55 }
49 if (isChrome || isSafari) { 56 if (isChrome || isSafari) {
src/utils/file/stream.ts 0 → 100644
  1 +/**
  2 + * @description: base64 to blob
  3 + */
  4 +export function dataURLtoBlob(base64Buf: string): Blob {
  5 + const arr = base64Buf.split(',');
  6 + const typeItem = arr[0];
  7 + const mime = typeItem.match(/:(.*?);/)![1];
  8 + const bstr = atob(arr[1]);
  9 + let n = bstr.length;
  10 + const u8arr = new Uint8Array(n);
  11 + while (n--) {
  12 + u8arr[n] = bstr.charCodeAt(n);
  13 + }
  14 + return new Blob([u8arr], { type: mime });
  15 +}
src/views/demo/feat/download/imgBase64.ts 0 → 100644
  1 +export default ``;
src/views/demo/feat/download/index.vue
@@ -4,14 +4,16 @@ @@ -4,14 +4,16 @@
4 <a-button type="primary" class="my-4" @click="handleDownByData"> 文件流下载 </a-button> 4 <a-button type="primary" class="my-4" @click="handleDownByData"> 文件流下载 </a-button>
5 5
6 <a-alert message="根据文件地址下载文件" /> 6 <a-alert message="根据文件地址下载文件" />
7 -  
8 <a-button type="primary" class="my-4" @click="handleDownloadByUrl"> 文件地址下载 </a-button> 7 <a-button type="primary" class="my-4" @click="handleDownloadByUrl"> 文件地址下载 </a-button>
  8 +
  9 + <a-alert message="base64流下载" />
  10 + <a-button type="primary" class="my-4" @click="handleDownloadByBase64"> base64流下载 </a-button>
9 </div> 11 </div>
10 </template> 12 </template>
11 <script lang="ts"> 13 <script lang="ts">
12 import { defineComponent } from 'vue'; 14 import { defineComponent } from 'vue';
13 - import { downloadByUrl, downloadByData } from '/@/utils/file/FileDownload';  
14 - 15 + import { downloadByUrl, downloadByData, downloadByBase64 } from '/@/utils/file/FileDownload';
  16 + import imgBase64 from './imgBase64';
15 export default defineComponent({ 17 export default defineComponent({
16 setup() { 18 setup() {
17 function handleDownByData() { 19 function handleDownByData() {
@@ -23,9 +25,14 @@ @@ -23,9 +25,14 @@
23 target: '_self', 25 target: '_self',
24 }); 26 });
25 } 27 }
  28 +
  29 + function handleDownloadByBase64() {
  30 + downloadByBase64(imgBase64, 'logo.png');
  31 + }
26 return { 32 return {
27 handleDownloadByUrl, 33 handleDownloadByUrl,
28 handleDownByData, 34 handleDownByData,
  35 + handleDownloadByBase64,
29 }; 36 };
30 }, 37 },
31 }); 38 });