Skip to content

An A-Frame WebXR component for cheaply creating thousands of boxy buildings.

License

Notifications You must be signed in to change notification settings

DougReeder/aframe-shader-buildings

Repository files navigation

aframe-shader-buildings

An A-Frame WebXR component for cheaply creating boxy buildings. Allows you to place thousands of buildings in your scene, without using up your GPU budget!

sample screenshot

live example scene On desktop, drag to turn and use WASD keys to move.

flyable city from Elfland Glider

Usage

Include using

<script src="https://unpkg.com/aframe-shader-buildings@^2.0.0/dist/main.js"></script>

Declaration of a single two-tiered building:

<a-assets>
    <img id="brick" src="./assets/TexturesCom_Wall_BrickPlain1_512_albedo.jpg">
</a-assets>
<a-shader-buildings sun-position="-0.5 1.0 1.0" wall-src="#brick" wall-color="#675342" window-opacity="0.9" buildings=
		'[{"x":5,"z":-995,"xCoreSections":7,"xWingSections":5,"zSections":12,"zWingSections":5,"ySections":30},{"x":0,"z":-1000,"y":120,"xCoreSections":5,"xWingSections":4,"zSections":9,"zWingSections":4,"ySections":30}]'
></a-shader-buildings>

The buildings attribute is JSON, which is problematic in HTML attributes. When creating buildings procedurally, the buildings attribute is set programmatically (see example.html). When hand-crafting buildings, you can set buildings as an HTML attribute. The recommended workaround is to use single quotes around the attribute value, which all modern browsers parse correctly (but an HTML linter will complain about). See the declaration above.

All buildings in the same entity will have the same style, so typically you'll define a number of a-shader-buildings entities, each with different attributes. The primitive varies the intensity of each building, so they look a tiny bit different. Each vertex also has a random tweak to the intensity, so the buildings looks less artificial.

For each building, you'll need to set x, z, y, xCoreSections, xWingSections, zCoreSections, zWingSections, and ySections.

Typically, you'll place the buildings on a flat plateau or valley floor, which may be at any elevation. (All buildings use the same value for y.) Placing a building on a slope requires careful planning to keep windows from being half-buried.

The buildings will look much better if you set wall-src with a seamless square image. They will look better yet, if you set equirectangular or px, nx, etc. with the cubemap of a cubical room with all furniture against the walls. Such cubemaps are hard to find, so if your skills allow, you might want to model a cubical room in your favorite tool and make the cubemap yourself.

Parameters

Typically, you'll set the positions of buildings in the buildings element, and leave the entity x, y, and z as zero, but you don't have to.

elevation-geometry, elevation-material

default: 0

Set these to the same value. Altitude of the base of the buildings.

x-proportion-geometry, x-proportion-material

default: 5

Set these to the same value - this is the length of a section of wall, in meters, that contains one window along the x-axis.

z-proportion-geometry, z-proportion-material

default: 5

Set these to the same value - this is the length of a section of wall, in meters, that contains one window along the z-axis.

y-proportion-geometry, y-proportion-material

default: 4

Set these to the same value - this is the height of a storey in meters.

window-width

default: 0.0 (half the section is window) min: -1.0 max: 1.0

proportion of a section which is window

window-height

default: -0.4 min: -1.0 max: 1.0

proportion of a storey which is window

wall-src

default: none

a reference to a square image for the texture of the walls, preferably seamless. The intensity of the texture will be tweaked so each building looks a bit different.

wall-zoom

default: 2.0 min: 0.001

the number of linear meters that one texture image will cover

wall-color

default: '#909090'

the base color of walls, if wall-src is not set or hasn't loaded yet. should be close to the average color of the wall-src image The intensity of the color will be tweaked so each building looks a bit different.

equirectangular

default: none

reference to equirectangular image of a building interior, for faux-3D windows. Only images of a room that's nearly cubical will look good. Overrides px, nx, etc.

px, nx, py, ny, nz

default: none

references to separate cubemap textures of a building interior, for faux-3D windows. No positive-z image is needed, as that's the window side. Only images of a room that's nearly cubical will look good. Overridden by equirectangular.

window-color

default: 'black'

the base color of windows. Mixed with the image in equirectangular or px, nx, etc.

window-opacity

default: 0.5

how much light the windows block. If neither equirectangular nor px, nx, etc. are set, should be set to about 0.9. Without a cubemap, this will be mixed with white.

sun-position

default: {x:-1.0, y:1.0, z:-1.0}

The direction from which the sun is shining. Any lights in the scene are ignored.

buildings

default: "[]" (no buildings)

JSON string of an array of objects, each object describing a building tier, containing some of the following properties. A building consists of 1 or more tiers. A tier contains 1 or more stories. Currently, all tiers are ell-shaped.

  • x: location of tier, relative to building. Should be a multiple of the x-proportion
  • z: location of tier, relative to building. Should be a multiple of the z-proportion
  • y: location of tier, relative to building. Should be a multiple of the y-proportion
  • xCoreSections: # sections (and thus, windows) in the tier core along the x axis. Add about 0.15 (depending on the windowWidth) to make a wall windowless.
  • xWingSections: # sections (and thus, windows) in the wing along the x axis. Add about 0.15 (depending on the windowWidth) to make a wall windowless.
  • zCoreSections: # sections (and thus, windows) in the tier core along the z axis. Add about 0.15 (depending on the windowWidth) to make a wall windowless.
  • zWingSections: # sections (and thus, windows) in the wing along the z axis. Add about 0.15 (depending on the windowWidth) to make a wall windowless.
  • ySections: # stories in the tier. Add up to about 0.4 to give a tier some attic space.

Development

  1. npm install

  2. edit files

  3. npm run watch

  4. when ready to commit: npm run build

Internals

The geometry of all buildings of a single entity is merged, so there's only one draw call.

Ell-shaped buildings use only 12 triangles per tier, regardless of the number of windows.

The windows aren't part of the geometry nor wall texture - they are added by the shader.

To Do

  • Use a modified version of a Phong shader, for more realism
  • rotate buildings for more variety
  • More building shapes: tee, cee, plus, eee and box, for starters.

Credits

Interior cubemap code by Mosen Heydari https://github.com/mohsenheydari/three-interior-mapping

Hotel room photos by Emil Persson, aka Humus. http://www.humus.name licensed under a Creative Commons Attribution 3.0 Unported License. http://creativecommons.org/licenses/by/3.0/

About

An A-Frame WebXR component for cheaply creating thousands of boxy buildings.

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published