diff --git a/client/browser/FinderBrowser.tsx b/client/browser/FinderBrowser.tsx new file mode 100644 index 000000000..3117818da --- /dev/null +++ b/client/browser/FinderBrowser.tsx @@ -0,0 +1,133 @@ +import React, {createRef, useEffect, useState} from 'react'; +import {File, Folder} from "../finder/Item"; +import ArrowDownIcon from '../icons/arrow-down.svg'; +import ArrowRightIcon from '../icons/arrow-right.svg'; +import FolderIcon from '../icons/folder.svg'; +import RootIcon from '../icons/root.svg'; + + +function InodeList(props) { + const {folderId} = props; + const [isLoading, setLoading] = useState(false); + const [inodes, setInodes] = useState([]); + const [searchQuery, setSearchQuery] = useState(() => { + const params = new URLSearchParams(window.location.search); + return params.get('q'); + }); + + useEffect(() => { + fetchInodes(); + }, [searchQuery]); + + async function fetchInodes() { + const params = new URLSearchParams({q: searchQuery}); + const fetchInodesUrl = `${props.baseurl}${folderId}/fetch${searchQuery ? `?${params.toString()}` : ''}`; + setLoading(true); + const response = await fetch(fetchInodesUrl); + if (response.ok) { + const body = await response.json(); + setInodes(body.inodes.map(inode => { + const elementRef = createRef(); + return {...inode, elementRef}; + })); + } else { + console.error(response); + } + setLoading(false); + } + + function selectInode(event: PointerEvent, inode) { + if (inode.disabled) + return; + let modifier, selectedIndex = -1; + // simple click + if (inode.selected) { + modifier = f => ({...f, selected: false}); + } else { + if (!(event.target as HTMLElement)?.classList.contains('inode-name')) { + // prevent selecting the inode when clicking on the name field to edit it + modifier = f => ({...f, selected: f.id === inode.id}); + selectedIndex = inodes.findIndex(f => f.id === inode.id); // remember for an upcoming shift-click + } else { + modifier = f => f; + } + } + const modifiedInodes = inodes.map((f, k) => ({...modifier(f, k), cutted: false, copied: false})); + setInodes(modifiedInodes); + } + + function deselectInodes() { + if (inodes.find(inode => inode.selected || inode.dragged)) { + setInodes(inodes.map(inode => ({...inode, selected: false, dragged: false}))); + } + } + + function renderInodes() { + if (isLoading) + return (
  • {gettext("Loading...")}
  • ); + + if (inodes.length === 0) { + if (searchQuery) + return (
  • {`No match while searching for “${searchQuery}”`}
  • ); + return (
  • {gettext("This folder is empty")}
  • ); + } + + return inodes.map(inode => inode.is_folder + ? + : + ); + } + + function cssClasses() { + const classes = ['inode-list']; + if (isLoading) + classes.push('loading'); + return classes.join(' '); + } + + return ( + + ); +} + + +function FolderStructure(props) { + const {folder} = props; + + return folder ? ( +
  • + {folder.is_root ? : <>{folder.name}} + {folder.children && (
      + {folder.children.map(child => ( + + ))} +
    )} +
  • + ) : null; +} + + +export function FinderBrowser(props) { + const [structure, setStructure] = useState({root_folder: null}); + + useEffect(() => { + getStructure(); + }, []); + + async function getStructure() { + const response = await fetch(`${props.baseurl}structure/${props.realm}`); + if (response.ok) { + setStructure(await response.json()); + } else { + console.error(response); + } + } + + return ( + + ); +} diff --git a/client/browserbuild.config.mjs b/client/browserbuild.config.mjs new file mode 100644 index 000000000..db9d5fd04 --- /dev/null +++ b/client/browserbuild.config.mjs @@ -0,0 +1,21 @@ +import {build} from 'esbuild'; +import svgr from 'esbuild-plugin-svgr'; +import parser from 'yargs-parser'; +const buildOptions = parser(process.argv.slice(2), { + boolean: ['debug', 'minify'], +}); + +await build({ + entryPoints: [ + 'client/finder-browser.ts', + ], + bundle: true, + minify: buildOptions.minify, + sourcemap: buildOptions.debug, + outfile: 'finder/static/finder/js/browser.js', + format: 'esm', + jsx: 'automatic', + plugins: [svgr()], + loader: {'.svg': 'text', '.jsx': 'jsx' }, + target: ['es2020', 'chrome84', 'firefox84', 'safari14', 'edge84'] +}).catch(() => process.exit(1)); diff --git a/client/finder-browser.scss b/client/finder-browser.scss new file mode 100644 index 000000000..af5659cbf --- /dev/null +++ b/client/finder-browser.scss @@ -0,0 +1,14 @@ +.folder-structure { + &, ul { + padding-left: 0; + list-style: none; + } + + li { + svg { + height: 20px; + vertical-align: text-bottom; + margin-right: 0.5em; + } + } +} diff --git a/client/finder-browser.ts b/client/finder-browser.ts new file mode 100644 index 000000000..0c90430a8 --- /dev/null +++ b/client/finder-browser.ts @@ -0,0 +1,9 @@ +import r2wc from '@r2wc/react-to-web-component'; +import {FinderBrowser} from 'browser/FinderBrowser'; + + +window.addEventListener('DOMContentLoaded', (event) => { + window.customElements.define('finder-browser', r2wc(FinderBrowser, { + props: {baseurl: 'string', realm: 'string'}} + )); +}); diff --git a/client/icons/arrow-down.svg b/client/icons/arrow-down.svg new file mode 100644 index 000000000..25a9e132a --- /dev/null +++ b/client/icons/arrow-down.svg @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/client/icons/arrow-right.svg b/client/icons/arrow-right.svg new file mode 100644 index 000000000..2dd539d23 --- /dev/null +++ b/client/icons/arrow-right.svg @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/client/icons/folder.svg b/client/icons/folder.svg new file mode 100644 index 000000000..fdf8187ed --- /dev/null +++ b/client/icons/folder.svg @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/finder/api/__init__.py b/finder/api/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/finder/api/urls.py b/finder/api/urls.py new file mode 100644 index 000000000..a4a36612a --- /dev/null +++ b/finder/api/urls.py @@ -0,0 +1,27 @@ +from django.urls import path +from django.views.i18n import JavaScriptCatalog + +from finder.api.views import BrowserView + + +app_name = 'finder-api' +urlpatterns = [ + path( + 'structure/', + BrowserView.as_view(action='structure'), + ), + path( + 'fetch/', + BrowserView.as_view(action='fetch'), + ), + path( + 'jsi18n/', + JavaScriptCatalog.as_view(packages=['finder']), + name="javascript-catalog", + ), + path( + '', + BrowserView.as_view(), + name="base-url", + ), +] diff --git a/finder/api/views.py b/finder/api/views.py new file mode 100644 index 000000000..fd04df994 --- /dev/null +++ b/finder/api/views.py @@ -0,0 +1,44 @@ +from django.contrib.sites.shortcuts import get_current_site +from django.http import JsonResponse, HttpResponseBadRequest, HttpResponseNotFound +from django.utils.translation import gettext +from django.views import View + +from finder.models.folder import FolderModel, RealmModel + + +class BrowserView(View): + """ + The view for web component . + """ + action = None + + def get(self, request, *args, **kwargs): + action = getattr(self, self.action, None) + if not callable(action): + return HttpResponseBadRequest(f"Action {self.action} not allowed.") + return action(request, *args, **kwargs) + + def structure(self, request, realm): + site = get_current_site(request) + try: + realm = RealmModel.objects.get(site=site, slug=realm) + except RealmModel.DoesNotExist: + return HttpResponseNotFound(f"Realm {realm} not found for {site.domain}.") + root_folder = FolderModel.objects.get_root_folder(realm) + children = FolderModel.objects.filter(parent=root_folder) + return JsonResponse({ + 'root_folder': { + 'id': str(root_folder.id), + 'name': None, # The root folder has no name. + 'is_root': True, + 'children': [{ + 'id': str(child.id), + 'name': child.name, + 'num_children': child.num_children + } for child in children], + 'num_children': root_folder.num_children, + } + }) + + def fetch(self, request, folder_id): + return JsonResponse({'message': f'Fetching folder {folder_id}'}) diff --git a/package.json b/package.json index eff6aca82..1c52432cb 100644 --- a/package.json +++ b/package.json @@ -4,13 +4,17 @@ "private": true, "scripts": { "adminbuild": "node client/adminbuild.config.mjs", + "browserbuild": "node client/browserbuild.config.mjs", "compilescss": "sass --load-path=. client:finder/static/finder/css/", + "buildall": "concurrently \"npm run adminbuild -- --debug \" \"npm run browserbuild -- --debug \" \"npm run compilescss\"" }, "devDependencies": { "@dnd-kit/core": "^6.1.0", "@dnd-kit/modifiers": "^6.0.1", + "@r2wc/react-to-web-component": "^2.0.3", "@types/react-dom": "^18.3.0", "@wavesurfer/react": "^1.0.7", + "bootstrap": "^5.3.3", "concurrently": "^8.2.0", "esbuild": "^0.19.12", "esbuild-plugin-svgr": "^2.1.0", diff --git a/testapp/templates/testapp.html b/testapp/templates/testapp.html index a90be71f7..d2ba73c17 100644 --- a/testapp/templates/testapp.html +++ b/testapp/templates/testapp.html @@ -8,8 +8,9 @@ Django-Filer - - + + + @@ -17,7 +18,7 @@

    Django Filer Demo

    -
    +
    diff --git a/testapp/urls.py b/testapp/urls.py index afb99eeda..78c8dc96d 100644 --- a/testapp/urls.py +++ b/testapp/urls.py @@ -18,9 +18,9 @@ from django.contrib import admin from django.http import HttpResponse from django.template.loader import get_template -from django.urls import path +from django.urls import include, path -from testapp.admin_site import admin_site +from finder.api import urls as finder_urls def render_landing(request): @@ -31,7 +31,7 @@ def render_landing(request): urlpatterns = [ path('admin/', admin.site.urls), - path('testapp/admin/', admin_site.urls), + path('finder-api/', include(finder_urls)), path('testapp/', render_landing), ] if settings.DEBUG: