Skip to content

Commit bf3c2f3

Browse files
authored
tweak buckets files list UI (#166)
* add upload hint when files table is empty * tweak the ui of bucket tooltip * tweak batch generate links when bucket protect
1 parent 27f99a8 commit bf3c2f3

File tree

13 files changed

+199
-78
lines changed

13 files changed

+199
-78
lines changed

src/renderer/components/empty-holder/index.tsx

+15-3
Original file line numberDiff line numberDiff line change
@@ -5,11 +5,15 @@ import {useI18n} from "@renderer/modules/i18n";
55
import LoadingHolder from "../loading-holder";
66

77
interface EmptyHolderProps {
8+
icon?: React.ReactElement,
9+
subtitle?: React.ReactElement,
810
loading?: boolean,
911
col?: number,
1012
}
1113

1214
const EmptyHolder: React.FC<EmptyHolderProps> = ({
15+
icon = null,
16+
subtitle = null,
1317
loading,
1418
col,
1519
}) => {
@@ -24,14 +28,22 @@ const EmptyHolder: React.FC<EmptyHolderProps> = ({
2428
if (col) {
2529
return (
2630
<tr>
27-
<td className="text-center text-muted" colSpan={col}>{translate("common.empty")}</td>
31+
<td className="text-center text-body text-opacity-25" colSpan={col}>
32+
<div className="d-flex flex-column justify-content-center align-items-center">
33+
{icon}
34+
<span>{translate("common.empty")}</span>
35+
{subtitle}
36+
</div>
37+
</td>
2838
</tr>
2939
);
3040
}
3141

3242
return (
33-
<div className="d-flex justify-content-center align-items-center text-muted w-100 h-100">
34-
{translate("common.empty")}
43+
<div className="d-flex flex-column justify-content-center align-items-center text-body text-opacity-25 w-100 h-100">
44+
{icon}
45+
<span>{translate("common.empty")}</span>
46+
{subtitle}
3547
</div>
3648
);
3749
};

src/renderer/components/modals/file/generate-file-links/index.tsx

+1-2
Original file line numberDiff line numberDiff line change
@@ -376,8 +376,7 @@ const GenerateFileLinks: React.FC<ModalProps & GenerateFileLinksProps> = (props)
376376
<Modal.Footer>
377377
{
378378
!memoFileItems.length ||
379-
batchProgressState.status === BatchTaskStatus.Ended ||
380-
domain?.protected
379+
batchProgressState.status === BatchTaskStatus.Ended
381380
? null
382381
: <span ref={submitButtonRef}/>
383382
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
import {useCallback, useEffect, useState} from "react";
2+
3+
const useIsOverflow = () => {
4+
const [element, setElement] = useState<HTMLElement>();
5+
const [isOverflow, setIsOverflow] = useState(false);
6+
7+
const ref = useCallback((node: HTMLElement | null) => {
8+
if (!node) {
9+
return;
10+
}
11+
12+
setElement(node);
13+
}, []);
14+
15+
useEffect(() => {
16+
if (!element) {
17+
return;
18+
}
19+
20+
const checkOverflow = () => {
21+
if (
22+
element.clientWidth < element.scrollWidth ||
23+
element.clientHeight < element.scrollHeight
24+
) {
25+
setIsOverflow(true);
26+
} else {
27+
setIsOverflow(false);
28+
}
29+
};
30+
31+
checkOverflow();
32+
33+
const observer = new ResizeObserver(checkOverflow);
34+
observer.observe(element);
35+
36+
return () => {
37+
observer.disconnect();
38+
};
39+
}, [element]);
40+
41+
return {
42+
ref,
43+
isOverflow,
44+
};
45+
};
46+
47+
export default useIsOverflow;

src/renderer/modules/i18n/lang/dict.ts

+1
Original file line numberDiff line numberDiff line change
@@ -172,6 +172,7 @@ export default interface Dictionary {
172172
fileStorageClass: string,
173173
fileModifyDate: string,
174174
fileOperation: string,
175+
emptyHint: string,
175176
loadMore: string,
176177
loadMoreFailed: string,
177178
},

src/renderer/modules/i18n/lang/en-us.ts

+3-2
Original file line numberDiff line numberDiff line change
@@ -168,6 +168,7 @@ const dict: Dictionary = {
168168
fileStorageClass: "Storage Class",
169169
fileModifyDate: "Last Modify Date",
170170
fileOperation: "Actions",
171+
emptyHint: "Drop local files or directors to here to upload",
171172
loadMore: "Load more",
172173
loadMoreFailed: "Load more failed. ",
173174
},
@@ -202,8 +203,8 @@ const dict: Dictionary = {
202203
},
203204
upload: {
204205
dropZone: {
205-
enter: "Drag here to upload",
206-
over: "Release to upload",
206+
enter: "Drag to here to upload",
207+
over: "Drop to upload",
207208
},
208209
dialog: {
209210
title: "Select upload files",

src/renderer/modules/i18n/lang/ja-jp.ts

+3-2
Original file line numberDiff line numberDiff line change
@@ -167,6 +167,7 @@ const dict: Dictionary = {
167167
fileStorageClass: "保管タイプ",
168168
fileModifyDate: "最終更新日",
169169
fileOperation: "アクション",
170+
emptyHint: "ここにファイルやフォルダをドラッグ&ドロップしてアップロード",
170171
loadMore: "もっと読み込む...",
171172
loadMoreFailed: "読み込みが失敗しました,"
172173
},
@@ -201,8 +202,8 @@ const dict: Dictionary = {
201202
},
202203
upload: {
203204
dropZone: {
204-
enter: "Drag here to upload",
205-
over: "Release to upload",
205+
enter: "ここにドラッグしてアップロード",
206+
over: "ドロップしてアップロード",
206207
},
207208
dialog: {
208209
title: "[ファイルのアップロード] を選択",

src/renderer/modules/i18n/lang/zh-cn.ts

+1
Original file line numberDiff line numberDiff line change
@@ -167,6 +167,7 @@ const dict: Dictionary = {
167167
fileStorageClass: "存储类型",
168168
fileModifyDate: "最后修改时间",
169169
fileOperation: "操作",
170+
emptyHint: "拖拽本地文件或目录至此处即可上传",
170171
loadMore: "加载更多",
171172
loadMoreFailed: "加载更多失败,",
172173
},

src/renderer/pages/browse/buckets/bucket-table-row.tsx

+63-51
Original file line numberDiff line numberDiff line change
@@ -4,43 +4,11 @@ import classNames from "classnames";
44
import moment from "moment";
55

66
import {BucketItem} from "@renderer/modules/qiniu-client";
7-
import TooltipText from "@renderer/components/tooltip-text";
87
import {useI18n} from "@renderer/modules/i18n";
8+
import useIsOverflow from "@renderer/modules/hooks/use-is-overflow";
9+
import TooltipText from "@renderer/components/tooltip-text";
910
import Duration from "@common/const/duration";
1011

11-
interface BucketNameTooltipProps {
12-
bucket: BucketItem
13-
}
14-
15-
const BucketNameTooltip: React.FC<BucketNameTooltipProps> = ({
16-
bucket,
17-
}) => {
18-
const {translate} = useI18n();
19-
20-
// granted permission text
21-
let grantedPermissionTip: string | null = null
22-
if (bucket.grantedPermission === "readonly") {
23-
grantedPermissionTip = translate("browse.bucketTable.bucketGrantedReadOnly");
24-
} else if (bucket.grantedPermission === "readwrite") {
25-
grantedPermissionTip = translate("browse.bucketTable.bucketGrantedReadWrite");
26-
}
27-
28-
return (
29-
<>
30-
<div className="text-start">
31-
{bucket.name}
32-
{
33-
grantedPermissionTip &&
34-
<span className="text-info">{grantedPermissionTip}</span>
35-
}
36-
</div>
37-
<div className="text-start">
38-
<small className="text-secondary">{bucket.remark}</small>
39-
</div>
40-
</>
41-
);
42-
};
43-
4412
interface BucketTableRowProps {
4513
data: BucketItem,
4614
isSelected: boolean,
@@ -54,6 +22,8 @@ const BucketTableRow: React.FC<BucketTableRowProps> = ({
5422
onClickRow,
5523
onClickBucket,
5624
}) => {
25+
const {translate} = useI18n();
26+
5727
// icon class name
5828
let iconClassName = "bi bi-database-fill me-1 text-brown";
5929
if (bucket.grantedPermission === "readonly") {
@@ -62,6 +32,24 @@ const BucketTableRow: React.FC<BucketTableRowProps> = ({
6232
iconClassName = "bic bic-database-fill-pencil me-1 text-slate";
6333
}
6434

35+
// granted permission text
36+
let grantedPermissionTip: string | null = null
37+
if (bucket.grantedPermission === "readonly") {
38+
grantedPermissionTip = translate("browse.bucketTable.bucketGrantedReadOnly");
39+
} else if (bucket.grantedPermission === "readwrite") {
40+
grantedPermissionTip = translate("browse.bucketTable.bucketGrantedReadWrite");
41+
}
42+
43+
// is overflow
44+
const {
45+
ref: bucketRemarkRef,
46+
isOverflow: bucketRemarkIsOverflow,
47+
} = useIsOverflow();
48+
const {
49+
ref: bucketNameRef,
50+
isOverflow: bucketNameIsOverflow,
51+
} = useIsOverflow();
52+
6553
// render
6654
return (
6755
<tr
@@ -77,24 +65,38 @@ const BucketTableRow: React.FC<BucketTableRowProps> = ({
7765
}}
7866
/>
7967
</td>
80-
<td>
81-
<TooltipText
82-
tooltipPlacement="right"
83-
delay={{
84-
show: Duration.Second,
85-
hide: 0,
86-
}}
87-
tooltipContent={<BucketNameTooltip bucket={bucket}/>}
88-
>
89-
<div
90-
className="d-inline-flex flex-column"
68+
{/* The `line height: 0` fixes height incorrect of row causing by `overflow-ellipsis-inline` */}
69+
<td style={{lineHeight: 0}}>
70+
<div>
71+
<TooltipText
72+
disabled={!bucket.grantedPermission && !bucketNameIsOverflow}
73+
tooltipPlacement="right"
74+
delay={{
75+
show: Duration.Second,
76+
hide: 0,
77+
}}
78+
tooltipContent={
79+
<div className="text-start">
80+
{
81+
bucketNameIsOverflow &&
82+
<div className="text-break-all">
83+
{bucket.name}
84+
</div>
85+
}
86+
<div className="text-info">
87+
{grantedPermissionTip}
88+
</div>
89+
</div>
90+
}
9191
>
9292
<span
9393
tabIndex={0}
94-
className="text-link overflow-ellipsis"
94+
className="text-link overflow-ellipsis-inline"
9595
style={{
9696
["--line-num" as any]: bucket.remark ? 1 : 2,
97+
lineHeight: "var(--bs-body-line-height)",
9798
}}
99+
ref={bucketNameRef}
98100
onKeyUp={e => {
99101
if (e.code === "Space") {
100102
e.stopPropagation();
@@ -109,16 +111,26 @@ const BucketTableRow: React.FC<BucketTableRowProps> = ({
109111
<i className={iconClassName}/>
110112
{bucket.name}
111113
</span>
112-
{
113-
bucket.remark &&
114+
</TooltipText>
115+
</div>
116+
{
117+
bucket.remark &&
118+
<div>
119+
<TooltipText
120+
disabled={!bucketRemarkIsOverflow}
121+
tooltipPlacement="right"
122+
tooltipContent={<div className="text-start">{bucket.remark}</div>}
123+
>
114124
<small
115-
className="text-secondary overflow-ellipsis"
125+
ref={bucketRemarkRef}
126+
className="text-secondary overflow-ellipsis-inline"
127+
style={{lineHeight: "var(--bs-body-line-height)"}}
116128
>
117129
{bucket.remark}
118130
</small>
119-
}
131+
</TooltipText>
120132
</div>
121-
</TooltipText>
133+
}
122134
</td>
123135
{/* may empty string, so use `||` instead of `??` */}
124136
<td className="align-middle">{bucket.regionName || bucket.regionId}</td>

src/renderer/pages/browse/buckets/bucket-table.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,7 @@ const BucketTable: React.FC<BucketTableProps> = ({
5050
>
5151
<colgroup>
5252
<col style={{width: "2rem"}}/>
53-
<col/>
53+
<col style={{width: "40rem"}}/>
5454
<col style={{minWidth: "12rem"}}/>
5555
<col style={{minWidth: "14rem"}}/>
5656
</colgroup>

src/renderer/pages/browse/files/file-grid/index.tsx

+24-1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ import StorageClass from "@common/models/storage-class";
66

77
import {FileItem} from "@renderer/modules/qiniu-client";
88

9+
import {useI18n} from "@renderer/modules/i18n";
910
import EmptyHolder from "@renderer/components/empty-holder";
1011
import BaseGrid from "@renderer/components/base-grid";
1112

@@ -40,6 +41,8 @@ const FileGrid: React.FC<FileGridProps> = ({
4041
onSelectFiles,
4142
onDoubleClickFile,
4243
}) => {
44+
const {translate} = useI18n();
45+
4346
const filesData: FileRowData[] = useMemo(() =>
4447
data.map(item => ({
4548
...item,
@@ -80,7 +83,27 @@ const FileGrid: React.FC<FileGridProps> = ({
8083
rowHeight={GRID_CELL_HEIGHT}
8184
endReachedThreshold={LOAD_MORE_THRESHOLD}
8285
onEndReached={handleEndReached}
83-
emptyRender={<EmptyHolder loading={loading}/>}
86+
emptyRender={
87+
<EmptyHolder
88+
icon={
89+
<i
90+
className="bi bi-inbox"
91+
style={{
92+
fontSize: '4rem',
93+
lineHeight: 1,
94+
}}
95+
/>
96+
}
97+
subtitle={
98+
<span
99+
className="text-body mt-2"
100+
>
101+
{translate("browse.fileTable.emptyHint")}
102+
</span>
103+
}
104+
loading={loading}
105+
/>
106+
}
84107
overlayRender={
85108
<OverlayHolder
86109
loadingMore={loadingMore}

0 commit comments

Comments
 (0)