using System; using System.Collections; using System.Collections.Generic; using System.Xml; using Server; using Server.Items; using Server.Mobiles; using Server.Gumps; using Server.Spells; using Server.Spells.Fourth; using Server.Spells.Sixth; using Server.Spells.Seventh; namespace Server.Regions { public enum SpawnZLevel { Lowest, Highest, Random } public class BaseRegion : Region { public static void Configure() { Region.DefaultRegionType = typeof( BaseRegion ); } private string m_RuneName; private bool m_NoLogoutDelay; private SpawnEntry[] m_Spawns; private SpawnZLevel m_SpawnZLevel; private bool m_ExcludeFromParentSpawns; public string RuneName{ get{ return m_RuneName; } set{ m_RuneName = value; } } public bool NoLogoutDelay{ get{ return m_NoLogoutDelay; } set{ m_NoLogoutDelay = value; } } public SpawnEntry[] Spawns { get{ return m_Spawns; } set { if ( m_Spawns != null ) { for ( int i = 0; i < m_Spawns.Length; i++ ) m_Spawns[i].Delete(); } m_Spawns = value; } } public SpawnZLevel SpawnZLevel{ get{ return m_SpawnZLevel; } set{ m_SpawnZLevel = value; } } public bool ExcludeFromParentSpawns{ get{ return m_ExcludeFromParentSpawns; } set{ m_ExcludeFromParentSpawns = value; } } public override void OnUnregister() { base.OnUnregister(); this.Spawns = null; } public static string GetRuneNameFor( Region region ) { while ( region != null ) { BaseRegion br = region as BaseRegion; if ( br != null && br.m_RuneName != null ) return br.m_RuneName; region = region.Parent; } return null; } public override TimeSpan GetLogoutDelay( Mobile m ) { if ( m_NoLogoutDelay ) { if ( m.Aggressors.Count == 0 && m.Aggressed.Count == 0 && !m.Criminal ) return TimeSpan.Zero; } return base.GetLogoutDelay( m ); } public static bool CanSpawn( Region region, params Type[] types ) { while ( region != null ) { if ( !region.AllowSpawn() ) return false; BaseRegion br = region as BaseRegion; if ( br != null ) { if ( br.Spawns != null ) { for ( int i = 0; i < br.Spawns.Length; i++ ) { SpawnEntry entry = br.Spawns[i]; if ( entry.Definition.CanSpawn( types ) ) return true; } } if ( br.ExcludeFromParentSpawns ) return false; } region = region.Parent; } return false; } public override void OnEnter( Mobile m ) { base.OnEnter(m); if ( m.Mounted && NoMounts( m, m.Location ) && m is PlayerMobile ) Server.Mobiles.StableMaster.DismountPlayer( m ); if ( m is BaseCreature && ((BaseCreature)m).SeaCreature ) { m.PlaySound( 0x026 ); Effects.SendLocationEffect( m.Location, m.Map, 0x35B2, 16 ); m.Delete(); } } public override void OnExit( Mobile m ) { base.OnExit(m); if ( !m.Mounted && m is PlayerMobile ) Server.Mobiles.StableMaster.GetLastMounted( m ); } public override bool OnBeginSpellCast( Mobile m, ISpell s ) { if ( ( s is MarkSpell || s is RecallSpell || s is GateTravelSpell ) && SpellHelper.NoRecall( m.Location, m ) ) { m.SendMessage( "For some strange reason, you cannot cast this spell!" ); m.FixedEffect( 0x3735, 6, 30 ); m.PlaySound( 0x5C ); return false; } return base.OnBeginSpellCast( m, s ); } public override bool AcceptsSpawnsFrom( Region region ) { if ( region == this || !m_ExcludeFromParentSpawns ) return base.AcceptsSpawnsFrom( region ); return false; } private Rectangle3D[] m_Rectangles; private int[] m_RectangleWeights; private int m_TotalWeight; private static List m_RectBuffer1 = new List(); private static List m_RectBuffer2 = new List(); private void InitRectangles() { if ( m_Rectangles != null ) return; // Test if area rectangles are overlapping, and in that case break them into smaller non overlapping rectangles for ( int i = 0; i < this.Area.Length; i++ ) { m_RectBuffer2.Add( this.Area[i] ); for ( int j = 0; j < m_RectBuffer1.Count && m_RectBuffer2.Count > 0; j++ ) { Rectangle3D comp = m_RectBuffer1[j]; for ( int k = m_RectBuffer2.Count - 1; k >= 0; k-- ) { Rectangle3D rect = m_RectBuffer2[k]; int l1 = rect.Start.X, r1 = rect.End.X, t1 = rect.Start.Y, b1 = rect.End.Y; int l2 = comp.Start.X, r2 = comp.End.X, t2 = comp.Start.Y, b2 = comp.End.Y; if ( l1 < r2 && r1 > l2 && t1 < b2 && b1 > t2 ) { m_RectBuffer2.RemoveAt( k ); int sz = rect.Start.Z; int ez = rect.End.X; if ( l1 < l2 ) { m_RectBuffer2.Add( new Rectangle3D( new Point3D( l1, t1, sz ), new Point3D( l2, b1, ez ) ) ); } if ( r1 > r2 ) { m_RectBuffer2.Add( new Rectangle3D( new Point3D( r2, t1, sz ), new Point3D( r1, b1, ez ) ) ); } if ( t1 < t2 ) { m_RectBuffer2.Add( new Rectangle3D( new Point3D( Math.Max( l1, l2 ), t1, sz ), new Point3D( Math.Min( r1, r2 ), t2, ez ) ) ); } if ( b1 > b2 ) { m_RectBuffer2.Add( new Rectangle3D( new Point3D( Math.Max( l1, l2 ), b2, sz ), new Point3D( Math.Min( r1, r2 ), b1, ez ) ) ); } } } } m_RectBuffer1.AddRange( m_RectBuffer2 ); m_RectBuffer2.Clear(); } m_Rectangles = m_RectBuffer1.ToArray(); m_RectBuffer1.Clear(); m_RectangleWeights = new int[m_Rectangles.Length]; for ( int i = 0; i < m_Rectangles.Length; i++ ) { Rectangle3D rect = m_Rectangles[i]; int weight = rect.Width * rect.Height; m_RectangleWeights[i] = weight; m_TotalWeight += weight; } } private static List m_SpawnBuffer1 = new List(); private static List m_SpawnBuffer2 = new List(); public Point3D RandomSpawnLocation( int spawnHeight, bool land, bool water, Point3D home, int range ) { Map map = this.Map; if ( map == Map.Internal ) return Point3D.Zero; InitRectangles(); if ( m_TotalWeight <= 0 ) return Point3D.Zero; for ( int i = 0; i < 10; i++ ) // Try 10 times { int x, y, minZ, maxZ; if ( home == Point3D.Zero ) { int rand = Utility.Random( m_TotalWeight ); x = int.MinValue; y = int.MinValue; minZ = int.MaxValue; maxZ = int.MinValue; for ( int j = 0; j < m_RectangleWeights.Length; j++ ) { int curWeight = m_RectangleWeights[j]; if ( rand < curWeight ) { Rectangle3D rect = m_Rectangles[j]; x = rect.Start.X + rand % rect.Width; y = rect.Start.Y + rand / rect.Width; minZ = rect.Start.Z; maxZ = rect.End.Z; break; } rand -= curWeight; } } else { x = Utility.RandomMinMax( home.X - range, home.X + range ); y = Utility.RandomMinMax( home.Y - range, home.Y + range ); minZ = int.MaxValue; maxZ = int.MinValue; for ( int j = 0; j < this.Area.Length; j++ ) { Rectangle3D rect = this.Area[j]; if ( x >= rect.Start.X && x < rect.End.X && y >= rect.Start.Y && y < rect.End.Y ) { minZ = rect.Start.Z; maxZ = rect.End.Z; break; } } if ( minZ == int.MaxValue ) continue; } if ( x < 0 || y < 0 || x >= map.Width || y >= map.Height ) continue; LandTile lt = map.Tiles.GetLandTile( x, y ); int ltLowZ = 0, ltAvgZ = 0, ltTopZ = 0; map.GetAverageZ( x, y, ref ltLowZ, ref ltAvgZ, ref ltTopZ ); TileFlag ltFlags = TileData.LandTable[lt.ID & TileData.MaxLandValue].Flags; bool ltImpassable = ( (ltFlags & TileFlag.Impassable) != 0 ); if ( !lt.Ignored && ltAvgZ >= minZ && ltAvgZ < maxZ ) if ( (ltFlags & TileFlag.Wet) != 0 ) { if ( water ) m_SpawnBuffer1.Add( ltAvgZ ); } else if ( land && !ltImpassable ) m_SpawnBuffer1.Add( ltAvgZ ); StaticTile[] staticTiles = map.Tiles.GetStaticTiles( x, y, true ); for ( int j = 0; j < staticTiles.Length; j++ ) { StaticTile tile = staticTiles[j]; ItemData id = TileData.ItemTable[tile.ID & TileData.MaxItemValue]; int tileZ = tile.Z + id.CalcHeight; if ( tileZ >= minZ && tileZ < maxZ ) if ( (id.Flags & TileFlag.Wet) != 0 ) { if ( water ) m_SpawnBuffer1.Add( tileZ ); } else if ( land && id.Surface && !id.Impassable ) m_SpawnBuffer1.Add( tileZ ); } Sector sector = map.GetSector( x, y ); for ( int j = 0; j < sector.Items.Count; j++ ) { Item item = sector.Items[j]; if ( !(item is BaseMulti) && item.ItemID <= TileData.MaxItemValue && item.AtWorldPoint( x, y ) ) { m_SpawnBuffer2.Add( item ); if ( !item.Movable ) { ItemData id = item.ItemData; int itemZ = item.Z + id.CalcHeight; if ( itemZ >= minZ && itemZ < maxZ ) if ( (id.Flags & TileFlag.Wet) != 0 ) { if ( water ) m_SpawnBuffer1.Add( itemZ ); } else if ( land && id.Surface && !id.Impassable ) m_SpawnBuffer1.Add( itemZ ); } } } if ( m_SpawnBuffer1.Count == 0 ) { m_SpawnBuffer1.Clear(); m_SpawnBuffer2.Clear(); continue; } int z; switch ( m_SpawnZLevel ) { case SpawnZLevel.Lowest: { z = int.MaxValue; for ( int j = 0; j < m_SpawnBuffer1.Count; j++ ) { int l = m_SpawnBuffer1[j]; if ( l < z ) z = l; } break; } case SpawnZLevel.Highest: { z = int.MinValue; for ( int j = 0; j < m_SpawnBuffer1.Count; j++ ) { int l = m_SpawnBuffer1[j]; if ( l > z ) z = l; } break; } default: // SpawnZLevel.Random { int index = Utility.Random( m_SpawnBuffer1.Count ); z = m_SpawnBuffer1[index]; break; } } m_SpawnBuffer1.Clear(); if ( !Region.Find( new Point3D( x, y, z ), map ).AcceptsSpawnsFrom( this ) ) { m_SpawnBuffer2.Clear(); continue; } int top = z + spawnHeight; bool ok = true; for ( int j = 0; j < m_SpawnBuffer2.Count; j++ ) { Item item = m_SpawnBuffer2[j]; ItemData id = item.ItemData; if ( ( id.Surface || id.Impassable ) && item.Z + id.CalcHeight > z && item.Z < top ) { ok = false; break; } } m_SpawnBuffer2.Clear(); if ( !ok ) continue; if ( ltImpassable && ltAvgZ > z && ltLowZ < top ) continue; for ( int j = 0; j < staticTiles.Length; j++ ) { StaticTile tile = staticTiles[j]; ItemData id = TileData.ItemTable[tile.ID & TileData.MaxItemValue]; if ( ( id.Surface || id.Impassable ) && tile.Z + id.CalcHeight > z && tile.Z < top ) { ok = false; break; } } if ( !ok ) continue; for ( int j = 0; j < sector.Mobiles.Count; j++ ) { Mobile m = sector.Mobiles[j]; if ( m.X == x && m.Y == y && ( m.AccessLevel == AccessLevel.Player || !m.Hidden ) ) if ( m.Z + 16 > z && m.Z < top ) { ok = false; break; } } if ( ok ) return new Point3D( x, y, z ); } return Point3D.Zero; } public override string ToString() { if ( this.Name != null ) return this.Name; else if ( this.RuneName != null ) return this.RuneName; else return this.GetType().Name; } public BaseRegion( string name, Map map, int priority, params Rectangle2D[] area ) : base( name, map, priority, area ) { } public BaseRegion( string name, Map map, int priority, params Rectangle3D[] area ) : base( name, map, priority, area ) { } public BaseRegion( string name, Map map, Region parent, params Rectangle2D[] area ) : base( name, map, parent, area ) { } public BaseRegion( string name, Map map, Region parent, params Rectangle3D[] area ) : base( name, map, parent, area ) { } public BaseRegion( XmlElement xml, Map map, Region parent ) : base( xml, map, parent ) { ReadString( xml["rune"], "name", ref m_RuneName, false ); bool logoutDelayActive = true; ReadBoolean( xml["logoutDelay"], "active", ref logoutDelayActive, false ); m_NoLogoutDelay = !logoutDelayActive; XmlElement spawning = xml["spawning"]; if ( spawning != null ) { ReadBoolean( spawning, "excludeFromParent", ref m_ExcludeFromParentSpawns, false ); SpawnZLevel zLevel = SpawnZLevel.Lowest; ReadEnum( spawning, "zLevel", ref zLevel, false ); m_SpawnZLevel = zLevel; List list = new List(); foreach ( XmlNode node in spawning.ChildNodes ) { XmlElement el = node as XmlElement; if ( el != null ) { SpawnDefinition def = SpawnDefinition.GetSpawnDefinition( el ); if ( def == null ) continue; int id = 0; if ( !ReadInt32( el, "id", ref id, true ) ) continue; int amount = 0; if ( !ReadInt32( el, "amount", ref amount, true ) ) continue; TimeSpan minSpawnTime = SpawnEntry.DefaultMinSpawnTime; ReadTimeSpan( el, "minSpawnTime", ref minSpawnTime, false ); TimeSpan maxSpawnTime = SpawnEntry.DefaultMaxSpawnTime; ReadTimeSpan( el, "maxSpawnTime", ref maxSpawnTime, false ); Point3D home = Point3D.Zero; int range = 0; XmlElement homeEl = el["home"]; if ( ReadPoint3D( homeEl, map, ref home, false ) ) ReadInt32( homeEl, "range", ref range, false ); Direction dir = SpawnEntry.InvalidDirection; ReadEnum( el["direction"], "value" , ref dir, false ); SpawnEntry entry = new SpawnEntry( id, this, home, range, dir, def, amount, minSpawnTime, maxSpawnTime ); list.Add( entry ); } } if ( list.Count > 0 ) { m_Spawns = list.ToArray(); } } } public static Point3D GetBoatWater( int x, int y, Map map, int range ) { bool WaterOk = false; Point3D loc = new Point3D(0, 0, 0); Map tm = map; int tx = 0; int ty = 0; int tz = 0; int r = 0; LandTile t = tm.Tiles.GetLandTile(tx, ty); while ( !WaterOk ) { tx = Utility.RandomMinMax( x+range, x-range ); ty = Utility.RandomMinMax( y+range, y-range ); tz = tm.GetAverageZ(tx, ty); t = tm.Tiles.GetLandTile(tx, ty); if ( IsWaterTile ( t.ID ) ) WaterOk = true; if ( WaterOk ) loc = new Point3D(tx, ty, tz); r++; // SAFETY CATCH if ( r > 50 ) { WaterOk = true; } } return loc; } public static bool IsWaterTile ( int id ) { if ( id==0x00A8 || id==0x00A9 || id==0x00AA || id==0x00AB ) return true; return false; } public static bool TestOcean( Map map, int x, int y, int distance ) { int results = 0; LandTile seaTile1 = map.Tiles.GetLandTile( x-distance, y-distance ); LandTile seaTile2 = map.Tiles.GetLandTile( x, y-distance ); LandTile seaTile3 = map.Tiles.GetLandTile( x+distance, y-distance ); LandTile seaTile4 = map.Tiles.GetLandTile( x-distance, y ); LandTile seaTile5 = map.Tiles.GetLandTile( x+distance, y ); LandTile seaTile6 = map.Tiles.GetLandTile( x-distance, y+distance ); LandTile seaTile7 = map.Tiles.GetLandTile( x, y+distance ); LandTile seaTile8 = map.Tiles.GetLandTile( x+distance, y+distance ); if ( !IsWaterTile( seaTile1.ID ) ){ results ++; } if ( !IsWaterTile( seaTile2.ID ) ){ results ++; } if ( !IsWaterTile( seaTile3.ID ) ){ results ++; } if ( !IsWaterTile( seaTile4.ID ) ){ results ++; } if ( !IsWaterTile( seaTile5.ID ) ){ results ++; } if ( !IsWaterTile( seaTile6.ID ) ){ results ++; } if ( !IsWaterTile( seaTile7.ID ) ){ results ++; } if ( !IsWaterTile( seaTile8.ID ) ){ results ++; } if ( results > 0 ) return false; return true; } public static Point3D GetOceanSpot() { bool IsWater = false; Point3D loc = new Point3D(0, 0, 0); Point3D failover = new Point3D(100, 100, -5); Map map = Map.Britannia; int tx = 0; int ty = 0; int tz = 0; bool run = true; int r = 0; while ( run ) { tx = Utility.RandomMinMax( 26, 7142 ); ty = Utility.RandomMinMax( 26, 4070 ); tz = map.GetAverageZ(tx, ty); LandTile t = map.Tiles.GetLandTile(tx, ty); if ( IsWaterTile( t.ID ) && TestOcean( map, tx, ty, 10 ) ) IsWater = true; Point3D locale = new Point3D(tx, ty, tz); Region reg = Region.Find( locale, map ); if ( tz != -5 ) IsWater = false; if ( IsWater && reg == map.DefaultRegion ) { loc = locale; run = false; } r++; // SAFETY CATCH if ( r > 5000 && run ) { loc = failover; run = false; } } return loc; } } }