box.cpp

来自「开放源码的编译器open watcom 1.6.0版的源代码」· C++ 代码 · 共 218 行

CPP
218
字号
/****************************************************************************
*
*                            Open Watcom Project
*
*    Portions Copyright (c) 1983-2002 Sybase, Inc. All Rights Reserved.
*
*  ========================================================================
*
*    This file contains Original Code and/or Modifications of Original
*    Code as defined in and that are subject to the Sybase Open Watcom
*    Public License version 1.0 (the 'License'). You may not use this file
*    except in compliance with the License. BY USING THIS FILE YOU AGREE TO
*    ALL TERMS AND CONDITIONS OF THE LICENSE. A copy of the License is
*    provided with the Original Code and Modifications, and is also
*    available at www.sybase.com/developer/opensource.
*
*    The Original Code and all software distributed under the License are
*    distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
*    EXPRESS OR IMPLIED, AND SYBASE AND ALL CONTRIBUTORS HEREBY DISCLAIM
*    ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF
*    MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR
*    NON-INFRINGEMENT. Please see the License for the specific language
*    governing rights and limitations under the License.
*
*  ========================================================================
*
* Description:  Low level classes for 3D demo.
*
****************************************************************************/


#include "box.hpp"

static float point::world_scale = 12 * 12;      // ie 12 feet
static float point::vertical_shift = 0;

// Functions which could be inlined
point::point( float x0, float y0, float z0 ) : x(x0), y(y0), z(z0) { }
point::point( point& src ) : x(src.x), y(src.y), z(src.z) { }
void point::move_left( float amount ) { x -= amount; }
void point::move_right( float amount ) { x += amount; }
void point::move_up( float amount ) { y += amount; }
void point::move_down( float amount ) { y -= amount; }
void point::move_forward( float amount ) { z -= amount; }
void point::move_back( float amount ) { z += amount; }

color::color(unsigned char red,unsigned char green,unsigned char blue)
                : r(red), g(green), b(blue) { }
color::color(const color& src) : r(src.r), g(src.g), b(src.b) { }
color::color() : r(0), g(0), b(0) { }
void color::rgb(unsigned char red, unsigned char green, unsigned char blue)
                { r = red; g = green; b = blue; }
void color::rgb(color& src) { rgb( src.r, src.g, src.b); }
void color::rgb(color* src) { rgb( src->r, src->g, src->b); }
void color::red( unsigned char intensity ) { r = intensity; }
void color::green( unsigned char intensity ) { g = intensity; }
void color::blue( unsigned char intensity ) { b = intensity; }

float box::width() { return size_vector.x; }
float box::height() { return size_vector.y; }
float box::depth() { return size_vector.z; }
void box::left( float x0 ) { x = x0; }
void box::right( float x0 ) { x = x0 - size_vector.x; }
void box::front( float z0 ) { z = z0 - size_vector.z; }
void box::back( float z0 ) { z = z0; }
void box::top( float y0 ) { y = y0 - size_vector.y; }
void box::bottom( float y0 ) { y = y0; }
float box::left() { return x; }
float box::right() { return x + size_vector.x; }
float box::top() { return y + size_vector.y; }
float box::bottom() { return y; }
float box::front() { return( z + size_vector.z ); }
float box::back() { return z; }
float box::mid_width() { return x + size_vector.x / 2; }
float box::mid_height() { return y + size_vector.y / 2; }
float box::mid_depth() { return z + size_vector.z / 2; }


void point::rotate( orientation move )
{
    float       temp;

    switch( move % 4 ) {
    case FRONT:
        // nothing to do
        break;
    case RIGHT:
        temp = x;
        x = z;
        z = -temp;
        break;
    case BACK:
        x = -x;
        z = -z;
        break;
    case LEFT:
        temp = x;
        x = -z;
        z = temp;
        break;
    }
}

void point::moveto( float x0, float y0, float z0 )
{
    x = x0;
    y = y0;
    z = z0;
}

void point::moveto( point& origin )
{
    x = origin.x;
    y = origin.y;
    z = origin.z;
}

void point::shift( float xdelta, float ydelta, float zdelta )
{
    x += xdelta;
    y += ydelta;
    z += zdelta;
}

void point::shift( point& vector )
{
    shift( vector.x, vector.y, vector.z );
}

box::box( float across, float depth, float height )
    : point(0,0,0), size_vector(across, height, depth), facing(FRONT)
{
    children = 0;
    sibling = 0;
}

box::~box()
{
    delete children;
    delete sibling;
}

static void point::world_size( float wide, float deep, float high )
{
    // To keep proportions correct, just remember the largest of the dimensions

    world_scale = wide;
    if( world_scale < deep ) {
        world_scale = deep;
    }
    if( world_scale < high ) {
        world_scale = high;
        vertical_shift = 0;
    } else {
        vertical_shift = (world_scale - high) / 2;
    }
}

void box::add(box* child)
{
    child->sibling = children;
    children = child;
}

void box::orient( orientation rotation )
{
    facing = (orientation) ((facing + rotation) % 4);
}

void box::orient( box * other )
{
    orient( other->facing );
}

void box::position_to_right_of( box& to_left )
{
    // Move to same position & facing as to_left
    moveto( to_left );
    orient( to_left.facing );
    // Now move across by the width of to_left
    point shift_vector( to_left.width(), 0, 0 );
    shift_vector.rotate( facing );
    shift( shift_vector );
}

void box::change_color( color newclr )
{
    rgb( newclr );
    for( box * curr = children; curr; curr = curr->sibling ) {
        curr->change_color( newclr );
    }
}

void box::draw( point& origin, orientation face )
{
    // First, combine the rotations
    face = (orientation) ((face + facing) % 4);
    if( children ) {
        for( box * curr = children; curr; curr = curr->sibling ) {
            point curr_origin = *curr;
            curr_origin.rotate( face );
            curr_origin.shift( origin );
            curr->draw( curr_origin, face );
        }
    } else {
        // Only draw outer box if there is no children
        point new_size = size_vector;
        new_size.rotate( face );
        point other_corner = origin;
        other_corner.shift( new_size );
        draw_box( origin.x / world_scale, other_corner.x / world_scale,
                  (origin.y + vertical_shift) / world_scale,
                  (other_corner.y + vertical_shift) / world_scale,
                  origin.z / world_scale, other_corner.z / world_scale,
                  r, g, b );
    }
}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?