home *** CD-ROM | disk | FTP | other *** search
/ PC Shareware 1997 June / PC_Shareware-1997-06.iso / programy / genesis / geomdoc.txt < prev    next >
Encoding:
Text File  |  1996-08-17  |  203.0 KB  |  6,250 lines

  1.  
  2. 3
  3.   
  4.             Genesis Programming Specification
  5.  
  6. Author:                              Steven Woodman
  7. Date:                                16/8/96
  8. Revision:                            6
  9.  
  10.  
  11.  
  12.  
  13.  
  14.  
  15.  
  16.  
  17.  
  18.  
  19.  
  20.  
  21.  
  22.  
  23.  
  24.           (C) Silicon Dream Ltd. 1995, 1996
  25.  
  26. TABLE OF CONTENTS
  27. GEOMETRY SPECIFICATION                                  5
  28. INTRODUCTION                                            5
  29.  WHICH COMPILER?                                       6
  30.  PROVIDING A NEW GEOMETRY ENGINE                       6
  31.  WRITING NEW TOOLS                                     6
  32.  USING GEOMETRY FOR APPLICATIONS                       7
  33. HOW THE GEOMETRY API WORKS                              7
  34.  VERTICES                                              8
  35.  PATCHES                                               8
  36.   HOW DO YOU SPECIFY WHICH SIDE IS WHICH               8
  37.   PENETRATING PATCHES                                  8
  38.  NORMALS                                               9
  39.  OBJECTS                                               9
  40.  LIGHTS                                               10
  41.   AMBIENT LIGHT                                       10
  42.  CO-ORDINATE SYSTEMS                                  10
  43.   HOW DO WE DEFINE THE POSITION OF A CO-ORDINATE SYSTEM11
  44.   MATRICES                                            11
  45.   CO-ORDINATE SYSTEM `HANDEDNESS'                     11
  46.  SURFACE TYPES                                        12
  47.   TEXTURES                                            12
  48.     PROJECTED TEXTURES                                12
  49.     WRAPPED TEXTURES                                  13
  50.     TILED TEXTURES                                    13
  51.     RECURSIVE TEXTURES                                13
  52.     BUMP MAPPING                                      13
  53.     TEXTURE ORIENTATION AND SIZE                      14
  54.     NON-LINEAR TEXTURE PROJECTIONS                    14
  55.   SPECULAR HIGHLIGHTS                                 14
  56.   TRANSPARENT SURFACES                                14
  57. USING THE API                                          14
  58. GEOMETRY API REFERENCE                                 15
  59.  USE OF C++ MATHS CLASSES                             15
  60.  GENERAL FUNCTIONS                                    16
  61.  CO-ORDINATE SYSTEM FUNCTIONS                         22
  62.  OBJECT CONSTRUCTION FUNCTIONS                        26
  63.  LIGHTING FUNCTIONS                                   36
  64.  LOADING AND SAVING                                   38
  65.  CALLBACK FUNCTIONS                                   41
  66.  STRUCTURES AND TYPES                                 42
  67. TOOL INTERFACE                                         52
  68. INTRODUCTION                                           52
  69. WRITING CUSTOM TOOLS                                   53
  70.  THE TOOL OBJECT                                      53
  71.  THE TOOL'S CONFIGURATION DIALOG BOX                  53
  72. HOW TO TELL THE EDITOR ABOUT YOUR TOOL                 54
  73. OVERRIDING TOOL FUNCTIONS                              54
  74. CALLING SEQUENCE                                       54
  75.  MODIFYING AND UNDOING                                55
  76.  DRAWING INTO THE VIEW                                55
  77.   DRAWING RETURN CODES                                55
  78.     REDRAW_ALL                                        55
  79.     REDRAW_NONE                                       56
  80.     REDRAW_OBJECT_WIRE                                56
  81.     REDRAW_TOOL                                       56
  82.     REDRAW_NOTOOL                                     56
  83.     REDRAW_REFRESH                                    56
  84.     REDRAW_SHADING                                    56
  85.   GETTING 2D SCREEN CO-ORDINATES                      57
  86. DRAGGING WITH THE MOUSE                                57
  87.  XOR'ING IN MULTIPLE VIEWS                            57
  88.   DRAWSOFAR PARAMETERS                                58
  89. GENERAL GUIDELINES                                     58
  90.  HELP ON USING TOOLS                                  58
  91.  DISPLAYING TEXT STRINGS                              58
  92.  CO-ORDINATE SYSTEM TYPES                             59
  93.  USER DATA                                            59
  94. A NOTE FOR C USERS                                     59
  95. OVERRIDABLES                                           60
  96. SUPPORT                                                66
  97. QUICK START TO WRITING TOOLS                           83
  98. MATHS LIBRARY                                          85
  99. FOR C USERS                                            85
  100. FOR C++ USERS                                          85
  101.  VECTORS                                              86
  102.  LONG VECTORS                                         87
  103.  POLAR VECTORS                                        87
  104.  MATRICES                                             88
  105. DEBUG LIBRARY                                          89
  106. INSTANCES OF DEBUG LIBRARY                             90
  107. WRITING A GEOMETRY ENGINE                              91
  108. ERRORS                                                 91
  109. HANDLES                                                91
  110. UNSUPPORTED FEATURES                                   92
  111. HELPER LIBRARY                                         92
  112. API'S                                                  92
  113. GEOMETRY SPECIFICATION
  114.  
  115. Introduction
  116.  
  117. This part of the spec describes the Geometry API. It
  118. assumes an understanding of programming in C++ and a
  119. basic knowledge of programming in Windows. Knowledge of
  120. mathematics is not a prerequisite although a basic
  121. understanding of vectors and matrices is helpful.
  122.  
  123. Figure 1 shows the various component parts of the Genesis
  124. package and how they relate to one another.
  125.  
  126.  
  127.  
  128.  
  129.  
  130.  
  131.  
  132.  
  133.  
  134.  
  135.  
  136.  
  137.  
  138.  
  139.  
  140.  
  141.  
  142.  
  143.  
  144.  
  145. Figure 1. Components of Genesis
  146.  
  147. Geometry is the part of Genesis that actually stores the
  148. data structures representing the 3D models and performs
  149. all the 3D rendering. The editor is an application that
  150. calls the Geometry API as the user performs various
  151. actions on the program. For instance when the user
  152. presses the `Pos' button on the control bar the editor
  153. makes a call to the SetCamera routine in order to
  154. position the camera ready for rendering. When the user
  155. selects the `Render' menu item, a call to Render is made.
  156. A user will never need to know these details, however
  157. there are three scenarios in which a programmer might
  158. want to understand how the API works.
  159.  
  160.      ·    Providing a new Geometry engine
  161. ·    Writing new tools for the editor
  162. ·    Using Geometry for applications
  163.  
  164. Whichever scenario you are interested in the maths and
  165. debug libraries supplied in this package will prove
  166. invaluable programming aids (See the `Maths Library' and
  167. `Debug Library' sections).
  168.  
  169.  
  170.  
  171. Which compiler?
  172.  
  173. The various Genesis library and import library files were
  174. produced using Microsoft Visual C++. Although any
  175. compiler can be used to compile source code for Genesis,
  176. some linkers might experience problems trying to link to
  177. these lib files. If you experience problems email the
  178. author; spike@silicond.demon.co.uk stating which compiler
  179. you are using and we'll attempt to furnish you with a new
  180. set of libs.
  181.  
  182. Providing a new Geometry engine
  183.  
  184. The default Geometry engine is a single DLL which
  185. provides software for 3D graphics processing. It doesn't
  186. support accelerated hardware because at the time of
  187. writing we couldn't possibly anticipate what manner of 3D
  188. acceleration hardware might become available in the
  189. future. When third parties release their new 3D
  190. processing  systems, whether hardware or software based,
  191. Genesis can support them simply by writing a DLL to
  192. interface the Geometry API to the new hardware/software.
  193. Imagine for example that the XYZ computer company came
  194. out with a board and a software interface that gave PC's
  195. the ability to render at over a million polygons a
  196. second, then with a few days work Genesis with all its
  197. powerful editing tools could be rendering much faster and
  198. completely transparently from the users viewpoint.
  199.  
  200. To create a new Geometry engine you will create a DLL
  201. implementing the Geometry API but which maps those
  202. functions onto the software functions provided by the
  203. third party 3D company. A set of routines is supplied in
  204. this package to make this job easier (See the `Geometry
  205. Helper Library' section).
  206.  
  207. Writing new tools
  208.  
  209. The Genesis editor is very powerful in the respect that
  210. if a function is needed which isn't available, with a
  211. little programming knowledge you can create your own
  212. extensions thereby tailoring Genesis to your own
  213. particular set of problems. For instance, an architect
  214. might spend most of his time creating doors, all of which
  215. are different but all share basic characteristics. By
  216. writing a door creation tool, he can position the cursor
  217. where he wants the door, press a button in the tool box
  218. representing his custom tool, enter a few basic
  219. parameters into a dialog box, and hey presto, a door
  220. appears. In this way users can build their own libraries
  221. of tools all of which are seemlessly integrated into the
  222. program. In this way Genesis can be considered as little
  223. more than a shell designed to support a vast variety of
  224. `plug in' tools to cover every conceivable problem in 3D
  225. design.
  226.  
  227. To create a new tool you will create a DLL implementing a
  228. C++ class derived from a class that we supply (See the
  229. `Writing Tools' section). Don't worry if you are not
  230. familiar with C++ as we provide a C++ template source
  231. file. All you need to do is to fill in the relevant
  232. functions using C. You will use the Geometry API plus
  233. functions in the base class to implement the
  234. functionality of the tool. Minimal knowledge of
  235. programming in Windows is required.
  236.  
  237. Some of the more general tools we thought might be useful
  238. are listed below. Basically anything which manipulates 3D
  239. models can be implemented as a tool. However there are in
  240. fact an unlimited number considering the huge range of
  241. applications Genesis might be applied to:
  242.  
  243. 1.   Primitive creation types (Creates primitive objects)
  244.      ·    Patches
  245.      ·    Spheres/Semi spheres
  246. ·    Cylinders
  247.      ·    Boxes
  248.      ·    Cones
  249. ·    Torus's
  250. ·    Ellipsoids/General curved area creators
  251. 2.   Traditional types (Often found on more conventional
  252.  modellers)
  253.      ·    Lathe (Spins a 2D outline into a 3D shape)
  254.      ·    Extrude (Expands a 2D outline to give it 3D depth)
  255. ·    Copy (Makes a copy of an object)
  256.     ·    Text input window (Calls other tools when you type
  257.       in commands like `sphere rad=10;')
  258. 3.   Savers/Loaders (for various file formats)
  259.      ·    3DS
  260.      ·    DXF
  261. ·    WAD (Doom game file format)
  262. 4.   CSG tools (Enables new objects to be created by
  263.  combining existing ones)
  264.      ·    Union operator (joins two objects as one object;
  265.       a=b|c)
  266.      ·    Intersection operator (creates one object from the
  267.       overlapping volume of two others; a=b&c)
  268.      ·    Subtraction operator (subtracts one object from
  269.       another; a=b-c)
  270.      ·    Exclusive OR operator (creates an object as the
  271.       volume of two other objects excluding any intersection;
  272.       a=b^c)
  273. 5.   Imaginative ideas
  274.      ·    Fractal landscape generator
  275.     ·    `Physical law' animation tools
  276.  
  277. Using Geometry for applications
  278.  
  279. The Geometry engine can be used via its API to construct
  280. new applications totally separate from the Genesis
  281. editor. For instance; Flight simulators, medical imaging
  282. applications, oil field data visualisation, 3D games to
  283. mention but a few. The advantages of using the Geometry
  284. API are twofold. Firstly all the difficult 3D stuff is
  285. already taken care of as well as a lot of other useful
  286. bits such as routines to manipulate 3D vectors and
  287. matrices etc. (See `Maths Library' section). Secondly,
  288. any new hardware appearing for which a geometry engine is
  289. written (see above) will automatically work with the new
  290. application. So your flight simulator might not look too
  291. quick on a slow 486 PC but when XYZ's new 3D processor
  292. board hits the market suddenly without any additional
  293. programming effort your 486 looks like a professional
  294. simulator.
  295.  
  296. To write a new application the Geometry DLL is used as a
  297. stand alone DLL, just like any other, which is called
  298. from your application according to the Geometry API.
  299.  
  300. How the Geometry API works
  301.  
  302. The Geometry design is a hierarchical one. Objects such
  303. as vertices, patches and normals belong to objects.
  304. Objects belong to co-ordinate systems as do lights. Co-
  305. ordinate systems belong to other co-ordinates system. Co-
  306. ordinate system along with all their associated objects
  307. can be rendered. The following sections describe the
  308. terminology further.
  309.  
  310.  
  311.  
  312.  
  313.  
  314.  
  315.  
  316.  
  317.  
  318.  
  319.  
  320.  
  321. Figure 2. Construction of a cube
  322.  
  323. Vertices
  324.  
  325. A vertex is a point in 3D space described by an x, y and
  326. a z value. A cube has 6 vertices or corners, 12 edges,
  327. and 8 faces.
  328.  
  329. Patches
  330.  
  331. A patch is a polygon or face defined by 3 or more
  332. vertices, such as the 8 faces of the cube. You build up
  333. 3D models by connecting patches together exactly like a
  334. patchwork quilt. A patch has a surface type indicator
  335. defining its colour, texture and reflection
  336. characteristics. Patches are always flat but can be made
  337. to appear curved by associating normals with each of its
  338. vertices (see below). Because they are always flat care
  339. should be taken when creating patches of more than 4
  340. vertices that all the vertices lie on a plane otherwise
  341. Geometry will politely inform you of your error.
  342.  
  343. Patches are one sided entities meaning they are intended
  344. to be viewed only from one side. An attempt to view it
  345. from the other side will in fact make it invisible. This
  346. might sound strange but is actually quite logical. For
  347. instance as you look around your cube you are always
  348. seeing the individual patches from the same side, to view
  349. them from the other side you will need to go inside the
  350. cube. If you did this you would actually see straight
  351. through to the outside because the patches would be
  352. invisible. What you need to do is design the inside of
  353. the cube to prevent you seeing out, i.e.  you need 8 more
  354. patches inside facing the other way. The reason for this
  355. is that it makes rendering quite a bit faster. In cases
  356. where you don't want to go inside the cube you only need
  357. 8 single sided patches which render much quicker than 8
  358. double sided ones. This is a technique used by almost all
  359. real time rendering systems.
  360.  
  361. How do you specify which side is which
  362.  
  363. Geometry adopts the convention that all patches must have
  364. their vertices ordered clockwise when viewed from the
  365. correct side. It is the ordering of the vertices which
  366. defines which side is the solid looking side.  When
  367. creating patches you must think about this, if you order
  368. them wrongly your object will be visible from the inside,
  369. and probably completely invisible from outside.
  370.  
  371.  
  372.  
  373.  
  374.  
  375.  
  376.  
  377.  
  378.  
  379.  
  380.  
  381.  
  382.  
  383. Figure 3. Penetrating patches
  384.  
  385. Penetrating patches
  386.  
  387. Care should be taken when constructing objects that one
  388. patch should not penetrate another as not all Geometry
  389. engines support penetrating surfaces. Although this is
  390. not a problem for the default engine it shouldn't be
  391. relied upon. If you want an object to look like it has
  392. been pierced with another the patches should be designed
  393. to give this effect without actually penetrating. The
  394. editor's CSG union tool will allow users to do this.
  395.  
  396.  
  397.  
  398.  
  399.  
  400.  
  401.  
  402.  
  403.  
  404.  
  405.  
  406.  
  407.  
  408.  
  409.  
  410.  
  411. Figure 4. Normals
  412.  
  413. Normals
  414.  
  415. A normal is a direction specified in 3D by an x, y and z
  416. value, rather like a vertex. Normals are used to make
  417. objects appear curved. By associating a normal with each
  418. corner of our cube, we are telling the renderer that when
  419. it is working out the light shading on those points, to
  420. assume that the gradient of the surface at that point is
  421. such the normal's direction sticks out at 90 degrees to
  422. it. In fact this isn't so, as we might have 3 or more
  423. patches converging at a vertex (as in the cube example),
  424. but thanks to the normal the shading is generated as if
  425. it were a single face at that point. If we tell Geometry
  426. that an object we are constructing is to appear curved
  427. the `autosmooth' feature can generate the normal
  428. information automatically for us. There are some instance
  429. however when we would like to create them explicitly.
  430.  
  431.  
  432.  
  433.  
  434.  
  435.  
  436.  
  437.  
  438.  
  439.  
  440.  
  441.  
  442. Figure 5. Missing or wrongly ordered patches
  443.  
  444. Objects
  445.  
  446. An object is a 3D model constructed out of patches. Since
  447. patches are single sided it should be fully enclosed by
  448. patches so that we cannot see the inside of any patch
  449. from any angle. As mentioned above you would in fact not
  450. see it at all giving the appearance of a hole in your
  451. object. Another way to think  about it is that each edge
  452. of every patch should have another patch connected to it.
  453. Geometry will not complain if you attempt to render an
  454. incomplete object but the results can look confusing seen
  455. from some angles.
  456.  
  457. The editor's `Enclosed' tool will highlight any
  458. unconnected edges.
  459.  
  460.  
  461.  
  462. Lights
  463.  
  464. As well as a 3D position, lights can have characteristics
  465. such as intensity, colour and direction although the
  466. default Geometry engine actually only takes notice of the
  467. position and intensity.
  468.  
  469. Ambient Light
  470.  
  471. Real life lighting very rarely leaves you completely in
  472. the dark. This is because there is always light coming
  473. from somewhere, e.g. sunlight, moonlight, a far off
  474. street lamp etc. In 3D modeller environments we find it
  475. difficult to account for all these things, so to ensure
  476. that at least a little bit of light falls even on
  477. surfaces where no light sources can reach, we use ambient
  478. light. The ambient light setting is simply an intensity
  479. value added to those generated by the light sources when
  480. rendering a scene.
  481.  
  482.  
  483.  
  484.  
  485.  
  486.  
  487.  
  488.  
  489.  
  490.  
  491.  
  492.  
  493.  
  494.  
  495.  
  496.  
  497. Figure 6. A co-ordinate system hierarchy
  498.  
  499. Co-ordinate systems
  500.  
  501. Every 3D co-ordinate you specify for lights, vertices,
  502. normals and therefore patches and objects is relative to
  503. an origin, i.e. point 0, 0, 0. The position of the origin
  504. relative to whatever other stuff might get rendered is
  505. defined by the co-ordinate system. All its associated
  506. objects, lights and textures can be moved quickly and
  507. easily relative to everything else simply by moving the
  508. co-ordinate system. The co-ordinate system moves and
  509. everything in it moves. As an example, imagine a huge
  510. landscape. This would be defined in the top level co-
  511. ordinate system by loads of patches. In this landscape
  512. there is a river, and on the river a large boat
  513. containing lots of people. We want to animate this boat
  514. moving down the river. As the boat moves twisting and
  515. turning as it goes, so do all the people.
  516. When creating the animation we have to define the
  517. position of the boat at each frame, but we shouldn't also
  518. have to define the position of every individual person.
  519. We do this by creating a co-ordinate system as a child of
  520. the landscape's co-ordinate system. and make the boat and
  521. everything in it belong it. We then only have to move the
  522. co-ordinate system.
  523.  
  524. Co-ordinate systems are also useful even if we're not
  525. making animation's as it provides us with a new origin
  526. and axis to work with when designing objects at awkward
  527. angles to the x, y and z axis's in the parent co-ordinate
  528. system.
  529.  
  530.  
  531.  
  532. How do we define the position of a co-ordinate system
  533.  
  534. This brings us to the most difficult mathematical concept
  535. used in Geometry. The straight answer is; a 4x4 matrix.
  536. Some of you will know what I mean, some will require a
  537. further explanation.
  538.  
  539.  
  540.  
  541.  
  542.  
  543.  
  544.  
  545.  
  546.  
  547.  
  548. Figure 7. Matrices
  549.  
  550. Matrices
  551.  
  552. A matrix is a set of numbers laid out in a rectangle.
  553. Matrices can have any number of columns by any number of
  554. rows. The matrices we're concerned with in Geometry are
  555. all 4x4. When defining the position of a new co-ordinate
  556. system its not just a matter of specifying the 3D co-
  557. ordinates of a new origin because we also have to define
  558. the `orientation' of the new axis. The `orientation'
  559. means the angle of the new x, y and z axis relative to
  560. the parent co-ordinate system. The x axis isn't
  561. necessarily parallel to the parents x axis it could be
  562. rotated a bit in the y axis and bit in the z axis and
  563. even enlarged or shrunk relative to the parent. Matrices
  564. were designed to represent this type of information.
  565.  
  566. Its not obvious from looking at a matrix what type of
  567. transformations it represents. Thankfully the maths
  568. library supplies a C++ class and a set of C functions
  569. that make working with matrices easy.
  570.  
  571.  
  572.  
  573.  
  574.  
  575. Figure 8. Co-ordinate system `handedness'
  576.  
  577. Co-ordinate system `handedness'
  578.  
  579. Two types of co-ordinate system can be defined; left and
  580. right handed. It is necessary to tell Geometry which type
  581. you have defined otherwise the clockwise ordering of
  582. vertices in a patch is meaningless. To find out which
  583. type you want, position your right hand so that you index
  584. finger and second finger are at right angles to each
  585. other and your thumb is pointing up. In a right handed co-
  586. ordinate system if the x axis has the direction of the
  587. index finger and the y the direction of the second finger
  588. then the z axis will have the direction of the thumb. The
  589. same applies to a left handed co-ordinate system if you
  590. use your left hand.
  591.  
  592. The first co-ordinate system you define will be a child
  593. of HCSYS_TOP. This is a left handed co-ordinate system,
  594. so unless the matrix used to describe the new system
  595. specifically reverses either the x, y or z axis, then
  596. this will also be a left handed system. Geometry cannot
  597. work this out from the matrix, so it needs to be told.
  598.  
  599. When constructing your data you must bear in mind the co-
  600. ordinate system it belongs too. If for example you
  601. designed a model of the earth for a right handed co-
  602. ordinate system but displayed it in a left handed system
  603. then America would end up on the right and Europe and
  604. Asia on the left. No amount of rotating or moving the
  605. object would correct this problem. If something like this
  606. happens to you then look closely at which type of co-
  607. ordinate system you should be using for you data.
  608.  
  609. Surface Types
  610.  
  611. All patches have a surface type. A surface type not only
  612. describes the colour of the surface but its texture and
  613. how highlights appear on the surface. The surface type
  614. does not say whether a patch is curved. This is an
  615. attribute of the patch itself.
  616.  
  617.  
  618.  
  619.  
  620.  
  621.                                    
  622.  
  623. Figure 9. Tiled projected textures  Figure 10. Wrapped
  624.                                     textures
  625.  
  626. Textures
  627.  
  628. Texturing is a way of making your object look more
  629. detailed and realistic by projecting an image onto it.
  630. For instance, if you have a wall you can use a brick
  631. texture or a wood grain. Several methods of applying
  632. texture could be used depending on the underlying
  633. geometry engine. Two are described here. The default
  634. engine uses the second one.
  635.  
  636.  
  637. Projected textures
  638.  
  639. The way to imagine a projected texture is to think of an
  640. invisible image positioned somewhere in your scene.
  641. Although you cannot see the image directly, the plane of
  642. the image is projected infinitely through the scene in
  643. both directions and the image `rubs off' on any patch
  644. which uses the surface type of this texture. Think of it
  645. a bit like a film projector that shines on anything
  646. placed in front of it. The only difference is the film
  647. projector image gets bigger the further in front the
  648. object or screen is moved. Also with a film projector
  649. everything in front gets projected onto, not just
  650. selected patches.
  651.  
  652.  
  653.  
  654.  
  655. Wrapped textures
  656.  
  657. The default engine uses a technique common to many real-
  658. time 3D games. Imagine that the texture is a piece of
  659. Christmas wrapping paper and the object being textured is
  660. a present. Folding the paper (texture) around a simple
  661. object such as a cube, until it is totally covered is a
  662. straight forward process. Suppose however that the object
  663. is a battleship created from tens of thousands of
  664. polygons which are to be wrapped in a rusty metallic
  665. texture. The principal is exactly the same.
  666.  
  667. Both methods have advantages and disadvantages. Wrapping
  668. is generally faster because as long as the texture
  669. doesn't move it can be rendered many times without
  670. needing re-wrapping. You are bound to get discontinuous
  671. edges on some textured objects. The default engine does
  672. its best to eliminate the effect of this on some objects
  673. by carefully selecting how the texture should be folded.
  674. The projection method is slower and although it doesn't
  675. give discontinuous textures at edges it does so at the
  676. expense of distorting the texture. Projection methods do
  677. allow for spherical and cylindrical type texture
  678. projections. Many other methods exist for applying
  679. texture to objects which are not discussed here.
  680.  
  681. Tiled textures
  682.  
  683. If you have a huge wall you want to texture and you have
  684. a small image of part of a brick wall to use as the
  685. texture you have a problem. To make the bitmap fit the
  686. wall you have to scale it up. This is quite easily done
  687. but now you have a huge wall made of a few huge bricks!
  688. What you need to do is make the texture `tiled'. Genesis
  689. can place multiple copies of the image alongside each
  690. other and above and below, like tiling an infinitely
  691. large bathroom wall. Every point in space will then be
  692. within `range' of the texture.
  693.  
  694. If you want to use tiled textures you should ensure that
  695. the image you use is `tileable', in other words, if you
  696. place multiple copies alongside each other, you should
  697. not see the joins. If you can see the joins the wall will
  698. look like you've simply pasted up posters of bricks
  699. rather than used the real thing.
  700.  
  701. If you choose not to tile your texture and you have a
  702. patch using the surface type which is not in range of the
  703. texture, then this patch is coloured using the basic
  704. surface colour of the surface type.
  705.  
  706.  
  707. Recursive textures
  708.  
  709. A recursive texture is where each pixel value in your
  710. image refers not to a  colour in the palette of the
  711. image, but rather to another Genesis surface type. This
  712. means you could define a surface type which appears to
  713. have brick on top and through holes in the brick you can
  714. see wood for instance. You can even define pixel values
  715. to represent real holes in your object that show through
  716. to objects behind. The default Genesis Geometry engine
  717. does not support recursive textures.
  718.  
  719.  
  720. Bump mapping
  721.  
  722. With bump mapping the pixel values in the image refer to
  723. the height of a bump which is to appear at that point on
  724. any patch its projected onto. Obviously no real physical
  725. bump appears on the object, but the shading at the point
  726. is modified to make the surface appear irregular at that
  727. point. The effect is extremely convincing as long as you
  728. don't get too close to the object. The default Geometry
  729. engine does not support bump mapping.
  730.  
  731.  
  732.  
  733.  
  734. Texture orientation and size
  735.  
  736. Having discussed how textures work, we still haven't
  737. addressed the question of how we specify the orientation
  738. of the texture. Its quite simple really, as before we use
  739. a 4x4 matrix to define the orientation relative to the
  740. origin of a co-ordinate system. If you use an identity
  741. matrix (this is really a null matrix, one that does
  742. nothing) then your texture will be aligned with the x and
  743. y axis of the co-ordinate system and will project through
  744. the z. The size will be such that if your image is 100
  745. pixels wide by 80 high then it will extend from the
  746. origin to x=100, and y=80. Any other matrix can be used
  747. to position, rotate, and scale the image away from the
  748. origin. This is true regardless of the method the engine
  749. uses to apply the texture (projected or wrapped etc.).
  750.  
  751.  
  752. Non-linear texture projections
  753.  
  754. Projection texture matrices can also be used to define
  755. non-linear projections. For instance, suppose you want to
  756. project bricks onto a sphere. With a normal linear
  757. projection as described above, your bricks will distort
  758. as they go around the sides of the sphere, like putting a
  759. football in front of a film projector. The way around
  760. this is to use a spherical mapping. In other words create
  761. a matrix to map points on a sphere onto a flat image.
  762. Cylindrical, conic and even toriodal projections might
  763. come in handy for other types of objects. Wrapped
  764. textures will not behave in this way, so this is really
  765. an engine dependent feature.
  766.  
  767.  
  768. Specular highlights
  769.  
  770. Specular reflections are what you see when you take a
  771. shiny surface such as a metal tray, and angle it towards
  772. a light. You will often see brightly lit areas where the
  773. light bounces off the object towards your eye. The exact
  774. shape, size and brightness of these highlights depends on
  775. what the surface is made of. Since the default Geometry
  776. engine (or any using Gouraud as opposed to Phong shading)
  777. performs its lighting calculations at each vertex only,
  778. the more vertices you have the more accurately the
  779. lighting will mimic the surface type. Even if you have a
  780. flat square surface, to get the most visually accurate
  781. highlights the surface should be made of lots of small
  782. flat patches rather than one large one.
  783.  
  784. In technical terms the specular reflection
  785. characteristics relates the amount of light reflected to
  786. the angle of incidence (the angle made by the light
  787. striking a point on the surface and the normal of that
  788. point).
  789.  
  790.  
  791. Transparent surfaces
  792.  
  793. As well as all the above a surface can have associated
  794. with it a transmissive value, in other words its degree
  795. of transparency. A value of 1.0 would make the surface
  796. completely invisible whereas 0 would be completely solid.
  797. The default Geometry engine does not support transparent
  798. surfaces.
  799.  
  800.  
  801. Using the API
  802.  
  803. Custom tool writers have an easy life as the editor does
  804. most things for you. Most of the time you will be using
  805. only a handful of calls specifically to do with create
  806. vertices and patches etc. However a full understanding of
  807. how the API works is useful for when you want to expand
  808. your tool writing to cover more advanced topics. See the
  809. separate section `Writing custom tools'.
  810.  
  811. Application writers using the API will need a full
  812. understanding of the API, as will those developing new
  813. Geometry engines.
  814.  
  815. The first call to the Geometry DLL must be Initialise,
  816. and it can only be called once. The last call must be
  817. Terminate. No furthers calls can be made after a
  818. Terminate. After Initialise the next likely call that a
  819. standalone application is likely to make will probably be
  820. AddCoorSystem, since every object must belong to a co-
  821. ordinate system. If we're writing a tool this would have
  822. been done for us by the editor. As explained co-ordinate
  823. systems are built up in hierarchies. A top level co-
  824. ordinate system has HCSYS_TOP (a constant defined in
  825. `geometry.h') as its parent. Other co-ordinate systems
  826. defined underneath these are known as its children.
  827.  
  828. After creating some patches you can render an image by
  829. placing the camera into one of these co-ordinate systems
  830. using the SetCamera call and then call Render. Again,
  831. custom tools writers wont need to worry about this. When
  832. a co-ordinate system is rendered everything in it, its
  833. child co-ordinate systems and everything in them, and its
  834. parent co-ordinate systems and everything in them are
  835. rendered. The only thing which isn't rendered are other
  836. top level co-ordinate systems. So each top level co-
  837. ordinate system can be considered a completely separate
  838. scene, much like separate documents in a word processor.
  839. The one that gets rendered is the one we SetCamera into.
  840.  
  841. The image gets rendered into a windows Device Independent
  842. Bitamp (DIB) which is created by geometry's Render
  843. function. The application can maintain as many DIBs or
  844. rendered images as it likes. Notice how the editor does
  845. this. It has at least one for each top level co-ordinate
  846. system. When the window's window needs to be painted (on
  847. the WM_PAINT message) with the bitmap, we can call
  848. another Geometry function, UpdateImage, to transfer the
  849. image to the screen.
  850.  
  851. Geometry API Reference
  852.  
  853. To use the Geometry API either from your own application
  854. or from a custom tool, you must include the `geometry.h'
  855. C++ include file and link to `geometry.lib'. All Geometry
  856. functions return a GeomErr value. Usually this will be
  857. GERR_OK (value 0) if the function succeeded otherwise
  858. will either represent an internal processing error such
  859. as `out of memory', or information such as `vertex not
  860. visible'. The calling code should check these return
  861. codes and display any serious internal errors to the user
  862. in a message box. The GetErrorText function can be used
  863. to get a complete text description of the error and
  864. whereabouts internally it occurred. Internal processing
  865. errors more often than not indicate a bug in the calling
  866. code, such as trying to create a patch out of just two
  867. vertices. When your code is debugged they should go away.
  868. If serious errors persist when they shouldn't do we would
  869. be grateful if you could fill in a bug report form and
  870. send it to us, in order for us to correct bugs in
  871. subsequent releases. We will also endeavour to send you
  872. an update as soon as the bug is fixed.
  873.  
  874. Many functions accept handles to objects. It is the
  875. responsibility of the tool/application writers to ensure
  876. that these handles are valid, because although there is a
  877. `invalid handle' error, you cannot rely on the Geometry
  878. engine being able to check its validity.
  879.  
  880.  
  881. Use of C++ maths classes
  882.  
  883. Often a Geometry function will require a C++ class
  884. defined in the maths library as a parameter, a Vec for
  885. instance would be passed to AddVertex. A pure C interface
  886. will soon be made available where instead you will pass a
  887. pointer to a maths `Vector' structure.
  888.  
  889.  
  890.  
  891. General functions
  892.  
  893.  
  894. GeomErr Initialise (void);
  895. Comments
  896.  
  897. Called to initialise Geometry.  Must be called first.
  898.  
  899. Return codes
  900.  
  901. GERR_OUT_OF_MEMORY
  902.  
  903.  
  904.  
  905.  
  906. GeomErr Terminate (void);
  907.  
  908. Comments
  909.  
  910. Called to terminate Geometry and release all its
  911. allocated resource.  Must be called last.
  912.  
  913. Return codes
  914.  
  915.  
  916.  
  917.  
  918. GeomErr DefSurfType (SurfType *pst,
  919.                      bool bReuse,
  920.                      HanSurf *phsur);
  921.  
  922. pst            Pointer to a SurfType structure.
  923.  
  924. bReuse         If TRUE this call will return the handle
  925. of an identical type if one exists.
  926.  
  927. phsur               Returns a handle to a surface type.
  928.  
  929. Comments
  930.  
  931. Creates a surface type definition. See the SurfType
  932. structure definition for more details. The user data
  933. field of the SurfType structure is ignored and is set to
  934. zero by the engine. To set this to another value the
  935. SetUserData call is used. The hsurNext field of the
  936. SurfType structure is also ignored but is used when
  937. querying an existing surface type.
  938.  
  939. Return codes
  940.  
  941. GERR_OUT_OF_MEMORY
  942. GERR_TOO_MANY_SURFACE_TYPES
  943. GERR_BITMAP_FILE_NOT_FOUND
  944. GERR_NOT_A_BMP_FILE
  945.  
  946. GeomErr DelSurfType (HanSurf hsur);
  947.  
  948. hsur           Handle to surface type to delete.
  949.  
  950. Comments
  951.  
  952. Deletes a surface type as long as no patches are using
  953. it.
  954.  
  955. Return codes
  956.  
  957. GERR_INVALID_HANDLE
  958. GERR_IN_USE
  959.  
  960.  
  961.  
  962.  
  963. GeomErr QrySurfType (HanSurf hsur,
  964.                      SurfType *pst,
  965.                      BITMAPINFOHEADER *pbmih,
  966.                      ulong *pulNumPats);
  967.  
  968. hsur          Handle to surface type to query.
  969.  
  970. pst           SurfType structure to receive the
  971.               information about the surface.
  972.  
  973. pbmih         Pointer to windows BITMAPINFOHEADER
  974.               structure which will receive details of
  975.               the texture bitmap if appropriate.
  976.  
  977. pulNumPats    Number of patches using this surface.
  978.  
  979. Comments
  980.  
  981. Queries information about a given surface type.
  982.  
  983. Return codes
  984.  
  985. GERR_INVALID_HANDLE
  986.  
  987.  
  988.  
  989.  
  990. GeomErr ModSurfType (HanSurf hsur,
  991.                       SurfType *pst);
  992.  
  993. hsur          Handle to surface type to modify.
  994.  
  995. pst           SurfType structure used to set surface
  996.               characteristics.
  997.  
  998. Comments
  999.  
  1000. Modifies a surface type.
  1001.  
  1002. The user data field in the SurfType structure is ignored.
  1003. The only way to change the user data of a surface type is
  1004. using the SetUserData call.
  1005.  
  1006. Return codes
  1007.  
  1008. GERR_INVALID_HANDLE
  1009.  
  1010.  
  1011.  
  1012.  
  1013. GeomErr GetFirstSurfType (HanSurf *phsur);
  1014.  
  1015. phsur         Handle of the first surface type defined
  1016.               in geometry.
  1017.  
  1018. Comments
  1019.  
  1020. Returns the handle of the first surface type defined. To
  1021. enumerate all other surface types, query this one and use
  1022. the hsurNext field of the SurfType structure to query the
  1023. next, and so on.
  1024.  
  1025. Return codes
  1026.  
  1027.  
  1028.  
  1029.  
  1030. GeomErr SetCamera (HanCoorSys hcsys,
  1031.                    Vec &vecPos,
  1032.                    Vec &vecDir,
  1033.                    Vec &vecUp,
  1034.                    ViewParams *pvp);
  1035.  
  1036. hcsys               Co-ordinate system to set the camera
  1037. in.
  1038.  
  1039. vecPos              The position of the camera within
  1040. this co-ordinate system.
  1041.  
  1042. vecDir        The direction in which the camera is
  1043.               facing. It takes the form of an absolute
  1044.               3D position within the co-ordinate system.
  1045.  
  1046. vecUp         The direction which will be `up' in the
  1047.               rendered image.  It takes the form of an
  1048.               absolute 3D position within the co-
  1049.               ordinate system.
  1050.  
  1051. pvp           Defines the viewing parameters such as
  1052.               viewing angle and depth of view of the
  1053.               camera.
  1054.  
  1055. Comments
  1056.  
  1057. Sets the camera at a position within the co-ordinate
  1058. system. The camera can only be in one co-ordinate system
  1059. at a time. Setting it in another co-ordinate system will
  1060. remove it from this one. If you need to use Get3DLine and
  1061. Get3DPoint to get 2D screen co-ordinates of 3D points,
  1062. you must SetCamera first into the appropriate co-ordinate
  1063. system.
  1064.  
  1065. Technical note: Apart from defining the viewing
  1066. parameters, and  setting which top level co-ordinate
  1067. system gets rendered, this call is really nothing more
  1068. than a convenient way of modifying the matrix of the top
  1069. level co-ordinate system. Each top level co-ordinate
  1070. system is a scene in its own right. Each is a child of
  1071. HCSYS_TOP which is in fact the camera itself! The Z axis
  1072. of HCSYS_TOP is the axis along which the camera points.
  1073. Setting the camera in any co-ordinate system simply has
  1074. the effect of modifying the matrix of the top level child
  1075. in the scene, such that the camera appears to be at the
  1076. position given within the specified system. In fact the
  1077. camera is not really in this system at all, it is the
  1078. great granddad of all co-ordinate systems! With a clever
  1079. piece of coding you could achieve the same result by
  1080. using ModCoorSys to modify the matrix of the top level
  1081. child in the scene. NB. It is not essential to understand
  1082. this concept in order to program the API.
  1083.  
  1084. Return codes
  1085.  
  1086.  
  1087.  
  1088.  
  1089. GeomErr Render (BITMAPINFO *huge *ppbmi);
  1090.  
  1091. ppbmi               Pointer to a pointer to a windows
  1092. DIB.
  1093.  
  1094. Comments
  1095.  
  1096. Before rendering the *ppbmi value should be set to NULL.
  1097. This will cause Geometry to create a DIB of the size
  1098. specified in the ViewParams structure of the last
  1099. SetCamera call and return a pointer to it in *ppbmi. On
  1100. subsequent renders if we pass in the same value of *ppbmi
  1101. Geometry will reuse the bitmap. If the requested size
  1102. changes on any subsequent SetCamera call or circumstances
  1103. within Geometry change, for instance, implementation
  1104. specific options might allow your Geometry engine to
  1105. switch between 8 and 24 bit output, then the old bitmap
  1106. is freed and a new bitmap created. Each time we pass a
  1107. value of NULL, we force a new bitmap to be created. To
  1108. transfer the image to a device context such as the
  1109. screen, call the UpdateImage function.
  1110.  
  1111. The render bitmaps created by this call are removed by
  1112. Geometry when Terminate is called, however since these
  1113. bitmaps can be very large it is advisable to remove them
  1114. if no longer required by calling DeleteRenBitmap.
  1115.  
  1116. If the render resulted in visible clipping at the front
  1117. clipping plane then GERR_VISIBLE_FRONT_CLIP is returned.
  1118. This should not be interpreted as an error. This is
  1119. useful for applications such as flight simulators where
  1120. the camera, viewpoint, cockpit or whatever you want to
  1121. call it, can crash into objects in the scene. If any
  1122. visible clipping occurs at the front clipping plane then
  1123. this can be considered as crashing into the object. The
  1124. application should be aware however that if you approach
  1125. a large patch and in a single render step move from one
  1126. side to the other, then geometry will not return this
  1127. code. In such cases the application might need to perform
  1128. additional checking.
  1129.  
  1130. The default Geometry engine is very sensitive about the
  1131. `handedness' of the co-ordinate systems being rendered.
  1132. When creating a co-ordinate system it has no way of
  1133. telling if the co-ordinate system you've created is
  1134. really of the type you've told it. Since Geometry's
  1135. renderer relies on this information to optimise its
  1136. performance it is likely to go wrong if say, you created
  1137. a left hand co-ordinate system but told Geometry it is
  1138. right handed. If you experience objects which appear
  1139. inside out when rendering check your co-ordinate systems
  1140. very carefully.
  1141.  
  1142. Return codes
  1143.  
  1144. GERR_IMAGE_SIZE_NOT_MULT_4
  1145. GERR_BITMAP_TOO_SMALL
  1146. GERR_TOO_MANY_BITMAPS
  1147. GERR_VISIBLE_FRONT_CLIP
  1148.  
  1149.  
  1150. GeomErr DeleteRenBitmap (BITMAPINFO huge *pbmi);
  1151.  
  1152. pbmi           A pointer the BITMAPINFO structure created
  1153. by the Render call.
  1154.  
  1155. Comments
  1156.  
  1157. Tells Geometry to remove a bitmap created by the Render
  1158. call.
  1159.  
  1160. Return codes
  1161.  
  1162.  
  1163.  
  1164.  
  1165. GeomErr UpdateImage (ulong ulHDC,
  1166.                      BITMAPINFO huge *pbmi,
  1167.                      ushort usX,
  1168.                      ushort usY);
  1169.  
  1170. ulHDC          A ulong value holding the HDC to paint the
  1171. rendered image to.
  1172.  
  1173. pbmi           Pointer to the rendered DIB (created by
  1174. calling Render).
  1175.  
  1176. usX           Horizontal position in the DC of where the
  1177.               image will appear.
  1178.  
  1179. usY           Vertical position in the DC of where the
  1180.               image will appear.
  1181.  
  1182. Comments
  1183.  
  1184. Transfers a rendered DIB to a device context. This call
  1185. could result in realising the colour palette.
  1186.  
  1187. Return codes
  1188.  
  1189.  
  1190.  
  1191.  
  1192. GeomErr SetUserData (Handle han,
  1193.                      HandleType htype,
  1194.                      ulong ulUser);
  1195.  
  1196. han            Handle of the item to set the user data
  1197. for.
  1198.  
  1199. htype               The type of the `han' handle;
  1200.  
  1201.                  COORSYS_HANDLE
  1202.                  OBJECT_HANDLE
  1203.                  PATCH_HANDLE
  1204.                  VERTEX_HANDLE
  1205.                  NORMAL_HANDLE
  1206.                  SURFACE_HANDLE
  1207.                  LIGHT_HANDLE
  1208.  
  1209. ulUser              The user data value.
  1210.  
  1211. Comments
  1212.  
  1213. This routine is provided so that the application can
  1214. associate instance data with a particular geometry
  1215. object, patch, light etc. This 32 bit data value can be
  1216. used to store a pointer to further data about the object.
  1217. For instance the application might associate a name with
  1218. each object or light. If the item is deleted either
  1219. directly or indirectly (e.g. patches get deleted as a
  1220. result of DelObject) geometry will call the application
  1221. back giving it a chance to remove its instance data.
  1222.  
  1223. Return codes
  1224.  
  1225.  
  1226.  
  1227. GeomErr  SetDelCallback (fnDelCallback *DelCallback);
  1228.  
  1229. DelCallback        The address of a callback function
  1230.               which is called when any geometry item is
  1231.               deleted.
  1232.  
  1233. Comments
  1234.  
  1235. This routine is used to set the address of a callback
  1236. function which is called every time any geometry item
  1237. (e.g. objects, lights, patches etc.) is deleted. The
  1238. application could associate instance data with a geometry
  1239. item which would need removing when the item is deleted.
  1240.  
  1241. The callback function is called before the item is
  1242. deleted so the handle is still valid.
  1243.  
  1244. Return codes
  1245.  
  1246.  
  1247. GeomErr GetErrorText (GeomErr gerr,
  1248.                      char *szBuff,
  1249.                      ushort usBuffSize);
  1250.  
  1251. gerr           The GeomErr value returned from a Geometry
  1252. call.
  1253.  
  1254. szBuff              Pointer to a character buffer to hold
  1255. the error string.
  1256.  
  1257. usBuffSize          Set to indicate the size of szBuff.
  1258.  
  1259. Comments
  1260.  
  1261. Can be called to get Geometry to return an error string
  1262. describing the error which occurred. The string will
  1263. contain the name of the module and the line number on
  1264. which the error occurred.
  1265.  
  1266. Return codes
  1267.  
  1268. GERR_BUFFER_TOO_SMALL
  1269.  
  1270.  
  1271.  
  1272. Co-ordinate system functions
  1273.  
  1274.  
  1275. GeomErr AddCoorSys (HanCoorSys hcsysParent,
  1276.                      Mat &matToParent,
  1277.                      Mat &matFromParent,
  1278.                      ushort usType,
  1279.                      HanCoorSys *phcsys);
  1280.  
  1281. hcsysParent   Handle to parent co-ordinate system
  1282.  
  1283. matToParent   Matrix describing the transformation from
  1284.               child to parent. In other words this
  1285.               matrix transforms a point in the child co-
  1286.               ordinate system to its corresponding
  1287.               position relative to the parent co-
  1288.               ordinate system.
  1289.  
  1290. matFromParent The inverse or opposite matrix to
  1291.               matToParent. NB. The maths library can
  1292.               easily generate the inverse of a matrix by
  1293.               preceding it with a minus sign (e.g. -
  1294.               mat).
  1295.  
  1296. usType        This value indicates whether the co-
  1297.               ordinate system being defined is right
  1298.               handed or left handed. It should be either
  1299.               CT_RIGHTHAND or CT_LEFTHAND.
  1300.  
  1301. phcsys        Returns a handle to a new co-ordinate
  1302.               system.
  1303.  
  1304. Comments
  1305.  
  1306. Creates a co-ordinate system as a child of the parent.
  1307. Use HCSYS_TOP to create a top level co-ordinate system.
  1308. If the matrices do not specifically reverse one of the co-
  1309. ordinate system axis's then this co-ordinate system will
  1310. be the same `handedness' as its parent. It is important
  1311. that the usType parameter correctly identifies the type
  1312. of the co-ordinate system.
  1313.  
  1314. The matrices can define scaling and shearing as well as
  1315. basic orientation. However a matrix is not acceptable if
  1316. it alters the basic topology of the geometry. For
  1317. instance, a mirror type matrix will alter
  1318. the `type' of the co-ordinate system (only use this if
  1319. you also specify the correct usType parameter). A matrix
  1320. that warps space into say, a cone or cylindrical shape is
  1321. completely unacceptable. A matrix consisting entirely of
  1322. zeros, would be the Genesis equivalent of a black hole.
  1323. In space physical laws break down inside black holes.
  1324. Genesis will also break down if you use a matrix like
  1325. this!
  1326.  
  1327. Return codes
  1328.  
  1329. GERR_OUT_OF_MEMORY
  1330.  
  1331.  
  1332.  
  1333.  
  1334. GeomErr DelCoorSys (HanCoorSys hcsys);
  1335.  
  1336. hcsys               Co-ordinate system to delete.
  1337.  
  1338. Comments
  1339.  
  1340. Deletes a co-ordinate system and any objects belonging to
  1341. that system including child co-ordinate systems and their
  1342. objects.
  1343.  
  1344. Return codes
  1345.  
  1346.  
  1347.  
  1348.  
  1349. GeomErr QryCoorSys (HanCoorSys hcsys, CoorSysInfo *pcsi);
  1350.  
  1351. hcsys               Co-ordinate system to query.
  1352.  
  1353. pcsi           Pointer to a CoorSysInfo structure to
  1354. receive the information.
  1355.  
  1356. Comments
  1357.  
  1358. Returns information about a co-ordinate system and all
  1359. its associated objects enabling us to navigate our way
  1360. around the co-ordinate system hierarchy.
  1361.  
  1362. If you query HCSYS_TOP, the only relevant field is
  1363. usType, which will be left-hand.
  1364.  
  1365. If you query any top level co-ordinate systems (i.e.
  1366. direct children of HCSYS_TOP) you cannot rely on the
  1367. matrix fields of the CoorSysInfo structure being what you
  1368. originally set them to. The reason for this apparent
  1369. oddity is that under the covers these matrices define the
  1370. position of the scene relative to the camera, so as the
  1371. camera 'moves', so these matrices change.
  1372.  
  1373. Return codes
  1374.  
  1375.  
  1376.  
  1377.  
  1378. GeomErr ModCoorSys (HanCoorSys hcsys,
  1379.                      Mat &matToParent,
  1380.                      Mat &matFromParent,
  1381.                      ushort usTransType,
  1382.                      ushort usNewType);
  1383.  
  1384. hcsys               Co-ordinate system to query.
  1385.  
  1386. matToParent   Matrix to define the new position of the
  1387.               co-ordinate system relative to its parent.
  1388.               It can be used either to replace the old
  1389.               matrix or to modify it.
  1390.  
  1391. matFromParent Inverse of matToParent. This must be
  1392.               supplied. If it is not known you can
  1393.               always pass in (-matToParent).
  1394.  
  1395. usTransType   Indication of how the new matToParent
  1396.               matrix is to modify the existing one. Can
  1397.               be one of the following values;
  1398.  
  1399.     MCS_TRANS_LAST      The new matrix is applied as;
  1400. old_matrix*new_matrix
  1401.  
  1402.     MCS_TRANS_FRIST     The new matrix is applied as;
  1403. new_matrix*old_matrix
  1404.  
  1405.     MCS_TRANS_REPLACE        The new matrix simply
  1406. replaces the old
  1407.  
  1408. usNewType     The new type of the co-ordinate system as
  1409.               defined by the transformation. Zero can be
  1410.               used if the transformation hasn't changed
  1411.               type.
  1412.  
  1413. Comments
  1414.  
  1415. Modifies the position of a co-ordinate system relative to
  1416. its parent. If the transformation results in the type of
  1417. the co-ordinate system changing (e.g. from right-hand to
  1418. left-hand) then we must tell the engine the new type. If
  1419. the co-ordinate system contains any objects or child co-
  1420. ordinate system then its type cannot be changed.
  1421.  
  1422. If the matrix of a top level child co-ordinate system
  1423. (any child of HCSYS_TOP) is changed and queried later it
  1424. cannot be guaranteed to be the same. The reason for this
  1425. is that the SetCamera call also modifies the matrix the
  1426. top level child of the scene it is set in (see the
  1427. technical note on the SetCamera call).
  1428.  
  1429. Return codes
  1430.  
  1431. GERR_INVALID_OPP_IN_TOP
  1432. GERR_CANT_CHANGE_TYPE
  1433.  
  1434.  
  1435.  
  1436.  
  1437. GeomErr Get3DLine (HanCoorSys hcsys,
  1438.                    Vec &vecA,
  1439.                    Vec &vecB,
  1440.                    float *pfStartX,
  1441.                    float *pfStartY,
  1442.                    float *pfEndX,
  1443.                    float *pfEndY);
  1444.  
  1445. hcsys               Handle of co-ordinate system points
  1446. belong to.
  1447.  
  1448. vecA           Position in co-ordinate system of first
  1449. point.
  1450.  
  1451. vecB           Position in co-ordinate system of second
  1452. point.
  1453.  
  1454. pfStartX       Returns the screen X co-ordinate of start
  1455. of line.
  1456.  
  1457. pfStartY       Returns the screen Y co-ordinate of start
  1458. of line.
  1459.  
  1460. pfEndX         Returns the screen X co-ordinate of end of
  1461. line.
  1462.  
  1463. pfEndY         Returns the screen Y co-ordinate of end of
  1464. line.
  1465.  
  1466. Comments
  1467.  
  1468. Returns the 2D screen co-ordinates of a line segment
  1469. defined by two points; vecA and vecB, within the co-
  1470. ordinate system hcsys. This routine can be used to
  1471. generate lines corresponding to points in a rendered
  1472. image which can then be superimposed on the rendered
  1473. image. This way it makes it look as if Geometry's
  1474. renderer can support 3D lines as well as patches. Of
  1475. course this is not quite true as the lines do not go
  1476. through the hidden surface/line process.
  1477.  
  1478. Unlike the 2D screen co-ordinates returned on calls like
  1479. QryVertex, this routine doesn't require a Render to have
  1480. taken place, however it does require that the camera is
  1481. set in an appropriate co-ordinate system for the point to
  1482. be visible, otherwise GERR_NOT_VISIBLE is returned.
  1483.  
  1484. Return codes
  1485.  
  1486. GERR_NOT_VISIBLE
  1487.  
  1488.  
  1489.  
  1490.  
  1491. GeomErr Get3DPoint (HanCoorSys hcsys,
  1492.                     Vec &vec,
  1493.                     float *pfX,
  1494.                     float *pfY);
  1495.  
  1496. hcsys               Handle of co-ordinate system points
  1497. belong to.
  1498.  
  1499. vec            Position of point in co-ordinate system.
  1500.  
  1501. pfX            Returns the screen X co-ordinate of the
  1502. point.
  1503.  
  1504. pfY            Returns the screen Y co-ordinate of the
  1505. point.
  1506.  
  1507. Comments
  1508.  
  1509. Returns the 2D screen co-ordinates of a point; vec,
  1510. within the co-ordinate system hcsys.
  1511.  
  1512. Unlike the 2D screen co-ordinates returned on calls like
  1513. QryVertex, this routine doesn't require a Render to have
  1514. taken place, however it does require that the camera is
  1515. set in an appropriate co-ordinate system for the point to
  1516. be visible, otherwise GERR_NOT_VISIBLE is returned.
  1517.  
  1518. Return codes
  1519.  
  1520. GERR_NOT_VISIBLE
  1521.  
  1522.  
  1523.  
  1524. Object construction functions
  1525.  
  1526.  
  1527. GeomErr AddObject (HanCoorSys hcsys,
  1528.                    float fNewVertTol,
  1529.                    HanObject *phobj);
  1530.  
  1531. hcsys               Handle to co-ordinate system to add
  1532. object to.
  1533.  
  1534. fNewVertTol         Vertices in this object cannot be
  1535. closer than this value in all of x, y and z.
  1536.  
  1537. phobj               Returns a handle to the new object.
  1538.  
  1539. Comments
  1540.  
  1541. Adds a new object to a co-ordinate system.
  1542.  
  1543. When new vertices are added to this object a check is
  1544. made to see if it is closer than fNewVertTol in x, y and
  1545. z. If it is the vertex is re-used. Notice that
  1546. fNewVertTol is not the distance from the existing
  1547. vertices, this is given by the formula;
  1548. MaxDistFromVert=sqrt((NewVertTol^2)*3).
  1549.  
  1550. Return codes
  1551.  
  1552.  
  1553.  
  1554.  
  1555. GeomErr DelObject (HanObject hobj);
  1556.  
  1557. hobj           Handle to the object being deleted.
  1558.  
  1559. Comments
  1560.  
  1561. Deletes an object along with all its patches, vertices
  1562. and normals from a co-ordinate system.
  1563.  
  1564. Return codes
  1565.  
  1566.  
  1567.  
  1568.  
  1569. GeomErr QryObject (HanObject hobj, ObjectInfo *poi);
  1570.  
  1571. hobj           Handle to the object to query.
  1572.  
  1573. poi            Points to an ObjectInfo structure to
  1574. receive the information.
  1575.  
  1576.  
  1577. Comments
  1578.  
  1579. Returns information on an object.
  1580.  
  1581. Return codes
  1582.  
  1583.  
  1584.  
  1585. GeomErr MoveObject (HanObject hobj, Vec &vec);
  1586.  
  1587. hobj           Handle to the object to move.
  1588.  
  1589. vec            The vector to move the object along.
  1590.  
  1591. Comments
  1592.  
  1593. This function moves an entire object within its co-
  1594. ordinate system. The most efficient way of moving objects
  1595. for instance, during a real-time animation, is to assign
  1596. the object to its own co-ordinate system which is then
  1597. moved relative to its parent. If however an object has to
  1598. be moved within the system, each vertex is turn has to
  1599. moved.
  1600.  
  1601. Although less efficient than moving a co-ordinate system,
  1602. this routine is far more efficient than calling ModVertex
  1603. for each vertex. The reason for this is that each time a
  1604. single vertex is moved the engine has to check if any of
  1605. the patches using it has four or more vertices. If they
  1606. do the chances are that the vertices in these patches no
  1607. longer lie on a plane and these patches might have to be
  1608. split in two to accommodate the move. If every vertex in
  1609. the object moves by the same vector this is not necessary
  1610. so using this function saves on processing and prevents
  1611. patches being split in two.
  1612.  
  1613. Return codes
  1614.  
  1615.  
  1616.  
  1617.  
  1618. GeomErr ScaleObject (HanObject hobj, float fScale);
  1619.  
  1620. hobj           Handle to the object to scale.
  1621.  
  1622. fScale              The factor to scale the object by.
  1623.  
  1624. Comments
  1625.  
  1626. This function scales an entire object within its co-
  1627. ordinate system. The most efficient way of scaling
  1628. objects for instance, during a real-time animation, is to
  1629. assign the object to its own co-ordinate system which is
  1630. then scaled relative to its parent. This routine is
  1631. however more efficient than moving each vertex in the
  1632. object (see MoveObject for more details).
  1633.  
  1634. Return codes
  1635.  
  1636.  
  1637.  
  1638.  
  1639. GeomErr PrtObject (HanObject hobj);
  1640.  
  1641. hobj           Handle to the object to print.
  1642.  
  1643. Comments
  1644.  
  1645. Used purely as a debugging aid for when writing Geometry
  1646. engines. This routine can be called to print out a text
  1647. description of an objects data structures. It can be
  1648. invoked from the editor using a special key combination.
  1649. Unless compiled with the _DEBUG macro definition this
  1650. routine should do nothing.
  1651.  
  1652. Return codes
  1653.  
  1654.  
  1655.  
  1656.  
  1657. GeomErr AddVertex (HanObject hobj,
  1658.                    Vec &vec,
  1659.                    HanVertex *phver);
  1660.  
  1661. hobj           Handle to object to add vertex to.
  1662.  
  1663. vec            3D position of the vertex within the co-
  1664. ordinate system.
  1665.  
  1666. phver               Returns a handle to the vertex.
  1667.  
  1668. Comments
  1669.  
  1670. Creates a new vertex belonging to the object.
  1671.  
  1672. Return codes
  1673.  
  1674.  
  1675.  
  1676.  
  1677. GeomErr DelVertex (HanObject hobj, HanVertex hver);
  1678.  
  1679. hobj           Handle to object we want to delete the
  1680. vertex from.
  1681.  
  1682. hver           Handle to the vertex to be deleted.
  1683.  
  1684. Comments
  1685.  
  1686. Deletes a vertex from an object. The vertex is only
  1687. deleted if no patches are using it.
  1688.  
  1689. Return Codes
  1690.  
  1691. GERR_IN_USE
  1692.  
  1693.  
  1694.  
  1695. GeomErr QryVertex (HanObject hobj,
  1696.                    HanVertex hver,
  1697.                    VertInfo *pvi);
  1698.  
  1699. hobj           Handle to object containing vertex to
  1700. query.
  1701.  
  1702. hver           Handle to the vertex to query.
  1703.  
  1704. pvi            Pointer to a VertInfo structure to receive
  1705. the information.
  1706.  
  1707. Comments
  1708.  
  1709. Returns information about a vertex. The fScrnX and fScrnY
  1710. elements of the VertInfo structure contain the screen x
  1711. and y co-ordinates of this vertex the last time a render
  1712. was performed. If the vertex was not visible for whatever
  1713. reason, e.g. it was out of view, or the last render was
  1714. applied to a different co-ordinate system, then
  1715. GERR_NOT_VISIBLE is returned.
  1716.  
  1717. The default Geometry engine will indicate that the vertex
  1718. was visible if it was within the view and used by a patch
  1719. which was facing the viewpoint even if the vertex was
  1720. obscured by a closer object. If on the other hand the
  1721. vertex was on the back face of an object, it will
  1722. indicate that it was not visible.
  1723.  
  1724. Return codes
  1725.  
  1726. GERR_NOT_VISIBLE
  1727.  
  1728.  
  1729.  
  1730.  
  1731. GeomErr ModVertex (HanObject hobj,
  1732.                     HanVertex hver,
  1733.                     Vec &vec);
  1734.  
  1735. hobj           Handle to object containing vertex to
  1736. modify.
  1737.  
  1738. hver           Handle to vertex to modify.
  1739.  
  1740. vec            New position.
  1741.  
  1742. Comments
  1743.  
  1744. Modifies the position of a vertex. The next time a render
  1745. is performed the vertex will appear in its new position.
  1746. If the new position affects a patch with 4 or more sides,
  1747. such that it no longer lies on a plane, then the patch
  1748. will be split into two or more patches.
  1749.  
  1750. To move or scale an entire object the MoveObject and
  1751. ScaleObject functions should be used.
  1752.  
  1753. Return codes
  1754.  
  1755. GERR_COINCIDENT_POINTS
  1756. GERR_THIN_SEGMENT
  1757. GERR_COMPLEX_OUTLINE
  1758.  
  1759. GeomErr AddNormal (HanObject hobj
  1760.                     Vec &vec,
  1761.                     HanNormal *phnor);
  1762.  
  1763. hobj           Handle to the object to add the normal to.
  1764.  
  1765. vec                3D direction vector of the normal. It
  1766.               specifies a 3D `direction' rather than an
  1767.               absolute point in 3D space.
  1768.  
  1769. phnor               Returns a handle to the new normal.
  1770.  
  1771. Comments
  1772.  
  1773. Adds a normal to an object.
  1774.  
  1775. Return codes
  1776.  
  1777.  
  1778.  
  1779.  
  1780. GeomErr DelNormal (HanObject hobj, HanNormal hnor);
  1781.  
  1782. hobj           Handle to the object containing the normal
  1783. to delete.
  1784.  
  1785. hnor           Handle to the normal to delete.
  1786.  
  1787. Comments
  1788.  
  1789. Deletes a normal from an object as long as no patches are
  1790. using it.
  1791.  
  1792. Return Codes
  1793.  
  1794. GERR_IN_USE
  1795.  
  1796.  
  1797.  
  1798.  
  1799. GeomErr QryNormal (HanObject hobj,
  1800.                     HanNormal hnor,
  1801.                     NormInfo *pni);
  1802.  
  1803. hobj           Handle to the object containing the normal
  1804. to query.
  1805.  
  1806. hnor           Handle to normal to query.
  1807.  
  1808. pni            Pointer to a NormInfo structure to receive
  1809. the information.
  1810.  
  1811. Comments
  1812.  
  1813. Returns information about a normal. Since a single normal
  1814. can be used by vertices from many patches we need to
  1815. supply the handle of a vertex in order to generate the 2D
  1816. screen co-ordinates of the normal when used by that
  1817. vertex. The normal makes a line extending from the vertex
  1818. to a distance of 1 co-ordinate system unit away in the 3D
  1819. direction given when the normal was created. If this line
  1820. or any part of it was visible at the last render, its 2D
  1821. screen co-ordinates are returned, otherwise the return
  1822. code is GERR_NOT_VISIBLE.
  1823.  
  1824. Return Codes
  1825.  
  1826. GERR_NOT_VISIBLE
  1827.  
  1828.  
  1829.  
  1830.  
  1831. GeomErr ModNormal (HanObject hobj,
  1832.                     HanNormal hnor,
  1833.                     Vec &vec);
  1834.  
  1835. hobj           Handle to the object containing the
  1836. normal.
  1837.  
  1838. hnor           Handle to normal to modify.
  1839.  
  1840. vec            New direction vector.
  1841.  
  1842. Comments
  1843.  
  1844. Modifies the direction of a normal vector and therefore
  1845. the surface gradient at all vertices using the normal. At
  1846. the next render the shading at all those vertices using
  1847. the normal will have changed to reflect the new surface
  1848. gradients.
  1849.  
  1850. Return codes
  1851.  
  1852.  
  1853.  
  1854.  
  1855. GeomErr QryEdge (HanObject hobj,
  1856.                  HanEdge hedg,
  1857.                  EdgeInfo *pei);
  1858.  
  1859. hobj           Handle to the object containing the edge.
  1860.  
  1861. hedg           Handle to the edge to be queried.
  1862.  
  1863. pei            Pointer to an EdgeInfo structure to
  1864. receive the information.
  1865.  
  1866. Comments
  1867.  
  1868. Returns information about an edge. Although we do not
  1869. explicitly create the edges, we create vertices and
  1870. patches and Geometry takes care of the edges, it is
  1871. sometimes useful to have access to the edges. Suppose we
  1872. want to draw a wire frame of the object (exactly as the
  1873. editor does). We could iteratively query each patch in
  1874. the object, query the screen positions of each of its
  1875. vertices in turn and draw them. However each edge of each
  1876. patch is also used by another patch meaning each edge
  1877. (and therefore the entire object) will get drawn twice.
  1878. Querying the edges directly avoids this.
  1879.  
  1880. Return codes
  1881.  
  1882.  
  1883.  
  1884. GeomErr DefPatch (HanObject hobj,
  1885.                   ushort usNumEdges,
  1886.                   ushort usFlags,
  1887.                   float fSmoothAng,
  1888.                   HanVertex *ahver,
  1889.                   HanNormal *ahnor,
  1890.                   HanSurf hsur,
  1891.                   ushort *pusNumPats,
  1892.                   HanPatch *ahpat);
  1893.  
  1894. hobj           Handle to object to add patch to.
  1895.  
  1896. usNumEdges    The number of edges, and therefore
  1897.               vertices and normals in the patch. This
  1898.               can be any number from 3 upwards. If the
  1899.               Geometry engine does not support patches
  1900.               of more than say 3 or 4 edges then the
  1901.               patch will be automatically split up into
  1902.               more than one patch resulting in more than
  1903.               one handle being returned.
  1904.  
  1905. usFlags       Flags controlling the creation of the
  1906.               patch. It can be any of the following
  1907.               values combined with C's OR operator
  1908.               (`|');
  1909.  
  1910.  DP_AUTOSMOOTH    Signifies that the patch should
  1911.                    appear curved even although we have
  1912.                    not defined any normals. The normals
  1913.                    should be created automatically by
  1914.                    Geometry by averaging out the plane
  1915.                    normals of all patches converging on
  1916.                    a vertex.
  1917.  
  1918.  DP_SMOOTH_BY_SURFSignifies that autosmoothing is
  1919.                    applied only to patches adjoining
  1920.                    this one if they have the same
  1921.                    surface type.
  1922.  
  1923.  DP_SMOOTH_BY_ANGLE    Signifies that autosmoothing is
  1924.                    applied only to patches adjoining
  1925.                    this one if they angle made at the
  1926.                    join is less than that given by the
  1927.                    fSmoothAng parameter.
  1928.  
  1929.  DP_DONT_VALIDATE Signifies that this patch should not
  1930.                    be validated. If your code is fully
  1931.                    debugged and tested it can use this
  1932.                    flag to say, `I am confident that
  1933.                    this patch definition is OK'. Using
  1934.                    this flag speeds up the creation
  1935.                    process, otherwise the following
  1936.                    checks are performed;
  1937.  
  1938.                     ·     No two points can be in exactly the same spot
  1939.                     ·     All points do not lie along a straight line
  1940. ·     All points lie on a plane (or within a small
  1941. tolerance of)
  1942.  
  1943.  DP_CONVEX        Signifies that the polygon is convex
  1944.                    (has no internal angles greater than
  1945.                    180 degrees). This flag should not be
  1946.                    used if we're not certain. As with
  1947.                    DP_DONT_VALIDATE it can speed up the
  1948.                    creation process, (depending on the
  1949.                    Geometry engine).
  1950.  
  1951.  DP_ATTEMPT_TO_FIX_PLANE Signifies that if the points
  1952.                    are not quite on a plane then we
  1953.                    should split the outline into two or
  1954.                    more planes in order to be able to
  1955.                    create it. If the points do not lie
  1956.                    anywhere near a common plane then
  1957.                    geometry will not be able to discern
  1958.                    any kind of clockwise/anticlockwise
  1959.                    ordering and will still generate a
  1960.                    GERR_NOT_ON_A_PLANE error code.
  1961.  
  1962. fSmoothAng    Specifies the angle to use for
  1963.               autosmoothing expressed in radians. Only
  1964.               takes effect if DP_SMOOTH_BY_ANGLE is
  1965.               used.
  1966.  
  1967. ahver         Array of handles to vertices. The number
  1968.               of vertices is given by the usNumEdges
  1969.               parameter. The vertices must be specified
  1970.               in a clockwise order when viewing the
  1971.               patch from the correct side. No attempt
  1972.               should be made to define a patch using
  1973.               vertices belonging to another object.
  1974.  
  1975. ahnor         Array of handles to normals. The number of
  1976.               normals is given by the usNumEdges
  1977.               parameter. Each normal relates to the
  1978.               corresponding vertex in ahver, e.g.
  1979.               ahnor[0] applies to aver[0] etc. This
  1980.               parameter is ignored if the DP_AUTOSMOOTH
  1981.               flag is given. No attempt should be made
  1982.               to define a patch using normals belonging
  1983.               to another object.
  1984.  
  1985. hsur          Handle to the surface type to be used for
  1986.               the patch.
  1987.  
  1988. pusNumPats    A pointer to a ushort used to return the
  1989.               number of patches actually created (Large
  1990.               concave outlines will almost definitely be
  1991.               split up into more than one patch). Before
  1992.               calling the ushort should be set to the
  1993.               size of the ahpat buffer.
  1994.  
  1995. ahpat         Buffer to hold the handle(s) of the
  1996.               patch(es) created.
  1997.  
  1998. Comments
  1999.  
  2000. Creates a patch out of  3 or more predefined vertices.
  2001. Any number of vertices can be used to define the outline
  2002. of the patch as long as they lie on a plane, do not form
  2003. a straight line or cross over at any point.
  2004.  
  2005. The autosmooth feature takes the worry out of having to
  2006. define normals for curved surfaces. If the patch is
  2007. deleted the vertices and normals defining it are not
  2008. deleted. This is true even if the autosmooth feature was
  2009. used to create the normals automatically. In this case
  2010. the patch should be queried to find the handles of the
  2011. normals it uses before it is deleted and then the normals
  2012. can be deleted using DelNormal.
  2013.  
  2014. The outline may be split into more than one patch
  2015. depending on how many vertices the Geometry engine can
  2016. use in a single patch and whether or not it can support
  2017. concave patch outlines. The default Geometry engine
  2018. supports patches of no more than four vertices and they
  2019. must be convex. An attempt to create a patch with more
  2020. than four vertices, or with a concave outline will result
  2021. in more than one patch being created.
  2022.  
  2023. Return codes
  2024.  
  2025. GERR_COINCIDENT_POINTS
  2026. GERR_THIN_SEGMENT
  2027. GERR_NOT_ON_PLANE
  2028. GERR_OUT_OF_MEMORY
  2029. GERR_NOT_ENOUGH_POINTS
  2030. GERR_COMPLEX_OUTLINE
  2031.  
  2032.  
  2033. GeomErr DelPatch (HanObject hobj, HanPatch hpat);
  2034.  
  2035. hobj           Handle to the object containing the patch
  2036. to delete.
  2037.  
  2038. hpat           Handle to the patch to delete.
  2039.  
  2040. Comments
  2041.  
  2042. Deletes a patch from an object. Deleting a patch does not
  2043. delete any vertices or normals used by the patch. Also be
  2044. aware that even if the autosmooth feature was used to
  2045. automatically create normals for the patch, these will
  2046. not be deleted either. To delete these the patch should
  2047. be queried to find the handles of the normals which can
  2048. then be deleted after the patch.
  2049.  
  2050. Return codes
  2051.  
  2052.  
  2053.  
  2054.  
  2055. GeomErr QryPatch (HanObject hobj,
  2056.                   HanPatch hpat,
  2057.                   PatchInfo *ppi);
  2058.  
  2059. hobj           Handle to the object containing the patch
  2060. to query.
  2061.  
  2062. hpat           Handle to the patch to query.
  2063.  
  2064. ppi           Pointer to a PatchInfo structure to hold
  2065.               the information.
  2066.  
  2067. Comments
  2068.  
  2069. Returns information about a patch. The ahver and ahnor
  2070. elements of the PatchInfo structure should point to
  2071. buffers to receive the handles of the vertices and
  2072. normals and the usNumEdges element should indicate how
  2073. many handles there is room for in the buffers. NB. No
  2074. Geometry engine implementation should have more than
  2075. MAX_PATCH_EDGES edges in any patch, so it is a good idea
  2076. to set usNumEdges to this. The ahver and ahnor fields can
  2077. be set to NULL if we're not interested in the handles. If
  2078. GERR_BUFFER_TOO_SMALL is returned the usNumEdges field of
  2079. the PatchInfo structure will contain the size required.
  2080.  
  2081. Return codes
  2082.  
  2083. GERR_BUFFER_TOO_SMALL
  2084.  
  2085.  
  2086.  
  2087.  
  2088. GeomErr QryVertPatch (ulong *pulRef, HanPatch *phpat);
  2089.  
  2090. pulRef        Pointer to a ulong holding a reference to
  2091.               a patch. This value is returned by the
  2092.               ulRef element of the VertInfo structure on
  2093.               a call to QryVertex.
  2094.  
  2095. phpat               Returns a handle to the next patch
  2096. using this vertex.
  2097.  
  2098. Comments
  2099.  
  2100. To query which patches use a particular vertex, first
  2101. query the vertex, and then pass the ulRef value to
  2102. QryVertPatch. This will return a handle to the first
  2103. patch using that vertex. Subsequent calls to QryVertPatch
  2104. will return other patches until a NULL_HANDLE is
  2105. returned.
  2106.  
  2107. Return codes
  2108.  
  2109.  
  2110.  
  2111.  
  2112. GeomErr ModPatchSurf (HanObject hobj,
  2113.                       HanPatch hpat,
  2114.                       HanSurf hsur)
  2115.  
  2116. hobj           The object the patch belongs to.
  2117.  
  2118. hpat          The patch to change.
  2119.  
  2120. hsur           The surface type to colour the patch with.
  2121.  
  2122. Comments
  2123.  
  2124. Changes the surface type used by a patch.
  2125.  
  2126. Return codes
  2127.  
  2128.  
  2129.  
  2130. Lighting functions
  2131.  
  2132.  
  2133. GeomErr AddLight (HanCoorSys hcsys,
  2134.                   LightDef *pld,
  2135.                   HanLight *phli);
  2136.  
  2137. hcsys               Handle to co-ordinate system to add
  2138. light to.
  2139.  
  2140. pld            Points to a LightDef structure containing
  2141. details of the light.
  2142.  
  2143. phli           Returns a handle to a new light.
  2144.  
  2145. Comments
  2146.  
  2147. Creates a new light belonging to the co-ordinate system.
  2148. The user data field of the LightDef structure is ignored
  2149. and is set to zero by the engine. To set this to another
  2150. value the SetUserData call is used. The hliNext field of
  2151. the LightDef structure is also ignored but is used when
  2152. querying an existing light.
  2153.  
  2154. Return codes
  2155.  
  2156.  
  2157.  
  2158.  
  2159. GeomErr DelLight (HanLight hli);
  2160.  
  2161. hli            Handle to the light to be deleted.
  2162.  
  2163. Comments
  2164.  
  2165. Deletes a light from the co-ordinate system.
  2166.  
  2167. Return Codes
  2168.  
  2169.  
  2170.  
  2171.  
  2172. GeomErr QryLight (HanLight hli, LightDef *pld);
  2173.  
  2174. hli            Handle to light to query.
  2175.  
  2176. pld            Pointer to a LightDef structure to receive
  2177. the information.
  2178.  
  2179. Comments
  2180.  
  2181. Returns information about a light.
  2182.  
  2183. Return codes
  2184.  
  2185.  
  2186.  
  2187.  
  2188. GeomErr ModLight (HanLight hli, LightDef *pld);
  2189.  
  2190. hli            Handle to light to modify.
  2191.  
  2192. pld            Pointer to LightDef structure containing
  2193. the new parameters.
  2194.  
  2195. Comments
  2196.  
  2197. Modifies the characteristics of a light, including
  2198. possibly its position.
  2199.  
  2200. The user data field in the LightDef structure is ignored.
  2201. The only way to change the user data of a light is using
  2202. the SetUserData call.
  2203.  
  2204. Return codes
  2205.  
  2206.  
  2207.  
  2208.  
  2209. GeomErr SetAmbient (float fInt);
  2210.  
  2211. fInt           Ambient light intensity.
  2212.  
  2213. Comments
  2214.  
  2215. Sets the intensity of the ambient light in a scene. The
  2216. ambient value applies to all co-ordinates systems.
  2217.  
  2218. Return codes
  2219.  
  2220. GERR_INVALID_LIGHT_INT
  2221.  
  2222.  
  2223.  
  2224.  
  2225.  
  2226.  
  2227. Loading and Saving
  2228.  
  2229.  
  2230. GeomErr SaveScene (HFILE hfile,
  2231.                    HanCoorSys hcsys,
  2232.                    ulong *pulNumCSys,
  2233.                    HanCoorSys *ahcsys,
  2234.                    ulong *pulNumObjs,
  2235.                    HanObject *ahobj,
  2236.                    fnLoadSaveCallback *Report);
  2237.  
  2238. hfile         Handle of a file to save to.
  2239.  
  2240. hcsys         Co-ordinate system to save.
  2241.  
  2242. pulNumCSys    Indicates the size of the ahcsys buffer in
  2243.               terms of how many handles it can hold. On
  2244.               return this value is modified to the total
  2245.               number of co-ordinate systems saved, and
  2246.               therefore the number of handles in the
  2247.               ahcsys array.
  2248.  
  2249. ahcsys        An array which on return contains the
  2250.               handles of all the co-ordinate systems
  2251.               saved in the order in which they were
  2252.               saved in the file. If you are not
  2253.               interested in the handles, this parameter
  2254.               can be NULL if we set *pulNumCSys to zero.
  2255.  
  2256. pulNumObjs    Indicates the size of the ahobj buffer in
  2257.               terms of how many handles it can hold. On
  2258.               return this value is modified to the total
  2259.               number of objects saved, and therefore the
  2260.               number of handles in the ahobj array.
  2261.  
  2262. ahobj         An array which on return contains the
  2263.               handles of all the objects saved in the
  2264.               order in which they were saved in the
  2265.               file. If you are not interested in the
  2266.               handles, this parameter can be NULL if we
  2267.               set *pulNumObjs to zero.
  2268.  
  2269. Report         The address of a callback function to
  2270.                inform the application of how far through
  2271.                the load its done (see the
  2272.                fnLoadSaveCallback function).
  2273.  
  2274. Comments
  2275.  
  2276. Saves a co-ordinate system to the file given by hfile.
  2277. All objects and lights belonging to this co-ordinate
  2278. system will be saved including any children and all their
  2279. objects. The last four parameters will return the
  2280. handles of all co-ordinate systems and objects saved.
  2281. This is very useful for applications like the editor. The
  2282. editor associates a name with each co-ordinate system and
  2283. object. These names are unknown to Geometry and so will
  2284. not get saved. However the editor can save the names
  2285. associated with each handle after the main Geometry
  2286. information in the file. When the file gets loaded again
  2287. by LoadScene we get told the new handles which we then
  2288. simply have to associate with the names in the file.
  2289.  
  2290. Return codes
  2291.  
  2292.  
  2293.  
  2294.  
  2295. GeomErr LoadScene (HFILE hfile,
  2296.                    HanCoorSys hcsys,
  2297.                    Char *szTexPath,
  2298.                    ulong *pulNumCSys,
  2299.                    HanCoorSys *ahcsys,
  2300.                    ulong *pulNumObjs,
  2301.                    HanObject *ahobj,
  2302.                    fnLoadSaveCallback *Report);
  2303.  
  2304. hfile         Handle of a file to load from.
  2305.  
  2306. hcsys         Co-ordinate system to load under.
  2307.  
  2308. szTexPath     A pointer to a path specification for
  2309.               where to search for texture bitmaps if
  2310.               they are not found in the current working
  2311.               directory. This would typically be set to
  2312.               the directory where the model file is, or
  2313.               else a special texture directory. Any
  2314.               number of paths can be seperated by semi
  2315.               colons, but each must have a terminating
  2316.               back slash e.g.;
  2317.               "c:\genesis\textures\;c:\windows\bmps\" or
  2318.               else be a null string. The pointer cannot
  2319.               be NULL.
  2320.  
  2321. pulNumCSys    Indicates the size of the ahcsys buffer in
  2322.               terms of how many handles it can hold. On
  2323.               return this value is modified to the total
  2324.               number of co-ordinate systems loaded, and
  2325.               therefore the number of handles in the
  2326.               ahcsys array.
  2327.  
  2328. ahcsys        An array which on return contains the
  2329.               handles of all the co-ordinate systems
  2330.               loaded in the order in which they existed
  2331.               in the file. If you are not interested in
  2332.               the handles, this parameter can be NULL if
  2333.               we set *pulNumCSys to zero.
  2334.  
  2335. pulNumObjs    Indicates the size of the ahobj buffer in
  2336.               terms of how many handles it can hold. On
  2337.               return this value is modified to the total
  2338.               number of objects loaded, and therefore
  2339.               the number of handles in the ahobj array.
  2340.  
  2341. ahobj         An array which on return contains the
  2342.               handles of all the objects loaded in the
  2343.               order in which they existed in the file.
  2344.               If you are not interested in the handles,
  2345.               this parameter can be NULL if we set
  2346.               *pulNumObjs to zero.
  2347.  
  2348. Report         The address of a callback function to
  2349.                inform the application of how far through
  2350.                the load its done (see the
  2351.                fnLoadSaveCallback function).
  2352. Comments
  2353.  
  2354. Loads a co-ordinate system from the file given by hfile.
  2355. All objects and lights belonging to this co-ordinate
  2356. system will be loaded including any children and all
  2357. their objects. The top level co-ordinate system in the
  2358. file will become a child of hcsys. To load an entirely
  2359. new scene hcsys should be set to HCSYS_TOP. For a fuller
  2360. description of the last four parameters see SaveScene.
  2361.  
  2362. Return codes
  2363.  
  2364. GERR_INVALID_GEN_FILE
  2365.  
  2366.  
  2367.  
  2368. GeomErr LoadLWObject (HFILE hfile,
  2369.                        HanCoorSys hcsys,
  2370.                        char *szTexPath,
  2371.                        HanObject *phobj,
  2372.                        fnLoadSaveCallback *Report);
  2373.  
  2374. hfile         Windows handle of the file to load from.
  2375.  
  2376. hcsys         Handle of the co-ordinate system the
  2377.               object will be added to.
  2378.  
  2379. szTexPath     A pointer to a path specification for
  2380.               where to search for texture bitmaps if
  2381.               they are not found in the current working
  2382.               directory. This would typically be set to
  2383.               the directory where the model file is, or
  2384.               else a special texture directory. Any
  2385.               number of paths can be seperated by semi
  2386.               colons, but each must have a terminating
  2387.               back slash e.g.;
  2388.               "c:\genesis\textures\;c:\windows\bmps\" or
  2389.               else be a null string. The pointer cannot
  2390.               be NULL.
  2391.  
  2392. phobj         A pointer to the handle of the object that
  2393.               will be created.
  2394.  
  2395. Report         The address of a callback function to
  2396.                inform the application of how far through
  2397.                the load its done (see the
  2398.                fnLoadSaveCallback function).
  2399.  
  2400. Comments
  2401.  
  2402. Loads a lightwave object from an .lwo file into the
  2403. specified co-ordinate system.
  2404.  
  2405.  
  2406.  
  2407.  
  2408.  
  2409.  
  2410. Callback functions
  2411.  
  2412.  
  2413. void fnLoadSaveCallback (ulong ulNumBytes,
  2414.                         ushort usReportType);
  2415.  
  2416. ulNumBytes    If usReport type is LSREPORT_LOADING this
  2417.               is the number of bytes actually written to
  2418.               the file. If usReport type is
  2419.               LSREPORT_SAVING this is the number of
  2420.               bytes currently read from the file. If
  2421.               usReport type is LSREPORT_POLYGONS this is
  2422.               actually the number of polygons processed
  2423.               during loading of non Genesis file formats
  2424.               (due to the differences in the way various
  2425.               file formats store polygons, processing
  2426.               can sometimes be a time consuming task).
  2427.  
  2428. usReportType  Determines what the ulNumBytes parameter
  2429.               refers to. Can be either;
  2430.  
  2431.                    LSREPORT_LOADING
  2432.                    LSREPORT_SAVING
  2433.                    LSREPORT_POLYGONS
  2434.  
  2435. Comments
  2436.  
  2437. This callback function is used by application to give the
  2438. user some feedback during the loading or saving of large
  2439. files to or from Genesis. The address of this function is
  2440. passed to the various loading/saving routines.
  2441.  
  2442.  
  2443. void fnDelCallback (Handle han,
  2444.                   HandleType htype,
  2445.                   ulong ulUser);
  2446.  
  2447. han            Handle of the item being deleted.
  2448.  
  2449. htype          The type of the `han' handle. Can be any
  2450.                of the *_HANDLE types that are used for
  2451.                the SetUserData call.
  2452.  
  2453. ulUser              The user data value of the item being
  2454. deleted.
  2455.  
  2456. Comments
  2457.  
  2458. This routine is called by geometry when any item such as
  2459. a patch, light, surface type etc. is deleted. The
  2460. SetDelCallback function is used to pass the address of
  2461. this routine to geometry. If SetDelCallback is not
  2462. called, then no callback is made.
  2463.  
  2464. This callback function is useful if you want to assign
  2465. additional data (or user data) to an item. The
  2466. application will often allocate memory and store the
  2467. pointer to this memory in the user dword (see SetUserData
  2468. call). The application can then use this callback to
  2469. delete this data.
  2470.  
  2471.  
  2472.  
  2473. Structures and types
  2474.  
  2475.  
  2476. typedef struct Colourtag      // col
  2477.      {
  2478.      byte                byRed;
  2479.      byte                byGrn;
  2480.      byte                byBlu;
  2481.      } Colour;
  2482.  
  2483. byRed               Red component.
  2484.  
  2485. byGrn               Green component.
  2486.  
  2487. byBlu               Blue component.
  2488.  
  2489. Comments
  2490.  
  2491. Specifies a true RGB colour value.
  2492.  
  2493.  
  2494.  
  2495.  
  2496. typedef struct VertInfotag         // vi
  2497.      {
  2498.      Vec                 vec;
  2499.      ulong               ulNumPatches;
  2500.      ulong               ulRef;
  2501.      float                    fScrnX;
  2502.      float                    fScrnY;
  2503.      ulong               ulUser;
  2504.      HanVertex           hverNext;
  2505.      } VertInfo;
  2506.  
  2507. vec           3D position of vertex.
  2508.  
  2509. ulNumPatches  Number of patches using this vertex.
  2510.  
  2511. ulRef         Reference to the first patch. To find all
  2512.               the patches using this vertex pass this
  2513.               value to QryVertPatch.
  2514.  
  2515. fScrnX        The screen X co-ordinate of this vertex
  2516.               after the last Render call.
  2517.  
  2518. fScrnY        The screen Y co-ordinate of this vertex
  2519.               after the last Render call.
  2520.  
  2521. ulUser              The user data value.
  2522.  
  2523. hverNext      Handle to the next vertex in this object.
  2524.  
  2525. Comments
  2526.  
  2527. Contains information about a queried vertex.
  2528.  
  2529.  
  2530.  
  2531.  
  2532. typedef struct NormInfotag         // ni
  2533.      {
  2534.      Vec                 vec;
  2535.      ulong               ulUser;
  2536.      HanNormal      hnorNext;
  2537.      } NormInfo;
  2538.  
  2539. vec           3D direction vector of normal.
  2540.  
  2541. ulUser              The user data value.
  2542.  
  2543. hnorNext      Handle to next normal in this object.
  2544.  
  2545. Comments
  2546.  
  2547. Contains information about a queried normal.
  2548.  
  2549.  
  2550.  
  2551.  
  2552. typedef struct PatchInfotag        // pi
  2553.      {
  2554.      ushort              usNumEdges;
  2555.      ushort              usFlags;
  2556.      HanSurf             hsur;
  2557.      HanVertex           *ahver;
  2558.      HanNormal      *ahnor;
  2559.      Vec                 vecNorm;
  2560.      ulong               ulUser;
  2561.      HanPatch            hpatNext;
  2562.      } PatchInfo;
  2563.  
  2564. usNumEdges    Describes the size of the buffers pointed
  2565.               to by ahver and ahnor.
  2566.  
  2567. usFlags       Set of bitwise OR'd flags describing the
  2568.               patch;
  2569.  
  2570.  PI_VISIBLE       Patch was visible at last render. The
  2571.                    default geometry engine sets this on
  2572.                    if the patch was facing us, but was
  2573.                    maybe obscured by a closer object
  2574.  PI_FLAT          If on object is flat, otherwise it is
  2575.                    curved
  2576.  
  2577. hsur          Handle to the surface used by the patch.
  2578.  
  2579. ahver         Pointer to a buffer in which the handles
  2580.               of the vertices will be returned.
  2581.  
  2582. ahnor         Pointer to a buffer in which the handles
  2583.               of the normals will be returned.
  2584.  
  2585. vecNorm       Normal of the plane of the patch.
  2586.  
  2587. ulUser              The user data value.
  2588.  
  2589. hpatNext      Handle to the next patch in the object.
  2590.  
  2591. Comments
  2592.  
  2593. Contains information about a queried patch. The ahver and
  2594. ahnor pointers should be set to point to the buffers to
  2595. receive the handles before the query call and usNumEdges
  2596. variable be set to the size of the ahver and ahnor
  2597. buffers.
  2598.  
  2599.  
  2600.  
  2601.  
  2602. typedef struct EdgeInfotag         // ei
  2603.      {
  2604.      HanVertex           hverA;
  2605.      HanVertex           hverB;
  2606.      HanPatch            hpatAB;
  2607.      HanPatch            hpatBA;
  2608.      HanEdge             hedgNext;
  2609.      } EdgeInfo;
  2610.  
  2611. hverA         Handle to the vertex at one endpoint.
  2612.  
  2613. hverB         Handle to the vertex at the other
  2614.               endpoint.
  2615.  
  2616. hpatAB        Handle to the patch which uses the edge
  2617.               from vertex A to vertex B NULL_HANDLE if
  2618.               not used.
  2619.  
  2620. hpatBA        Handle to the patch which uses the edge
  2621.               from vertex B to vertex A NULL_HANDLE if
  2622.               not used.
  2623.  
  2624. hedgNext      Handle to the next edge in this object.
  2625.  
  2626. Comments
  2627.  
  2628. Contains information about a queried edge. Each edge
  2629. should be used twice if the object is fully enclosed by
  2630. patches. If the object isn't fully enclosed one of the
  2631. patch handles will be set to NULL_HANDLE. Since edges are
  2632. created and deleted automatically by the geometry engine
  2633. you will not receive an edge used by no patches as any
  2634. such edges would have been deleted.
  2635.  
  2636.  
  2637.  
  2638.  
  2639. typedef struct ObjectInfotag       // oi
  2640.      {
  2641.      ulong               ulNumVerts;
  2642.      ulong               ulNumNorms;
  2643.      ulong               ulNumPatches;
  2644.      ulong               ulNumEdges;
  2645.      float                    fNewVertTol;
  2646.      ulong               ulMemUsed;
  2647.      ushort              usCoorType;
  2648.      HanCoorSys          hcsys;
  2649.      HanVertex           hverFirst;
  2650.      HanNormal      hnorFirst;
  2651.      HanPatch            hpatFirst;
  2652.      HanEdge             hedgFirst;
  2653.      ulong               ulUser;
  2654.      HanObject           hobjNext;
  2655.      } ObjectInfo;
  2656.  
  2657. ulNumVerts    Number of vertices defined in the object
  2658.  
  2659. ulNumNorms    Number of normals defined in the object
  2660.  
  2661. ulNumPatches  Number of patches defined in the object.
  2662.  
  2663. ulNumEdges    Number of edges defined in the object.
  2664.  
  2665. float         Indication of how close new vertices can
  2666.               be to neighbouring vertices without
  2667.               getting re-used.
  2668.  
  2669. ulMemUsed     Total memory used by object and all its
  2670.               vertices, patches etc. (in bytes).
  2671.  
  2672. usCoorType    Type of co-ordinate system object belongs
  2673.               to;
  2674.  
  2675.  CT_LEFTHAND      Co-ordinate system is left handed
  2676.  CT_RIGHTHAND     Co-ordinate system is right handed
  2677.  
  2678. hcsys         Handle of the co-ordinate system this
  2679.               object belongs to.
  2680.  
  2681. hverFirst     Handle to the first vertex in this object.
  2682.  
  2683. hnorFirst     Handle to the first normal in this object.
  2684.  
  2685. hpatFirst     Handle to the first patch in this object.
  2686.  
  2687. hedgFirst     Handle to the first edge in this object.
  2688.  
  2689. ulUser              The user data value.
  2690.  
  2691. hobjNext      Handle to the next object in the co-
  2692.               ordinate system.
  2693.  
  2694. Comments
  2695.  
  2696. Contains information about a queried object.
  2697.  
  2698.  
  2699.  
  2700.  
  2701. typedef struct CoorSysInfotag      // csi
  2702.      {
  2703.      ushort              usType;
  2704.      Mat                 matToParent;
  2705.      Mat                 matFromParent;
  2706.      HanCoorSys          hcsysParent;
  2707.      HanCoorSys          hcsysNext;
  2708.      HanCoorSys          hcsysFirstChild;
  2709.      HanLight            hliFirst;
  2710.      HanObject           hobjFirst;
  2711.      ulong               ulUser;
  2712.      } CoorSysInfo;
  2713.  
  2714. usType             Type of co-ordinate system.
  2715.  
  2716.  CT_LEFTHAND      Co-ordinate system is left handed
  2717.  CT_RIGHTHAND     Co-ordinate system is right handed
  2718.  
  2719. matToParent   Matrix describing the orientation of this
  2720.               co-ordinate system relative to the parent.
  2721.               The matrix will convert a point in our co-
  2722.               ordinate system into that of the parent.
  2723.  
  2724. matFromParent The inverse of matToParent. This matrix
  2725.               will convert a point in the parent co-
  2726.               ordinate system into ours.
  2727.  
  2728. hcsysParent   Handle to parent co-ordinate system.
  2729.  
  2730. hcsysNext     Handle to next sibling co-ordinate system,
  2731.               i.e. the next co-ordinate system which
  2732.               also has hcsysParent as its parent.
  2733.  
  2734. hcsysFirstChild    Handle to the first of our child co-
  2735.               ordinate system, i.e. the first co-
  2736.               ordinate system which has us as the parent
  2737.               (query this co-ordinate system to find the
  2738.               next one).
  2739.  
  2740. hliFirst           Handle to the first light in this co-
  2741.               ordinate system (query this light to find
  2742.               the next one).
  2743.  
  2744. hobjFirst     Handle to the first object in this co-
  2745.               ordinate system (query this object to find
  2746.               the next one).
  2747.  
  2748. ulUser              The user data value.
  2749.  
  2750. Comments
  2751.  
  2752. Contains information about a queried co-ordinate system.
  2753.  
  2754.  
  2755.  
  2756.  
  2757. typedef struct LightDeftag         // ld
  2758.      {
  2759.      Vec                 vecPos;
  2760.      Vec                 vecDir;
  2761.      ushort              usFlags;
  2762.      float                    fInt;
  2763.      float                    fAng;
  2764.      Colour              col;
  2765.      float                    fHalfIntDist;
  2766.      HanCoorSys          hcsys;
  2767.      ulong               ulUser;
  2768.      HanLight            hliNext;
  2769.      } LightDef;
  2770.  
  2771. vecPos        3D position of light.
  2772.  
  2773. vecDir        3D position towards which the light is
  2774.               directed.
  2775.  
  2776. usFlags       Flags describing the light;
  2777.  
  2778.  LI_DIRECTIONAL   Light has a directional beam
  2779.  LI_DIMINISHING   Light diminishes with distance
  2780.  
  2781. fInt          Intensity of the light from 0 to 1.
  2782.  
  2783. fAng          Angle of the lights beam (if directional).
  2784.  
  2785. col           Colour of the light.
  2786.  
  2787. fHalfIntDist  If the LI_DIMINISHING flag is set, this
  2788.               value show how far the light can travel
  2789.               before its intensity halves.
  2790.  
  2791. hcsys         The co-ordinate system the light belongs
  2792.               to.
  2793.  
  2794. ulUser        The user data value.
  2795.  
  2796. hliNext       Handle of the next light in this co-
  2797.               ordinate system.
  2798.  
  2799. Comments
  2800.  
  2801. Contains information about a light. Can be used to set
  2802. the light or query the light. NB. The default Geometry
  2803. engine does not support directional or coloured lights.
  2804.  
  2805.  
  2806.  
  2807.  
  2808. typedef struct SurfTypetag         // st
  2809.      {
  2810.      Colour              col;
  2811.      ushort              usFlags;
  2812.      ushort              usSpecRefChar;
  2813.      float                    fShineCoef;
  2814.      float                    fTransCoef;
  2815.      HanCoorSys          hcsysTex;
  2816.      Mat                 matTex;
  2817.      HanSurf             hsurSurf0;
  2818.      HanSurf             hsurSurf1;
  2819.      HanSurf             hsurSurf2;
  2820.      HanSurf             hsurSurf3;
  2821.      float                    fBumpHeight;
  2822.      ulong               ulUser;
  2823.      HanSurf             hsurNext;
  2824.      char                szFNTex[128];
  2825.      } SurfType;
  2826.  
  2827. col           Basic colour of the surface.
  2828.  
  2829. usFlags       A set of bitwise OR'd flags defining the
  2830.               type of surface.
  2831.  
  2832.  DS_TEXTURE       Surface is textured
  2833.  DS_RECURSIVE_TEXTURE  The texture is a recursive
  2834.                    texture. The default Geometry engine
  2835.                    does not support recursive textures.
  2836.  DS_TILE_TEXTURE  The texture will be tiled. If the
  2837.                    texture is not tiled the basic
  2838.                    surface colour `col' will show
  2839.                    through where the texture does not
  2840.                    reach, or where hsurSurf0-4 is set to
  2841.                    NULL_HANDLE for recursive textures.
  2842.  DS_BUMPED        The texture will be bump mapped. It
  2843.                    is not valid to have this on as well
  2844.                    as DS_RECURSIVE_TEXTURE. The default
  2845.                    Geometry engine does not support bump
  2846.                    mapped textures.
  2847.  
  2848. usSpecRefChar This value describes only how the specular
  2849.               highlights will appear on the surface. It
  2850.               does not mean that an SRC_GOLD surface
  2851.               will have a gold colour but given enough
  2852.               patches the positioning and size of the
  2853.               specular highlights will mimic a surface
  2854.               made of gold
  2855.  
  2856.               The following types are permitted;
  2857.  
  2858.                  SRC_DULL   (Surface has no specular
  2859.                              highlights)
  2860.                  SRC_CONSTANT    (50% reflection
  2861.                              regardless of incidence
  2862.                              angle)
  2863.                  SRC_LINEAR (Amount of reflected light
  2864.                              increases in proportion to
  2865.                              incidence angle)
  2866.                  SRC_SILVER (Produces large diffuse
  2867.                              highlights)
  2868.                  SRC_GOLD   (Produces dull ring shaped
  2869.                              highlights)
  2870.                  SRC_GLASS  (Produces highlights only on
  2871.                              the horizon edges of
  2872.                              objects)
  2873.                  SRC_BRASS  (The default Geometry engine
  2874.                              does not implement
  2875.                  SRC_COPPER any further ,but maps them
  2876.                              to the nearest of the
  2877.                  SRC_ALUMINIUM    above...)
  2878.                  SRC_IRON
  2879.                  SRC_PLASTIC
  2880.                  SRC_WATER
  2881.                  SRC_CHINA
  2882.                  SRC_LEATHER
  2883.                  SRC_SILK
  2884.  
  2885.               For completeness sake the following types
  2886.               are defined but mean the same as SRC_DULL
  2887.               as they are not shiny surfaces;
  2888.  
  2889.                  SRC_ROCK   (same as SRC_DULL...)
  2890.                  SRC_RUBBER
  2891.                  SRC_WOOD
  2892.                  SRC_FABRIC
  2893.  
  2894. fShineCoef    Shinyness coefficient. This values ranges
  2895.               from 1 to 10. Large values mean highly
  2896.               polished, and therefore very reflective
  2897.               surfaces. These generate very small but
  2898.               bright highlights with sharply defined
  2899.               edges (e.g. a snooker ball). Small values
  2900.               produces large, dull, diffuse highlights
  2901.               (e.g. a piece of paper).
  2902.  
  2903. fTransCoef    Transmissive coefficient. A value ranging
  2904.               from 0 to 1 to describe how transparent
  2905.               the surface is. A value of 1 would make
  2906.               the surface invisible. The default
  2907.               Geometry engine does not support
  2908.               transmissive surfaces.
  2909.  
  2910. hcsysTex      The handle of the co-ordinate system the
  2911.               texture is defined relative to. See below
  2912.               for further details. NB. Because a texture
  2913.               is defined in one co-ordinate system,
  2914.               doesn't mean it cant be used by patches in
  2915.               another. Textures pervade all of 3D space,
  2916.               however because top level co-ordinate
  2917.               systems aren't related to each other by
  2918.               matrices like parent-child-sibling
  2919.               relationships, it does means that the
  2920.               orientation of a texture in one scene
  2921.               won't have any meaning in other and may
  2922.               come out looking confusing. The
  2923.               ModSurfType call can be used to modify the
  2924.               co-ordinate system a texture is defined
  2925.               in.
  2926.  
  2927. matTex        The orientation of the texture bitmap
  2928.               within the co-ordinate system. When a
  2929.               texture is defined the bitmap is projected
  2930.               through 3D space and `rubs off' on any
  2931.               patch using this surface type. If you
  2932.               supply the identity matrix here then the
  2933.               orientation of the bitmap is such that the
  2934.               bottom left is at the origin of the co-
  2935.               ordinate system, and the bottom edge
  2936.               proceeds along the x axis for however many
  2937.               pixels wide the bitmap is, e.g. If we have
  2938.               a 100 pixel wide bitmap the bottom left
  2939.               will be at 0, 0, 0 and the bottom right at
  2940.               100, 0, 0 in the co-ordinate system.
  2941.               Similarly the top of the bitmap proceeds
  2942.               up the y axis of the co-ordinate system.
  2943.               The bitmap is projected through the z axis
  2944.               to +/- infinity. By changing this matrix
  2945.               you can alter the plane of the bitmap or
  2946.               scale it to any size.
  2947.  
  2948. hsurfSurf0-3  A recursive bitmap is one where the colour
  2949.               values in the bitmap refer not to colours
  2950.               in the bitmaps palette, but to other
  2951.               surfaces types which can in turn be
  2952.               textured, or even recursive textures. The
  2953.               surface handles for the colour values 0 to
  2954.               3 can be set here. A value of NULL_HANDLE
  2955.               means the surface's colour (defined by
  2956.               `col') shows through, whereas a value of
  2957.               HSUR_SEE_THROUGH means that the object has
  2958.               a hole in it at those points showing
  2959.               objects that may be behind. The default
  2960.               Geometry engine does not support recursive
  2961.               textures.
  2962.  
  2963. fBumpHeight   If the DS_BUMPED flag is set, the colour
  2964.               values in the bitmap refer to height
  2965.               values for a bumpy surface. The effective
  2966.               height of the maximum colour value is set
  2967.               using this parameter. The default Geometry
  2968.               engine does not support bump mapping.
  2969.  
  2970. ulUser        The user data value.
  2971.  
  2972. hsurNext      The next surface type defined in geometry.
  2973.  
  2974. szFNTex       The full name (including path and
  2975.               extension) of a windows bitmap file to use
  2976.               as a texture.
  2977.  
  2978. Comments
  2979.  
  2980. Defines a surface type. This structure can be used to
  2981. both set, query and modify a surface type.
  2982.  
  2983.  
  2984.  
  2985.  
  2986. typedef struct ViewParamstag  // vp
  2987.      {
  2988.      float                    fAngX;
  2989.      float                    fAngY;
  2990.      float                    fClipFront;
  2991.      float                    fClipBack;
  2992.      ushort              usResX;
  2993.      ushort              usResY;
  2994.      } ViewParams;
  2995.  
  2996. fAngX               The horizontal viewing angle.
  2997.  
  2998. fAnyY               The vertical viewing angle.
  2999.  
  3000. fClipFront          The distance from the viewpoint of
  3001. the front clipping plane.
  3002.  
  3003. fClipBack      The distance from the viewpoint of the
  3004. back clipping plane.
  3005.  
  3006. usResX         The horizontal size of bitmap to render.
  3007.  
  3008. usResY         The vertical size of bitmap to render.
  3009.  
  3010. Comments
  3011.  
  3012. Defines the viewing parameters for the SetCamera
  3013. function.
  3014.  
  3015.  
  3016.  
  3017. Tool Interface
  3018.  
  3019. Introduction
  3020.  
  3021. Although custom tools are implemented as a C++ class, an
  3022. understanding of the C++ language is not essential as we
  3023. have supplied a template source file; toolexmp.cpp,
  3024. defining the class. All you need to do is to fill in the
  3025. functions which can be written in normal C language. The
  3026. source to the patch tool is also supplied as an example
  3027. (see patch.cpp). Only a basic understanding of
  3028. programming the windows operating system is required as
  3029. the only windows calls you are likely to make are the GDI
  3030. drawing calls such as, MoveTo and LineTo. Dialog box
  3031. calls will also be required if your tool needs a
  3032. configuration dialog box. An understanding of the
  3033. Geometry API is required.
  3034.  
  3035. The screen shot below is useful to refer to when reading
  3036. this section.
  3037.  
  3038.  
  3039.  
  3040.  
  3041.  
  3042.  
  3043.  
  3044.  
  3045.  
  3046.  
  3047.  
  3048.  
  3049.  
  3050.  
  3051.  
  3052.  
  3053.  
  3054.  
  3055.  
  3056.  
  3057.  
  3058.  
  3059.  
  3060.  
  3061.  
  3062.  
  3063.  
  3064.  
  3065.  
  3066.  
  3067.  
  3068.  
  3069.  
  3070. You will notice that some of the Geometry API functions
  3071. are also available through the tool interface and have
  3072. been slightly modified. This does not mean that you can't
  3073. use the original Geometry versions There are two reasons
  3074. we have done this; Some functions, DefPatch for example
  3075. are built on top of Geometry's DefPatch and provide
  3076. additional functionality, for instance if an error occurs
  3077. during the DefPatch call (say `out of memory') the error
  3078. will be reported to the user in a message box and
  3079. everything created by the tool up to that point will be
  3080. erased. This is the sort of thing many tools would have
  3081. to implement, so to save time and effort the tool base
  3082. class DefPatch does it for you (if you want it to).
  3083.  
  3084. The other reason for duplicating some functions is that
  3085. the editor gives names to things like objects and co-
  3086. ordinates systems. Since Geometry only has the concept of
  3087. handles and not names, if your tool needs to create an
  3088. object called `MyBox' it needs to call the editor version
  3089. of AddObject for the name to appear in the list box on
  3090. the control bar. If you use the Geometry version it will
  3091. still be visible and will be rendered, but without a name
  3092. it wont be selectable at a single mouse click from the
  3093. listbox and any tools that require the name of an object
  3094. to modify will not be able to process the object.
  3095.  
  3096.  
  3097. Writing custom tools
  3098.  
  3099. Each tool is implemented in a windows DLL by writing a
  3100. C++ class derived from the base tool class; `Tool'. The
  3101. base class is implemented in tool.dll. Your derived class
  3102. should include tool.h and should be linked to tool.lib.
  3103.  
  3104. Your DLL should include a resource file defining a 16
  3105. pixel wide by 15 pixel high bitmap with a resource ID of
  3106. one. This bitmap contains an image which will
  3107. automatically be placed on a button in the toolbox to
  3108. enable your tool to be invoked. If you are using
  3109. Microsoft App Studio this can be done by entering; <id>=1
  3110. in the id field of the properties dialog for the bitmap.
  3111.  
  3112. The tool object
  3113.  
  3114. The DLL should provide a routine to create and delete an
  3115. object belonging to your tool class which will be called
  3116. by the editor. The IMPLEMENT_OBJECT macro will do this
  3117. for you. The following line must be added to your main
  3118. source file;
  3119.  
  3120.      IMPLEMENT_OBJECT(<your_class_name>)
  3121.  
  3122. You must also add the following to the exports section of
  3123. your .def file;
  3124.  
  3125.      CreateTool
  3126.      DeleteTool
  3127.  
  3128. The editor will create one of your tool objects for each
  3129. scene that exists. Note, this is not the same as the
  3130. view. You can have several views into a single scene. It
  3131. works in the same in same way that a word processor can
  3132. have several documents loaded at once, and you might have
  3133. several windows showing different parts of the same
  3134. document. The reason for this is that it makes tool
  3135. writing considerably simpler as a tool can be
  3136. constructing in different scenes at once without the tool
  3137. code having to manage a complete set of variables
  3138. relating to each scene. Therefore a tool defines just one
  3139. set of member variables that it needs to do its job.
  3140.  
  3141. The IMPLEMENT_OBJECT macro also calls the Debug library's
  3142. DebStart function, so if your tool uses the library it
  3143. should not call it again.
  3144.  
  3145. The tool's configuration dialog box
  3146.  
  3147. If your tool defines a configuration dialog box, you
  3148. generally want the dialog to apply to all tools belonging
  3149. to your object. For instance when editing in scene A, you
  3150. configure your tool in a certain way. You then switch to
  3151. scene B, since you see only one tool of your type in the
  3152. toolbox, you assumes it is configured the way you just
  3153. set it. Now, if your configuration data is stored as
  3154. member variables within  your object there will be a
  3155. separate configuration for each tool, i.e. one for each
  3156. scene. So generally it is best to make your configuration
  3157. variables static members or global.
  3158.  
  3159. How to tell the editor about your tool
  3160.  
  3161. The name of your DLL should be added to the Gened.ini
  3162. file in the main windows directory, under the section
  3163. [Tools] as follows;
  3164.  
  3165.                [Tools]
  3166.                0=patch
  3167.                1=sphere
  3168.                2=box
  3169.                :
  3170.                n=<name of your DLL>
  3171.  
  3172. The value of n must be the next unused consequecutive
  3173. number.
  3174.  
  3175. Overriding tool functions
  3176.  
  3177. Whenever the `Set' or `Do' buttons on the control bar is
  3178. pressed, or the `Undo' menu item is selected an
  3179. appropriate function is called in the currently selected
  3180. tool. The tool base class provide default functions which
  3181. actually do nothing. To detect these events you can
  3182. override these functions in your derived class. The
  3183. following table shows the events which trigger tool
  3184. functions;
  3185.  
  3186. Event        Called when
  3187. OnSelect     The tool has been selected from the
  3188.              toolbox
  3189. OnUnSelect   An attempt has been made to select another
  3190.              tool
  3191. OnButtonDow  The mouse is in the view and the left
  3192. n            button is pressed
  3193. OnMouseMove  The left button is pressed and the mouse
  3194.              moves
  3195. OnButtonUp   The left mouse button is released
  3196. OnSet        The `Set' button on the control bar is
  3197.              pressed
  3198. OnDo         The `Do' button on the control bar is
  3199.              pressed
  3200. OnViewChang  The another view into the scene becomes
  3201. e            active
  3202. OnUndo       The `Undo' menu item is selected and this
  3203.              was the last active tool
  3204. FreeUndo     When another tool modifies the scene
  3205. OnCommand    The user selects a menu/toolbar button
  3206.              create by this tool
  3207. OnCommandUp  Windows requires the status of a tools
  3208. date         menu/toolbar button is updated
  3209. OnPaste      A clipboard paste operation is performed
  3210.              and this tool is the paste receiver
  3211. OnConfigure  The user requests to configure the tool
  3212.  
  3213. Calling sequence
  3214.  
  3215. The very first call to all tools is Initialise. Override
  3216. it if you need to initialise any variables. The next
  3217. likely call to your tool will be when it is selected from
  3218. the toolbox, the OnSelect routine will be invoked. Then
  3219. as the user clicks and moves the mouse in the view you
  3220. will receive OnButtonDown, OnMouseMove and OnButtonUp
  3221. events. Then when the user presses the `Set' button,
  3222. having positioned the 3D cursor, the OnSet call is
  3223. invoked. This can be invoked as many times as your tool
  3224. requires. If your tool requires an indefinite number of
  3225. Set presses, then the `Do' button can be used to do
  3226. whatever is required to complete the tools action. For
  3227. instance, the patch tool uses Set to position each
  3228. vertex. Since you can define an indefinite number of
  3229. vertices, the Do button is used to create the patch. The
  3230. interpretation of the buttons is entirely up to you.
  3231.  
  3232. All events are directed to the currently selected tool
  3233. only, apart from, obviously OnSelect, OnUndo and
  3234. FreeUndo.
  3235.  
  3236. Modifying and undoing
  3237.  
  3238. For your custom tool to be seamlessly integrated with the
  3239. editor, it must be capable of undoing any action it has
  3240. performed on the scene. If and when a tool modifies any
  3241. aspect of a scene, whether its creating or deleting part
  3242. of an object, moving a light or even changing the colour
  3243. of a patch it must call the `ModifiedScene' function.
  3244. This has two effects. First of all it marks the scene as
  3245. having changed and will give the user a warning if he
  3246. tries to close the editor down without having saved the
  3247. scene first. Secondly it marks this tool for calling
  3248. should the user select the `Undo' menu item. This tool
  3249. can now be called to undo its action even after it has
  3250. been de-selected, at least until another tool calls the
  3251. ModifiedScene routine.
  3252.  
  3253. Drawing into the view
  3254.  
  3255. During the construction of a primitive the tool will
  3256. probably need to draw into the views showing the object
  3257. being constructed, to give the user some feedback about
  3258. what is going on. There are two ways of doing this. The
  3259. tool can implement the DrawSoFar routine so that whenever
  3260. the view gets updated it gets a chance to draw into the
  3261. DC representing the view window. The patch tool draws a
  3262. dotted line between the cursor positions at each point
  3263. Set was pressed using this technique. The patch isn't
  3264. actually created on Geometry until the Do button is
  3265. pressed. Alternatively, more complex tools might
  3266. construct the object on Geometry in several steps (e.g.
  3267. during Set presses, or on mouse move events) in which
  3268. case the editor can redraw it for you. The sphere tool
  3269. does this. A combination of these techniques can also be
  3270. used.
  3271.  
  3272. When the screen view needs repainting, say if part of it
  3273. is uncovered by a window on top, the editor redraws the
  3274. rendered image along with its wire frame and then calls
  3275. the selected tool's DrawSoFar routine passing it the
  3276. device context handle of the view window. The DrawSoFar
  3277. routine will be called for each view into the object
  3278. which needs updating. If we attempt to draw into the view
  3279. from any call other than DrawSoFar, then we have to
  3280. consider the fact that there might be several views into
  3281. this object all of which will need updating. For this
  3282. reason drawing should be reserved exclusively for the
  3283. DrawSoFar routine.
  3284.  
  3285. The view will probably need updating after a Set/Do press
  3286. or an Undo selection. Although we cannot draw on these
  3287. events we can return a value which will cause the view(s)
  3288. to be repainted, and therefore DrawSoFar to be called.
  3289.  
  3290. Drawing return codes
  3291.  
  3292. For each view into the scene the editor maintains two
  3293. bitmaps. The first is the full colour rendered bitmap
  3294. generated by Geometry's Render call. The second is a
  3295. monochrome overlay containing the image of the x, y, and
  3296. z axis, the objects wire frame, and optionally a set of
  3297. grid points. When the object changes all this needs to be
  3298. updated; the image must be re-rendered and the overlay
  3299. regenerated. When a tool is in a constructing state, we
  3300. might want to draw the object quickly. For instance the
  3301. sphere or box tool requires the radius or corner to be
  3302. dragged by the mouse and released when you release the
  3303. button. Given that there could be several views into the
  3304. object there is not time to re-render and regenerate the
  3305. overlay for every view as the mouse moves. The whole
  3306. thing would be too slow. The following return codes from
  3307. event functions can tell the editor how to redraw
  3308. quickly.
  3309.  
  3310. REDRAW_ALL
  3311.  
  3312. This is typically used only once when the tool has
  3313. finished its business and tells the editor to re-render
  3314. the main bitmaps and regenerate all overlays for each
  3315. view. The screen image for each view is then refreshed
  3316. from both of these. This takes the most amount of time
  3317. and so it is unlikely you will want to use this from a
  3318. mouse move event.
  3319.  
  3320. Generally it takes the windows GDI longer to draw the
  3321. overlay than it takes the engine to render the image. For
  3322. this reason the box and sphere tools use this return code
  3323. from the mouse move event only when the overlay is turned
  3324. off.
  3325.  
  3326. The current tools DrawSoFar routine will also be called.
  3327.  
  3328. REDRAW_NONE
  3329.  
  3330. Pretty much self explanatory. This return code will not
  3331. affect the on screen image in any way.
  3332.  
  3333. REDRAW_OBJECT_WIRE
  3334.  
  3335. This causes the editor to draw the wire frame of the last
  3336. object created by the tool base class's AddObject
  3337. function directly onto the screen using the GDI XOR mode.
  3338. The XOR mode has the effect that if you draw something
  3339. into a DC twice, it actually removes it leaving you with
  3340. the original image. Using this return code does not cause
  3341. the screen image to get refreshed from the rendered and
  3342. overlay bitmaps, so it is very fast.
  3343.  
  3344. The box and sphere tools use this return code to update
  3345. the wire frame of the box on the mouse move event when
  3346. dragging a corner to get the correct box size (see the
  3347. `Dragging with the mouse' section).
  3348.  
  3349. Again the current tools DrawSoFar routine is called.
  3350.  
  3351. REDRAW_TOOL
  3352.  
  3353. The REDRAW_ALL and REDRAW_OBJECT_WIRE return codes result
  3354. in the editor redrawing the object as it exists within
  3355. Geometry (i.e.. as seen through Geometry's query
  3356. functions). If the object within Geometry hasn't changed,
  3357. returning these codes will be wasteful as the on screen
  3358. image wont change. Instead the REDRAW_TOOL return code
  3359. results in just the tools DrawSoFar routine being called.
  3360. The patch tool uses this after setting a vertex.
  3361.  
  3362. REDRAW_NOTOOL
  3363.  
  3364. This causes the editor to simply refresh the screen from
  3365. its internal bitmaps, thereby effectively removing any
  3366. image drawn as a result of a REDRAW_OBJECT_WIRE code, or
  3367. anything drawn by a tool's DrawSoFar routine. If a tool
  3368. has used either of these to draw an object during a
  3369. construction phase but undo was selected before the
  3370. operation completed, then this return code can be used to
  3371. erase the `construction image' on the screen. If we
  3372. actually constructed parts of an object on Geometry then
  3373. we must remember to remove them as well otherwise the
  3374. next time a render is performed the object will reappear!
  3375.  
  3376. The DrawSoFar routine is not called as a result.
  3377.  
  3378.  
  3379. REDRAW_REFRESH
  3380.  
  3381. This is the same as a REDRAW_NOTOOL except that the
  3382. tool's DrawSoFar routine is called as well.
  3383.  
  3384.  
  3385. REDRAW_SHADING
  3386.  
  3387. This return code re-renders the main bitmap but doesn't
  3388. redraw the overlay. This is only really use then if the
  3389. geometry itself doesn't change, otherwise the overlay
  3390. won't match the rendered picture. An example is the when
  3391. the light tool moves a light The objects don't move so
  3392. the overlay doesn't need updating but the shading on the
  3393. objects changes.
  3394.  
  3395. Again the DrawSoFar routine is called.
  3396.  
  3397.  
  3398. Getting 2D screen co-ordinates
  3399.  
  3400. Any drawing performed in DrawSoFar is done using the
  3401. windows GDI. The 2D window co-ordinates you need to
  3402. perform the drawing operations can be got using Geometry
  3403. calls such as Get3DPoint and Get3DLine. The Geometry API
  3404. spec states that before you can use these calls you must
  3405. set the camera in the appropriate co-ordinate system.
  3406. Obviously without specifying a camera view the 3D co-
  3407. ordinates you work with will not have a 2D counterpart.
  3408. Before DrawSoFar is called the editor sets the camera at
  3409. the viewpoint in the co-ordinate system of the view being
  3410. redrawn so the tool doesn't have to worry about this. On
  3411. event calls like OnSet and OnMouseMove the camera is set
  3412. in the currently active view, i.e.. the one that has the
  3413. focus.
  3414.  
  3415.  
  3416. Dragging with the mouse
  3417.  
  3418. The sphere tool creates a small sphere when Set is
  3419. pressed, if the `quickdraw' configuration option is
  3420. switched off. It then traps the OnMouseMove event and
  3421. scales the sphere to the correct radius. By returning
  3422. REDRAW_OBJECT_WIRE it gets the editor to draw the wire
  3423. frame of the sphere (assuming the overlay is on).
  3424.  
  3425. If it is more appropriate to perform drawing ourselves
  3426. then simply use GDI's ROP2 setting of XORPEN to draw and
  3427. erase your image in the DrawSoFar routine. You may of
  3428. course prefer to save the background before drawing on
  3429. it, but in general we have found that XOR drawing is
  3430. faster. The only thing to remember is that if you change
  3431. any of the DC's attributes such as the selected pen,
  3432. drawing mode etc. you must set them back to the originals
  3433. before DrawSoFar returns. NB. This is a general
  3434. requirement of programming in Windows.
  3435.  
  3436. XOR'ing in multiple views
  3437.  
  3438. If you wish to implement dragging on the mouse move event
  3439. as described above there is just one other thing to bear
  3440. in mind. Consider the following sequence of events.
  3441. Assume that this is part of a box construction tool. We
  3442. `Set' the cursor at one corner then move the cursor to
  3443. the opposite corner (during which the box is being
  3444. dynamically sized in the view[s]), and press `Do'.
  3445. Further assume we have two views into this scene.
  3446.  
  3447.      1.   OnSet event - Set the 3D co-ordinates of the first
  3448.        corner to those of the cursor. Set the 3D co-ordinates of
  3449.        the far corner to the same as the first and return
  3450.        REDRAW_TOOL to start the XOR drawing (initially a point
  3451.        sized cube as both corners are in the same place)
  3452.      2.   OnDrawSoFar call - Called for the first view as a
  3453.        result of returning REDRAW_TOOL from OnSet. Being the
  3454.        first DrawSoFar for this view there is no previous
  3455.        construction image to remove, so simply draw the image in
  3456.        XOR mode (i.e.. a cube) using Get3DPoint to convert your
  3457.        3D corners into 2D window co-ordinates and draw them
  3458.        using the GDI. Store the 2D co-ordinates of your
  3459.        construction image so you can erase it later
  3460.      3.   OnDrawSoFar call - Called for the second view. Do
  3461.        exactly the same as for the previous DrawSoFar but for
  3462.        the second view
  3463. 4.   OnMouseMove event - Set the 3D co-ordinates of the
  3464. opposite corner to the cursor position, and return
  3465. REDRAW_TOOL
  3466.      5.   OnDrawSoFar call - Called for the first view as a
  3467.        result of returning REDRAW_TOOL from the OnMouseMove
  3468.        event. Redraw the old construction image in XOR mode to
  3469.        erase the image and redraw the image in XOR mode using
  3470.        the opposite corner in its new position
  3471.      6.   OnDrawSoFar call - Called for the second view. Do
  3472.        exactly the same as for the previous DrawSoFar but for
  3473.        the second view
  3474.      7.   Do steps 4, 5 and 6 - For however many mouse move
  3475.        events we get
  3476.      8.   OnDo event - Construct the cube using Geometry and
  3477.        return REDRAW_ALL to get it rendered properly
  3478.  
  3479. You will notice that because we have two views we get two
  3480. DrawSoFar calls every time we return REDRAW_TOOL from an
  3481. event. This means that we must store two sets of co-
  3482. ordinates in order to remove the previous XOR image. For
  3483. this purpose DrawSoFar gets called with a parameter to
  3484. instance data relating to each particular view.
  3485.  
  3486.  
  3487. DrawSoFar parameters
  3488.  
  3489. The first parameter contains the DC handle of the view.
  3490. You need this when using GDI drawing commands.
  3491.  
  3492. The second parameter is a pointer to a 60 byte area of
  3493. memory stored as part of each view known as the view
  3494. instance data. You can use this memory to store whatever
  3495. you like, e.g. co-ordinates of the construction image. If
  3496. you need more memory than this you can allocate your own
  3497. and store a pointer to the memory in the view instance
  3498. data. The SetViewData call in the tool base class can be
  3499. used to initialise the instance data at any time (say,
  3500. from the OnSet event).
  3501.  
  3502. The final parameter is a boolean value which says why
  3503. this routine has been called. Not only will you receive
  3504. DrawSoFar calls as a result of return codes from events
  3505. but also if part of the view was invalidated by a window
  3506. on top being removed. In normal windows programming this
  3507. will generate a WM_PAINT message. If this happens and
  3508. your tool is in the process of constructing an XOR image
  3509. it must not erase the previous image because the view is
  3510. invalid in that region. As a general rule, if this
  3511. bInvalid parameter is TRUE simply draw the XOR image as
  3512. if it were the first OnDrawSoFar for this view.
  3513.  
  3514. General guidelines
  3515.  
  3516. Help on using tools
  3517.  
  3518. As a general rule any tool should provide instructions to
  3519. the user on how to operate the tool. The recommended
  3520. approach is at each stage inform the user what to do next
  3521. by writing a message in the status bar at the bottom of
  3522. the main window using the WriteMessage function. See the
  3523. Sphere or Patch tools as an example.
  3524.  
  3525. Displaying text strings
  3526.  
  3527. Whenever your tool displays any text it is a good idea to
  3528. load the text from a string table resource. This way if
  3529. your tool ever gets used in other countries, language
  3530. translation is a simple matter of going through the
  3531. string table and recompiling to get a new DLL, rather
  3532. than having to search through the source code for literal
  3533. strings. The tool base class functions such as
  3534. WriteMessage and MessageBox aid this by allowing you to
  3535. present string resource ids as well as character
  3536. pointers.
  3537.  
  3538.  
  3539. Co-ordinate system types
  3540.  
  3541. Any tool which creates patches has to consider the fact
  3542. it could be invoked in either a left-hand or a right-hand
  3543. co-ordinate system. The significance is in the ordering
  3544. of the vertices. Remember vertices should always be seen
  3545. to progress clockwise around the patch. If your tool
  3546. works fine in one system but not in the other then you
  3547. probably need to reverse the ordering in the system which
  3548. doesn't work.
  3549.  
  3550.  
  3551. User Data
  3552.  
  3553. User data can be associated with geometry items such as
  3554. objects, patches, lights, surfaces etc. The user data is
  3555. a single 32 bit value. Tools cannot modify this value
  3556. directly as the editor uses the user dwords to store
  3557. pointers to its internal information. If a tool needs to
  3558. store instance for a geometry item, it must use the
  3559. appropriate tool functions such as AddUserData. This way
  3560. the editor can chain together the instance data it needs
  3561. as well as those from any tools.
  3562.  
  3563.  
  3564. A note for C Users
  3565.  
  3566. For those not familiar with C++, you can use the
  3567. toolexmp.cpp file as a base to work from. The functions
  3568. defined in the `Overidables' section are functions you
  3569. can implement yourself if you're interested in trapping
  3570. those particular events. Empty functions are defined but
  3571. commented out in toolexmp.cpp. If you want to trap mouse
  3572. moves events simply uncomment it and add your code using
  3573. normal C. Remember C++ is a superset of the C language.
  3574.  
  3575. The functions defined in the section titled `Support' are
  3576. supplied for you to call should the need arise. One thing
  3577. you will realise about C++ is that a function is
  3578. specified not just by its name but also by its
  3579. parameters, meaning you can have two or more functions
  3580. with the same name as long as they have different
  3581. parameters (NB. The MessageBox support function is an
  3582. example).
  3583.  
  3584. Some variables are also defined in the `Support' section.
  3585. In C++ terminology these are public variables defined in
  3586. the base class. Non C++ programmers can just think of
  3587. them as global variables containing information which may
  3588. or may not be of use to your tool. You must not attempt
  3589. to alter these variables.
  3590.  
  3591.  
  3592.  
  3593. Overridables
  3594.  
  3595.  
  3596. virtual ~Tool();
  3597.  
  3598. Comments
  3599.  
  3600. Destructor for your tool object. This function must be
  3601. overridden even if it does nothing. As with any normal
  3602. C++ class, if you create a constructor for your object
  3603. which allocates resources, the destructor must release
  3604. these resources. C users can leave this function empty.
  3605.  
  3606.  
  3607.  
  3608.  
  3609. virtual void Initialise();
  3610.  
  3611. Comments
  3612.  
  3613. Called once when the DLL is loaded. This call can be used
  3614. to initialise variables.
  3615.  
  3616.  
  3617.  
  3618.  
  3619. virtual Redraw OnSelect(Vec &vec);
  3620.  
  3621. vec            Current position of the 3D cursor.
  3622.  
  3623. Comments
  3624.  
  3625. Called when the user selects the tool from the toolbox.
  3626.  
  3627.  
  3628.  
  3629.  
  3630. virtual Redraw OnUnSelect(bool *pbOkToChange)
  3631.  
  3632. pbOkToChange   Set this to TRUE if it is OK for the
  3633. editor to change the current tool.
  3634.  
  3635. Comments
  3636.  
  3637. Called when the user tries to select another tool from
  3638. the toolbox. If the tool is in the middle of a
  3639. complicated operation it can either set *pbOkToChange to
  3640. FALSE or it can either abort its operation or complete it
  3641. without further action from the user. What happens here
  3642. is up to the tool writer. Remember the tool can always
  3643. undo what its just done if the user isn't happy.
  3644.  
  3645. When the clipboard pastes an object into the scene the
  3646. paste receiver tool is automatically selected in order
  3647. for the user to position the object. This means that an
  3648. OnUnSelect event is sent to the current tool if this is
  3649. not the paste receiver. If the tool sets *pbyOkToChange
  3650. to FALSE the user is presented with a message saying that
  3651. the paste cannot be carried out until the selected tool
  3652. completes its operation.
  3653.  
  3654.  
  3655. virtual void OnConfigure();
  3656.  
  3657. Comments
  3658.  
  3659. Called when the tool is selected and the user then
  3660. selects the Configure item from the Tool menu. Typically
  3661. this call is overridden to construct a dialog box
  3662. enabling the user to enter configuration information for
  3663. the tool.
  3664.  
  3665.  
  3666.  
  3667.  
  3668. virtual Redraw OnButtonDown(Vec &vec);
  3669.  
  3670. vec            Current position of the 3D cursor.
  3671.  
  3672. Comments
  3673.  
  3674. Called when the tool is selected and the user presses the
  3675. left mouse button when the mouse is in a view allowing
  3676. the 3D cursor to be moved. The vec parameter gives the co-
  3677. ordinates of the 3D cursor. The editor assumes the tool
  3678. is in a constructing state only after the first Set
  3679. press, so if you override this function, construction
  3680. should not start until Set has been pressed. The Redraw
  3681. return code can be used to tell the editor how to
  3682. efficiently redraw the view.
  3683.  
  3684.  
  3685.  
  3686.  
  3687. virtual Redraw OnMouseMove(Vec &vec);
  3688.  
  3689. vec            Current position of the 3D cursor.
  3690.  
  3691. Comments
  3692.  
  3693. Called when the 3D cursor moves within a view. NB.
  3694. Windows programmers should realise that this is not quite
  3695. the same as a WM_MOUSEMOVE message as the left button has
  3696. to be depressed while the mouse is over a view.
  3697.  
  3698.  
  3699.  
  3700.  
  3701. virtual Redraw OnButtonUp(Vec &vec);
  3702.  
  3703. vec            Current position of the 3D cursor.
  3704.  
  3705. Comments
  3706.  
  3707. Called when the tool is selected and the user releases
  3708. the left mouse button when the mouse is in a view.
  3709.  
  3710.  
  3711.  
  3712.  
  3713. virtual Redraw OnSet(Vec &vec);
  3714.  
  3715. vec            Current position of the 3D cursor.
  3716.  
  3717. Comments
  3718.  
  3719. Called when the 'Set' button is clicked after a tool has
  3720. been selected.
  3721.  
  3722.  
  3723.  
  3724.  
  3725. virtual Redraw OnUndo();
  3726.  
  3727. Comments
  3728.  
  3729. Called when user wishes to undo the effect of a tool.
  3730. This can be called even if the tool is not selected. If
  3731. the currently selected tool has not modified the scene in
  3732. any way and the user selects `Undo' from the `Edit' menu,
  3733. then the last tool to call ModifiedScene is called to
  3734. undo.
  3735.  
  3736. If we are not the selected tool we should only return
  3737. REDRAW_ALL after undoing, as anything else might effect
  3738. whatever the selected tool has drawn into the view(s).
  3739. For instance returning REDRAW_NOTOOL will erase whatever
  3740. the selected tool has drawn.
  3741.  
  3742.  
  3743.  
  3744.  
  3745. virtual Redraw OnDo(Vec &vec);
  3746.  
  3747. vec            Current position of the 3D cursor.
  3748.  
  3749. Comments
  3750.  
  3751. Called when the user presses the `Do' button on the
  3752. control bar. Typically this is used to end use of the
  3753. selected tool.
  3754.  
  3755.  
  3756.  
  3757.  
  3758. virtual void OnViewChange(HanCoorSys hcsys, Vec &vec);
  3759.  
  3760. hcsys               The co-ordinate system of the new
  3761. active view.
  3762.  
  3763. vec            Current position of the 3D cursor.
  3764.  
  3765. Comments
  3766.  
  3767. Called when the user clicks the mouse in a new view
  3768. making it active. Since there is a separate object of
  3769. each tool type for each scene the tool will not be
  3770. notified of a view change into a different scene. In fact
  3771. the new active scene could have a different tool type
  3772. currently selected.
  3773.  
  3774. This event is also called if a tool calls the base class
  3775. version of AddCoorSys with the bAddView parameter set to
  3776. TRUE. In this case the OnViewChange event is called
  3777. before the AddCoorSys call returns.
  3778.  
  3779.  
  3780.  
  3781.  
  3782. virtual Redraw OnPaste(Vec &vec,
  3783.                      HanObject *ahobj,
  3784.                      ulong ulNumObjects);
  3785.                      
  3786. vec            Current position of the 3D cursor.
  3787.  
  3788. ahobj               A list of ulNumObjects object
  3789. handles.
  3790.  
  3791. ulNumObjects        The number of objects pasted into the
  3792. view.
  3793.  
  3794. Comments
  3795.  
  3796. Only the paste receiver tool receives this event. When a
  3797. clipboard paste operation is perform the tool is
  3798. automatically selected and this event is called to
  3799. position one or more objects in the view. To make a tool
  3800. the paste receiver tool the PasteReceiver function must
  3801. be called. In the default configuration the move tool is
  3802. also the paste receiver.
  3803.  
  3804. When the paste receiver moves the pasted objects it
  3805. should not consider this as modifying the scene, so it
  3806. should not call ModifyScene as a result. When a paste is
  3807. performed the `undo' tool (the one which gets called to
  3808. undo the last action if undo is selected from the menu)
  3809. is set internally to the editors clipboard. If undo is
  3810. selected the editor deletes all pasted objects. If the
  3811. paste receiver calls ModifiedScene then the it will be
  3812. called to undo the move only (not the entire paste).
  3813.  
  3814.  
  3815.  
  3816.  virtual void FreeUndo();
  3817.  
  3818. vec           Position to set the cursor too.
  3819.  
  3820. Comments
  3821.  
  3822. When a tool modifies a scene, whether its creating or
  3823. deleting part of an object, moving something or creating
  3824. a light etc. it must call ModifiedScene. This marks the
  3825. tool as the one which must perform an `undo' if the user
  3826. requests it. Sometimes a tool will need to allocate
  3827. memory or other resources in order to be able to undo
  3828. what it has done. If another tool then modifies the scene
  3829. this one becomes the `undo' tool and the previous tool
  3830. can then free up its resources.
  3831.  
  3832. This event is used to tell a tool that if it allocated
  3833. any resources for the purposes of undoing, then they are
  3834. no longer required and it can free them. If the same tool
  3835. modifies a scene twice, say the sphere tool is used to
  3836. create two spheres without another tool being selected in
  3837. the mean time, it can (but doesn't need to) call
  3838. ModifiedScene twice. However it will only receive one
  3839. FreeUndo event when another tool modifies the scene. The
  3840. tool should therefore free any resources it has kept
  3841. before the second operation.
  3842.  
  3843. The subtract tool creates two archives of the objects it
  3844. uses in a subtract operation so that it can re-create the
  3845. original objects if undo is selected. When it receives
  3846. this event if deletes the archives.
  3847.  
  3848.  
  3849. virtual void OnCommand(ushort usID)
  3850.                      
  3851. usID           The command ID of the menu item or toolbar
  3852. button.
  3853.  
  3854. Comments
  3855.  
  3856. Called when a menu item or toolbar button created by this
  3857. tool is selected by the user. Menu items and toolbars can
  3858. be added to the main frame window using standard windows
  3859. calls. The tool must however use only the range of IDs
  3860. from 0x9000 to 0x9009.
  3861.  
  3862. The editor alone will receive notifications that these
  3863. have been selected. These notifications are passed on to
  3864. the tool using this event.
  3865.  
  3866.  
  3867.  
  3868.  
  3869. virtual CmdUpdate OnCommandUpdate(ushort usID,
  3870.                                     char *szText,
  3871.                                     ushort usBuffSize)
  3872.                      
  3873. usID           The command ID of the menu item or toolbar
  3874. button.
  3875.  
  3876. szText         Can be used to set the text of a menu
  3877.                item. This is ignored unless we return
  3878.                SETTEXT.
  3879.  
  3880. usBuffSize          The size of the supplied szText
  3881. buffer.
  3882.  
  3883. Comments
  3884.  
  3885. This event is called when windows requires the status of
  3886. a menu item or toolbar button to be updated and is
  3887. usually called when the menu drops down or when there are
  3888. no other messages in the queue for toolbar buttons. The
  3889. value returned determines the status of the control. If
  3890. SETTEXT is returned the supplied buffer should contain
  3891. the text to set the menu item to. Possible return values
  3892. are;
  3893.  
  3894.                ENABLE
  3895.                DISABLE
  3896.                CHECK
  3897.                UNCHECK
  3898.                INDETER
  3899.                RADIOON
  3900.                RADIOOFF
  3901.                SETTEXT
  3902.  
  3903.  
  3904. virtual void DrawSoFar(HDC hdc,
  3905.                      HanCoorSys hcsys,
  3906.                      void *pvViewData,
  3907.                      bool bInvalid);
  3908.  
  3909. hdc            A handle to the device context of the view
  3910. being redrawn.
  3911.  
  3912. hcsys               Co-ordinate system handle of the view
  3913. being redrawn.
  3914.  
  3915. pvViewData          Pointer to instance data for the
  3916. view.
  3917.  
  3918. bInvalid      The region being redrawn was invalidated,
  3919.               probably by a window on top being removed.
  3920.  
  3921. Comments
  3922.  
  3923. Called when the view is being redrawn to give the tool a
  3924. chance to draw the `object so far' into the view.
  3925.  
  3926.  
  3927.  
  3928. Support
  3929.  
  3930.  
  3931. Protected member variables
  3932.  
  3933. hinst               Instance handle of the derived tool
  3934. DLL.
  3935.  
  3936. hwnd           Handle to the main frame window.
  3937.  
  3938. hcsysTopLevel  Handle of top level co-ordinate system in
  3939. this scene.
  3940.  
  3941. hsurActive     Currently active surface type. This is the
  3942.                selected palette entry or the selected
  3943.                surface in the surface type dialog box.
  3944.                The value is NULL_HANDLE if no surface is
  3945.                selected.
  3946.  
  3947. usSmoothFlags  A set of autosmooth flags as defined by
  3948.                the Autosmooth dialog box. The flags are
  3949.                the same set of autosmooth flags that are
  3950.                passed to geometry's DefPatch call.
  3951.  
  3952. fSmoothAng     The autosmooth threshold angle as set by
  3953.                the autosmooth dialog box expressed in
  3954.                radians.
  3955.  
  3956. hcsysActive    The co-ordinate system of the currently
  3957.                active view.
  3958.  
  3959. bOverlay       Is TRUE if the overlay is switched on in
  3960.                the active view (hcsysActive).
  3961.  
  3962. Mode           The mode of the active view (hcsysActive)
  3963.  
  3964. ahverPrim     Array of handles to vertices created
  3965.               (ulNumVerts contains how many). See
  3966.               StartPrim for details.
  3967.  
  3968. hverPrim       Last handle created by the base class's
  3969. AddVertex call.
  3970.  
  3971. ulNumVerts          Number of vertex handles in
  3972. ahverPrim.
  3973.  
  3974. ahnorPrim     Array of handles to normals created
  3975.               (ulNumNorms contains how many). See
  3976.               StartPrim for details.
  3977.  
  3978. hnorPrim       Last handle created by the base class's
  3979. AddNormal call.
  3980.  
  3981. ulNumNorms          Number of normal handles in
  3982. ahnorPrim.
  3983.  
  3984. ahpatPrim      Array of handles to patches created
  3985.                (ulNumPats contains how many). See
  3986.                StartPrim for details
  3987.  
  3988. ulNumPats      Number of patch handles in ahpatPrim.
  3989.  
  3990.  
  3991.  
  3992.  
  3993. void WriteMessage(int iResId);
  3994. void WriteMessage(const char *szMsg);
  3995.  
  3996. iResId              The resource id of the string within
  3997. the DLLs resource file.
  3998.  
  3999. szMsg               Pointer to a null terminated string.
  4000.  
  4001. Comments
  4002.  
  4003. Called to write a message in status bar at the bottom of
  4004. the main window.
  4005.  
  4006.  
  4007.  
  4008.  
  4009. int MessageBox(int iTitleId, int iMsgId,
  4010.               UINT uiStyle =MB_ICONEXCLAMATION);
  4011. int MessageBox(int iTitleId, char *szMsg,
  4012.               UINT uiStyle =MB_ICONEXCLAMATION);
  4013. int MessageBox(int iTitleId, GeomErr gerr);
  4014.  
  4015. iTitleId       The resource id of a string to use for the
  4016. message box title.
  4017.  
  4018. iMsgId              The resource id of the main message
  4019. string.
  4020.  
  4021. szMsg               A pointer to the message string.
  4022.  
  4023. gerr          A Geometry error value. The routine will
  4024.               query an appropriate error string from
  4025.               Geometry, and will use the
  4026.               MB_ICONEXCLAMATION style.
  4027.  
  4028. uiStyle       One of windows MB_ values to indicate the
  4029.               type of message box. This defaults to
  4030.               MB_ICONEXCLAMATION if omitted.
  4031.  
  4032. Comments
  4033.  
  4034. Puts a message box on the screen. There are three
  4035. variations on this function. If a tool gets an internal
  4036. processing error return code from Geometry it should use
  4037. the third form to notify the user, and then abort
  4038. whatever action it was performing.
  4039.  
  4040. The return code is the same as for the windows MessageBox
  4041. function.
  4042.  
  4043.  
  4044.  
  4045.  
  4046. bool StartPrim (HanObject hobj,
  4047.              ulong ulMaxVerts,
  4048.              ulong ulMaxNorms,
  4049.              ulong ulMaxPats,
  4050.              bool bReportErrs,
  4051.              bool bUndoOnErr);
  4052.  
  4053. hobj           Handle to the object we want to add to.
  4054.  
  4055. ulMaxVerts          Maximum number of vertices we will be
  4056. adding.
  4057.  
  4058. ulMaxNorms          Maximum number of normals we will be
  4059. adding.
  4060.  
  4061. ulMaxPats      Maximum number of patches we will be
  4062. adding.
  4063.  
  4064. bReportErrs   If this is set to TRUE any Geometry errors
  4065.               that occur during AddVertex, AddNormal, or
  4066.               DefPatch (e.g. `out of memory' or
  4067.               `vertices not on a plane') will be
  4068.               reported to the user in an message box.
  4069.  
  4070. bUndoOnErr    If this is set to TRUE and a Geometry
  4071.               error occurs then every patch, vertex and
  4072.               normal added to the object since the
  4073.               StartPrim call will be removed.
  4074.  
  4075. Comments
  4076.  
  4077. This routine can be used along with the base class's
  4078. AddVertex, AddNormal and DefPatch routines to add
  4079. elements to an object. They are usually more convenient
  4080. than Geometry's version of these functions because the
  4081. handles to the elements are automatically stored in
  4082. arrays allocated by this routine. For instance a tool
  4083. will usually want to access say the handle for the 8th
  4084. vertex that it created. If we use the base class version
  4085. of the AddVertex function we can access this handle using
  4086. the ahverPrim protected member variable of the tool base
  4087. class, i.e.. ahverPrim[7] for the 8th vertex. Similarly
  4088. ahnorPrim[7] will access the 8th normal created and
  4089. ahpatPrim[7] will give you the 8th patch. What is more,
  4090. if we need to undo, the RemovePrim function will remove
  4091. all the elements created so far.
  4092.  
  4093. It should be considered that some geometry engines will
  4094. only support triangular patches so each call to DefPatch
  4095. could result in several patches being created. Make sure
  4096. you leave enough room in the ulMaxPats parameter for the
  4097. maximum number of patches that could result.
  4098.  
  4099. Once the tool has finished constructing and we have no
  4100. further interest in the handles, we must call the EndPrim
  4101. function to release the handle arrays. If during any
  4102. `Add' function a serious error is returned by Geometry a
  4103. description of the error can be automatically reported to
  4104. the user and the RemovePrim function called to remove all
  4105. elements created. The return code is then passed back to
  4106. the tool. In this case you must not call EndPrim.
  4107.  
  4108. If the tool has any reason to call RemovePrim itself it
  4109. should not call EndPrim as the RemovePrim routine does
  4110. this for you.
  4111.  
  4112. If you call say AddVertex to create more vertices than
  4113. you originally specified on the StartPrim call the
  4114. function will not fail, but the handle will not be stored
  4115. either, therefore if you call RemovePrim you might find
  4116. it doesn't remove all of your object.
  4117.  
  4118. StartPrim can be called with an object already containing
  4119. vertices, patches and normals.
  4120.  
  4121. This routine returns TRUE if it succeeded.
  4122.  
  4123.  
  4124.  
  4125.  
  4126. void EndPrim ();
  4127.  
  4128. Comments
  4129.  
  4130. Called after a StartPrim and when the tool has no further
  4131. need for handles to the elements its created. This
  4132. routine removes the buffers allocated by StartPrim.
  4133.  
  4134.  
  4135.  
  4136.  
  4137. void RemovePrim ();
  4138.  
  4139. Comments
  4140.  
  4141. Removes all primitive elements (verts, normals and
  4142. patches) added to the object the tool is working on.
  4143. Should be called only if a StartPrim has been called.
  4144.  
  4145.  
  4146.  
  4147.  
  4148. GeomErr AddVertex (Vec &vec);
  4149.  
  4150. vec            Position of vertex.
  4151.  
  4152. Comments
  4153.  
  4154. Adds a vertex  to the object specified by a call to
  4155. StartPrim and records its handle in the ahverPrim array.
  4156. The hver base class variable can be used as a more
  4157. convenient way of accessing the last handle created by an
  4158. AddVertex call.
  4159.  
  4160. If Geometry reports an error this is passed to the user
  4161. in a message box.
  4162.  
  4163. This call can only be used after a call to StartPrim. See
  4164. StartPrim comment and Geometry's AddVertex function for a
  4165. full description.
  4166.  
  4167.  
  4168.  
  4169.  
  4170. GeomErr AddNormal (Vec &vec);
  4171.  
  4172.  
  4173. vec            Direction of normal.
  4174.  
  4175. Comments
  4176.  
  4177. Adds a normal  to the object specified by a call to
  4178. StartPrim and records its handle in the ahnorPrim array.
  4179. The hnor base class variable can be used as a more
  4180. convenient way of accessing the last handle created by an
  4181. AddNormal call.
  4182.  
  4183. If Geometry reports an error this is passed to the user
  4184. in a message box.
  4185.  
  4186. This call can only be used after a call to StartPrim. See
  4187. StartPrim comment and Geometry's AddNormal function for a
  4188. full description.
  4189.  
  4190.  
  4191.  
  4192.  
  4193. GeomErr DefPatch (ushort usNumEdges,
  4194.                  ushort usFlags,
  4195.                  float fSmoothAng,
  4196.                  HanVertex *ahver,
  4197.                  HanNormal *ahnor,
  4198.                  HanSurf hsur);
  4199.  
  4200. usNumEdges    The number of edges, and therefore
  4201.               vertices and normals in the patch. This
  4202.               can be any number from 3 upwards.
  4203.  
  4204. usFlags       Flags controlling the creation of the
  4205.               patch.
  4206.  
  4207.  fSmoothAng   Specifies the angle to use for
  4208.               autosmoothing expressed in radians. Only
  4209.               takes effect if DP_SMOOTH_BY_ANGLE flag is
  4210.               used.
  4211.  
  4212. ahver         Array of handles to vertices. The number
  4213.               of vertices is given by the usNumEdges
  4214.               parameter.
  4215.  
  4216. ahnor         Array of handles to normals. The number of
  4217.               normals is given by the usNumEdges
  4218.               parameter.
  4219.  
  4220. hsur          Handle to the surface type to be used for
  4221.               the patch. If you use NULL_HANDLE here
  4222.               then the currently selected surface type
  4223.               will be used. If no surface type has been
  4224.               selected then any available type will be
  4225.               used. If no surface types exists then one
  4226.               will be created.
  4227.  
  4228. Comments
  4229.  
  4230. Adds a patch to the object specified by a call to
  4231. StartPrim and records its handle in the ahpatPrim array.
  4232. Unlike AddVertex and AddNormal, DefPatch doesn't have an
  4233. equivalent hpat variable as the call could result in
  4234. several patches being created. The ahpat array must be
  4235. used to access the handles.
  4236.  
  4237. If Geometry reports an error this is passed to the user
  4238. in a message box.
  4239.  
  4240. This call can only be used after a call to StartPrim. See
  4241. StartPrim comment and Geometry's DefPatch function for a
  4242. full description.
  4243.  
  4244.  
  4245.  
  4246.  
  4247. HanObject AddObject(HanCoorSys hcsys,
  4248.                     char *szName,
  4249.                     ushort usBuffSize,
  4250.                     float fNewVertTol,
  4251.                     HanObject hobj)
  4252.  
  4253. hcsys         Handle to co-ordinate system to add object
  4254.               too.
  4255.  
  4256. szName        The name of the object to appear in the
  4257.               object list box. The actual name used is
  4258.               returned in this buffer as the name
  4259.               supplied might already be in use. If this
  4260.               is the case a number is added to the end
  4261.               of the name.
  4262.  
  4263. usBuffSize    Size of the szName buffer. To ensure that
  4264.               the modified name will fit you can set
  4265.               this to MAX_NAME_BUFF_LEN but ensure that
  4266.               the supplied string uses no more than
  4267.               MAX_NAME_LEN of the buffer. If the actual
  4268.               name used is of no interest this can be
  4269.               set to zero,  so the buffer will not be
  4270.               modified. This allows use of a literal
  4271.               string, e.g. AddObject(hcsys, "box", 0...)
  4272.  
  4273. fNewVertTol   The tolerance setting for creating new
  4274.               vertices rather than reusing old ones.
  4275.  
  4276. hobj          If the object already exists within
  4277.               Geometry and you simply want to give it a
  4278.               name, the existing object handle can be
  4279.               supplied here.
  4280.  
  4281. Comments
  4282.  
  4283. This function does not have to be used in conjunction
  4284. with StartPrim and EndPrim. It simply calls Geometry's
  4285. AddObject function to start an empty object in the given
  4286. co-ordinate system. However by supplying a name, the
  4287. object gets added to the object list on the control panel
  4288. thereby allowing the entire object to be selected by
  4289. highlighting its name.
  4290.  
  4291. To delete this object the standard geometry function
  4292. ::DelObject can be used.
  4293.  
  4294.  
  4295.  
  4296.  
  4297. HanCoorSys AddCoorSys(HanCoorSys hcsysParent,
  4298.                        char *szName,
  4299.                        ushort usBuffSize,
  4300.                        HanCoorSys hcsys,
  4301.                        Mat &matToParent,
  4302.                        Mat &matFromParent,
  4303.                        ushort usType,
  4304.                        bool bAddView);
  4305.  
  4306. hcsysParent   Handle to the parent co-ordinate system.
  4307.  
  4308. szName        The name of the coor sys to appear in the
  4309.               list box. The actual name used is returned
  4310.               in this buffer as the name supplied might
  4311.               already be in use. If this is the case a
  4312.               number is added to the end of the name.
  4313.  
  4314. usBuffSize    Size of the szName buffer. To ensure that
  4315.               the modified name will fit you can set
  4316.               this to MAX_NAME_BUFF_LEN but ensure that
  4317.               the supplied string uses no more than
  4318.               MAX_NAME_LEN of the buffer. If the actual
  4319.               name used is of no interest this can be
  4320.               set to zero,  so the buffer will not be
  4321.               modified. This allows use of a literal
  4322.               string, e.g. AddCoorSys(hcsys, "new",
  4323.               0...)
  4324.  
  4325. hcsys         If the coor sys already exists within
  4326.               Geometry and you simply want to give it a
  4327.               name, the existing handle can be supplied
  4328.               here.
  4329.  
  4330. matToParent   Matrix describing the transformation from
  4331.               child to parent. In other words this
  4332.               matrix transforms a point in the child co-
  4333.               ordinate system to its corresponding
  4334.               position relative to the parent co-
  4335.               ordinate system.
  4336.  
  4337. matFromParent The inverse or opposite matrix to
  4338.               matToParent. NB. The maths library can
  4339.               easily generate the inverse of a matrix by
  4340.               preceding it with a minus sign (e.g. -
  4341.               mat).
  4342.  
  4343. usType             This value indicates whether the co-
  4344.               ordinate system being defined is right
  4345.               handed or left handed. It should be either
  4346.               CT_RIGHTHAND or CT_LEFTHAND.
  4347.  
  4348. bAddView       If this is set to TRUE a new view window
  4349. will appear.
  4350.  
  4351. Comments
  4352.  
  4353. Creates a new co-ordinate system as a child hcsysParent.
  4354. Its name will appear in the coor sys list box. There are
  4355. many advantages of using a named co-ordinate system
  4356. rather than creating an unnamed one using the Geometry
  4357. API, for instance, a named co-ordinate system can be used
  4358. in the surface type dialog to specify an orientation for
  4359. a texture.
  4360.  
  4361. To delete this co-ordinate system the standard geometry
  4362. function ::DelCoorSys can be used.
  4363.  
  4364.  
  4365.  
  4366.  
  4367.  
  4368. HanObject GetObjectHandle(const char *szName);
  4369.  
  4370. szName         The name of the object whose handle is to
  4371. be returned.
  4372.  
  4373. Comments
  4374.  
  4375. Returns the handle of the named object. A NULL_HANDLE is
  4376. returned if the name was not recognised.
  4377.  
  4378.  
  4379.  
  4380.  
  4381. HanObject GetCoorSysHandle(const char *szName);
  4382.  
  4383. szName         The name of the coor sys whose handle is
  4384. to be returned.
  4385.  
  4386. Comments
  4387.  
  4388. Returns the handle of the named co-ordinate system. A
  4389. NULL_HANDLE is returned if the name was not recognised.
  4390.  
  4391.  
  4392.  
  4393.  
  4394. void GetObjectName(HanObject hobj, char *szName,
  4395.                    ushort usBuffSize);
  4396.  
  4397. hobj          The handle of the object whose name we
  4398.               want returned.
  4399.  
  4400. szName        The name of the object.
  4401.  
  4402. usBuffSize    Length of the szName buffer.
  4403.  
  4404. Comments
  4405.  
  4406. Returns the name of a given object. A zero length string
  4407. is returned if the handle was not recognised.
  4408.  
  4409.  
  4410.  
  4411.  
  4412. void GetCoorSysName(HanCoorSys hcsys, char *szName,
  4413.                     ushort usBuffSize);
  4414.  
  4415. hcsys         The handle of the coor sys whose name we
  4416.               want returned.
  4417.  
  4418. szName        The name of the coor sys.
  4419.  
  4420. usBuffSize    Length of the szName buffer.
  4421.  
  4422. Comments
  4423.  
  4424. Returns the name of a given coor sys. A zero length
  4425. string is returned if the handle was not recognised.
  4426.  
  4427.  
  4428.  
  4429.  
  4430. void ForceRedraw(Redraw rd,
  4431.                 HanCoorSys hcsys=NULL_HANDLE);
  4432.  
  4433. rd            The redraw code to use.
  4434.  
  4435. hcsys         The handle of the co-ordinate system whose
  4436.               view is to be redrawn. If more than one
  4437.               view window exists for this coor sys they
  4438.               are all redrawn. If a NULL_HANDLE is
  4439.               passed, every view into the scene (i.e..
  4440.               all co-ordinate systems) is redrawn.
  4441.  
  4442. Comments
  4443.  
  4444. Forces a redraw of one or more views. This function is
  4445. useful if a tool constructs a dialog box which has
  4446. buttons that can be invoked independently of any events
  4447. from the editor. For instance the coor sys editing tool
  4448. has a dialog whose buttons could invoke a redraw. All
  4449. resulting DrawSoFar calls will be made before this call
  4450. returns.
  4451.  
  4452.  
  4453.  
  4454.  
  4455. void DrawMarker(HDC hdc, int iX, int iY, ushort usType);
  4456.  
  4457. hdc            Handle to device context to draw into.
  4458.  
  4459. iX             Device context X co-ordinate for marker.
  4460.  
  4461. iY             Device context Y Co-ordinate for marker.
  4462.  
  4463. usType              Type of marker to draw. Can be one of
  4464. the following values;
  4465.  
  4466.                MT_BOX         A small square marker
  4467.                MT_PLUS        A `+' shaped marker
  4468.                MT_CROSS       An `X' type cross marker
  4469.                MT_DIAMOND          A small diamond shaped
  4470. marker
  4471.  
  4472. Comments
  4473.  
  4474. Draws a marker into the device context at the specified
  4475. position using the current attribute (pen, colour, mode
  4476. etc.) settings of the DC.
  4477.  
  4478.  
  4479.  
  4480.  
  4481. void DrawArrow(HDC hdc, int iX, int iY, ushort usFlags);
  4482.  
  4483. hdc           Device context to draw into.
  4484.  
  4485. iX            Device context x co-ordinate of the arrow
  4486.               head.
  4487.  
  4488. iY            Device context y co-ordinate of the arrow
  4489.               head.
  4490.  
  4491. usFlags       Type of marker to draw. Can be one of the
  4492.               following values;
  4493.  
  4494.               DA_FILLED     The arrow head is filled in
  4495.               DA_FIXEDSIZE  The arrow head is fixed in
  4496.                              size as opposed to
  4497.                              proportional to the length
  4498.                              of the arrow body
  4499.  
  4500. Comments
  4501.  
  4502. Draws an arrow from the current position in the device
  4503. context to iX, iY.
  4504.  
  4505.  
  4506.  
  4507.  
  4508. bool ActivateCSysWin(HanCoorSys hcsys);
  4509.  
  4510. hcsys         Handle of co-ordinate system whose view is
  4511.               to be activated.
  4512.  
  4513. Comments
  4514.  
  4515. Activates the view window associated with a co-ordinate
  4516. system. This means bringing it on top of the other
  4517. windows and highlighting its title bar. An OnViewChange
  4518. event will be received before this call returns.
  4519.  
  4520.  
  4521.  
  4522.  
  4523. void ModifiedScene();
  4524.  
  4525. Comments
  4526.  
  4527. This routine must be called after any tool modifies the
  4528. scene in any way. Not only does it mark the tool as the
  4529. one which should perform an undo if the user requests it,
  4530. but also it marks the document as modified such that a
  4531. warning will be issued if an attempt is made to close the
  4532. editor without saving it first.
  4533.  
  4534.  
  4535.  
  4536.  
  4537. void SetViewData(void *pvData, ushort usSize);
  4538.  
  4539. pvData        Pointer to the view data.
  4540.  
  4541. usSize        Size of the view data.
  4542.  
  4543. Comments
  4544.  
  4545. Sets the view data for every view in the scene to the
  4546. contents of pvData.
  4547.  
  4548.  
  4549.  
  4550.  
  4551. void BackgroundYield(const char *szWrite=NULL);
  4552.  
  4553. szWrite       An optional string which is written to the
  4554.               status bar at the bottom of the screen.
  4555.  
  4556. Comments
  4557.  
  4558. If a tool has a lot of processing to perform which could
  4559. take several seconds, even minutes, then somewhere in its
  4560. processing loop it should call this function. This
  4561. prevents the entire system being `locked up' during the
  4562. processing. The optional string can be used to print a
  4563. message to status bar to say for instance; "25% done".
  4564. Notice how the CSG tools use this.
  4565.  
  4566.  
  4567.  
  4568.  
  4569. void *AddUserData(HanObject hobj,
  4570.                   Handle han,
  4571.                   HandleType htype,
  4572.                   ulong ulSize);
  4573.  
  4574. hobj          The parent object of the item to add the
  4575.               data to. This parameter is only needed if
  4576.               the item is a patch, a vertex or a normal.
  4577.  
  4578. han           The handle of the item. This can be any
  4579.               standard geometry handle.
  4580.  
  4581. htype         The type of the `han' handle. This can be
  4582.               any one of;
  4583.  
  4584.                 COORSYS_HANDLE
  4585.                 OBJECT_HANDLE
  4586.                 PATCH_HANDLE
  4587.                 VERTEX_HANDLE
  4588.                 NORMAL_HANDLE
  4589.                 SURFACE_HANDLE
  4590.                 LIGHT_HANDLE
  4591.  
  4592. ulSize        Size of the user data required.
  4593.  
  4594. Comments
  4595.  
  4596. If a tool needs to store instance data for a geometry
  4597. item, say a patch, light or co-ordinate system, this
  4598. routine should be used. It is not safe to allocate memory
  4599. and use geometry's SetUserData call to store the pointer
  4600. in the items user data, as many other tools and the
  4601. editor itself may also need to store instance data. Using
  4602. this routine not only allocates the memory for you but
  4603. chains together the user data blocks from all other
  4604. tools. The returned pointer points to a block of memory
  4605. of ulSize bytes.
  4606.  
  4607. Only one set of user data can exist for each tool. If
  4608. more is needed the first lot should be deleted using
  4609. RemoveUserData before calling here again.
  4610.  
  4611. The user data is automatically deleted when the item is
  4612. deleted from geometry.
  4613.  
  4614.  
  4615.  
  4616.  
  4617. void *GetUserData(HanObject hobj,
  4618.                  Handle han,
  4619.                  HandleType htype);
  4620.  
  4621. hobj          The parent object the item belongs to.
  4622.               This is only required if the item is a
  4623.               patch, a vertex or a normal.
  4624.  
  4625. han           The handle of the item we want the user
  4626.               data for.
  4627.  
  4628. htype         The type of the handle (see AddUserData).
  4629.  
  4630. Comments
  4631.  
  4632. The pointer returned points to the user data created by
  4633.               this tool for the geometry item given by
  4634.               `han'.
  4635.  
  4636.  
  4637.  
  4638.  
  4639. bool RemoveUserData(HanObject hobj,
  4640.                     Handle han,
  4641.                     HandleType htype);
  4642.  
  4643. hobj          The parent object the item belongs to.
  4644.               This is only required if the item is a
  4645.               patch, a vertex or a normal.
  4646.  
  4647. han           The handle of the item we want the user
  4648.               data deleted from.
  4649.  
  4650. htype         The type of the handle (see AddUserData).
  4651.  
  4652. Comments
  4653.  
  4654. Removes the user data allocated by a tool using the
  4655. AddUserData call. If TRUE is returned the memory was
  4656. freed successfully.
  4657.  
  4658.  
  4659.  
  4660.  
  4661. bool IsSelected(HanObject hobj,
  4662.               Handle han,
  4663.               HandleType htype);
  4664.  
  4665. hobj          The parent object of the patch or vertex
  4666.               we want to query. This is ignored if we
  4667.               are querying an object.
  4668.  
  4669. han           The handle of the item we want to query.
  4670.  
  4671. htype         The type of the handle. This can be any
  4672.               one of;
  4673.                 
  4674.                 OBJECT_HANDLE
  4675.                 PATCH_HANDLE
  4676.                 VERTEX_HANDLE
  4677.  
  4678. Comments
  4679.  
  4680. Returns TRUE if the geometry item is selected.
  4681.  
  4682.  
  4683.  
  4684. Handle GetNextSelected(Handle hanParent,
  4685.                        ulong *pulRef,
  4686.                        HandleType htype);
  4687.  
  4688. hanParent     The parent item of the item we are
  4689.               querying. If we want to find the next
  4690.               selected object in a specific co-ordinate
  4691.               system we must set this to the handle of
  4692.               the co-ordinate system. If we want to find
  4693.               the next selected patch or vertex in an
  4694.               object, this must be set to the owning
  4695.               object.
  4696.  
  4697. pulRef        A reference value used by this routine. To
  4698.               query the first selected object in a co-
  4699.               ordinate system or the first vertex/patch
  4700.               in an object we must first set the value
  4701.               pointed to, to zero.
  4702.  
  4703. htype         The type of the handle. This can be any
  4704.               one of;
  4705.                 
  4706.                 OBJECT_HANDLE
  4707.                 PATCH_HANDLE
  4708.                 VERTEX_HANDLE
  4709.  
  4710. Comments
  4711.  
  4712. Returns the handle of the next selected object in a co-
  4713. ordinate system or the next selected vertex or patch in
  4714. an object. Object, vertices and patches are selected
  4715. using the editor `select mode'. The first call must be
  4716. made with (*pulRef) set to zero. Subsequent calls will
  4717. then return all selected items. When NULL_HANDLE is
  4718. returned there are no more selected items in the
  4719. object/co-ordinate system. An object is selected if all
  4720. of its vertices are selected.
  4721.  
  4722.  
  4723.  
  4724. HanCoorSys QryNextCoorSys(ulong *pulRef,
  4725.                             CoorSysInfo *pcsi);
  4726.  
  4727. pulRef        A pointer to a reference value that this
  4728.               routine uses to find its place in the
  4729.               enumeration. On the first call to this
  4730.               routine set (*pulRef) to NULL_HANDLE).
  4731.  
  4732. pcsi          A pointer to a structure used to receive
  4733.               information about the next co-ordinate
  4734.               system.
  4735.  
  4736. Comments
  4737.  
  4738. It is quite common to have to write code which performs
  4739. processing for every co-ordinate system in a scene.
  4740. Because of the hierarchical nature of the co-ordinate
  4741. systems the code needed to enumerate them (return each in
  4742. sequence) can be quite complex. This is a handy routine
  4743. which can be called multiple times to return a query
  4744. structure for each co-ordinate system in the scene.
  4745.  
  4746. The first time the routine is called (*pulRef) should be
  4747. set to NULL_HANDLE. This routine then returns the handle
  4748. of the next coordinate system in the scene along with its
  4749. query structure.When NULL_HANDLE is returned there are no
  4750. more co-ordinate systems to process. The following code
  4751. can be used to perform processing for all co-ordinate
  4752. systems in a scene;
  4753.  
  4754.       ulRef=NULL_HANDLE;
  4755.       hcsys=QryNextCoorSys(&ulRef, &csi);
  4756.       while (hcsys!=NULL_HANDLE)
  4757.            {
  4758.            /* Perform processing here using info from
  4759.       csi... */
  4760.       
  4761.            QryNextCoorSys(&ulRef, &csi);
  4762.            }
  4763.  
  4764.  
  4765.  
  4766. void EditInThisView(HanCoorSys hcsys);
  4767.  
  4768. hcsys         Co-ordinate system to edit in. Use
  4769.               NULL_HANDLE to indicate any co-ordinate
  4770.               system can be used.
  4771.  
  4772. Comments
  4773.  
  4774. Most primitive creation type tools (for instance the
  4775. editor's sphere and box tools) will only allow you to
  4776. create the primitive in the view window (or co-ordinate
  4777. system) you started it in. For instance the sphere tool
  4778. wont allow you to press `Set' (for the sphere centre) in
  4779. one window and `Do' in another. This routine can be used
  4780. to say that any button presses must  be made when the
  4781. view for the specified co-ordinate system is active. If
  4782. any other is active the tool is not notified of the
  4783. event. This forces creation to happen in a specific
  4784. window. If this is used, when creation has finished or if
  4785. `Undo' is selected then you must call it again with
  4786. NULL_HANDLE. This ensures that the next operation can be
  4787. started in any window.
  4788.  
  4789.  
  4790.  
  4791. void MakeObjWireframe(HanObject hobj);
  4792.  
  4793. hobj          Object to add to the wireframe redraw
  4794.               list.
  4795.  
  4796. Comments
  4797.  
  4798. If a tool returns a REDRAW_OBJECT_WIRE code from an
  4799. event, any objects (using the base class AddObject call)
  4800. created since the tool was selected will be drawn in wire
  4801. frame rather than rendered. Subsequent return codes from
  4802. events cause the old image to be erased (using the GDI's
  4803. XOR mode) and redrawn enabling fast wireframe animation
  4804. which can be performed on the OnMouseMove event for
  4805. instance. However it might be that the tool requires an
  4806. existing object drawn  in wire frame. The move tool does
  4807. this for any objects which are selected when the mouse
  4808. moves. This function adds other objects to the wire frame
  4809. redraw list.
  4810.  
  4811.  
  4812.  
  4813. void NoneWireframe();
  4814.  
  4815. Comments
  4816.  
  4817. Clears the wire frame redraw list. See the
  4818. MakeObjWireframe call.
  4819.  
  4820.  
  4821.  
  4822. void PasteReceiver();
  4823.  
  4824. Comments
  4825.  
  4826. Nominates this tool as the paste receiver tool. Only one
  4827. tool can nominate itself as a paste receiver. In the
  4828. default configuration the move tool is the paste receiver
  4829. tool. When a clipboard paste operation is performed on
  4830. the editor the pasted object(s) need positioning in
  4831. exactly the same way you can paste into a 2D graphics
  4832. paint program (except in 3D). The paste receiver tool
  4833. will be automatically selected and will receive a special
  4834. OnPaste event when pasting occurs. The OnPaste event
  4835. passes the tool a list of handles for objects to be
  4836. positioned.
  4837.  
  4838.  
  4839.  
  4840. void AutosmoothDlg();
  4841.  
  4842. Comments
  4843.  
  4844. This can be called by a tool to bring up the Autosmooth
  4845. dialog box. Any changes made in this dialog will be
  4846. reflected in the public base class variables
  4847. usSmoothFlags and fSmoothAng. Most primitive creation
  4848. type tools use these to control the smoothness of the
  4849. primitives they create. Some tools also include a button
  4850. on their configuration dialogs to access this dialog.
  4851.  
  4852.  
  4853.  
  4854. void SurfaceDlg();
  4855.  
  4856. Comments
  4857.  
  4858. This can be called by a tool to bring up the Surface Type
  4859. dialog box. If the currently active surface type changes
  4860. as a result of the user clicking on another type then the
  4861. public base class variables hsurActive will reflect this.
  4862. Most primitive creation type tools use this as the
  4863. surface type to use for new primitives. Some tools also
  4864. include a button on their configuration dialogs to access
  4865. this dialog.
  4866.  
  4867.  
  4868.  
  4869. void ChangeMode(Mode mode);
  4870.  
  4871. mode          The mode to set the active view to.
  4872.               Possible values are;
  4873.                   
  4874.                   MODE_EDIT
  4875.                   MODE_VIEW
  4876.                   MODE_FLIGHT
  4877.                   MODE_SELECT
  4878.  
  4879. Comments
  4880.  
  4881. This call can be used to change the mode of the active
  4882. view. The edit mode is most useful for tools as mouse
  4883. movement in edit mode causes the 3D cursor to move. Most
  4884. tools require use of the 3D cursor to position and create
  4885. primitives. The view tool on the other hand is a very
  4886. basic tool which does nothing more than put the active
  4887. view into view mode. Using this call, tools have the
  4888. ability to change the mode on the users behalf to allow
  4889. the user to better view something it has just done or
  4890. select a set of patches to be worked on etc.
  4891.  
  4892.  
  4893.  
  4894. void SetCursor(Vec &vec);
  4895.  
  4896. vec           Position to set the cursor too.
  4897.  
  4898. Comments
  4899.  
  4900. Sets the position of the 3D cursor in the active view.
  4901. The light tool uses this to set the cursor to the
  4902. position of the light selected in its dialog box.
  4903.  
  4904.  
  4905.  
  4906. HanArchive ArchiveObject(HanCoorSys hcsysOrigin,
  4907.                          HanObject hobj,
  4908.                          bool bRemove);
  4909.  
  4910. hcsysOrigin   The co-ordinate system which the archive
  4911.               is stored relative to. All vertices and
  4912.               textures used by the object are converted
  4913.               relative to this co-ordinate system.
  4914.  
  4915. hobj          The object to store in the archive.
  4916.  
  4917. bRemove       Set to TRUE if the object should be
  4918.               deleted from geometry, as in a clipboard
  4919.               cut operation.
  4920.  
  4921. Comments
  4922.  
  4923. This call archives an object. An object archive is a
  4924. complete description of an object including all its
  4925. vertices, normals and any surface types that it uses
  4926. (including textures), in a block of memory. The handle
  4927. returned is not a geometry handle but a special base
  4928. class handle which identifies the archive. Archives are
  4929. extremely useful for recreating objects which have been
  4930. deleted (see the CreateFromArchive function).
  4931.  
  4932. The hcsysOrigin parameter does not have to be the co-
  4933. ordinate system the object belongs to. All vertex
  4934. position and texture positions will be converted relative
  4935. to hcsysOrigin.
  4936.  
  4937. The subtract tool uses archives to undo the effect of a
  4938. subtraction operation.
  4939.  
  4940. The archive handle is returned. If the function fails
  4941. NULL_HANDLE is returned.
  4942.  
  4943.  
  4944.  
  4945. HanObject CreateFromArchive(HanArchive harch,
  4946.                             HanCoorSys hcsys);
  4947.  
  4948. harch         The handle of the archive to extract from.
  4949.  
  4950. hcsys         The co-ordinate system we want to extract
  4951.               to. If NULL_HANDLE objects is extracted
  4952.               back into the co-ordinate system it was
  4953.               originally copied from.
  4954.  
  4955. Comments
  4956.  
  4957. This function recreates an object from an archive created
  4958. by the ArchiveObject call. The hcsys parameter determines
  4959. which co-ordinate system it will be created in. If this
  4960. is the same as the hcsysOrigin parameter used on the
  4961. ArchiveObject call then the object will be created in
  4962. exactly the same place in 3D space that it was taken
  4963. from, assuming of course it was deleted when originally
  4964. archived. If it was not deleted you will end up with two
  4965. identical objects in exactly the same place.
  4966.  
  4967. Just because the object might re-appear in the same place
  4968. doesn't mean of course that it belongs to the same co-
  4969. ordinate system. The co-ordinate system it belongs to now
  4970. is given by hcsys.
  4971.  
  4972. If hcsys is NULL_HANDLE the object is copied back into
  4973. its original co-ordinate system. The editor uses this
  4974. facility to undo a clipboard cut operation.
  4975.  
  4976. The return value is the handle of the newly created
  4977. object.
  4978.  
  4979.  
  4980.  
  4981. void DeleteArchive(HanArchive harch)
  4982.  
  4983. harch         The handle of the archive to delete.
  4984.  
  4985. Comments
  4986.  
  4987. Deletes an object archive.
  4988.  
  4989. Quick start to writing tools
  4990.  
  4991. The following source file is provided as a template to
  4992. work from when creating new tools. Remember you must
  4993. define a 16x15 pixel bitmap with an id of one in your
  4994. resource file, and you must export CreateTool and
  4995. DeleteTool in the .def file (the functions are
  4996. implemented by the IMPLEMENT_OBJECT macro).
  4997.  
  4998. From then on simply uncomment any events you are
  4999. interested in trapping. The whole of the geometry, tool
  5000. interface, maths and debug API's are available to you as
  5001. well as the windows API.
  5002.  
  5003. Finally put an entry in the Gened.ini file so the editor
  5004. knows of the existence of your tool.
  5005.  
  5006.  
  5007. /*-------------------------------------------------------
  5008. --------------------
  5009.                ExampleTool
  5010.                -----------
  5011.  
  5012.      Example tool class
  5013.  
  5014.      (C) Silicon Dream Ltd 1995
  5015.  
  5016.   -------------------------------------------------------
  5017. --------------------
  5018. Changes:                           Date:
  5019. * Created file                     01/06/96
  5020. */
  5021.  
  5022. #include <tool.h>
  5023.  
  5024. #include "resource.h"
  5025.  
  5026. // YOU DO: Change the class name below to describe your
  5027. tool and anywhere else
  5028. //      this class name is used in this file
  5029. class ExampleTool: public Tool
  5030.      {
  5031.      private:
  5032.           // YOU DO: Declare your private variables here.
  5033. eg.
  5034.           //  Vec        vecCursorPos;
  5035.           //      (Non C++ users: 'private' means those
  5036.           //       variables which have global scope from
  5037.           //       within the functions you define here)
  5038.  
  5039.           // YOU DO: Uncomment any functions to trap
  5040. events you're interested in
  5041.           // void _cppdyn Initialise();           //
  5042. Called once when DLL loaded
  5043.           // Redraw _cppdyn OnSelect(Vec &vec);        //
  5044. Called when tool is selected
  5045.           // Redraw _cppdyn OnUnSelect(bool
  5046. *pbOkToChange);     // Called when tool is selected
  5047.           // void _cppdyn OnConfigure();               //
  5048. Called to configure tool
  5049.           // Redraw _cppdyn OnButtonDown(Vec &vec);
  5050. // Called when button pressed
  5051.           // Redraw _cppdyn OnMouseMove(Vec &vec);
  5052. // Called when mouse moves with button down
  5053.           // Redraw _cppdyn OnButtonUp(Vec &vec);      //
  5054. Called when button is released
  5055.           // Redraw _cppdyn OnSet(Vec &vec);      //
  5056. Called when the 'Set' button is clicked
  5057.           // Redraw _cppdyn OnDo(Vec &vec);            //
  5058. Called to end use of tool
  5059.           // Redraw _cppdyn OnUndo();             //
  5060. Called when user wishes to undo effect of tool
  5061.           // void _cppdyn FreeUndo();             //
  5062. Called when tool no longer needs to keep..
  5063.           // void _cppdyn OnViewChange(HanCoorSys hcsys,
  5064. Vec &vec); // Called when active view changes
  5065.           // Redraw _cppdyn OnPaste(Vec &vec, HanObject
  5066. *ahobj,   // Called when objects are pasted into the..
  5067.           //           ulong ulNumObjects);
  5068.           // void _cppdyn OnCommand(ushort usID);      //
  5069. Called when a tool specific menu item..
  5070.           // CmdUpdate _cppdyn OnCommandUpdate(ushort
  5071. usID,,    // Called when a tool specific menu item..
  5072. //                char *szText, ushort usBuffSize);
  5073.           // void _cppdyn DrawSoFar(HDC hdc, HanCoorSys
  5074. hcsys,    // Called to draw object so far into the view
  5075.           //           void *pvViewData, bool bInvalid);
  5076.  
  5077.      public:
  5078.           _cppdyn ~ExampleTool() {};
  5079.      };
  5080.  
  5081. IMPLEMENT_OBJECT(ExampleTool)
  5082.  
  5083. // YOU DO: Uncomment and fill in any functions for events
  5084. you want to trap.
  5085. //      (Non C++ users: If you create any functions of
  5086. your own which need
  5087. //       to access your private variables (see above),
  5088. you must make the
  5089. //       funtion part of your class by declaring it along
  5090. with the other
  5091. //       functions (see above), and prefixing the
  5092. function name with
  5093. //       'classname::' in the implementation)
  5094.  
  5095. /*
  5096. void _cppdyn ExampleTool::Initialise()
  5097.      {
  5098.      }
  5099. */
  5100. /*
  5101. Redraw _cppdyn ExampleTool::OnSelect(Vec &vec)
  5102.      {
  5103.      return REDRAW_NONE;
  5104.      }
  5105. */
  5106. /*
  5107. Redraw _cppdyn ExampleTool::OnUnSelect(bool
  5108. *pbOkToChange)
  5109.      {
  5110.      *pbOkToChange=TRUE;
  5111.      return REDRAW_NONE;
  5112.      }
  5113. */
  5114. /*
  5115. void _cppdyn ExampleTool::OnConfigure()
  5116.      {
  5117.      }
  5118. */
  5119. /*
  5120. Redraw _cppdyn ExampleTool::OnButtonDown(Vec &vec)
  5121.      {
  5122.      return REDRAW_NONE;
  5123.      }
  5124. */
  5125. /*
  5126. Redraw _cppdyn ExampleTool::OnMouseMove(Vec &vec)
  5127.      {
  5128.      return REDRAW_NONE;
  5129.      }
  5130. */
  5131. /*
  5132. Redraw _cppdyn ExampleTool::OnButtonUp(Vec &vec)
  5133.      {
  5134.      return REDRAW_NONE;
  5135.      }
  5136. */
  5137. /*
  5138. Redraw _cppdyn ExampleTool::OnSet(Vec &vec)
  5139.      {
  5140.      return REDRAW_NONE;
  5141.      }
  5142. */
  5143. /*
  5144. Redraw _cppdyn ExampleTool::OnDo(Vec &vec)
  5145.      {
  5146.      return REDRAW_NONE;
  5147.      }
  5148. */
  5149. /*
  5150. Redraw _cppdyn ExampleTool::OnUndo()
  5151.      {
  5152.      return REDRAW_NONE;
  5153.      }
  5154. */
  5155. /*
  5156. void _cppdyn ExampleTool::FreeUndo()
  5157.      {
  5158.      }
  5159. */
  5160. /*
  5161. void _cppdyn ExampleTool::OnViewChange(HanCoorSys hcsys,
  5162. Vec &vec)
  5163.      {
  5164.      }
  5165. */
  5166. /*
  5167. Redraw _cppdyn ExampleTool::OnPaste(Vec &vec, HanObject
  5168. *ahobj, ulong ulNumObjects)
  5169.      {
  5170.      return REDRAW_NONE;
  5171.      }
  5172. */
  5173. /*
  5174. void _cppdyn ExampleTool::OnCommand(ushort usID)
  5175.      {
  5176.      }
  5177. */
  5178. /*
  5179. CmdUpdate _cppdyn ExampleTool::OnCommandUpdate(ushort
  5180. usID, char *szText, ushort usBuffSize)
  5181.      {
  5182.      return ENABLE;
  5183.      }
  5184. */
  5185. /*
  5186. void _cppdyn ExampleTool::DrawSoFar(HDC hdc, HanCoorSys
  5187. hcsys, void *pvViewData, bool bInvalid)
  5188.      {
  5189.      }
  5190. */
  5191.  
  5192.  
  5193.  
  5194.  
  5195. Maths Library
  5196.  
  5197. The maths library is implemented in a single DLL
  5198. containing useful functions and classes for manipulating
  5199. graphical related objects such as vectors and matrices.
  5200. Currently Geometry accepts parameters to its functions as
  5201. C++ classes so the C version of the library is not really
  5202. useful for programming tools or applications. However
  5203. eventually a C version of the API will also be included.
  5204.  
  5205. This section is split into two, the first part describing
  5206. the C support, the second describing the C++ support.
  5207.  
  5208. To use the DLL functions and classes you must include
  5209. either the `maths.h' include file (not to be confused
  5210. with C's math.h), for C users, or `cppmaths.h' for C++
  5211. users. Both must then link to `maths.lib'.
  5212.  
  5213. The C++ class interface is built on top of the C maths
  5214. library and provides no additional functionality that is
  5215. not available in the C library, however the C++ version
  5216. allows programs to be written which manipulate vectors
  5217. and matrices using algebraic formula and are therefore
  5218. easier to read. For instance;
  5219.  
  5220.      VectorA=VectorB+VectorC;
  5221.  
  5222. is easier to understand than;
  5223.  
  5224.      MthAddVec(&VectorA, &VectorB, &VectorC);
  5225.  
  5226. For C Users
  5227.  
  5228. Not yet written
  5229.  
  5230. For C++ Users
  5231.  
  5232. The C++ support is probably best described in terms  of
  5233. the class definitions and examples of the operations
  5234. available on those classes, rather than a function by
  5235. function breakdown.
  5236.  
  5237. Four types of class are available;
  5238.  
  5239. Vec       Implements a three element floating point
  5240. Cartesian co-ordinate (x, y and z).
  5241. Lvec          Implements a four element floating point
  5242.          Cartesian co-ordinate (x, y, z and w).
  5243. Polar          Implements a three element floating point
  5244. polar co-ordinate (theta, phi and rho).
  5245. Mat       Implements a 4x4 floating point matrix.
  5246.  
  5247.  
  5248.  
  5249. Vectors
  5250.  
  5251. A vector can be used to store the position of a point in
  5252. 3D space or alternatively a direction relative to another
  5253. point. The following operations are defined for vectors;
  5254.  
  5255.    Vec vec(x, y, z)         Initialise a vector with up
  5256.    to three values
  5257.    Vec vecA(vecB)      Initialise a vector with another
  5258.    vector
  5259.    Vec vec(lvec)            Initialise a vector with a
  5260.    long vector (loses last component)
  5261.    ((Vec) lvec)             Convert a long vector to a
  5262.    vector (loses last component)
  5263.    Vec vec(pol)         Initialise a vector with a polar
  5264.                          vector (conversion performed)
  5265.    ((Vec) pol)              Convert a polar vector to a
  5266.    vector (conversion performed)
  5267.    vecA+vecB           Add two vectors
  5268.    vecA-vecB           Subtract a vector from another
  5269.    vec*f                    Multiply a vector by a
  5270.    scalar
  5271.    vec*mat             Multiply a vector by a matrix
  5272.    vec/f                    Divide a vector by a scalar
  5273.    vecA+=vecB               Add two vectors (overwrite
  5274.    original)
  5275.    vecA-=vecB               Subtract a vector from
  5276.    another (overwrite original)
  5277.    vec*=f                   Multiply a vector by a
  5278.    scalar (overwrite original)
  5279.    vec*=mat            Multiply a vector by a matrix
  5280.    (overwrite original)
  5281.    vec/=f                   Divide a vector by a scalar
  5282.    (overwrite original)
  5283.    -vec                Returns the vector inverted
  5284.    vecA==vecB               Are two vectors equal?
  5285.    vecA!=vecB               Are two vectors not equal?
  5286.    vec.Set(x, y, z)         Set a vector's components
  5287.    vec.SetX(y)              Set a vector's X component
  5288.    vec.SetY(y)              Set a vector's Y component
  5289.    vec.SetZ(z)              Set a vector's Z component
  5290.    vec.X()                  Get x component
  5291.    vec.Y()                  Get y component
  5292.    vec.Z()                  Get z component
  5293.    vec.Len()           Get length of vector
  5294.    vecA.Dot(vecB)      Get dot product of two vectors
  5295.    vecA.Cross(vecB)         Get cross product of two
  5296.    vectors
  5297.    vec.AddrVec()        Gets address of Vector structure
  5298.                          member (WARNING: Use only to
  5299.                          interface to maths C functions)
  5300.  
  5301.  
  5302.  
  5303. Long Vectors
  5304.  
  5305. Long vectors are designed primarily to allow
  5306. multiplication by matrices. The following operations are
  5307. defined for long vectors;
  5308.  
  5309.    LVec lvec(x, y, z, w)         Initialise a long
  5310.    vector with up to four values
  5311.    LVec lvecA(lvecB)        Initialise a long vector
  5312.    with another long vector
  5313.    LVec lvec(vec)       Initialise a long vector with a
  5314.                          vector (last component becomes
  5315.                          1.0)
  5316.    ((LVec) vec)         Convert a long vector to a
  5317.                          vector (last component becomes
  5318.                          1.0)
  5319.    lvec*mat            Multiply a long vector with a
  5320.    matrix
  5321.    lvec*=mat           Multiply a long vector with a
  5322.    matrix (overwrite original)
  5323.    lvecA==lvecB        Are two long vectors equal?
  5324.    lvecA!=lvecB             Are two long vectors not
  5325.    equal?
  5326.    lvec.Set(x, y, z, w)          Set a long vector's
  5327.    components
  5328.    lvec.SetX(x)             Set a long vector's X
  5329.    component
  5330.    lvec.SetY(y)             Set a long vector's Y
  5331.    component
  5332.    lvec.SetZ(z)             Set a long vector's Z
  5333.    component
  5334.    lvec.SetW(w)             Set a long vector's W
  5335.    component
  5336.    lvec.X()            Get x component
  5337.    lvec.Y()            Get y component
  5338.    lvec.Z()            Get z component
  5339.    lvec.W()            Get w component (homogeneous co-
  5340.    ordinate)
  5341.    lvec.AddrLVec()      Gets address of LongVec
  5342.                          structure member (WARNING: Use
  5343.                          only to interface to maths C
  5344.                          functions)
  5345.  
  5346. Polar Vectors
  5347.  
  5348. The following operations are defined for polar vectors;
  5349.  
  5350.    Polar pol(t, p, r)       Initialise a polar with up
  5351.    to three values
  5352.    Polar polA(polB)         Initialise a polar with
  5353.    another polar
  5354.    Polar pol(vec)      Initialise a polar with a vector
  5355.    (conversion performed)
  5356.    ((Polar) vec)            Convert a vector to a polar
  5357.    (conversion performed)
  5358.    polA==polB               Are two polars equal?
  5359.    polA!=polB               Are two polars not equal?
  5360.    pol.Set(t, p, r)     Set a polar's components
  5361.    pol.SetTheta(t)      Set a polar's theta component
  5362.    pol.SetPhi(p)        Set a polar's phi component
  5363.    pol.SetRho(r)        Set a polar's rho component
  5364.    pol.Theta()              Get theta component
  5365.    pol.Phi()           Get phi component
  5366.    pol.Rho()           Get rho component
  5367.    pol.AddrVec()        Gets address of Vector structure
  5368.                          member (WARNING: Use only to
  5369.                          interface to maths C functions)
  5370.    
  5371.  
  5372. Matrices
  5373.  
  5374. The following operations are defined for matrices;
  5375.  
  5376.    
  5377.    Mat mat             Initialise a matrix with the
  5378.    identity matrix
  5379.    Mat matA(matB)      Initialise a matrix with another
  5380.    matrix
  5381.    Mat mat(XROT, a)     Initialise a matrix with a
  5382.                          rotation or scale value (or
  5383.                          leave it unset)
  5384.    Mat mat(TRANSL, vec) Initialise a matrix with a
  5385.                          translation value (or leave it
  5386.                          unset)
  5387.    Mat mat(vecOrigin, vecZAxis, vecYDir)  Initialise a
  5388.                          matrix by defining a new origin,
  5389.                          a direction for the z axis, and
  5390.                          given these constraints, a
  5391.                          direction towards which the y
  5392.                          axis will point.
  5393.    matA*matB           Multiply two matrices
  5394.    matA*=matB               Multiply two matrices
  5395.    (overwrite original)
  5396.    matA==matB               Are two matrices equal?
  5397.    matA!=matB               Are two matrices not equal?
  5398.    -mat                 Compute inverse of matrix (i.e.
  5399.                          when multiplied by this gives
  5400.                          identity)
  5401.    mat.Set(f1,..f16)        Sets elements of a matrix
  5402.    mat.MoveParent(vec) Moves coor sys defined by matrix
  5403.    relative to its parent
  5404.    mat.MoveChild(vec)  Moves coor sys defined by matrix
  5405.    relative to itself
  5406.    mat.ScaleParent(x, y, z)      Scale coor sys defined
  5407.    by matrix relative to its parent
  5408.    mat.ScaleChild(x, y, z)  Scale coor sys defined by
  5409.    matrix relative to its own origin
  5410.    mat.XRotParent(a)        Rotates coor sys defined by
  5411.    matrix about parents x axis
  5412.    mat.XRotChild(a)         Rotates coor sys defined by
  5413.    matrix about its own x axis
  5414.    mat.YRotParent(a)        Rotates coor sys defined by
  5415.    matrix about parents y axis
  5416.    mat.YRotChild(a)         Rotates coor sys defined by
  5417.    matrix about its own y axis
  5418.    mat.ZRotParent(a)        Rotates coor sys defined by
  5419.    matrix about parents z axis
  5420.    mat.ZRotChild(a)         Rotates coor sys defined by
  5421.    matrix about its own z axis
  5422.    mat.ReverseX()      Reverses direction of x axis of
  5423.    coor sys defined by matrix
  5424.    mat.ReverseY()      Reverses direction of y axis of
  5425.    coor sys defined by matrix
  5426.    mat.ReverseZ()      Reverses direction of z axis of
  5427.    coor sys defined by matrix
  5428.    mat.AddrMat()        Gets address of Matrix structure
  5429.                          member (WARNING: Use only to
  5430.                          interface to maths C functions)
  5431.  
  5432.  
  5433.  
  5434. Debug Library
  5435.  
  5436. The debug library is not really part of Genesis but is a
  5437. very helpful debugging aid whether you're writing tools,
  5438. geometry engines, applications, in fact any code what so
  5439. ever. It can be used in windows or non-windows programs,
  5440. DLLs, C, or C++. What is more it presents you with just
  5441. one set of very fast and very simple memory management
  5442. functions.
  5443.  
  5444. The first call for any application or DLL must be to
  5445. DebStart.
  5446.  
  5447. To allocate memory use Debug's Malloc and Free calls
  5448. (note the capitals letters to distinguish from C's memory
  5449. functions). If using C++ you can use New and Delete
  5450. (again with capitals). At any time, a call to DebListMem
  5451. will output to a file showing any memory which has been
  5452. allocated, the size of the memory, the module and line it
  5453. was allocated from and whether it was allocated with
  5454. Malloc or New. This is usually called when an application
  5455. or DLL terminates to ensure all memory was freed.
  5456.  
  5457. If you're programming under windows 3.x the Malloc and
  5458. New functions are not limited to allocating 64K segments.
  5459.  
  5460. The DebOut function can be used exactly like printf but
  5461. will output to a file. Each line appearing in the debug
  5462. output file is timestamped to within the accuracy of the
  5463. system clock. The time given is the time in seconds since
  5464. the first call to DebOut. What is more since the output
  5465. is buffered into a 64K area of memory it has practically
  5466. zero impact on the performance of your application and so
  5467. can be used to time `speed critical' parts of your code.
  5468. When the buffer becomes full and the file has to be
  5469. flushed to disk a line appears in the file saying that a
  5470. flush happened at this point in case there is a
  5471. discrepancy in the time stamps.
  5472.  
  5473. The debug filename is set by the string passed on the
  5474. very first call to DebOut. This string should contain a
  5475. valid filename. Directly after the filename you can put a
  5476. `+' character to indicate that the data should also be
  5477. directed to an additional output stream. Under windows
  5478. the additional output stream will be the debug window, in
  5479. a regular C program it will be the screen. Because many
  5480. DLL's could all be printing on the additional output
  5481. stream simultaneously the output can look confusing and
  5482. the timestamps wont necessarily be continuous.
  5483.  
  5484. If writing C code simply include `debug.h' and C++ users
  5485. should include `cppdebug.h'. In both cases you should
  5486. define the _DEBUG macro, this can be done with /D _DEBUG
  5487. on the compile line. The program must then be linked to
  5488. debug.lib.
  5489.  
  5490. Here is an example run;
  5491.  
  5492.  
  5493.      DebStart();                   // Must be first call
  5494.      DebOut("C:\app.deb+");             // Open debug
  5495. file (with debug win output also)
  5496.      DebOut("Debug file has been opened");
  5497.      DebOut("Variable fLength is %f", fLength);     //
  5498. Output floating point variable
  5499.      pvoid1=Malloc(20);            // Allocates 20 bytes
  5500.      pvoid2=Malloc(30);            // Allocates 30 bytes
  5501.      pobj=New Obj;                 // Allocate a C++
  5502. object called Obj
  5503.      DebOut(FLUSH);           // Forces the debug file to
  5504. be flushed to disk
  5505.      DebOut("More debug");
  5506.      DebListMem();                 // List all unfreed
  5507. memory to debug file
  5508.      DebOut("Closing file now");
  5509.      DebOut(STOP);                 // Flushes and closes
  5510. the debug file
  5511.      Free(pvoid1);                 // Free memory...
  5512.      Free(pvoid2);
  5513.      Delete(pobj);
  5514.  
  5515. The following output will be generated;
  5516.  
  5517.     0.000: Debug file has been opened
  5518.     0.000: Variable fLength is 5.75
  5519.     0.000: --------: (Flushed)
  5520.     0.030: More debug
  5521.     0.030: Allocated memory dump:
  5522.     0.033:   Size   |     File     | Line  |  Alloc type
  5523.     0.033: ---------+--------------+-------+-------------
  5524. -
  5525.     0.033:     20   |     app.c    |  128  |   Malloc
  5526.     0.033:     30   |     app.c    |  129  |   Malloc
  5527.     0.033:    106   |     app.c    |  130  |    New
  5528.     0.040: Closing file now
  5529.  
  5530.  
  5531. If your code is compiled without the _DEBUG macro defined
  5532. then the memory allocation will not store the debugging
  5533. information required by DebMemList. What is more the
  5534. DebOut routine will not be included as part of your
  5535. application and so calls to it should be enclosed in
  5536. #ifdef's to prevent linker errors, e.g.;
  5537.  
  5538.      #ifdef _DEBUG
  5539.      DebOut("A line of debug");
  5540.      #endif
  5541.  
  5542. Instances of debug library
  5543.  
  5544. If you are building a large application consisting of
  5545. many DLLs, and statically linked libraries, then it is
  5546. sometimes not clear how many instances of the debug
  5547. library you have. Basically whenever you use the linker
  5548. and pass it the debug library's name, then you get an
  5549. instance of the library. The upshot of this is that you
  5550. cannot open a debug file in your application and expect
  5551. one of your DLLs to write to the same file. Any DLLs your
  5552. application use we're linked separately and therefore
  5553. have their own copy of the debug library. Therefore they
  5554. should open their own debug file on the first call to
  5555. DebOut. Statically linked libraries however are not
  5556. linked independently and are therefore considered part of
  5557. the application or DLL using it.
  5558.  
  5559. Writing a Geometry engine
  5560.  
  5561. Writing a Geometry engine is the simple process of
  5562. writing a DLL whose interface conforms to the Geometry
  5563. API specification and which maps those functions onto its
  5564. own or a third party rendering engine or a hardware based
  5565. interface.
  5566.  
  5567. Of course there may be many differences between the new
  5568. renderer and Genesis's Geometry engine. For instance the
  5569. renderer might not have the concept of a co-ordinate
  5570. system. However as long as it has the ability to move
  5571. objects independently of one another then a co-ordinate
  5572. system can be implemented in the DLL which hides the
  5573. inadequacies of the renderer. It might also be that the
  5574. renderer uses integer rather than floating point co-
  5575. ordinates. Again this is easy to remedy, you could adopt
  5576. the convention that floating point numbers in the range 0
  5577. to 100 get scaled to integer numbers 0 to 100,000 meaning
  5578. that floating point numbers with an effective resolution
  5579. of 0.001 are supported. As long as the co-ordinates are
  5580. converted back on any query function then a Geometry
  5581. application will not know the difference.
  5582.  
  5583.  
  5584. Errors
  5585.  
  5586. The geometry.h file includes all of the standard geometry
  5587. errors. If a particular error code is listed in the
  5588. Geometry API spec as being returned from a particular
  5589. function, for instance, DefPatch returns
  5590. GERR_NOT_ON_PLANE if its points do not lie on a plane,
  5591. then any Geometry engine must return the same error
  5592. should the condition occur. This is because some
  5593. applications may check for this specific return code and
  5594. if it returns a more general error, such as
  5595. GERR_INT_PROCESSING_ERROR then the behaviour of the
  5596. application might change. The mark of a good Geometry
  5597. engine is one whose applications work identically to the
  5598. default one.
  5599.  
  5600. If an error occurs in the new engine which a standard
  5601. error does not describe adequately then you can use
  5602. GERR_INT_PROCESSING_ERROR or define your own code whose
  5603. value must be GERR_UNKNOWN_ERROR (defined in geometry.h)
  5604. or greater. Such errors can have their own descriptions
  5605. implemented in the engine's GetErrorText function.
  5606.  
  5607.  
  5608. Handles
  5609.  
  5610. All handles are defined as 32 bit unsigned integers. The
  5611. geometry engine can use this value to mean whatever is
  5612. most convenient. For instance the default Geometry stores
  5613. a pointer to the appropriate object in this value. It can
  5614. also be used as an index into an array, however, the
  5615. value 0 cannot be used as this is reserved for the
  5616. NULL_HANDLE, so any such arrays must be based at one.
  5617.  
  5618. Handles do not have to be validated according to the
  5619. spec. To perform proper validation on all handles at
  5620. every call could be time consuming, so if it is not
  5621. quickly and easily achieved, then do not worry about
  5622. implementing it. The onus is on the application and tool
  5623. writers to provide valid handles to calls.
  5624.  
  5625.  
  5626. Unsupported features
  5627.  
  5628. If a Geometry engine does not support a particular
  5629. feature say, bumped mapped textures, it should still
  5630. accept DefSurfType calls defining bump mapped textures
  5631. without returning a bad error. When rendered, the texture
  5632. should just appear as a normal image mapped texture. What
  5633. is more if the surface is queried with QrySurfType it
  5634. should return a SurfType structure with the bump mapped
  5635. flag set, otherwise as a particular model is saved and
  5636. loaded into editors with different engines then the model
  5637. will change. If it is loaded then saved again by an
  5638. editor whose engine does not support bump mapping, and
  5639. then the resulting file is loaded into an engine that
  5640. does, the texture will have lost its `bumpy' quality.
  5641. Likewise engines not supporting `curved' patches must
  5642. still store the normal information.
  5643.  
  5644.  
  5645. Helper library
  5646.  
  5647. A library is provided which contains functions common to
  5648. any Geometry engine thereby preventing any duplication of
  5649. work on behalf of the programmer. The library includes;
  5650.  
  5651.      ·    Validating patches (checks all points lie on a plane
  5652.       etc.)
  5653. ·    Splitting patches into smaller (possibly concave)
  5654. patches
  5655. ·    Computing patch normals
  5656. ·    Computing normals for `autosmooth' patches
  5657. ·    Loading and saving scenes
  5658.      ·    Loading bitmaps for textures
  5659.      ·    Providing text descriptions of Geometry errors
  5660.  
  5661. If the Geometry engine is compiled with the _DEBUG macro
  5662. defined then we must link to the debug version of the
  5663. helper library genhelpd.lib. Otherwise we should link to
  5664. the release version genhelp.lib. Notice there is no
  5665. include file for the helper library, everything you need
  5666. is defined in geometry.h.
  5667.  
  5668.  
  5669. API's
  5670.  
  5671. Any Geometry engine has at its disposal the whole of the
  5672. maths, debug and helper APIs. It should not attempt to
  5673. use the tool interface as there is no concept of tools in
  5674. the Geometry engine. Tools are purely an invention of the
  5675. editor application.
  5676.  
  5677.  
  5678. GeomErr RegisterError (GeomErr gerrIn,
  5679.                       char *szFNIn, ushort usLineIn);
  5680.  
  5681. gerrIn        Geometry error to register.
  5682.  
  5683. szFNIn        Filename of the module in which the error
  5684.               occurred
  5685.  
  5686. usLineIn      The line number within the module at which
  5687.               the error occurred.
  5688.  
  5689. Comments
  5690.  
  5691. Registers a Geometry error. If an error occurs somewhere
  5692. within the Geometry engine, whether it is the result of
  5693. the application passing in bad parameters (e.g.
  5694. GERR_INVALID_HANDLE), or due to circumstances beyond the
  5695. apps control (e.g. GERR_OUT_OF_MEMORY) then the error
  5696. must be passed back to the app. It is quite likely that
  5697. the app will then call the GetErrorText function to see
  5698. what went wrong and possibly to report the error to the
  5699. user. If the Geometry engine registers the error with
  5700. this routine before returning the error code then it
  5701. gives the GetErrorText routine in the helper library the
  5702. ability to build a more helpful error string indicating
  5703. where exactly the error occurred.
  5704.  
  5705. The Gerr() macro defined in geometry.h calls the
  5706. RegisterError function without worrying about how to work
  5707. out the module name and line number.
  5708.  
  5709. The following example code shows the recommended way of
  5710. returning errors;
  5711.      
  5712.      if (pmem==NULL)
  5713.           return Gerr(GERR_OUT_OF_MEMORY);
  5714.  
  5715. If the app then passes the returned error to GetErrorText
  5716. which in turn invokes the helper library function
  5717. GetHlpErrorText, then the following text will be
  5718. returned;
  5719.  
  5720.      Out of memory. Error occurred at line 850 in module
  5721. geometry.cpp.
  5722.  
  5723. If we returned the error without invoking the macro;
  5724.  
  5725.      if (pmem==NULL)
  5726.           return GERR_OUT_OF_MEMORY;
  5727.  
  5728. then the following text will be returned from
  5729. GetErrorText;
  5730.  
  5731.      Out of memory.
  5732.  
  5733. Implementation specific errors with values above
  5734. GERR_UNKNOWN_ERROR can also be registered.
  5735.  
  5736.  
  5737.  
  5738.  
  5739. GeomErr GetHlpErrorText (GeomErr gerr,
  5740.                          char *szBuff, ushort
  5741. usBuffSize);
  5742.  
  5743. gerr          The error whose text is to be returned.
  5744.  
  5745. szBuff        Buffer to contain message.
  5746.  
  5747. usBuffSize    Size of the supplied buffer.
  5748.  
  5749. Comments
  5750.  
  5751. This function can be called from the Geometry engine's
  5752. GetErrorText function. It supplies standard text
  5753. descriptions for all of Geometry's errors. If an
  5754. implementation specific error is used (one with a value
  5755. above GERR_UNKNOWN_ERROR) then the buffer will contain
  5756. the following;
  5757.  
  5758.          `An implementation specific error occurred.
  5759.          Error was reported at line 850 in module
  5760.          geometry.cpp. Implementation interprets error
  5761.          as:'
  5762.  
  5763. and GERR_UNKNOWN_ERROR will be returned. The GetErrorText
  5764. function can then append its own implementation defined
  5765. description of the error if there is space in the buffer.
  5766. strlen can be used to find the length of the returned
  5767. string. If the GetHlpErrorText function runs out of
  5768. buffer space then it copies as much of the error text as
  5769. possible and puts three period characters, `...' in the
  5770. end of the buffer and returns GERR_BUFFER_TOO_SMALL.
  5771.  
  5772.  
  5773.  
  5774.  
  5775. GeomErr ValidPatch (Vec *avec,
  5776.                    ushort usNumVecs, ushort usFlags);
  5777.  
  5778. avec          A list of points to be used for the
  5779.               vertices.
  5780.  
  5781. usNumVecs     Number of points in avec.
  5782.  
  5783. usFlags       Flags used to indicate the kind of
  5784.               validity checks to perform;
  5785.  
  5786.      VP_CHECK_THIN_SEGMENT   Checks that no `thin'
  5787.                          segments appear in the patch
  5788.                          outline. A thin segment is where
  5789.                          part of the outline doubles back
  5790.                          on itself at a 180 degree angle,
  5791.                          forming an infinitely thin line
  5792.                          rather than an area.
  5793.  
  5794.      VP_CHECK_ON_PLANE  Checks that all the points lie
  5795.                          on a plane or within a tolerance
  5796.                          of the plane surface. The
  5797.                          tolerance used is proportional
  5798.                          to the size of the patch and is
  5799.                          approximately 1/50th the
  5800.                          greatest distance across any two
  5801.                          points on the patch outline.
  5802.                          This really only applies to
  5803.                          patches with more 4 or more
  5804.                          vertices.
  5805.  
  5806. Comments
  5807.  
  5808. Two checks are performed regardless of the flag setting,
  5809. first that at least three points have been supplied, and
  5810. secondly that no consequetive points are coincident, i.e.
  5811. are in exactly the same place.
  5812. One check which is isn't performed is whether any line
  5813. segments in the patch's outline cross over thereby making
  5814. the ordering clockwise in one part and anticlockwise in
  5815. the other. We also do not check for concave outlines as
  5816. to some renderers a concave patch might well be a valid
  5817. patch. To those that don't support concave patches the
  5818. split function can be called to chop it up into smaller
  5819. convex patches.
  5820.  
  5821. In general this routine should not be called from
  5822. DefPatch if the DP_DONT_VALIDATE flag is used in the call
  5823. to DefPatch. This is because this call can be time
  5824. consuming if many patches are being created, and if the
  5825. application programmer is confident that no invalidate
  5826. patches will be passed then DP_DONT_VALIDATE can speed
  5827. things up.
  5828.  
  5829.  
  5830.  
  5831. GeomErr ExtractVecs (HanObject hobj, ushort usType,
  5832.                     Handle *ahan, ushort usNum,
  5833.                     Vec **pavec);
  5834.  
  5835. hobj          Handle to object containing vertices.
  5836.  
  5837. usType        Are we extracting vertices or normals?
  5838.               Value can be EV_NORMS or EV_VERTS.
  5839.  
  5840. ahan          Array of handles to vertices or normals.
  5841.  
  5842. usNum         Number of handles in ahan array.
  5843.  
  5844. pavec         The address of a pointer to Vec. The
  5845.               pointer to the allocated memory holding
  5846.               the vectors is returned here.
  5847.  
  5848. Comments
  5849.  
  5850. This routine extracts the vectors from a list of vertices
  5851. or normals using QryVertex or QryNormal respectively. A
  5852. buffer is allocated to contain the vectors. This buffer
  5853. must be freed by the calling code using the Free call.
  5854.  
  5855. This routine is useful for extracting vertex positions in
  5856. order to pass to routines such as ValidPatch, Split and
  5857. CompNormal.
  5858.  
  5859.  
  5860.  
  5861.  
  5862. GeomErr CompNormal (Vec *avec, ushort usNumVecs,
  5863.                       ushort usFlags, Vec *pvecNorm);
  5864.  
  5865. avec          Set of points to use for patch.
  5866.  
  5867. usNumVecs     Number of points in avec.
  5868.  
  5869. usFlags       Flags to specify the significance of the
  5870.               normal;
  5871.  
  5872.    CN_DIRECTION_IMPORTANT  The normal will point to the
  5873.                        side from which the points proceed
  5874.                        in a clockwise direction. Also if
  5875.                        we set this flag we must try and
  5876.                        set the CONVEX flag (if we know
  5877.                        for sure that the outline is
  5878.                        convex) and the co-ordinate system
  5879.                        type flag (appropriate to the co-
  5880.                        ordinate system the points belong
  5881.                        to).
  5882.    
  5883.    CN_CONVEX          If we know for certain that the
  5884.                        outline is convex, then we should
  5885.                        set this flag as it can reduce the
  5886.                        amount of processing this routine
  5887.                        has to do. If
  5888.                        CN_DIRECTION_IMPORTANT is not set
  5889.                        then this is not important.
  5890.    
  5891.    CT_LEFTHAND        If we know that this patch belongs
  5892.                        to a left-hand coor sys then we
  5893.                        should set this flag as it can
  5894.                        reduce the amount of processing
  5895.                        this routine has to do. If
  5896.                        CN_DIRECTION_IMPORTANT is not set
  5897.                        then this is not important.
  5898.    
  5899.    CT_RIGHTHAND       If we know that this patch belongs
  5900.                        to a right-hand coor sys then we
  5901.                        should set this flag as it can
  5902.                        reduce the amount of processing
  5903.                        this routine has to do. If
  5904.                        CN_DIRECTION_IMPORTANT is not set
  5905.                        then this is not important.
  5906.  
  5907. pvecNorm      A unit length normal vector is returned
  5908.               here.
  5909.  
  5910. Comments
  5911.  
  5912. Computes the normal of a set of points. No check is made
  5913. to see if the points lie on a plane (use ValidPatch for
  5914. this). If the CN_DIRECTION_IMPORTANT is set then a lot
  5915. more processing is needed to work out which side the
  5916. normal should be facing. This processing can be reduced
  5917. by specifying as many of the other parameters as
  5918. possible. Concave patches always need the full processing
  5919. if CN_DIRECTION_IMPORTANT is set.
  5920.  
  5921. If the patch outline contains some points very close
  5922. together then this routine is selective in deciding which
  5923. three points to use to generate the normal. This ensures
  5924. we get an accurate a value as possible as the accuracy of
  5925. this normal is crucial to the operation of some of the
  5926. editors tools.
  5927.  
  5928.  
  5929.  
  5930.  
  5931. GeomErr Split (Vec *avecMain, ushort usNumMainVecs,
  5932.               ushort usMaxPerPatch, ushort usFlags,
  5933.               ushort *pusNumSets, void **apvSets,
  5934.               ushort *ausNumInSets);
  5935.  
  5936. avecMain      Set of points to split up.
  5937.  
  5938. usNumMainVecs Number of points in avecMain.
  5939.  
  5940. usMaxPerPatch The maximum number vertices per patch that
  5941.               this Geometry engine will allow.
  5942.  
  5943. usFlags       Flags controlling the operation;
  5944.  
  5945.    SP_SUPPORT_CONCAVE  Indicates whether the Geometry
  5946.                         engine supports concave patch
  5947.                         outlines.
  5948.  
  5949.    SP_RETURN_INDEX     Rather than returning an array of
  5950.                         vector arrays in (*apvSets)
  5951.                         instead we return an array of
  5952.                         ushort arrays, where each ushort
  5953.                         is an index into the original set
  5954.                         of points passed in.
  5955.  
  5956. pusNumSets    The number of sets returned, i.e.. the
  5957.               number of smaller outlines the main
  5958.               outline was split into.
  5959.  
  5960. apvSets       A pointer to an array of size
  5961.               (*pusNumSets) containing pointers to
  5962.               either arrays of vectors arrays of ushorts
  5963.               (if SP_RETURN_INDEX was given). Each of
  5964.               these arrays describes a set of points. If
  5965.               SP_RETURN_INDEX was specified than a
  5966.               variable declared; ushort **aausIndex,
  5967.               should be passed to apvSets, otherwise the
  5968.               variable should be; Vec **aavec; All
  5969.               arrays are allocated by this routine.
  5970.  
  5971. ausNumInSet   An array of ushorts indicating the size of
  5972.               the individual set arrays. For example
  5973.               ausNumInSet[2] contains the size of the
  5974.               array pointed to by apvSet[2]. In other
  5975.               words set 2 consists of elements
  5976.               apvSet[2][0] to
  5977.               apvSet[2][ausNumIntSet[2]].
  5978.  
  5979. Comments
  5980.  
  5981. This routine splits up an outline defined by a set of
  5982. points into a number of sets. The sets can be expressed
  5983. themselves as points or as indexes into the original
  5984. array avecMain. The sets are allocated by this routine,
  5985. but it is the responsibility of the calling code to Free
  5986. the memory when it has finished with it.
  5987.  
  5988. The following code fragment will free up all memory;
  5989.  
  5990.                for (us=usNumSets; us--;)
  5991.                     Free(aausIndex[us]);     // Free all
  5992. sets
  5993.                Free(aausIndex);              // Free
  5994. array of pointers to sets
  5995.                Free(ausNumInSet);       // Free array of
  5996. sizes of sets
  5997.  
  5998. Although we can pass in any number for usMaxPerPatch, in
  5999. practice the sets produced will not contain more than 4
  6000. points each, although these can be concave if
  6001. SP_SUPPORT_CONCAVE is set.
  6002.  
  6003.  
  6004.  
  6005.  
  6006. GeomErr Autosmooth (HanObject hobj, HanVertex *ahver,
  6007.                     ushort usNumVerts, Vec &vecNorm,
  6008.                     ushort usFlags, HanSurf hsur,
  6009.                     float fSmoothAng, HanNormal *ahnor,
  6010.                     ASUndoBuff *pasub);
  6011.  
  6012. hobj          Handle to object to add `autosmoothed'
  6013.               patch too.
  6014.  
  6015. ahver         Array of vertex handles defining the
  6016.               patch.
  6017.  
  6018. usNumVerts    Number of handles in ahver. Also specifies
  6019.               the size of the ahnor buffer.
  6020.  
  6021. vecNorm       The normal of the patch to be
  6022.               `autosmoothed'. The normal should point to
  6023.               the empty side of the patch.
  6024.  
  6025. usFlags       Flags controlling the operation;
  6026.  
  6027.    DP_SMOOTH_BY_SURFThis patch is to be smoothed only
  6028.                      with neighbouring patches which have
  6029.                      the same surface type.
  6030.  
  6031.    DP_SMOOTH_BY_ANGLE    This patch is to be smoothed
  6032.                      only with neighbouring patches which
  6033.                      form an angle less than fSmoothAng
  6034.                      with this patch.
  6035.  
  6036. hsur          Surface type of the `autosmoothed' patch.
  6037.  
  6038. fSmoothAng    Specifies autosmooth angle threshold if
  6039.               DP_SMOOTH_BY_ANGLE is set.
  6040.  
  6041. ahnor         An array of handles to normals returned.
  6042.  
  6043. pasub         Pointer to an autosmooth undo buffer. If
  6044.               the geometry engine wishes to abort this
  6045.               operation it can pass this to the UndoAS
  6046.               routine to undo any changes made. If not
  6047.               it must pass it to the ComitAS.
  6048.  
  6049. Comments
  6050.  
  6051. Given an array of vertex handles defining a new patch,
  6052. this routine can determine the set of normals to use to
  6053. get this patch to appear smooth given the criteria
  6054. defined in usFlags. It does this by looking at the
  6055. surrounding patches and averaging out the surface normals
  6056. of the patches to create the normals. If normals already
  6057. exist at these points they are modified, otherwise new
  6058. ones are created. The set of normals to use in defining
  6059. the patch is passed back in the ahnor parameter. The
  6060. returned pointer pasub should be passed either to UndoAS
  6061. or to CommitAS if the patch creation succeeded.
  6062.  
  6063.  
  6064.  
  6065.  
  6066. GeomErr CommitAS (ASUndoBuff *pasub);
  6067.  
  6068. pasub         Pointer to the autosmooth undo buffer.
  6069.  
  6070. Comments
  6071.  
  6072. Commits the changes made to the normals by the Autosmooth
  6073. function. Either this or UndoAS must be called at some
  6074. point after an Autosmooth call.
  6075.  
  6076.  
  6077.  
  6078.  
  6079. GeomErr UndoAS (ASUndoBuff *pasub);
  6080.  
  6081. pasub         Pointer to the autosmooth undo buffer.
  6082.  
  6083. Comments
  6084.  
  6085. Undoes the changes made to the normals by the Autosmooth
  6086. function. Either this or CommitAS must be called at some
  6087. point after an Autosmooth call.
  6088.  
  6089.  
  6090.  
  6091.  
  6092. GeomErr LoadBmp (char *szFN,
  6093.                   BITMAPINFO huge **ppbmi);
  6094.  
  6095. szFN          The name of a .bmp file to open.
  6096.  
  6097. ppbmi         The address of a pointer to a bitmap which
  6098.               will be returned after the bitmap is
  6099.               loaded. The memory is allocated by this
  6100.               routine.
  6101.  
  6102. Comments
  6103.  
  6104. Loads a windows bitmap or IFF file with the specified
  6105. name. Even if the bitmap is an IFF file it is presented
  6106. to the calling code as a windows BITMAPINFO structure.
  6107. HAM and Extra halfbrite IFF files are also supported.
  6108.  
  6109. The correct amount of memory is automatically allocated
  6110. and a pointer to it is returned in (*ppbmi). The calling
  6111. code must free the memory using the Debug library's Free
  6112. call. If the bitmap does not exist then
  6113. GERR_BITMAP_FILE_NOT_FOUND is returned. If the file has
  6114. an incorrect format GERR_NOT_A_BMP_FILE is returned.
  6115.  
  6116.  
  6117.  
  6118.  
  6119. GeomErr SaveSceneHlp (HFILE hfile,
  6120.                        HanCoorSys hcsys,
  6121.                        ulong *pulNumCSys,
  6122.                        HanCoorSys *ahcsys,
  6123.                        ulong *pulNumObjs,
  6124.                        HanObject *ahobj,
  6125.                        fnLoadSaveCallback *Report);
  6126.  
  6127. hfile         Windows handle of the file to save too.
  6128.  
  6129. hcsys         Handle of the co-ordinate system to save.
  6130.  
  6131. pulNumCSys    Size of ahcsys (in number of handles). On
  6132.               return holds the total number of co-
  6133.               ordinate systems saved.
  6134.  
  6135. ahcsys        An array of handles to co-ordinate systems
  6136.               saved.
  6137.  
  6138. pulNumObjs    Size of ahobj (in number of handles). On
  6139.               return holds the total number of objects
  6140.               saved.
  6141.  
  6142. ahobj         An array of handles to objects saved.
  6143.  
  6144. Report         The address of a callback function to
  6145.                inform the application of how far through
  6146.                the save its done.
  6147.  
  6148. Comments
  6149.  
  6150. This function takes exactly the same parameters as
  6151. Geometry's SaveScene function, so the implementation of
  6152. the geometry function is nothing more than calling this
  6153. helper routine.
  6154.  
  6155.  
  6156.  
  6157.  
  6158. GeomErr LoadSceneHlp (HFILE hfile,
  6159.                        HanCoorSys csysParent,
  6160.                        Char *szTexPath,
  6161.                        ulong *pulNumCSys,
  6162.                        HanCoorSys *ahcsys,
  6163.                        ulong *pulNumObjs,
  6164.                        HanObject *ahobj,
  6165.                        fnLoadSaveCallback *Report);
  6166.  
  6167. hfile         Windows handle of the file to load from.
  6168.  
  6169. hcsysParent   Handle of the co-ordinate system which
  6170.               will be the parent of the one being
  6171.               loaded.
  6172.  
  6173. szTexPath     A pointer to a path specification for
  6174.               where to search for texture bitmaps if
  6175.               they are not found in the current working
  6176.               directory. This would typically be set to
  6177.               the directory where the model file is, or
  6178.               else a special texture directory. Any
  6179.               number of paths can be seperated by semi
  6180.               colons, but each must have a terminating
  6181.               back slash e.g.;
  6182.               "c:\genesis\textures\;c:\windows\bmps\" or
  6183.               else be a null string. The pointer cannot
  6184.               be NULL.
  6185.  
  6186. pulNumCSys    Size of ahcsys (in number of handles). On
  6187.               return holds the total number of co-
  6188.               ordinate systems loaded.
  6189.  
  6190. ahcsys        An array of handles to co-ordinate systems
  6191.               loaded.
  6192.  
  6193. pulNumObjs    Size of ahobj (in number of handles). On
  6194.               return holds the total number of objects
  6195.               loaded.
  6196.  
  6197. ahobj         An array of handles to objects loaded.
  6198.  
  6199. Report         The address of a callback function to
  6200.                inform the application of how far through
  6201.                the load its done.
  6202.  
  6203. Comments
  6204.  
  6205. This function takes exactly the same parameters as
  6206. Geometry's LoadScene function, so the implementation of
  6207. the geometry function is nothing more than calling this
  6208. helper routine.
  6209.  
  6210.  
  6211.  
  6212.  
  6213. GeomErr LoadLWObjectHlp (HFILE hfile,
  6214.                            HanCoorSys hcsys,
  6215.                            char *szTexPath,
  6216.                            HanObject *phobj,
  6217.                            fnLoadSaveCallback *Report);
  6218.  
  6219. hfile         Windows handle of the file to load from.
  6220.  
  6221. hcsys         Handle of the co-ordinate system the
  6222.               object will be added to.
  6223.  
  6224. szTexPath     A pointer to a path specification for
  6225.               where to search for texture bitmaps if
  6226.               they are not found in the current working
  6227.               directory. This would typically be set to
  6228.               the directory where the model file is, or
  6229.               else a special texture directory. Any
  6230.               number of paths can be seperated by semi
  6231.               colons, but each must have a terminating
  6232.               back slash e.g.;
  6233.               "c:\genesis\textures\;c:\windows\bmps\" or
  6234.               else be a null string. The pointer cannot
  6235.               be NULL.
  6236.  
  6237. phobj         A pointer to the handle of the object that
  6238.               will be created.
  6239.  
  6240. Report         The address of a callback function to
  6241.                inform the application of how far through
  6242.                the load its done.
  6243.  
  6244. Comments
  6245.  
  6246. This function takes exactly the same parameters as
  6247. Geometry's LoadLWObject function, so the implementation
  6248. of the geometry function is nothing more than calling
  6249. this helper routine.
  6250.