mirror of
https://github.com/minio/minio.git
synced 2025-01-25 13:43:17 -05:00
NewUX: On filename click, select file instead of downloading (#6023)
This commit has been done according to @abperiasamy's feedback as we are going to reserve the click on file icon to open the preview modal in the future. Also, when the user now selects a single file, the file itself is downloaded instead of a .zip file containing the file. Fixes #6019
This commit is contained in:
parent
617a6d8e47
commit
94ec6f374e
@ -37,7 +37,7 @@ export const ObjectContainer = ({
|
|||||||
if (checkedObjectsCount == 0) {
|
if (checkedObjectsCount == 0) {
|
||||||
props.actionButtons = <ObjectActions object={object} />
|
props.actionButtons = <ObjectActions object={object} />
|
||||||
}
|
}
|
||||||
return <ObjectItem {...props} onClick={() => downloadObject(object.name)} />
|
return <ObjectItem {...props} />
|
||||||
}
|
}
|
||||||
|
|
||||||
const mapStateToProps = state => {
|
const mapStateToProps = state => {
|
||||||
@ -46,10 +46,4 @@ const mapStateToProps = state => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const mapDispatchToProps = dispatch => {
|
export default connect(mapStateToProps)(ObjectContainer)
|
||||||
return {
|
|
||||||
downloadObject: object => dispatch(actionsObjects.downloadObject(object))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(ObjectContainer)
|
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import React from "react"
|
import React from "react"
|
||||||
|
import classNames from "classnames"
|
||||||
import { connect } from "react-redux"
|
import { connect } from "react-redux"
|
||||||
import humanize from "humanize"
|
import humanize from "humanize"
|
||||||
import Moment from "moment"
|
import Moment from "moment"
|
||||||
@ -34,19 +35,22 @@ export const ObjectItem = ({
|
|||||||
onClick
|
onClick
|
||||||
}) => {
|
}) => {
|
||||||
return (
|
return (
|
||||||
<div className={"fesl-row"} data-type={getDataType(name, contentType)}>
|
<div
|
||||||
|
className={
|
||||||
|
"fesl-row " +
|
||||||
|
classNames({
|
||||||
|
"fesl-row-selected": checked
|
||||||
|
})
|
||||||
|
}
|
||||||
|
data-type={getDataType(name, contentType)}
|
||||||
|
>
|
||||||
<div className="fesl-item fesl-item-icon">
|
<div className="fesl-item fesl-item-icon">
|
||||||
<div className="fi-select">
|
<div className="fi-select">
|
||||||
<input
|
<input
|
||||||
type="checkbox"
|
type="checkbox"
|
||||||
name={name}
|
name={name}
|
||||||
checked={checked}
|
|
||||||
onChange={() => {
|
|
||||||
checked ? uncheckObject(name) : checkObject(name)
|
|
||||||
}}
|
|
||||||
/>
|
/>
|
||||||
<i className="fis-icon" />
|
<i className="fis-icon" />
|
||||||
<i className="fis-helper" />
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="fesl-item fesl-item-name">
|
<div className="fesl-item fesl-item-name">
|
||||||
@ -54,7 +58,7 @@ export const ObjectItem = ({
|
|||||||
href="#"
|
href="#"
|
||||||
onClick={e => {
|
onClick={e => {
|
||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
onClick()
|
checked ? uncheckObject(name) : checkObject(name)
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{name}
|
{name}
|
||||||
|
@ -28,6 +28,15 @@ export class ObjectsBulkActions extends React.Component {
|
|||||||
showDeleteConfirmation: false
|
showDeleteConfirmation: false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
handleDownload() {
|
||||||
|
const { checkedObjects, resetCheckedList, downloadChecked, downloadObject } = this.props
|
||||||
|
if (checkedObjects.length === 1) {
|
||||||
|
downloadObject(checkedObjects[0])
|
||||||
|
resetCheckedList()
|
||||||
|
} else {
|
||||||
|
downloadChecked()
|
||||||
|
}
|
||||||
|
}
|
||||||
deleteChecked() {
|
deleteChecked() {
|
||||||
const { deleteChecked } = this.props
|
const { deleteChecked } = this.props
|
||||||
deleteChecked()
|
deleteChecked()
|
||||||
@ -39,24 +48,28 @@ export class ObjectsBulkActions extends React.Component {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
render() {
|
render() {
|
||||||
const { checkedObjectsCount, downloadChecked, clearChecked } = this.props
|
const { checkedObjects, clearChecked } = this.props
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={
|
className={
|
||||||
"list-actions" +
|
"list-actions" +
|
||||||
classNames({
|
classNames({
|
||||||
" list-actions-toggled": checkedObjectsCount > 0
|
" list-actions-toggled": checkedObjects.length > 0
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<span className="la-label">
|
<span className="la-label">
|
||||||
<i className="fa fa-check-circle" /> {checkedObjectsCount} Objects
|
<i className="fa fa-check-circle" /> {checkedObjects.length}
|
||||||
|
{checkedObjects.length === 1 ? " Object " : " Objects "}
|
||||||
selected
|
selected
|
||||||
</span>
|
</span>
|
||||||
<span className="la-actions pull-right">
|
<span className="la-actions pull-right">
|
||||||
<button id="download-checked" onClick={downloadChecked}>
|
<button
|
||||||
|
id="download-checked"
|
||||||
|
onClick={this.handleDownload.bind(this)}
|
||||||
|
>
|
||||||
{" "}
|
{" "}
|
||||||
Download all as zip{" "}
|
Download {checkedObjects.length === 1 ? "object" : "all as zip"}{" "}
|
||||||
</button>
|
</button>
|
||||||
</span>
|
</span>
|
||||||
<span className="la-actions pull-right">
|
<span className="la-actions pull-right">
|
||||||
@ -86,13 +99,15 @@ export class ObjectsBulkActions extends React.Component {
|
|||||||
|
|
||||||
const mapStateToProps = state => {
|
const mapStateToProps = state => {
|
||||||
return {
|
return {
|
||||||
checkedObjectsCount: getCheckedList(state).length
|
checkedObjects: getCheckedList(state)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const mapDispatchToProps = dispatch => {
|
const mapDispatchToProps = dispatch => {
|
||||||
return {
|
return {
|
||||||
downloadChecked: () => dispatch(actions.downloadCheckedObjects()),
|
downloadChecked: () => dispatch(actions.downloadCheckedObjects()),
|
||||||
|
downloadObject: object => dispatch(actions.downloadObject(object)),
|
||||||
|
resetCheckedList: () => dispatch(actions.resetCheckedList()),
|
||||||
clearChecked: () => dispatch(actions.resetCheckedList()),
|
clearChecked: () => dispatch(actions.resetCheckedList()),
|
||||||
deleteChecked: () => dispatch(actions.deleteCheckedObjects())
|
deleteChecked: () => dispatch(actions.deleteCheckedObjects())
|
||||||
}
|
}
|
||||||
|
@ -28,33 +28,22 @@ describe("ObjectItem", () => {
|
|||||||
expect(wrapper.prop("data-type")).toBe("image")
|
expect(wrapper.prop("data-type")).toBe("image")
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should call onClick when the object isclicked", () => {
|
it("should call checkObject when the object is selected", () => {
|
||||||
const onClick = jest.fn()
|
|
||||||
const wrapper = shallow(<ObjectItem name={"test"} onClick={onClick} />)
|
|
||||||
wrapper.find("a").simulate("click", { preventDefault: jest.fn() })
|
|
||||||
expect(onClick).toHaveBeenCalled()
|
|
||||||
})
|
|
||||||
|
|
||||||
it("should call checkObject when the object/prefix is checked", () => {
|
|
||||||
const checkObject = jest.fn()
|
const checkObject = jest.fn()
|
||||||
const wrapper = shallow(
|
const wrapper = shallow(<ObjectItem name={"test"} checked={false} checkObject={checkObject} />)
|
||||||
<ObjectItem name={"test"} checked={false} checkObject={checkObject} />
|
wrapper.find("a").simulate("click", { preventDefault: jest.fn() })
|
||||||
)
|
|
||||||
wrapper.find("input[type='checkbox']").simulate("change")
|
|
||||||
expect(checkObject).toHaveBeenCalledWith("test")
|
expect(checkObject).toHaveBeenCalledWith("test")
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should render checked checkbox", () => {
|
it("should render highlighted row when object is selected", () => {
|
||||||
const wrapper = shallow(<ObjectItem name={"test"} checked={true} />)
|
const wrapper = shallow(<ObjectItem name={"test"} checked={true} />)
|
||||||
expect(wrapper.find("input[type='checkbox']").prop("checked")).toBeTruthy()
|
expect(wrapper.find(".fesl-row").hasClass("fesl-row-selected")).toBeTruthy()
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should call uncheckObject when the object/prefix is unchecked", () => {
|
it("should call uncheckObject when the object is deselected", () => {
|
||||||
const uncheckObject = jest.fn()
|
const uncheckObject = jest.fn()
|
||||||
const wrapper = shallow(
|
const wrapper = shallow(<ObjectItem name={"test"} checked={true} uncheckObject={uncheckObject} />)
|
||||||
<ObjectItem name={"test"} checked={true} uncheckObject={uncheckObject} />
|
wrapper.find("a").simulate("click", { preventDefault: jest.fn() })
|
||||||
)
|
|
||||||
wrapper.find("input[type='checkbox']").simulate("change")
|
|
||||||
expect(uncheckObject).toHaveBeenCalledWith("test")
|
expect(uncheckObject).toHaveBeenCalledWith("test")
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
@ -20,19 +20,33 @@ import { ObjectsBulkActions } from "../ObjectsBulkActions"
|
|||||||
|
|
||||||
describe("ObjectsBulkActions", () => {
|
describe("ObjectsBulkActions", () => {
|
||||||
it("should render without crashing", () => {
|
it("should render without crashing", () => {
|
||||||
shallow(<ObjectsBulkActions checkedObjectsCount={0} />)
|
shallow(<ObjectsBulkActions checkedObjects={0} />)
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should show actions when checkObjectsCount is more than 0", () => {
|
it("should show actions when checkObjectsCount is more than 0", () => {
|
||||||
const wrapper = shallow(<ObjectsBulkActions checkedObjectsCount={1} />)
|
const wrapper = shallow(<ObjectsBulkActions checkedObjects={["test"]} />)
|
||||||
expect(wrapper.hasClass("list-actions-toggled")).toBeTruthy()
|
expect(wrapper.hasClass("list-actions-toggled")).toBeTruthy()
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should call downloadChecked when download button is clicked", () => {
|
it("should call downloadObject for single object when download button is clicked", () => {
|
||||||
|
const downloadObject = jest.fn()
|
||||||
|
const resetCheckedList = jest.fn()
|
||||||
|
const wrapper = shallow(
|
||||||
|
<ObjectsBulkActions
|
||||||
|
checkedObjects={["test1"]}
|
||||||
|
downloadObject={downloadObject}
|
||||||
|
resetCheckedList={resetCheckedList}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
wrapper.find("#download-checked").simulate("click")
|
||||||
|
expect(downloadObject).toHaveBeenCalled()
|
||||||
|
})
|
||||||
|
|
||||||
|
it("should call downloadChecked for multiple objects when download button is clicked", () => {
|
||||||
const downloadChecked = jest.fn()
|
const downloadChecked = jest.fn()
|
||||||
const wrapper = shallow(
|
const wrapper = shallow(
|
||||||
<ObjectsBulkActions
|
<ObjectsBulkActions
|
||||||
checkedObjectsCount={1}
|
checkedObjects={["test1", "test2"]}
|
||||||
downloadChecked={downloadChecked}
|
downloadChecked={downloadChecked}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
@ -43,14 +57,14 @@ describe("ObjectsBulkActions", () => {
|
|||||||
it("should call clearChecked when close button is clicked", () => {
|
it("should call clearChecked when close button is clicked", () => {
|
||||||
const clearChecked = jest.fn()
|
const clearChecked = jest.fn()
|
||||||
const wrapper = shallow(
|
const wrapper = shallow(
|
||||||
<ObjectsBulkActions checkedObjectsCount={1} clearChecked={clearChecked} />
|
<ObjectsBulkActions checkedObjects={["test"]} clearChecked={clearChecked} />
|
||||||
)
|
)
|
||||||
wrapper.find("#close-bulk-actions").simulate("click")
|
wrapper.find("#close-bulk-actions").simulate("click")
|
||||||
expect(clearChecked).toHaveBeenCalled()
|
expect(clearChecked).toHaveBeenCalled()
|
||||||
})
|
})
|
||||||
|
|
||||||
it("shoud show DeleteObjectConfirmModal when delete-checked button is clicked", () => {
|
it("shoud show DeleteObjectConfirmModal when delete-checked button is clicked", () => {
|
||||||
const wrapper = shallow(<ObjectsBulkActions checkedObjectsCount={1} />)
|
const wrapper = shallow(<ObjectsBulkActions checkedObjects={["test"]} />)
|
||||||
wrapper.find("#delete-checked").simulate("click")
|
wrapper.find("#delete-checked").simulate("click")
|
||||||
wrapper.update()
|
wrapper.update()
|
||||||
expect(wrapper.find("DeleteObjectConfirmModal").length).toBe(1)
|
expect(wrapper.find("DeleteObjectConfirmModal").length).toBe(1)
|
||||||
@ -60,7 +74,7 @@ describe("ObjectsBulkActions", () => {
|
|||||||
const deleteChecked = jest.fn()
|
const deleteChecked = jest.fn()
|
||||||
const wrapper = shallow(
|
const wrapper = shallow(
|
||||||
<ObjectsBulkActions
|
<ObjectsBulkActions
|
||||||
checkedObjectsCount={1}
|
checkedObjects={["test"]}
|
||||||
deleteChecked={deleteChecked}
|
deleteChecked={deleteChecked}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
|
@ -74,7 +74,7 @@ div.fesl-row {
|
|||||||
border-bottom: 1px solid transparent;
|
border-bottom: 1px solid transparent;
|
||||||
cursor: default;
|
cursor: default;
|
||||||
.transition(background-color);
|
.transition(background-color);
|
||||||
.transition-duration(500ms);
|
.transition-duration(300ms);
|
||||||
|
|
||||||
@media (max-width: (@screen-xs-max - 100px)) {
|
@media (max-width: (@screen-xs-max - 100px)) {
|
||||||
padding: 5px 20px;
|
padding: 5px 20px;
|
||||||
@ -87,16 +87,8 @@ div.fesl-row {
|
|||||||
}
|
}
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
.fis-icon {
|
&:not(.fesl-row-selected) {
|
||||||
&:before {
|
background: lighten(@text-muted-color, 22%);
|
||||||
.opacity(0)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.fis-helper {
|
|
||||||
&:before {
|
|
||||||
.opacity(1);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -137,6 +129,14 @@ div.fesl-row {
|
|||||||
.fesl-row-selected {
|
.fesl-row-selected {
|
||||||
background-color: @list-row-selected-bg;
|
background-color: @list-row-selected-bg;
|
||||||
|
|
||||||
|
&:nth-child(even) {
|
||||||
|
background-color: @list-row-selected-even-bg;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background: @list-row-selected-hover;
|
||||||
|
}
|
||||||
|
|
||||||
&, .fesl-item a {
|
&, .fesl-item a {
|
||||||
color: darken(@text-color, 10%);
|
color: darken(@text-color, 10%);
|
||||||
}
|
}
|
||||||
@ -161,27 +161,6 @@ div.fesl-row {
|
|||||||
height: 35px;
|
height: 35px;
|
||||||
z-index: 8;
|
z-index: 8;
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
cursor: pointer;
|
|
||||||
|
|
||||||
&:checked {
|
|
||||||
& ~ .fis-icon {
|
|
||||||
background-color: #32393F;
|
|
||||||
|
|
||||||
&:before {
|
|
||||||
opacity: 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
& ~ .fis-helper {
|
|
||||||
&:before {
|
|
||||||
.scale(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
&:after {
|
|
||||||
.scale(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -210,38 +189,6 @@ div.fesl-row {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.fis-helper {
|
|
||||||
&:before,
|
|
||||||
&:after {
|
|
||||||
position: absolute;
|
|
||||||
.transition(all);
|
|
||||||
.transition-duration(250ms);
|
|
||||||
}
|
|
||||||
|
|
||||||
&:before {
|
|
||||||
content: '';
|
|
||||||
width: 15px;
|
|
||||||
height: 15px;
|
|
||||||
border: 2px solid @white;
|
|
||||||
z-index: 7;
|
|
||||||
border-radius: 2px;
|
|
||||||
top: 10px;
|
|
||||||
left: 10px;
|
|
||||||
opacity: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
&:after {
|
|
||||||
font-family: @font-family-icon;
|
|
||||||
content: '\f00c';
|
|
||||||
top: 8px;
|
|
||||||
left: 9px;
|
|
||||||
color: @white;
|
|
||||||
font-size: 14px;
|
|
||||||
.scale(0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*--------------------------
|
/*--------------------------
|
||||||
Files and Folders
|
Files and Folders
|
||||||
----------------------------*/
|
----------------------------*/
|
||||||
|
@ -97,5 +97,7 @@
|
|||||||
/*-------------------------
|
/*-------------------------
|
||||||
List
|
List
|
||||||
--------------------------*/
|
--------------------------*/
|
||||||
@list-row-selected-bg: #fbf2bf;
|
@list-row-selected-bg: #fffad6;
|
||||||
|
@list-row-selected-even-bg: #faf5d1;
|
||||||
|
@list-row-selected-hover: #f5f0cc;
|
||||||
@list-row-even-bg: #fafafa;
|
@list-row-even-bg: #fafafa;
|
File diff suppressed because one or more lines are too long
Loading…
x
Reference in New Issue
Block a user