Skip to content
2 changes: 1 addition & 1 deletion src/frontend/packages/list-components/dist/index.cjs.js

Large diffs are not rendered by default.

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion src/frontend/packages/list-components/dist/index.es.js

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion src/frontend/packages/list-components/dist/index.es.js.map

Large diffs are not rendered by default.

164 changes: 164 additions & 0 deletions src/frontend/packages/list-components/src/ReferenceFilterTree.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,164 @@
import React from 'react';
import { useGetList } from 'react-admin';
import {TreeItem, TreeView, useTreeItem} from '@mui/lab';
import { useListFilterContext } from 'ra-core';
import LabelIcon from '@mui/icons-material/Label';
import { IconButton } from '@mui/material';
import ExpandMoreIcon from '@mui/icons-material/ExpandMore';
import ChevronRightIcon from '@mui/icons-material/ChevronRight';
import clsx from 'clsx';
import Typography from '@mui/material/Typography';

/**
* @example
* const FilterAside = () => (
* <Card>
* <CardContent>
* <FilterLiveSearch source="pair:label" />
* <ReferenceFilterTree
* reference="Theme"
* source="pair:broader"
* label="pair:label"
* icon={icon}
* predicate={"http://virtual-assembly.org/ontologies/pair#hasTopic"}
* />
* </CardContent>
* </Card>
* );
*/
Comment thread
BastienSig marked this conversation as resolved.

function GenerateTreeItem(source, label, allItems, routeTree, parentId) {
// If !parentId it's a trunkItem
const isParentLevel = !parentId;
const listToUse = isParentLevel ? routeTree : allItems.filter(({ [source]: itemSource }) => itemSource === parentId);
return (
listToUse.map((route) =>
<CustomTreeItem nodeId={route["id"]} label={route[label]} key={route["id"]}>
{GenerateTreeItem(source, label, allItems, [], route["id"])}
</CustomTreeItem>
)
)
}

const CustomContent = React.forwardRef(function CustomContent(props, ref) {
const {
classes,
className,
label,
nodeId,
icon: iconProp,
expansionIcon,
displayIcon,
} = props;

const {
disabled,
expanded,
selected,
focused,
handleExpansion,
handleSelection,
preventSelection,
} = useTreeItem(nodeId);

const icon = iconProp || expansionIcon || displayIcon;

const handleMouseDown = (event) => {
preventSelection(event);
};

const handleExpansionClick = (event) => {
handleExpansion(event);
};

const handleSelectionClick = (event) => {
handleSelection(event);
};

return (
// eslint-disable-next-line jsx-a11y/no-static-element-interactions
<div
className={clsx(className, classes.root, {
[classes.expanded]: expanded,
[classes.selected]: selected,
[classes.focused]: focused,
[classes.disabled]: disabled,
})}
onMouseDown={handleMouseDown}
ref={ref}
>
{/* eslint-disable-next-line jsx-a11y/click-events-have-key-events,jsx-a11y/no-static-element-interactions */}
<div onClick={handleExpansionClick} className={classes.iconContainer}>
{icon}
</div>
<Typography
onClick={handleSelectionClick}
component="div"
className={classes.label}
>
{label}
</Typography>
</div>
);
});

function CustomTreeItem(props) {
return <TreeItem ContentComponent={CustomContent} {...props} />;
}

const ReferenceFilterTree = ({ reference, source, label, limit, sort, filter, icon, predicate }) => {
const { data } = useGetList(reference, { page: 1, perPage: Infinity }, sort, filter);
const { filterValues, setFilters } = useListFilterContext();

let routeTree = [], allItems = [];
for (const item in data) {
if (data[item][source] === undefined ) {
routeTree.push(data[item]);
}
allItems = allItems.concat(data[item]);
}

const handleSelect = (event, nodes) => {
const sparqlWhere = {
"type": "bgp",
"triples": nodes.map((node) => {
return {
subject: {
termType: 'Variable',
value: 's1',
},
predicate: {
termType: 'NamedNode',
value: predicate,
},
object: {
termType: 'NamedNode',
value: node,
},
};
}),
}

const encodedQuery = encodeURIComponent(JSON.stringify(sparqlWhere));
setFilters({...filterValues, "sparqlWhere": encodedQuery })
}

return (
<div>
{!icon ? icon = <IconButton size="small" edge="start"> <LabelIcon style={{ color: 'black', }} /> </IconButton> : icon}
{reference}
<TreeView
multiSelect
onNodeSelect={handleSelect}
aria-label="icon expansion"
defaultCollapseIcon={<ExpandMoreIcon />}
defaultExpandIcon={<ChevronRightIcon />}
>
{GenerateTreeItem(source, label, allItems, routeTree)}
</TreeView>
</div>
)
};

export default ReferenceFilterTree;

1 change: 1 addition & 0 deletions src/frontend/packages/list-components/src/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ export { default as ChipList } from './ChipList';
export { default as GridList } from './GridList';
export { default as MasonryList } from './MasonryList';
export { default as ReferenceFilter } from './ReferenceFilter';
export { default as ReferenceFilterTree } from './ReferenceFilterTree';

export { default as MultiViewsList } from './MultiViewsList/MultiViewsList';
export { default as ListActionsWithViews } from './MultiViewsList/ListActionsWithViews';
Expand Down

Large diffs are not rendered by default.

Large diffs are not rendered by default.

Large diffs are not rendered by default.

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,10 @@ const fetchSparqlEndpoints = async (containers, resourceId, params, config) => {
getBlankNodesFromDataServers(containers[serverKey], dataServers);

const predicates = params.filter?._predicates || dataModel.list?.predicates;

//When the sparql request come from URI, it's a string who must must be decode.
if (params.filter.sparqlWhere && (typeof params.filter.sparqlWhere === 'string' || params.filter.sparqlWhere instanceof String)) {
params.filter.sparqlWhere = JSON.parse(decodeURIComponent(params.filter.sparqlWhere));
}
const sparqlQuery = buildSparqlQuery({
containers: containers[serverKey],
params: { ...params, filter: { ...dataModel.list?.filter, ...params.filter } },
Expand Down