1
+ import { css } from '@emotion/react'
1
2
import { Canvas , useFrame , useThree } from '@react-three/fiber'
2
3
import { SMAA , ToneMapping } from '@react-three/postprocessing'
4
+ import { type TilesRenderer as TilesRendererImpl } from '3d-tiles-renderer'
3
5
import {
4
6
GLTFExtensionsPlugin ,
5
7
GoogleCloudAuthPlugin ,
@@ -14,12 +16,19 @@ import {
14
16
TilesPlugin ,
15
17
TilesRenderer
16
18
} from '3d-tiles-renderer/r3f'
17
- import { useAtomValue } from 'jotai'
19
+ import { useAtomValue , useSetAtom } from 'jotai'
18
20
import {
19
21
EffectMaterial ,
20
22
type EffectComposer as EffectComposerImpl
21
23
} from 'postprocessing'
22
- import { Fragment , useLayoutEffect , useRef , type FC } from 'react'
24
+ import {
25
+ Fragment ,
26
+ useEffect ,
27
+ useLayoutEffect ,
28
+ useRef ,
29
+ useState ,
30
+ type FC
31
+ } from 'react'
23
32
import { DRACOLoader } from 'three-stdlib'
24
33
25
34
import { TileCreasedNormalsPlugin } from '@takram/three-3d-tiles-support'
@@ -40,7 +49,7 @@ import {
40
49
41
50
import { EffectComposer } from '../helpers/EffectComposer'
42
51
import { HaldLUT } from '../helpers/HaldLUT'
43
- import { googleMapsApiKeyAtom } from '../helpers/states'
52
+ import { googleMapsApiKeyAtom , needsApiKeyAtom } from '../helpers/states'
44
53
import { Stats } from '../helpers/Stats'
45
54
import { useColorGradingControls } from '../helpers/useColorGradingControls'
46
55
import { useControls } from '../helpers/useControls'
@@ -56,9 +65,26 @@ dracoLoader.setDecoderPath('https://www.gstatic.com/draco/v1/decoders/')
56
65
57
66
const Globe : FC = ( ) => {
58
67
const apiKey = useAtomValue ( googleMapsApiKeyAtom )
68
+
69
+ const [ tiles , setTiles ] = useState < TilesRendererImpl | null > ( null )
70
+ const setNeedsApiKey = useSetAtom ( needsApiKeyAtom )
71
+ useEffect ( ( ) => {
72
+ if ( tiles == null ) {
73
+ return
74
+ }
75
+ const callback = ( ) : void => {
76
+ setNeedsApiKey ( true )
77
+ }
78
+ tiles . addEventListener ( 'load-error' , callback )
79
+ return ( ) => {
80
+ tiles . removeEventListener ( 'load-error' , callback )
81
+ }
82
+ } , [ tiles , setNeedsApiKey ] )
83
+
59
84
return (
60
85
< TilesRenderer
61
86
key = { apiKey } // Reconstruct tiles when API key changes.
87
+ ref = { setTiles }
62
88
>
63
89
{ apiKey !== '' ? (
64
90
< TilesPlugin
@@ -237,11 +263,40 @@ const Scene: FC<SceneProps> = ({
237
263
238
264
export const Story : FC < SceneProps > = props => {
239
265
useGoogleMapsAPIKeyControls ( )
266
+ const needsApiKey = useAtomValue ( needsApiKeyAtom )
240
267
return (
241
- < Canvas gl = { { depth : false } } frameloop = 'demand' >
242
- < Stats />
243
- < Scene { ...props } />
244
- </ Canvas >
268
+ < >
269
+ < Canvas gl = { { depth : false } } frameloop = 'demand' >
270
+ < Stats />
271
+ < Scene { ...props } />
272
+ </ Canvas >
273
+ { needsApiKey && (
274
+ < div
275
+ css = { css `
276
+ position : absolute;
277
+ top : 50% ;
278
+ left : 50% ;
279
+ color : white;
280
+ text-align : center;
281
+ line-height : 1.5 ;
282
+ transform : translate (-50% , -50% );
283
+ ` }
284
+ >
285
+ Our API key has seemingly exceeded its daily quota.
286
+ < br />
287
+ Enter your{ ' ' }
288
+ < a
289
+ href = 'https://developers.google.com/maps/documentation/tile/get-api-key'
290
+ target = '_blank'
291
+ rel = 'noreferrer'
292
+ style = { { color : 'inherit' } }
293
+ >
294
+ Google Maps API key
295
+ </ a > { ' ' }
296
+ at the top right of this screen, or check back tomorrow.
297
+ </ div >
298
+ ) }
299
+ </ >
245
300
)
246
301
}
247
302
0 commit comments