Skip to content

Commit fe94d16

Browse files
committed
Add basic vanilla story for the clouds
1 parent 9885a64 commit fe94d16

File tree

2 files changed

+264
-0
lines changed

2 files changed

+264
-0
lines changed
+262
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,262 @@
1+
import { type StoryFn } from '@storybook/react'
2+
import {
3+
EffectComposer,
4+
EffectPass,
5+
NormalPass,
6+
RenderPass,
7+
ToneMappingEffect,
8+
ToneMappingMode
9+
} from 'postprocessing'
10+
import { useLayoutEffect } from 'react'
11+
import {
12+
Group,
13+
HalfFloatType,
14+
LinearFilter,
15+
LinearMipMapLinearFilter,
16+
Mesh,
17+
MeshBasicMaterial,
18+
NoColorSpace,
19+
NoToneMapping,
20+
PerspectiveCamera,
21+
RedFormat,
22+
RepeatWrapping,
23+
Scene,
24+
TextureLoader,
25+
TorusKnotGeometry,
26+
Vector3,
27+
WebGLRenderer,
28+
type Data3DTexture,
29+
type Texture
30+
} from 'three'
31+
import { OrbitControls } from 'three-stdlib'
32+
import invariant from 'tiny-invariant'
33+
34+
import {
35+
AerialPerspectiveEffect,
36+
getSunDirectionECEF,
37+
PrecomputedTexturesLoader,
38+
type PrecomputedTextures
39+
} from '@takram/three-atmosphere'
40+
import {
41+
CLOUD_SHAPE_DETAIL_TEXTURE_SIZE,
42+
CLOUD_SHAPE_TEXTURE_SIZE,
43+
CloudsEffect,
44+
type CloudsEffectChangeEvent
45+
} from '@takram/three-clouds'
46+
import {
47+
createData3DTextureLoaderClass,
48+
Ellipsoid,
49+
Geodetic,
50+
parseUint8Array,
51+
radians,
52+
STBNLoader
53+
} from '@takram/three-geospatial'
54+
import {
55+
DitheringEffect,
56+
LensFlareEffect
57+
} from '@takram/three-geospatial-effects'
58+
59+
let renderer: WebGLRenderer
60+
let camera: PerspectiveCamera
61+
let controls: OrbitControls
62+
let scene: Scene
63+
let aerialPerspective: AerialPerspectiveEffect
64+
let clouds: CloudsEffect
65+
let composer: EffectComposer
66+
67+
const date = new Date('2000-06-01T10:00:00Z')
68+
const geodetic = new Geodetic(0, radians(67), 500)
69+
const position = geodetic.toECEF()
70+
const up = Ellipsoid.WGS84.getSurfaceNormal(position)
71+
72+
function init(): void {
73+
const container = document.getElementById('container')
74+
invariant(container != null)
75+
76+
const aspect = window.innerWidth / window.innerHeight
77+
camera = new PerspectiveCamera(75, aspect, 10, 1e6)
78+
camera.position.copy(position)
79+
camera.up.copy(up)
80+
81+
controls = new OrbitControls(camera, container)
82+
controls.enableDamping = true
83+
controls.minDistance = 1e3
84+
controls.target.copy(position)
85+
86+
scene = new Scene()
87+
88+
const group = new Group()
89+
Ellipsoid.WGS84.getEastNorthUpFrame(position).decompose(
90+
group.position,
91+
group.quaternion,
92+
group.scale
93+
)
94+
scene.add(group)
95+
96+
const torusKnotGeometry = new TorusKnotGeometry(200, 60, 256, 64)
97+
torusKnotGeometry.computeVertexNormals()
98+
const torusKnot = new Mesh(
99+
torusKnotGeometry,
100+
new MeshBasicMaterial({ color: 'white' })
101+
)
102+
group.add(torusKnot)
103+
104+
// Demonstrates deferred lighting here.
105+
aerialPerspective = new AerialPerspectiveEffect(camera)
106+
aerialPerspective.sky = true
107+
aerialPerspective.sunIrradiance = true
108+
aerialPerspective.skyIrradiance = true
109+
110+
// For the lighting in AerialPerspectiveEffect to work, we must provide a
111+
// normal buffer. Alternatively, this can be sourced from the MRT output.
112+
const normalPass = new NormalPass(scene, camera)
113+
aerialPerspective.normalBuffer = normalPass.texture
114+
115+
clouds = new CloudsEffect(camera)
116+
clouds.coverage = 0.4
117+
clouds.localWeatherVelocity.set(0.001, 0)
118+
clouds.events.addEventListener('change', onCloudsChange)
119+
120+
// Define the direction to the sun.
121+
const sunDirection = new Vector3()
122+
getSunDirectionECEF(date, sunDirection)
123+
aerialPerspective.sunDirection.copy(sunDirection)
124+
clouds.sunDirection.copy(sunDirection)
125+
126+
renderer = new WebGLRenderer({
127+
depth: false,
128+
logarithmicDepthBuffer: false
129+
})
130+
renderer.setPixelRatio(window.devicePixelRatio)
131+
renderer.setSize(window.innerWidth, window.innerHeight)
132+
renderer.toneMapping = NoToneMapping
133+
renderer.toneMappingExposure = 10
134+
135+
// Use floating-point render buffer, as radiance/luminance is stored here.
136+
composer = new EffectComposer(renderer, {
137+
frameBufferType: HalfFloatType,
138+
multisampling: 0
139+
})
140+
composer.addPass(new RenderPass(scene, camera))
141+
composer.addPass(normalPass)
142+
composer.addPass(new EffectPass(camera, clouds, aerialPerspective))
143+
composer.addPass(
144+
new EffectPass(
145+
camera,
146+
new LensFlareEffect(),
147+
new ToneMappingEffect({ mode: ToneMappingMode.AGX }),
148+
new DitheringEffect()
149+
)
150+
)
151+
152+
// Load precomputed textures.
153+
new PrecomputedTexturesLoader().load('atmosphere', onPrecomputedTexturesLoad)
154+
155+
// Load textures for the clouds.
156+
new TextureLoader().load('clouds/local_weather.png', onLocalWeatherLoad)
157+
new (createData3DTextureLoaderClass(parseUint8Array, {
158+
width: CLOUD_SHAPE_TEXTURE_SIZE,
159+
height: CLOUD_SHAPE_TEXTURE_SIZE,
160+
depth: CLOUD_SHAPE_TEXTURE_SIZE
161+
}))().load('clouds/shape.bin', onShapeLoad)
162+
new (createData3DTextureLoaderClass(parseUint8Array, {
163+
width: CLOUD_SHAPE_DETAIL_TEXTURE_SIZE,
164+
height: CLOUD_SHAPE_DETAIL_TEXTURE_SIZE,
165+
depth: CLOUD_SHAPE_DETAIL_TEXTURE_SIZE
166+
}))().load('clouds/shape_detail.bin', onShapeDetailLoad)
167+
new TextureLoader().load('clouds/turbulence.png', onTurbulenceLoad)
168+
new STBNLoader().load('core/stbn.bin', onSTBNLoad)
169+
170+
container.appendChild(renderer.domElement)
171+
window.addEventListener('resize', onWindowResize)
172+
}
173+
174+
function onCloudsChange(event: CloudsEffectChangeEvent): void {
175+
switch (event.property) {
176+
case 'atmosphereOverlay':
177+
aerialPerspective.overlay = clouds.atmosphereOverlay
178+
break
179+
case 'atmosphereShadow':
180+
aerialPerspective.shadow = clouds.atmosphereShadow
181+
break
182+
case 'atmosphereShadowLength':
183+
aerialPerspective.shadowLength = clouds.atmosphereShadowLength
184+
break
185+
}
186+
}
187+
188+
function onPrecomputedTexturesLoad(textures: PrecomputedTextures): void {
189+
Object.assign(aerialPerspective, textures)
190+
Object.assign(clouds, textures)
191+
192+
renderer.setAnimationLoop(render)
193+
}
194+
195+
function onLocalWeatherLoad(texture: Texture): void {
196+
texture.minFilter = LinearMipMapLinearFilter
197+
texture.magFilter = LinearFilter
198+
texture.wrapS = RepeatWrapping
199+
texture.wrapT = RepeatWrapping
200+
texture.colorSpace = NoColorSpace
201+
texture.needsUpdate = true
202+
clouds.localWeatherTexture = texture
203+
}
204+
205+
function onShapeLoad(texture: Data3DTexture): void {
206+
texture.format = RedFormat
207+
texture.minFilter = LinearFilter
208+
texture.magFilter = LinearFilter
209+
texture.wrapS = RepeatWrapping
210+
texture.wrapT = RepeatWrapping
211+
texture.wrapR = RepeatWrapping
212+
texture.colorSpace = NoColorSpace
213+
texture.needsUpdate = true
214+
clouds.shapeTexture = texture
215+
}
216+
217+
function onShapeDetailLoad(texture: Data3DTexture): void {
218+
texture.format = RedFormat
219+
texture.minFilter = LinearFilter
220+
texture.magFilter = LinearFilter
221+
texture.wrapS = RepeatWrapping
222+
texture.wrapT = RepeatWrapping
223+
texture.wrapR = RepeatWrapping
224+
texture.colorSpace = NoColorSpace
225+
texture.needsUpdate = true
226+
clouds.shapeDetailTexture = texture
227+
}
228+
229+
function onTurbulenceLoad(texture: Texture): void {
230+
texture.minFilter = LinearMipMapLinearFilter
231+
texture.magFilter = LinearFilter
232+
texture.wrapS = RepeatWrapping
233+
texture.wrapT = RepeatWrapping
234+
texture.colorSpace = NoColorSpace
235+
texture.needsUpdate = true
236+
clouds.turbulenceTexture = texture
237+
}
238+
239+
function onSTBNLoad(texture: Data3DTexture): void {
240+
aerialPerspective.stbnTexture = texture
241+
clouds.stbnTexture = texture
242+
}
243+
244+
function onWindowResize(): void {
245+
camera.aspect = window.innerWidth / window.innerHeight
246+
camera.updateProjectionMatrix()
247+
renderer.setSize(window.innerWidth, window.innerHeight)
248+
}
249+
250+
function render(): void {
251+
controls.update()
252+
composer.render()
253+
}
254+
255+
const Story: StoryFn = () => {
256+
useLayoutEffect(() => {
257+
init()
258+
}, [])
259+
return <div id='container' />
260+
}
261+
262+
export default Story

storybook/src/clouds/Clouds.stories.tsx

+2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ import { type Meta } from '@storybook/react'
33
import _Basic from './Clouds-Basic'
44
import _CustomLayers from './Clouds-CustomLayers'
55
import _MovingEllipsoid from './Clouds-MovingEllipsoid'
6+
import _Vanilla from './Clouds-Vanilla'
67

78
export default {
89
title: 'clouds/Clouds',
@@ -14,3 +15,4 @@ export default {
1415
export const Basic = _Basic
1516
export const CustomLayers = _CustomLayers
1617
export const MovingEllipsoid = _MovingEllipsoid
18+
export const Vanilla = _Vanilla

0 commit comments

Comments
 (0)