aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/client/views/collections/CollectionView.tsx1
-rw-r--r--src/client/views/collections/CollectionViewChromes.scss30
-rw-r--r--src/client/views/collections/CollectionViewChromes.tsx57
-rw-r--r--src/client/views/collections/collectionGrid/CollectionGridView.tsx248
-rw-r--r--src/client/views/collections/collectionGrid/Grid.tsx8
-rw-r--r--src/client/views/nodes/ContentFittingDocumentView.tsx2
-rw-r--r--src/client/views/nodes/DocumentView.tsx1
7 files changed, 225 insertions, 122 deletions
diff --git a/src/client/views/collections/CollectionView.tsx b/src/client/views/collections/CollectionView.tsx
index 5b344813d..311e8d3a1 100644
--- a/src/client/views/collections/CollectionView.tsx
+++ b/src/client/views/collections/CollectionView.tsx
@@ -227,6 +227,7 @@ export class CollectionView extends Touchable<FieldViewProps & CollectionViewCus
subItems.push({ description: "Carousel", event: () => func(CollectionViewType.Carousel), icon: "columns" });
subItems.push({ description: "Pivot/Time", event: () => func(CollectionViewType.Time), icon: "columns" });
subItems.push({ description: "Map", event: () => func(CollectionViewType.Map), icon: "globe-americas" });
+ subItems.push({ description: "Grid", event: () => func(CollectionViewType.Grid), icon: "border-all" });
if (addExtras && this.props.Document._viewType === CollectionViewType.Freeform) {
subItems.push({ description: "Custom", icon: "fingerprint", event: AddCustomFreeFormLayout(this.props.Document, this.props.fieldKey) });
}
diff --git a/src/client/views/collections/CollectionViewChromes.scss b/src/client/views/collections/CollectionViewChromes.scss
index e4581eb46..9795a3a22 100644
--- a/src/client/views/collections/CollectionViewChromes.scss
+++ b/src/client/views/collections/CollectionViewChromes.scss
@@ -3,7 +3,7 @@
.collectionViewChrome-cont {
position: absolute;
- width:100%;
+ width: 100%;
opacity: 0.9;
z-index: 9001;
transition: top .5s;
@@ -33,7 +33,7 @@
outline-color: black;
}
- .collectionViewBaseChrome-button{
+ .collectionViewBaseChrome-button {
font-size: 75%;
text-transform: uppercase;
letter-spacing: 2px;
@@ -44,6 +44,7 @@
padding: 12px 10px 11px 10px;
margin-left: 10px;
}
+
.collectionViewBaseChrome-cmdPicker {
margin-left: 3px;
margin-right: 0px;
@@ -51,14 +52,16 @@
border: none;
color: grey;
}
+
.commandEntry-outerDiv {
pointer-events: all;
background-color: gray;
display: flex;
flex-direction: row;
+
.commandEntry-drop {
- color:white;
- width:25px;
+ color: white;
+ width: 25px;
margin-top: auto;
margin-bottom: auto;
}
@@ -72,15 +75,17 @@
pointer-events: all;
// margin-top: 10px;
}
+
.collectionViewBaseChrome-template,
.collectionViewBaseChrome-viewModes {
display: grid;
background: rgb(238, 238, 238);
- color:grey;
- margin-top:auto;
- margin-bottom:auto;
+ color: grey;
+ margin-top: auto;
+ margin-bottom: auto;
margin-left: 5px;
}
+
.collectionViewBaseChrome-viewModes {
margin-left: 25px;
}
@@ -88,7 +93,7 @@
.collectionViewBaseChrome-viewSpecs {
margin-left: 5px;
display: grid;
-
+
.collectionViewBaseChrome-filterIcon {
position: relative;
display: flex;
@@ -202,7 +207,7 @@
.collectionStackingViewChrome-pivotField,
.collectionTreeViewChrome-pivotField {
color: white;
- width:100%;
+ width: 100%;
min-width: 100px;
text-align: center;
background: rgb(238, 238, 238);
@@ -232,6 +237,10 @@
.collectionTreeViewChrome-pivotField:hover {
cursor: text;
}
+
+ .collectionGridViewChrome-entryBox {
+ width: 50%;
+ }
}
}
@@ -292,8 +301,9 @@
flex-direction: column;
height: 40px;
}
+
.commandEntry-inputArea {
- display:flex;
+ display: flex;
flex-direction: row;
width: 150px;
margin: auto auto auto auto;
diff --git a/src/client/views/collections/CollectionViewChromes.tsx b/src/client/views/collections/CollectionViewChromes.tsx
index 5dc0b09ac..def20ae9b 100644
--- a/src/client/views/collections/CollectionViewChromes.tsx
+++ b/src/client/views/collections/CollectionViewChromes.tsx
@@ -2,7 +2,7 @@ import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
import { action, computed, observable, runInAction } from "mobx";
import { observer } from "mobx-react";
import * as React from "react";
-import { Doc, DocListCast } from "../../../fields/Doc";
+import { Doc, DocListCast, HeightSym } from "../../../fields/Doc";
import { Id } from "../../../fields/FieldSymbols";
import { List } from "../../../fields/List";
import { listSpec } from "../../../fields/Schema";
@@ -15,6 +15,8 @@ import { COLLECTION_BORDER_WIDTH } from "../globalCssVariables.scss";
import { CollectionViewType } from "./CollectionView";
import { CollectionView } from "./CollectionView";
import "./CollectionViewChromes.scss";
+import { CollectionGridView } from "./collectionGrid/CollectionGridView";
+import HeightLabel from "./collectionMulticolumn/MultirowHeightLabel";
const datepicker = require('js-datepicker');
interface CollectionViewChromeProps {
@@ -205,6 +207,7 @@ export class CollectionViewBaseChrome extends React.Component<CollectionViewChro
case CollectionViewType.Schema: return (<CollectionSchemaViewChrome key="collchrome" PanelWidth={this.props.PanelWidth} CollectionView={this.props.CollectionView} type={this.props.type} />);
case CollectionViewType.Tree: return (<CollectionTreeViewChrome key="collchrome" PanelWidth={this.props.PanelWidth} CollectionView={this.props.CollectionView} type={this.props.type} />);
case CollectionViewType.Masonry: return (<CollectionStackingViewChrome key="collchrome" PanelWidth={this.props.PanelWidth} CollectionView={this.props.CollectionView} type={this.props.type} />);
+ case CollectionViewType.Grid: return (<CollectionGridViewChrome key="collchrome" PanelWidth={this.props.PanelWidth} CollectionView={this.props.CollectionView} type={this.props.type} />);
default: return null;
}
}
@@ -504,3 +507,55 @@ export class CollectionTreeViewChrome extends React.Component<CollectionViewChro
}
}
+/**
+ * Chrome for grid view.
+ */
+@observer
+export class CollectionGridViewChrome extends React.Component<CollectionViewChromeProps> {
+
+ /**
+ * Sets the value of `numCols` on the grid's Document to the value entered.
+ */
+ @action
+ onNumColsEnter = (e: React.KeyboardEvent<HTMLInputElement>) => {
+ if (e.key === "Enter" || e.key === "Tab") {
+ if (this.props.CollectionView.props.Document.numCols as number !== e.currentTarget.valueAsNumber) {
+ this.props.CollectionView.props.Document.numCols = e.currentTarget.valueAsNumber;
+ //this.props.CollectionView.forceUpdate(); // to be used if CollectionGridView is not an observer
+ }
+
+ }
+ }
+
+ /**
+ * Sets the value of `rowHeight` on the grid's Document to the value entered.
+ */
+ @action
+ onRowHeightEnter = (e: React.KeyboardEvent<HTMLInputElement>) => {
+ if (e.key === "Enter" || e.key === "Tab") {
+ if (this.props.CollectionView.props.Document.rowHeight as number !== e.currentTarget.valueAsNumber) {
+ this.props.CollectionView.props.Document.rowHeight = e.currentTarget.valueAsNumber;
+ //this.props.CollectionView.forceUpdate();
+ }
+ }
+ }
+
+ render() {
+ return (
+ <div className="collectionTreeViewChrome-cont">
+ <span className={"search-icon"}>
+ <span className="icon-background">
+ <FontAwesomeIcon icon="columns" size="1x" />
+ </span>
+ <input className="collectionGridViewChrome-entryBox" type="number" placeholder={this.props.CollectionView.props.Document.numCols as string} onKeyDown={this.onNumColsEnter} autoFocus />
+ </span>
+ <span className={"search-icon"}>
+ <span className="icon-background">
+ <FontAwesomeIcon icon="text-height" size="1x" />
+ </span>
+ <input className="collectionGridViewChrome-entryBox" type="number" placeholder={this.props.CollectionView.props.Document.rowHeight as string} onKeyDown={this.onRowHeightEnter} />
+ </span>
+ </div>
+ );
+ }
+} \ No newline at end of file
diff --git a/src/client/views/collections/collectionGrid/CollectionGridView.tsx b/src/client/views/collections/collectionGrid/CollectionGridView.tsx
index cd1e0f7f0..ebb710af6 100644
--- a/src/client/views/collections/collectionGrid/CollectionGridView.tsx
+++ b/src/client/views/collections/collectionGrid/CollectionGridView.tsx
@@ -1,39 +1,58 @@
-import { action, computed, observable } from 'mobx';
-import { observer } from 'mobx-react';
+import { computed, observable, action } from 'mobx';
import * as React from "react";
-import { Doc, DataSym, DocListCast } from '../../../../new_fields/Doc';
-import { documentSchema } from '../../../../new_fields/documentSchemas';
-import { makeInterface } from '../../../../new_fields/Schema';
-import { BoolCast, NumCast, ScriptCast, StrCast, Cast } from '../../../../new_fields/Types';
+import { Doc, DocListCast } from '../../../../fields/Doc';
+import { documentSchema } from '../../../../fields/documentSchemas';
+import { makeInterface, createSchema } from '../../../../fields/Schema';
+import { BoolCast, NumCast, ScriptCast, StrCast, Cast } from '../../../../fields/Types';
import { DragManager } from '../../../util/DragManager';
import { Transform } from '../../../util/Transform';
import { undoBatch } from '../../../util/UndoManager';
import { ContentFittingDocumentView } from '../../nodes/ContentFittingDocumentView';
import { CollectionSubView } from '../CollectionSubView';
-import { List } from '../../../../new_fields/List';
+import { SubCollectionViewProps } from '../CollectionSubView';
+import { List } from '../../../../fields/List';
import { returnZero } from '../../../../Utils';
-import Grid from "./Grid";
-import { Layout } from "./Grid";
+import Grid, { Layout } from "./Grid";
+import { Id } from '../../../../fields/FieldSymbols';
+import { observer } from 'mobx-react';
type GridSchema = makeInterface<[typeof documentSchema]>;
const GridSchema = makeInterface(documentSchema);
+@observer
export class CollectionGridView extends CollectionSubView(GridSchema) {
- private layouts: Layout[] = [];
- private layoutDocs: Doc[] = [];
- @observable private numCols: number = 10;
- @observable private rowHeight: number = 100;
- @observable private isMounted: boolean = false;
+ constructor(props: Readonly<SubCollectionViewProps>) {
+ super(props);
- componentDidMount() {
- this.isMounted = true;
+ this.props.Document.numCols = this.props.Document.numCols ? this.props.Document.numCols : 10;
+ this.props.Document.rowHeight = this.props.Document.rowHeight ? this.props.Document.rowHeight : 100;
}
- componentWillUnmount() {
- this.isMounted = false;
- console.log("hola");
+ componentDidMount() {
+ if (!(this.props.Document.gridLayouts as List<Doc>)?.length) {
+
+ console.log("no layouts stored on doc");
+
+ this.props.Document.gridLayouts = new List<Doc>();
+
+ for (let i = 0; i < this.childLayoutPairs.length; i++) {
+
+ const layoutDoc: Doc = new Doc();
+ layoutDoc.i = layoutDoc[Id];
+ layoutDoc.x = 2 * (i % 5);
+ layoutDoc.y = 2 * Math.floor(i / 5);
+ layoutDoc.w = 2;
+ layoutDoc.h = 2;
+
+ (this.props.Document.gridLayouts as List<Doc>).push(layoutDoc);
+
+ // use childlayoutpairs length instead
+ }
+
+ }
+
}
/**
@@ -42,18 +61,26 @@ export class CollectionGridView extends CollectionSubView(GridSchema) {
* the sum of all the resolved column widths of the
* documents before the target.
*/
- private lookupIndividualTransform = (layout: Layout) => {
+ private lookupIndividualTransform = (doc: Doc) => {
- const yTranslation = this.rowHeight * layout.y;// + 15 * (layout.y - 1);
- console.log(yTranslation);
- return this.props.ScreenToLocalTransform().translate(-this.props.PanelWidth() / this.numCols * layout.x, -yTranslation);
+ const yTranslation = (this.props.Document.rowHeight as number) * (doc.y as number) + 10 * (doc.y as number);
+ return this.props.ScreenToLocalTransform().translate(-this.props.PanelWidth() / (this.props.Document.numCols as number) * (doc.x as number), -yTranslation);
}
@computed get onChildClickHandler() { return ScriptCast(this.Document.onChildClick); }
- @observable private width = (layout: Layout) => layout.w * this.props.PanelWidth() / this.numCols;
- @observable private height = (layout: Layout) => layout.h * this.rowHeight;
+ /**
+ * Sets the width of the decorating box.
+ * @param Doc doc
+ */
+ @observable private width = (doc: Doc) => doc.w as number * this.props.PanelWidth() / (this.props.Document.numCols as number);
+
+ /**
+ * Sets the height of the decorating box.
+ * @param doc `Doc`
+ */
+ @observable private height = (doc: Doc) => doc.h as number * (this.props.Document.rowHeight as number);
addDocTab = (doc: Doc, where: string) => {
if (where === "inPlace" && this.layoutDoc.isInPlaceContainer) {
@@ -81,15 +108,23 @@ export class CollectionGridView extends CollectionSubView(GridSchema) {
onClick={this.onChildClickHandler}
renderDepth={this.props.renderDepth + 1}
parentActive={this.props.active}
- //Display={"contents"}
+ display={"contents"}
/>;
}
-
//@action
+ /**
+ * Saves the layouts received from the Grid to the Document.
+ * @param layouts `Layout[]`
+ */
set layout(layouts: Layout[]) {
- this.layouts = layouts;
- this.props.Document.gridLayouts = new List<Doc>();
+
+ console.log("setting layout in CollectionGridView");
+ console.log(layouts?.[0].w);
+ //this.props.Document.gridLayouts = new List<Doc>();
+
+ const docList: Doc[] = [];
+
for (const layout of layouts) {
const layoutDoc = new Doc();
layoutDoc.i = layout.i;
@@ -98,126 +133,119 @@ export class CollectionGridView extends CollectionSubView(GridSchema) {
layoutDoc.w = layout.w;
layoutDoc.h = layout.h;
- (this.props.Document.gridLayouts as List<Doc>).push(layoutDoc);
- console.log("gazoinks");
-
+ docList.push(layoutDoc);
}
- this.forceUpdate(); // better way to do this?
+
+ this.props.Document.gridLayouts = new List<Doc>(docList);
}
- get layout() {
- //console.log(this.layouts.length === 0);
- if (this.layouts.length === 0) {
- if (this.props.Document.gridLayouts) {
- //console.log(this.props.Document.gridLayouts);
- // for (const layout of (this.props.Document.gridLayouts as List<Doc>)) {
- // if (layout instanceof Doc) {
- // this.layouts.push(
- // { i: layout.i as string, x: layout.x as number, y: layout.y as number, w: layout.w as number, h: layout.h as number }
- // );
- // }
- // else {
- // layout.then((layout: Doc) => {
- // this.layouts.push(
- // { i: layout.i as string, x: layout.x as number, y: layout.y as number, w: layout.w as number, h: layout.h as number }
- // );
- // console.log(layout.i);
- // });
- // }
- // }
- // }
- for (const layout of DocListCast(this.props.Document.gridLayouts)) {
- this.layouts.push(
- { i: layout.i as string, x: layout.x as number, y: layout.y as number, w: layout.w as number, h: layout.h as number }
- );
- }
- }
- else {
- for (let i = 0; i < this.childLayoutPairs.length; i++) {
- this.layouts.push(
- { i: 'wrapper' + i, x: 2 * (i % 5), y: 2 * Math.floor(i / 5), w: 2, h: 2 }
- );
-
- const layoutDoc: Doc = new Doc();
- layoutDoc.i = "wrapper" + i;
- layoutDoc.x = 2 * (i % 5);
- layoutDoc.y = 2 * Math.floor(i / 5);
- layoutDoc.w = 2;
- layoutDoc.h = 2;
-
- this.layoutDocs.push(layoutDoc);
- this.props.Document.highest = i;
- }
- this.props.Document.gridLayouts = new List<Doc>(this.layoutDocs);
- }
- }
+ // _.reject() on item removal?
- return this.layouts;
- }
+ /**
+ * @returns a list of `ContentFittingDocumentView`s inside wrapper divs.
+ * The key of the wrapper div must be the same as the `i` value of the corresponding layout.
+ */
@computed
- private get contents(): [JSX.Element[], Layout[]] {
+ private get contents(): JSX.Element[] {
const { childLayoutPairs } = this;
const collector: JSX.Element[] = [];
- const layoutArray: Layout[] = [];
+ //const layoutArray: Layout[] = [];
+ const docList: Doc[] = DocListCast(this.props.Document.gridLayouts);
- const previousLength = this.layout.length;
- layoutArray.push(...this.layout);
+ const previousLength = docList.length;
+ // layoutArray.push(...this.layout);
- if (!layoutArray.length) {
- return [[], []];
- }
-
- if (this.childLayoutPairs.length > previousLength) {
- layoutArray.push(
- { i: 'wrapper' + previousLength, x: 2 * (previousLength % 5), y: 2 * Math.floor(previousLength / 5), w: 2, h: 2 }
- // add values to document
- );
- // this.layout.push(
- // { i: 'wrapper' + previousLength, x: 2 * (previousLength % 5), y: 2 * Math.floor(previousLength / 5), w: 2, h: 2 }
- // );
+ if (!previousLength) {
+ // console.log("early return");
+ return [];
}
for (let i = 0; i < childLayoutPairs.length; i++) {
const { layout } = childLayoutPairs[i];
- const dxf = () => this.lookupIndividualTransform(layoutArray[i]);//.translate(-NumCast(Document._xMargin), -NumCast(Document._yMargin));
- const width = () => this.width(layoutArray[i]); //this.lookupPixels(layout);
- const height = () => this.height(layoutArray[i]);//PanelHeight() - 2 * NumCast(Document._yMargin) - (BoolCast(Document.showWidthLabels) ? 20 : 0);
+ const dxf = () => this.lookupIndividualTransform(docList?.[i]);
+ const width = () => this.width(docList?.[i]);
+ const height = () => this.height(docList?.[i]);
collector.push(
<div className={"document-wrapper"}
- key={"wrapper" + i}
+ key={docList?.[i].i as string}
>
{this.getDisplayDoc(layout, dxf, width, height)}
</div>
);
}
- return [collector, layoutArray];
+ return collector;
+ }
+
+ /**
+ * @returns a list of Layouts from a list of Docs
+ * @param docLayoutList `Doc[]`
+ */
+ toLayoutList(docLayoutList: Doc[]): Layout[] {
+
+ const layouts: Layout[] = [];
+ for (const layout of docLayoutList) {
+ layouts.push(
+ { i: layout.i as string, x: layout.x as number, y: layout.y as number, w: layout.w as number, h: layout.h as number }
+ );
+ }
+ return layouts;
+ }
+
+ /**
+ * Checks whether a new node has been added to the grid and updates the Document accordingly.
+ */
+ checkUpdate() {
+ const previousLength = (this.props.Document.gridLayouts as List<Doc>)?.length;
+ if (this.childLayoutPairs.length > previousLength) {
+ console.log("adding doc");
+ const layoutDoc: Doc = new Doc();
+ layoutDoc.i = layoutDoc[Id];
+ layoutDoc.x = 2 * (previousLength % 5);
+ layoutDoc.y = 2 * Math.floor(previousLength / 5);
+ layoutDoc.w = 2;
+ layoutDoc.h = 2;
+
+ (this.props.Document.gridLayouts as List<Doc>).push(layoutDoc);
+ }
}
render(): JSX.Element {
- const contents: JSX.Element[] = this.contents?.[0];
- const layout: Layout[] = this.contents?.[1];
- // if (this.isMounted) {
+ this.checkUpdate();
+
+ const contents: JSX.Element[] = this.contents;
+ const layout: Layout[] = this.toLayoutList(DocListCast(this.props.Document.gridLayouts));
+
+ if (layout.length === 0) {
+ console.log("layouts not loaded");
+ }
+ else {
+ console.log("rendering with this");
+ console.log(layout[0].w);
+ }
+
+
return (
<div className="collectionGridView_contents"
style={{
marginLeft: NumCast(this.props.Document._xMargin), marginRight: NumCast(this.props.Document._xMargin),
marginTop: NumCast(this.props.Document._yMargin), marginBottom: NumCast(this.props.Document._yMargin)
- }} ref={this.createDashEventsTarget}>
-
+ }}
+ ref={this.createDashEventsTarget}
+ onPointerDown={(e: React.PointerEvent) => e.stopPropagation()}
+ >
<Grid
width={this.props.PanelWidth()}
nodeList={contents}
layout={layout}
gridView={this}
- numCols={this.numCols}
- rowHeight={this.rowHeight}
+ numCols={this.props.Document.numCols as number}
+ rowHeight={this.props.Document.rowHeight as number}
/>
</div>
);
- // }
}
}
diff --git a/src/client/views/collections/collectionGrid/Grid.tsx b/src/client/views/collections/collectionGrid/Grid.tsx
index ce0173e94..d400f2810 100644
--- a/src/client/views/collections/collectionGrid/Grid.tsx
+++ b/src/client/views/collections/collectionGrid/Grid.tsx
@@ -20,9 +20,16 @@ interface GridProps {
rowHeight: number;
}
+/**
+ * Wrapper around the actual GridLayout of `react-grid-layout`.
+ */
@observer
export default class Grid extends React.Component<GridProps, GridLayout.ResponsiveProps> {
+ /**
+ * If there has been a change in layout, calls a method in CollectionGridView to set the layouts on the Document.
+ * @param layout `Layout[]`
+ */
onLayoutChange(layout: Layout[]) {
this.props.gridView.layout = layout;
}
@@ -36,6 +43,7 @@ export default class Grid extends React.Component<GridProps, GridLayout.Responsi
width={this.props.width}
compactType={null}
isDroppable={true}
+ margin={[10, 10]}
onLayoutChange={layout => this.onLayoutChange(layout)}
>
{this.props.nodeList}
diff --git a/src/client/views/nodes/ContentFittingDocumentView.tsx b/src/client/views/nodes/ContentFittingDocumentView.tsx
index 39097865a..77555061f 100644
--- a/src/client/views/nodes/ContentFittingDocumentView.tsx
+++ b/src/client/views/nodes/ContentFittingDocumentView.tsx
@@ -83,7 +83,7 @@ export class ContentFittingDocumentView extends React.Component<DocumentViewProp
return (<div className="contentFittingDocumentView" style={{
width: Math.abs(this.centeringYOffset) > 0.001 ? "auto" : this.props.PanelWidth(),
height: Math.abs(this.centeringOffset) > 0.0001 ? "auto" : this.props.PanelHeight(),
- display: this.props.Display /* just added for grid */
+ display: this.props.display /* just added for grid */
}}>
{!this.props.Document || !this.props.PanelWidth ? (null) : (
<div className="contentFittingDocumentView-previewDoc"
diff --git a/src/client/views/nodes/DocumentView.tsx b/src/client/views/nodes/DocumentView.tsx
index 340fa06a8..95af52f13 100644
--- a/src/client/views/nodes/DocumentView.tsx
+++ b/src/client/views/nodes/DocumentView.tsx
@@ -96,6 +96,7 @@ export interface DocumentViewProps {
dontRegisterView?: boolean;
layoutKey?: string;
radialMenu?: String[];
+ display?: string;
}
@observer