diff --git a/.gitignore b/.gitignore
index 3ff87f131..a0031ecff 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,3 +1,4 @@
node_modules/
package-lock.json
-.vscode
\ No newline at end of file
+.vscode
+public
\ No newline at end of file
diff --git a/README.md b/README.md
index 320d7c467..c334a3435 100644
--- a/README.md
+++ b/README.md
@@ -1,23 +1,33 @@
-# prismarine-template
-[![NPM version](https://img.shields.io/npm/v/prismarine-template.svg)](http://npmjs.com/package/prismarine-template)
-[![Build Status](https://github.com/PrismarineJS/prismarine-template/workflows/CI/badge.svg)](https://github.com/PrismarineJS/prismarine-template/actions?query=workflow%3A%22CI%22)
+# prismarine-web-client
+[![NPM version](https://img.shields.io/npm/v/prismarine-web-client.svg)](http://npmjs.com/package/prismarine-web-client)
+[![Build Status](https://github.com/PrismarineJS/prismarine-web-client/workflows/CI/badge.svg)](https://github.com/PrismarineJS/prismarine-web-client/actions?query=workflow%3A%22CI%22)
[![Discord](https://img.shields.io/badge/chat-on%20discord-brightgreen.svg)](https://discord.gg/GsEFRM8)
-[![Gitter](https://img.shields.io/badge/chat-on%20gitter-brightgreen.svg)](https://gitter.im/PrismarineJS/general)
-[![Irc](https://img.shields.io/badge/chat-on%20irc-brightgreen.svg)](https://irc.gitter.im/)
-[![Try it on gitpod](https://img.shields.io/badge/try-on%20gitpod-brightgreen.svg)](https://gitpod.io/#https://github.com/PrismarineJS/prismarine-template)
+[![Try it on gitpod](https://img.shields.io/badge/try-on%20gitpod-brightgreen.svg)](https://gitpod.io/#https://github.com/PrismarineJS/prismarine-web-client)
-A template repository to make it easy to create new prismarine repo
+A minecraft client running in a web page.
-## Usage
+It runs mineflayer in the browser which connects to a websocket minecraft server.
+It provides a simple websocket to tcp proxy as a backend to make it possible to connect to any minecraft server.
-```js
-const template = require('prismarine-template')
+## Features
+
+* display blocks
+* display entities as colored rectangles
+* movement sync
+
+## Roadmap
+
+* chat
+* block placing and breaking
-template.helloWorld()
+## Run
+
+```js
+npm install
+npm run build-start
```
-## API
+Then connect to http://localhost:8080
+
-### helloWorld()
-Prints hello world
diff --git a/dns.js b/dns.js
new file mode 100644
index 000000000..be7206ae3
--- /dev/null
+++ b/dns.js
@@ -0,0 +1,39 @@
+/* global XMLHttpRequest */
+
+// Custom DNS resolver made by SiebeDW. Powered by google dns.
+// Supported: SRV (not all errors support)
+module.exports.resolveSrv = function (hostname, callback) {
+ const Http = new XMLHttpRequest()
+ const url = `https://dns.google.com/resolve?name=${hostname}&type=SRV`
+ Http.open('GET', url)
+ Http.responseType = 'json'
+ Http.send()
+
+ Http.onload = function () {
+ const response = Http.response
+ if (response.Status === 3) {
+ const err = new Error('querySrv ENOTFOUND')
+ err.code = 'ENOTFOUND'
+ callback(err)
+ return
+ }
+ if (!response.Answer || response.Answer.length < 1) {
+ const err = new Error('querySrv ENODATA')
+ err.code = 'ENODATA'
+ callback(err)
+ return
+ }
+ const willreturn = []
+ response.Answer.forEach(function (object) {
+ const data = object.data.split(' ')
+ willreturn.push({
+ priority: data[0],
+ weight: data[1],
+ port: data[2],
+ name: data[3]
+ })
+ })
+ console.log(willreturn)
+ callback(null, willreturn)
+ }
+}
diff --git a/example.js b/example.js
deleted file mode 100644
index ceadd46e3..000000000
--- a/example.js
+++ /dev/null
@@ -1,3 +0,0 @@
-const template = require('prismarine-template')
-
-template.helloWorld()
diff --git a/index.html b/index.html
new file mode 100644
index 000000000..beddd6437
--- /dev/null
+++ b/index.html
@@ -0,0 +1,30 @@
+
+
+
+ Prismarine Viewer
+
+
+
+
+
+
diff --git a/index.js b/index.js
index 4bd4c7509..23ee193af 100644
--- a/index.js
+++ b/index.js
@@ -1,9 +1,132 @@
-if (typeof process !== 'undefined' && parseInt(process.versions.node.split('.')[0]) < 14) {
- console.error('Your node version is currently', process.versions.node)
- console.error('Please update it to a version >= 14.x.x from https://nodejs.org/')
- process.exit(1)
-}
+/* global THREE, prompt */
+
+// Workaround for process.versions.node not existing in the browser
+process.versions.node = '14.0.0'
+
+const mineflayer = require('mineflayer')
+const { WorldView, Viewer } = require('prismarine-viewer/viewer')
+global.THREE = require('three')
+
+async function main () {
+ const viewDistance = 6
+ const host = prompt('Host', '95.111.249.143')
+ const port = parseInt(prompt('Port', '10000'))
+ const username = prompt('Username', 'pviewer_person')
+ let password = prompt('Password (blank for offline)')
+ password = password === '' ? undefined : password
+ console.log(`connecting to ${host} ${port} with ${username}`)
+
+ const bot = mineflayer.createBot({
+ host,
+ port,
+ username,
+ password
+ })
+
+ bot.on('end', () => {
+ console.log('disconnected')
+ })
+
+ bot.once('spawn', () => {
+ console.log('bot spawned - starting viewer')
+
+ const version = bot.version
+
+ const center = bot.entity.position
+
+ const worldView = new WorldView(bot.world, viewDistance, center)
+
+ // Create three.js context, add to page
+ const renderer = new THREE.WebGLRenderer()
+ renderer.setPixelRatio(window.devicePixelRatio || 1)
+ renderer.setSize(window.innerWidth, window.innerHeight)
+ document.body.appendChild(renderer.domElement)
+
+ // Create viewer
+ const viewer = new Viewer(renderer)
+ viewer.setVersion(version)
+
+ worldView.listenToBot(bot)
+ worldView.init(bot.entity.position)
+
+ function botPosition () {
+ viewer.setFirstPersonCamera(bot.entity.position, bot.entity.yaw, bot.entity.pitch)
+ worldView.updatePosition(bot.entity.position)
+ }
+
+ bot.on('move', botPosition)
+
+ // Link WorldView and Viewer
+ viewer.listen(worldView)
+ viewer.camera.position.set(center.x, center.y, center.z)
+
+ function moveCallback (e) {
+ bot.entity.pitch -= e.movementY * 0.01
+ bot.entity.yaw -= e.movementX * 0.01
+ viewer.setFirstPersonCamera(bot.entity.position, bot.entity.yaw, bot.entity.pitch)
+ }
+ function changeCallback () {
+ if (document.pointerLockElement === renderer.domElement ||
+ document.mozPointerLockElement === renderer.domElement ||
+ document.webkitPointerLockElement === renderer.domElement) {
+ document.addEventListener('mousemove', moveCallback, false)
+ } else {
+ document.removeEventListener('mousemove', moveCallback, false)
+ }
+ }
+ document.addEventListener('pointerlockchange', changeCallback, false)
+ document.addEventListener('mozpointerlockchange', changeCallback, false)
+ document.addEventListener('webkitpointerlockchange', changeCallback, false)
+ renderer.domElement.requestPointerLock = renderer.domElement.requestPointerLock ||
+ renderer.domElement.mozRequestPointerLock ||
+ renderer.domElement.webkitRequestPointerLock
+ document.addEventListener('mousedown', (e) => {
+ renderer.domElement.requestPointerLock()
+ })
+
+ document.addEventListener('contextmenu', (e) => e.preventDefault(), false)
+ document.addEventListener('keydown', (e) => {
+ console.log(e.code)
+ if (e.code === 'KeyW') {
+ bot.setControlState('forward', true)
+ } else if (e.code === 'KeyS') {
+ bot.setControlState('back', true)
+ } else if (e.code === 'KeyA') {
+ bot.setControlState('right', true)
+ } else if (e.code === 'KeyD') {
+ bot.setControlState('left', true)
+ } else if (e.code === 'Space') {
+ bot.setControlState('jump', true)
+ } else if (e.code === 'ShiftLeft') {
+ bot.setControlState('sneak', true)
+ } else if (e.code === 'ControlLeft') {
+ bot.setControlState('sprint', true)
+ }
+ }, false)
+ document.addEventListener('keyup', (e) => {
+ if (e.code === 'KeyW') {
+ bot.setControlState('forward', false)
+ } else if (e.code === 'KeyS') {
+ bot.setControlState('back', false)
+ } else if (e.code === 'KeyA') {
+ bot.setControlState('right', false)
+ } else if (e.code === 'KeyD') {
+ bot.setControlState('left', false)
+ } else if (e.code === 'Space') {
+ bot.setControlState('jump', false)
+ } else if (e.code === 'ShiftLeft') {
+ bot.setControlState('sneak', false)
+ } else if (e.code === 'ControlLeft') {
+ bot.setControlState('sprint', false)
+ }
+ }, false)
-module.exports.helloWorld = function () {
- console.log('Hello world !')
+ // Browser animation loop
+ const animate = () => {
+ window.requestAnimationFrame(animate)
+ renderer.render(viewer.scene, viewer.camera)
+ }
+ animate()
+ })
}
+main()
diff --git a/package.json b/package.json
index 3f1a05d8b..50f0e3a1c 100644
--- a/package.json
+++ b/package.json
@@ -1,31 +1,50 @@
{
- "name": "prismarine-template",
- "version": "1.0.0",
- "description": "A template repository to make it easy to create new prismarine repo",
- "main": "index.js",
- "scripts": {
- "test": "jest --verbose",
- "pretest": "npm run lint",
- "lint": "standard",
- "fix": "standard --fix"
- },
- "repository": {
- "type": "git",
- "url": "git+https://github.com/PrismarineJS/prismarine-template.git"
- },
- "keywords": [
- "prismarine",
- "template"
- ],
- "author": "Romain Beaumont",
- "license": "MIT",
- "bugs": {
- "url": "https://github.com/PrismarineJS/prismarine-template/issues"
- },
- "homepage": "https://github.com/PrismarineJS/prismarine-template#readme",
- "devDependencies": {
- "jest": "^26.1.0",
- "prismarine-template": "file:.",
- "standard": "^16.0.1"
- }
+ "name": "web_client",
+ "private": true,
+ "version": "1.0.0",
+ "description": "web_client",
+ "main": "index.js",
+ "scripts": {
+ "prepare": "webpack",
+ "start": "webpack serve",
+ "prod-start": "node server.js",
+ "build-start": "npm run prepare && npm run prod-start",
+ "lint": "standard",
+ "fix": "standard --fix",
+ "test": "npm run lint && mocha"
+ },
+ "dependencies": {
+ "assert": "^2.0.0",
+ "browserify-zlib": "^0.2.0",
+ "buffer": "^6.0.3",
+ "clean-webpack-plugin": "^3.0.0",
+ "compression": "^1.7.4",
+ "constants-browserify": "^1.0.0",
+ "copy-webpack-plugin": "^7.0.0",
+ "crypto-browserify": "^3.12.0",
+ "events": "^3.2.0",
+ "express": "^4.17.1",
+ "http-browserify": "^1.7.0",
+ "https-browserify": "^1.0.0",
+ "memfs": "^3.2.0",
+ "mineflayer": "^2.39.2",
+ "net-browserify": "^0.2.4",
+ "os-browserify": "^0.3.0",
+ "path-browserify": "^1.0.1",
+ "prismarine-viewer": "^1.14.0",
+ "process": "^0.11.10",
+ "request": "^2.88.2",
+ "stream-browserify": "^3.0.0",
+ "three": "^0.124.0",
+ "timers-browserify": "^2.0.12",
+ "webpack": "^5.11.0",
+ "webpack-cli": "^4.2.0",
+ "webpack-dev-server": "^3.11.0"
+ },
+ "devDependencies": {
+ "http-server": "^0.12.3",
+ "lodash-webpack-plugin": "^0.11.6",
+ "mocha": "^8.3.0",
+ "standard": "^16.0.3"
+ }
}
diff --git a/perf_hooks_replacement.js b/perf_hooks_replacement.js
new file mode 100644
index 000000000..69b0e2ed5
--- /dev/null
+++ b/perf_hooks_replacement.js
@@ -0,0 +1 @@
+module.exports.performance = window.performance
diff --git a/server.js b/server.js
new file mode 100644
index 000000000..0163072c5
--- /dev/null
+++ b/server.js
@@ -0,0 +1,76 @@
+const express = require('express')
+const netApi = require('net-browserify')
+const bodyParser = require('body-parser')
+const request = require('request')
+const compression = require('compression')
+
+// Create our app
+const app = express()
+
+app.use(function (req, res, next) {
+ res.header('Access-Control-Allow-Origin', req.get('Origin') || '*')
+ res.header('Access-Control-Allow-Credentials', 'true')
+ res.header('Access-Control-Allow-Methods', 'GET,HEAD,PUT,PATCH,POST,DELETE')
+ res.header('Access-Control-Expose-Headers', 'Content-Length')
+ res.header(
+ 'Access-Control-Allow-Headers',
+ 'Accept, Authorization, Content-Type, X-Requested-With, Range'
+ )
+ if (req.method === 'OPTIONS') {
+ return res.send(200)
+ } else {
+ return next()
+ }
+})
+
+app.use(compression())
+app.use(netApi())
+app.use(express.static('./public'))
+
+app.use(bodyParser.json({ limit: '100kb' }))
+
+app.all('*', function (req, res, next) {
+ // Set CORS headers: allow all origins, methods, and headers: you may want to lock this down in a production environment
+ res.header('Access-Control-Allow-Origin', '*')
+ res.header('Access-Control-Allow-Methods', 'GET, PUT, PATCH, POST, DELETE')
+ res.header(
+ 'Access-Control-Allow-Headers',
+ req.header('access-control-request-headers')
+ )
+
+ if (req.method === 'OPTIONS') {
+ // CORS Preflight
+ res.send()
+ } else {
+ const targetURL = req.header('Target-URL')
+ if (!targetURL) {
+ res.status(404).send({ error: '404 Not Found' })
+ return
+ }
+ const newHeaders = req.headers
+ newHeaders.host = targetURL
+ .replace('https://', '')
+ .replace('http://', '')
+ .split('/')[0]
+ request(
+ {
+ url: targetURL + req.url,
+ method: req.method,
+ json: req.body,
+ headers: req.headers
+ },
+ function (error, response, body) {
+ if (error) {
+ console.error(error)
+ console.error('error: ' + response.statusCode)
+ }
+ // console.log(body);
+ }
+ ).pipe(res)
+ }
+})
+
+// Start the server
+const server = app.listen(8080, function () {
+ console.log('Server listening on port ' + server.address().port)
+})
diff --git a/test/basic.test.js b/test/basic.test.js
index 968ec21f8..d461258c8 100644
--- a/test/basic.test.js
+++ b/test/basic.test.js
@@ -1,7 +1,7 @@
-/* eslint-env jest */
+/* eslint-env mocha */
describe('basic', () => {
- test('test', () => {
+ it('test', () => {
})
})
diff --git a/webpack.config.js b/webpack.config.js
new file mode 100644
index 000000000..12b1fa2b4
--- /dev/null
+++ b/webpack.config.js
@@ -0,0 +1,82 @@
+const webpack = require('webpack')
+const path = require('path')
+const LodashModuleReplacementPlugin = require('lodash-webpack-plugin')
+const CopyPlugin = require('copy-webpack-plugin')
+const { CleanWebpackPlugin } = require('clean-webpack-plugin')
+
+const config = {
+ // devtool: 'inline-source-map',
+ // mode: 'development',
+ mode: 'production',
+ entry: path.resolve(__dirname, './index.js'),
+ output: {
+ path: path.resolve(__dirname, './public'),
+ filename: './index.js'
+ },
+ resolve: {
+ alias: {
+ 'minecraft-protocol': path.resolve(
+ __dirname,
+ 'node_modules/minecraft-protocol/src/index.js'
+ ), // Hack to allow creating the client in a browser
+ express: false,
+ net: 'net-browserify',
+ fs: 'memfs'
+ },
+ fallback: {
+ zlib: require.resolve('browserify-zlib'),
+ stream: require.resolve('stream-browserify'),
+ buffer: require.resolve('buffer/'),
+ events: require.resolve('events/'),
+ assert: require.resolve('assert/'),
+ crypto: require.resolve('crypto-browserify'),
+ path: require.resolve('path-browserify'),
+ constants: require.resolve('constants-browserify'),
+ os: require.resolve('os-browserify/browser'),
+ http: require.resolve('http-browserify'),
+ https: require.resolve('https-browserify'),
+ timers: require.resolve('timers-browserify'),
+ // fs: require.resolve("fs-memory/singleton"),
+ child_process: false,
+ perf_hooks: path.resolve(__dirname, 'perf_hooks_replacement.js'),
+ dns: path.resolve(__dirname, 'dns.js')
+ }
+ },
+ plugins: [
+ // fix "process is not defined" error:
+ new CleanWebpackPlugin(),
+ new webpack.ProvidePlugin({
+ process: 'process/browser'
+ }),
+ new webpack.ProvidePlugin({
+ Buffer: ['buffer', 'Buffer']
+ }),
+ new webpack.NormalModuleReplacementPlugin(
+ /prismarine-viewer[/|\\]viewer[/|\\]lib[/|\\]utils/,
+ './utils.web.js'
+ ),
+ new CopyPlugin({
+ patterns: [
+ { from: 'index.html', to: './index.html' },
+ { from: 'node_modules/prismarine-viewer/public/blocksStates/', to: './blocksStates/' },
+ { from: 'node_modules/prismarine-viewer/public/textures/', to: './textures/' },
+ { from: 'node_modules/prismarine-viewer/public/worker.js', to: './' },
+ { from: 'node_modules/prismarine-viewer/public/supportedVersions.json', to: './' }
+ ]
+ }),
+ new webpack.optimize.ModuleConcatenationPlugin(),
+ new LodashModuleReplacementPlugin()
+ ],
+ devServer: {
+ contentBase: path.resolve(__dirname, './public'),
+ compress: true,
+ inline: true,
+ // open: true,
+ hot: true,
+ watchOptions: {
+ ignored: /node_modules/
+ }
+ }
+}
+
+module.exports = config