diff --git a/src/components/BrowserCell/BrowserCell.react.js b/src/components/BrowserCell/BrowserCell.react.js
index aa134b58c3..f6d141fb50 100644
--- a/src/components/BrowserCell/BrowserCell.react.js
+++ b/src/components/BrowserCell/BrowserCell.react.js
@@ -127,8 +127,8 @@ export default class BrowserCell extends Component {
} else if (this.props.type === 'Object' || this.props.type === 'Bytes') {
this.copyableValue = content = JSON.stringify(this.props.value);
} else if (this.props.type === 'File') {
- const fileName = this.props.value.url() ? getFileName(this.props.value) : 'Uploading\u2026';
- content = ;
+ const fileName = this.props.value.url?.() ? getFileName(this.props.value) : 'Uploading\u2026';
+ content = ;
this.copyableValue = fileName;
} else if (this.props.type === 'ACL') {
let pieces = [];
diff --git a/src/components/FileEditor/FileEditor.react.js b/src/components/FileEditor/FileEditor.react.js
index abc87b654b..b2adc35fd6 100644
--- a/src/components/FileEditor/FileEditor.react.js
+++ b/src/components/FileEditor/FileEditor.react.js
@@ -80,7 +80,7 @@ export default class FileEditor extends React.Component {
return (
diff --git a/src/components/Modal/Modal.scss b/src/components/Modal/Modal.scss
index 171c32e4cf..ad34f3123b 100644
--- a/src/components/Modal/Modal.scss
+++ b/src/components/Modal/Modal.scss
@@ -43,7 +43,7 @@
position: absolute;
font-size: 14px;
color: white;
- top: 52px;
+ top: 56px;
left: 28px;
}
diff --git a/src/dashboard/Data/Browser/Browser.react.js b/src/dashboard/Data/Browser/Browser.react.js
index 7a0f65472a..a69f2a1397 100644
--- a/src/dashboard/Data/Browser/Browser.react.js
+++ b/src/dashboard/Data/Browser/Browser.react.js
@@ -16,6 +16,7 @@ import DeleteRowsDialog from 'dashboard/Data/Browser/DeleteRow
import DropClassDialog from 'dashboard/Data/Browser/DropClassDialog.react';
import EmptyState from 'components/EmptyState/EmptyState.react';
import ExportDialog from 'dashboard/Data/Browser/ExportDialog.react';
+import ImportDialog from 'dashboard/Data/Browser/ImportDialog.react';
import AttachRowsDialog from 'dashboard/Data/Browser/AttachRowsDialog.react';
import AttachSelectedRowsDialog from 'dashboard/Data/Browser/AttachSelectedRowsDialog.react';
import CloneSelectedRowsDialog from 'dashboard/Data/Browser/CloneSelectedRowsDialog.react';
@@ -57,6 +58,7 @@ class Browser extends DashboardView {
showRemoveColumnDialog: false,
showDropClassDialog: false,
showExportDialog: false,
+ showImportDialog: false,
showExportSchemaDialog: false,
showAttachRowsDialog: false,
showEditRowDialog: false,
@@ -105,6 +107,7 @@ class Browser extends DashboardView {
this.showDeleteRows = this.showDeleteRows.bind(this);
this.showDropClass = this.showDropClass.bind(this);
this.showExport = this.showExport.bind(this);
+ this.showImport = this.showImport.bind(this);
this.login = this.login.bind(this);
this.logout = this.logout.bind(this);
this.toggleMasterKeyUsage = this.toggleMasterKeyUsage.bind(this);
@@ -282,6 +285,10 @@ class Browser extends DashboardView {
this.setState({ showExportDialog: true });
}
+ showImport() {
+ this.setState({ showImportDialog: true });
+ }
+
async login(username, password) {
if (Parse.User.current()) {
await Parse.User.logOut();
@@ -1125,6 +1132,7 @@ class Browser extends DashboardView {
this.state.showRemoveColumnDialog ||
this.state.showDropClassDialog ||
this.state.showExportDialog ||
+ this.state.showImportDialog ||
this.state.showExportSchema ||
this.state.rowsToDelete ||
this.state.showAttachRowsDialog ||
@@ -1445,6 +1453,89 @@ class Browser extends DashboardView {
}
}
+ async confirmImport(file) {
+ this.setState({ showImportDialog: null });
+ const className = this.props.params.className;
+ const classColumns = this.getClassColumns(className, false);
+ const columnsObject = {};
+ classColumns.forEach((column) => {
+ columnsObject[column.name] = column;
+ });
+ const { base64, type} = file._source;
+ if (type === 'text/csv') {
+ const csvToArray =(text) => {
+ let p = '', row = [''], ret = [row], i = 0, r = 0, s = !0, l;
+ for (l of text) {
+ if ('"' === l) {
+ if (s && l === p) row[i] += l;
+ s = !s;
+ } else if (',' === l && s) l = row[++i] = '';
+ else if ('\n' === l && s) {
+ if ('\r' === p) row[i] = row[i].slice(0, -1);
+ row = ret[++r] = [l = '']; i = 0;
+ } else row[i] += l;
+ p = l;
+ }
+ return ret;
+ };
+ const csv = atob(base64);
+ const [columns, ...rows] = csvToArray(csv);
+ await Parse.Object.saveAll(rows.filter(row => row.join() !== '').map(row => {
+ const json = {className};
+ for (let i = 1; i < row.length; i++) {
+ const column = columns[i];
+ const value = row[i];
+ if (value === 'null') {
+ continue;
+ }
+ const {type, targetClass, name} = columnsObject[column] || {};
+ if (type === 'Relation') {
+ json[column] = {
+ __type: 'Relation',
+ className: targetClass,
+ };
+ continue;
+ }
+ if (type === 'Pointer') {
+ json[column] = {
+ __type: 'Pointer',
+ className: targetClass,
+ objectId: value,
+ };
+ continue;
+ }
+ if (name === 'ACL') {
+ json.ACL = new Parse.ACL(JSON.parse(value));
+ continue;
+ }
+ if (type === 'Date') {
+ json[column] = new Date(value);
+ continue;
+ }
+ if (type === 'Boolean') {
+ json[column] = value === 'true';
+ continue;
+ }
+ if (type === 'String') {
+ json[column] = value;
+ continue;
+ }
+ if (type === 'Number') {
+ json[column] = Number(value);
+ continue;
+ }
+ try {
+ json[column] = JSON.parse(value);
+ } catch (e) {
+ /* */
+ }
+ }
+ return Parse.Object.fromJSON(json, false, true);
+ }), {useMasterKey: true});
+ }
+ this.refresh();
+ }
+
getClassRelationColumns(className) {
const currentClassName = this.props.params.className;
return this.getClassColumns(className, false)
@@ -1640,6 +1731,7 @@ class Browser extends DashboardView {
onExport={this.showExport}
onChangeCLP={this.handleCLPChange}
onRefresh={this.refresh}
+ onImport={this.showImport}
onAttachRows={this.showAttachRowsDialog}
onAttachSelectedRows={this.showAttachSelectedRowsDialog}
onCloneSelectedRows={this.showCloneSelectedRowsDialog}
@@ -1761,6 +1853,14 @@ class Browser extends DashboardView {
onCancel={() => this.setState({ showExportDialog: false })}
onConfirm={() => this.exportClass(className)} />
);
+ }
+ else if (this.state.showImportDialog) {
+ extras = (
+ this.setState({ showImportDialog: false })}
+ onConfirm={(file) => this.confirmImport(file)} />
+ );
} else if (this.state.showExportSchemaDialog) {
extras = (
- )
+ );
} else if (this.state.showAttachSelectedRowsDialog) {
extras = (
)}
{onAddRow && }
+
+
+ Import
+
+
Refresh
diff --git a/src/dashboard/Data/Browser/ImportDialog.react.js b/src/dashboard/Data/Browser/ImportDialog.react.js
new file mode 100644
index 0000000000..40a3199052
--- /dev/null
+++ b/src/dashboard/Data/Browser/ImportDialog.react.js
@@ -0,0 +1,66 @@
+import Modal from 'components/Modal/Modal.react';
+import FileEditor from 'components/FileEditor/FileEditor.react';
+import React from 'react';
+import Pill from 'components/Pill/Pill.react';
+import getFileName from 'lib/getFileName';
+import { CurrentApp } from 'context/currentApp';
+
+export default class ImportDialog extends React.Component {
+ constructor() {
+ super();
+ this.state = {
+ file: null,
+ showFileEditor: false
+ };
+ }
+
+ openFileEditor() {
+ this.setState({
+ showFileEditor: true
+ });
+ }
+
+ hideFileEditor(file) {
+ this.setState({
+ showFileEditor: false,
+ file
+ });
+ }
+
+ render() {
+ return (
+
+
this.props.onConfirm(this.state.file)}>
+
+ {this.state.file &&
}
+
+
this.openFileEditor()}
+ />
+ {this.state.showFileEditor && (
+ this.hideFileEditor(file)}
+ onCancel={() => this.hideFileEditor()}
+ />
+ )}
+
+
+
+
+ );
+ }
+}