Finished grid demo

This commit is contained in:
Robert Long 2021-08-16 16:18:45 -07:00
parent e68c9bee4a
commit f303cb345f
4 changed files with 88 additions and 30 deletions

27
package-lock.json generated
View file

@ -11,6 +11,7 @@
"classnames": "^2.3.1", "classnames": "^2.3.1",
"color-hash": "^2.0.1", "color-hash": "^2.0.1",
"events": "^3.3.0", "events": "^3.3.0",
"lodash-move": "^1.1.1",
"matrix-js-sdk": "^12.0.1", "matrix-js-sdk": "^12.0.1",
"react": "^17.0.0", "react": "^17.0.0",
"react-dom": "^17.0.0", "react-dom": "^17.0.0",
@ -1099,6 +1100,19 @@
"verror": "1.10.0" "verror": "1.10.0"
} }
}, },
"node_modules/lodash": {
"version": "4.17.21",
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
"integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg=="
},
"node_modules/lodash-move": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/lodash-move/-/lodash-move-1.1.1.tgz",
"integrity": "sha1-WfduDxrFfm2Gg/UxvsB8W26k40g=",
"dependencies": {
"lodash": "^4.6.1"
}
},
"node_modules/loglevel": { "node_modules/loglevel": {
"version": "1.7.1", "version": "1.7.1",
"resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.7.1.tgz", "resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.7.1.tgz",
@ -2530,6 +2544,19 @@
"verror": "1.10.0" "verror": "1.10.0"
} }
}, },
"lodash": {
"version": "4.17.21",
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
"integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg=="
},
"lodash-move": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/lodash-move/-/lodash-move-1.1.1.tgz",
"integrity": "sha1-WfduDxrFfm2Gg/UxvsB8W26k40g=",
"requires": {
"lodash": "^4.6.1"
}
},
"loglevel": { "loglevel": {
"version": "1.7.1", "version": "1.7.1",
"resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.7.1.tgz", "resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.7.1.tgz",

View file

@ -10,6 +10,7 @@
"classnames": "^2.3.1", "classnames": "^2.3.1",
"color-hash": "^2.0.1", "color-hash": "^2.0.1",
"events": "^3.3.0", "events": "^3.3.0",
"lodash-move": "^1.1.1",
"matrix-js-sdk": "^12.0.1", "matrix-js-sdk": "^12.0.1",
"react": "^17.0.0", "react": "^17.0.0",
"react-dom": "^17.0.0", "react-dom": "^17.0.0",

View file

@ -1,8 +1,9 @@
import React, { useCallback, useEffect, useRef, useState } from "react"; import React, { useCallback, useEffect, useRef, useState } from "react";
import { useDrag } from "react-use-gesture"; import { useDrag } from "react-use-gesture";
import { useSprings, animated, useSpring } from "@react-spring/web"; import { useSprings, animated } from "@react-spring/web";
import styles from "./GridDemo.module.css"; import styles from "./GridDemo.module.css";
import useMeasure from "react-use-measure"; import useMeasure from "react-use-measure";
import moveArrItem from "lodash-move";
function isInside([x, y], targetTile) { function isInside([x, y], targetTile) {
const left = targetTile.x; const left = targetTile.x;
@ -128,7 +129,9 @@ export function GridDemo() {
const stream = await navigator.mediaDevices.getUserMedia({ video: true }); const stream = await navigator.mediaDevices.getUserMedia({ video: true });
setStream(stream); setStream(stream);
tileOrderRef.current.push(tileOrderRef.current.length); tileOrderRef.current.push(tileOrderRef.current.length);
setTileState(() => { setTileState(() => {
console.log("startWebcam");
const tiles = [{ stream, key: tileKey.current++, remove: false }]; const tiles = [{ stream, key: tileKey.current++, remove: false }];
const tilePositions = getTilePositions(tiles, gridBounds); const tilePositions = getTilePositions(tiles, gridBounds);
return { tiles, tilePositions }; return { tiles, tilePositions };
@ -141,6 +144,7 @@ export function GridDemo() {
tileOrderRef.current.push(tileOrderRef.current.length); tileOrderRef.current.push(tileOrderRef.current.length);
setTileState(({ tiles }) => { setTileState(({ tiles }) => {
console.log("addTile");
const newTiles = [ const newTiles = [
...tiles, ...tiles,
{ stream: newStream, key: tileKey.current++, remove: false }, { stream: newStream, key: tileKey.current++, remove: false },
@ -193,10 +197,11 @@ export function GridDemo() {
}, [gridBounds]); }, [gridBounds]);
const animate = useCallback( const animate = useCallback(
(order) => (index) => { (tileIndex) => {
const tileIndex = order[index]; const tileOrder = tileOrderRef.current;
const order = tileOrder.indexOf(tileIndex);
const tile = tiles[tileIndex]; const tile = tiles[tileIndex];
const tilePosition = tilePositions[tileIndex]; const tilePosition = tilePositions[order];
const draggingTile = draggingTileRef.current; const draggingTile = draggingTileRef.current;
const dragging = draggingTile && tile.key === draggingTile.key; const dragging = draggingTile && tile.key === draggingTile.key;
const remove = tile.remove; const remove = tile.remove;
@ -205,8 +210,8 @@ export function GridDemo() {
return { return {
width: tilePosition.width, width: tilePosition.width,
height: tilePosition.height, height: tilePosition.height,
x: tilePosition.x + draggingTile.x, x: draggingTile.offsetX + draggingTile.x,
y: tilePosition.y + draggingTile.y, y: draggingTile.offsetY + draggingTile.y,
scale: 1.1, scale: 1.1,
opacity: 1, opacity: 1,
zIndex: 1, zIndex: 1,
@ -237,47 +242,62 @@ export function GridDemo() {
[tiles, tilePositions] [tiles, tilePositions]
); );
const [springs, api] = useSprings( const [springs, api] = useSprings(tiles.length, animate, [
tiles.length, tilePositions,
animate(tileOrderRef.current), tiles,
[tilePositions, tiles] ]);
);
const bind = useDrag(({ args: [index], active, xy, movement }) => { const bind = useDrag(({ args: [key], active, xy, movement }) => {
let order = tileOrderRef.current; const tileOrder = tileOrderRef.current;
const tileIndex = tileOrderRef.current[index]; const dragTileIndex = tiles.findIndex((tile) => tile.key === key);
const tile = tiles[tileIndex]; const dragTile = tiles[dragTileIndex];
const dragTileOrder = tileOrder.indexOf(dragTileIndex);
const cursorPosition = [xy[0] - gridBounds.left, xy[1] - gridBounds.top]; const cursorPosition = [xy[0] - gridBounds.left, xy[1] - gridBounds.top];
for (let i = 0; i < tileOrderRef.current.length; i++) { for (
if (i === index) { let hoverTileIndex = 0;
hoverTileIndex < tiles.length;
hoverTileIndex++
) {
const hoverTile = tiles[hoverTileIndex];
const hoverTileOrder = tileOrder.indexOf(hoverTileIndex);
const hoverTilePosition = tilePositions[hoverTileOrder];
if (hoverTile.key === key) {
continue; continue;
} }
const hoverTileIndex = tileOrderRef.current[i];
const hoverTilePosition = tilePositions[hoverTileIndex];
if (isInside(cursorPosition, hoverTilePosition)) { if (isInside(cursorPosition, hoverTilePosition)) {
// TODO: Figure out swapping tileOrderRef.current = moveArrItem(
// order[i] = tileIndex; tileOrder,
// order[index] = i; dragTileOrder,
hoverTileOrder
);
break; break;
} }
} }
if (active) { if (active) {
if (!draggingTileRef.current) {
const tilePosition = tilePositions[dragTileOrder];
draggingTileRef.current = { draggingTileRef.current = {
key: tile.key, key: dragTile.key,
x: movement[0], offsetX: tilePosition.x,
y: movement[1], offsetY: tilePosition.y,
}; };
}
draggingTileRef.current.x = movement[0];
draggingTileRef.current.y = movement[1];
} else { } else {
draggingTileRef.current = null; draggingTileRef.current = null;
} }
api.start(animate(order)); api.start(animate);
}); });
return ( return (
@ -300,7 +320,7 @@ export function GridDemo() {
return ( return (
<ParticipantTile <ParticipantTile
{...bind(i)} {...bind(tile.key)}
key={tile.key} key={tile.key}
style={{ style={{
boxShadow: shadow.to( boxShadow: shadow.to(
@ -308,6 +328,7 @@ export function GridDemo() {
), ),
...style, ...style,
}} }}
tileKey={tile.key}
{...tile} {...tile}
/> />
); );
@ -331,6 +352,7 @@ function ParticipantTile({ style, stream, remove, tileKey, ...rest }) {
return ( return (
<animated.div className={styles.participantTile} style={style} {...rest}> <animated.div className={styles.participantTile} style={style} {...rest}>
<div className={styles.participantName}>{tileKey}</div>
<video ref={videoRef} playsInline /> <video ref={videoRef} playsInline />
</animated.div> </animated.div>
); );

View file

@ -27,3 +27,11 @@
height: 100%; height: 100%;
object-fit: cover; object-fit: cover;
} }
.participantName {
position: absolute;
bottom: 0;
right: 0;
background-color: rgba(0, 0, 0, 0.2);
padding: 8px 24px;
}