X-Git-Url: https://deadsoftware.ru/gitweb?a=blobdiff_plain;f=src%2Fgame%2Fg_items.pas;h=253a0311607bbc2f12f27e4ee4c6a85f2b234e99;hb=2b647061c5ce08e02c046c3ef8e0a31917cc28f3;hp=27dbc75bf19abf1ced60905a43afd6212a3ac526;hpb=082b546b38711030b4490facbefa37a331cb1a37;p=d2df-sdl.git diff --git a/src/game/g_items.pas b/src/game/g_items.pas index 27dbc75..253a031 100644 --- a/src/game/g_items.pas +++ b/src/game/g_items.pas @@ -68,6 +68,8 @@ function g_Items_ObjByIdx (idx: Integer): PObj; procedure g_Items_EmitPickupSound (idx: Integer); // at item position procedure g_Items_EmitPickupSoundAt (idx, x, y: Integer); +procedure g_Items_AddDynLights(); + type TItemEachAliveCB = function (it: PItem): Boolean is nested; // return `true` to stop @@ -77,7 +79,7 @@ function g_Items_ForEachAlive (cb: TItemEachAliveCB; backwards: Boolean=false): var gItemsTexturesID: Array [1..ITEM_MAX] of DWORD; - gMaxDist: Integer = 1; + gMaxDist: Integer = 1; // for sounds ITEM_RESPAWNTIME: Integer = 60 * 36; implementation @@ -86,7 +88,7 @@ uses g_basic, e_graphics, g_sound, g_main, g_gfx, g_map, Math, g_game, g_triggers, g_console, SysUtils, g_player, g_net, g_netmsg, e_log, - g_grid, binheap; + g_grid, binheap, idpool; var @@ -95,7 +97,7 @@ var // ////////////////////////////////////////////////////////////////////////// // var - freeIds: TBinaryHeapInt = nil; // free item ids + freeIds: TIdPool = nil; // ////////////////////////////////////////////////////////////////////////// // @@ -263,7 +265,7 @@ begin InitTextures(); - freeIds := binHeapNewIntLess(); + freeIds := TIdPool.Create(); end; @@ -321,6 +323,7 @@ begin g_Texture_Delete('ITEM_JETPACK'); freeIds.Free(); + freeIds := nil; end; @@ -329,8 +332,9 @@ var it: PItem; begin if (idx < 0) or (idx > High(ggItems)) then raise Exception.Create('releaseItem: invalid item id'); + if not freeIds.hasAlloced[LongWord(idx)] then raise Exception.Create('releaseItem: trying to release unallocated item (0)'); it := @ggItems[idx]; - if not it.slotIsUsed then raise Exception.Create('releaseItem: trying to release unallocated item'); + if not it.slotIsUsed then raise Exception.Create('releaseItem: trying to release unallocated item (1)'); if (it.arrIdx <> idx) then raise Exception.Create('releaseItem: arrIdx inconsistency'); it.slotIsUsed := false; if (it.Animation <> nil) then @@ -341,37 +345,38 @@ begin it.Live := False; it.SpawnTrigger := -1; it.ItemType := ITEM_NONE; - freeIds.insert(it.arrIdx); + freeIds.release(LongWord(idx)); end; -function allocItem (): DWORD; +procedure growItemArrayTo (newsz: Integer); var i, olen: Integer; it: PItem; begin - if (freeIds.count = 0) then + if (newsz < Length(ggItems)) then exit; + // no free slots + olen := Length(ggItems); + SetLength(ggItems, newsz); + for i := olen to High(ggItems) do begin - // no free slots - olen := Length(ggItems); - SetLength(ggItems, olen+64); - for i := olen to High(ggItems) do - begin - it := @ggItems[i]; - it.slotIsUsed := false; - it.arrIdx := i; - it.ItemType := ITEM_NONE; - it.Animation := nil; - it.Live := false; - it.SpawnTrigger := -1; - it.Respawnable := false; - freeIds.insert(i); - end; + it := @ggItems[i]; + it.slotIsUsed := false; + it.arrIdx := i; + it.ItemType := ITEM_NONE; + it.Animation := nil; + it.Live := false; + it.SpawnTrigger := -1; + it.Respawnable := false; + //if not freeIds.hasFree[LongWord(i)] then raise Exception.Create('internal error in item idx manager'); end; +end; - result := freeIds.front; - freeIds.popFront(); +function allocItem (): DWORD; +begin + result := freeIds.alloc(); + if (result >= Length(ggItems)) then growItemArrayTo(Integer(result)+64); if (Integer(result) > High(ggItems)) then raise Exception.Create('allocItem: freeid list corrupted'); if (ggItems[result].arrIdx <> Integer(result)) then raise Exception.Create('allocItem: arrIdx inconsistency'); end; @@ -380,51 +385,24 @@ end; // it will be slow if the slot is free (we have to rebuild the heap) function wantItemSlot (slot: Integer): Integer; var - i, olen: Integer; + olen: Integer; it: PItem; - rebuildFreeList: Boolean = true; begin if (slot < 0) or (slot > $0fffffff) then raise Exception.Create('wantItemSlot: bad item slot request'); // do we need to grow item storate? olen := Length(ggItems); - if (slot >= olen) then - begin - // need more spice! - SetLength(ggItems, slot+64); - // add free slots to free list - for i := olen to High(ggItems) do - begin - it := @ggItems[i]; - it.slotIsUsed := false; - it.arrIdx := i; - it.ItemType := ITEM_NONE; - it.Animation := nil; - it.Live := false; - it.SpawnTrigger := -1; - it.Respawnable := false; - if (i <> slot) then freeIds.insert(i); - end; - rebuildFreeList := false; - end; + if (slot >= olen) then growItemArrayTo(slot+64); it := @ggItems[slot]; if not it.slotIsUsed then begin - // this is unused slot; get it, and rebuild id list - if rebuildFreeList then - begin - freeIds.clear(); - for i := 0 to High(ggItems) do - begin - if (i <> slot) and (not it.slotIsUsed) then freeIds.insert(i); - end; - end; + freeIds.alloc(LongWord(slot)); end else begin - // it will be readded - it.slotIsUsed := false; + if not freeIds.hasAlloced[slot] then raise Exception.Create('wantItemSlot: internal error in item idx manager'); end; + it.slotIsUsed := false; result := slot; end; @@ -463,7 +441,7 @@ var begin if ForcedID < 0 then find_id := allocItem() else find_id := wantItemSlot(ForcedID); - {$IF DEFINED(D2F_DEBUG)}e_WriteLog(Format('allocated item #%d', [Integer(find_id)]), MSG_NOTIFY);{$ENDIF} + //{$IF DEFINED(D2F_DEBUG)}e_WriteLog(Format('allocated item #%d', [Integer(find_id)]), MSG_NOTIFY);{$ENDIF} it := @ggItems[find_id]; @@ -652,7 +630,7 @@ begin with ggItems[i] do begin - if g_Collide(Obj.X, Obj.Y, Obj.Rect.Width, Obj.Rect.Height, sX, sY, sWidth, sHeight) then + if g_dbg_scale_05 or g_Collide(Obj.X, Obj.Y, Obj.Rect.Width, Obj.Rect.Height, sX, sY, sWidth, sHeight) then begin if (Animation = nil) then begin @@ -903,4 +881,31 @@ begin end; end; + +procedure g_Items_AddDynLights(); +var + f: Integer; + it: PItem; +begin + for f := 0 to High(ggItems) do + begin + it := @ggItems[f]; + if not it.Live then continue; + case it.ItemType of + ITEM_KEY_RED: g_AddDynLight(it.Obj.X+(it.Obj.Rect.Width div 2), it.Obj.Y+(it.Obj.Rect.Height div 2), 24, 1.0, 0.0, 0.0, 0.6); + ITEM_KEY_GREEN: g_AddDynLight(it.Obj.X+(it.Obj.Rect.Width div 2), it.Obj.Y+(it.Obj.Rect.Height div 2), 24, 0.0, 1.0, 0.0, 0.6); + ITEM_KEY_BLUE: g_AddDynLight(it.Obj.X+(it.Obj.Rect.Width div 2), it.Obj.Y+(it.Obj.Rect.Height div 2), 24, 0.0, 0.0, 1.0, 0.6); + ITEM_ARMOR_GREEN: g_AddDynLight(it.Obj.X+(it.Obj.Rect.Width div 2), it.Obj.Y+(it.Obj.Rect.Height div 2), 42, 0.0, 1.0, 0.0, 0.6); + ITEM_ARMOR_BLUE: g_AddDynLight(it.Obj.X+(it.Obj.Rect.Width div 2), it.Obj.Y+(it.Obj.Rect.Height div 2), 42, 0.0, 0.0, 1.0, 0.6); + ITEM_SPHERE_BLUE: g_AddDynLight(it.Obj.X+(it.Obj.Rect.Width div 2), it.Obj.Y+(it.Obj.Rect.Height div 2), 32, 0.0, 1.0, 0.0, 0.6); + ITEM_SPHERE_WHITE: g_AddDynLight(it.Obj.X+(it.Obj.Rect.Width div 2), it.Obj.Y+(it.Obj.Rect.Height div 2), 32, 1.0, 1.0, 1.0, 0.6); + ITEM_INVUL: g_AddDynLight(it.Obj.X+(it.Obj.Rect.Width div 2), it.Obj.Y+(it.Obj.Rect.Height div 2), 32, 1.0, 0.0, 0.0, 0.6); + ITEM_INVIS: g_AddDynLight(it.Obj.X+(it.Obj.Rect.Width div 2), it.Obj.Y+(it.Obj.Rect.Height div 2), 32, 1.0, 1.0, 0.0, 0.6); + ITEM_BOTTLE: g_AddDynLight(it.Obj.X+(it.Obj.Rect.Width div 2), it.Obj.Y+(it.Obj.Rect.Height div 2), 16, 0.0, 0.0, 0.8, 0.6); + ITEM_HELMET: g_AddDynLight(it.Obj.X+(it.Obj.Rect.Width div 2), it.Obj.Y+(it.Obj.Rect.Height div 2), 16, 0.0, 0.8, 0.0, 0.6); + end; + end; +end; + + end.