-
Notifications
You must be signed in to change notification settings - Fork 1
/
Parse.hs
133 lines (119 loc) · 5.52 KB
/
Parse.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
-- Copyright 2012 Mitchell Kember.
module Parse () where
import Control.Applicative ((<$>), (<*>))
import Text.JSON
(JSON, JSValue(..), Result(..), fromJSObject, makeObj, readJSON, showJSON)
import qualified Data.Map as M
import Intersect
import Trace
import Vector
lookupM :: (Monad m) => String -> [(String, a)] -> m a
lookupM a as = maybe (fail $ "Missing element: " ++ a) return (lookup a as)
ensure :: (Monad m) => (a -> Bool) -> String -> a -> m a
ensure p s x | p x = return x
| otherwise = fail s
instance (JSON a, Num a) => JSON (VectorT a) where
showJSON (Vector x y z) = showJSON [x, y, z]
readJSON v = case readJSON v of
Ok [x, y, z] -> Ok $ Vector x y z
Ok _ -> fail "Vector is not a 3-element array."
Error s -> fail s
instance JSON Scene where
showJSON (Scene settings world camera objects lights materials) = makeObj
[ ("settings", showJSON settings)
, ("world", showJSON world)
, ("camera", showJSON camera)
, ("objects", showJSON objects)
, ("lights", showJSON lights)
, ("materials", showJSON materials) ]
readJSON (JSObject obj) = Scene
<$> f "settings" <*> f "world" <*> f "camera" <*> f "objects"
<*> f "lights" <*> f "materials" >>= ensure
(\s -> all (`M.member` mMaterials s) $ map mMaterialID (mObjects s))
"Reference to non-existant material-id."
where f x = lookupM x (fromJSObject obj) >>= readJSON
readJSON _ = fail "Scene must be an object."
instance JSON Settings where
showJSON (Settings resolutionX resolutionY samples depth) = makeObj
[ ("resolution-x", showJSON resolutionX)
, ("resolution-y", showJSON resolutionY)
, ("samples", showJSON samples)
, ("depth", showJSON depth) ]
readJSON (JSObject obj) = Settings
<$> (f "resolution-x" >>= ensure (> 0) "resolution-x must be > 0.")
<*> (f "resolution-y" >>= ensure (> 0) "resolution-y must be > 0.")
<*> (f "samples" >>= ensure (>= 0) "samples must be >= 0.")
<*> (f "depth" >>= ensure (> 0) "depth must be > 0.")
where f x = lookupM x (fromJSObject obj) >>= readJSON
readJSON _ = fail "Settings must be an object."
instance JSON World where
showJSON (World sky) = makeObj [ ("sky", showJSON sky) ]
readJSON (JSObject obj) = World <$> f "sky"
where f x = lookupM x (fromJSObject obj) >>= readJSON
readJSON _ = fail "World must be an object."
instance JSON Camera where
showJSON (Orthographic (Ray x v) upward orthoScale) = makeObj
[ ("projection", showJSON "orthographic")
, ("position", showJSON x)
, ("direction", showJSON v)
, ("upward", showJSON upward)
, ("ortho-scale", showJSON orthoScale) ]
showJSON (Perspective (Ray x v) upward focalLength) = makeObj
[ ("projection", showJSON "perspective")
, ("position", showJSON x)
, ("direction", showJSON v)
, ("upward", showJSON upward)
, ("focal-length", showJSON focalLength) ]
readJSON (JSObject obj) = case (f "projection") of
Ok "orthographic" -> Orthographic
<$> (Ray <$> f "position"
<*> fmap normalize (f "direction"))
<*> fmap normalize (f "upward") <*> f "ortho-scale" >>= g
Ok "perspective" -> Perspective
<$> (Ray <$> f "position"
<*> fmap normalize (f "direction"))
<*> fmap normalize (f "upward") <*> f "focal-length" >>= g
Ok _ -> fail "Invalid Camera projection."
Error s -> fail s
where
f x = lookupM x (fromJSObject obj) >>= readJSON
g = ensure (\c -> let (Ray _ d) = mSight c in d <.> mUpward c == 0)
"Camera direction and upward vectors must be orthogonal."
readJSON _ = fail "Camera must be an object."
instance JSON Object where
showJSON (Object surface materialID) = makeObj
[ ("surface", showJSON surface)
, ("material-id", showJSON materialID) ]
readJSON (JSObject obj) = Object <$> f "surface" <*> f "material-id"
where f x = lookupM x (fromJSObject obj) >>= readJSON
readJSON _ = fail "Object must be an object."
instance JSON Surface where
showJSON (Sphere c r) = makeObj
[ ("type", showJSON "sphere")
, ("position", showJSON c)
, ("radius", showJSON r) ]
showJSON (Plane n d) = makeObj
[ ("type", showJSON "plane")
, ("normal", showJSON n)
, ("distance", showJSON d) ]
readJSON (JSObject obj) = case (f "type") of
Ok "sphere" -> Sphere <$> f "position" <*> f "radius"
Ok "plane" -> Plane <$> fmap normalize (f "normal") <*> f "distance"
Ok _ -> fail "Invalid Surface type."
Error s -> fail s
where f x = lookupM x (fromJSObject obj) >>= readJSON
readJSON _ = fail "Surface must be an object."
instance JSON Light where
showJSON (Light position intensity) = makeObj
[ ("position", showJSON position)
, ("intensity", showJSON intensity) ]
readJSON (JSObject obj) = Light <$> f "position" <*> f "intensity"
where f x = lookupM x (fromJSObject obj) >>= readJSON
readJSON _ = fail "Light must be an object."
instance JSON Material where
showJSON (Material diffuse reflect) = makeObj
[ ("diffuse", showJSON diffuse)
, ("reflect", showJSON reflect) ]
readJSON (JSObject obj) = Material <$> f "diffuse" <*> f "reflect"
where f x = lookupM x (fromJSObject obj) >>= readJSON
readJSON _ = fail "Material must be an object."