summaryrefslogtreecommitdiff
path: root/client
diff options
context:
space:
mode:
Diffstat (limited to 'client')
-rw-r--r--client/actions/index.js17
-rw-r--r--client/components/Browser/BrowserView.jsx17
-rw-r--r--client/components/Browser/Files/FileListView.jsx68
-rw-r--r--client/components/Browser/Files/FileUploadButton.jsx1
-rw-r--r--client/components/Browser/Files/Files.jsx81
-rw-r--r--client/components/Browser/Folders/FolderForm.jsx47
-rw-r--r--client/components/Browser/Folders/FolderListView.jsx24
-rw-r--r--client/components/Browser/Folders/Folders.jsx56
-rw-r--r--client/components/Browser/Folders/NewFolderView.jsx24
-rw-r--r--client/components/Tasks/Tasks.jsx30
-rw-r--r--client/containers/browser.js12
-rw-r--r--client/containers/fileList.js21
-rw-r--r--client/containers/folderList.js23
-rw-r--r--client/containers/newFolder.js29
-rw-r--r--client/reducers/folders.js25
-rw-r--r--client/reducers/index.js2
-rw-r--r--client/reducers/newFolder.js32
-rw-r--r--client/store.js3
18 files changed, 284 insertions, 228 deletions
diff --git a/client/actions/index.js b/client/actions/index.js
index 5b180ea..94d64c8 100644
--- a/client/actions/index.js
+++ b/client/actions/index.js
@@ -55,14 +55,29 @@ export const loadOpenFolders = (folders) => ({
type: 'LOAD_OPEN_FOLDERS',
folders,
})
-export const openFolder = (folder) => ({
+
+export const initNewFolder = () => ({
+ type: 'INIT_NEW_FOLDER'
+})
+export const cancelNewFolder = () => ({
+ type: 'CANCEL_NEW_FOLDER'
+})
+export const createNewFolder = (name, cb) => ({
+ type: 'CREATE_NEW_FOLDER',
+ name,
+ cb
+})
+
+export const openFolder = (folder, cb) => ({
type: 'OPEN_FOLDER',
folder,
+ cb
})
export const closeFolder = (folder) => ({
type: 'CLOSE_FOLDER',
folder,
})
+
export const addFolder = (folder) => ({
type: 'ADD_FOLDER',
folder,
diff --git a/client/components/Browser/BrowserView.jsx b/client/components/Browser/BrowserView.jsx
index 604d952..e510fc7 100644
--- a/client/components/Browser/BrowserView.jsx
+++ b/client/components/Browser/BrowserView.jsx
@@ -1,7 +1,8 @@
import { h, Component } from 'preact'
-import Folders from './Folders/Folders.jsx'
-import Files from './Files/Files.jsx'
+import NewFolder from '../../containers/newFolder.js'
+import FolderList from '../../containers/folderList.js'
+import FileList from '../../containers/fileList.js'
export default function BrowserView (props) {
const openFolders = (props.openFolders || []).map((folder_id) => {
@@ -9,20 +10,14 @@ export default function BrowserView (props) {
if (! folder_list.length) return
const folder = folder_list[0]
return (
- <Files
- folder={folder}
- addFiles={props.addFiles}
- onClose={() => props.closeFolder(folder)}
- />
+ <FileList folder={folder} />
)
})
return (
<div class='column'>
- <Folders folders={props.folders}
- openFolder={(folder) => props.openFolder(folder)}
- addFolder={(folder) => props.addFolder(folder)}
- />
+ <NewFolder />
+ <FolderList />
{openFolders}
</div>
)
diff --git a/client/components/Browser/Files/FileListView.jsx b/client/components/Browser/Files/FileListView.jsx
new file mode 100644
index 0000000..1c5f952
--- /dev/null
+++ b/client/components/Browser/Files/FileListView.jsx
@@ -0,0 +1,68 @@
+import { h, Component } from 'preact'
+
+import FileUploadButton from './FileUploadButton.jsx'
+import FileLink from '../../../containers/fileLink.js'
+import TaskContentLink from '../../../containers/taskContentLink.js'
+import TaskStyleLink from '../../../containers/taskStyleLink.js'
+
+export default function FileListView (props) {
+ console.log(props)
+
+ let file_list;
+ if (props.folder && props.folder.files) {
+ file_list = props.folder.files
+ }
+ else {
+ file_list = []
+ }
+ const files = file_list.map(toFilenamePair)
+ .sort(sortByFilename)
+ .map(fromPair)
+ .map( (file, i) => {
+ if (! file) return
+ return (
+ <div key={i} class={props.selected === file ? 'selected' : ''}>
+ <span class='name'><FileLink file={file}>{file.name}</FileLink></span>
+ <span class='mime'>{file.processed ? file.mime : 'working...'}</span>
+ <span class='duration'>{file.duration ? (file.duration.toFixed(1) + 's') : ''}</span>
+ <span class='actions'>
+ <TaskContentLink file={file}>content</TaskContentLink>
+ <TaskStyleLink file={file}>style</TaskStyleLink>
+ </span>
+ </div>
+ )
+ })
+ return (
+ <div class='window'>
+ <div class='heading'>
+ <b>{props.folder.name}</b>
+ <div class='buttons'>
+ <FileUploadButton folder={props.folder} addFiles={props.addFiles} />
+ <button onClick={props.onClose}>&times;</button>
+ </div>
+ </div>
+ <div class='list'>
+ {files}
+ </div>
+ </div>
+ )
+
+}
+
+
+function toFilenamePair (file) { return [file.name.toLowerCase(), file] }
+function sortByFilename (a,b) { return a[0] < b[0] ? -1 : a[0] == b[0] ? 0 : 1 }
+function fromPair (pair) { return pair[1] }
+
+function filepath (file) {
+ return '/data/' + file.folder_id + '/' + encodeURIComponent(file.name)
+}
+function mp3path (file) {
+ if (file.mime !== 'audio/mp3') {
+ return filepath(file) + '.mp3'
+ }
+ return filepath(file)
+}
+function pngpath (file) {
+ return filepath(file) + '.png'
+} \ No newline at end of file
diff --git a/client/components/Browser/Files/FileUploadButton.jsx b/client/components/Browser/Files/FileUploadButton.jsx
index 2d25d3e..8a1bdda 100644
--- a/client/components/Browser/Files/FileUploadButton.jsx
+++ b/client/components/Browser/Files/FileUploadButton.jsx
@@ -15,7 +15,6 @@ export default class FileUploadButton extends Component {
this.props.addFiles(got_files)
})
}
-
render() {
return (
<div class='fileUploadButton'>
diff --git a/client/components/Browser/Files/Files.jsx b/client/components/Browser/Files/Files.jsx
deleted file mode 100644
index c4df74d..0000000
--- a/client/components/Browser/Files/Files.jsx
+++ /dev/null
@@ -1,81 +0,0 @@
-import { h, Component } from 'preact'
-
-import { audioPlayFile } from '../../../actions'
-
-import client from '../../../client.js'
-
-import FileUploadButton from './FileUploadButton.jsx'
-import FileLink from '../../../containers/fileLink.js'
-import TaskContentLink from '../../../containers/taskContentLink.js'
-import TaskStyleLink from '../../../containers/taskStyleLink.js'
-
-export default class Files extends Component {
- constructor(props) {
- super()
- this.state = {
- selected: null,
- }
- this.addFiles = this.addFiles.bind(this)
- }
- addFiles(newFiles) {
- if (! newFiles) return
- this.props.addFiles( newFiles )
- }
- handleClick(file) {
- this.setState({ selected: file })
- }
- render() {
- let file_list;
- if (this.props.folder && this.props.folder.files) {
- file_list = this.props.folder.files
- }
- else {
- file_list = []
- }
- const files = file_list.map(toFilenamePair).sort(sortByFilename).map(fromPair).map( (file, i) => {
- if (! file) return
- return (
- <div key={i} class={this.state.selected === file ? 'selected' : ''}>
- <span class='name'><FileLink file={file}>{file.name}</FileLink></span>
- <span class='mime'>{file.processed ? file.mime : 'working...'}</span>
- <span class='duration'>{file.duration ? (file.duration.toFixed(1) + 's') : ''}</span>
- <span class='actions'>
- <TaskContentLink file={file}>content</TaskContentLink>
- <TaskStyleLink file={file}>style</TaskStyleLink>
- </span>
- </div>
- )
- })
- return (
- <div class='window'>
- <div class='heading'>
- <b>{this.props.folder.name}</b>
- <div class='buttons'>
- <FileUploadButton folder={this.props.folder} addFiles={this.addFiles} />
- <button onClick={this.props.onClose}>&times;</button>
- </div>
- </div>
- <div class='list'>
- {files}
- </div>
- </div>
- )
- }
-}
-
-function toFilenamePair (file) { return [file.name.toLowerCase(), file] }
-function sortByFilename (a,b) { return a[0] < b[0] ? -1 : a[0] == b[0] ? 0 : 1 }
-function fromPair (pair) { return pair[1] }
-
-function filepath (file) {
- return '/data/' + file.folder_id + '/' + encodeURIComponent(file.name)
-}
-function mp3path (file) {
- if (file.mime !== 'audio/mp3') {
- return filepath(file) + '.mp3'
- }
- return filepath(file)
-}
-function pngpath (file) {
- return filepath(file) + '.png'
-} \ No newline at end of file
diff --git a/client/components/Browser/Folders/FolderForm.jsx b/client/components/Browser/Folders/FolderForm.jsx
deleted file mode 100644
index ad3e321..0000000
--- a/client/components/Browser/Folders/FolderForm.jsx
+++ /dev/null
@@ -1,47 +0,0 @@
-import { h, Component } from 'preact'
-
-import client from '../../../client.js'
-
-export default class FolderForm extends Component {
- constructor(props) {
- super()
- this.state = {
- name: '',
- }
- this.updateState = this.updateState.bind(this)
- this.handleSubmit = this.handleSubmit.bind(this)
- }
- updateState(event){
- const name = event.target.name
- let value = event.target.value
- console.log(name, value)
- this.setState({
- [name]: value,
- error: null,
- })
- }
- handleSubmit(event) {
- event.preventDefault()
- let rec = Object.assign({}, this.state)
- delete rec.error
- this.props.onClose()
- client.folder.create( rec ).then( (data) => {
- this.props.addFolder( data )
- })
- }
- render() {
- return (
- <div class='form'>
- <h1>new folder</h1>
- <div>
- <label for='folders_folder_name'>Name</label>
- <input type='text' id='folders_folder_name' name='name' value={this.state.name} onChange={this.updateState} />
- </div>
- <div>
- <label></label>
- <button onClick={this.handleSubmit}>Create</button>
- </div>
- </div>
- )
- }
-}
diff --git a/client/components/Browser/Folders/FolderListView.jsx b/client/components/Browser/Folders/FolderListView.jsx
new file mode 100644
index 0000000..1ee76d8
--- /dev/null
+++ b/client/components/Browser/Folders/FolderListView.jsx
@@ -0,0 +1,24 @@
+import { h, Component } from 'preact'
+
+export default function FolderListView (props) {
+ console.log(props)
+ const folders = props.folders.map( (folder, i) => (
+ <div key={i} onClick={() => props.openFolder(folder)}>
+ <span class='name'>{folder.name}</span>
+ </div>
+ ))
+
+ return (
+ <div class='window'>
+ <div class='heading'>
+ <b>browser</b>
+ <div class='buttons'>
+ <button onClick={props.initNewFolder}>+ folder</button>
+ </div>
+ </div>
+ <div class='list'>
+ {folders}
+ </div>
+ </div>
+ )
+}
diff --git a/client/components/Browser/Folders/Folders.jsx b/client/components/Browser/Folders/Folders.jsx
deleted file mode 100644
index 2c85e80..0000000
--- a/client/components/Browser/Folders/Folders.jsx
+++ /dev/null
@@ -1,56 +0,0 @@
-import { h, Component } from 'preact'
-
-import Modal from '../../UI/Modal.jsx'
-import FolderForm from './FolderForm.jsx'
-
-import client from '../../../client.js'
-
-export default class Folders extends Component {
- constructor(props) {
- super()
- this.state = {
- adding: false,
- }
- }
- openModal() {
- this.setState({ adding: true })
- }
- closeModal() {
- this.setState({ adding: false })
- }
- toggle(folder) {
- folder.open = ! folder.open
- if (folder.open && ! folder.files) {
- client.file.index({ 'folder_id': folder.id }).then( files => {
- console.log(files)
- folder.files = files
- this.props.openFolder(folder)
- })
- }
- }
- render() {
- const folders = (this.props.folders || []).map( (folder,i) => {
- return (
- <div key={i} onClick={() => this.toggle(folder)}>
- <span class='name'>{folder.name}</span>
- </div>
- )
- })
- return (
- <div class='window'>
- <div class='heading'>
- <b>browser</b>
- <div class='buttons'>
- <button onClick={() => this.openModal()}>+ folder</button>
- </div>
- </div>
- <div class='list'>
- {folders}
- </div>
- <Modal visible={this.state.adding} onClose={() => this.closeModal()}>
- <FolderForm addFolder={this.props.addFolder} onClose={() => this.closeModal()} />
- </Modal>
- </div>
- )
- }
-}
diff --git a/client/components/Browser/Folders/NewFolderView.jsx b/client/components/Browser/Folders/NewFolderView.jsx
new file mode 100644
index 0000000..7573450
--- /dev/null
+++ b/client/components/Browser/Folders/NewFolderView.jsx
@@ -0,0 +1,24 @@
+import { h, Component } from 'preact'
+
+export default function NewFolderView (props) {
+ console.log(props)
+ if (! props.visible) {
+ return null
+ }
+ return (
+ <div class='window'>
+ <div class='heading'>
+ <b>new folder</b>
+ <div class='buttons'>
+ <button onClick={props.onClose}>x</button>
+ </div>
+ </div>
+ <input type='text'
+ autofocus
+ placeholder='Enter a name'
+ value={props.name}
+ onChange={props.onChange}
+ />
+ </div>
+ )
+}
diff --git a/client/components/Tasks/Tasks.jsx b/client/components/Tasks/Tasks.jsx
index 79ee0a1..c76e61d 100644
--- a/client/components/Tasks/Tasks.jsx
+++ b/client/components/Tasks/Tasks.jsx
@@ -4,23 +4,21 @@ import { Link } from 'react-router-dom'
import TaskForm from '../../containers/taskForm.js'
import TaskList from '../../containers/taskList.js'
-export default class Tasks extends Component {
- render() {
- return (
- <div class='column'>
- <div class='window'>
- <div class='heading'>
- <b>create task</b>
- </div>
- <TaskForm />
+export default function Tasks (props) {
+ return (
+ <div class='column'>
+ <div class='window'>
+ <div class='heading'>
+ <b>create task</b>
</div>
- <div class='window'>
- <div class='heading'>
- <b>recent tasks</b>
- </div>
- <TaskList />
+ <TaskForm />
+ </div>
+ <div class='window'>
+ <div class='heading'>
+ <b>recent tasks</b>
</div>
+ <TaskList />
</div>
- )
- }
+ </div>
+ )
}
diff --git a/client/containers/browser.js b/client/containers/browser.js
index 74c0e8d..e92dbbe 100644
--- a/client/containers/browser.js
+++ b/client/containers/browser.js
@@ -1,5 +1,5 @@
import { connect } from 'react-redux'
-import { addFile, addFiles, addFolder, openFolder, closeFolder } from '../actions'
+import { addFile, addFiles } from '../actions'
import BrowserView from '../components/Browser/BrowserView.jsx'
const mapStateToProps = state => state.folders
@@ -11,16 +11,6 @@ const mapDispatchToProps = (dispatch, ownProps) => ({
addFiles: (files) => {
dispatch(addFiles(files))
},
- addFolder: (folder) => {
- dispatch(addFolder(folder))
- },
- openFolder: (folder) => {
- dispatch(openFolder(folder))
- },
- closeFolder: (folder) => {
- console.log(folder)
- dispatch(closeFolder(folder))
- }
})
const Browser = connect(
diff --git a/client/containers/fileList.js b/client/containers/fileList.js
new file mode 100644
index 0000000..3e6554d
--- /dev/null
+++ b/client/containers/fileList.js
@@ -0,0 +1,21 @@
+import { connect } from 'react-redux'
+import { addFiles, closeFolder } from '../actions'
+import FolderListView from '../components/Browser/Files/FileListView.jsx'
+
+const mapStateToProps = (state) => (state.folders)
+
+const mapDispatchToProps = (dispatch, ownProps) => ({
+ addFiles: (files) => {
+ dispatch(addFiles(files))
+ },
+ onClose: () => {
+ dispatch(closeFolder(ownProps.folder))
+ },
+})
+
+const FolderList = connect(
+ mapStateToProps,
+ mapDispatchToProps
+)(FolderListView)
+
+export default FolderList
diff --git a/client/containers/folderList.js b/client/containers/folderList.js
new file mode 100644
index 0000000..6c58ee5
--- /dev/null
+++ b/client/containers/folderList.js
@@ -0,0 +1,23 @@
+import { connect } from 'react-redux'
+import { initNewFolder, openFolder, addFiles } from '../actions'
+import FolderListView from '../components/Browser/Folders/FolderListView.jsx'
+
+const mapStateToProps = (state) => (state.folders)
+
+const mapDispatchToProps = (dispatch, ownProps) => ({
+ initNewFolder: () => {
+ dispatch(initNewFolder())
+ },
+ openFolder: (folder) => {
+ dispatch(openFolder(folder, (files) => {
+ dispatch(addFiles(files))
+ }))
+ },
+})
+
+const FolderList = connect(
+ mapStateToProps,
+ mapDispatchToProps
+)(FolderListView)
+
+export default FolderList
diff --git a/client/containers/newFolder.js b/client/containers/newFolder.js
new file mode 100644
index 0000000..047edde
--- /dev/null
+++ b/client/containers/newFolder.js
@@ -0,0 +1,29 @@
+import { connect } from 'react-redux'
+import {
+ cancelNewFolder,
+ createNewFolder,
+ addFolder,
+} from '../actions'
+import NewFolderView from '../components/Browser/Folders/NewFolderView.jsx'
+
+const mapStateToProps = (state) => (state.newFolder)
+
+const mapDispatchToProps = (dispatch, ownProps) => ({
+ onClose: () => {
+ dispatch(cancelNewFolder())
+ },
+ onChange: (e) => {
+ const name = e.currentTarget.value
+ if (! name) return
+ dispatch(createNewFolder(name, (folder) => {
+ dispatch(addFolder(folder))
+ }))
+ }
+})
+
+const NewFolder = connect(
+ mapStateToProps,
+ mapDispatchToProps
+)(NewFolderView)
+
+export default NewFolder
diff --git a/client/reducers/folders.js b/client/reducers/folders.js
index a69df74..bec4ad3 100644
--- a/client/reducers/folders.js
+++ b/client/reducers/folders.js
@@ -2,7 +2,7 @@
import client from '../client'
const folders = (state = {}, action) => {
- let file, files, folders, folder, openFolders, folder_id
+ let file, files, folders, folder, openFolders, folder_id, filesAreLoaded
console.log(action)
switch (action.type) {
case 'LOAD_FOLDERS':
@@ -24,7 +24,14 @@ const folders = (state = {}, action) => {
})
return {
...state,
- folders: folders,
+ folders
+ }
+
+ case 'ADD_FOLDER':
+ folders = [ action.folder ].concat(state.folders)
+ return {
+ ...state,
+ folders
}
case 'LOAD_OPEN_FOLDERS':
@@ -36,10 +43,22 @@ const folders = (state = {}, action) => {
case 'OPEN_FOLDER':
openFolders = state.openFolders
folder = action.folder
+ folder_id = folder.id
if (openFolders.indexOf(folder.id) === -1) {
openFolders = openFolders.concat(folder.id)
localStorage['openFolders'] = JSON.stringify(openFolders)
}
+ filesAreLoaded = state.folders.some( (folder) => {
+ if (folder.id === folder_id && folder.files) {
+ return true
+ }
+ return false
+ })
+ if (! filesAreLoaded) {
+ client.file.index({ folder_id }).then( (files) => {
+ action.cb && action.cb(files)
+ })
+ }
return {
...state,
openFolders,
@@ -57,7 +76,7 @@ const folders = (state = {}, action) => {
case 'ADD_FILES':
files = action.files
- folder_id = Number(files[0].folder_id)
+ folder_id = Number(files[0] && files[0].folder_id)
folders = state.folders.map( (folder) => {
if (folder.id === folder_id) {
folder.files = ( folder.files || [] ).concat(files)
diff --git a/client/reducers/index.js b/client/reducers/index.js
index 8fa287b..8a6ea2f 100644
--- a/client/reducers/index.js
+++ b/client/reducers/index.js
@@ -4,12 +4,14 @@ import audioPlayer from './audioPlayer'
import currentTask from './currentTask'
import tasks from './tasks'
import folders from './folders'
+import newFolder from './newFolder'
const cortexApp = combineReducers({
audioPlayer,
currentTask,
tasks,
folders,
+ newFolder,
})
export default cortexApp
diff --git a/client/reducers/newFolder.js b/client/reducers/newFolder.js
new file mode 100644
index 0000000..5c0f40d
--- /dev/null
+++ b/client/reducers/newFolder.js
@@ -0,0 +1,32 @@
+import client from '../client'
+
+const newFolder = (state = {}, action) => {
+ switch (action.type) {
+ case 'INIT_NEW_FOLDER':
+ return {
+ name: '',
+ visible: true,
+ }
+
+ case 'CANCEL_NEW_FOLDER':
+ return {
+ name: '',
+ visible: false,
+ }
+
+ case 'CREATE_NEW_FOLDER':
+ if (action.name) {
+ client.folder.create({ name: action.name })
+ .then(action.cb)
+ }
+ return {
+ name: '',
+ visible: false,
+ }
+
+ default:
+ return state
+ }
+}
+
+export default newFolder
diff --git a/client/store.js b/client/store.js
index 1ac7be1..4e0b368 100644
--- a/client/store.js
+++ b/client/store.js
@@ -9,8 +9,9 @@ const store = createStore(reducer, {
},
tasks: [],
folders: {
- openFolders: [],
+ newFolder: {},
folders: [],
+ openFolders: [],
},
audioPlayer: {
file: null,