/* MOTION_SENSOR.C Copyright (C) 1991-2001 and beyond by Bungie Studios, Inc. and the "Aleph One" developers. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. This license is contained in the file "COPYING", which is included with this source code; it is available online at http://www.gnu.org/licenses/gpl.html Saturday, June 11, 1994 1:37:32 AM Friday, September 16, 1994 2:04:47 PM (Jason') removed all get_shape_information() calls. Monday, December 5, 1994 9:19:55 PM (Jason) flickers in magnetic environments Feb 18, 2000 (Loren Petrich): Made VacBobs display properly Mar 23, 2000 (Loren Petrich): Made motion-sensor monster typing more generic; since it's now read off of a table, it can easily be changed without rebuilding the app. Sep 2, 2000 (Loren Petrich): Idiot-proofed the shapes display, since the shape accessor now returns NULL pointers for nonexistent bitmaps. */ #include "cseries.h" #include "map.h" #include "monsters.h" #include "render.h" #include "interface.h" #include "motion_sensor.h" #include "player.h" #include "network_games.h" #include "HUDRenderer_SW.h" #include "HUDRenderer_OGL.h" #include #include #include #ifdef env68k #pragma segment texture #endif // LP change: creating an array describing the shape assignments // of the various monsters, so as to allow better customization enum { MType_Friend, // What you, friendly players, and the Bobs are MType_Alien, // What the other critters are MType_Enemy, // What hostile players are NUMBER_OF_MDISPTYPES }; static short MonsterDisplays[NUMBER_OF_MONSTER_TYPES] = { // Marine MType_Friend, // Ticks MType_Alien, MType_Alien, MType_Alien, // S'pht MType_Alien, MType_Alien, MType_Alien, MType_Alien, // Pfhor MType_Alien, MType_Alien, MType_Alien, MType_Alien, // Bob MType_Friend, MType_Friend, MType_Friend, MType_Friend, // Drone MType_Alien, MType_Alien, MType_Alien, MType_Alien, MType_Alien, // Cyborg MType_Alien, MType_Alien, MType_Alien, MType_Alien, // Enforcer MType_Alien, MType_Alien, // Hunter MType_Alien, MType_Alien, // Trooper MType_Alien, MType_Alien, // Big Cyborg, Hunter MType_Alien, MType_Alien, // F'lickta MType_Alien, MType_Alien, MType_Alien, // S'pht'Kr MType_Alien, MType_Alien, // Juggernauts MType_Alien, MType_Alien, // Tiny ones MType_Alien, MType_Alien, MType_Alien, // VacBobs MType_Friend, MType_Friend, MType_Friend, MType_Friend, }; /* ??monsters which translate only on frame changes periodically drop off the motion sensor making it appear jumpy //when entities first appear on the sensor they are initially visible //entities are drawn one at a time, so the dark spot from one entity can cover the light spot from another //sometimes the motion sensor shapes are not masked correctly (fixed from SHAPES.C) //must save old points in real world coordinates because entities get redrawn when the player turns */ #define MAXIMUM_MOTION_SENSOR_ENTITIES 12 #define NUMBER_OF_PREVIOUS_LOCATIONS 6 struct motion_sensor_definition { uint16 update_frequency; uint16 rescan_frequency; uint16 range; int16 scale; }; struct motion_sensor_definition motion_sensor_settings = { 5, // update_frequency 15, // rescan_frequency (8 * WORLD_ONE), // range 64 // scale }; #define MOTION_SENSOR_UPDATE_FREQUENCY motion_sensor_settings.update_frequency #define MOTION_SENSOR_RESCAN_FREQUENCY motion_sensor_settings.rescan_frequency #define MOTION_SENSOR_RANGE motion_sensor_settings.range #define MOTION_SENSOR_SCALE motion_sensor_settings.scale #define OBJECT_IS_VISIBLE_TO_MOTION_SENSOR(o) true #define FLICKER_FREQUENCY 0xf /* ---------- structures */ /* an entity canÕt just be jerked from the array, because his signal should fade out, so we mark him as Ōbeing removedÕ and wait until his last signal fades away to actually remove him */ #define SLOT_IS_BEING_REMOVED_BIT 0x4000 #define SLOT_IS_BEING_REMOVED(e) ((e)->flags&(uint16)SLOT_IS_BEING_REMOVED_BIT) #define MARK_SLOT_AS_BEING_REMOVED(e) ((e)->flags|=(uint16)SLOT_IS_BEING_REMOVED_BIT) struct entity_data { uint16 flags; /* [slot_used.1] [slot_being_removed.1] [unused.14] */ short monster_index; shape_descriptor shape; short remove_delay; /* only valid if this entity is being removed [0,NUMBER_OF_PREVIOUS_LOCATIONS) */ point2d previous_points[NUMBER_OF_PREVIOUS_LOCATIONS]; bool visible_flags[NUMBER_OF_PREVIOUS_LOCATIONS]; world_point3d last_location; angle last_facing; }; struct region_data { short x0, x1; }; /* ---------- globals */ static short motion_sensor_player_index; static short motion_sensor_side_length; static struct region_data *sensor_region; static struct entity_data *entities; static short network_compass_state; static shape_descriptor mount_shape, virgin_mount_shapes, compass_shapes; static shape_descriptor alien_shapes, friendly_shapes, enemy_shapes; static bool motion_sensor_changed; static long ticks_since_last_update, ticks_since_last_rescan; /* ---------- private code */ static void precalculate_sensor_region(short side_length); static short find_or_add_motion_sensor_entity(short monster_index); static shape_descriptor get_motion_sensor_entity_shape(short monster_index); static void clipped_transparent_sprite_copy(struct bitmap_definition *source, struct bitmap_definition *destination, struct region_data *region, short x0, short y0); static void bitmap_window_copy(struct bitmap_definition *source, struct bitmap_definition *destination, short x0, short y0, short x1, short y1); static void unclipped_solid_sprite_copy(struct bitmap_definition *source, struct bitmap_definition *destination, short x0, short y0); /* ---------- code */ void initialize_motion_sensor( shape_descriptor mount, shape_descriptor virgin_mounts, shape_descriptor aliens, shape_descriptor friends, shape_descriptor enemies, shape_descriptor compasses, short side_length) { mount_shape= mount; virgin_mount_shapes= virgin_mounts; enemy_shapes= enemies; friendly_shapes= friends; alien_shapes= aliens; compass_shapes= compasses; entities= new entity_data[MAXIMUM_MOTION_SENSOR_ENTITIES]; assert(entities); for (int i = 0; i < MAXIMUM_MOTION_SENSOR_ENTITIES; i++) { entities[i].flags = 0; } sensor_region= new region_data[side_length]; assert(sensor_region); /* precalculate the sensor region */ precalculate_sensor_region(side_length); /* reset_motion_sensor() should be called before the motion sensor is used, but after itÕs shapes are loaded (because it will do bitmap copying) */ } void reset_motion_sensor( short player_index) { struct bitmap_definition *mount, *virgin_mount; short i; motion_sensor_player_index= player_index; ticks_since_last_update= ticks_since_last_rescan= 0; get_shape_bitmap_and_shading_table(mount_shape, &mount, (void **) NULL, NONE); if (!mount) return; get_shape_bitmap_and_shading_table(virgin_mount_shapes, &virgin_mount, (void **) NULL, NONE); if (!virgin_mount) return; assert(mount->width==virgin_mount->width); assert(mount->height==virgin_mount->height); bitmap_window_copy(virgin_mount, mount, 0, 0, mount->width, mount->height); for (i= 0; iobject_index); /* if we need to scan for new objects, flood around the owner monster looking for other, visible monsters within our range */ if ((ticks_since_last_rescan-= ticks_elapsed)<0 || ticks_elapsed==NONE) { struct monster_data *monster; short monster_index; for (monster_index=0,monster=monsters;monster_indexobject_index); world_distance distance= guess_distance2d((world_point2d *) &object->location, (world_point2d *) &owner_object->location); if (distancepermutation); } } } ticks_since_last_rescan= MOTION_SENSOR_RESCAN_FREQUENCY; } render_motion_sensor(ticks_elapsed); } void HUD_SW_Class::render_motion_sensor(short ticks_elapsed) { // If we need to update the motion sensor, draw all active entities if ((ticks_since_last_update -= ticks_elapsed) < 0 || ticks_elapsed == NONE) { erase_all_entity_blips(); /*if (dynamic_world->player_count > 1)*/ draw_network_compass(); draw_all_entity_blips(); ticks_since_last_update = MOTION_SENSOR_UPDATE_FREQUENCY; motion_sensor_changed = true; } } void HUD_OGL_Class::render_motion_sensor(short ticks_elapsed) { // Draw background screen_rectangle *r = get_interface_rectangle(_motion_sensor_rect); DrawShapeAtXY(BUILD_DESCRIPTOR(_collection_interface, _motion_sensor_virgin_mount), r->left, r->top); // We allways draw all active entities because we have to update the // display on every frame if ((ticks_since_last_update -= ticks_elapsed) < 0 || ticks_elapsed == NONE) { erase_all_entity_blips(); ticks_since_last_update = MOTION_SENSOR_UPDATE_FREQUENCY; } /*if (dynamic_world->player_count > 1)*/ draw_network_compass(); draw_all_entity_blips(); } /* the interface code will call this function and only draw the motion sensor if we return true */ bool motion_sensor_has_changed(void) { bool changed = motion_sensor_changed; motion_sensor_changed = false; return changed; } /* toggle through the ranges */ void adjust_motion_sensor_range(void) { } /* ---------- private code */ void HUD_Class::draw_network_compass(void) { #if !defined(DISABLE_NETWORKING) short new_state= get_network_compass_state(motion_sensor_player_index); short difference= (new_state^network_compass_state)|new_state; if (difference&_network_compass_nw) draw_or_erase_unclipped_shape(36, 36, compass_shapes, (new_state&_network_compass_nw) != 0); if (difference&_network_compass_ne) draw_or_erase_unclipped_shape(61, 36, compass_shapes+1, (new_state&_network_compass_ne) != 0); if (difference&_network_compass_se) draw_or_erase_unclipped_shape(61, 61, compass_shapes+3, (new_state&_network_compass_se) != 0); if (difference&_network_compass_sw) draw_or_erase_unclipped_shape(36, 61, compass_shapes+2, (new_state&_network_compass_sw) != 0); network_compass_state= new_state; #endif // !defined(DISABLE_NETWORKING) } void HUD_Class::erase_all_entity_blips(void) { struct object_data *owner_object= get_object_data(get_player_data(motion_sensor_player_index)->object_index); struct entity_data *entity; short entity_index; /* first erase all locations where the entity changed locations and then did not change locations, and erase itÕs last location */ for (entity_index=0,entity=entities;entity_indexmonster_index)) { struct object_data *object= get_object_data(get_monster_data(entity->monster_index)->object_index); world_distance distance= guess_distance2d((world_point2d *) &object->location, (world_point2d *) &owner_object->location); /* verify that weÕre still in range (and mark us as being removed if weÕre not */ if (distance>MOTION_SENSOR_RANGE || !OBJECT_IS_VISIBLE_TO_MOTION_SENSOR(object)) { // dprintf("removed2"); MARK_SLOT_AS_BEING_REMOVED(entity); } } else { // dprintf("removed1"); MARK_SLOT_AS_BEING_REMOVED(entity); } /* erase the blip specified by NUMBER_OF_PREVIOUS_LOCATIONS-1 */ if (entity->visible_flags[NUMBER_OF_PREVIOUS_LOCATIONS-1]) erase_entity_blip(&entity->previous_points[NUMBER_OF_PREVIOUS_LOCATIONS-1], entity->shape); /* adjust the arrays to make room for new entries */ memmove(entity->visible_flags+1, entity->visible_flags, (NUMBER_OF_PREVIOUS_LOCATIONS-1)*sizeof(bool)); memmove(entity->previous_points+1, entity->previous_points, (NUMBER_OF_PREVIOUS_LOCATIONS-1)*sizeof(point2d)); entity->visible_flags[0]= false; /* if weÕre not being removed, make room for a new location and calculate it */ if (!SLOT_IS_BEING_REMOVED(entity)) { struct monster_data *monster= get_monster_data(entity->monster_index); struct object_data *object= get_object_data(monster->object_index); /* remember if this entity is visible or not */ if (object->transfer_mode!=_xfer_invisibility && object->transfer_mode!=_xfer_subtle_invisibility && (!(static_world->environment_flags&_environment_magnetic) || !((dynamic_world->tick_count+4*monster->object_index)&FLICKER_FREQUENCY))) { if (object->location.x!=entity->last_location.x || object->location.y!=entity->last_location.y || object->location.z!=entity->last_location.z || object->facing!=entity->last_facing) { entity->visible_flags[0]= true; entity->last_location= object->location; entity->last_facing= object->facing; } } /* calculate the 2d position on the motion sensor */ entity->previous_points[0]= *(point2d *)&object->location; transform_point2d((world_point2d *)&entity->previous_points[0], (world_point2d *)&owner_object->location, NORMALIZE_ANGLE(owner_object->facing+QUARTER_CIRCLE)); //entity->previous_points[0].x>>= MOTION_SENSOR_SCALE; entity->previous_points[0].x /= MOTION_SENSOR_SCALE; // entity->previous_points[0].y>>= MOTION_SENSOR_SCALE; entity->previous_points[0].y /= MOTION_SENSOR_SCALE; } else { /* erase the blip specified by entity->remove_delay */ if (entity->visible_flags[entity->remove_delay]) erase_entity_blip(&entity->previous_points[entity->remove_delay], entity->shape); /* if this is the last point of an entity which was being removed; mark it as unused */ if ((entity->remove_delay+= 1)>=NUMBER_OF_PREVIOUS_LOCATIONS) { MARK_SLOT_AS_FREE(entity); } } } } } void HUD_Class::draw_all_entity_blips(void) { struct entity_data *entity; short entity_index, intensity; for (intensity=NUMBER_OF_PREVIOUS_LOCATIONS-1;intensity>=0;--intensity) { for (entity_index=0,entity=entities;entity_indexvisible_flags[intensity]) { draw_entity_blip(&entity->previous_points[intensity], entity->shape + intensity); } } } } } void HUD_SW_Class::draw_or_erase_unclipped_shape(short x, short y, shape_descriptor shape, bool draw) { struct bitmap_definition *mount, *virgin_mount, *blip; get_shape_bitmap_and_shading_table(mount_shape, &mount, (void **) NULL, NONE); if (!mount) return; get_shape_bitmap_and_shading_table(virgin_mount_shapes, &virgin_mount, (void **) NULL, NONE); if (!virgin_mount) return; get_shape_bitmap_and_shading_table(shape, &blip, (void **) NULL, NONE); if (!blip) return; draw ? unclipped_solid_sprite_copy(blip, mount, x, y) : bitmap_window_copy(virgin_mount, mount, x, y, x+blip->width, y+blip->height); } void HUD_OGL_Class::draw_or_erase_unclipped_shape(short x, short y, shape_descriptor shape, bool draw) { if (draw) { screen_rectangle *r = get_interface_rectangle(_motion_sensor_rect); DrawShapeAtXY(shape, x + r->left, y + r->top); } } void HUD_SW_Class::erase_entity_blip(point2d *location, shape_descriptor shape) { struct bitmap_definition *mount, *virgin_mount, *blip; short x, y; get_shape_bitmap_and_shading_table(mount_shape, &mount, (void **) NULL, NONE); if (!mount) return; get_shape_bitmap_and_shading_table(virgin_mount_shapes, &virgin_mount, (void **) NULL, NONE); if (!virgin_mount) return; get_shape_bitmap_and_shading_table(shape, &blip, (void **) NULL, NONE); if (!blip) return; x= location->x + (motion_sensor_side_length>>1) - (blip->width>>1); y= location->y + (motion_sensor_side_length>>1) - (blip->height>>1); bitmap_window_copy(virgin_mount, mount, x, y, x+blip->width, y+blip->height); } void HUD_SW_Class::draw_entity_blip(point2d *location, shape_descriptor shape) { bitmap_definition *mount, *blip; get_shape_bitmap_and_shading_table(mount_shape, &mount, (void **) NULL, NONE); if (!mount) return; get_shape_bitmap_and_shading_table(shape, &blip, (void **) NULL, NONE); if (!blip) return; clipped_transparent_sprite_copy(blip, mount, sensor_region, location->x + (motion_sensor_side_length>>1) - (blip->width>>1), location->y + (motion_sensor_side_length>>1) - (blip->height>>1)); } void HUD_OGL_Class::draw_entity_blip(point2d *location, shape_descriptor shape) { bitmap_definition *blip; get_shape_bitmap_and_shading_table(shape, &blip, (void **) NULL, NONE); if (!blip) return; screen_rectangle *r = get_interface_rectangle(_motion_sensor_rect); int x = location->x, y = location->y; int c_x = r->left + (motion_sensor_side_length >> 1); int c_y = r->top + (motion_sensor_side_length >> 1); SetClipPlane(x, y, c_x, c_y, motion_sensor_side_length >> 1); DrawShapeAtXY(shape, x + c_x - (blip->width >> 1), y + c_y - (blip->height >> 1), true); DisableClipPlane(); } /* if we find an entity that is being removed, we continue with the removal process and ignore the new signal; the new entity will probably not be added to the sensor again for a full second or so (the range should be set so that this is reasonably hard to do) */ static short find_or_add_motion_sensor_entity( short monster_index) { struct entity_data *entity; short entity_index, best_unused_index; best_unused_index= NONE; for (entity_index=0,entity=entities;entity_indexmonster_index==monster_index) break; } else { if (best_unused_index==NONE) best_unused_index= entity_index; } } if (entity_index==MAXIMUM_MOTION_SENSOR_ENTITIES) { /* not found; add new entity if we can */ if (best_unused_index!=NONE) { struct monster_data *monster= get_monster_data(monster_index); struct object_data *object= get_object_data(monster->object_index); short i; entity= entities+best_unused_index; entity->flags= 0; entity->monster_index= monster_index; entity->shape= get_motion_sensor_entity_shape(monster_index); for (i=0;ivisible_flags[i]= false; entity->last_location= object->location; entity->last_facing= object->facing; entity->remove_delay= 0; MARK_SLOT_AS_USED(entity); // dprintf("new index, pointer: %d, %p", best_unused_index, entity); } entity_index= best_unused_index; } return entity_index; } static void precalculate_sensor_region( short side_length) { double half_side_length= side_length/2.0; double r= half_side_length + 1.0; short i; /* save length for assert() during rendering */ motion_sensor_side_length= side_length; /* precompute [x0,x1] clipping values for each y value in the circular sensor */ for (i=0;i=r) x= r-1.0; sensor_region[i].x0= int16(half_side_length-x); sensor_region[i].x1= int16(x+half_side_length); } } /* (x0,y0) and (x1,y1) specify a window to be copied from the source to (x2,y2) in the destination. pixel index zero is transparent (handles clipping) */ static void bitmap_window_copy( struct bitmap_definition *source, struct bitmap_definition *destination, short x0, short y0, short x1, short y1) { short count; short y; assert(x0<=x1&&y0<=y1); if (x0<0) x0= 0; if (y0<0) y0= 0; if (x1>source->width) x1= source->width; if (y1>source->height) y1= source->height; assert(source->width==destination->width); assert(source->height==destination->height); assert(destination->width==motion_sensor_side_length); assert(destination->height==motion_sensor_side_length); for (y=y0;yrow_addresses[y]+x0; register pixel8 *write= destination->row_addresses[y]+x0; for (count=x1-x0;count>0;--count) *write++= *read++; } } static void clipped_transparent_sprite_copy( struct bitmap_definition *source, struct bitmap_definition *destination, struct region_data *region, short x0, short y0) { short height, y; assert(destination->width==motion_sensor_side_length); assert(destination->height==motion_sensor_side_length); y= 0; height= source->height; if (y0+height>destination->height) height= destination->height-y0; if (y0<0) { y= -y0; height+= y0; } while ((height-= 1)>=0) { register pixel8 pixel, *read, *write; register short width= source->width; short clip_left= region[y0+y].x0, clip_right= region[y0+y].x1; short offset= 0; if (x0clip_right) width= clip_right-x0-offset; assert(y>=0&&yheight); assert(y0+y>=0&&y0+yheight); read= source->row_addresses[y]+offset; write= destination->row_addresses[y0+y]+x0+offset; while ((width-= 1)>=0) { if ((pixel= *read++)!=0) { *write++= pixel; } else { write+= 1; } } y+= 1; } } static void unclipped_solid_sprite_copy( struct bitmap_definition *source, struct bitmap_definition *destination, short x0, short y0) { short height, y; y= 0; height= source->height; while ((height-= 1)>=0) { register pixel8 *read, *write; register short width= source->width; assert(y>=0&&yheight); assert(y0+y>=0&&y0+yheight); read= source->row_addresses[y]; write= destination->row_addresses[y0+y]+x0; while ((width-= 1)>=0) *write++= *read++; y+= 1; } } static shape_descriptor get_motion_sensor_entity_shape( short monster_index) { struct monster_data *monster= get_monster_data(monster_index); shape_descriptor shape; if (MONSTER_IS_PLAYER(monster)) { struct player_data *player= get_player_data(monster_index_to_player_index(monster_index)); struct player_data *owner= get_player_data(motion_sensor_player_index); shape= ((player->team==owner->team && !(GET_GAME_OPTIONS()&_force_unique_teams)) || GET_GAME_TYPE()==_game_of_cooperative_play) ? friendly_shapes : enemy_shapes; } else { /* switch (monster->type) { case _civilian_crew: case _civilian_science: case _civilian_security: case _civilian_assimilated: // LP additions: the VacBobs case _civilian_fusion_crew: case _civilian_fusion_science: case _civilian_fusion_security: case _civilian_fusion_assimilated: shape= friendly_shapes; break; default: shape= alien_shapes; break; } */ switch(MonsterDisplays[monster->type]) { case MType_Friend: shape = friendly_shapes; break; case MType_Alien: shape = alien_shapes; break; case MType_Enemy: default: shape = enemy_shapes; break; } } return shape; } // XML elements for parsing motion-sensor specification; // this is a specification of what monster type gets what // motion-sensor blip short *OriginalMonsterDisplays = NULL; // This assigns an motion-sensir blip to a monster type class XML_MotSensAssignParser: public XML_ElementParser { bool IsPresent[2]; short Monster, Type; public: bool Start(); bool HandleAttribute(const char *Tag, const char *Value); bool AttributesDone(); bool ResetValues(); XML_MotSensAssignParser(): XML_ElementParser("assign") {} }; bool XML_MotSensAssignParser::Start() { // back up old values first if (!OriginalMonsterDisplays) { OriginalMonsterDisplays = (short *) malloc(sizeof(short) * NUMBER_OF_MONSTER_TYPES); assert(OriginalMonsterDisplays); for (int i = 0; i < NUMBER_OF_MONSTER_TYPES; i++) OriginalMonsterDisplays[i] = MonsterDisplays[i]; } for (int k=0; k<2; k++) IsPresent[k] = false; return true; } bool XML_MotSensAssignParser::HandleAttribute(const char *Tag, const char *Value) { if (StringsEqual(Tag,"monster")) { if (ReadBoundedInt16Value(Value,Monster,0,NUMBER_OF_MONSTER_TYPES-1)) { IsPresent[0] = true; return true; } else return false; } else if (StringsEqual(Tag,"type")) { if (ReadBoundedInt16Value(Value,Type,0,NUMBER_OF_MDISPTYPES-1)) { IsPresent[1] = true; return true; } else return false; } UnrecognizedTag(); return false; } bool XML_MotSensAssignParser::AttributesDone() { // Verify... bool AllPresent = true; for (int k=0; k<2; k++) if (!IsPresent[k]) AllPresent = false; if (!AllPresent) { AttribsMissing(); return false; } // Put into place MonsterDisplays[Monster] = Type; return true; } bool XML_MotSensAssignParser::ResetValues() { if (OriginalMonsterDisplays) { for (int i = 0; i < NUMBER_OF_MONSTER_TYPES; i++) MonsterDisplays[i] = OriginalMonsterDisplays[i]; free(OriginalMonsterDisplays); OriginalMonsterDisplays = NULL; } return true; } static XML_MotSensAssignParser MotSensAssignParser; struct motion_sensor_definition *original_motion_sensor_settings = NULL; // Subclassed to set the color objects appropriately class XML_MotSensParser: public XML_ElementParser { public: bool Start() { // back up old values first if (!original_motion_sensor_settings) { original_motion_sensor_settings = (struct motion_sensor_definition *) malloc(sizeof(struct motion_sensor_definition)); assert(original_motion_sensor_settings); *original_motion_sensor_settings = motion_sensor_settings; } return true; } bool HandleAttribute(const char *Tag, const char *Value) { if (StringsEqual(Tag, "scale")) { return (ReadInt16Value(Value, MOTION_SENSOR_SCALE)); } else if (StringsEqual(Tag, "range")) { float range; if (ReadFloatValue(Value, range)) { MOTION_SENSOR_RANGE = range * WORLD_ONE; return true; } else { return false; } } else if (StringsEqual(Tag, "update_frequency")) { return (ReadUInt16Value(Value, MOTION_SENSOR_UPDATE_FREQUENCY)); } else if (StringsEqual(Tag, "rescan_frequency")) { return (ReadUInt16Value(Value, MOTION_SENSOR_RESCAN_FREQUENCY)); } UnrecognizedTag(); return false; } bool ResetValues() { if (original_motion_sensor_settings) { motion_sensor_settings = *original_motion_sensor_settings; free(original_motion_sensor_settings); original_motion_sensor_settings = NULL; } return true; } XML_MotSensParser(): XML_ElementParser("motion_sensor") {} }; static XML_MotSensParser MotSensParser; // LP change: added infravision-parser export XML_ElementParser *MotionSensor_GetParser() { MotSensParser.AddChild(&MotSensAssignParser); return &MotSensParser; }