// @configure_input@ /**************************************************************************\ * Copyright (c) Kongsberg Oil & Gas Technologies AS * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * Neither the name of the copyright holder nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. \**************************************************************************/ /*! \class So@Gui@Cursor \brief The So@Gui@Cursor class is used to set cursors for GUI components. \ingroup misc The class provides both a set of pre-defined cursor shapes, aswell as the option to use custom bitmap graphics. Instances of this class is usually made for passing in to the So@Gui@Component::setComponentCursor() and So@Gui@Component::setWidgetCursor() methods. */ #include #include #include /*! \enum So@Gui@Cursor::Shape For enumerating built-in shape types, that can be used without having to specify cursor graphics. These are mapped to pre-defined cursor shapes from the underlying toolkit. */ /*! \var So@Gui@Cursor::Shape So@Gui@Cursor::CUSTOM_BITMAP Cursor is specified with our own bitmap graphics. The bitmap data must be passed into the constructor. */ /*! \var So@Gui@Cursor::Shape So@Gui@Cursor::DEFAULT Using the native Window system's default cursor. */ /*! \var So@Gui@Cursor::Shape So@Gui@Cursor::BUSY A busy cursor. */ /*! \var So@Gui@Cursor::Shape So@Gui@Cursor::CROSSHAIR Two thin lines in a cross. */ /*! \var So@Gui@Cursor::Shape So@Gui@Cursor::UPARROW Thick, upward-pointing arrow. */ /*** Graphic data for the mouse pointer bitmaps. ***********************/ /*! \class So@Gui@Cursor::CustomCursor \brief The So@Gui@Cursor::CustomCursor class is used to specify bitmap data for So@Gui@Cursor custom cursors. */ /*! \var SbVec2s So@Gui@Cursor::CustomCursor::dim Size of cursor data. Can be anything from 1 pixel wide and high, up to maximum 32 pixels in each dimension. */ /*! \var SbVec2s So@Gui@Cursor::CustomCursor::hotspot Position of cursor hotspot, ie the pixel of the cursor which is considered to be the "pick point" when interacting with the canvas or widgets. The hotspot position should be specified relative to the uppermost, leftmost corner (which is position <0, 0>). */ /*! \var unsigned char * So@Gui@Cursor::CustomCursor::bitmap The size of the array of bitmap data should be equal to (width+7)/8*height, as each cursor pixel is specified with just two bits, one in the bitmap array, one in the mask array. The bitmap and mask bits should be laid out as follows: \verbatim B=1 and M=1 == black B=0 and M=1 == white B=0 and M=0 == transparency B=1 and M=0 will give undefined results for some platforms, so avoid. \endverbatim */ /*! \var unsigned char * So@Gui@Cursor::CustomCursor::mask See doc of So@Gui@Cursor::CustomCursor::bitmap. */ /*** zoom-style cursor ******/ #define ZOOM_WIDTH 16 #define ZOOM_HEIGHT 16 #define ZOOM_BYTES ((ZOOM_WIDTH + 7) / 8) * ZOOM_HEIGHT #define ZOOM_HOT_X 5 #define ZOOM_HOT_Y 7 static unsigned char zoom_bitmap[ZOOM_BYTES] = { 0x00, 0x0f, 0x80, 0x1c, 0x40, 0x38, 0x20, 0x70, 0x90, 0xe4, 0xc0, 0xcc, 0xf0, 0xfc, 0x00, 0x0c, 0x00, 0x0c, 0xf0, 0xfc, 0xc0, 0xcc, 0x90, 0xe4, 0x20, 0x70, 0x40, 0x38, 0x80, 0x1c, 0x00, 0x0f }; static unsigned char zoom_mask_bitmap[ZOOM_BYTES] = { 0x00,0x0f,0x80,0x1f,0xc0,0x3f,0xe0,0x7f,0xf0,0xff,0xf0,0xff,0xf0,0xff,0x00, 0x0f,0x00,0x0f,0xf0,0xff,0xf0,0xff,0xf0,0xff,0xe0,0x7f,0xc0,0x3f,0x80,0x1f, 0x00,0x0f }; /*** pan-style cursor *******/ #define PAN_WIDTH 16 #define PAN_HEIGHT 16 #define PAN_BYTES ((PAN_WIDTH + 7) / 8) * PAN_HEIGHT #define PAN_HOT_X 7 #define PAN_HOT_Y 7 static unsigned char pan_bitmap[PAN_BYTES] = { 0xc0, 0x03, 0x60, 0x02, 0x20, 0x04, 0x10, 0x08, 0x68, 0x16, 0x54, 0x2a, 0x73, 0xce, 0x01, 0x80, 0x01, 0x80, 0x73, 0xce, 0x54, 0x2a, 0x68, 0x16, 0x10, 0x08, 0x20, 0x04, 0x40, 0x02, 0xc0, 0x03 }; static unsigned char pan_mask_bitmap[PAN_BYTES] = { 0xc0,0x03,0xe0,0x03,0xe0,0x07,0xf0,0x0f,0xe8,0x17,0xdc,0x3b,0xff,0xff,0xff, 0xff,0xff,0xff,0xff,0xff,0xdc,0x3b,0xe8,0x17,0xf0,0x0f,0xe0,0x07,0xc0,0x03, 0xc0,0x03 }; /*** rotate-style cursor ****/ #define ROTATE_WIDTH 16 #define ROTATE_HEIGHT 16 #define ROTATE_BYTES ((ROTATE_WIDTH + 7) / 8) * ROTATE_HEIGHT #define ROTATE_HOT_X 6 #define ROTATE_HOT_Y 8 static unsigned char rotate_bitmap[ROTATE_BYTES] = { 0xf0, 0xef, 0x18, 0xb8, 0x0c, 0x90, 0xe4, 0x83, 0x34, 0x86, 0x1c, 0x83, 0x00, 0x81, 0x00, 0xff, 0xff, 0x00, 0x81, 0x00, 0xc1, 0x38, 0x61, 0x2c, 0xc1, 0x27, 0x09, 0x30, 0x1d, 0x18, 0xf7, 0x0f }; static unsigned char rotate_mask_bitmap[ROTATE_BYTES] = { 0xf0,0xef,0xf8,0xff,0xfc,0xff,0xfc,0xff,0x3c,0xfe,0x1c,0xff,0x00,0xff,0x00, 0xff,0xff,0x00,0xff,0x00,0xff,0x38,0x7f,0x3c,0xff,0x3f,0xff,0x3f,0xff,0x1f, 0xf7,0x0f }; /*** just a blank cursor ****/ #define BLANK_WIDTH 1 #define BLANK_HEIGHT 1 #define BLANK_BYTES ((BLANK_WIDTH + 7) / 8) * BLANK_HEIGHT #define BLANK_HOT_X 0 #define BLANK_HOT_Y 0 static unsigned char blank_bitmap[BLANK_BYTES] = { 0x00 }; static unsigned char blank_mask_bitmap[BLANK_BYTES] = { 0x00 }; /***********************************************************************/ static So@Gui@Cursor::CustomCursor zoom; static So@Gui@Cursor::CustomCursor pan; static So@Gui@Cursor::CustomCursor rotate; static So@Gui@Cursor::CustomCursor blank; static So@Gui@Cursor * soguicursor_zoomcursor = NULL; static So@Gui@Cursor * soguicursor_pancursor = NULL; static So@Gui@Cursor * soguicursor_rotatecursor = NULL; static So@Gui@Cursor * soguicursor_blankcursor = NULL; static void soguicursor_atexit_cleanup(void) { delete soguicursor_zoomcursor; delete soguicursor_pancursor; delete soguicursor_rotatecursor; delete soguicursor_blankcursor; soguicursor_zoomcursor = NULL; soguicursor_pancursor = NULL; soguicursor_rotatecursor = NULL; soguicursor_blankcursor = NULL; } void So@Gui@Cursor::initClass(void) { zoom.dim = SbVec2s(ZOOM_WIDTH, ZOOM_HEIGHT); zoom.hotspot = SbVec2s(ZOOM_HOT_X, ZOOM_HOT_Y); zoom.bitmap = zoom_bitmap; zoom.mask = zoom_mask_bitmap; pan.dim = SbVec2s(PAN_WIDTH, PAN_HEIGHT); pan.hotspot = SbVec2s(PAN_HOT_X, PAN_HOT_Y); pan.bitmap = pan_bitmap; pan.mask = pan_mask_bitmap; rotate.dim = SbVec2s(ROTATE_WIDTH, ROTATE_HEIGHT); rotate.hotspot = SbVec2s(ROTATE_HOT_X, ROTATE_HOT_Y); rotate.bitmap = rotate_bitmap; rotate.mask = rotate_mask_bitmap; blank.dim = SbVec2s(BLANK_WIDTH, BLANK_HEIGHT); blank.hotspot = SbVec2s(BLANK_HOT_X, BLANK_HOT_Y); blank.bitmap = blank_bitmap; blank.mask = blank_mask_bitmap; soguicursor_zoomcursor = new So@Gui@Cursor(&zoom); soguicursor_pancursor = new So@Gui@Cursor(&pan); soguicursor_rotatecursor = new So@Gui@Cursor(&rotate); soguicursor_blankcursor = new So@Gui@Cursor(&blank); SoAny::atexit((SoAny::atexit_f*)soguicursor_atexit_cleanup, 0); } /*! Default constructor. Creates a default cursor. */ So@Gui@Cursor::So@Gui@Cursor(void) { this->commonConstructor(So@Gui@Cursor::DEFAULT, NULL); } /*! Constructor with Shape argument, for setting up the cursor with a pre-defined shape from the underlying native toolkit. */ So@Gui@Cursor::So@Gui@Cursor(const Shape shape) { assert(shape != CUSTOM_BITMAP && "don't use this constructor for making CUSTOM_BITMAP cursors"); this->commonConstructor(shape, NULL); } /*! Constructs a new custom bitmap cursor from \a cc. Note that only a \e shallow copy will be made of the CustomCursor bitmap and mask references, so don't deallocate the memory they use until the So@Gui@Cursor has been destructed. As for the format of the cursor bitmap data, see documentation of So@Gui@Cursor::CustomCursor. */ So@Gui@Cursor::So@Gui@Cursor(const CustomCursor * ccarg) { this->commonConstructor(CUSTOM_BITMAP, ccarg); } // Private common constructor. void So@Gui@Cursor::commonConstructor(const Shape shapearg, const CustomCursor * ccarg) { this->shape = shapearg; this->cc = NULL; if (ccarg) { assert(shape == CUSTOM_BITMAP); this->cc = new CustomCursor; *(this->cc) = *ccarg; } } /*! Constructor. */ So@Gui@Cursor::So@Gui@Cursor(const So@Gui@Cursor & cursor) { this->commonConstructor(cursor.shape, cursor.cc); } /*! Destructor */ So@Gui@Cursor::~So@Gui@Cursor() { delete this->cc; } /*! Equal operator. */ So@Gui@Cursor & So@Gui@Cursor::operator=(const So@Gui@Cursor & c) { delete this->cc; this->cc = NULL; this->commonConstructor(c.shape, c.cc); return *this; } /*! Returns the shape type. If the cursor has been set by the application programmer to a bitmap cursor, So@Gui@Cursor::CUSTOM_BITMAP is returned. */ So@Gui@Cursor::Shape So@Gui@Cursor::getShape(void) const { return this->shape; } /*! Sets the cursor to a predefined shape. */ void So@Gui@Cursor::setShape(const Shape shapearg) { this->shape = shapearg; } /*! Returns a reference to the current custom bitmap cursor. Do not call this method unless So@Gui@Cursor::getShape() returns So@Gui@Cursor::CUSTOM_BITMAP. */ const So@Gui@Cursor::CustomCursor & So@Gui@Cursor::getCustomCursor(void) const { assert(cc!=NULL && "not a custom bitmap cursor"); return *(this->cc); } /*! Returns a "zoom" indicator cursor. */ const So@Gui@Cursor & So@Gui@Cursor::getZoomCursor(void) { return *soguicursor_zoomcursor; } /*! Returns a cursor with "pan" graphics (ie for translation in the camera normal plane). */ const So@Gui@Cursor & So@Gui@Cursor::getPanCursor(void) { return *soguicursor_pancursor; } /*! Returns a "rotate" indicator cursor. */ const So@Gui@Cursor & So@Gui@Cursor::getRotateCursor(void) { return *soguicursor_rotatecursor; } /*! Because all toolkits don't easily support setting up a blank cursor from pre-defined shapes or API functions, we also provide a simple completely transparent cursor. */ const So@Gui@Cursor & So@Gui@Cursor::getBlankCursor(void) { return *soguicursor_blankcursor; }