Skip to content
Snippets Groups Projects
Commit 7535ab22 authored by Louis's avatar Louis :fire:
Browse files

Initial version, basic interactions such as 'push' and 'you'. Emergent...

Initial version, basic interactions such as 'push' and 'you'. Emergent features implemented such as noun 'is' noun
parent e534486c
No related branches found
No related tags found
No related merge requests found
Showing with 719 additions and 44 deletions
*.png filter=lfs diff=lfs merge=lfs -text
......@@ -916,6 +916,11 @@
"minimist": "^1.2.0"
}
},
"@commander-lol/redux-reducer": {
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/@commander-lol/redux-reducer/-/redux-reducer-1.0.1.tgz",
"integrity": "sha1-u2Vvl13M8/3qd/aB9768ul6PiG8="
},
"@csstools/convert-colors": {
"version": "1.4.0",
"resolved": "https://registry.npmjs.org/@csstools/convert-colors/-/convert-colors-1.4.0.tgz",
......@@ -1469,6 +1474,16 @@
}
}
},
"@wasm-tool/wasm-pack-plugin": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/@wasm-tool/wasm-pack-plugin/-/wasm-pack-plugin-1.0.0.tgz",
"integrity": "sha512-8Z0rGItT0sMSj2gDQT6ws2pxbwVQzXmfaeArcQdh+AUQK0zjpbBBreIs9aEQS4G+2XnvDwUbuibj/C2ezAFXyQ==",
"requires": {
"chalk": "^2.4.1",
"command-exists": "^1.2.7",
"watchpack": "^1.6.0"
}
},
"@webassemblyjs/ast": {
"version": "1.8.5",
"resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.8.5.tgz",
......@@ -3464,6 +3479,11 @@
"resolved": "https://registry.npmjs.org/comma-separated-tokens/-/comma-separated-tokens-1.0.7.tgz",
"integrity": "sha512-Jrx3xsP4pPv4AwJUDWY9wOXGtwPXARej6Xd99h4TUGotmf8APuquKMpK+dnD3UgyxK7OEWaisjZz+3b5jtL6xQ=="
},
"command-exists": {
"version": "1.2.8",
"resolved": "https://registry.npmjs.org/command-exists/-/command-exists-1.2.8.tgz",
"integrity": "sha512-PM54PkseWbiiD/mMsbvW351/u+dafwTJ0ye2qB60G1aGQP9j3xK2gmMDc+R34L3nDtx4qMCitXT75mkbkGJDLw=="
},
"commander": {
"version": "2.20.0",
"resolved": "https://registry.npmjs.org/commander/-/commander-2.20.0.tgz",
......@@ -3614,6 +3634,55 @@
"resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz",
"integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40="
},
"copy-webpack-plugin": {
"version": "5.0.3",
"resolved": "https://registry.npmjs.org/copy-webpack-plugin/-/copy-webpack-plugin-5.0.3.tgz",
"integrity": "sha512-PlZRs9CUMnAVylZq+vg2Juew662jWtwOXOqH4lbQD9ZFhRG9R7tVStOgHt21CBGVq7k5yIJaz8TXDLSjV+Lj8Q==",
"requires": {
"cacache": "^11.3.2",
"find-cache-dir": "^2.1.0",
"glob-parent": "^3.1.0",
"globby": "^7.1.1",
"is-glob": "^4.0.1",
"loader-utils": "^1.2.3",
"minimatch": "^3.0.4",
"normalize-path": "^3.0.0",
"p-limit": "^2.2.0",
"schema-utils": "^1.0.0",
"serialize-javascript": "^1.7.0",
"webpack-log": "^2.0.0"
},
"dependencies": {
"globby": {
"version": "7.1.1",
"resolved": "https://registry.npmjs.org/globby/-/globby-7.1.1.tgz",
"integrity": "sha1-+yzP+UAfhgCUXfral0QMypcrhoA=",
"requires": {
"array-union": "^1.0.1",
"dir-glob": "^2.0.0",
"glob": "^7.1.2",
"ignore": "^3.3.5",
"pify": "^3.0.0",
"slash": "^1.0.0"
}
},
"ignore": {
"version": "3.3.10",
"resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.10.tgz",
"integrity": "sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug=="
},
"normalize-path": {
"version": "3.0.0",
"resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
"integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA=="
},
"slash": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/slash/-/slash-1.0.0.tgz",
"integrity": "sha1-xB8vbDn8FtHNF61LXYlhFK5HDVU="
}
}
},
"core-js": {
"version": "3.0.1",
"resolved": "https://registry.npmjs.org/core-js/-/core-js-3.0.1.tgz",
......@@ -5801,6 +5870,14 @@
"minimalistic-crypto-utils": "^1.0.1"
}
},
"hoist-non-react-statics": {
"version": "3.3.0",
"resolved": "https://registry.npmjs.org/hoist-non-react-statics/-/hoist-non-react-statics-3.3.0.tgz",
"integrity": "sha512-0XsbTXxgiaCDYDIWFcwkmerZPSwywfUqYmwT4jzewKTQSWoE6FCMoUVOeBJWK3E/CrWbxRG3m5GzY4lnIwGRBA==",
"requires": {
"react-is": "^16.7.0"
}
},
"hosted-git-info": {
"version": "2.7.1",
"resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.7.1.tgz",
......@@ -10150,6 +10227,29 @@
"resolved": "https://registry.npmjs.org/react-is/-/react-is-16.8.6.tgz",
"integrity": "sha512-aUk3bHfZ2bRSVFFbbeVS4i+lNPZr3/WM5jT2J5omUVV1zzcs1nAaf3l51ctA5FFvCRbhrH0bdAsRRQddFJZPtA=="
},
"react-redux": {
"version": "7.1.0",
"resolved": "https://registry.npmjs.org/react-redux/-/react-redux-7.1.0.tgz",
"integrity": "sha512-hyu/PoFK3vZgdLTg9ozbt7WF3GgX5+Yn3pZm5/96/o4UueXA+zj08aiSC9Mfj2WtD1bvpIb3C5yvskzZySzzaw==",
"requires": {
"@babel/runtime": "^7.4.5",
"hoist-non-react-statics": "^3.3.0",
"invariant": "^2.2.4",
"loose-envify": "^1.4.0",
"prop-types": "^15.7.2",
"react-is": "^16.8.6"
},
"dependencies": {
"@babel/runtime": {
"version": "7.5.4",
"resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.5.4.tgz",
"integrity": "sha512-Na84uwyImZZc3FKf4aUF1tysApzwf3p2yuFBIyBfbzT5glzKTdvYI4KVW4kcgjrzoGUjC7w3YyCHcJKaRxsr2Q==",
"requires": {
"regenerator-runtime": "^0.13.2"
}
}
}
},
"react-scripts": {
"version": "3.0.1",
"resolved": "https://registry.npmjs.org/react-scripts/-/react-scripts-3.0.1.tgz",
......@@ -10269,6 +10369,15 @@
"minimatch": "3.0.4"
}
},
"redux": {
"version": "4.0.4",
"resolved": "https://registry.npmjs.org/redux/-/redux-4.0.4.tgz",
"integrity": "sha512-vKv4WdiJxOWKxK0yRoaK3Y4pxxB0ilzVx6dszU2W8wLxlb2yikRph4iV/ymtdJ6ZxpBLFbyrxklnT5yBbQSl3Q==",
"requires": {
"loose-envify": "^1.4.0",
"symbol-observable": "^1.2.0"
}
},
"regenerate": {
"version": "1.4.0",
"resolved": "https://registry.npmjs.org/regenerate/-/regenerate-1.4.0.tgz",
......@@ -11508,6 +11617,11 @@
"util.promisify": "~1.0.0"
}
},
"symbol-observable": {
"version": "1.2.0",
"resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-1.2.0.tgz",
"integrity": "sha512-e900nM8RRtGhlV36KGEU9k65K3mPb1WV70OdjfxlG2EAuM1noi/E/BaW/uMhL7bPEssK8QV57vN3esixjUvcXQ=="
},
"symbol-tree": {
"version": "3.2.4",
"resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz",
......
......@@ -3,9 +3,15 @@
"version": "0.1.0",
"private": true,
"dependencies": {
"@commander-lol/redux-reducer": "^1.0.1",
"@wasm-tool/wasm-pack-plugin": "^1.0.0",
"copy-webpack-plugin": "^5.0.3",
"react": "^16.8.6",
"react-dom": "^16.8.6",
"react-scripts": "3.0.1"
"react-redux": "^7.1.0",
"react-scripts": "3.0.1",
"redux": "^4.0.4",
"rimraf": "^2.6.3"
},
"scripts": {
"start": "react-scripts start",
......
public/charles-text.png

128 B

public/charles.png

129 B

......@@ -20,6 +20,7 @@
Learn how to configure a non-root public URL by running `npm run build`.
-->
<title>React App</title>
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/normalize/8.0.1/normalize.min.css" integrity="sha256-l85OmPOjvil/SOvVt3HnSSjzF1TUMyT9eV0c2BzEGzU=" crossorigin="anonymous" />
</head>
<body>
<noscript>You need to enable JavaScript to run this app.</noscript>
......
public/is-text.png

128 B

public/push-text.png

128 B

public/stop-text.png

129 B

public/wall-text.png

128 B

public/wall.png 0 → 100644 LFS
public/wall.png

128 B

public/you-text.png

128 B

.App {
text-align: center;
#root {
width: 100%;
height: 100%;
position: relative;
}
.App-logo {
animation: App-logo-spin infinite 20s linear;
height: 40vmin;
pointer-events: none;
.root-container {
margin-left: 0;
transition: margin-left 0.2s linear;
min-width: min-content;
}
.App-header {
background-color: #282c34;
min-height: 100vh;
.level-container {
position: relative;
width: 800px;
height: 800px;
}
.level-tile {
background-color: rgba(12,55,66,0.2);
position: absolute;
z-index: -1;
}
.level-entity {
position: absolute;
z-index: 10;
}
.render-controls {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
font-size: calc(10px + 2vmin);
color: white;
width: 400px;
padding: 10px;
position: absolute;
margin-left: -420px;
transition: margin-left 0.2s linear;
}
.App-link {
color: #61dafb;
.render-controls-row {
display: flex;
height: 100px;
align-items: center;
}
@keyframes App-logo-spin {
from {
transform: rotate(0deg);
}
to {
transform: rotate(360deg);
}
.render-controls-row label{
flex: 1;
}
.render-controls-row input {
flex: 3;
}
\ No newline at end of file
import React from 'react';
import React, {Component} from 'react';
import logo from './logo.svg';
import './App.css';
import Level from './render/Level'
import LevelData from './game/Level'
import testLevel from './data/testlevel1'
function App() {
return (
<div className="App">
<header className="App-header">
<img src={logo} className="App-logo" alt="logo" />
<p>
Edit <code>src/App.js</code> and save to reload.
</p>
<a
className="App-link"
href="https://reactjs.org"
target="_blank"
rel="noopener noreferrer"
>
Learn React
</a>
</header>
</div>
);
class App extends Component<{}> {
state = {
margin: 1,
scale: 1,
showOptions: false,
levelData: null
}
async componentDidMount() {
const levelData = await LevelData.from(testLevel)
levelData.tick()
this.setState({ levelData })
}
render() {
const { margin, scale, showOptions, levelData } = this.state
const opts = {margin, scale}
return (
<React.Fragment>
<div className="render-controls" style={showOptions && {marginLeft: 0} || {}}>
<div className="render-controls-row"><label>Margin: </label><input type="number" value={margin}
onChange={this.set('margin')}/>
</div>
<div className="render-controls-row"><label>Scale: </label><input type="number" value={scale}
onChange={this.set('scale')}
step={0.25}/></div>
</div>
<div className="root-container" style={showOptions && {marginLeft: 420} || {}}>
<button onClick={this.set('showOptions', s => !s.showOptions)}>Show Options</button>
<br/>
{ levelData == null ? <span>LOADING</span> : <Level level={levelData} {...opts} /> }
</div>
</React.Fragment>
);
}
set = (name, value = null) => e => this.setState(s => ({ [name]: value && value.call ? value(s) : value || e.currentTarget.value }))
}
export default App;
{
"id": "push",
"interactions": {
"stop": "stop",
"push": "is-moved"
}
}
\ No newline at end of file
{
"id": "you",
"interactions": {
"push": "is-moved",
"stop": "stop"
}
}
\ No newline at end of file
{
"name": "Ba Ba Test Sheep",
"width": 10,
"height": 10,
"connectors": [
{
"type": "is",
"x": 2,
"y": 5
},
{
"type": "is",
"x": 2,
"y": 2
}
],
"nouns": [
{
"name": "charles",
"images": {
"text": "/charles-text.png",
"entity": "/charles.png"
},
"text_start_locations": [
{
"x": 1,
"y": 5
}
],
"entity_start_locations": [
{
"x": 6,
"y": 5
}
]
},
{
"name": "wall",
"images": {
"text": "/wall-text.png",
"entity": "/wall.png"
},
"text_start_locations": [
{
"x": 1,
"y": 2
}
],
"entity_start_locations": [
{
"x": 0,
"y": 0
},
{
"x": 1,
"y": 0
},
{
"x": 2,
"y": 0
},
{
"x": 3,
"y": 0
},
{
"x": 4,
"y": 0
},
{
"x": 5,
"y": 0
},
{
"x": 6,
"y": 0
},
{
"x": 7,
"y": 0
},
{
"x": 8,
"y": 0
},
{
"x": 9,
"y": 0
}
]
}
],
"adjectives": [
{
"name": "you",
"source": {
"type": "internal",
"data": "adjectives/you"
},
"images": {
"text": "/you-text.png"
},
"text_start_locations": [
{
"x": 3,
"y": 5
}
]
},
{
"name": "push",
"source": {
"type": "internal",
"data": "adjectives/push"
},
"images": {
"text": "/push-text.png"
},
"text_start_locations": [
{
"x": 8,
"y": 8
}
]
},
{
"name": "stop",
"source": {
"type": "inline",
"data": {
"id": "stop",
"interactions": {}
}
},
"images": {
"text": "/stop-text.png"
},
"text_start_locations": [
{
"x": 3,
"y": 2
}
]
}
]
}
\ No newline at end of file
export default class Adjective {
id: string
interactions: { [id: string]: string }
meta = {}
static async resolveInteractionSource(source) {
switch (source.type) {
case "inline": return source.data.interactions
case "internal": {
const data = await import(`../data/${ source.data }`)
if (data.default) {
return data.default.interactions
}
return data.interactions
}
default:
throw TypeError(`Invalid interactions source for ${ JSON.stringify(source) }`)
}
}
constructor(id, interactions) {
this.id = id
this.interactions = interactions
}
interact(other) {
console.log(this.id, this.interactions)
console.log(other.id, other.interactions)
return this.interactions[other.id]
}
}
\ No newline at end of file
// @flow
import typeof Noun from './Noun'
export default class Entity {
x: number
y: number
noun: Noun
fallbackImage: ?string = null
meta = {}
constructor([x, y], noun) {
this.x = x
this.y = y
this.noun = noun
}
get position(): [number, number] {
return [this.x, this.y]
}
get renderProps() {
return {
x: this.x,
y: this.y,
image: this.noun.entityImage || this.fallbackImage,
}
}
onCollide(other: Entity) {
return this.noun.interact(other.noun, this.position, other.position)
}
}
\ No newline at end of file
import Adjective from './Adjective'
import Noun from './Noun'
import Entity from './Entity'
type MapOf<Type> = { [id: string]: Type }
type ConstructorParams = { adjectives: MapOf<Adjective>, nouns: MapOf<Noun>, entities: Entity[], width: number, height: number, name: string, connectors: Object[] }
export const Inputs = {
Controls: {
Up: 'controls_up',
Left: 'controls_left',
Right: 'controls_right',
Down: 'controls_down',
},
Actions: {
Wait: 'action_wait',
},
}
export default class Level {
adjectives: MapOf<Adjective>
nouns: MapOf<Noun>
entities: Entity[]
width: number
height: number
name: string
connectors: Object[]
static async from(source) {
const width = source.width
const height = source.height
const name = source.name || 'A Level'
const entities = []
const nouns = {}
const adjectives = {}
const textNoun = await Noun.createDefaultTextNoun()
const rawNouns = source.nouns || []
for (const src of rawNouns) {
const noun = new Noun(src.images)
nouns[src.name] = noun
if (src.hasOwnProperty('text_start_locations')) {
for (const {x, y} of src.text_start_locations) {
const entity = new Entity([x, y], textNoun)
entity.fallbackImage = noun.textImage
entity.meta = {
type: 'noun',
name: src.name,
}
entities.push(entity)
}
}
if (src.hasOwnProperty('entity_start_locations')) {
for (const {x, y} of src.entity_start_locations) {
const entity = new Entity([x, y], noun)
entities.push(entity)
}
}
}
const rawAdjectives = source.adjectives || []
for (const src of rawAdjectives) {
const interactions = await Adjective.resolveInteractionSource(src.source)
const adjective = new Adjective(src.name, interactions)
adjectives[src.name] = adjective
if (src.hasOwnProperty('text_start_locations')) {
for (const {x, y} of src.text_start_locations) {
const entity = new Entity([x, y], textNoun)
entity.fallbackImage = src.images.text
entity.meta = {
type: 'adjective',
name: src.name,
}
entities.push(entity)
}
}
}
if (source.hasOwnProperty('connectors')) {
for (const { x, y, type: cType } of source.connectors) {
const entity = new Entity([x, y], textNoun)
entity.fallbackImage = `/${ cType }-text.png`
entity.meta = {
type: 'connector',
name: cType,
}
entities.push(entity)
}
}
return new Level({
width,
height,
name,
entities,
nouns,
adjectives,
})
}
constructor(opts: ConstructorParams) {
this.adjectives = opts.adjectives;
this.nouns = opts.nouns;
this.entities = opts.entities;
this.width = opts.width;
this.height = opts.height;
this.name = opts.name;
}
get connectors() {
const connectors = this.findEntities(e => e.meta.type === 'connector')
return connectors.map(c => ({ x: c.x, y: c.y, type: c.meta.name }))
}
findEntities(p) {
return this.entities.filter(p)
}
findEntitiesAt(x, y, p = () => true) {
return this.entities.filter(e => e.x === x && e.y === y && p(e))
}
findEntitiesFor(noun) {
return this.entities.filter(e => e.noun === noun || e.noun.meta.name === noun)
}
findEntitiesThat(adjective) {
if (typeof adjective === 'string') {
return this.entities.filter(e => {
return Array.from(e.noun.adjectives).some(a => a.id === adjective)
})
} else {
return this.entities.filter(e => e.noun.adjectives.has(adjective))
}
}
swapNouns(oldNoun, newNoun) {
this.entities.forEach(entity => {
if (entity.noun === oldNoun) {
entity.noun = newNoun
}
})
}
moveYou({ x = 0, y = 0 } = {}) {
const you = this.adjectives.you
if (you) {
console.log(you)
this.findEntitiesThat(you).forEach(entity => {
entity.x += x
entity.y += y
})
}
}
async processInput(input = Inputs.Actions.Wait) {
switch(input) {
case Inputs.Controls.Up: {
const md = { x: 0, y: -1 }
this.moveYou(md)
return this.tick(md)
}
case Inputs.Controls.Left: {
const md = { x: -1, y: 0 }
this.moveYou(md)
return this.tick(md)
}
case Inputs.Controls.Right: {
const md = { x: 1, y: 0 }
this.moveYou(md)
return this.tick(md)
}
case Inputs.Controls.Down: {
const md = { x: 0, y: 1 }
this.moveYou(md)
return this.tick(md)
}
case Inputs.Actions.Wait: {
return this.tick()
}
}
}
async resolveEntityInteractions(entity, delta) {
const others = this.findEntitiesAt(entity.x, entity.y, e => e !== entity)
if (others.length > 0) {
for (const other of others) {
const result = await entity.onCollide(other)
if (result === 'is-moved' && delta) {
other.x += delta.x
other.y += delta.y
await this.resolveEntityInteractions(other, delta)
} else if (result === 'stop' && delta) {
entity.x -= delta.x
entity.y -= delta.y
await this.resolveEntityInteractions(entity, delta)
}
}
}
}
async tick(delta= null) {
for (const entity of this.entities) {
await this.resolveEntityInteractions(entity, delta)
}
const newDelta = this.connectors.map(connector => {
let leftright = null
const [left] = this.findEntitiesAt(connector.x - 1, connector.y, e => e.meta && e.meta.type)
const [right] = this.findEntitiesAt(connector.x + 1, connector.y, e => e.meta && e.meta.type)
if (left && right) {
leftright = { from: left, to: right }
}
let updown = null
const [up] = this.findEntitiesAt(connector.x, connector.y - 1, e => e.meta && e.meta.type)
const [down] = this.findEntitiesAt(connector.x, connector.y + 1, e => e.meta && e.meta.type)
if (up && down) {
updown = { from: up, to: down }
}
return [leftright, updown].filter(Boolean)
}).reduce((acc, c) => (acc || []).concat(c)).reduce((acc, connection) => {
if (connection.from.meta.type === 'noun') {
const { from, to } = connection
const fromName = from.meta.name
acc[fromName] = acc[fromName] || []
acc[fromName].push(to)
}
return acc
}, {})
Object.values(this.nouns).forEach(noun => noun.clear())
outer: for (const [name, newSet] of Object.entries(newDelta)) {
const noun = this.nouns[name]
if (noun) {
const adjectives = []
for (const updated of newSet) {
if (updated.meta.type === 'noun') {
this.swapNouns(noun, this.nouns[updated.meta.name])
continue outer
} else if (updated.meta.type === 'adjective') {
adjectives.push(this.adjectives[updated.meta.name])
}
}
noun.set(adjectives.filter(Boolean))
}
}
}
}
\ No newline at end of file
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment