CameraHandler
CameraInterface
Availability LightWave® 9.0
Component Layout
Header lwcamera.h
The CameraHandler provides ray starting positions and directions for each pixel being raytraced.
In this way any type of camera projection can be created. The projection defined by the plugin
may be previewed in Layout, if possible, by supplying an OpenGL-like projection matrix.
A camera handler is attached to a camera. The camera handler works in world coordinates, not
camera coordinates. The definition of the projection may involve items in the scene other than
the camera to which the handler is attached. The camera handler maps filmplane and lens positions to rays.
A position on the film is given in filmplane coordinates. Filmplane coordinates are given in
units of meters. They define a location on the hypothetical plane on which the film frame is placed.
The film frame is considered to be at the origin of the filmplane. The filmplane coordinate system has y axis
going up the film frame, and the x axis in the same direction as the frame's horizontal direction.
Note that filmplane coordinates are unaffected by image resolution, film size, or pixel aspect ratio.
A position on the lens is also given in a normalized range from -1 to 1, with the origin considered to
be the centre of the lens. The lens positions define points on a hypothetical lens. This is particularly
important for depth of field effects. No assumptions are made about the shape of the lens, except that the
user can influence the distribution of the sample points that will be supplied by setting a diaphragm shape.
It is up to the camera handler to map the combination of filmplane and lens sample position to a ray.
For some cameras there may be no concept of a lens. Those cameras can indicate that they don't support and
depth of field rendering and can ignore the lens sample positions.
Note that there were substantial changes to this API between version 1 and version 2.
As a result, version 1 can no longer be supported.
Handler Activation Function
XCALL_( int ) MyCamera( int version, GlobalFunc *global,
LWCameraHandler *local, void *serverData );
The local argument to a CameraHandler activation function is
an LWCameraHandler.
typedef struct st_LWCameraHandler
{
LWInstanceFuncs* inst;
LWItemFuncs* item;
LWRenderFuncs* rend;
int (*preview) (LWInstance, double lpx, double lpy, LWDMatrix4 projection,
const LWCameraAccess* camaccess);
LWError (*newFrame) (LWInstance, const LWFrameInfo* frameinfo,
const LWCameraAccess* camaccess);
int (*evaluate) (LWInstance, double fpx, double fpy, double lpx, double lpy,
double fractime, LWCameraRay* ray, const LWCameraAccess* camaccess);
unsigned int (*flags) (LWInstance);
} LWCameraHandler;
The first three members of this structure are the standard handler
functions. The context argument to the inst->create function is the LWItemID of the associated object.
- result = preview( instance, lpx, lpy, projection, access )
- Fills in the given matrix with an OpenGL projection matrix that approximates the camera view, if possible.
The (lpx, lpy) coordinates is the sample position on the lens, with values in the range between -1 and 1.
This may be non-zero when constructing motion blur previews for example.
The projection matrix should project from camera space to the camera's filmplane.
The matrix follows the OpenGL matrix conventions. It is expected that the matrix projects to the film size
(unlike the standard OpenGL frustum which projects to a normalized coordinate system). The effects of aspect
ratio of the film and viewport will be automatically handled by LW. Set the projection matrix to fit the
projection to the height of the film with a square aspect ratio. The frame number and time for the preview is
as given by the most recent call to newTime().
Returns 1 if the matrix has been filled in, or 0 if the matrix could not be filled in, for example
because there is no good projection matrix approximation for the camera.
- result = newFrame( instance, frameinfo, access )
- This is called whenever a new set of evaluations is started, usually at the start of a frame render, or when
constructing the view frustum preview of a selected camera in Layout. This is a good place to do some pre-computation.
The frameinfo and camera access structures are explained below.
Returns a pointer to an error string of something went wrong, or NULL if all is well.
- result = evaluate( instance, fpx, fpy, lpx, lpy, fractime, ray, access )
- Given a position (fpx, fpy) on the camera filmplane, a position (lpx, lpy) on the lens, and a fractional time
into the frame, compute a starting position and direction for the ray to be cast.
The filmplane coordinates may fall outside of the film frame. There is no defined order to the filmplane and lens
positions with which the evaluation function is called.
The supplied time value is a fraction between 0 and 1 that ranges over the time duration of the frame (which can be
retrieved from the frameinfo structure supplied to newFrame()). 0 means the start time of the frame, while 1 means
the time at the end of the duration of the frame (which may be different than that implied by the current frames per second
setting, as it includes the effects of fields and motion blur length).
There is no defined ordering to the time values that will be given. It is recommended that
the camera handler pre-computes any animation of settings in newFrame() and uses the fractime parameter
to interpolate. LightWave can do a default interpolation on the returned ray, which consists of transforming the ray
coordinates according to the change in position and orientation of the camera at the fractional time compared with the
start of the frame. If this is sufficient, the camera handler should compute the ray position and direction as it would
be at the start of the frame, ignoring the fractional time, and return a result of LWCAMTYPEEVAL_DO_DEF_INTERP
(see below).
The evaluation function must also be thread-safe, for example using
the multithreading utilities if needed.
The evaluation fills in the ray structure if the camera can cast a ray for the given coordinates and time.
The return value is one of:
- LWCAMTYPEEVAL_NO_RAY
- No ray can be cast.
- LWCAMTYPEEVAL_RAY
- A ray can be cast.
- LWCAMTYPEEVAL_DO_DEF_INTERP
- A ray can be cast, but let LightWave do a defauly time interpolation.
- flags = flags( instance )
-
- Returns bit flags combined using bitwise-or. Currently defined flags are:
- LWCAMTYPEF_SUPPORTSDOF
- The camera support depth of field rendering.
Interface Activation Function
XCALL_( int ) MyInterface( int version, GlobalFunc *global,
LWInterface *local, void *serverData );
This is the standard interface activation
for handlers. Users open a Camera interface by pressing an Properties
button on the Camera Properties panel.
To help implement the user interface, the standard zoom factor controller is
available as an XPanel control of type "zoomfactor-env", which uses a vparm
as its underlying data type. This control is available from LightWave® 9.3 onwards.
The vparm must have been created in the channel group of
the camera. The control will handle getting and setting the camera zoom settings. The
vparm does not have to be saved or loaded by the camera plugin.
Frame Info
The newFrame() function is supplied with some basic information about the frame that is about to be evaluated.
More detailed information may be obtained through the camera info global.
typedef struct st_LWFrameInfo {
LWFrame frame;
LWTime start;
LWTime duration;
double framesPerSecond;
unsigned int motionSteps;
} LWFrameInfo;
- frame
- The frame number of the frame to be evaluated.
- start
- The start time of the frame to be evaluated.
- duration
- The duration of the frame to be evaluated. Together with the start time this defines the range
of time values over which the fractime parameter passed to the evaluation function is
normalized. Finer-grained information can be obtained through caminfo->frameTiming()
- framesPerSecond
- The current FPS setting. This is the time from the start of one frame to the start of the next
frame. Note that this is often different that the duration of the frame due to field rendering and
motion blur length.
- motionSteps
- The suggested number of time steps to use for constructing animation interpolation. To compute
the positions, orientations, and other parameters for items at any time within the frame duration,
LightWave pre-computes the animations at regular intervals along the frame's duration, and then
interpolates during during rendering to determine the parameters at any arbitrary time within the
frame duration. The number of intervals used is given by motionSteps. Camera handlers
can use this as a guide to determine how many intervals to split the frame duration up in to
pre-compute the animation of its settings. Note that this value may be 0.
Camera Access
The access structure information that may be useful for computing the camera.
typedef struct st_LWCameraAccess {
LWItemID cameraID;
LWDVector worldPos;
LWDVector toWorld[3];
LWDVector toCamera[3];
double filmSize[2];
} LWCameraAccess;
- cameraID
- The ItemID of the camera that the CameraHandler plugin is be applied to.
- worldPos
- The position of the camera item in world coordinates.
- toWorld[3]
- Right, Up and Forward directions of the camera in world coordinates.
- toCamera[3]
- Inverse of toWorld[3] .
- filmSize[2]
- The width and height of a film frame, given in meters.
Ray structure
The evaluation function fills in a ray structure describing the ray to be cast.
typedef struct st_LWCameraRay {
LWDVector rayPos;
LWDVector rayDir;
LWDVector filmNorm;
LWDVector filter[3];
} LWCameraRay;
- rayPos
- The starting position of the ray, in world coordinates.
- rayDir
- The ray direction, in world coordinates.
- filmNorm
- The normal to the film plane for the ray. This is used to compute the depth buffer value
for any point along the ray. The depth buffer value is computed by projecting the point
(relative to the start position of the ray) onto the film normal for that ray. The length
of the resulting vector is taken to be the depth value. For example, for a perspective
camera the film normal is the camera's Z axis. For a spherical projection camera it may
be more proper to set the film normal equal to the ray direction, so that the depth value
is equal to distance along the ray.
- filter
- A camera lens may affect the color, for example darkening towards the edges causing vignetting.
The camera handler can set a color filter for each of the red, green, and blue channels. A
filter is applied to a channel by doing a dot-product of the filter with the raytraced RGB color.
Note that this is not meant to replace the functionality of pixel filters
or image filters. The camera filter is used to apply shading effects which
are inherent to the camera, not the film or scene.
|