diff options
Diffstat (limited to 'src')
| -rw-r--r-- | src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx | 8 | ||||
| -rw-r--r-- | src/client/views/nodes/KeyFrame.tsx | 16 | ||||
| -rw-r--r-- | src/client/views/nodes/Timeline.scss | 42 | ||||
| -rw-r--r-- | src/client/views/nodes/Timeline.tsx | 251 |
4 files changed, 316 insertions, 1 deletions
diff --git a/src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx b/src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx index 1a1614ac7..e741953d6 100644 --- a/src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx +++ b/src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx @@ -18,6 +18,8 @@ import "./CollectionFreeFormView.scss"; import { MarqueeView } from "./MarqueeView"; import React = require("react"); import v5 = require("uuid/v5"); +// import { BooleanField } from "../../../../fields/BooleanField"; +import { Timeline } from "../../nodes/Timeline"; import { createSchema, makeInterface, listSpec } from "../../../../new_fields/Schema"; import { Doc, WidthSym, HeightSym } from "../../../../new_fields/Doc"; import { FieldValue, Cast, NumCast, BoolCast } from "../../../../new_fields/Types"; @@ -347,8 +349,12 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) { <CollectionFreeFormRemoteCursors {...this.props} key="remoteCursors" /> </CollectionFreeFormViewPannableContents> </MarqueeView> +<<<<<<< HEAD + <Timeline {...this.props} /> +======= <CollectionFreeFormOverlayView {...this.getDocumentViewProps(this.props.Document)} {...this.props} /> - </div> +>>>>>>> 6f49d067b58caf6297f7ae7687cf05b627c27a1d + </div > ); } } diff --git a/src/client/views/nodes/KeyFrame.tsx b/src/client/views/nodes/KeyFrame.tsx new file mode 100644 index 000000000..fb072f4d6 --- /dev/null +++ b/src/client/views/nodes/KeyFrame.tsx @@ -0,0 +1,16 @@ +import { Doc } from '../../../new_fields/Doc'; + +export class KeyFrame { + private _document: Doc; + constructor() { + this._document = new Doc(); + + + } + + get doc():Doc { + return this._document; + } + + +}
\ No newline at end of file diff --git a/src/client/views/nodes/Timeline.scss b/src/client/views/nodes/Timeline.scss new file mode 100644 index 000000000..cf63c987c --- /dev/null +++ b/src/client/views/nodes/Timeline.scss @@ -0,0 +1,42 @@ +.timeline-container { + height: 100px; + width: 500px; + background-color: rgb(160, 226, 243); + position: absolute; + left: 15%; + top: 1%; + font-size: 75%; + + .timeline { + height: 60px; + width: 500px; + bottom: 0px; + background-color: grey; + position: absolute; + } + + .inner { + height: 44px; + width: 484px; + background-color: black; + //opacity: 0.5; + position: absolute; + margin: 8px; + z-index: 10; + } + + button { + height: 30px; + width: 100px; + font-size: 1em; + position: relative; + margin: 5px; + } + + input { + height: 30px; + width: 100px; + font-size: 1em; + margin: 5px; + } +}
\ No newline at end of file diff --git a/src/client/views/nodes/Timeline.tsx b/src/client/views/nodes/Timeline.tsx new file mode 100644 index 000000000..cdd355fb3 --- /dev/null +++ b/src/client/views/nodes/Timeline.tsx @@ -0,0 +1,251 @@ +import * as React from "react"; +import * as ReactDOM from "react-dom"; +import { observer } from "mobx-react"; +import { observable, reaction, action, IReactionDisposer, observe, IObservableArray } from "mobx"; +import "./Timeline.scss"; +import { KeyFrame } from "./KeyFrame"; +import { CollectionViewProps } from "../collections/CollectionBaseView"; +import { CollectionSubView, SubCollectionViewProps } from "../collections/CollectionSubView"; +import { DocumentViewProps, DocumentView } from "./DocumentView"; + +import { CollectionFreeFormView } from "../collections/collectionFreeForm/CollectionFreeFormView"; +import { Doc, Self } from "../../../new_fields/Doc"; +import { Document, listSpec } from "../../../new_fields/Schema"; +import { FieldValue, Cast } from "../../../new_fields/Types"; +import { emptyStatement } from "babel-types"; +import { DocumentManager } from "../../util/DocumentManager"; +import { SelectionManager } from "../../util/SelectionManager"; +import { List } from "../../../new_fields/List"; + + +@observer +export class Timeline extends CollectionSubView(Document) { + @observable private _inner = React.createRef<HTMLDivElement>(); + @observable private _timeInput = React.createRef<HTMLInputElement>(); + + @observable private _isRecording: Boolean = false; + @observable private _currentBar: any = null; + @observable private _newBar: any = null; + private _reactionDisposers: IReactionDisposer[] = []; + private _keyFrames: KeyFrame[] = []; + private _keyBars: HTMLDivElement[] = []; + + private _currentBarX: number = 0; + @observable private _onBar: Boolean = false; + @observable private _keys = ["x", "y"]; + @observable private _frames: Doc[] = []; + + + @action + onRecord = (e: React.MouseEvent) => { + this._isRecording = true; + let children = Cast(this.props.Document[this.props.fieldKey], listSpec(Doc)); + if (!children) { + return; + } + let childrenList = (children[Self] as any).__fields; + // let keys = ["x", "y"]; //maybe make this an instance var? + + const addReaction = (element: Doc) => { + element = (element as any).value(); + return reaction(() => { + return this._keys.map(key => FieldValue(element[key])); + }, data => { //where is the data index actually being set? + if (this._inner.current) { + let keyFrame: KeyFrame; //keyframe reference + //is KeyFrame just a wrapper for a doc? cause then could just be of type doc... + let exists: boolean = false; + let time: number = this._currentBarX; //time that indicates what frame you are in. Whole numbers. + // let frames: List<Doc>; //don't know if this should be an instance... + this._keyFrames.forEach(kf => { //checks if there is a keyframe that is tracking htis document. + if (kf.doc.document === element) { + keyFrame = kf; + this._frames = Cast(keyFrame.doc.frames, listSpec(Doc))!; + exists = true; + } + }); + + if (!exists) { + keyFrame = new KeyFrame(); + let bar: HTMLDivElement = this.createBar(5, time, "yellow"); + this._inner.current.appendChild(bar); + // keyFrame.doc.bar = bar; + keyFrame.doc.frames = new List<Doc>(); + + this._keyFrames.push(keyFrame); + } + + this._keys.forEach((key, index) => { + console.log(data[index]); + if (keyFrame.doc.frames !== undefined) { + this._frames.forEach(frame => { + if (frame.time === this._currentBarX) { + frame[key] = data[index]; + } + }); + } + + //keyFrame.document[key] = data[index]; + }); + } + }); + }; + + + observe(childrenList as IObservableArray<Doc>, change => { + + if (change.type === "update") { + this._reactionDisposers[change.index](); + this._reactionDisposers[change.index] = addReaction(change.newValue); + } else { + let removed = this._reactionDisposers.splice(change.index, change.removedCount, ...change.added.map(addReaction)); + removed.forEach(disp => disp()); + } + }, true); + + } + + //maybe to be called in innerPointerDown + //this is the only method that should have access to time changes + //should run every time the time has been changed + //should look through the array of times and positions and set docs to the positions associated with the specific time + //if there is not an entry in the array associated with the time (for now) choose the next existing lower/higher time entry, etc. + //eventually there will be interpolation between these existing times + //will be similar to last block of code in the onRecord reaction, but kind of doing the opposite... + + + @action + timeChange = async (time: number) => { //x position of mouse relative to inner box can be passed in? + let element = (Doc as any).value(); + //data is list of docs + //this.props.document + //GOOD EXAMPLES: collection subview has childdox getter : gets the data part (field key) + //to get keyframes out of document can also use similar code + let keyFrame: KeyFrame; //keyframe reference + //potentially correct looping?? + const docs = await DocListCastAsync(this.props.Document[this.props.fieldKey]); + for (let i in docs) { + let oneDoc = docs[i]; + let kfs = this._keyFrames[i]; //what is this doing (is this where child getter comes in?) + } + //end of potentially correct looping + this._keyFrames.forEach(kf => { //checks if there is a keyframe that is tracking this document + if (kf.doc.document === element) { + keyFrame = kf; + this._frames = Cast(keyFrame.doc.frames, listSpec(Doc))!; + } + }, + this._keys.forEach((key) => { //for each key + if (keyFrame.doc.frames !== undefined) { + this._frames.forEach(frame => { + if (frame.time === time) { + keyFrame.doc[key] = frame[key]; //set the doc's position to the frames' stored values + } + }); + } + } + } + + get keyFrames() { + return Cast(this.props.Document[this.props.fieldKey], listSpec(Doc), []).filter(doc => KeyFrame); + } + + @action + displayKeyFrames = (dv: DocumentView) => { + console.log(dv); + dv.props.Document; + + } + + + @action + onInnerPointerUp = (e: React.PointerEvent) => { + if (this._inner.current) { + this._inner.current.removeEventListener("pointermove", this.onInnerPointerMove); + } + } + + @action + onInnerPointerDown = (e: React.PointerEvent) => { + e.preventDefault(); + e.stopPropagation(); + if (this._isRecording) { + if (this._inner.current) { + let mouse = e.nativeEvent; + let offsetX = Math.round(mouse.offsetX); + this._currentBarX = offsetX; + this._currentBar.style.transform = `translate(${offsetX}px)`; + this._inner.current.removeEventListener("pointermove", this.onInnerPointerMove); //reset + this._inner.current.addEventListener("pointermove", this.onInnerPointerMove); + } + } + } + + @action + onInnerPointerMove = (e: PointerEvent) => { + e.preventDefault(); + e.stopPropagation(); + let offsetX = Math.round(e.offsetX); + this._currentBarX = offsetX; + this._currentBar.style.transform = `translate(${offsetX}px)`; //styling should not have to be done this way...maybe done through react?? + console.log(offsetX); + console.log(this._currentBarX); + } + + createBar = (width: number, pos: number = 0, color: string = "green"): HTMLDivElement => { + let bar = document.createElement("div"); + bar.style.height = "100%"; + bar.style.width = `${width}px`; + bar.style.backgroundColor = color; + bar.style.transform = `translate(${pos}px)`; //repeated code from previous method + bar.style.position = "absolute"; + bar.style.pointerEvents = "none"; + return bar; + } + + onTimeEntered = (e: React.KeyboardEvent) => { + if (this._timeInput.current) { + if (e.keyCode === 13) { + let input = parseInt(this._timeInput.current.value) || 0; + this._currentBar.style.transform = `translate(${input}px)`; + this._currentBarX = input; + } + } + } + + componentDidMount() { + if (this._inner.current && this._currentBar === null) { + this._currentBar = this.createBar(5); + this._inner.current.appendChild(this._currentBar); + } + + let doc: Doc = this.props.Document; + let test = this.props.Document[this.props.fieldKey]; + + } + + componentWillUnmount() { + this._reactionDisposers.forEach(disp => disp()); + this._reactionDisposers = []; + } + + render() { + return ( + <div> + <div className="timeline-container"> + <div className="timeline"> + <div className="inner" ref={this._inner} onPointerDown={this.onInnerPointerDown} onPointerUp={this.onInnerPointerUp}> + { + SelectionManager.SelectedDocuments().map((dv) => { + this.displayKeyFrames(dv); + }) + } + </div> + </div> + <button onClick={this.onRecord}>Record</button> + <input placeholder="Time" ref={this._timeInput} onKeyDown={this.onTimeEntered}></input> + </div> + </div> + ); + } +}
\ No newline at end of file |
