3 {$INCLUDE ../shared/a_modes.inc}
8 LCLIntf
, LCLType
, SysUtils
, Variants
, Classes
, Graphics
,
9 Controls
, Forms
, Dialogs
, StdCtrls
, Buttons
,
10 ComCtrls
, ValEdit
, Types
, Menus
, ExtCtrls
,
11 CheckLst
, Grids
, OpenGLContext
, utils
, UTF8Process
;
17 TMainForm
= class(TForm
)
22 miMenuFile
: TMenuItem
;
26 miSaveMapAs
: TMenuItem
;
27 miOpenWadMap
: TMenuItem
;
29 miReopenMap
: TMenuItem
;
30 miSaveMiniMap
: TMenuItem
;
31 miDeleteMap
: TMenuItem
;
36 miMenuEdit
: TMenuItem
;
43 miSelectAll
: TMenuItem
;
48 miMenuTools
: TMenuItem
;
49 miSnapToGrid
: TMenuItem
;
51 miSwitchGrid
: TMenuItem
;
52 miShowEdges
: TMenuItem
;
64 miMenuService
: TMenuItem
;
65 miCheckMap
: TMenuItem
;
66 miOptimmization
: TMenuItem
;
67 miMapPreview
: TMenuItem
;
70 miMenuSettings
: TMenuItem
;
71 miMapOptions
: TMenuItem
;
75 miMapTestSettings
: TMenuItem
;
77 miMenuHelp
: TMenuItem
;
79 // Скрытый пункт меню для Ctrl+Tab:
83 // Панель инструментов:
84 MainToolBar
: TToolBar
;
86 pLoadProgress
: TPanel
;
87 RenderPanel
: TOpenGLControl
;
88 tbNewMap
: TToolButton
;
89 tbOpenMap
: TToolButton
;
90 tbSaveMap
: TToolButton
;
91 tbOpenWadMap
: TToolButton
;
93 tbShowMap
: TToolButton
;
97 tbGridOn
: TToolButton
;
100 tbTestMap
: TToolButton
;
101 // Всплывающее меню для кнопки слоев:
103 miLayerP1
: TMenuItem
;
104 miLayerP2
: TMenuItem
;
105 miLayerP3
: TMenuItem
;
106 miLayerP4
: TMenuItem
;
107 miLayerP5
: TMenuItem
;
108 miLayerP6
: TMenuItem
;
109 miLayerP7
: TMenuItem
;
110 miLayerP8
: TMenuItem
;
111 miLayerP9
: TMenuItem
;
112 // Всплывающее меню для кнопки теста карты:
113 pmMapTest
: TPopupMenu
;
114 miMapTestPMSet
: TMenuItem
;
119 sbHorizontal
: TScrollBar
;
120 sbVertical
: TScrollBar
;
124 // Панель применения свойств:
125 PanelPropApply
: TPanel
;
126 bApplyProperty
: TButton
;
127 // Редактор свойств объектов:
128 vleObjectProperty
: TValueListEditor
;
130 // Панель объектов - вкладки:
132 pcObjects
: TPageControl
;
135 lbTextureList
: TListBox
;
136 // Панель настройки текстур:
137 PanelTextures
: TPanel
;
139 lTextureWidth
: TLabel
;
141 lTextureHeight
: TLabel
;
142 cbPreview
: TCheckBox
;
143 bbAddTexture
: TBitBtn
;
144 bbRemoveTexture
: TBitBtn
;
145 bClearTexture
: TButton
;
146 // Панель типов панелей:
147 PanelPanelType
: TPanel
;
148 lbPanelType
: TListBox
;
149 // Вкладка "Предметы":
151 lbItemList
: TListBox
;
154 // Вкладка "Монстры":
155 tsMonsters
: TTabSheet
;
156 lbMonsterList
: TListBox
;
157 rbMonsterLeft
: TRadioButton
;
158 rbMonsterRight
: TRadioButton
;
159 // Вкладка "Области":
161 lbAreasList
: TListBox
;
162 rbAreaLeft
: TRadioButton
;
163 rbAreaRight
: TRadioButton
;
164 // Вкладка "Триггеры":
165 tsTriggers
: TTabSheet
;
166 lbTriggersList
: TListBox
;
167 clbActivationType
: TCheckListBox
;
168 clbKeys
: TCheckListBox
;
171 Splitter1
: TSplitter
;
172 Splitter2
: TSplitter
;
173 StatusBar
: TStatusBar
;
175 // Специальные объекты:
176 ImageList
: TImageList
;
177 ilToolbar
: TImageList
;
178 OpenDialog
: TOpenDialog
;
179 SaveDialog
: TSaveDialog
;
180 selectall1
: TMenuItem
;
181 ColorDialog
: TColorDialog
;
183 procedure aAboutExecute(Sender
: TObject
);
184 procedure aCheckMapExecute(Sender
: TObject
);
185 procedure aMoveToFore(Sender
: TObject
);
186 procedure aMoveToBack(Sender
: TObject
);
187 procedure aCopyObjectExecute(Sender
: TObject
);
188 procedure aCutObjectExecute(Sender
: TObject
);
189 procedure aEditorOptionsExecute(Sender
: TObject
);
190 procedure aExitExecute(Sender
: TObject
);
191 procedure aMapOptionsExecute(Sender
: TObject
);
192 procedure aNewMapExecute(Sender
: TObject
);
193 procedure aOpenMapExecute(Sender
: TObject
);
194 procedure aOptimizeExecute(Sender
: TObject
);
195 procedure aPasteObjectExecute(Sender
: TObject
);
196 procedure aSelectAllExecute(Sender
: TObject
);
197 procedure aSaveMapExecute(Sender
: TObject
);
198 procedure aSaveMapAsExecute(Sender
: TObject
);
199 procedure aUndoExecute(Sender
: TObject
);
200 procedure aDeleteMap(Sender
: TObject
);
201 procedure bApplyPropertyClick(Sender
: TObject
);
202 procedure bbAddTextureClick(Sender
: TObject
);
203 procedure bbRemoveTextureClick(Sender
: TObject
);
204 procedure FormActivate(Sender
: TObject
);
205 procedure FormCloseQuery(Sender
: TObject
; var CanClose
: Boolean);
206 procedure FormCreate(Sender
: TObject
);
207 procedure FormDestroy(Sender
: TObject
);
208 procedure FormDropFiles(Sender
: TObject
; const FileNames
: array of String);
209 procedure FormKeyDown(Sender
: TObject
; var Key
: Word; Shift
: TShiftState
);
210 procedure FormResize(Sender
: TObject
);
211 procedure lbTextureListClick(Sender
: TObject
);
212 procedure lbTextureListDrawItem(Control
: TWinControl
; Index
: Integer;
213 ARect
: TRect
; State
: TOwnerDrawState
);
214 procedure miReopenMapClick(Sender
: TObject
);
215 procedure RenderPanelMouseDown(Sender
: TObject
; Button
: TMouseButton
; Shift
: TShiftState
; X
, Y
: Integer);
216 procedure RenderPanelMouseMove(Sender
: TObject
; Shift
: TShiftState
; X
, Y
: Integer);
217 procedure RenderPanelMouseUp(Sender
: TObject
; Button
: TMouseButton
; Shift
: TShiftState
; X
, Y
: Integer);
218 procedure RenderPanelPaint(Sender
: TObject
);
219 procedure RenderPanelResize(Sender
: TObject
);
220 procedure Splitter1Moved(Sender
: TObject
);
221 procedure vleObjectPropertyEditButtonClick(Sender
: TObject
);
222 procedure vleObjectPropertyApply(Sender
: TObject
);
223 procedure vleObjectPropertyGetPickList(Sender
: TObject
; const KeyName
: String; Values
: TStrings
);
224 procedure vleObjectPropertyKeyDown(Sender
: TObject
; var Key
: Word;
226 procedure tbGridOnClick(Sender
: TObject
);
227 procedure miMapPreviewClick(Sender
: TObject
);
228 procedure miLayer1Click(Sender
: TObject
);
229 procedure miLayer2Click(Sender
: TObject
);
230 procedure miLayer3Click(Sender
: TObject
);
231 procedure miLayer4Click(Sender
: TObject
);
232 procedure miLayer5Click(Sender
: TObject
);
233 procedure miLayer6Click(Sender
: TObject
);
234 procedure miLayer7Click(Sender
: TObject
);
235 procedure miLayer8Click(Sender
: TObject
);
236 procedure miLayer9Click(Sender
: TObject
);
237 procedure tbShowClick(Sender
: TObject
);
238 procedure miSnapToGridClick(Sender
: TObject
);
239 procedure miMiniMapClick(Sender
: TObject
);
240 procedure miSwitchGridClick(Sender
: TObject
);
241 procedure miShowEdgesClick(Sender
: TObject
);
242 procedure minexttabClick(Sender
: TObject
);
243 procedure miSaveMiniMapClick(Sender
: TObject
);
244 procedure bClearTextureClick(Sender
: TObject
);
245 procedure miPackMapClick(Sender
: TObject
);
246 procedure aRecentFileExecute(Sender
: TObject
);
247 procedure miMapTestSettingsClick(Sender
: TObject
);
248 procedure miTestMapClick(Sender
: TObject
);
249 procedure sbVerticalScroll(Sender
: TObject
; ScrollCode
: TScrollCode
;
250 var ScrollPos
: Integer);
251 procedure sbHorizontalScroll(Sender
: TObject
; ScrollCode
: TScrollCode
;
252 var ScrollPos
: Integer);
253 procedure miOpenWadMapClick(Sender
: TObject
);
254 procedure selectall1Click(Sender
: TObject
);
255 procedure Splitter1CanResize(Sender
: TObject
; var NewSize
: Integer;
256 var Accept
: Boolean);
257 procedure Splitter2CanResize(Sender
: TObject
; var NewSize
: Integer;
258 var Accept
: Boolean);
259 procedure vleObjectPropertyEnter(Sender
: TObject
);
260 procedure vleObjectPropertyExit(Sender
: TObject
);
261 procedure FormKeyUp(Sender
: TObject
; var Key
: Word;
265 procedure OnIdle(Sender
: TObject
; var Done
: Boolean);
267 procedure RefreshRecentMenu();
268 procedure OpenMapFile(FileName
: String);
269 function RenderMousePos(): TPoint
;
270 procedure RecountSelectedObjects();
276 LAYER_FOREGROUND
= 2;
284 TEST_MAP_NAME
= '$$$_TEST_$$$';
285 LANGUAGE_FILE_NAME
= '_Editor.txt';
296 DotStepOne
, DotStepTwo
: Byte;
298 DrawTexturePanel
: Boolean;
299 DrawPanelSize
: Boolean;
301 PreviewColor
: TColor
;
302 UseCheckerboard
: Boolean;
304 RecentCount
: Integer;
305 RecentFiles
: TStringList
;
306 slInvalidTextures
: TStringList
;
308 TestGameMode
: String;
310 TestLimScore
: String;
311 TestOptionsTwoPlayers
: Boolean;
312 TestOptionsTeamDamage
: Boolean;
313 TestOptionsAllowExit
: Boolean;
314 TestOptionsWeaponStay
: Boolean;
315 TestOptionsMonstersDM
: Boolean;
317 TestMapOnce
: Boolean;
319 LayerEnabled
: Array [LAYER_BACK
..LAYER_TRIGGERS
] of Boolean =
320 (True, True, True, True, True, True, True, True, True);
321 PreviewMode
: Byte = 0;
327 procedure OpenMap(FileName
: String; mapN
: String);
328 function AddTexture(aWAD
, aSection
, aTex
: String; silent
: Boolean): Boolean;
329 procedure RemoveSelectFromObjects();
330 procedure ChangeShownProperty(Name
: String; NewValue
: String);
335 f_options
, e_graphics
, e_log
, GL
, Math
,
336 f_mapoptions
, g_basic
, f_about
, f_mapoptimization
,
337 f_mapcheck
, f_addresource_texture
, g_textures
,
338 f_activationtype
, f_keys
,
339 MAPREADER
, f_selectmap
, f_savemap
, WADEDITOR
, WADSTRUCT
, MAPDEF
,
340 g_map
, f_saveminimap
, f_addresource
, CONFIG
, f_packmap
,
341 f_addresource_sound
, f_maptest
, f_choosetype
,
342 g_language
, f_selectlang
, ClipBrd
, g_resources
;
345 UNDO_DELETE_PANEL
= 1;
346 UNDO_DELETE_ITEM
= 2;
347 UNDO_DELETE_AREA
= 3;
348 UNDO_DELETE_MONSTER
= 4;
349 UNDO_DELETE_TRIGGER
= 5;
353 UNDO_ADD_MONSTER
= 9;
354 UNDO_ADD_TRIGGER
= 10;
355 UNDO_MOVE_PANEL
= 11;
358 UNDO_MOVE_MONSTER
= 14;
359 UNDO_MOVE_TRIGGER
= 15;
360 UNDO_RESIZE_PANEL
= 16;
361 UNDO_RESIZE_TRIGGER
= 17;
363 MOUSEACTION_NONE
= 0;
364 MOUSEACTION_DRAWPANEL
= 1;
365 MOUSEACTION_DRAWTRIGGER
= 2;
366 MOUSEACTION_MOVEOBJ
= 3;
367 MOUSEACTION_RESIZE
= 4;
368 MOUSEACTION_MOVEMAP
= 5;
369 MOUSEACTION_DRAWPRESS
= 6;
370 MOUSEACTION_NOACTION
= 7;
373 RESIZETYPE_VERTICAL
= 1;
374 RESIZETYPE_HORIZONTAL
= 2;
383 SELECTFLAG_TELEPORT
= 1;
385 SELECTFLAG_TEXTURE
= 3;
387 SELECTFLAG_MONSTER
= 5;
388 SELECTFLAG_SPAWNPOINT
= 6;
389 SELECTFLAG_SHOTPANEL
= 7;
390 SELECTFLAG_SELECTED
= 8;
392 RECENT_FILES_MENU_START
= 12;
394 CLIPBOARD_SIG
= 'DF:ED';
400 UNDO_DELETE_PANEL
: (Panel
: ^TPanel
);
401 UNDO_DELETE_ITEM
: (Item
: TItem
);
402 UNDO_DELETE_AREA
: (Area
: TArea
);
403 UNDO_DELETE_MONSTER
: (Monster
: TMonster
);
404 UNDO_DELETE_TRIGGER
: (Trigger
: TTrigger
);
409 UNDO_ADD_TRIGGER
: (AddID
: DWORD
);
414 UNDO_MOVE_TRIGGER
: (MoveID
: DWORD
; dX
, dY
: Integer);
416 UNDO_RESIZE_TRIGGER
: (ResizeID
: DWORD
; dW
, dH
: Integer);
423 OBJECT_PANEL
: (Panel
: ^TPanel
);
424 OBJECT_ITEM
: (Item
: TItem
);
425 OBJECT_AREA
: (Area
: TArea
);
426 OBJECT_MONSTER
: (Monster
: TMonster
);
427 OBJECT_TRIGGER
: (Trigger
: TTrigger
);
430 TCopyRecArray
= Array of TCopyRec
;
434 gDataLoaded
: Boolean = False;
435 ShowMap
: Boolean = False;
436 DrawRect
: PRect
= nil;
437 SnapToGrid
: Boolean = True;
439 MousePos
: Types
.TPoint
;
440 LastMovePoint
: Types
.TPoint
;
443 MouseLDownPos
: Types
.TPoint
;
444 MouseRDownPos
: Types
.TPoint
;
447 SelectFlag
: Byte = SELECTFLAG_NONE
;
448 MouseAction
: Byte = MOUSEACTION_NONE
;
449 ResizeType
: Byte = RESIZETYPE_NONE
;
450 ResizeDirection
: Byte = RESIZEDIR_NONE
;
452 DrawPressRect
: Boolean = False;
453 EditingProperties
: Boolean = False;
455 UndoBuffer
: Array of Array of TUndoRec
= nil;
460 //----------------------------------------
461 //Далее идут вспомогательные процедуры
462 //----------------------------------------
464 function NameToBool(Name
: String): Boolean;
466 if Name
= BoolNames
[True] then
472 function NameToDir(Name
: String): TDirection
;
474 if Name
= DirNames
[D_LEFT
] then
480 function NameToDirAdv(Name
: String): Byte;
482 if Name
= DirNamesAdv
[1] then
485 if Name
= DirNamesAdv
[2] then
488 if Name
= DirNamesAdv
[3] then
494 function ActivateToStr(ActivateType
: Byte): String;
498 if ByteBool(ACTIVATE_PLAYERCOLLIDE
and ActivateType
) then
499 Result
:= Result
+ '+PC';
500 if ByteBool(ACTIVATE_MONSTERCOLLIDE
and ActivateType
) then
501 Result
:= Result
+ '+MC';
502 if ByteBool(ACTIVATE_PLAYERPRESS
and ActivateType
) then
503 Result
:= Result
+ '+PP';
504 if ByteBool(ACTIVATE_MONSTERPRESS
and ActivateType
) then
505 Result
:= Result
+ '+MP';
506 if ByteBool(ACTIVATE_SHOT
and ActivateType
) then
507 Result
:= Result
+ '+SH';
508 if ByteBool(ACTIVATE_NOMONSTER
and ActivateType
) then
509 Result
:= Result
+ '+NM';
511 if (Result
<> '') and (Result
[1] = '+') then
512 Delete(Result
, 1, 1);
515 function StrToActivate(Str
: String): Byte;
519 if Pos('PC', Str
) > 0 then
520 Result
:= ACTIVATE_PLAYERCOLLIDE
;
521 if Pos('MC', Str
) > 0 then
522 Result
:= Result
or ACTIVATE_MONSTERCOLLIDE
;
523 if Pos('PP', Str
) > 0 then
524 Result
:= Result
or ACTIVATE_PLAYERPRESS
;
525 if Pos('MP', Str
) > 0 then
526 Result
:= Result
or ACTIVATE_MONSTERPRESS
;
527 if Pos('SH', Str
) > 0 then
528 Result
:= Result
or ACTIVATE_SHOT
;
529 if Pos('NM', Str
) > 0 then
530 Result
:= Result
or ACTIVATE_NOMONSTER
;
533 function KeyToStr(Key
: Byte): String;
537 if ByteBool(KEY_RED
and Key
) then
538 Result
:= Result
+ '+RK';
539 if ByteBool(KEY_GREEN
and Key
) then
540 Result
:= Result
+ '+GK';
541 if ByteBool(KEY_BLUE
and Key
) then
542 Result
:= Result
+ '+BK';
543 if ByteBool(KEY_REDTEAM
and Key
) then
544 Result
:= Result
+ '+RT';
545 if ByteBool(KEY_BLUETEAM
and Key
) then
546 Result
:= Result
+ '+BT';
548 if (Result
<> '') and (Result
[1] = '+') then
549 Delete(Result
, 1, 1);
552 function StrToKey(Str
: String): Byte;
556 if Pos('RK', Str
) > 0 then
558 if Pos('GK', Str
) > 0 then
559 Result
:= Result
or KEY_GREEN
;
560 if Pos('BK', Str
) > 0 then
561 Result
:= Result
or KEY_BLUE
;
562 if Pos('RT', Str
) > 0 then
563 Result
:= Result
or KEY_REDTEAM
;
564 if Pos('BT', Str
) > 0 then
565 Result
:= Result
or KEY_BLUETEAM
;
568 function EffectToStr(Effect
: Byte): String;
570 if Effect
in [EFFECT_TELEPORT
..EFFECT_FIRE
] then
571 Result
:= EffectNames
[Effect
]
573 Result
:= EffectNames
[EFFECT_NONE
];
576 function StrToEffect(Str
: String): Byte;
580 Result
:= EFFECT_NONE
;
581 for i
:= EFFECT_TELEPORT
to EFFECT_FIRE
do
582 if EffectNames
[i
] = Str
then
589 function MonsterToStr(MonType
: Byte): String;
591 if MonType
in [MONSTER_DEMON
..MONSTER_MAN
] then
592 Result
:= MonsterNames
[MonType
]
594 Result
:= MonsterNames
[MONSTER_ZOMBY
];
597 function StrToMonster(Str
: String): Byte;
601 Result
:= MONSTER_ZOMBY
;
602 for i
:= MONSTER_DEMON
to MONSTER_MAN
do
603 if MonsterNames
[i
] = Str
then
610 function ItemToStr(ItemType
: Byte): String;
612 if ItemType
in [ITEM_MEDKIT_SMALL
..ITEM_MAX
] then
613 Result
:= ItemNames
[ItemType
]
615 Result
:= ItemNames
[ITEM_AMMO_BULLETS
];
618 function StrToItem(Str
: String): Byte;
622 Result
:= ITEM_AMMO_BULLETS
;
623 for i
:= ITEM_MEDKIT_SMALL
to ITEM_MAX
do
624 if ItemNames
[i
] = Str
then
631 function ShotToStr(ShotType
: Byte): String;
633 if ShotType
in [TRIGGER_SHOT_PISTOL
..TRIGGER_SHOT_MAX
] then
634 Result
:= ShotNames
[ShotType
]
636 Result
:= ShotNames
[TRIGGER_SHOT_PISTOL
];
639 function StrToShot(Str
: String): Byte;
643 Result
:= TRIGGER_SHOT_PISTOL
;
644 for i
:= TRIGGER_SHOT_PISTOL
to TRIGGER_SHOT_MAX
do
645 if ShotNames
[i
] = Str
then
652 function SelectedObjectCount(): Word;
658 if SelectedObjects
= nil then
661 for a
:= 0 to High(SelectedObjects
) do
662 if SelectedObjects
[a
].Live
then
663 Result
:= Result
+ 1;
666 function GetFirstSelected(): Integer;
672 if SelectedObjects
= nil then
675 for a
:= 0 to High(SelectedObjects
) do
676 if SelectedObjects
[a
].Live
then
683 function Normalize16(x
: Integer): Integer;
685 Result
:= (x
div 16) * 16;
688 procedure MoveMap(X
, Y
: Integer);
690 rx
, ry
, ScaleSz
: Integer;
692 with MainForm
.RenderPanel
do
694 ScaleSz
:= 16 div Scale
;
695 // Размер видимой части карты:
696 rx
:= min(Normalize16(Width
), Normalize16(gMapInfo
.Width
)) div 2;
697 ry
:= min(Normalize16(Height
), Normalize16(gMapInfo
.Height
)) div 2;
698 // Место клика на мини-карте:
699 MapOffset
.X
:= X
- (Width
-max(gMapInfo
.Width
div ScaleSz
, 1)-1);
700 MapOffset
.Y
:= Y
- 1;
701 // Это же место на "большой" карте:
702 MapOffset
.X
:= MapOffset
.X
* ScaleSz
;
703 MapOffset
.Y
:= MapOffset
.Y
* ScaleSz
;
704 // Левый верхний угол новой видимой части карты:
705 MapOffset
.X
:= MapOffset
.X
- rx
;
706 MapOffset
.Y
:= MapOffset
.Y
- ry
;
708 if MapOffset
.X
< 0 then
710 if MapOffset
.Y
< 0 then
712 if MapOffset
.X
> MainForm
.sbHorizontal
.Max
then
713 MapOffset
.X
:= MainForm
.sbHorizontal
.Max
;
714 if MapOffset
.Y
> MainForm
.sbVertical
.Max
then
715 MapOffset
.Y
:= MainForm
.sbVertical
.Max
;
717 MapOffset
.X
:= Normalize16(MapOffset
.X
);
718 MapOffset
.Y
:= Normalize16(MapOffset
.Y
);
721 MainForm
.sbHorizontal
.Position
:= MapOffset
.X
;
722 MainForm
.sbVertical
.Position
:= MapOffset
.Y
;
724 MapOffset
.X
:= -MapOffset
.X
;
725 MapOffset
.Y
:= -MapOffset
.Y
;
730 function IsTexturedPanel(PanelType
: Word): Boolean;
732 Result
:= WordBool(PanelType
and (PANEL_WALL
or PANEL_BACK
or PANEL_FORE
or
733 PANEL_STEP
or PANEL_OPENDOOR
or PANEL_CLOSEDOOR
or
734 PANEL_WATER
or PANEL_ACID1
or PANEL_ACID2
));
737 procedure FillProperty();
742 MainForm
.vleObjectProperty
.Strings
.Clear();
743 MainForm
.RecountSelectedObjects();
745 // Отображаем свойства если выделен только один объект:
746 if SelectedObjectCount() <> 1 then
749 _id
:= GetFirstSelected();
750 if not SelectedObjects
[_id
].Live
then
753 with MainForm
.vleObjectProperty
do
754 with ItemProps
[InsertRow(_lc
[I_PROP_ID
], IntToStr(SelectedObjects
[_id
].ID
), True)] do
756 EditStyle
:= esSimple
;
760 case SelectedObjects
[0].ObjectType
of
763 with MainForm
.vleObjectProperty
,
764 gPanels
[SelectedObjects
[_id
].ID
] do
766 with ItemProps
[InsertRow(_lc
[I_PROP_X
], IntToStr(X
), True)] do
768 EditStyle
:= esSimple
;
772 with ItemProps
[InsertRow(_lc
[I_PROP_Y
], IntToStr(Y
), True)] do
774 EditStyle
:= esSimple
;
778 with ItemProps
[InsertRow(_lc
[I_PROP_WIDTH
], IntToStr(Width
), True)] do
780 EditStyle
:= esSimple
;
784 with ItemProps
[InsertRow(_lc
[I_PROP_HEIGHT
], IntToStr(Height
), True)] do
786 EditStyle
:= esSimple
;
790 with ItemProps
[InsertRow(_lc
[I_PROP_PANEL_TYPE
], GetPanelName(PanelType
), True)] do
792 EditStyle
:= esEllipsis
;
796 if IsTexturedPanel(PanelType
) then
797 begin // Может быть текстура
798 with ItemProps
[InsertRow(_lc
[I_PROP_PANEL_TEX
], TextureName
, True)] do
800 EditStyle
:= esEllipsis
;
804 if TextureName
<> '' then
805 begin // Есть текстура
806 with ItemProps
[InsertRow(_lc
[I_PROP_PANEL_ALPHA
], IntToStr(Alpha
), True)] do
808 EditStyle
:= esSimple
;
812 with ItemProps
[InsertRow(_lc
[I_PROP_PANEL_BLEND
], BoolNames
[Blending
], True)] do
814 EditStyle
:= esPickList
;
824 with MainForm
.vleObjectProperty
,
825 gItems
[SelectedObjects
[_id
].ID
] do
827 with ItemProps
[InsertRow(_lc
[I_PROP_X
], IntToStr(X
), True)] do
829 EditStyle
:= esSimple
;
833 with ItemProps
[InsertRow(_lc
[I_PROP_Y
], IntToStr(Y
), True)] do
835 EditStyle
:= esSimple
;
839 with ItemProps
[InsertRow(_lc
[I_PROP_DM_ONLY
], BoolNames
[OnlyDM
], True)] do
841 EditStyle
:= esPickList
;
845 with ItemProps
[InsertRow(_lc
[I_PROP_ITEM_FALLS
], BoolNames
[Fall
], True)] do
847 EditStyle
:= esPickList
;
855 with MainForm
.vleObjectProperty
,
856 gMonsters
[SelectedObjects
[_id
].ID
] do
858 with ItemProps
[InsertRow(_lc
[I_PROP_X
], IntToStr(X
), True)] do
860 EditStyle
:= esSimple
;
864 with ItemProps
[InsertRow(_lc
[I_PROP_Y
], IntToStr(Y
), True)] do
866 EditStyle
:= esSimple
;
870 with ItemProps
[InsertRow(_lc
[I_PROP_DIRECTION
], DirNames
[Direction
], True)] do
872 EditStyle
:= esPickList
;
880 with MainForm
.vleObjectProperty
,
881 gAreas
[SelectedObjects
[_id
].ID
] do
883 with ItemProps
[InsertRow(_lc
[I_PROP_X
], IntToStr(X
), True)] do
885 EditStyle
:= esSimple
;
889 with ItemProps
[InsertRow(_lc
[I_PROP_Y
], IntToStr(Y
), True)] do
891 EditStyle
:= esSimple
;
895 with ItemProps
[InsertRow(_lc
[I_PROP_DIRECTION
], DirNames
[Direction
], True)] do
897 EditStyle
:= esPickList
;
905 with MainForm
.vleObjectProperty
,
906 gTriggers
[SelectedObjects
[_id
].ID
] do
908 with ItemProps
[InsertRow(_lc
[I_PROP_TR_TYPE
], GetTriggerName(TriggerType
), True)] do
910 EditStyle
:= esSimple
;
914 with ItemProps
[InsertRow(_lc
[I_PROP_X
], IntToStr(X
), True)] do
916 EditStyle
:= esSimple
;
920 with ItemProps
[InsertRow(_lc
[I_PROP_Y
], IntToStr(Y
), True)] do
922 EditStyle
:= esSimple
;
926 with ItemProps
[InsertRow(_lc
[I_PROP_WIDTH
], IntToStr(Width
), True)] do
928 EditStyle
:= esSimple
;
932 with ItemProps
[InsertRow(_lc
[I_PROP_HEIGHT
], IntToStr(Height
), True)] do
934 EditStyle
:= esSimple
;
938 with ItemProps
[InsertRow(_lc
[I_PROP_TR_ENABLED
], BoolNames
[Enabled
], True)] do
940 EditStyle
:= esPickList
;
944 with ItemProps
[InsertRow(_lc
[I_PROP_TR_TEXTURE_PANEL
], IntToStr(TexturePanel
), True)] do
946 EditStyle
:= esEllipsis
;
950 with ItemProps
[InsertRow(_lc
[I_PROP_TR_ACTIVATION
], ActivateToStr(ActivateType
), True)] do
952 EditStyle
:= esEllipsis
;
956 with ItemProps
[InsertRow(_lc
[I_PROP_TR_KEYS
], KeyToStr(Key
), True)] do
958 EditStyle
:= esEllipsis
;
965 str
:= win2utf(Data
.MapName
);
966 with ItemProps
[InsertRow(_lc
[I_PROP_TR_NEXT_MAP
], str
, True)] do
968 EditStyle
:= esEllipsis
;
975 with ItemProps
[InsertRow(_lc
[I_PROP_TR_TELEPORT_TO
], Format('(%d:%d)', [Data
.TargetPoint
.X
, Data
.TargetPoint
.Y
]), True)] do
977 EditStyle
:= esEllipsis
;
981 with ItemProps
[InsertRow(_lc
[I_PROP_TR_D2D
], BoolNames
[Data
.d2d_teleport
], True)] do
983 EditStyle
:= esPickList
;
987 with ItemProps
[InsertRow(_lc
[I_PROP_TR_TELEPORT_SILENT
], BoolNames
[Data
.silent_teleport
], True)] do
989 EditStyle
:= esPickList
;
993 with ItemProps
[InsertRow(_lc
[I_PROP_TR_TELEPORT_DIR
], DirNamesAdv
[Data
.TlpDir
], True)] do
995 EditStyle
:= esPickList
;
1000 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
,
1001 TRIGGER_DOOR
, TRIGGER_DOOR5
:
1003 with ItemProps
[InsertRow(_lc
[I_PROP_TR_DOOR_PANEL
], IntToStr(Data
.PanelID
), True)] do
1005 EditStyle
:= esEllipsis
;
1009 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SILENT
], BoolNames
[Data
.NoSound
], True)] do
1011 EditStyle
:= esPickList
;
1015 with ItemProps
[InsertRow(_lc
[I_PROP_TR_D2D
], BoolNames
[Data
.d2d_doors
], True)] do
1017 EditStyle
:= esPickList
;
1022 TRIGGER_CLOSETRAP
, TRIGGER_TRAP
:
1024 with ItemProps
[InsertRow(_lc
[I_PROP_TR_TRAP_PANEL
], IntToStr(Data
.PanelID
), True)] do
1026 EditStyle
:= esEllipsis
;
1030 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SILENT
], BoolNames
[Data
.NoSound
], True)] do
1032 EditStyle
:= esPickList
;
1036 with ItemProps
[InsertRow(_lc
[I_PROP_TR_D2D
], BoolNames
[Data
.d2d_doors
], True)] do
1038 EditStyle
:= esPickList
;
1043 TRIGGER_PRESS
, TRIGGER_ON
, TRIGGER_OFF
,
1046 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EX_AREA
],
1047 Format('(%d:%d %d:%d)', [Data
.tX
, Data
.tY
, Data
.tWidth
, Data
.tHeight
]), True)] do
1049 EditStyle
:= esEllipsis
;
1053 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EX_DELAY
], IntToStr(Data
.Wait
), True)] do
1055 EditStyle
:= esSimple
;
1059 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EX_COUNT
], IntToStr(Data
.Count
), True)] do
1061 EditStyle
:= esSimple
;
1065 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EX_MONSTER
], IntToStr(Data
.MonsterID
-1), True)] do
1067 EditStyle
:= esEllipsis
;
1071 if TriggerType
= TRIGGER_PRESS
then
1072 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EX_RANDOM
], BoolNames
[Data
.ExtRandom
], True)] do
1074 EditStyle
:= esPickList
;
1082 TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
:
1084 with ItemProps
[InsertRow(_lc
[I_PROP_TR_LIFT_PANEL
], IntToStr(Data
.PanelID
), True)] do
1086 EditStyle
:= esEllipsis
;
1090 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SILENT
], BoolNames
[Data
.NoSound
], True)] do
1092 EditStyle
:= esPickList
;
1096 with ItemProps
[InsertRow(_lc
[I_PROP_TR_D2D
], BoolNames
[Data
.d2d_doors
], True)] do
1098 EditStyle
:= esPickList
;
1105 with ItemProps
[InsertRow(_lc
[I_PROP_TR_TEXTURE_ONCE
], BoolNames
[Data
.ActivateOnce
], True)] do
1107 EditStyle
:= esPickList
;
1111 with ItemProps
[InsertRow(_lc
[I_PROP_TR_TEXTURE_ANIM_ONCE
], BoolNames
[Data
.AnimOnce
], True)] do
1113 EditStyle
:= esPickList
;
1120 str
:= win2utf(Data
.SoundName
);
1121 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SOUND_NAME
], str
, True)] do
1123 EditStyle
:= esEllipsis
;
1127 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SOUND_VOLUME
], IntToStr(Data
.Volume
), True)] do
1129 EditStyle
:= esSimple
;
1133 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SOUND_PAN
], IntToStr(Data
.Pan
), True)] do
1135 EditStyle
:= esSimple
;
1139 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SOUND_COUNT
], IntToStr(Data
.PlayCount
), True)] do
1141 EditStyle
:= esSimple
;
1145 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SOUND_LOCAL
], BoolNames
[Data
.Local
], True)] do
1147 EditStyle
:= esPickList
;
1151 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SOUND_SWITCH
], BoolNames
[Data
.SoundSwitch
], True)] do
1153 EditStyle
:= esPickList
;
1158 TRIGGER_SPAWNMONSTER
:
1160 with ItemProps
[InsertRow(_lc
[I_PROP_TR_MONSTER_TYPE
], MonsterToStr(Data
.MonType
), True)] do
1162 EditStyle
:= esEllipsis
;
1166 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SPAWN_TO
],
1167 Format('(%d:%d)', [Data
.MonPos
.X
, Data
.MonPos
.Y
]), True)] do
1169 EditStyle
:= esEllipsis
;
1173 with ItemProps
[InsertRow(_lc
[I_PROP_DIRECTION
], DirNames
[TDirection(Data
.MonDir
)], True)] do
1175 EditStyle
:= esPickList
;
1179 with ItemProps
[InsertRow(_lc
[I_PROP_TR_HEALTH
], IntToStr(Data
.MonHealth
), True)] do
1181 EditStyle
:= esSimple
;
1185 with ItemProps
[InsertRow(_lc
[I_PROP_TR_MONSTER_ACTIVE
], BoolNames
[Data
.MonActive
], True)] do
1187 EditStyle
:= esPickList
;
1191 with ItemProps
[InsertRow(_lc
[I_PROP_TR_COUNT
], IntToStr(Data
.MonCount
), True)] do
1193 EditStyle
:= esSimple
;
1197 with ItemProps
[InsertRow(_lc
[I_PROP_TR_FX_TYPE
], EffectToStr(Data
.MonEffect
), True)] do
1199 EditStyle
:= esEllipsis
;
1203 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SPAWN_MAX
], IntToStr(Data
.MonMax
), True)] do
1205 EditStyle
:= esSimple
;
1209 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SPAWN_DELAY
], IntToStr(Data
.MonDelay
), True)] do
1211 EditStyle
:= esSimple
;
1215 case Data
.MonBehav
of
1216 1: str
:= _lc
[I_PROP_TR_MONSTER_BEHAVIOUR_1
];
1217 2: str
:= _lc
[I_PROP_TR_MONSTER_BEHAVIOUR_2
];
1218 3: str
:= _lc
[I_PROP_TR_MONSTER_BEHAVIOUR_3
];
1219 4: str
:= _lc
[I_PROP_TR_MONSTER_BEHAVIOUR_4
];
1220 5: str
:= _lc
[I_PROP_TR_MONSTER_BEHAVIOUR_5
];
1221 else str
:= _lc
[I_PROP_TR_MONSTER_BEHAVIOUR_0
];
1223 with ItemProps
[InsertRow(_lc
[I_PROP_TR_MONSTER_BEHAVIOUR
], str
, True)] do
1225 EditStyle
:= esPickList
;
1232 with ItemProps
[InsertRow(_lc
[I_PROP_TR_ITEM_TYPE
], ItemToStr(Data
.ItemType
), True)] do
1234 EditStyle
:= esEllipsis
;
1238 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SPAWN_TO
],
1239 Format('(%d:%d)', [Data
.ItemPos
.X
, Data
.ItemPos
.Y
]), True)] do
1241 EditStyle
:= esEllipsis
;
1245 with ItemProps
[InsertRow(_lc
[I_PROP_DM_ONLY
], BoolNames
[Data
.ItemOnlyDM
], True)] do
1247 EditStyle
:= esPickList
;
1251 with ItemProps
[InsertRow(_lc
[I_PROP_ITEM_FALLS
], BoolNames
[Data
.ItemFalls
], True)] do
1253 EditStyle
:= esPickList
;
1257 with ItemProps
[InsertRow(_lc
[I_PROP_TR_COUNT
], IntToStr(Data
.ItemCount
), True)] do
1259 EditStyle
:= esSimple
;
1263 with ItemProps
[InsertRow(_lc
[I_PROP_TR_FX_TYPE
], EffectToStr(Data
.ItemEffect
), True)] do
1265 EditStyle
:= esEllipsis
;
1269 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SPAWN_MAX
], IntToStr(Data
.ItemMax
), True)] do
1271 EditStyle
:= esSimple
;
1275 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SPAWN_DELAY
], IntToStr(Data
.ItemDelay
), True)] do
1277 EditStyle
:= esSimple
;
1284 str
:= win2utf(Data
.MusicName
);
1285 with ItemProps
[InsertRow(_lc
[I_PROP_TR_MUSIC_NAME
], str
, True)] do
1287 EditStyle
:= esEllipsis
;
1291 if Data
.MusicAction
= 1 then
1292 str
:= _lc
[I_PROP_TR_MUSIC_ON
]
1294 str
:= _lc
[I_PROP_TR_MUSIC_OFF
];
1296 with ItemProps
[InsertRow(_lc
[I_PROP_TR_MUSIC_ACT
], str
, True)] do
1298 EditStyle
:= esPickList
;
1305 with ItemProps
[InsertRow(_lc
[I_PROP_TR_PUSH_ANGLE
], IntToStr(Data
.PushAngle
), True)] do
1307 EditStyle
:= esSimple
;
1310 with ItemProps
[InsertRow(_lc
[I_PROP_TR_PUSH_FORCE
], IntToStr(Data
.PushForce
), True)] do
1312 EditStyle
:= esSimple
;
1315 with ItemProps
[InsertRow(_lc
[I_PROP_TR_PUSH_RESET
], BoolNames
[Data
.ResetVel
], True)] do
1317 EditStyle
:= esPickList
;
1324 case Data
.ScoreAction
of
1325 1: str
:= _lc
[I_PROP_TR_SCORE_ACT_1
];
1326 2: str
:= _lc
[I_PROP_TR_SCORE_ACT_2
];
1327 3: str
:= _lc
[I_PROP_TR_SCORE_ACT_3
];
1328 else str
:= _lc
[I_PROP_TR_SCORE_ACT_0
];
1330 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SCORE_ACT
], str
, True)] do
1332 EditStyle
:= esPickList
;
1335 with ItemProps
[InsertRow(_lc
[I_PROP_TR_COUNT
], IntToStr(Data
.ScoreCount
), True)] do
1337 EditStyle
:= esSimple
;
1340 case Data
.ScoreTeam
of
1341 1: str
:= _lc
[I_PROP_TR_SCORE_TEAM_1
];
1342 2: str
:= _lc
[I_PROP_TR_SCORE_TEAM_2
];
1343 3: str
:= _lc
[I_PROP_TR_SCORE_TEAM_3
];
1344 else str
:= _lc
[I_PROP_TR_SCORE_TEAM_0
];
1346 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SCORE_TEAM
], str
, True)] do
1348 EditStyle
:= esPickList
;
1351 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SCORE_CON
], BoolNames
[Data
.ScoreCon
], True)] do
1353 EditStyle
:= esPickList
;
1356 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SCORE_MSG
], BoolNames
[Data
.ScoreMsg
], True)] do
1358 EditStyle
:= esPickList
;
1365 case Data
.MessageKind
of
1366 1: str
:= _lc
[I_PROP_TR_MESSAGE_KIND_1
];
1367 else str
:= _lc
[I_PROP_TR_MESSAGE_KIND_0
];
1369 with ItemProps
[InsertRow(_lc
[I_PROP_TR_MESSAGE_KIND
], str
, True)] do
1371 EditStyle
:= esPickList
;
1374 case Data
.MessageSendTo
of
1375 1: str
:= _lc
[I_PROP_TR_MESSAGE_TO_1
];
1376 2: str
:= _lc
[I_PROP_TR_MESSAGE_TO_2
];
1377 3: str
:= _lc
[I_PROP_TR_MESSAGE_TO_3
];
1378 4: str
:= _lc
[I_PROP_TR_MESSAGE_TO_4
];
1379 5: str
:= _lc
[I_PROP_TR_MESSAGE_TO_5
];
1380 else str
:= _lc
[I_PROP_TR_MESSAGE_TO_0
];
1382 with ItemProps
[InsertRow(_lc
[I_PROP_TR_MESSAGE_TO
], str
, True)] do
1384 EditStyle
:= esPickList
;
1387 str
:= win2utf(Data
.MessageText
);
1388 with ItemProps
[InsertRow(_lc
[I_PROP_TR_MESSAGE_TEXT
], str
, True)] do
1390 EditStyle
:= esSimple
;
1393 with ItemProps
[InsertRow(_lc
[I_PROP_TR_MESSAGE_TIME
], IntToStr(Data
.MessageTime
), True)] do
1395 EditStyle
:= esSimple
;
1402 with ItemProps
[InsertRow(_lc
[I_PROP_TR_DAMAGE_VALUE
], IntToStr(Data
.DamageValue
), True)] do
1404 EditStyle
:= esSimple
;
1407 with ItemProps
[InsertRow(_lc
[I_PROP_TR_INTERVAL
], IntToStr(Data
.DamageInterval
), True)] do
1409 EditStyle
:= esSimple
;
1416 with ItemProps
[InsertRow(_lc
[I_PROP_TR_HEALTH
], IntToStr(Data
.HealValue
), True)] do
1418 EditStyle
:= esSimple
;
1421 with ItemProps
[InsertRow(_lc
[I_PROP_TR_INTERVAL
], IntToStr(Data
.HealInterval
), True)] do
1423 EditStyle
:= esSimple
;
1426 with ItemProps
[InsertRow(_lc
[I_PROP_TR_HEALTH_MAX
], BoolNames
[Data
.HealMax
], True)] do
1428 EditStyle
:= esPickList
;
1431 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SILENT
], BoolNames
[Data
.HealSilent
], True)] do
1433 EditStyle
:= esPickList
;
1440 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SHOT_TYPE
], ShotToStr(Data
.ShotType
), True)] do
1442 EditStyle
:= esEllipsis
;
1446 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SHOT_SOUND
], BoolNames
[Data
.ShotSound
], True)] do
1448 EditStyle
:= esPickList
;
1452 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SHOT_PANEL
], IntToStr(Data
.ShotPanelID
), True)] do
1454 EditStyle
:= esEllipsis
;
1458 case Data
.ShotTarget
of
1459 1: str
:= _lc
[I_PROP_TR_SHOT_TO_1
];
1460 2: str
:= _lc
[I_PROP_TR_SHOT_TO_2
];
1461 3: str
:= _lc
[I_PROP_TR_SHOT_TO_3
];
1462 4: str
:= _lc
[I_PROP_TR_SHOT_TO_4
];
1463 5: str
:= _lc
[I_PROP_TR_SHOT_TO_5
];
1464 6: str
:= _lc
[I_PROP_TR_SHOT_TO_6
];
1465 else str
:= _lc
[I_PROP_TR_SHOT_TO_0
];
1467 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SHOT_TO
], str
, True)] do
1469 EditStyle
:= esPickList
;
1473 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SHOT_SIGHT
], IntToStr(Data
.ShotIntSight
), True)] do
1475 EditStyle
:= esSimple
;
1479 case Data
.ShotAim
of
1480 1: str
:= _lc
[I_PROP_TR_SHOT_AIM_1
];
1481 2: str
:= _lc
[I_PROP_TR_SHOT_AIM_2
];
1482 3: str
:= _lc
[I_PROP_TR_SHOT_AIM_3
];
1483 else str
:= _lc
[I_PROP_TR_SHOT_AIM_0
];
1485 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SHOT_AIM
], str
, True)] do
1487 EditStyle
:= esPickList
;
1491 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SPAWN_TO
],
1492 Format('(%d:%d)', [Data
.ShotPos
.X
, Data
.ShotPos
.Y
]), True)] do
1494 EditStyle
:= esEllipsis
;
1498 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SHOT_ANGLE
], IntToStr(Data
.ShotAngle
), True)] do
1500 EditStyle
:= esSimple
;
1504 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EX_DELAY
], IntToStr(Data
.ShotWait
), True)] do
1506 EditStyle
:= esSimple
;
1510 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SHOT_ACC
], IntToStr(Data
.ShotAccuracy
), True)] do
1512 EditStyle
:= esSimple
;
1516 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SHOT_AMMO
], IntToStr(Data
.ShotAmmo
), True)] do
1518 EditStyle
:= esSimple
;
1522 with ItemProps
[InsertRow(_lc
[I_PROP_TR_SHOT_RELOAD
], IntToStr(Data
.ShotIntReload
), True)] do
1524 EditStyle
:= esSimple
;
1531 with ItemProps
[InsertRow(_lc
[I_PROP_TR_COUNT
], IntToStr(Data
.FXCount
), True)] do
1533 EditStyle
:= esSimple
;
1537 if Data
.FXType
= 0 then
1538 str
:= _lc
[I_PROP_TR_EFFECT_PARTICLE
]
1540 str
:= _lc
[I_PROP_TR_EFFECT_ANIMATION
];
1541 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EFFECT_TYPE
], str
, True)] do
1543 EditStyle
:= esEllipsis
;
1548 if Data
.FXType
= 0 then
1549 case Data
.FXSubType
of
1550 TRIGGER_EFFECT_SLIQUID
:
1551 str
:= _lc
[I_PROP_TR_EFFECT_SLIQUID
];
1552 TRIGGER_EFFECT_LLIQUID
:
1553 str
:= _lc
[I_PROP_TR_EFFECT_LLIQUID
];
1554 TRIGGER_EFFECT_DLIQUID
:
1555 str
:= _lc
[I_PROP_TR_EFFECT_DLIQUID
];
1556 TRIGGER_EFFECT_BLOOD
:
1557 str
:= _lc
[I_PROP_TR_EFFECT_BLOOD
];
1558 TRIGGER_EFFECT_SPARK
:
1559 str
:= _lc
[I_PROP_TR_EFFECT_SPARK
];
1560 TRIGGER_EFFECT_BUBBLE
:
1561 str
:= _lc
[I_PROP_TR_EFFECT_BUBBLE
];
1563 if Data
.FXType
= 1 then
1565 if (Data
.FXSubType
= 0) or (Data
.FXSubType
> EFFECT_FIRE
) then
1566 Data
.FXSubType
:= EFFECT_TELEPORT
;
1567 str
:= EffectToStr(Data
.FXSubType
);
1569 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EFFECT_SUBTYPE
], str
, True)] do
1571 EditStyle
:= esEllipsis
;
1575 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EFFECT_COLOR
], IntToStr(Data
.FXColorR
or (Data
.FXColorG
shl 8) or (Data
.FXColorB
shl 16)), True)] do
1577 EditStyle
:= esEllipsis
;
1581 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EFFECT_CENTER
], BoolNames
[Data
.FXPos
= 0], True)] do
1583 EditStyle
:= esPickList
;
1587 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EX_DELAY
], IntToStr(Data
.FXWait
), True)] do
1589 EditStyle
:= esSimple
;
1593 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EFFECT_VELX
], IntToStr(Data
.FXVelX
), True)] do
1595 EditStyle
:= esSimple
;
1599 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EFFECT_VELY
], IntToStr(Data
.FXVelY
), True)] do
1601 EditStyle
:= esSimple
;
1605 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EFFECT_SPL
], IntToStr(Data
.FXSpreadL
), True)] do
1607 EditStyle
:= esSimple
;
1611 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EFFECT_SPR
], IntToStr(Data
.FXSpreadR
), True)] do
1613 EditStyle
:= esSimple
;
1617 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EFFECT_SPU
], IntToStr(Data
.FXSpreadU
), True)] do
1619 EditStyle
:= esSimple
;
1623 with ItemProps
[InsertRow(_lc
[I_PROP_TR_EFFECT_SPD
], IntToStr(Data
.FXSpreadD
), True)] do
1625 EditStyle
:= esSimple
;
1629 end; //case TriggerType
1631 end; // OBJECT_TRIGGER:
1635 procedure ChangeShownProperty(Name
: String; NewValue
: String);
1639 if SelectedObjectCount() <> 1 then
1641 if not SelectedObjects
[GetFirstSelected()].Live
then
1644 // Есть ли такой ключ:
1645 if MainForm
.vleObjectProperty
.FindRow(Name
, row
) then
1647 MainForm
.vleObjectProperty
.Values
[Name
] := NewValue
;
1651 procedure SelectObject(fObjectType
: Byte; fID
: DWORD
; Multi
: Boolean);
1660 // Уже выделен - убираем:
1661 if SelectedObjects
<> nil then
1662 for a
:= 0 to High(SelectedObjects
) do
1663 with SelectedObjects
[a
] do
1664 if Live
and (ID
= fID
) and
1665 (ObjectType
= fObjectType
) then
1674 SetLength(SelectedObjects
, Length(SelectedObjects
)+1);
1676 with SelectedObjects
[High(SelectedObjects
)] do
1678 ObjectType
:= fObjectType
;
1685 SetLength(SelectedObjects
, 1);
1687 with SelectedObjects
[0] do
1689 ObjectType
:= fObjectType
;
1695 MainForm
.miCopy
.Enabled
:= True;
1696 MainForm
.miCut
.Enabled
:= True;
1698 if fObjectType
= OBJECT_PANEL
then
1700 MainForm
.miToFore
.Enabled
:= True;
1701 MainForm
.miToBack
.Enabled
:= True;
1705 procedure RemoveSelectFromObjects();
1707 SelectedObjects
:= nil;
1708 DrawPressRect
:= False;
1709 MouseLDown
:= False;
1710 MouseRDown
:= False;
1711 MouseAction
:= MOUSEACTION_NONE
;
1712 SelectFlag
:= SELECTFLAG_NONE
;
1713 ResizeType
:= RESIZETYPE_NONE
;
1714 ResizeDirection
:= RESIZEDIR_NONE
;
1716 MainForm
.vleObjectProperty
.Strings
.Clear();
1718 MainForm
.miCopy
.Enabled
:= False;
1719 MainForm
.miCut
.Enabled
:= False;
1720 MainForm
.miToFore
.Enabled
:= False;
1721 MainForm
.miToBack
.Enabled
:= False;
1724 procedure DeleteSelectedObjects();
1729 if SelectedObjects
= nil then
1735 for a
:= 0 to High(SelectedObjects
) do
1736 with SelectedObjects
[a
] do
1741 SetLength(UndoBuffer
, Length(UndoBuffer
)+1);
1742 i
:= High(UndoBuffer
);
1746 SetLength(UndoBuffer
[i
], Length(UndoBuffer
[i
])+1);
1747 ii
:= High(UndoBuffer
[i
]);
1752 UndoBuffer
[i
, ii
].UndoType
:= UNDO_DELETE_PANEL
;
1753 New(UndoBuffer
[i
, ii
].Panel
);
1754 UndoBuffer
[i
, ii
].Panel
^ := gPanels
[ID
];
1758 UndoBuffer
[i
, ii
].UndoType
:= UNDO_DELETE_ITEM
;
1759 UndoBuffer
[i
, ii
].Item
:= gItems
[ID
];
1763 UndoBuffer
[i
, ii
].UndoType
:= UNDO_DELETE_AREA
;
1764 UndoBuffer
[i
, ii
].Area
:= gAreas
[ID
];
1768 UndoBuffer
[i
, ii
].UndoType
:= UNDO_DELETE_TRIGGER
;
1769 UndoBuffer
[i
, ii
].Trigger
:= gTriggers
[ID
];
1773 RemoveObject(ID
, ObjectType
);
1776 RemoveSelectFromObjects();
1778 MainForm
.miUndo
.Enabled
:= UndoBuffer
<> nil;
1779 MainForm
.RecountSelectedObjects();
1782 procedure Undo_Add(ObjectType
: Byte; ID
: DWORD
; Group
: Boolean = False);
1786 if (not Group
) or (Length(UndoBuffer
) = 0) then
1787 SetLength(UndoBuffer
, Length(UndoBuffer
)+1);
1788 SetLength(UndoBuffer
[High(UndoBuffer
)], Length(UndoBuffer
[High(UndoBuffer
)])+1);
1789 i
:= High(UndoBuffer
);
1790 ii
:= High(UndoBuffer
[i
]);
1794 UndoBuffer
[i
, ii
].UndoType
:= UNDO_ADD_PANEL
;
1796 UndoBuffer
[i
, ii
].UndoType
:= UNDO_ADD_ITEM
;
1798 UndoBuffer
[i
, ii
].UndoType
:= UNDO_ADD_MONSTER
;
1800 UndoBuffer
[i
, ii
].UndoType
:= UNDO_ADD_AREA
;
1802 UndoBuffer
[i
, ii
].UndoType
:= UNDO_ADD_TRIGGER
;
1805 UndoBuffer
[i
, ii
].AddID
:= ID
;
1807 MainForm
.miUndo
.Enabled
:= UndoBuffer
<> nil;
1810 procedure FullClear();
1812 RemoveSelectFromObjects();
1814 LoadSky(gMapInfo
.SkyName
);
1816 slInvalidTextures
.Clear();
1817 MapCheckForm
.lbErrorList
.Clear();
1818 MapCheckForm
.mErrorDescription
.Clear();
1820 MainForm
.miUndo
.Enabled
:= False;
1821 MainForm
.sbHorizontal
.Position
:= 0;
1822 MainForm
.sbVertical
.Position
:= 0;
1823 MainForm
.FormResize(nil);
1824 MainForm
.Caption
:= FormCaption
;
1829 procedure ErrorMessageBox(str
: String);
1831 MessageBox(0, PChar(str
), PChar(_lc
[I_MSG_ERROR
]),
1832 MB_ICONINFORMATION
or MB_OK
or MB_DEFBUTTON1
);
1835 function CheckProperty(): Boolean;
1841 _id
:= GetFirstSelected();
1843 if SelectedObjects
[_id
].ObjectType
= OBJECT_PANEL
then
1844 with gPanels
[SelectedObjects
[_id
].ID
] do
1846 if TextureWidth
<> 0 then
1847 if StrToIntDef(MainForm
.vleObjectProperty
.Values
[_lc
[I_PROP_WIDTH
]], 1) mod TextureWidth
<> 0 then
1849 ErrorMessageBox(Format(_lc
[I_MSG_WRONG_TEXWIDTH
],
1854 if TextureHeight
<> 0 then
1855 if StrToIntDef(Trim(MainForm
.vleObjectProperty
.Values
[_lc
[I_PROP_HEIGHT
]]), 1) mod TextureHeight
<> 0 then
1857 ErrorMessageBox(Format(_lc
[I_MSG_WRONG_TEXHEIGHT
],
1862 if IsTexturedPanel(PanelType
) and (TextureName
<> '') then
1863 if not (StrToIntDef(MainForm
.vleObjectProperty
.Values
[_lc
[I_PROP_PANEL_ALPHA
]], -1) in [0..255]) then
1865 ErrorMessageBox(_lc
[I_MSG_WRONG_ALPHA
]);
1870 if SelectedObjects
[_id
].ObjectType
in [OBJECT_PANEL
, OBJECT_TRIGGER
] then
1871 if (StrToIntDef(MainForm
.vleObjectProperty
.Values
[_lc
[I_PROP_WIDTH
]], 0) <= 0) or
1872 (StrToIntDef(MainForm
.vleObjectProperty
.Values
[_lc
[I_PROP_HEIGHT
]], 0) <= 0) then
1874 ErrorMessageBox(_lc
[I_MSG_WRONG_SIZE
]);
1878 if (Trim(MainForm
.vleObjectProperty
.Values
[_lc
[I_PROP_X
]]) = '') or
1879 (Trim(MainForm
.vleObjectProperty
.Values
[_lc
[I_PROP_Y
]]) = '') then
1881 ErrorMessageBox(_lc
[I_MSG_WRONG_XY
]);
1888 procedure SelectTexture(ID
: Integer);
1890 MainForm
.lbTextureList
.ItemIndex
:= ID
;
1891 MainForm
.lbTextureListClick(nil);
1894 function AddTexture(aWAD
, aSection
, aTex
: String; silent
: Boolean): Boolean;
1896 a
, FrameLen
: Integer;
1899 ResourceName
: String;
1900 FullResourceName
: String;
1901 SectionName
: String;
1903 Width
, Height
: Word;
1911 if aSection
= '..' then
1914 SectionName
:= aSection
;
1917 aWAD
:= _lc
[I_WAD_SPECIAL_MAP
];
1919 if aWAD
= _lc
[I_WAD_SPECIAL_MAP
] then
1921 g_ProcessResourceStr(OpenedMap
, @fn
, nil, nil);
1922 //FileName := EditorDir+'maps\'+ExtractFileName(fn);
1924 ResourceName
:= ':'+SectionName
+'\'+aTex
;
1927 if aWAD
= _lc
[I_WAD_SPECIAL_TEXS
] then
1928 begin // Спец. текстуры
1930 ResourceName
:= aTex
;
1933 begin // Внешний WAD
1934 FileName
:= EditorDir
+'wads/'+aWAD
;
1935 ResourceName
:= aWAD
+':'+SectionName
+'\'+aTex
;
1940 // Есть ли уже такая текстура:
1941 for a
:= 0 to MainForm
.lbTextureList
.Items
.Count
-1 do
1942 if ResourceName
= MainForm
.lbTextureList
.Items
[a
] then
1945 ErrorMessageBox(Format(_lc
[I_MSG_TEXTURE_ALREADY
],
1950 // Название ресурса <= 64 символов:
1951 if Length(ResourceName
) > 64 then
1954 ErrorMessageBox(Format(_lc
[I_MSG_RES_NAME_64
],
1962 if aWAD
= _lc
[I_WAD_SPECIAL_TEXS
] then
1964 a
:= MainForm
.lbTextureList
.Items
.Add(ResourceName
);
1971 FullResourceName
:= FileName
+':'+SectionName
+'\'+aTex
;
1973 if IsAnim(FullResourceName
) then
1974 begin // Аним. текстура
1975 GetFrame(FullResourceName
, Data
, FrameLen
, Width
, Height
);
1977 if not g_CreateTextureMemorySize(Data
, FrameLen
, ResourceName
, 0, 0, Width
, Height
, 1) then
1979 a
:= MainForm
.lbTextureList
.Items
.Add(ResourceName
);
1981 else // Обычная текстура
1983 if not g_CreateTextureWAD(ResourceName
, FullResourceName
) then
1985 a
:= MainForm
.lbTextureList
.Items
.Add(ResourceName
);
1987 if (not ok
) and (slInvalidTextures
.IndexOf(ResourceName
) = -1) then
1989 slInvalidTextures
.Add(ResourceName
);
1992 if (a
> -1) and (not silent
) then
1999 procedure UpdateCaption(sMap
, sFile
, sRes
: String);
2002 if (sFile
= '') and (sRes
= '') and (sMap
= '') then
2003 Caption
:= FormCaption
2006 Caption
:= Format('%s - %s:%s', [FormCaption
, sFile
, sRes
])
2008 if (sFile
<> '') and (sRes
<> '') then
2009 Caption
:= Format('%s - %s (%s:%s)', [FormCaption
, sMap
, sFile
, sRes
])
2011 Caption
:= Format('%s - %s', [FormCaption
, sMap
]);
2014 procedure OpenMap(FileName
: String; mapN
: String);
2019 SelectMapForm
.Caption
:= _lc
[I_CAP_OPEN
];
2020 SelectMapForm
.GetMaps(FileName
);
2022 if (FileName
= OpenedWAD
) and
2023 (OpenedMap
<> '') then
2025 MapName
:= OpenedMap
;
2026 while (Pos(':\', MapName
) > 0) do
2027 Delete(MapName
, 1, Pos(':\', MapName
) + 1);
2029 idx
:= SelectMapForm
.lbMapList
.Items
.IndexOf(MapName
);
2030 SelectMapForm
.lbMapList
.ItemIndex
:= idx
;
2033 if SelectMapForm
.lbMapList
.Count
> 0 then
2034 SelectMapForm
.lbMapList
.ItemIndex
:= 0
2036 SelectMapForm
.lbMapList
.ItemIndex
:= -1;
2041 idx
:= SelectMapForm
.lbMapList
.Items
.IndexOf(mapN
);
2045 if (SelectMapForm
.ShowModal() = mrOK
) and
2046 (SelectMapForm
.lbMapList
.ItemIndex
<> -1) then
2047 idx
:= SelectMapForm
.lbMapList
.ItemIndex
2052 MapName
:= SelectMapForm
.lbMapList
.Items
[idx
];
2058 pLoadProgress
.Left
:= (RenderPanel
.Width
div 2)-(pLoadProgress
.Width
div 2);
2059 pLoadProgress
.Top
:= (RenderPanel
.Height
div 2)-(pLoadProgress
.Height
div 2);
2060 pLoadProgress
.Show();
2062 OpenedMap
:= FileName
+':\'+MapName
;
2063 OpenedWAD
:= FileName
;
2065 idx
:= RecentFiles
.IndexOf(OpenedMap
);
2066 // Такая карта уже недавно открывалась:
2068 RecentFiles
.Delete(idx
);
2069 RecentFiles
.Insert(0, OpenedMap
);
2070 RefreshRecentMenu();
2074 pLoadProgress
.Hide();
2077 lbTextureList
.Sorted
:= True;
2078 lbTextureList
.Sorted
:= False;
2080 UpdateCaption(gMapInfo
.Name
, ExtractFileName(FileName
), MapName
);
2084 procedure MoveSelectedObjects(Wall
, alt
: Boolean; dx
, dy
: Integer);
2089 if SelectedObjects
= nil then
2096 for a
:= 0 to High(SelectedObjects
) do
2097 if SelectedObjects
[a
].Live
then
2099 if ObjectCollideLevel(SelectedObjects
[a
].ID
, SelectedObjects
[a
].ObjectType
, dx
, 0) then
2102 if ObjectCollideLevel(SelectedObjects
[a
].ID
, SelectedObjects
[a
].ObjectType
, 0, dy
) then
2105 if (not okX
) or (not okY
) then
2111 for a
:= 0 to High(SelectedObjects
) do
2112 if SelectedObjects
[a
].Live
then
2115 MoveObject(SelectedObjects
[a
].ObjectType
, SelectedObjects
[a
].ID
, dx
, 0);
2118 MoveObject(SelectedObjects
[a
].ObjectType
, SelectedObjects
[a
].ID
, 0, dy
);
2120 if alt
and (SelectedObjects
[a
].ObjectType
= OBJECT_TRIGGER
) then
2122 if gTriggers
[SelectedObjects
[a
].ID
].TriggerType
in [TRIGGER_PRESS
,
2123 TRIGGER_ON
, TRIGGER_OFF
, TRIGGER_ONOFF
] then
2124 begin // Двигаем зону Расширителя
2126 gTriggers
[SelectedObjects
[a
].ID
].Data
.tX
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.tX
+dx
;
2128 gTriggers
[SelectedObjects
[a
].ID
].Data
.tY
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.tY
+dy
;
2131 if gTriggers
[SelectedObjects
[a
].ID
].TriggerType
in [TRIGGER_TELEPORT
] then
2132 begin // Двигаем точку назначения Телепорта
2134 gTriggers
[SelectedObjects
[a
].ID
].Data
.TargetPoint
.X
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.TargetPoint
.X
+dx
;
2136 gTriggers
[SelectedObjects
[a
].ID
].Data
.TargetPoint
.Y
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.TargetPoint
.Y
+dy
;
2139 if gTriggers
[SelectedObjects
[a
].ID
].TriggerType
in [TRIGGER_SPAWNMONSTER
] then
2140 begin // Двигаем точку создания монстра
2142 gTriggers
[SelectedObjects
[a
].ID
].Data
.MonPos
.X
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.MonPos
.X
+dx
;
2144 gTriggers
[SelectedObjects
[a
].ID
].Data
.MonPos
.Y
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.MonPos
.Y
+dy
;
2147 if gTriggers
[SelectedObjects
[a
].ID
].TriggerType
in [TRIGGER_SPAWNITEM
] then
2148 begin // Двигаем точку создания предмета
2150 gTriggers
[SelectedObjects
[a
].ID
].Data
.ItemPos
.X
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.ItemPos
.X
+dx
;
2152 gTriggers
[SelectedObjects
[a
].ID
].Data
.ItemPos
.Y
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.ItemPos
.Y
+dy
;
2155 if gTriggers
[SelectedObjects
[a
].ID
].TriggerType
in [TRIGGER_SHOT
] then
2156 begin // Двигаем точку создания выстрела
2158 gTriggers
[SelectedObjects
[a
].ID
].Data
.ShotPos
.X
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.ShotPos
.X
+dx
;
2160 gTriggers
[SelectedObjects
[a
].ID
].Data
.ShotPos
.Y
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.ShotPos
.Y
+dy
;
2165 LastMovePoint
:= MousePos
;
2169 procedure ShowLayer(Layer
: Byte; show
: Boolean);
2171 LayerEnabled
[Layer
] := show
;
2176 MainForm
.miLayer1
.Checked
:= show
;
2177 MainForm
.miLayerP1
.Checked
:= show
;
2181 MainForm
.miLayer2
.Checked
:= show
;
2182 MainForm
.miLayerP2
.Checked
:= show
;
2186 MainForm
.miLayer3
.Checked
:= show
;
2187 MainForm
.miLayerP3
.Checked
:= show
;
2191 MainForm
.miLayer4
.Checked
:= show
;
2192 MainForm
.miLayerP4
.Checked
:= show
;
2196 MainForm
.miLayer5
.Checked
:= show
;
2197 MainForm
.miLayerP5
.Checked
:= show
;
2201 MainForm
.miLayer6
.Checked
:= show
;
2202 MainForm
.miLayerP6
.Checked
:= show
;
2206 MainForm
.miLayer7
.Checked
:= show
;
2207 MainForm
.miLayerP7
.Checked
:= show
;
2211 MainForm
.miLayer8
.Checked
:= show
;
2212 MainForm
.miLayerP8
.Checked
:= show
;
2216 MainForm
.miLayer9
.Checked
:= show
;
2217 MainForm
.miLayerP9
.Checked
:= show
;
2221 RemoveSelectFromObjects();
2224 procedure SwitchLayer(Layer
: Byte);
2226 ShowLayer(Layer
, not LayerEnabled
[Layer
]);
2229 procedure SwitchMap();
2231 ShowMap
:= not ShowMap
;
2232 MainForm
.tbShowMap
.Down
:= ShowMap
;
2235 procedure ShowEdges();
2237 if drEdge
[3] < 255 then
2240 drEdge
[3] := gAlphaEdge
;
2243 function SelectedTexture(): String;
2245 if MainForm
.lbTextureList
.ItemIndex
<> -1 then
2246 Result
:= MainForm
.lbTextureList
.Items
[MainForm
.lbTextureList
.ItemIndex
]
2251 function IsSpecialTextureSel(): Boolean;
2253 Result
:= (MainForm
.lbTextureList
.ItemIndex
<> -1) and
2254 IsSpecialTexture(MainForm
.lbTextureList
.Items
[MainForm
.lbTextureList
.ItemIndex
]);
2257 function CopyBufferToString(var CopyBuf
: TCopyRecArray
): String;
2262 procedure AddInt(x
: Integer);
2264 Res
:= Res
+ IntToStr(x
) + ' ';
2270 if Length(CopyBuf
) = 0 then
2273 Res
:= CLIPBOARD_SIG
+ ' ';
2275 for i
:= 0 to High(CopyBuf
) do
2277 if (CopyBuf
[i
].ObjectType
= OBJECT_PANEL
) and
2278 (CopyBuf
[i
].Panel
= nil) then
2282 AddInt(CopyBuf
[i
].ObjectType
);
2285 // Свойства объекта:
2286 case CopyBuf
[i
].ObjectType
of
2288 with CopyBuf
[i
].Panel
^ do
2295 Res
:= Res
+ '"' + TextureName
+ '" ';
2297 AddInt(IfThen(Blending
, 1, 0));
2301 with CopyBuf
[i
].Item
do
2306 AddInt(IfThen(OnlyDM
, 1, 0));
2307 AddInt(IfThen(Fall
, 1, 0));
2311 with CopyBuf
[i
].Monster
do
2313 AddInt(MonsterType
);
2316 AddInt(IfThen(Direction
= D_LEFT
, 1, 0));
2320 with CopyBuf
[i
].Area
do
2325 AddInt(IfThen(Direction
= D_LEFT
, 1, 0));
2329 with CopyBuf
[i
].Trigger
do
2331 AddInt(TriggerType
);
2336 AddInt(ActivateType
);
2338 AddInt(IfThen(Enabled
, 1, 0));
2339 AddInt(TexturePanel
);
2341 for j
:= 0 to 127 do
2342 AddInt(Data
.Default
[j
]);
2350 procedure StringToCopyBuffer(Str
: String; var CopyBuf
: TCopyRecArray
;
2355 function GetNext(): String;
2360 if Str
[1] = '"' then
2372 Result
:= Copy(Str
, 1, p
-1);
2388 Result
:= Copy(Str
, 1, p
-1);
2398 if GetNext() <> CLIPBOARD_SIG
then
2404 t
:= StrToIntDef(GetNext(), 0);
2406 if (t
< OBJECT_PANEL
) or (t
> OBJECT_TRIGGER
) or
2407 (GetNext() <> ';') then
2408 begin // Что-то не то => пропускаем:
2416 i
:= Length(CopyBuf
);
2417 SetLength(CopyBuf
, i
+ 1);
2419 CopyBuf
[i
].ObjectType
:= t
;
2420 CopyBuf
[i
].Panel
:= nil;
2422 // Свойства объекта:
2426 New(CopyBuf
[i
].Panel
);
2428 with CopyBuf
[i
].Panel
^ do
2430 PanelType
:= StrToIntDef(GetNext(), PANEL_WALL
);
2431 X
:= StrToIntDef(GetNext(), 0);
2432 Y
:= StrToIntDef(GetNext(), 0);
2433 pmin
.X
:= Min(X
, pmin
.X
);
2434 pmin
.Y
:= Min(Y
, pmin
.Y
);
2435 Width
:= StrToIntDef(GetNext(), 16);
2436 Height
:= StrToIntDef(GetNext(), 16);
2437 TextureName
:= GetNext();
2438 Alpha
:= StrToIntDef(GetNext(), 0);
2439 Blending
:= (GetNext() = '1');
2444 with CopyBuf
[i
].Item
do
2446 ItemType
:= StrToIntDef(GetNext(), ITEM_MEDKIT_SMALL
);
2447 X
:= StrToIntDef(GetNext(), 0);
2448 Y
:= StrToIntDef(GetNext(), 0);
2449 pmin
.X
:= Min(X
, pmin
.X
);
2450 pmin
.Y
:= Min(Y
, pmin
.Y
);
2451 OnlyDM
:= (GetNext() = '1');
2452 Fall
:= (GetNext() = '1');
2456 with CopyBuf
[i
].Monster
do
2458 MonsterType
:= StrToIntDef(GetNext(), MONSTER_DEMON
);
2459 X
:= StrToIntDef(GetNext(), 0);
2460 Y
:= StrToIntDef(GetNext(), 0);
2461 pmin
.X
:= Min(X
, pmin
.X
);
2462 pmin
.Y
:= Min(Y
, pmin
.Y
);
2464 if GetNext() = '1' then
2467 Direction
:= D_RIGHT
;
2471 with CopyBuf
[i
].Area
do
2473 AreaType
:= StrToIntDef(GetNext(), AREA_PLAYERPOINT1
);
2474 X
:= StrToIntDef(GetNext(), 0);
2475 Y
:= StrToIntDef(GetNext(), 0);
2476 pmin
.X
:= Min(X
, pmin
.X
);
2477 pmin
.Y
:= Min(Y
, pmin
.Y
);
2478 if GetNext() = '1' then
2481 Direction
:= D_RIGHT
;
2485 with CopyBuf
[i
].Trigger
do
2487 TriggerType
:= StrToIntDef(GetNext(), TRIGGER_EXIT
);
2488 X
:= StrToIntDef(GetNext(), 0);
2489 Y
:= StrToIntDef(GetNext(), 0);
2490 pmin
.X
:= Min(X
, pmin
.X
);
2491 pmin
.Y
:= Min(Y
, pmin
.Y
);
2492 Width
:= StrToIntDef(GetNext(), 16);
2493 Height
:= StrToIntDef(GetNext(), 16);
2494 ActivateType
:= StrToIntDef(GetNext(), 0);
2495 Key
:= StrToIntDef(GetNext(), 0);
2496 Enabled
:= (GetNext() = '1');
2497 TexturePanel
:= StrToIntDef(GetNext(), 0);
2499 for j
:= 0 to 127 do
2500 Data
.Default
[j
] := StrToIntDef(GetNext(), 0);
2505 pmin
.X
:= Min(Data
.TargetPoint
.X
, pmin
.X
);
2506 pmin
.Y
:= Min(Data
.TargetPoint
.Y
, pmin
.Y
);
2508 TRIGGER_PRESS
, TRIGGER_ON
, TRIGGER_OFF
, TRIGGER_ONOFF
:
2510 pmin
.X
:= Min(Data
.tX
, pmin
.X
);
2511 pmin
.Y
:= Min(Data
.tY
, pmin
.Y
);
2513 TRIGGER_SPAWNMONSTER
:
2515 pmin
.X
:= Min(Data
.MonPos
.X
, pmin
.X
);
2516 pmin
.Y
:= Min(Data
.MonPos
.Y
, pmin
.Y
);
2520 pmin
.X
:= Min(Data
.ItemPos
.X
, pmin
.X
);
2521 pmin
.Y
:= Min(Data
.ItemPos
.Y
, pmin
.Y
);
2525 pmin
.X
:= Min(Data
.ShotPos
.X
, pmin
.X
);
2526 pmin
.Y
:= Min(Data
.ShotPos
.Y
, pmin
.Y
);
2534 //----------------------------------------
2535 //Закончились вспомогательные процедуры
2536 //----------------------------------------
2538 procedure TMainForm
.RefreshRecentMenu();
2543 // Лишние запомненные карты:
2544 while RecentFiles
.Count
> RecentCount
do
2545 RecentFiles
.Delete(RecentFiles
.Count
-1);
2547 // Лишние строки меню:
2548 while MainMenu
.Items
[0].Count
> RECENT_FILES_MENU_START
do
2549 MainMenu
.Items
[0].Delete(MainMenu
.Items
[0].Count
-1);
2551 // Отделение списка карт от строки "Выход":
2552 if RecentFiles
.Count
> 0 then
2554 MI
:= TMenuItem
.Create(MainMenu
.Items
[0]);
2556 MainMenu
.Items
[0].Add(MI
);
2559 // Добавление в меню списка запомненных карт:
2560 for i
:= 0 to RecentFiles
.Count
-1 do
2562 MI
:= TMenuItem
.Create(MainMenu
.Items
[0]);
2563 MI
.Caption
:= IntToStr(i
+1) + ' ' + RecentFiles
[i
];
2564 MI
.OnClick
:= aRecentFileExecute
;
2565 MainMenu
.Items
[0].Add(MI
);
2569 procedure TMainForm
.aRecentFileExecute(Sender
: TObject
);
2575 s
:= LowerCase((Sender
as TMenuItem
).Caption
);
2576 Delete(s
, Pos('&', s
), 1);
2577 s
:= Trim(Copy(s
, 1, 2));
2578 n
:= StrToIntDef(s
, 0) - 1;
2580 if (n
< 0) or (n
>= RecentFiles
.Count
) then
2583 s
:= RecentFiles
[n
];
2584 pw
:= Pos('.wad:\', LowerCase(s
));
2588 begin // Map name included
2589 fn
:= Copy(s
, 1, pw
+ 3);
2590 Delete(s
, 1, pw
+ 5);
2591 if (FileExists(fn
)) then
2597 else // Only wad name
2598 if (FileExists(s
)) then
2604 if (not b
) and (MessageBox(0, PChar(_lc
[I_MSG_DEL_RECENT_PROMT
]),
2605 PChar(_lc
[I_MSG_DEL_RECENT
]), MB_ICONQUESTION
or MB_YESNO
) = idYes
) then
2607 RecentFiles
.Delete(n
);
2608 RefreshRecentMenu();
2612 procedure TMainForm
.aEditorOptionsExecute(Sender
: TObject
);
2614 OptionsForm
.ShowModal();
2617 procedure LoadStdFont(cfgres
, texture
: string; var FontID
: DWORD
);
2627 g_ReadResource(EditorDir
+ 'data/Game.wad', 'FONTS', cfgres
, cfgdata
, cfglen
);
2628 if cfgdata
<> nil then
2630 if not g_CreateTextureWAD('FONT_STD', EditorDir
+ 'data/Game.wad:FONTS\' + texture
) then
2631 e_WriteLog('ERROR ERROR ERROR', MSG_WARNING
);
2633 config
:= TConfig
.CreateMem(cfgdata
, cfglen
);
2634 cwdt
:= Min(Max(config
.ReadInt('FontMap', 'CharWidth', 0), 0), 255);
2635 chgt
:= Min(Max(config
.ReadInt('FontMap', 'CharHeight', 0), 0), 255);
2636 spc
:= Min(Max(config
.ReadInt('FontMap', 'Kerning', 0), -128), 127);
2638 if g_GetTexture('FONT_STD', ID
) then
2639 e_TextureFontBuild(ID
, FontID
, cwdt
, chgt
, spc
- 2);
2646 e_WriteLog('Could not load FONT_STD', MSG_WARNING
)
2650 procedure TMainForm
.FormCreate(Sender
: TObject
);
2658 EditorDir
:= ExtractFilePath(Application
.ExeName
);
2660 e_InitLog(EditorDir
+'Editor.log', WM_NEWFILE
);
2662 slInvalidTextures
:= TStringList
.Create
;
2664 ShowLayer(LAYER_BACK
, True);
2665 ShowLayer(LAYER_WALLS
, True);
2666 ShowLayer(LAYER_FOREGROUND
, True);
2667 ShowLayer(LAYER_STEPS
, True);
2668 ShowLayer(LAYER_WATER
, True);
2669 ShowLayer(LAYER_ITEMS
, True);
2670 ShowLayer(LAYER_MONSTERS
, True);
2671 ShowLayer(LAYER_AREAS
, True);
2672 ShowLayer(LAYER_TRIGGERS
, True);
2676 FormCaption
:= MainForm
.Caption
;
2680 config
:= TConfig
.CreateFile(EditorDir
+'Editor.cfg');
2682 if config
.ReadInt('Editor', 'XPos', -1) = -1 then
2683 Position
:= poDesktopCenter
2685 Left
:= config
.ReadInt('Editor', 'XPos', Left
);
2686 Top
:= config
.ReadInt('Editor', 'YPos', Top
);
2687 Width
:= config
.ReadInt('Editor', 'Width', Width
);
2688 Height
:= config
.ReadInt('Editor', 'Height', Height
);
2690 if config
.ReadBool('Editor', 'Maximize', False) then
2691 WindowState
:= wsMaximized
;
2692 ShowMap
:= config
.ReadBool('Editor', 'Minimap', False);
2693 PanelProps
.Width
:= config
.ReadInt('Editor', 'PanelProps', PanelProps
.ClientWidth
);
2694 Splitter1
.Left
:= PanelProps
.Left
;
2695 PanelObjs
.Height
:= config
.ReadInt('Editor', 'PanelObjs', PanelObjs
.ClientHeight
);
2696 Splitter2
.Top
:= PanelObjs
.Top
;
2697 StatusBar
.Top
:= PanelObjs
.BoundsRect
.Bottom
;
2698 DotEnable
:= config
.ReadBool('Editor', 'DotEnable', True);
2699 DotColor
:= config
.ReadInt('Editor', 'DotColor', $FFFFFF);
2700 DotStepOne
:= config
.ReadInt('Editor', 'DotStepOne', 16);
2701 DotStepTwo
:= config
.ReadInt('Editor', 'DotStepTwo', 8);
2702 DotStep
:= config
.ReadInt('Editor', 'DotStep', DotStepOne
);
2703 DrawTexturePanel
:= config
.ReadBool('Editor', 'DrawTexturePanel', True);
2704 DrawPanelSize
:= config
.ReadBool('Editor', 'DrawPanelSize', True);
2705 BackColor
:= config
.ReadInt('Editor', 'BackColor', $7F6040);
2706 PreviewColor
:= config
.ReadInt('Editor', 'PreviewColor', $00FF00);
2707 UseCheckerboard
:= config
.ReadBool('Editor', 'UseCheckerboard', True);
2708 gColorEdge
:= config
.ReadInt('Editor', 'EdgeColor', COLOR_EDGE
);
2709 gAlphaEdge
:= config
.ReadInt('Editor', 'EdgeAlpha', ALPHA_EDGE
);
2710 if gAlphaEdge
= 255 then
2711 gAlphaEdge
:= ALPHA_EDGE
;
2712 drEdge
[0] := GetRValue(gColorEdge
);
2713 drEdge
[1] := GetGValue(gColorEdge
);
2714 drEdge
[2] := GetBValue(gColorEdge
);
2715 if not config
.ReadBool('Editor', 'EdgeShow', True) then
2718 drEdge
[3] := gAlphaEdge
;
2719 gAlphaTriggerLine
:= config
.ReadInt('Editor', 'LineAlpha', ALPHA_LINE
);
2720 if gAlphaTriggerLine
= 255 then
2721 gAlphaTriggerLine
:= ALPHA_LINE
;
2722 gAlphaTriggerArea
:= config
.ReadInt('Editor', 'TriggerAlpha', ALPHA_AREA
);
2723 if gAlphaTriggerArea
= 255 then
2724 gAlphaTriggerArea
:= ALPHA_AREA
;
2725 gAlphaMonsterRect
:= config
.ReadInt('Editor', 'MonsterRectAlpha', 0);
2726 gAlphaAreaRect
:= config
.ReadInt('Editor', 'AreaRectAlpha', 0);
2727 if config
.ReadInt('Editor', 'Scale', 0) = 1 then
2731 if config
.ReadInt('Editor', 'DotSize', 0) = 1 then
2735 OpenDialog
.InitialDir
:= config
.ReadStr('Editor', 'LastOpenDir', EditorDir
);
2736 SaveDialog
.InitialDir
:= config
.ReadStr('Editor', 'LastSaveDir', EditorDir
);
2738 s
:= config
.ReadStr('Editor', 'Language', '');
2741 RecentCount
:= config
.ReadInt('Editor', 'RecentCount', 5);
2742 if RecentCount
> 10 then
2744 if RecentCount
< 2 then
2747 RecentFiles
:= TStringList
.Create();
2748 for i
:= 0 to RecentCount
-1 do
2750 s
:= config
.ReadStr('RecentFiles', IntToStr(i
+1), '');
2754 RefreshRecentMenu();
2758 tbShowMap
.Down
:= ShowMap
;
2759 tbGridOn
.Down
:= DotEnable
;
2760 pcObjects
.ActivePageIndex
:= 0;
2761 Application
.Title
:= _lc
[I_EDITOR_TITLE
];
2763 Application
.OnIdle
:= OnIdle
;
2766 procedure PrintBlack(X
, Y
: Integer; Text: string; FontID
: DWORD
);
2768 // NOTE: all the font printing routines assume CP1251
2769 e_TextureFontPrintEx(X
, Y
, Text, FontID
, 0, 0, 0, 1.0);
2772 procedure TMainForm
.Draw();
2777 Width
, Height
: Word;
2780 aX
, aY
, aX2
, aY2
, XX
, ScaleSz
: Integer;
2789 e_Clear(GL_COLOR_BUFFER_BIT
,
2790 GetRValue(BackColor
)/255,
2791 GetGValue(BackColor
)/255,
2792 GetBValue(BackColor
)/255);
2796 ObjCount
:= SelectedObjectCount();
2798 // Обводим выделенные объекты красной рамкой:
2799 if ObjCount
> 0 then
2801 for a
:= 0 to High(SelectedObjects
) do
2802 if SelectedObjects
[a
].Live
then
2804 Rect
:= ObjectGetRect(SelectedObjects
[a
].ObjectType
, SelectedObjects
[a
].ID
);
2808 e_DrawQuad(X
+MapOffset
.X
, Y
+MapOffset
.Y
,
2809 X
+MapOffset
.X
+Width
-1, Y
+MapOffset
.Y
+Height
-1,
2812 // Рисуем точки изменения размеров:
2813 if (ObjCount
= 1) and
2814 (SelectedObjects
[GetFirstSelected
].ObjectType
in [OBJECT_PANEL
, OBJECT_TRIGGER
]) then
2816 e_DrawPoint(5, X
+MapOffset
.X
, Y
+MapOffset
.Y
+(Height
div 2), 255, 255, 255);
2817 e_DrawPoint(5, X
+MapOffset
.X
+Width
-1, Y
+MapOffset
.Y
+(Height
div 2), 255, 255, 255);
2818 e_DrawPoint(5, X
+MapOffset
.X
+(Width
div 2), Y
+MapOffset
.Y
, 255, 255, 255);
2819 e_DrawPoint(5, X
+MapOffset
.X
+(Width
div 2), Y
+MapOffset
.Y
+Height
-1, 255, 255, 255);
2821 e_DrawPoint(3, X
+MapOffset
.X
, Y
+MapOffset
.Y
+(Height
div 2), 255, 0, 0);
2822 e_DrawPoint(3, X
+MapOffset
.X
+Width
-1, Y
+MapOffset
.Y
+(Height
div 2), 255, 0, 0);
2823 e_DrawPoint(3, X
+MapOffset
.X
+(Width
div 2), Y
+MapOffset
.Y
, 255, 0, 0);
2824 e_DrawPoint(3, X
+MapOffset
.X
+(Width
div 2), Y
+MapOffset
.Y
+Height
-1, 255, 0, 0);
2831 if DotEnable
and (PreviewMode
= 0) then
2838 for x
:= 0 to (RenderPanel
.Width
div DotStep
) do
2839 for y
:= 0 to (RenderPanel
.Height
div DotStep
) do
2840 e_DrawPoint(DotSize
, x
*DotStep
+ a
, y
*DotStep
+ a
,
2841 GetRValue(DotColor
),
2842 GetGValue(DotColor
),
2843 GetBValue(DotColor
));
2847 if (lbTextureList
.ItemIndex
<> -1) and (cbPreview
.Checked
) and
2848 (not IsSpecialTextureSel()) and (PreviewMode
= 0) then
2850 if not g_GetTexture(SelectedTexture(), ID
) then
2851 g_GetTexture('NOTEXTURE', ID
);
2852 g_GetTextureSizeByID(ID
, Width
, Height
);
2853 if UseCheckerboard
then
2855 if g_GetTexture('PREVIEW', PID
) then
2856 e_DrawFill(PID
, RenderPanel
.Width
-Width
, RenderPanel
.Height
-Height
, Width
div 16 + 1, Height
div 16 + 1, 0, True, False);
2858 e_DrawFillQuad(RenderPanel
.Width
-Width
-2, RenderPanel
.Height
-Height
-2,
2859 RenderPanel
.Width
-1, RenderPanel
.Height
-1,
2860 GetRValue(PreviewColor
), GetGValue(PreviewColor
), GetBValue(PreviewColor
), 0);
2861 e_Draw(ID
, RenderPanel
.Width
-Width
, RenderPanel
.Height
-Height
, 0, True, False);
2864 // Подсказка при выборе точки Телепорта:
2865 if SelectFlag
= SELECTFLAG_TELEPORT
then
2867 with gTriggers
[SelectedObjects
[GetFirstSelected()].ID
] do
2868 if Data
.d2d_teleport
then
2869 e_DrawLine(2, MousePos
.X
-16, MousePos
.Y
-1,
2870 MousePos
.X
+16, MousePos
.Y
-1,
2873 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+AreaSize
[AREA_DMPOINT
].Width
-1,
2874 MousePos
.Y
+AreaSize
[AREA_DMPOINT
].Height
-1, 255, 255, 255);
2876 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 192, 192, 192, 127);
2877 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 255, 255, 255);
2878 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, _glc
[I_HINT_TELEPORT
], gEditorFont
);
2881 // Подсказка при выборе точки появления:
2882 if SelectFlag
= SELECTFLAG_SPAWNPOINT
then
2884 e_DrawLine(2, MousePos
.X
-16, MousePos
.Y
-1,
2885 MousePos
.X
+16, MousePos
.Y
-1,
2887 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 192, 192, 192, 127);
2888 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 255, 255, 255);
2889 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, _glc
[I_HINT_SPAWN
], gEditorFont
);
2892 // Подсказка при выборе панели двери:
2893 if SelectFlag
= SELECTFLAG_DOOR
then
2895 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 192, 192, 192, 127);
2896 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 255, 255, 255);
2897 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, _glc
[I_HINT_PANEL_DOOR
], gEditorFont
);
2900 // Подсказка при выборе панели с текстурой:
2901 if SelectFlag
= SELECTFLAG_TEXTURE
then
2903 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+196, MousePos
.Y
+18, 192, 192, 192, 127);
2904 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+196, MousePos
.Y
+18, 255, 255, 255);
2905 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, _glc
[I_HINT_PANEL_TEXTURE
], gEditorFont
);
2908 // Подсказка при выборе панели индикации выстрела:
2909 if SelectFlag
= SELECTFLAG_SHOTPANEL
then
2911 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+316, MousePos
.Y
+18, 192, 192, 192, 127);
2912 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+316, MousePos
.Y
+18, 255, 255, 255);
2913 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, _glc
[I_HINT_PANEL_SHOT
], gEditorFont
);
2916 // Подсказка при выборе панели лифта:
2917 if SelectFlag
= SELECTFLAG_LIFT
then
2919 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 192, 192, 192, 127);
2920 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 255, 255, 255);
2921 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, _glc
[I_HINT_PANEL_LIFT
], gEditorFont
);
2924 // Подсказка при выборе монстра:
2925 if SelectFlag
= SELECTFLAG_MONSTER
then
2927 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+120, MousePos
.Y
+18, 192, 192, 192, 127);
2928 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+120, MousePos
.Y
+18, 255, 255, 255);
2929 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, _glc
[I_HINT_MONSTER
], gEditorFont
);
2932 // Подсказка при выборе области воздействия:
2933 if DrawPressRect
then
2935 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+204, MousePos
.Y
+18, 192, 192, 192, 127);
2936 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+204, MousePos
.Y
+18, 255, 255, 255);
2937 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, _glc
[I_HINT_EXT_AREA
], gEditorFont
);
2940 // Рисуем текстуры, если чертим панель:
2941 if (MouseAction
= MOUSEACTION_DRAWPANEL
) and (DrawTexturePanel
) and
2942 (lbTextureList
.ItemIndex
<> -1) and (DrawRect
<> nil) and
2943 (lbPanelType
.ItemIndex
in [0..8]) and not IsSpecialTextureSel() then
2945 if not g_GetTexture(SelectedTexture(), ID
) then
2946 g_GetTexture('NOTEXTURE', ID
);
2947 g_GetTextureSizeByID(ID
, Width
, Height
);
2949 if (Abs(Right
-Left
) >= Width
) and (Abs(Bottom
-Top
) >= Height
) then
2950 e_DrawFill(ID
, Min(Left
, Right
), Min(Top
, Bottom
), Abs(Right
-Left
) div Width
,
2951 Abs(Bottom
-Top
) div Height
, 64, True, False);
2954 // Прямоугольник выделения:
2955 if DrawRect
<> nil then
2957 e_DrawQuad(Left
, Top
, Right
-1, Bottom
-1, 255, 255, 255);
2959 // Чертим мышью панель/триггер или меняем мышью их размер:
2960 if (((MouseAction
in [MOUSEACTION_DRAWPANEL
, MOUSEACTION_DRAWTRIGGER
]) and
2961 not(ssCtrl
in GetKeyShiftState())) or (MouseAction
= MOUSEACTION_RESIZE
)) and
2962 (DrawPanelSize
) then
2964 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+88, MousePos
.Y
+33, 192, 192, 192, 127);
2965 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+88, MousePos
.Y
+33, 255, 255, 255);
2967 if MouseAction
in [MOUSEACTION_DRAWPANEL
, MOUSEACTION_DRAWTRIGGER
] then
2968 begin // Чертим новый
2969 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, Format(_glc
[I_HINT_WIDTH
],
2970 [Abs(MousePos
.X
-MouseLDownPos
.X
)]), gEditorFont
);
2971 PrintBlack(MousePos
.X
+2, MousePos
.Y
+16, Format(_glc
[I_HINT_HEIGHT
],
2972 [Abs(MousePos
.Y
-MouseLDownPos
.Y
)]), gEditorFont
);
2974 else // Растягиваем существующий
2975 if SelectedObjects
[GetFirstSelected
].ObjectType
in [OBJECT_PANEL
, OBJECT_TRIGGER
] then
2977 if SelectedObjects
[GetFirstSelected
].ObjectType
= OBJECT_PANEL
then
2979 Width
:= gPanels
[SelectedObjects
[GetFirstSelected
].ID
].Width
;
2980 Height
:= gPanels
[SelectedObjects
[GetFirstSelected
].ID
].Height
;
2984 Width
:= gTriggers
[SelectedObjects
[GetFirstSelected
].ID
].Width
;
2985 Height
:= gTriggers
[SelectedObjects
[GetFirstSelected
].ID
].Height
;
2988 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, Format(_glc
[I_HINT_WIDTH
], [Width
]),
2990 PrintBlack(MousePos
.X
+2, MousePos
.Y
+16, Format(_glc
[I_HINT_HEIGHT
], [Height
]),
2995 // Ближайшая к курсору мыши точка на сетке:
2996 e_DrawPoint(3, MousePos
.X
, MousePos
.Y
, 0, 0, 255);
3001 // Сколько пикселов карты в 1 пикселе мини-карты:
3002 ScaleSz
:= 16 div Scale
;
3003 // Размеры мини-карты:
3004 aX
:= max(gMapInfo
.Width
div ScaleSz
, 1);
3005 aY
:= max(gMapInfo
.Height
div ScaleSz
, 1);
3006 // X-координата на RenderPanel нулевой x-координаты карты:
3007 XX
:= RenderPanel
.Width
- aX
- 1;
3009 e_DrawFillQuad(XX
-1, 0, RenderPanel
.Width
-1, aY
+1, 0, 0, 0, 0);
3010 e_DrawQuad(XX
-1, 0, RenderPanel
.Width
-1, aY
+1, 197, 197, 197);
3012 if gPanels
<> nil then
3015 for a
:= 0 to High(gPanels
) do
3017 if PanelType
<> 0 then
3019 // Левый верхний угол:
3020 aX
:= XX
+ (X
div ScaleSz
);
3021 aY
:= 1 + (Y
div ScaleSz
);
3023 aX2
:= max(Width
div ScaleSz
, 1);
3024 aY2
:= max(Height
div ScaleSz
, 1);
3025 // Правый нижний угол:
3026 aX2
:= aX
+ aX2
- 1;
3027 aY2
:= aY
+ aY2
- 1;
3030 PANEL_WALL
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 208, 208, 208, 0);
3031 PANEL_WATER
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 0, 0, 192, 0);
3032 PANEL_ACID1
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 0, 176, 0, 0);
3033 PANEL_ACID2
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 176, 0, 0, 0);
3034 PANEL_STEP
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 128, 128, 128, 0);
3035 PANEL_LIFTUP
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 116, 72, 36, 0);
3036 PANEL_LIFTDOWN
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 116, 124, 96, 0);
3037 PANEL_LIFTLEFT
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 200, 80, 4, 0);
3038 PANEL_LIFTRIGHT
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 252, 140, 56, 0);
3039 PANEL_OPENDOOR
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 100, 220, 92, 0);
3040 PANEL_CLOSEDOOR
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 212, 184, 64, 0);
3041 PANEL_BLOCKMON
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 192, 0, 192, 0);
3045 // Рисуем красным выделенные панели:
3046 if SelectedObjects
<> nil then
3047 for b
:= 0 to High(SelectedObjects
) do
3048 with SelectedObjects
[b
] do
3049 if Live
and (ObjectType
= OBJECT_PANEL
) then
3050 with gPanels
[SelectedObjects
[b
].ID
] do
3051 if PanelType
and not(PANEL_BACK
or PANEL_FORE
) <> 0 then
3053 // Левый верхний угол:
3054 aX
:= XX
+ (X
div ScaleSz
);
3055 aY
:= 1 + (Y
div ScaleSz
);
3057 aX2
:= max(Width
div ScaleSz
, 1);
3058 aY2
:= max(Height
div ScaleSz
, 1);
3059 // Правый нижний угол:
3060 aX2
:= aX
+ aX2
- 1;
3061 aY2
:= aY
+ aY2
- 1;
3063 e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 255, 0, 0, 0)
3067 if (gMapInfo
.Width
> RenderPanel
.Width
) or
3068 (gMapInfo
.Height
> RenderPanel
.Height
) then
3070 // Окно, показывающее текущее положение экрана на карте:
3072 x
:= max(min(RenderPanel
.Width
, gMapInfo
.Width
) div ScaleSz
, 1);
3073 y
:= max(min(RenderPanel
.Height
, gMapInfo
.Height
) div ScaleSz
, 1);
3074 // Левый верхний угол:
3075 aX
:= XX
+ ((-MapOffset
.X
) div ScaleSz
);
3076 aY
:= 1 + ((-MapOffset
.Y
) div ScaleSz
);
3077 // Правый нижний угол:
3081 e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 127, 192, 127, 127, B_BLEND
);
3082 e_DrawQuad(aX
, aY
, aX2
, aY2
, 255, 0, 0);
3087 RenderPanel
.SwapBuffers();
3090 procedure TMainForm
.FormResize(Sender
: TObject
);
3092 e_SetViewPort(0, 0, RenderPanel
.Width
, RenderPanel
.Height
);
3094 if gMapInfo
.Width
>= RenderPanel
.Width
then
3095 sbHorizontal
.Max
:= Normalize16(gMapInfo
.Width
-RenderPanel
.Width
+16)
3097 sbHorizontal
.Max
:= 0;
3099 if gMapInfo
.Height
>= RenderPanel
.Height
then
3100 sbVertical
.Max
:= Normalize16(gMapInfo
.Height
-RenderPanel
.Height
+16)
3102 sbVertical
.Max
:= 0;
3104 MapOffset
.X
:= -Normalize16(sbHorizontal
.Position
);
3105 MapOffset
.Y
:= -Normalize16(sbVertical
.Position
);
3108 procedure SelectNextObject(X
, Y
: Integer; ObjectType
: Byte; ID
: DWORD
);
3113 j_max
:= 0; // shut up compiler
3117 res
:= (gPanels
<> nil) and
3118 PanelInShownLayer(gPanels
[ID
].PanelType
) and
3119 g_CollidePoint(X
, Y
, gPanels
[ID
].X
, gPanels
[ID
].Y
,
3121 gPanels
[ID
].Height
);
3122 j_max
:= Length(gPanels
) - 1;
3127 res
:= (gItems
<> nil) and
3128 LayerEnabled
[LAYER_ITEMS
] and
3129 g_CollidePoint(X
, Y
, gItems
[ID
].X
, gItems
[ID
].Y
,
3130 ItemSize
[gItems
[ID
].ItemType
][0],
3131 ItemSize
[gItems
[ID
].ItemType
][1]);
3132 j_max
:= Length(gItems
) - 1;
3137 res
:= (gMonsters
<> nil) and
3138 LayerEnabled
[LAYER_MONSTERS
] and
3139 g_CollidePoint(X
, Y
, gMonsters
[ID
].X
, gMonsters
[ID
].Y
,
3140 MonsterSize
[gMonsters
[ID
].MonsterType
].Width
,
3141 MonsterSize
[gMonsters
[ID
].MonsterType
].Height
);
3142 j_max
:= Length(gMonsters
) - 1;
3147 res
:= (gAreas
<> nil) and
3148 LayerEnabled
[LAYER_AREAS
] and
3149 g_CollidePoint(X
, Y
, gAreas
[ID
].X
, gAreas
[ID
].Y
,
3150 AreaSize
[gAreas
[ID
].AreaType
].Width
,
3151 AreaSize
[gAreas
[ID
].AreaType
].Height
);
3152 j_max
:= Length(gAreas
) - 1;
3157 res
:= (gTriggers
<> nil) and
3158 LayerEnabled
[LAYER_TRIGGERS
] and
3159 g_CollidePoint(X
, Y
, gTriggers
[ID
].X
, gTriggers
[ID
].Y
,
3160 gTriggers
[ID
].Width
,
3161 gTriggers
[ID
].Height
);
3162 j_max
:= Length(gTriggers
) - 1;
3172 // Перебор ID: от ID-1 до 0; потом от High до ID+1:
3181 if j
= Integer(ID
) then
3186 res
:= PanelInShownLayer(gPanels
[j
].PanelType
) and
3187 g_CollidePoint(X
, Y
, gPanels
[j
].X
, gPanels
[j
].Y
,
3191 res
:= (gItems
[j
].ItemType
<> ITEM_NONE
) and
3192 g_CollidePoint(X
, Y
, gItems
[j
].X
, gItems
[j
].Y
,
3193 ItemSize
[gItems
[j
].ItemType
][0],
3194 ItemSize
[gItems
[j
].ItemType
][1]);
3196 res
:= (gMonsters
[j
].MonsterType
<> MONSTER_NONE
) and
3197 g_CollidePoint(X
, Y
, gMonsters
[j
].X
, gMonsters
[j
].Y
,
3198 MonsterSize
[gMonsters
[j
].MonsterType
].Width
,
3199 MonsterSize
[gMonsters
[j
].MonsterType
].Height
);
3201 res
:= (gAreas
[j
].AreaType
<> AREA_NONE
) and
3202 g_CollidePoint(X
, Y
, gAreas
[j
].X
, gAreas
[j
].Y
,
3203 AreaSize
[gAreas
[j
].AreaType
].Width
,
3204 AreaSize
[gAreas
[j
].AreaType
].Height
);
3206 res
:= (gTriggers
[j
].TriggerType
<> TRIGGER_NONE
) and
3207 g_CollidePoint(X
, Y
, gTriggers
[j
].X
, gTriggers
[j
].Y
,
3209 gTriggers
[j
].Height
);
3216 SetLength(SelectedObjects
, 1);
3218 SelectedObjects
[0].ObjectType
:= ObjectType
;
3219 SelectedObjects
[0].ID
:= j
;
3220 SelectedObjects
[0].Live
:= True;
3228 procedure TMainForm
.RenderPanelMouseDown(Sender
: TObject
;
3229 Button
: TMouseButton
; Shift
: TShiftState
; X
, Y
: Integer);
3233 c1
, c2
, c3
, c4
: Boolean;
3239 MainForm
.ActiveControl
:= RenderPanel
;
3240 RenderPanel
.SetFocus();
3242 RenderPanelMouseMove(RenderPanel
, Shift
, X
, Y
);
3244 if Button
= mbLeft
then // Left Mouse Button
3246 // Двигаем карту с помощью мыши и мини-карты:
3248 g_CollidePoint(X
, Y
,
3249 RenderPanel
.Width
-max(gMapInfo
.Width
div (16 div Scale
), 1)-1,
3251 max(gMapInfo
.Width
div (16 div Scale
), 1),
3252 max(gMapInfo
.Height
div (16 div Scale
), 1) ) then
3255 MouseAction
:= MOUSEACTION_MOVEMAP
;
3257 else // Ставим предмет/монстра/область:
3258 if (pcObjects
.ActivePageIndex
in [1, 2, 3]) and
3259 (not (ssShift
in Shift
)) then
3261 case pcObjects
.ActivePageIndex
of
3263 if lbItemList
.ItemIndex
= -1 then
3264 ErrorMessageBox(_lc
[I_MSG_CHOOSE_ITEM
])
3267 item
.ItemType
:= lbItemList
.ItemIndex
+ ITEM_MEDKIT_SMALL
;
3268 if item
.ItemType
>= ITEM_WEAPON_KASTET
then
3269 item
.ItemType
:= item
.ItemType
+ 2;
3270 item
.X
:= MousePos
.X
-MapOffset
.X
;
3271 item
.Y
:= MousePos
.Y
-MapOffset
.Y
;
3273 if not (ssCtrl
in Shift
) then
3275 item
.X
:= item
.X
- (ItemSize
[item
.ItemType
][0] div 2);
3276 item
.Y
:= item
.Y
- ItemSize
[item
.ItemType
][1];
3279 item
.OnlyDM
:= cbOnlyDM
.Checked
;
3280 item
.Fall
:= cbFall
.Checked
;
3281 Undo_Add(OBJECT_ITEM
, AddItem(item
));
3284 if lbMonsterList
.ItemIndex
= -1 then
3285 ErrorMessageBox(_lc
[I_MSG_CHOOSE_MONSTER
])
3288 monster
.MonsterType
:= lbMonsterList
.ItemIndex
+ MONSTER_DEMON
;
3289 monster
.X
:= MousePos
.X
-MapOffset
.X
;
3290 monster
.Y
:= MousePos
.Y
-MapOffset
.Y
;
3292 if not (ssCtrl
in Shift
) then
3294 monster
.X
:= monster
.X
- (MonsterSize
[monster
.MonsterType
].Width
div 2);
3295 monster
.Y
:= monster
.Y
- MonsterSize
[monster
.MonsterType
].Height
;
3298 if rbMonsterLeft
.Checked
then
3299 monster
.Direction
:= D_LEFT
3301 monster
.Direction
:= D_RIGHT
;
3302 Undo_Add(OBJECT_MONSTER
, AddMonster(monster
));
3305 if lbAreasList
.ItemIndex
= -1 then
3306 ErrorMessageBox(_lc
[I_MSG_CHOOSE_AREA
])
3308 if (lbAreasList
.ItemIndex
+ 1) <> AREA_DOMFLAG
then
3310 area
.AreaType
:= lbAreasList
.ItemIndex
+ AREA_PLAYERPOINT1
;
3311 area
.X
:= MousePos
.X
-MapOffset
.X
;
3312 area
.Y
:= MousePos
.Y
-MapOffset
.Y
;
3314 if not (ssCtrl
in Shift
) then
3316 area
.X
:= area
.X
- (AreaSize
[area
.AreaType
].Width
div 2);
3317 area
.Y
:= area
.Y
- AreaSize
[area
.AreaType
].Height
;
3320 if rbAreaLeft
.Checked
then
3321 area
.Direction
:= D_LEFT
3323 area
.Direction
:= D_RIGHT
;
3324 Undo_Add(OBJECT_AREA
, AddArea(area
));
3330 i
:= GetFirstSelected();
3332 // Выбираем объект под текущим:
3333 if (SelectedObjects
<> nil) and
3334 (ssShift
in Shift
) and (i
>= 0) and
3335 (SelectedObjects
[i
].Live
) then
3337 if SelectedObjectCount() = 1 then
3338 SelectNextObject(X
-MapOffset
.X
, Y
-MapOffset
.Y
,
3339 SelectedObjects
[i
].ObjectType
,
3340 SelectedObjects
[i
].ID
);
3344 // Рисуем область триггера "Расширитель":
3345 if DrawPressRect
and (i
>= 0) and
3346 (SelectedObjects
[i
].ObjectType
= OBJECT_TRIGGER
) and
3347 (gTriggers
[SelectedObjects
[i
].ID
].TriggerType
in
3348 [TRIGGER_PRESS
, TRIGGER_ON
, TRIGGER_OFF
, TRIGGER_ONOFF
]) then
3349 MouseAction
:= MOUSEACTION_DRAWPRESS
3350 else // Рисуем панель:
3351 if pcObjects
.ActivePageIndex
= 0 then
3353 if (lbPanelType
.ItemIndex
>= 0) then
3354 MouseAction
:= MOUSEACTION_DRAWPANEL
3356 else // Рисуем триггер:
3357 if (lbTriggersList
.ItemIndex
>= 0) then
3359 MouseAction
:= MOUSEACTION_DRAWTRIGGER
;
3363 end; // if Button = mbLeft
3365 if Button
= mbRight
then // Right Mouse Button
3367 // Клик по мини-карте:
3369 g_CollidePoint(X
, Y
,
3370 RenderPanel
.Width
-max(gMapInfo
.Width
div (16 div Scale
), 1)-1,
3372 max(gMapInfo
.Width
div (16 div Scale
), 1),
3373 max(gMapInfo
.Height
div (16 div Scale
), 1) ) then
3375 MouseAction
:= MOUSEACTION_NOACTION
;
3377 else // Нужно что-то выбрать мышью:
3378 if SelectFlag
<> SELECTFLAG_NONE
then
3381 SELECTFLAG_TELEPORT
:
3382 // Точку назначения телепортации:
3383 with gTriggers
[SelectedObjects
[
3384 GetFirstSelected() ].ID
].Data
.TargetPoint
do
3386 X
:= MousePos
.X
-MapOffset
.X
;
3387 Y
:= MousePos
.Y
-MapOffset
.Y
;
3390 SELECTFLAG_SPAWNPOINT
:
3391 // Точку создания монстра:
3392 with gTriggers
[SelectedObjects
[GetFirstSelected()].ID
] do
3393 if TriggerType
= TRIGGER_SPAWNMONSTER
then
3395 Data
.MonPos
.X
:= MousePos
.X
-MapOffset
.X
;
3396 Data
.MonPos
.Y
:= MousePos
.Y
-MapOffset
.Y
;
3398 else if TriggerType
= TRIGGER_SPAWNITEM
then
3399 begin // Точка создания предмета:
3400 Data
.ItemPos
.X
:= MousePos
.X
-MapOffset
.X
;
3401 Data
.ItemPos
.Y
:= MousePos
.Y
-MapOffset
.Y
;
3403 else if TriggerType
= TRIGGER_SHOT
then
3404 begin // Точка создания выстрела:
3405 Data
.ShotPos
.X
:= MousePos
.X
-MapOffset
.X
;
3406 Data
.ShotPos
.Y
:= MousePos
.Y
-MapOffset
.Y
;
3412 IDArray
:= ObjectInRect(X
-MapOffset
.X
,
3414 2, 2, OBJECT_PANEL
, True);
3415 if IDArray
<> nil then
3417 for i
:= 0 to High(IDArray
) do
3418 if (gPanels
[IDArray
[i
]].PanelType
= PANEL_OPENDOOR
) or
3419 (gPanels
[IDArray
[i
]].PanelType
= PANEL_CLOSEDOOR
) then
3421 gTriggers
[SelectedObjects
[
3422 GetFirstSelected() ].ID
].Data
.PanelID
:= IDArray
[i
];
3427 gTriggers
[SelectedObjects
[
3428 GetFirstSelected() ].ID
].Data
.PanelID
:= -1;
3432 // Панель с текстурой:
3434 IDArray
:= ObjectInRect(X
-MapOffset
.X
,
3436 2, 2, OBJECT_PANEL
, True);
3437 if IDArray
<> nil then
3439 for i
:= 0 to High(IDArray
) do
3440 if ((gPanels
[IDArray
[i
]].PanelType
in
3441 [PANEL_WALL
, PANEL_BACK
, PANEL_FORE
,
3442 PANEL_WATER
, PANEL_ACID1
, PANEL_ACID2
,
3444 (gPanels
[IDArray
[i
]].PanelType
= PANEL_OPENDOOR
) or
3445 (gPanels
[IDArray
[i
]].PanelType
= PANEL_CLOSEDOOR
)) and
3446 (gPanels
[IDArray
[i
]].TextureName
<> '') then
3448 gTriggers
[SelectedObjects
[
3449 GetFirstSelected() ].ID
].TexturePanel
:= IDArray
[i
];
3454 gTriggers
[SelectedObjects
[
3455 GetFirstSelected() ].ID
].TexturePanel
:= -1;
3461 IDArray
:= ObjectInRect(X
-MapOffset
.X
,
3463 2, 2, OBJECT_PANEL
, True);
3464 if IDArray
<> nil then
3466 for i
:= 0 to High(IDArray
) do
3467 if (gPanels
[IDArray
[i
]].PanelType
= PANEL_LIFTUP
) or
3468 (gPanels
[IDArray
[i
]].PanelType
= PANEL_LIFTDOWN
) or
3469 (gPanels
[IDArray
[i
]].PanelType
= PANEL_LIFTLEFT
) or
3470 (gPanels
[IDArray
[i
]].PanelType
= PANEL_LIFTRIGHT
) then
3472 gTriggers
[SelectedObjects
[
3473 GetFirstSelected() ].ID
].Data
.PanelID
:= IDArray
[i
];
3478 gTriggers
[SelectedObjects
[
3479 GetFirstSelected() ].ID
].Data
.PanelID
:= -1;
3485 IDArray
:= ObjectInRect(X
-MapOffset
.X
,
3487 2, 2, OBJECT_MONSTER
, False);
3488 if IDArray
<> nil then
3489 gTriggers
[SelectedObjects
[
3490 GetFirstSelected() ].ID
].Data
.MonsterID
:= IDArray
[0]+1
3492 gTriggers
[SelectedObjects
[
3493 GetFirstSelected() ].ID
].Data
.MonsterID
:= 0;
3496 SELECTFLAG_SHOTPANEL
:
3497 // Панель индикации выстрела:
3499 if gTriggers
[SelectedObjects
[
3500 GetFirstSelected() ].ID
].TriggerType
= TRIGGER_SHOT
then
3502 IDArray
:= ObjectInRect(X
-MapOffset
.X
,
3504 2, 2, OBJECT_PANEL
, True);
3505 if IDArray
<> nil then
3507 for i
:= 0 to High(IDArray
) do
3508 if ((gPanels
[IDArray
[i
]].PanelType
in
3509 [PANEL_WALL
, PANEL_BACK
, PANEL_FORE
,
3510 PANEL_WATER
, PANEL_ACID1
, PANEL_ACID2
,
3512 (gPanels
[IDArray
[i
]].PanelType
= PANEL_OPENDOOR
) or
3513 (gPanels
[IDArray
[i
]].PanelType
= PANEL_CLOSEDOOR
)) and
3514 (gPanels
[IDArray
[i
]].TextureName
<> '') then
3516 gTriggers
[SelectedObjects
[
3517 GetFirstSelected() ].ID
].Data
.ShotPanelID
:= IDArray
[i
];
3522 gTriggers
[SelectedObjects
[
3523 GetFirstSelected() ].ID
].Data
.ShotPanelID
:= -1;
3528 SelectFlag
:= SELECTFLAG_SELECTED
;
3530 else // if SelectFlag <> SELECTFLAG_NONE...
3532 // Что уже выбрано и не нажат Ctrl:
3533 if (SelectedObjects
<> nil) and
3534 (not (ssCtrl
in Shift
)) then
3535 for i
:= 0 to High(SelectedObjects
) do
3536 with SelectedObjects
[i
] do
3539 if (ObjectType
in [OBJECT_PANEL
, OBJECT_TRIGGER
]) and
3540 (SelectedObjectCount() = 1) then
3542 Rect
:= ObjectGetRect(ObjectType
, ID
);
3544 c1
:= g_Collide(X
-MapOffset
.X
-1, Y
-MapOffset
.Y
-1, 2, 2,
3545 Rect
.X
-2, Rect
.Y
+(Rect
.Height
div 2)-2, 4, 4);
3546 c2
:= g_Collide(X
-MapOffset
.X
-1, Y
-MapOffset
.Y
-1, 2, 2,
3547 Rect
.X
+Rect
.Width
-3, Rect
.Y
+(Rect
.Height
div 2)-2, 4, 4);
3548 c3
:= g_Collide(X
-MapOffset
.X
-1, Y
-MapOffset
.Y
-1, 2, 2,
3549 Rect
.X
+(Rect
.Width
div 2)-2, Rect
.Y
-2, 4, 4);
3550 c4
:= g_Collide(X
-MapOffset
.X
-1, Y
-MapOffset
.Y
-1, 2, 2,
3551 Rect
.X
+(Rect
.Width
div 2)-2, Rect
.Y
+Rect
.Height
-3, 4, 4);
3553 // Меняем размер панели или триггера:
3554 if c1
or c2
or c3
or c4
then
3556 MouseAction
:= MOUSEACTION_RESIZE
;
3557 LastMovePoint
:= MousePos
;
3561 ResizeType
:= RESIZETYPE_HORIZONTAL
;
3563 ResizeDirection
:= RESIZEDIR_LEFT
3565 ResizeDirection
:= RESIZEDIR_RIGHT
;
3566 RenderPanel
.Cursor
:= crSizeWE
;
3570 ResizeType
:= RESIZETYPE_VERTICAL
;
3572 ResizeDirection
:= RESIZEDIR_UP
3574 ResizeDirection
:= RESIZEDIR_DOWN
;
3575 RenderPanel
.Cursor
:= crSizeNS
;
3582 // Перемещаем панель или триггер:
3583 if ObjectCollide(ObjectType
, ID
,
3585 Y
-MapOffset
.Y
-1, 2, 2) then
3587 MouseAction
:= MOUSEACTION_MOVEOBJ
;
3588 LastMovePoint
:= MousePos
;
3594 end; // if Button = mbRight
3596 MouseRDown
:= Button
= mbRight
;
3598 MouseRDownPos
:= MousePos
;
3600 MouseLDown
:= Button
= mbLeft
;
3602 MouseLDownPos
:= MousePos
;
3605 procedure TMainForm
.RenderPanelMouseUp(Sender
: TObject
;
3606 Button
: TMouseButton
; Shift
: TShiftState
; X
, Y
: Integer);
3611 rSelectRect
: Boolean;
3612 wWidth
, wHeight
: Word;
3615 procedure SelectObjects(ObjectType
: Byte);
3620 IDArray
:= ObjectInRect(rRect
.X
, rRect
.Y
,
3621 rRect
.Width
, rRect
.Height
,
3622 ObjectType
, rSelectRect
);
3624 if IDArray
<> nil then
3625 for i
:= 0 to High(IDArray
) do
3626 SelectObject(ObjectType
, IDArray
[i
], (ssCtrl
in Shift
) or rSelectRect
);
3629 if Button
= mbLeft
then
3630 MouseLDown
:= False;
3631 if Button
= mbRight
then
3632 MouseRDown
:= False;
3635 ResizeType
:= RESIZETYPE_NONE
;
3638 if Button
= mbLeft
then // Left Mouse Button
3640 if MouseAction
<> MOUSEACTION_NONE
then
3641 begin // Было действие мышью
3642 // Мышь сдвинулась во время удержания клавиши,
3643 // либо активирован режим быстрого рисования:
3644 if ((MousePos
.X
<> MouseLDownPos
.X
) and
3645 (MousePos
.Y
<> MouseLDownPos
.Y
)) or
3646 ((MouseAction
in [MOUSEACTION_DRAWPANEL
, MOUSEACTION_DRAWTRIGGER
]) and
3647 (ssCtrl
in Shift
)) then
3650 MOUSEACTION_DRAWPANEL
:
3652 // Фон или передний план без текстуры - ошибка:
3653 if (lbPanelType
.ItemIndex
in [1, 2]) and
3654 (lbTextureList
.ItemIndex
= -1) then
3655 ErrorMessageBox(_lc
[I_MSG_CHOOSE_TEXTURE
])
3656 else // Назначаем параметры панели:
3658 case lbPanelType
.ItemIndex
of
3659 0: Panel
.PanelType
:= PANEL_WALL
;
3660 1: Panel
.PanelType
:= PANEL_BACK
;
3661 2: Panel
.PanelType
:= PANEL_FORE
;
3662 3: Panel
.PanelType
:= PANEL_OPENDOOR
;
3663 4: Panel
.PanelType
:= PANEL_CLOSEDOOR
;
3664 5: Panel
.PanelType
:= PANEL_STEP
;
3665 6: Panel
.PanelType
:= PANEL_WATER
;
3666 7: Panel
.PanelType
:= PANEL_ACID1
;
3667 8: Panel
.PanelType
:= PANEL_ACID2
;
3668 9: Panel
.PanelType
:= PANEL_LIFTUP
;
3669 10: Panel
.PanelType
:= PANEL_LIFTDOWN
;
3670 11: Panel
.PanelType
:= PANEL_LIFTLEFT
;
3671 12: Panel
.PanelType
:= PANEL_LIFTRIGHT
;
3672 13: Panel
.PanelType
:= PANEL_BLOCKMON
;
3675 Panel
.X
:= Min(MousePos
.X
-MapOffset
.X
, MouseLDownPos
.X
-MapOffset
.X
);
3676 Panel
.Y
:= Min(MousePos
.Y
-MapOffset
.Y
, MouseLDownPos
.Y
-MapOffset
.Y
);
3677 if ssCtrl
in Shift
then
3681 if (lbTextureList
.ItemIndex
<> -1) and
3682 (not IsSpecialTextureSel()) then
3684 if not g_GetTexture(SelectedTexture(), TextureID
) then
3685 g_GetTexture('NOTEXTURE', TextureID
);
3686 g_GetTextureSizeByID(TextureID
, wWidth
, wHeight
);
3688 Panel
.Width
:= wWidth
;
3689 Panel
.Height
:= wHeight
;
3693 Panel
.Width
:= Abs(MousePos
.X
-MouseLDownPos
.X
);
3694 Panel
.Height
:= Abs(MousePos
.Y
-MouseLDownPos
.Y
);
3697 // Лифты, блокМон или отсутствие текстуры - пустая текстура:
3698 if (lbPanelType
.ItemIndex
in [9, 10, 11, 12, 13]) or
3699 (lbTextureList
.ItemIndex
= -1) then
3701 Panel
.TextureHeight
:= 1;
3702 Panel
.TextureWidth
:= 1;
3703 Panel
.TextureName
:= '';
3704 Panel
.TextureID
:= TEXTURE_SPECIAL_NONE
;
3706 else // Есть текстура:
3708 Panel
.TextureName
:= SelectedTexture();
3710 // Обычная текстура:
3711 if not IsSpecialTextureSel() then
3713 g_GetTextureSizeByName(Panel
.TextureName
,
3714 Panel
.TextureWidth
, Panel
.TextureHeight
);
3715 g_GetTexture(Panel
.TextureName
, Panel
.TextureID
);
3717 else // Спец.текстура:
3719 Panel
.TextureHeight
:= 1;
3720 Panel
.TextureWidth
:= 1;
3721 Panel
.TextureID
:= SpecialTextureID(SelectedTexture());
3726 Panel
.Blending
:= False;
3728 Undo_Add(OBJECT_PANEL
, AddPanel(Panel
));
3732 // Рисовали триггер:
3733 MOUSEACTION_DRAWTRIGGER
:
3735 trigger
.X
:= Min(MousePos
.X
-MapOffset
.X
, MouseLDownPos
.X
-MapOffset
.X
);
3736 trigger
.Y
:= Min(MousePos
.Y
-MapOffset
.Y
, MouseLDownPos
.Y
-MapOffset
.Y
);
3737 if ssCtrl
in Shift
then
3741 trigger
.Width
:= wWidth
;
3742 trigger
.Height
:= wHeight
;
3746 trigger
.Width
:= Abs(MousePos
.X
-MouseLDownPos
.X
);
3747 trigger
.Height
:= Abs(MousePos
.Y
-MouseLDownPos
.Y
);
3750 trigger
.Enabled
:= True;
3751 trigger
.TriggerType
:= lbTriggersList
.ItemIndex
+1;
3752 trigger
.TexturePanel
:= -1;
3755 trigger
.ActivateType
:= 0;
3757 if clbActivationType
.Checked
[0] then
3758 trigger
.ActivateType
:= Trigger
.ActivateType
or ACTIVATE_PLAYERCOLLIDE
;
3759 if clbActivationType
.Checked
[1] then
3760 trigger
.ActivateType
:= Trigger
.ActivateType
or ACTIVATE_MONSTERCOLLIDE
;
3761 if clbActivationType
.Checked
[2] then
3762 trigger
.ActivateType
:= Trigger
.ActivateType
or ACTIVATE_PLAYERPRESS
;
3763 if clbActivationType
.Checked
[3] then
3764 trigger
.ActivateType
:= Trigger
.ActivateType
or ACTIVATE_MONSTERPRESS
;
3765 if clbActivationType
.Checked
[4] then
3766 trigger
.ActivateType
:= Trigger
.ActivateType
or ACTIVATE_SHOT
;
3767 if clbActivationType
.Checked
[5] then
3768 trigger
.ActivateType
:= Trigger
.ActivateType
or ACTIVATE_NOMONSTER
;
3770 // Необходимые для активации ключи:
3773 if clbKeys
.Checked
[0] then
3774 trigger
.Key
:= Trigger
.Key
or KEY_RED
;
3775 if clbKeys
.Checked
[1] then
3776 trigger
.Key
:= Trigger
.Key
or KEY_GREEN
;
3777 if clbKeys
.Checked
[2] then
3778 trigger
.Key
:= Trigger
.Key
or KEY_BLUE
;
3779 if clbKeys
.Checked
[3] then
3780 trigger
.Key
:= Trigger
.Key
or KEY_REDTEAM
;
3781 if clbKeys
.Checked
[4] then
3782 trigger
.Key
:= Trigger
.Key
or KEY_BLUETEAM
;
3784 // Параметры триггера:
3785 FillByte(trigger
.Data
.Default
[0], 128, 0);
3787 case trigger
.TriggerType
of
3788 // Переключаемая панель:
3789 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
,
3790 TRIGGER_DOOR5
, TRIGGER_CLOSETRAP
, TRIGGER_TRAP
,
3791 TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
:
3793 Trigger
.Data
.PanelID
:= -1;
3799 trigger
.Data
.TargetPoint
.X
:= trigger
.X
-64;
3800 trigger
.Data
.TargetPoint
.Y
:= trigger
.Y
-64;
3801 trigger
.Data
.d2d_teleport
:= True;
3802 trigger
.Data
.TlpDir
:= 0;
3805 // Изменение других триггеров:
3806 TRIGGER_PRESS
, TRIGGER_ON
, TRIGGER_OFF
,
3809 trigger
.Data
.Count
:= 1;
3815 trigger
.Data
.Volume
:= 255;
3816 trigger
.Data
.Pan
:= 127;
3817 trigger
.Data
.PlayCount
:= 1;
3818 trigger
.Data
.Local
:= True;
3819 trigger
.Data
.SoundSwitch
:= False;
3825 trigger
.Data
.MusicAction
:= 1;
3828 // Создание монстра:
3829 TRIGGER_SPAWNMONSTER
:
3831 trigger
.Data
.MonType
:= MONSTER_ZOMBY
;
3832 trigger
.Data
.MonPos
.X
:= trigger
.X
-64;
3833 trigger
.Data
.MonPos
.Y
:= trigger
.Y
-64;
3834 trigger
.Data
.MonHealth
:= 0;
3835 trigger
.Data
.MonActive
:= False;
3836 trigger
.Data
.MonCount
:= 1;
3839 // Создание предмета:
3842 trigger
.Data
.ItemType
:= ITEM_AMMO_BULLETS
;
3843 trigger
.Data
.ItemPos
.X
:= trigger
.X
-64;
3844 trigger
.Data
.ItemPos
.Y
:= trigger
.Y
-64;
3845 trigger
.Data
.ItemOnlyDM
:= False;
3846 trigger
.Data
.ItemFalls
:= False;
3847 trigger
.Data
.ItemCount
:= 1;
3848 trigger
.Data
.ItemMax
:= 0;
3849 trigger
.Data
.ItemDelay
:= 0;
3855 trigger
.Data
.PushAngle
:= 90;
3856 trigger
.Data
.PushForce
:= 10;
3857 trigger
.Data
.ResetVel
:= True;
3862 trigger
.Data
.ScoreCount
:= 1;
3863 trigger
.Data
.ScoreCon
:= True;
3864 trigger
.Data
.ScoreMsg
:= True;
3869 trigger
.Data
.MessageKind
:= 0;
3870 trigger
.Data
.MessageSendTo
:= 0;
3871 trigger
.Data
.MessageText
:= '';
3872 trigger
.Data
.MessageTime
:= 144;
3877 trigger
.Data
.DamageValue
:= 5;
3878 trigger
.Data
.DamageInterval
:= 12;
3883 trigger
.Data
.HealValue
:= 5;
3884 trigger
.Data
.HealInterval
:= 36;
3889 trigger
.Data
.ShotType
:= TRIGGER_SHOT_BULLET
;
3890 trigger
.Data
.ShotSound
:= True;
3891 trigger
.Data
.ShotPanelID
:= -1;
3892 trigger
.Data
.ShotTarget
:= 0;
3893 trigger
.Data
.ShotIntSight
:= 0;
3894 trigger
.Data
.ShotAim
:= TRIGGER_SHOT_AIM_DEFAULT
;
3895 trigger
.Data
.ShotPos
.X
:= trigger
.X
-64;
3896 trigger
.Data
.ShotPos
.Y
:= trigger
.Y
-64;
3897 trigger
.Data
.ShotAngle
:= 0;
3898 trigger
.Data
.ShotWait
:= 18;
3899 trigger
.Data
.ShotAccuracy
:= 0;
3900 trigger
.Data
.ShotAmmo
:= 0;
3901 trigger
.Data
.ShotIntReload
:= 0;
3906 trigger
.Data
.FXCount
:= 1;
3907 trigger
.Data
.FXType
:= TRIGGER_EFFECT_PARTICLE
;
3908 trigger
.Data
.FXSubType
:= TRIGGER_EFFECT_SLIQUID
;
3909 trigger
.Data
.FXColorR
:= 0;
3910 trigger
.Data
.FXColorG
:= 0;
3911 trigger
.Data
.FXColorB
:= 255;
3912 trigger
.Data
.FXPos
:= TRIGGER_EFFECT_POS_CENTER
;
3913 trigger
.Data
.FXWait
:= 1;
3914 trigger
.Data
.FXVelX
:= 0;
3915 trigger
.Data
.FXVelY
:= -20;
3916 trigger
.Data
.FXSpreadL
:= 5;
3917 trigger
.Data
.FXSpreadR
:= 5;
3918 trigger
.Data
.FXSpreadU
:= 4;
3919 trigger
.Data
.FXSpreadD
:= 0;
3923 Undo_Add(OBJECT_TRIGGER
, AddTrigger(trigger
));
3926 // Рисовали область триггера "Расширитель":
3927 MOUSEACTION_DRAWPRESS
:
3928 with gTriggers
[SelectedObjects
[GetFirstSelected
].ID
] do
3930 Data
.tX
:= Min(MousePos
.X
-MapOffset
.X
, MouseLDownPos
.X
-MapOffset
.X
);
3931 Data
.tY
:= Min(MousePos
.Y
-MapOffset
.Y
, MouseLDownPos
.Y
-MapOffset
.Y
);
3932 Data
.tWidth
:= Abs(MousePos
.X
-MouseLDownPos
.X
);
3933 Data
.tHeight
:= Abs(MousePos
.Y
-MouseLDownPos
.Y
);
3935 DrawPressRect
:= False;
3939 MouseAction
:= MOUSEACTION_NONE
;
3941 end // if Button = mbLeft...
3942 else // Right Mouse Button:
3944 if MouseAction
= MOUSEACTION_NOACTION
then
3946 MouseAction
:= MOUSEACTION_NONE
;
3950 // Объект передвинут или изменен в размере:
3951 if MouseAction
in [MOUSEACTION_MOVEOBJ
, MOUSEACTION_RESIZE
] then
3953 MouseAction
:= MOUSEACTION_NONE
;
3958 // Еще не все выбрали:
3959 if SelectFlag
<> SELECTFLAG_NONE
then
3961 if SelectFlag
= SELECTFLAG_SELECTED
then
3962 SelectFlag
:= SELECTFLAG_NONE
;
3967 // Мышь сдвинулась во время удержания клавиши:
3968 if (MousePos
.X
<> MouseRDownPos
.X
) and
3969 (MousePos
.Y
<> MouseRDownPos
.Y
) then
3971 rSelectRect
:= True;
3973 rRect
.X
:= Min(MousePos
.X
, MouseRDownPos
.X
)-MapOffset
.X
;
3974 rRect
.Y
:= Min(MousePos
.Y
, MouseRDownPos
.Y
)-MapOffset
.Y
;
3975 rRect
.Width
:= Abs(MousePos
.X
-MouseRDownPos
.X
);
3976 rRect
.Height
:= Abs(MousePos
.Y
-MouseRDownPos
.Y
);
3978 else // Мышь не сдвинулась - нет прямоугольника:
3980 rSelectRect
:= False;
3982 rRect
.X
:= X
-MapOffset
.X
-1;
3983 rRect
.Y
:= Y
-MapOffset
.Y
-1;
3988 // Если зажат Ctrl - выделять еще, иначе только один выделенный объект:
3989 if not (ssCtrl
in Shift
) then
3990 RemoveSelectFromObjects();
3992 // Выделяем всё в выбранном прямоугольнике:
3993 if (ssCtrl
in Shift
) and (ssAlt
in Shift
) then
3995 SelectObjects(OBJECT_PANEL
);
3996 SelectObjects(OBJECT_ITEM
);
3997 SelectObjects(OBJECT_MONSTER
);
3998 SelectObjects(OBJECT_AREA
);
3999 SelectObjects(OBJECT_TRIGGER
);
4002 SelectObjects(pcObjects
.ActivePageIndex
+1);
4008 procedure TMainForm
.RenderPanelPaint(Sender
: TObject
);
4013 function TMainForm
.RenderMousePos(): Types
.TPoint
;
4015 Result
:= RenderPanel
.ScreenToClient(Mouse
.CursorPos
);
4018 procedure TMainForm
.RecountSelectedObjects();
4020 if SelectedObjectCount() = 0 then
4021 StatusBar
.Panels
[0].Text := ''
4023 StatusBar
.Panels
[0].Text := Format(_lc
[I_CAP_STAT_SELECTED
], [SelectedObjectCount()]);
4026 procedure TMainForm
.RenderPanelMouseMove(Sender
: TObject
;
4027 Shift
: TShiftState
; X
, Y
: Integer);
4030 dWidth
, dHeight
: Integer;
4033 wWidth
, wHeight
: Word;
4035 _id
:= GetFirstSelected();
4038 // Рисуем панель с текстурой, сетка - размеры текстуры:
4039 if (MouseAction
= MOUSEACTION_DRAWPANEL
) and
4040 (lbPanelType
.ItemIndex
in [0..8]) and
4041 (lbTextureList
.ItemIndex
<> -1) and
4042 (not IsSpecialTextureSel()) then
4044 sX
:= StrToIntDef(lTextureWidth
.Caption
, DotStep
);
4045 sY
:= StrToIntDef(lTextureHeight
.Caption
, DotStep
);
4048 // Меняем размер панели с текстурой, сетка - размеры текстуры:
4049 if (MouseAction
= MOUSEACTION_RESIZE
) and
4050 ( (SelectedObjects
[_id
].ObjectType
= OBJECT_PANEL
) and
4051 IsTexturedPanel(gPanels
[SelectedObjects
[_id
].ID
].PanelType
) and
4052 (gPanels
[SelectedObjects
[_id
].ID
].TextureName
<> '') and
4053 (not IsSpecialTexture(gPanels
[SelectedObjects
[_id
].ID
].TextureName
)) ) then
4055 sX
:= gPanels
[SelectedObjects
[_id
].ID
].TextureWidth
;
4056 sY
:= gPanels
[SelectedObjects
[_id
].ID
].TextureHeight
;
4059 // Выравнивание по сетке:
4065 else // Нет выравнивания по сетке:
4071 // Новая позиция мыши:
4073 begin // Зажата левая кнопка мыши
4074 MousePos
.X
:= (Round((X
-MouseLDownPos
.X
)/sX
)*sX
)+MouseLDownPos
.X
;
4075 MousePos
.Y
:= (Round((Y
-MouseLDownPos
.Y
)/sY
)*sY
)+MouseLDownPos
.Y
;
4079 begin // Зажата правая кнопка мыши
4080 MousePos
.X
:= (Round((X
-MouseRDownPos
.X
)/sX
)*sX
)+MouseRDownPos
.X
;
4081 MousePos
.Y
:= (Round((Y
-MouseRDownPos
.Y
)/sY
)*sY
)+MouseRDownPos
.Y
;
4084 begin // Кнопки мыши не зажаты
4085 MousePos
.X
:= (Round(X
/sX
)*sX
);
4086 MousePos
.Y
:= (Round(Y
/sY
)*sY
);
4089 // Изменение размера закончилось - ставим обычный курсор:
4090 if ResizeType
= RESIZETYPE_NONE
then
4091 RenderPanel
.Cursor
:= crDefault
;
4093 // Зажата только правая кнопка мыши:
4094 if (not MouseLDown
) and (MouseRDown
) then
4096 // Рисуем прямоугольник выделения:
4097 if MouseAction
= MOUSEACTION_NONE
then
4099 if DrawRect
= nil then
4101 DrawRect
.Top
:= MouseRDownPos
.y
;
4102 DrawRect
.Left
:= MouseRDownPos
.x
;
4103 DrawRect
.Bottom
:= MousePos
.y
;
4104 DrawRect
.Right
:= MousePos
.x
;
4107 // Двигаем выделенные объекты:
4108 if MouseAction
= MOUSEACTION_MOVEOBJ
then
4110 MoveSelectedObjects(ssShift
in Shift
, ssCtrl
in Shift
,
4111 MousePos
.X
-LastMovePoint
.X
+WASDOffset
.X
,
4112 MousePos
.Y
-LastMovePoint
.Y
+WASDOffset
.Y
);
4117 // Меняем размер выделенного объекта:
4118 if MouseAction
= MOUSEACTION_RESIZE
then
4120 if (SelectedObjectCount
= 1) and
4121 (SelectedObjects
[GetFirstSelected
].Live
) then
4123 dWidth
:= MousePos
.X
-LastMovePoint
.X
+WASDOffset
.X
;
4124 dHeight
:= MousePos
.Y
-LastMovePoint
.Y
+WASDOffset
.Y
;
4129 RESIZETYPE_VERTICAL
: dWidth
:= 0;
4130 RESIZETYPE_HORIZONTAL
: dHeight
:= 0;
4133 case ResizeDirection
of
4134 RESIZEDIR_UP
: dHeight
:= -dHeight
;
4135 RESIZEDIR_LEFT
: dWidth
:= -dWidth
;
4138 ResizeObject(SelectedObjects
[GetFirstSelected
].ObjectType
,
4139 SelectedObjects
[GetFirstSelected
].ID
,
4140 dWidth
, dHeight
, ResizeDirection
);
4142 LastMovePoint
:= MousePos
;
4147 // Зажата только левая кнопка мыши:
4148 if (not MouseRDown
) and (MouseLDown
) then
4150 // Рисуем прямоугольник планирования панели:
4151 if MouseAction
in [MOUSEACTION_DRAWPANEL
,
4152 MOUSEACTION_DRAWTRIGGER
,
4153 MOUSEACTION_DRAWPRESS
] then
4155 if DrawRect
= nil then
4157 if ssCtrl
in Shift
then
4161 if (lbTextureList
.ItemIndex
<> -1) and (not IsSpecialTextureSel()) and
4162 (MouseAction
= MOUSEACTION_DRAWPANEL
) then
4164 if not g_GetTexture(SelectedTexture(), TextureID
) then
4165 g_GetTexture('NOTEXTURE', TextureID
);
4166 g_GetTextureSizeByID(TextureID
, wWidth
, wHeight
);
4168 DrawRect
.Top
:= MouseLDownPos
.y
;
4169 DrawRect
.Left
:= MouseLDownPos
.x
;
4170 DrawRect
.Bottom
:= DrawRect
.Top
+ wHeight
;
4171 DrawRect
.Right
:= DrawRect
.Left
+ wWidth
;
4175 DrawRect
.Top
:= MouseLDownPos
.y
;
4176 DrawRect
.Left
:= MouseLDownPos
.x
;
4177 DrawRect
.Bottom
:= MousePos
.y
;
4178 DrawRect
.Right
:= MousePos
.x
;
4181 else // Двигаем карту:
4182 if MouseAction
= MOUSEACTION_MOVEMAP
then
4188 // Клавиши мыши не зажаты:
4189 if (not MouseRDown
) and (not MouseLDown
) then
4192 // Строка состояния - координаты мыши:
4193 StatusBar
.Panels
[1].Text := Format('(%d:%d)',
4194 [MousePos
.X
-MapOffset
.X
, MousePos
.Y
-MapOffset
.Y
]);
4197 procedure TMainForm
.FormCloseQuery(Sender
: TObject
; var CanClose
: Boolean);
4199 CanClose
:= MessageBox(0, PChar(_lc
[I_MSG_EXIT_PROMT
]),
4200 PChar(_lc
[I_MSG_EXIT
]),
4201 MB_ICONQUESTION
or MB_YESNO
or
4202 MB_DEFBUTTON1
) = idYes
;
4205 procedure TMainForm
.aExitExecute(Sender
: TObject
);
4210 procedure TMainForm
.FormDestroy(Sender
: TObject
);
4215 config
:= TConfig
.CreateFile(EditorDir
+'Editor.cfg');
4217 if WindowState
<> wsMaximized
then
4219 config
.WriteInt('Editor', 'XPos', Left
);
4220 config
.WriteInt('Editor', 'YPos', Top
);
4221 config
.WriteInt('Editor', 'Width', Width
);
4222 config
.WriteInt('Editor', 'Height', Height
);
4226 config
.WriteInt('Editor', 'XPos', RestoredLeft
);
4227 config
.WriteInt('Editor', 'YPos', RestoredTop
);
4228 config
.WriteInt('Editor', 'Width', RestoredWidth
);
4229 config
.WriteInt('Editor', 'Height', RestoredHeight
);
4231 config
.WriteBool('Editor', 'Maximize', WindowState
= wsMaximized
);
4232 config
.WriteBool('Editor', 'Minimap', ShowMap
);
4233 config
.WriteInt('Editor', 'PanelProps', PanelProps
.ClientWidth
);
4234 config
.WriteInt('Editor', 'PanelObjs', PanelObjs
.ClientHeight
);
4235 config
.WriteBool('Editor', 'DotEnable', DotEnable
);
4236 config
.WriteInt('Editor', 'DotStep', DotStep
);
4237 config
.WriteStr('Editor', 'LastOpenDir', OpenDialog
.InitialDir
);
4238 config
.WriteStr('Editor', 'LastSaveDir', SaveDialog
.InitialDir
);
4239 config
.WriteBool('Editor', 'EdgeShow', drEdge
[3] < 255);
4240 config
.WriteInt('Editor', 'EdgeColor', gColorEdge
);
4241 config
.WriteInt('Editor', 'EdgeAlpha', gAlphaEdge
);
4242 config
.WriteInt('Editor', 'LineAlpha', gAlphaTriggerLine
);
4243 config
.WriteInt('Editor', 'TriggerAlpha', gAlphaTriggerArea
);
4244 config
.WriteInt('Editor', 'MonsterRectAlpha', gAlphaMonsterRect
);
4245 config
.WriteInt('Editor', 'AreaRectAlpha', gAlphaAreaRect
);
4247 for i
:= 0 to RecentCount
-1 do
4248 if i
< RecentFiles
.Count
then
4249 config
.WriteStr('RecentFiles', IntToStr(i
+1), RecentFiles
[i
])
4251 config
.WriteStr('RecentFiles', IntToStr(i
+1), '');
4254 config
.SaveFile(EditorDir
+'Editor.cfg');
4257 slInvalidTextures
.Free
;
4260 procedure TMainForm
.FormDropFiles(Sender
: TObject
;
4261 const FileNames
: array of String);
4263 if Length(FileNames
) <> 1 then
4266 OpenMapFile(FileNames
[0]);
4269 procedure TMainForm
.RenderPanelResize(Sender
: TObject
);
4271 if MainForm
.Visible
then
4275 procedure TMainForm
.Splitter1Moved(Sender
: TObject
);
4280 procedure TMainForm
.aMapOptionsExecute(Sender
: TObject
);
4284 MapOptionsForm
.ShowModal();
4286 ResName
:= OpenedMap
;
4287 while (Pos(':\', ResName
) > 0) do
4288 Delete(ResName
, 1, Pos(':\', ResName
) + 1);
4290 UpdateCaption(gMapInfo
.Name
, ExtractFileName(OpenedWAD
), ResName
);
4293 procedure TMainForm
.aAboutExecute(Sender
: TObject
);
4295 AboutForm
.ShowModal();
4298 procedure TMainForm
.FormKeyDown(Sender
: TObject
; var Key
: Word;
4299 Shift
: TShiftState
);
4304 if (not EditingProperties
) then
4306 if Key
= Ord('1') then
4307 SwitchLayer(LAYER_BACK
);
4308 if Key
= Ord('2') then
4309 SwitchLayer(LAYER_WALLS
);
4310 if Key
= Ord('3') then
4311 SwitchLayer(LAYER_FOREGROUND
);
4312 if Key
= Ord('4') then
4313 SwitchLayer(LAYER_STEPS
);
4314 if Key
= Ord('5') then
4315 SwitchLayer(LAYER_WATER
);
4316 if Key
= Ord('6') then
4317 SwitchLayer(LAYER_ITEMS
);
4318 if Key
= Ord('7') then
4319 SwitchLayer(LAYER_MONSTERS
);
4320 if Key
= Ord('8') then
4321 SwitchLayer(LAYER_AREAS
);
4322 if Key
= Ord('9') then
4323 SwitchLayer(LAYER_TRIGGERS
);
4324 if Key
= Ord('0') then
4325 tbShowClick(tbShow
);
4327 if Key
= Ord('V') then
4328 begin // Поворот монстров и областей:
4329 if (SelectedObjects
<> nil) then
4331 for i
:= 0 to High(SelectedObjects
) do
4332 if (SelectedObjects
[i
].Live
) then
4334 if (SelectedObjects
[i
].ObjectType
= OBJECT_MONSTER
) then
4336 g_ChangeDir(gMonsters
[SelectedObjects
[i
].ID
].Direction
);
4339 if (SelectedObjects
[i
].ObjectType
= OBJECT_AREA
) then
4341 g_ChangeDir(gAreas
[SelectedObjects
[i
].ID
].Direction
);
4347 if pcObjects
.ActivePage
= tsMonsters
then
4349 if rbMonsterLeft
.Checked
then
4350 rbMonsterRight
.Checked
:= True
4352 rbMonsterLeft
.Checked
:= True;
4354 if pcObjects
.ActivePage
= tsAreas
then
4356 if rbAreaLeft
.Checked
then
4357 rbAreaRight
.Checked
:= True
4359 rbAreaLeft
.Checked
:= True;
4364 if not (ssCtrl
in Shift
) then
4366 // Быстрое превью карты:
4367 if Key
= Ord('E') then
4369 if PreviewMode
= 0 then
4373 // Вертикальный скролл карты:
4376 if Key
= Ord('W') then
4378 if (MouseLDown
or MouseRDown
) and (Position
>= DotStep
) then
4380 Dec(WASDOffset
.Y
, DotStep
);
4381 RenderPanelMouseMove(Sender
, Shift
, RenderMousePos().X
, RenderMousePos().Y
);
4383 Position
:= IfThen(Position
> DotStep
, Position
-DotStep
, 0);
4384 MapOffset
.Y
:= -Round(Position
/16) * 16;
4387 if Key
= Ord('S') then
4389 if (MouseLDown
or MouseRDown
) and (Position
+DotStep
<= Max
) then
4391 Inc(WASDOffset
.Y
, DotStep
);
4392 RenderPanelMouseMove(Sender
, Shift
, RenderMousePos().X
, RenderMousePos().Y
);
4394 Position
:= IfThen(Position
+DotStep
< Max
, Position
+DotStep
, Max
);
4395 MapOffset
.Y
:= -Round(Position
/16) * 16;
4399 // Горизонтальный скролл карты:
4400 with sbHorizontal
do
4402 if Key
= Ord('A') then
4404 if (MouseLDown
or MouseRDown
) and (Position
>= DotStep
) then
4406 Dec(WASDOffset
.X
, DotStep
);
4407 RenderPanelMouseMove(Sender
, Shift
, RenderMousePos().X
, RenderMousePos().Y
);
4409 Position
:= IfThen(Position
> DotStep
, Position
-DotStep
, 0);
4410 MapOffset
.X
:= -Round(Position
/16) * 16;
4413 if Key
= Ord('D') then
4415 if (MouseLDown
or MouseRDown
) and (Position
+DotStep
<= Max
) then
4417 Inc(WASDOffset
.X
, DotStep
);
4418 RenderPanelMouseMove(Sender
, Shift
, RenderMousePos().X
, RenderMousePos().Y
);
4420 Position
:= IfThen(Position
+DotStep
< Max
, Position
+DotStep
, Max
);
4421 MapOffset
.X
:= -Round(Position
/16) * 16;
4425 else // ssCtrl in Shift
4427 if ssShift
in Shift
then
4429 // Вставка по абсолютному смещению:
4430 if Key
= Ord('V') then
4431 aPasteObjectExecute(Sender
);
4433 RenderPanelMouseMove(Sender
, Shift
, RenderMousePos().X
, RenderMousePos().Y
);
4437 // Удалить выделенные объекты:
4438 if (Key
= VK_DELETE
) and (SelectedObjects
<> nil) and
4439 RenderPanel
.Focused() then
4440 DeleteSelectedObjects();
4443 if (Key
= VK_ESCAPE
) and (SelectedObjects
<> nil) then
4444 RemoveSelectFromObjects();
4446 // Передвинуть объекты:
4447 if MainForm
.ActiveControl
= RenderPanel
then
4452 if Key
= VK_NUMPAD4
then
4453 dx
:= IfThen(ssAlt
in Shift
, -1, -DotStep
);
4454 if Key
= VK_NUMPAD6
then
4455 dx
:= IfThen(ssAlt
in Shift
, 1, DotStep
);
4456 if Key
= VK_NUMPAD8
then
4457 dy
:= IfThen(ssAlt
in Shift
, -1, -DotStep
);
4458 if Key
= VK_NUMPAD5
then
4459 dy
:= IfThen(ssAlt
in Shift
, 1, DotStep
);
4461 if (dx
<> 0) or (dy
<> 0) then
4463 MoveSelectedObjects(ssShift
in Shift
, ssCtrl
in Shift
, dx
, dy
);
4468 if ssCtrl
in Shift
then
4470 // Выбор панели с текстурой для триггера
4471 if Key
= Ord('T') then
4473 DrawPressRect
:= False;
4474 if SelectFlag
= SELECTFLAG_TEXTURE
then
4476 SelectFlag
:= SELECTFLAG_NONE
;
4479 vleObjectProperty
.FindRow(_lc
[I_PROP_TR_TEXTURE_PANEL
], i
);
4481 SelectFlag
:= SELECTFLAG_TEXTURE
;
4484 if Key
= Ord('D') then
4486 SelectFlag
:= SELECTFLAG_NONE
;
4487 if DrawPressRect
then
4489 DrawPressRect
:= False;
4494 // Выбор области воздействия, в зависимости от типа триггера
4495 vleObjectProperty
.FindRow(_lc
[I_PROP_TR_EX_AREA
], i
);
4498 DrawPressRect
:= True;
4501 vleObjectProperty
.FindRow(_lc
[I_PROP_TR_DOOR_PANEL
], i
);
4503 vleObjectProperty
.FindRow(_lc
[I_PROP_TR_TRAP_PANEL
], i
);
4506 SelectFlag
:= SELECTFLAG_DOOR
;
4509 vleObjectProperty
.FindRow(_lc
[I_PROP_TR_LIFT_PANEL
], i
);
4512 SelectFlag
:= SELECTFLAG_LIFT
;
4515 vleObjectProperty
.FindRow(_lc
[I_PROP_TR_TELEPORT_TO
], i
);
4518 SelectFlag
:= SELECTFLAG_TELEPORT
;
4521 vleObjectProperty
.FindRow(_lc
[I_PROP_TR_SPAWN_TO
], i
);
4524 SelectFlag
:= SELECTFLAG_SPAWNPOINT
;
4528 // Выбор основного параметра, в зависимости от типа триггера
4529 vleObjectProperty
.FindRow(_lc
[I_PROP_TR_NEXT_MAP
], i
);
4532 g_ProcessResourceStr(OpenedMap
, @FileName
, nil, nil);
4533 SelectMapForm
.Caption
:= _lc
[I_CAP_SELECT
];
4534 SelectMapForm
.GetMaps(FileName
);
4536 if SelectMapForm
.ShowModal() = mrOK
then
4538 vleObjectProperty
.Cells
[1, i
] := SelectMapForm
.lbMapList
.Items
[SelectMapForm
.lbMapList
.ItemIndex
];
4539 bApplyProperty
.Click();
4543 vleObjectProperty
.FindRow(_lc
[I_PROP_TR_SOUND_NAME
], i
);
4545 vleObjectProperty
.FindRow(_lc
[I_PROP_TR_MUSIC_NAME
], i
);
4548 AddSoundForm
.OKFunction
:= nil;
4549 AddSoundForm
.lbResourcesList
.MultiSelect
:= False;
4550 AddSoundForm
.SetResource
:= vleObjectProperty
.Cells
[1, i
];
4552 if (AddSoundForm
.ShowModal() = mrOk
) then
4554 vleObjectProperty
.Cells
[1, i
] := AddSoundForm
.ResourceName
;
4555 bApplyProperty
.Click();
4559 vleObjectProperty
.FindRow(_lc
[I_PROP_TR_PUSH_ANGLE
], i
);
4561 vleObjectProperty
.FindRow(_lc
[I_PROP_TR_MESSAGE_TEXT
], i
);
4564 vleObjectProperty
.Row
:= i
;
4565 vleObjectProperty
.SetFocus();
4572 procedure TMainForm
.aOptimizeExecute(Sender
: TObject
);
4574 RemoveSelectFromObjects();
4575 MapOptimizationForm
.ShowModal();
4578 procedure TMainForm
.aCheckMapExecute(Sender
: TObject
);
4580 MapCheckForm
.ShowModal();
4583 procedure TMainForm
.bbAddTextureClick(Sender
: TObject
);
4585 AddTextureForm
.lbResourcesList
.MultiSelect
:= True;
4586 AddTextureForm
.ShowModal();
4589 procedure TMainForm
.lbTextureListClick(Sender
: TObject
);
4592 TextureWidth
, TextureHeight
: Word;
4597 if (lbTextureList
.ItemIndex
<> -1) and
4598 (not IsSpecialTextureSel()) then
4600 if g_GetTexture(SelectedTexture(), TextureID
) then
4602 g_GetTextureSizeByID(TextureID
, TextureWidth
, TextureHeight
);
4604 lTextureWidth
.Caption
:= IntToStr(TextureWidth
);
4605 lTextureHeight
.Caption
:= IntToStr(TextureHeight
);
4608 lTextureWidth
.Caption
:= _lc
[I_NOT_ACCESSIBLE
];
4609 lTextureHeight
.Caption
:= _lc
[I_NOT_ACCESSIBLE
];
4614 lTextureWidth
.Caption
:= '';
4615 lTextureHeight
.Caption
:= '';
4619 procedure TMainForm
.lbTextureListDrawItem(Control
: TWinControl
; Index
: Integer;
4620 ARect
: TRect
; State
: TOwnerDrawState
);
4622 with Control
as TListBox
do
4624 if LCLType
.odSelected
in State
then
4626 Canvas
.Brush
.Color
:= clHighlight
;
4627 Canvas
.Font
.Color
:= clHighlightText
;
4629 if (Items
<> nil) and (Index
>= 0) then
4630 if slInvalidTextures
.IndexOf(Items
[Index
]) > -1 then
4632 Canvas
.Brush
.Color
:= clRed
;
4633 Canvas
.Font
.Color
:= clWhite
;
4635 Canvas
.FillRect(ARect
);
4636 Canvas
.TextRect(ARect
, ARect
.Left
, ARect
.Top
, Items
[Index
]);
4640 procedure TMainForm
.miReopenMapClick(Sender
: TObject
);
4642 FileName
, Resource
: String;
4644 if OpenedMap
= '' then
4647 if MessageBox(0, PChar(_lc
[I_MSG_REOPEN_MAP_PROMT
]),
4648 PChar(_lc
[I_MENU_FILE_REOPEN
]), MB_ICONQUESTION
or MB_YESNO
) <> idYes
then
4651 g_ProcessResourceStr(OpenedMap
, @FileName
, nil, @Resource
);
4652 OpenMap(FileName
, Resource
);
4655 procedure TMainForm
.vleObjectPropertyGetPickList(Sender
: TObject
;
4656 const KeyName
: String; Values
: TStrings
);
4658 if vleObjectProperty
.ItemProps
[KeyName
].EditStyle
= esPickList
then
4660 if KeyName
= _lc
[I_PROP_DIRECTION
] then
4662 Values
.Add(DirNames
[D_LEFT
]);
4663 Values
.Add(DirNames
[D_RIGHT
]);
4665 else if KeyName
= _lc
[I_PROP_TR_TELEPORT_DIR
] then
4667 Values
.Add(DirNamesAdv
[0]);
4668 Values
.Add(DirNamesAdv
[1]);
4669 Values
.Add(DirNamesAdv
[2]);
4670 Values
.Add(DirNamesAdv
[3]);
4672 else if KeyName
= _lc
[I_PROP_TR_MUSIC_ACT
] then
4674 Values
.Add(_lc
[I_PROP_TR_MUSIC_ON
]);
4675 Values
.Add(_lc
[I_PROP_TR_MUSIC_OFF
]);
4677 else if KeyName
= _lc
[I_PROP_TR_MONSTER_BEHAVIOUR
] then
4679 Values
.Add(_lc
[I_PROP_TR_MONSTER_BEHAVIOUR_0
]);
4680 Values
.Add(_lc
[I_PROP_TR_MONSTER_BEHAVIOUR_1
]);
4681 Values
.Add(_lc
[I_PROP_TR_MONSTER_BEHAVIOUR_2
]);
4682 Values
.Add(_lc
[I_PROP_TR_MONSTER_BEHAVIOUR_3
]);
4683 Values
.Add(_lc
[I_PROP_TR_MONSTER_BEHAVIOUR_4
]);
4684 Values
.Add(_lc
[I_PROP_TR_MONSTER_BEHAVIOUR_5
]);
4686 else if KeyName
= _lc
[I_PROP_TR_SCORE_ACT
] then
4688 Values
.Add(_lc
[I_PROP_TR_SCORE_ACT_0
]);
4689 Values
.Add(_lc
[I_PROP_TR_SCORE_ACT_1
]);
4690 Values
.Add(_lc
[I_PROP_TR_SCORE_ACT_2
]);
4691 Values
.Add(_lc
[I_PROP_TR_SCORE_ACT_3
]);
4693 else if KeyName
= _lc
[I_PROP_TR_SCORE_TEAM
] then
4695 Values
.Add(_lc
[I_PROP_TR_SCORE_TEAM_0
]);
4696 Values
.Add(_lc
[I_PROP_TR_SCORE_TEAM_1
]);
4697 Values
.Add(_lc
[I_PROP_TR_SCORE_TEAM_2
]);
4698 Values
.Add(_lc
[I_PROP_TR_SCORE_TEAM_3
]);
4700 else if KeyName
= _lc
[I_PROP_TR_MESSAGE_KIND
] then
4702 Values
.Add(_lc
[I_PROP_TR_MESSAGE_KIND_0
]);
4703 Values
.Add(_lc
[I_PROP_TR_MESSAGE_KIND_1
]);
4705 else if KeyName
= _lc
[I_PROP_TR_MESSAGE_TO
] then
4707 Values
.Add(_lc
[I_PROP_TR_MESSAGE_TO_0
]);
4708 Values
.Add(_lc
[I_PROP_TR_MESSAGE_TO_1
]);
4709 Values
.Add(_lc
[I_PROP_TR_MESSAGE_TO_2
]);
4710 Values
.Add(_lc
[I_PROP_TR_MESSAGE_TO_3
]);
4711 Values
.Add(_lc
[I_PROP_TR_MESSAGE_TO_4
]);
4712 Values
.Add(_lc
[I_PROP_TR_MESSAGE_TO_5
]);
4714 else if KeyName
= _lc
[I_PROP_TR_SHOT_TO
] then
4716 Values
.Add(_lc
[I_PROP_TR_SHOT_TO_0
]);
4717 Values
.Add(_lc
[I_PROP_TR_SHOT_TO_1
]);
4718 Values
.Add(_lc
[I_PROP_TR_SHOT_TO_2
]);
4719 Values
.Add(_lc
[I_PROP_TR_SHOT_TO_3
]);
4720 Values
.Add(_lc
[I_PROP_TR_SHOT_TO_4
]);
4721 Values
.Add(_lc
[I_PROP_TR_SHOT_TO_5
]);
4722 Values
.Add(_lc
[I_PROP_TR_SHOT_TO_6
]);
4724 else if KeyName
= _lc
[I_PROP_TR_SHOT_AIM
] then
4726 Values
.Add(_lc
[I_PROP_TR_SHOT_AIM_0
]);
4727 Values
.Add(_lc
[I_PROP_TR_SHOT_AIM_1
]);
4728 Values
.Add(_lc
[I_PROP_TR_SHOT_AIM_2
]);
4729 Values
.Add(_lc
[I_PROP_TR_SHOT_AIM_3
]);
4731 else if (KeyName
= _lc
[I_PROP_PANEL_BLEND
]) or
4732 (KeyName
= _lc
[I_PROP_DM_ONLY
]) or
4733 (KeyName
= _lc
[I_PROP_ITEM_FALLS
]) or
4734 (KeyName
= _lc
[I_PROP_TR_ENABLED
]) or
4735 (KeyName
= _lc
[I_PROP_TR_D2D
]) or
4736 (KeyName
= _lc
[I_PROP_TR_SILENT
]) or
4737 (KeyName
= _lc
[I_PROP_TR_TELEPORT_SILENT
]) or
4738 (KeyName
= _lc
[I_PROP_TR_EX_RANDOM
]) or
4739 (KeyName
= _lc
[I_PROP_TR_TEXTURE_ONCE
]) or
4740 (KeyName
= _lc
[I_PROP_TR_TEXTURE_ANIM_ONCE
]) or
4741 (KeyName
= _lc
[I_PROP_TR_SOUND_LOCAL
]) or
4742 (KeyName
= _lc
[I_PROP_TR_SOUND_SWITCH
]) or
4743 (KeyName
= _lc
[I_PROP_TR_MONSTER_ACTIVE
]) or
4744 (KeyName
= _lc
[I_PROP_TR_PUSH_RESET
]) or
4745 (KeyName
= _lc
[I_PROP_TR_SCORE_CON
]) or
4746 (KeyName
= _lc
[I_PROP_TR_SCORE_MSG
]) or
4747 (KeyName
= _lc
[I_PROP_TR_HEALTH_MAX
]) or
4748 (KeyName
= _lc
[I_PROP_TR_SHOT_SOUND
]) or
4749 (KeyName
= _lc
[I_PROP_TR_EFFECT_CENTER
]) then
4751 Values
.Add(BoolNames
[True]);
4752 Values
.Add(BoolNames
[False]);
4757 procedure TMainForm
.bApplyPropertyClick(Sender
: TObject
);
4759 _id
, a
, r
, c
: Integer;
4769 if SelectedObjectCount() <> 1 then
4771 if not SelectedObjects
[GetFirstSelected()].Live
then
4775 if not CheckProperty() then
4781 _id
:= GetFirstSelected();
4783 r
:= vleObjectProperty
.Row
;
4784 c
:= vleObjectProperty
.Col
;
4786 case SelectedObjects
[_id
].ObjectType
of
4789 with gPanels
[SelectedObjects
[_id
].ID
] do
4791 X
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_X
]]));
4792 Y
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_Y
]]));
4793 Width
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_WIDTH
]]));
4794 Height
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_HEIGHT
]]));
4796 PanelType
:= GetPanelType(vleObjectProperty
.Values
[_lc
[I_PROP_PANEL_TYPE
]]);
4798 // Сброс ссылки на триггеры смены текстуры:
4799 if not WordBool(PanelType
and (PANEL_WALL
or PANEL_FORE
or PANEL_BACK
)) then
4800 if gTriggers
<> nil then
4801 for a
:= 0 to High(gTriggers
) do
4803 if (gTriggers
[a
].TriggerType
<> 0) and
4804 (gTriggers
[a
].TexturePanel
= Integer(SelectedObjects
[_id
].ID
)) then
4805 gTriggers
[a
].TexturePanel
:= -1;
4806 if (gTriggers
[a
].TriggerType
= TRIGGER_SHOT
) and
4807 (gTriggers
[a
].Data
.ShotPanelID
= Integer(SelectedObjects
[_id
].ID
)) then
4808 gTriggers
[a
].Data
.ShotPanelID
:= -1;
4811 // Сброс ссылки на триггеры лифта:
4812 if not WordBool(PanelType
and (PANEL_LIFTUP
or PANEL_LIFTDOWN
or PANEL_LIFTLEFT
or PANEL_LIFTRIGHT
)) then
4813 if gTriggers
<> nil then
4814 for a
:= 0 to High(gTriggers
) do
4815 if (gTriggers
[a
].TriggerType
in [TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
]) and
4816 (gTriggers
[a
].Data
.PanelID
= Integer(SelectedObjects
[_id
].ID
)) then
4817 gTriggers
[a
].Data
.PanelID
:= -1;
4819 // Сброс ссылки на триггеры двери:
4820 if not WordBool(PanelType
and (PANEL_OPENDOOR
or PANEL_CLOSEDOOR
)) then
4821 if gTriggers
<> nil then
4822 for a
:= 0 to High(gTriggers
) do
4823 if (gTriggers
[a
].TriggerType
in [TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
,
4824 TRIGGER_DOOR5
, TRIGGER_CLOSETRAP
, TRIGGER_TRAP
]) and
4825 (gTriggers
[a
].Data
.PanelID
= Integer(SelectedObjects
[_id
].ID
)) then
4826 gTriggers
[a
].Data
.PanelID
:= -1;
4828 if IsTexturedPanel(PanelType
) then
4829 begin // Может быть текстура
4830 if TextureName
<> '' then
4831 begin // Была текстура
4832 Alpha
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_PANEL_ALPHA
]]));
4833 Blending
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_PANEL_BLEND
]]);
4842 TextureName
:= vleObjectProperty
.Values
[_lc
[I_PROP_PANEL_TEX
]];
4844 if TextureName
<> '' then
4845 begin // Есть текстура
4846 // Обычная текстура:
4847 if not IsSpecialTexture(TextureName
) then
4849 g_GetTextureSizeByName(TextureName
,
4850 TextureWidth
, TextureHeight
);
4852 // Проверка кратности размеров панели:
4854 if TextureWidth
<> 0 then
4855 if gPanels
[SelectedObjects
[_id
].ID
].Width
mod TextureWidth
<> 0 then
4857 ErrorMessageBox(Format(_lc
[I_MSG_WRONG_TEXWIDTH
],
4861 if Res
and (TextureHeight
<> 0) then
4862 if gPanels
[SelectedObjects
[_id
].ID
].Height
mod TextureHeight
<> 0 then
4864 ErrorMessageBox(Format(_lc
[I_MSG_WRONG_TEXHEIGHT
],
4871 if not g_GetTexture(TextureName
, TextureID
) then
4872 // Не удалось загрузить текстуру, рисуем NOTEXTURE
4873 if g_GetTexture('NOTEXTURE', NoTextureID
) then
4875 TextureID
:= TEXTURE_SPECIAL_NOTEXTURE
;
4876 g_GetTextureSizeByID(NoTextureID
, NW
, NH
);
4878 TextureHeight
:= NH
;
4881 TextureID
:= TEXTURE_SPECIAL_NONE
;
4891 TextureID
:= TEXTURE_SPECIAL_NONE
;
4894 else // Спец.текстура
4898 TextureID
:= SpecialTextureID(TextureName
);
4901 else // Нет текстуры
4905 TextureID
:= TEXTURE_SPECIAL_NONE
;
4908 else // Не может быть текстуры
4915 TextureID
:= TEXTURE_SPECIAL_NONE
;
4922 with gItems
[SelectedObjects
[_id
].ID
] do
4924 X
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_X
]]));
4925 Y
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_Y
]]));
4926 OnlyDM
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_DM_ONLY
]]);
4927 Fall
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_ITEM_FALLS
]]);
4933 with gMonsters
[SelectedObjects
[_id
].ID
] do
4935 X
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_X
]]));
4936 Y
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_Y
]]));
4937 Direction
:= NameToDir(vleObjectProperty
.Values
[_lc
[I_PROP_DIRECTION
]]);
4943 with gAreas
[SelectedObjects
[_id
].ID
] do
4945 X
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_X
]]));
4946 Y
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_Y
]]));
4947 Direction
:= NameToDir(vleObjectProperty
.Values
[_lc
[I_PROP_DIRECTION
]]);
4953 with gTriggers
[SelectedObjects
[_id
].ID
] do
4955 X
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_X
]]));
4956 Y
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_Y
]]));
4957 Width
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_WIDTH
]]));
4958 Height
:= StrToInt(Trim(vleObjectProperty
.Values
[_lc
[I_PROP_HEIGHT
]]));
4959 Enabled
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_ENABLED
]]);
4960 ActivateType
:= StrToActivate(vleObjectProperty
.Values
[_lc
[I_PROP_TR_ACTIVATION
]]);
4961 Key
:= StrToKey(vleObjectProperty
.Values
[_lc
[I_PROP_TR_KEYS
]]);
4966 s
:= utf2win(vleObjectProperty
.Values
[_lc
[I_PROP_TR_NEXT_MAP
]]);
4967 FillByte(Data
.MapName
[0], 16, 0);
4969 Move(s
[1], Data
.MapName
[0], Min(Length(s
), 16));
4974 Data
.ActivateOnce
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_TEXTURE_ONCE
]]);
4975 Data
.AnimOnce
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_TEXTURE_ANIM_ONCE
]]);
4978 TRIGGER_PRESS
, TRIGGER_ON
, TRIGGER_OFF
, TRIGGER_ONOFF
:
4980 Data
.Wait
:= Min(StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EX_DELAY
]], 0), 65535);
4981 Data
.Count
:= Min(StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EX_COUNT
]], 0), 65535);
4982 if Data
.Count
< 1 then
4984 if TriggerType
= TRIGGER_PRESS
then
4985 Data
.ExtRandom
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EX_RANDOM
]]);
4988 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
, TRIGGER_DOOR5
,
4989 TRIGGER_CLOSETRAP
, TRIGGER_TRAP
, TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
,
4992 Data
.NoSound
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SILENT
]]);
4993 Data
.d2d_doors
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_D2D
]]);
4998 Data
.d2d_teleport
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_D2D
]]);
4999 Data
.silent_teleport
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_TELEPORT_SILENT
]]);
5000 Data
.TlpDir
:= NameToDirAdv(vleObjectProperty
.Values
[_lc
[I_PROP_TR_TELEPORT_DIR
]]);
5005 s
:= utf2win(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SOUND_NAME
]]);
5006 FillByte(Data
.SoundName
[0], 64, 0);
5008 Move(s
[1], Data
.SoundName
[0], Min(Length(s
), 64));
5010 Data
.Volume
:= Min(StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SOUND_VOLUME
]], 0), 255);
5011 Data
.Pan
:= Min(StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SOUND_PAN
]], 0), 255);
5012 Data
.PlayCount
:= Min(StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SOUND_COUNT
]], 0), 255);
5013 Data
.Local
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SOUND_LOCAL
]]);
5014 Data
.SoundSwitch
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SOUND_SWITCH
]]);
5017 TRIGGER_SPAWNMONSTER
:
5019 Data
.MonType
:= StrToMonster(vleObjectProperty
.Values
[_lc
[I_PROP_TR_MONSTER_TYPE
]]);
5020 Data
.MonDir
:= Byte(NameToDir(vleObjectProperty
.Values
[_lc
[I_PROP_DIRECTION
]]));
5021 Data
.MonHealth
:= Min(StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_HEALTH
]], 0), 1000000);
5022 if Data
.MonHealth
< 0 then
5023 Data
.MonHealth
:= 0;
5024 Data
.MonActive
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_MONSTER_ACTIVE
]]);
5025 Data
.MonCount
:= Min(StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_COUNT
]], 0), 64);
5026 if Data
.MonCount
< 1 then
5028 Data
.MonEffect
:= StrToEffect(vleObjectProperty
.Values
[_lc
[I_PROP_TR_FX_TYPE
]]);
5029 Data
.MonMax
:= Min(StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SPAWN_MAX
]], 0), 65535);
5030 Data
.MonDelay
:= Min(StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SPAWN_DELAY
]], 0), 65535);
5032 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_MONSTER_BEHAVIOUR
]] = _lc
[I_PROP_TR_MONSTER_BEHAVIOUR_1
] then
5034 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_MONSTER_BEHAVIOUR
]] = _lc
[I_PROP_TR_MONSTER_BEHAVIOUR_2
] then
5036 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_MONSTER_BEHAVIOUR
]] = _lc
[I_PROP_TR_MONSTER_BEHAVIOUR_3
] then
5038 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_MONSTER_BEHAVIOUR
]] = _lc
[I_PROP_TR_MONSTER_BEHAVIOUR_4
] then
5040 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_MONSTER_BEHAVIOUR
]] = _lc
[I_PROP_TR_MONSTER_BEHAVIOUR_5
] then
5046 Data
.ItemType
:= StrToItem(vleObjectProperty
.Values
[_lc
[I_PROP_TR_ITEM_TYPE
]]);
5047 Data
.ItemOnlyDM
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_DM_ONLY
]]);
5048 Data
.ItemFalls
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_ITEM_FALLS
]]);
5049 Data
.ItemCount
:= Min(StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_COUNT
]], 0), 64);
5050 if Data
.ItemCount
< 1 then
5051 Data
.ItemCount
:= 1;
5052 Data
.ItemEffect
:= StrToEffect(vleObjectProperty
.Values
[_lc
[I_PROP_TR_FX_TYPE
]]);
5053 Data
.ItemMax
:= Min(StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SPAWN_MAX
]], 0), 65535);
5054 Data
.ItemDelay
:= Min(StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SPAWN_DELAY
]], 0), 65535);
5059 s
:= utf2win(vleObjectProperty
.Values
[_lc
[I_PROP_TR_MUSIC_NAME
]]);
5060 FillByte(Data
.MusicName
[0], 64, 0);
5062 Move(s
[1], Data
.MusicName
[0], Min(Length(s
), 64));
5064 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_MUSIC_ACT
]] = _lc
[I_PROP_TR_MUSIC_ON
] then
5065 Data
.MusicAction
:= 1
5067 Data
.MusicAction
:= 2;
5072 Data
.PushAngle
:= Min(
5073 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_PUSH_ANGLE
]], 0), 360);
5074 Data
.PushForce
:= Min(
5075 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_PUSH_FORCE
]], 0), 255);
5076 Data
.ResetVel
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_PUSH_RESET
]]);
5081 Data
.ScoreAction
:= 0;
5082 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SCORE_ACT
]] = _lc
[I_PROP_TR_SCORE_ACT_1
] then
5083 Data
.ScoreAction
:= 1
5084 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SCORE_ACT
]] = _lc
[I_PROP_TR_SCORE_ACT_2
] then
5085 Data
.ScoreAction
:= 2
5086 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SCORE_ACT
]] = _lc
[I_PROP_TR_SCORE_ACT_3
] then
5087 Data
.ScoreAction
:= 3;
5088 Data
.ScoreCount
:= Min(Max(
5089 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_COUNT
]], 0), 0), 255);
5090 Data
.ScoreTeam
:= 0;
5091 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SCORE_TEAM
]] = _lc
[I_PROP_TR_SCORE_TEAM_1
] then
5093 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SCORE_TEAM
]] = _lc
[I_PROP_TR_SCORE_TEAM_2
] then
5095 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SCORE_TEAM
]] = _lc
[I_PROP_TR_SCORE_TEAM_3
] then
5096 Data
.ScoreTeam
:= 3;
5097 Data
.ScoreCon
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SCORE_CON
]]);
5098 Data
.ScoreMsg
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SCORE_MSG
]]);
5103 Data
.MessageKind
:= 0;
5104 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_MESSAGE_KIND
]] = _lc
[I_PROP_TR_MESSAGE_KIND_1
] then
5105 Data
.MessageKind
:= 1;
5107 Data
.MessageSendTo
:= 0;
5108 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_MESSAGE_TO
]] = _lc
[I_PROP_TR_MESSAGE_TO_1
] then
5109 Data
.MessageSendTo
:= 1
5110 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_MESSAGE_TO
]] = _lc
[I_PROP_TR_MESSAGE_TO_2
] then
5111 Data
.MessageSendTo
:= 2
5112 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_MESSAGE_TO
]] = _lc
[I_PROP_TR_MESSAGE_TO_3
] then
5113 Data
.MessageSendTo
:= 3
5114 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_MESSAGE_TO
]] = _lc
[I_PROP_TR_MESSAGE_TO_4
] then
5115 Data
.MessageSendTo
:= 4
5116 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_MESSAGE_TO
]] = _lc
[I_PROP_TR_MESSAGE_TO_5
] then
5117 Data
.MessageSendTo
:= 5;
5119 s
:= utf2win(vleObjectProperty
.Values
[_lc
[I_PROP_TR_MESSAGE_TEXT
]]);
5120 FillByte(Data
.MessageText
[0], 100, 0);
5122 Move(s
[1], Data
.MessageText
[0], Min(Length(s
), 100));
5124 Data
.MessageTime
:= Min(Max(
5125 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_MESSAGE_TIME
]], 0), 0), 65535);
5130 Data
.DamageValue
:= Min(Max(
5131 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_DAMAGE_VALUE
]], 0), 0), 65535);
5132 Data
.DamageInterval
:= Min(Max(
5133 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_INTERVAL
]], 0), 0), 65535);
5138 Data
.HealValue
:= Min(Max(
5139 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_HEALTH
]], 0), 0), 65535);
5140 Data
.HealInterval
:= Min(Max(
5141 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_INTERVAL
]], 0), 0), 65535);
5142 Data
.HealMax
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_HEALTH_MAX
]]);
5143 Data
.HealSilent
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SILENT
]]);
5148 Data
.ShotType
:= StrToShot(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_TYPE
]]);
5149 Data
.ShotSound
:= NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_SOUND
]]);
5150 Data
.ShotTarget
:= 0;
5151 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_TO
]] = _lc
[I_PROP_TR_SHOT_TO_1
] then
5152 Data
.ShotTarget
:= 1
5153 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_TO
]] = _lc
[I_PROP_TR_SHOT_TO_2
] then
5154 Data
.ShotTarget
:= 2
5155 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_TO
]] = _lc
[I_PROP_TR_SHOT_TO_3
] then
5156 Data
.ShotTarget
:= 3
5157 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_TO
]] = _lc
[I_PROP_TR_SHOT_TO_4
] then
5158 Data
.ShotTarget
:= 4
5159 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_TO
]] = _lc
[I_PROP_TR_SHOT_TO_5
] then
5160 Data
.ShotTarget
:= 5
5161 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_TO
]] = _lc
[I_PROP_TR_SHOT_TO_6
] then
5162 Data
.ShotTarget
:= 6;
5163 Data
.ShotIntSight
:= Min(Max(
5164 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_SIGHT
]], 0), 0), 65535);
5166 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_AIM
]] = _lc
[I_PROP_TR_SHOT_AIM_1
] then
5168 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_AIM
]] = _lc
[I_PROP_TR_SHOT_AIM_2
] then
5170 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_AIM
]] = _lc
[I_PROP_TR_SHOT_AIM_3
] then
5172 Data
.ShotAngle
:= Min(
5173 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_ANGLE
]], 0), 360);
5174 Data
.ShotWait
:= Min(Max(
5175 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EX_DELAY
]], 0), 0), 65535);
5176 Data
.ShotAccuracy
:= Min(Max(
5177 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_ACC
]], 0), 0), 65535);
5178 Data
.ShotAmmo
:= Min(Max(
5179 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_AMMO
]], 0), 0), 65535);
5180 Data
.ShotIntReload
:= Min(Max(
5181 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_SHOT_RELOAD
]], 0), 0), 65535);
5186 Data
.FXCount
:= Min(Max(
5187 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_COUNT
]], 0), 0), 255);
5188 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_TYPE
]] = _lc
[I_PROP_TR_EFFECT_PARTICLE
] then
5190 Data
.FXType
:= TRIGGER_EFFECT_PARTICLE
;
5191 Data
.FXSubType
:= TRIGGER_EFFECT_SLIQUID
;
5192 if vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_SUBTYPE
]] = _lc
[I_PROP_TR_EFFECT_SLIQUID
] then
5193 Data
.FXSubType
:= TRIGGER_EFFECT_SLIQUID
5194 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_SUBTYPE
]] = _lc
[I_PROP_TR_EFFECT_LLIQUID
] then
5195 Data
.FXSubType
:= TRIGGER_EFFECT_LLIQUID
5196 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_SUBTYPE
]] = _lc
[I_PROP_TR_EFFECT_DLIQUID
] then
5197 Data
.FXSubType
:= TRIGGER_EFFECT_DLIQUID
5198 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_SUBTYPE
]] = _lc
[I_PROP_TR_EFFECT_BLOOD
] then
5199 Data
.FXSubType
:= TRIGGER_EFFECT_BLOOD
5200 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_SUBTYPE
]] = _lc
[I_PROP_TR_EFFECT_SPARK
] then
5201 Data
.FXSubType
:= TRIGGER_EFFECT_SPARK
5202 else if vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_SUBTYPE
]] = _lc
[I_PROP_TR_EFFECT_BUBBLE
] then
5203 Data
.FXSubType
:= TRIGGER_EFFECT_BUBBLE
;
5206 Data
.FXType
:= TRIGGER_EFFECT_ANIMATION
;
5207 Data
.FXSubType
:= StrToEffect(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_SUBTYPE
]]);
5210 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_COLOR
]], 0), 0), $FFFFFF);
5211 Data
.FXColorR
:= a
and $FF;
5212 Data
.FXColorG
:= (a
shr 8) and $FF;
5213 Data
.FXColorB
:= (a
shr 16) and $FF;
5214 if NameToBool(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_CENTER
]]) then
5218 Data
.FXWait
:= Min(Max(
5219 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EX_DELAY
]], 0), 0), 65535);
5220 Data
.FXVelX
:= Min(Max(
5221 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_VELX
]], 0), -128), 127);
5222 Data
.FXVelY
:= Min(Max(
5223 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_VELY
]], 0), -128), 127);
5224 Data
.FXSpreadL
:= Min(Max(
5225 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_SPL
]], 0), 0), 255);
5226 Data
.FXSpreadR
:= Min(Max(
5227 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_SPR
]], 0), 0), 255);
5228 Data
.FXSpreadU
:= Min(Max(
5229 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_SPU
]], 0), 0), 255);
5230 Data
.FXSpreadD
:= Min(Max(
5231 StrToIntDef(vleObjectProperty
.Values
[_lc
[I_PROP_TR_EFFECT_SPD
]], 0), 0), 255);
5240 vleObjectProperty
.Row
:= r
;
5241 vleObjectProperty
.Col
:= c
;
5244 procedure TMainForm
.bbRemoveTextureClick(Sender
: TObject
);
5248 i
:= lbTextureList
.ItemIndex
;
5252 if MessageBox(0, PChar(Format(_lc
[I_MSG_DEL_TEXTURE_PROMT
],
5253 [SelectedTexture()])),
5254 PChar(_lc
[I_MSG_DEL_TEXTURE
]),
5255 MB_ICONQUESTION
or MB_YESNO
or
5256 MB_DEFBUTTON1
) <> idYes
then
5259 if gPanels
<> nil then
5260 for a
:= 0 to High(gPanels
) do
5261 if (gPanels
[a
].PanelType
<> 0) and
5262 (gPanels
[a
].TextureName
= SelectedTexture()) then
5264 ErrorMessageBox(_lc
[I_MSG_DEL_TEXTURE_CANT
]);
5268 g_DeleteTexture(SelectedTexture());
5269 i
:= slInvalidTextures
.IndexOf(lbTextureList
.Items
[i
]);
5271 slInvalidTextures
.Delete(i
);
5272 if lbTextureList
.ItemIndex
> -1 then
5273 lbTextureList
.Items
.Delete(lbTextureList
.ItemIndex
)
5276 procedure TMainForm
.aNewMapExecute(Sender
: TObject
);
5278 if (MessageBox(0, PChar(_lc
[I_MSG_CLEAR_MAP_PROMT
]),
5279 PChar(_lc
[I_MSG_CLEAR_MAP
]),
5280 MB_ICONQUESTION
or MB_YESNO
or
5281 MB_DEFBUTTON1
) = mrYes
) then
5285 procedure TMainForm
.aUndoExecute(Sender
: TObject
);
5289 if UndoBuffer
= nil then
5291 if UndoBuffer
[High(UndoBuffer
)] = nil then
5294 for a
:= 0 to High(UndoBuffer
[High(UndoBuffer
)]) do
5295 with UndoBuffer
[High(UndoBuffer
)][a
] do
5303 UNDO_DELETE_ITEM
: AddItem(Item
);
5304 UNDO_DELETE_AREA
: AddArea(Area
);
5305 UNDO_DELETE_MONSTER
: AddMonster(Monster
);
5306 UNDO_DELETE_TRIGGER
: AddTrigger(Trigger
);
5307 UNDO_ADD_PANEL
: RemoveObject(AddID
, OBJECT_PANEL
);
5308 UNDO_ADD_ITEM
: RemoveObject(AddID
, OBJECT_ITEM
);
5309 UNDO_ADD_AREA
: RemoveObject(AddID
, OBJECT_AREA
);
5310 UNDO_ADD_MONSTER
: RemoveObject(AddID
, OBJECT_MONSTER
);
5311 UNDO_ADD_TRIGGER
: RemoveObject(AddID
, OBJECT_TRIGGER
);
5315 SetLength(UndoBuffer
, Length(UndoBuffer
)-1);
5317 RemoveSelectFromObjects();
5319 miUndo
.Enabled
:= UndoBuffer
<> nil;
5323 procedure TMainForm
.aCopyObjectExecute(Sender
: TObject
);
5326 CopyBuffer
: TCopyRecArray
;
5330 function CB_Compare(I1
, I2
: TCopyRec
): Integer;
5332 Result
:= Integer(I1
.ObjectType
) - Integer(I2
.ObjectType
);
5334 if Result
= 0 then // Одного типа
5335 Result
:= Integer(I1
.ID
) - Integer(I2
.ID
);
5338 procedure QuickSortCopyBuffer(L
, R
: Integer);
5346 P
:= CopyBuffer
[(L
+ R
) shr 1];
5349 while CB_Compare(CopyBuffer
[I
], P
) < 0 do
5351 while CB_Compare(CopyBuffer
[J
], P
) > 0 do
5357 CopyBuffer
[I
] := CopyBuffer
[J
];
5365 QuickSortCopyBuffer(L
, J
);
5372 if SelectedObjects
= nil then
5378 // Копируем объекты:
5379 for a
:= 0 to High(SelectedObjects
) do
5380 if SelectedObjects
[a
].Live
then
5381 with SelectedObjects
[a
] do
5383 SetLength(CopyBuffer
, Length(CopyBuffer
)+1);
5384 b
:= High(CopyBuffer
);
5385 CopyBuffer
[b
].ID
:= ID
;
5386 CopyBuffer
[b
].Panel
:= nil;
5391 CopyBuffer
[b
].ObjectType
:= OBJECT_PANEL
;
5392 New(CopyBuffer
[b
].Panel
);
5393 CopyBuffer
[b
].Panel
^ := gPanels
[ID
];
5398 CopyBuffer
[b
].ObjectType
:= OBJECT_ITEM
;
5399 CopyBuffer
[b
].Item
:= gItems
[ID
];
5404 CopyBuffer
[b
].ObjectType
:= OBJECT_MONSTER
;
5405 CopyBuffer
[b
].Monster
:= gMonsters
[ID
];
5410 CopyBuffer
[b
].ObjectType
:= OBJECT_AREA
;
5411 CopyBuffer
[b
].Area
:= gAreas
[ID
];
5416 CopyBuffer
[b
].ObjectType
:= OBJECT_TRIGGER
;
5417 CopyBuffer
[b
].Trigger
:= gTriggers
[ID
];
5422 // Сортировка по ID:
5423 if CopyBuffer
<> nil then
5425 QuickSortCopyBuffer(0, b
);
5428 // Пестановка ссылок триггеров:
5429 for a
:= 0 to Length(CopyBuffer
)-1 do
5430 if CopyBuffer
[a
].ObjectType
= OBJECT_TRIGGER
then
5432 case CopyBuffer
[a
].Trigger
.TriggerType
of
5433 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
,
5434 TRIGGER_DOOR5
, TRIGGER_CLOSETRAP
, TRIGGER_TRAP
,
5435 TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
:
5436 if CopyBuffer
[a
].Trigger
.Data
.PanelID
<> -1 then
5440 for b
:= 0 to Length(CopyBuffer
)-1 do
5441 if (CopyBuffer
[b
].ObjectType
= OBJECT_PANEL
) and
5442 (Integer(CopyBuffer
[b
].ID
) = CopyBuffer
[a
].Trigger
.Data
.PanelID
) then
5444 CopyBuffer
[a
].Trigger
.Data
.PanelID
:= b
;
5449 // Этих панелей нет среди копируемых:
5451 CopyBuffer
[a
].Trigger
.Data
.PanelID
:= -1;
5454 TRIGGER_PRESS
, TRIGGER_ON
,
5455 TRIGGER_OFF
, TRIGGER_ONOFF
:
5456 if CopyBuffer
[a
].Trigger
.Data
.MonsterID
<> 0 then
5460 for b
:= 0 to Length(CopyBuffer
)-1 do
5461 if (CopyBuffer
[b
].ObjectType
= OBJECT_MONSTER
) and
5462 (Integer(CopyBuffer
[b
].ID
) = CopyBuffer
[a
].Trigger
.Data
.MonsterID
-1) then
5464 CopyBuffer
[a
].Trigger
.Data
.MonsterID
:= b
+1;
5469 // Этих монстров нет среди копируемых:
5471 CopyBuffer
[a
].Trigger
.Data
.MonsterID
:= 0;
5475 if CopyBuffer
[a
].Trigger
.Data
.ShotPanelID
<> -1 then
5479 for b
:= 0 to Length(CopyBuffer
)-1 do
5480 if (CopyBuffer
[b
].ObjectType
= OBJECT_PANEL
) and
5481 (Integer(CopyBuffer
[b
].ID
) = CopyBuffer
[a
].Trigger
.Data
.ShotPanelID
) then
5483 CopyBuffer
[a
].Trigger
.Data
.ShotPanelID
:= b
;
5488 // Этих панелей нет среди копируемых:
5490 CopyBuffer
[a
].Trigger
.Data
.ShotPanelID
:= -1;
5494 if CopyBuffer
[a
].Trigger
.TexturePanel
<> -1 then
5498 for b
:= 0 to Length(CopyBuffer
)-1 do
5499 if (CopyBuffer
[b
].ObjectType
= OBJECT_PANEL
) and
5500 (Integer(CopyBuffer
[b
].ID
) = CopyBuffer
[a
].Trigger
.TexturePanel
) then
5502 CopyBuffer
[a
].Trigger
.TexturePanel
:= b
;
5507 // Этих панелей нет среди копируемых:
5509 CopyBuffer
[a
].Trigger
.TexturePanel
:= -1;
5514 str
:= CopyBufferToString(CopyBuffer
);
5515 ClipBoard
.AsText
:= str
;
5517 for a
:= 0 to Length(CopyBuffer
)-1 do
5518 if (CopyBuffer
[a
].ObjectType
= OBJECT_PANEL
) and
5519 (CopyBuffer
[a
].Panel
<> nil) then
5520 Dispose(CopyBuffer
[a
].Panel
);
5525 procedure TMainForm
.aPasteObjectExecute(Sender
: TObject
);
5528 CopyBuffer
: TCopyRecArray
;
5530 swad
, ssec
, sres
: String;
5536 pmin
.X
:= High(pmin
.X
);
5537 pmin
.Y
:= High(pmin
.Y
);
5539 StringToCopyBuffer(ClipBoard
.AsText
, CopyBuffer
, pmin
);
5540 rel
:= not(ssShift
in GetKeyShiftState());
5542 if CopyBuffer
= nil then
5545 RemoveSelectFromObjects();
5547 h
:= High(CopyBuffer
);
5549 with CopyBuffer
[a
] do
5553 if Panel
<> nil then
5557 Panel
^.X
:= Panel
^.X
- pmin
.X
- MapOffset
.X
+ 32;
5558 Panel
^.Y
:= Panel
^.Y
- pmin
.Y
- MapOffset
.Y
+ 32;
5561 Panel
^.TextureID
:= TEXTURE_SPECIAL_NONE
;
5562 Panel
^.TextureWidth
:= 1;
5563 Panel
^.TextureHeight
:= 1;
5565 if (Panel
^.PanelType
= PANEL_LIFTUP
) or
5566 (Panel
^.PanelType
= PANEL_LIFTDOWN
) or
5567 (Panel
^.PanelType
= PANEL_LIFTLEFT
) or
5568 (Panel
^.PanelType
= PANEL_LIFTRIGHT
) or
5569 (Panel
^.PanelType
= PANEL_BLOCKMON
) or
5570 (Panel
^.TextureName
= '') then
5571 begin // Нет или не может быть текстуры:
5573 else // Есть текстура:
5575 // Обычная текстура:
5576 if not IsSpecialTexture(Panel
^.TextureName
) then
5578 res
:= g_GetTexture(Panel
^.TextureName
, Panel
^.TextureID
);
5582 g_ProcessResourceStr(Panel
^.TextureName
, swad
, ssec
, sres
);
5583 AddTexture(swad
, ssec
, sres
, True);
5584 res
:= g_GetTexture(Panel
^.TextureName
, Panel
^.TextureID
);
5588 g_GetTextureSizeByName(Panel
^.TextureName
,
5589 Panel
^.TextureWidth
, Panel
^.TextureHeight
)
5591 if g_GetTexture('NOTEXTURE', NoTextureID
) then
5593 Panel
^.TextureID
:= TEXTURE_SPECIAL_NOTEXTURE
;
5594 g_GetTextureSizeByID(NoTextureID
, Panel
^.TextureWidth
, Panel
^.TextureHeight
);
5597 else // Спец.текстура:
5599 Panel
^.TextureID
:= SpecialTextureID(Panel
^.TextureName
);
5600 with MainForm
.lbTextureList
.Items
do
5601 if IndexOf(Panel
^.TextureName
) = -1 then
5602 Add(Panel
^.TextureName
);
5606 ID
:= AddPanel(Panel
^);
5608 Undo_Add(OBJECT_PANEL
, ID
, a
> 0);
5609 SelectObject(OBJECT_PANEL
, ID
, True);
5616 Item
.X
:= Item
.X
- pmin
.X
- MapOffset
.X
+ 32;
5617 Item
.Y
:= Item
.Y
- pmin
.Y
- MapOffset
.Y
+ 32;
5620 ID
:= AddItem(Item
);
5621 Undo_Add(OBJECT_ITEM
, ID
, a
> 0);
5622 SelectObject(OBJECT_ITEM
, ID
, True);
5629 Monster
.X
:= Monster
.X
- pmin
.X
- MapOffset
.X
+ 32;
5630 Monster
.Y
:= Monster
.Y
- pmin
.Y
- MapOffset
.Y
+ 32;
5633 ID
:= AddMonster(Monster
);
5634 Undo_Add(OBJECT_MONSTER
, ID
, a
> 0);
5635 SelectObject(OBJECT_MONSTER
, ID
, True);
5642 Area
.X
:= Area
.X
- pmin
.X
- MapOffset
.X
+ 32;
5643 Area
.Y
:= Area
.Y
- pmin
.Y
- MapOffset
.Y
+ 32;
5646 ID
:= AddArea(Area
);
5647 Undo_Add(OBJECT_AREA
, ID
, a
> 0);
5648 SelectObject(OBJECT_AREA
, ID
, True);
5656 X
:= X
- pmin
.X
- MapOffset
.X
+ 32;
5657 Y
:= Y
- pmin
.Y
- MapOffset
.Y
+ 32;
5662 Data
.TargetPoint
.X
:=
5663 Data
.TargetPoint
.X
- pmin
.X
- MapOffset
.X
+ 32;
5664 Data
.TargetPoint
.Y
:=
5665 Data
.TargetPoint
.Y
- pmin
.Y
- MapOffset
.Y
+ 32;
5667 TRIGGER_PRESS
, TRIGGER_ON
, TRIGGER_OFF
, TRIGGER_ONOFF
:
5669 Data
.tX
:= Data
.tX
- pmin
.X
- MapOffset
.X
+ 32;
5670 Data
.tY
:= Data
.tY
- pmin
.Y
- MapOffset
.Y
+ 32;
5672 TRIGGER_SPAWNMONSTER
:
5675 Data
.MonPos
.X
- pmin
.X
- MapOffset
.X
+ 32;
5677 Data
.MonPos
.Y
- pmin
.Y
- MapOffset
.Y
+ 32;
5682 Data
.ItemPos
.X
- pmin
.X
- MapOffset
.X
+ 32;
5684 Data
.ItemPos
.Y
- pmin
.Y
- MapOffset
.Y
+ 32;
5689 Data
.ShotPos
.X
- pmin
.X
- MapOffset
.X
+ 32;
5691 Data
.ShotPos
.Y
- pmin
.Y
- MapOffset
.Y
+ 32;
5696 ID
:= AddTrigger(Trigger
);
5697 Undo_Add(OBJECT_TRIGGER
, ID
, a
> 0);
5698 SelectObject(OBJECT_TRIGGER
, ID
, True);
5703 // Переставляем ссылки триггеров:
5704 for a
:= 0 to High(CopyBuffer
) do
5705 if CopyBuffer
[a
].ObjectType
= OBJECT_TRIGGER
then
5707 case CopyBuffer
[a
].Trigger
.TriggerType
of
5708 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
,
5709 TRIGGER_DOOR5
, TRIGGER_CLOSETRAP
, TRIGGER_TRAP
,
5710 TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
:
5711 if CopyBuffer
[a
].Trigger
.Data
.PanelID
<> -1 then
5712 gTriggers
[CopyBuffer
[a
].ID
].Data
.PanelID
:=
5713 CopyBuffer
[CopyBuffer
[a
].Trigger
.Data
.PanelID
].ID
;
5715 TRIGGER_PRESS
, TRIGGER_ON
,
5716 TRIGGER_OFF
, TRIGGER_ONOFF
:
5717 if CopyBuffer
[a
].Trigger
.Data
.MonsterID
<> 0 then
5718 gTriggers
[CopyBuffer
[a
].ID
].Data
.MonsterID
:=
5719 CopyBuffer
[CopyBuffer
[a
].Trigger
.Data
.MonsterID
-1].ID
+1;
5722 if CopyBuffer
[a
].Trigger
.Data
.ShotPanelID
<> -1 then
5723 gTriggers
[CopyBuffer
[a
].ID
].Data
.ShotPanelID
:=
5724 CopyBuffer
[CopyBuffer
[a
].Trigger
.Data
.ShotPanelID
].ID
;
5727 if CopyBuffer
[a
].Trigger
.TexturePanel
<> -1 then
5728 gTriggers
[CopyBuffer
[a
].ID
].TexturePanel
:=
5729 CopyBuffer
[CopyBuffer
[a
].Trigger
.TexturePanel
].ID
;
5738 procedure TMainForm
.aCutObjectExecute(Sender
: TObject
);
5741 DeleteSelectedObjects();
5744 procedure TMainForm
.vleObjectPropertyEditButtonClick(Sender
: TObject
);
5746 Key
, FileName
: String;
5749 Key
:= vleObjectProperty
.Keys
[vleObjectProperty
.Row
];
5751 if Key
= _lc
[I_PROP_PANEL_TYPE
] then
5753 with ChooseTypeForm
, vleObjectProperty
do
5754 begin // Выбор типа панели:
5755 Caption
:= _lc
[I_PROP_PANEL_TYPE
];
5756 lbTypeSelect
.Items
.Clear();
5758 for b
:= 0 to High(PANELNAMES
) do
5760 lbTypeSelect
.Items
.Add(PANELNAMES
[b
]);
5761 if Values
[Key
] = PANELNAMES
[b
] then
5762 lbTypeSelect
.ItemIndex
:= b
;
5765 if ShowModal() = mrOK
then
5767 b
:= lbTypeSelect
.ItemIndex
;
5768 Values
[Key
] := PANELNAMES
[b
];
5769 vleObjectPropertyApply(Sender
);
5773 else if Key
= _lc
[I_PROP_TR_TELEPORT_TO
] then
5774 SelectFlag
:= SELECTFLAG_TELEPORT
5775 else if Key
= _lc
[I_PROP_TR_SPAWN_TO
] then
5776 SelectFlag
:= SELECTFLAG_SPAWNPOINT
5777 else if (Key
= _lc
[I_PROP_TR_DOOR_PANEL
]) or
5778 (Key
= _lc
[I_PROP_TR_TRAP_PANEL
]) then
5779 SelectFlag
:= SELECTFLAG_DOOR
5780 else if Key
= _lc
[I_PROP_TR_TEXTURE_PANEL
] then
5782 DrawPressRect
:= False;
5783 SelectFlag
:= SELECTFLAG_TEXTURE
;
5785 else if Key
= _lc
[I_PROP_TR_SHOT_PANEL
] then
5786 SelectFlag
:= SELECTFLAG_SHOTPANEL
5787 else if Key
= _lc
[I_PROP_TR_LIFT_PANEL
] then
5788 SelectFlag
:= SELECTFLAG_LIFT
5789 else if key
= _lc
[I_PROP_TR_EX_MONSTER
] then
5790 SelectFlag
:= SELECTFLAG_MONSTER
5791 else if Key
= _lc
[I_PROP_TR_EX_AREA
] then
5793 SelectFlag
:= SELECTFLAG_NONE
;
5794 DrawPressRect
:= True;
5796 else if Key
= _lc
[I_PROP_TR_NEXT_MAP
] then
5797 begin // Выбор следующей карты:
5798 g_ProcessResourceStr(OpenedMap
, @FileName
, nil, nil);
5799 SelectMapForm
.Caption
:= _lc
[I_CAP_SELECT
];
5800 SelectMapForm
.GetMaps(FileName
);
5802 if SelectMapForm
.ShowModal() = mrOK
then
5804 vleObjectProperty
.Values
[Key
] := SelectMapForm
.lbMapList
.Items
[SelectMapForm
.lbMapList
.ItemIndex
];
5805 vleObjectPropertyApply(Sender
);
5808 else if (Key
= _lc
[I_PROP_TR_SOUND_NAME
]) or
5809 (Key
= _lc
[I_PROP_TR_MUSIC_NAME
]) then
5810 begin // Выбор файла звука/музыки:
5811 AddSoundForm
.OKFunction
:= nil;
5812 AddSoundForm
.lbResourcesList
.MultiSelect
:= False;
5813 AddSoundForm
.SetResource
:= vleObjectProperty
.Values
[Key
];
5815 if (AddSoundForm
.ShowModal() = mrOk
) then
5817 vleObjectProperty
.Values
[Key
] := AddSoundForm
.ResourceName
;
5818 vleObjectPropertyApply(Sender
);
5821 else if Key
= _lc
[I_PROP_TR_ACTIVATION
] then
5822 with ActivationTypeForm
, vleObjectProperty
do
5823 begin // Выбор типов активации:
5824 cbPlayerCollide
.Checked
:= Pos('PC', Values
[Key
]) > 0;
5825 cbMonsterCollide
.Checked
:= Pos('MC', Values
[Key
]) > 0;
5826 cbPlayerPress
.Checked
:= Pos('PP', Values
[Key
]) > 0;
5827 cbMonsterPress
.Checked
:= Pos('MP', Values
[Key
]) > 0;
5828 cbShot
.Checked
:= Pos('SH', Values
[Key
]) > 0;
5829 cbNoMonster
.Checked
:= Pos('NM', Values
[Key
]) > 0;
5831 if ShowModal() = mrOK
then
5834 if cbPlayerCollide
.Checked
then
5835 b
:= ACTIVATE_PLAYERCOLLIDE
;
5836 if cbMonsterCollide
.Checked
then
5837 b
:= b
or ACTIVATE_MONSTERCOLLIDE
;
5838 if cbPlayerPress
.Checked
then
5839 b
:= b
or ACTIVATE_PLAYERPRESS
;
5840 if cbMonsterPress
.Checked
then
5841 b
:= b
or ACTIVATE_MONSTERPRESS
;
5842 if cbShot
.Checked
then
5843 b
:= b
or ACTIVATE_SHOT
;
5844 if cbNoMonster
.Checked
then
5845 b
:= b
or ACTIVATE_NOMONSTER
;
5847 Values
[Key
] := ActivateToStr(b
);
5848 vleObjectPropertyApply(Sender
);
5851 else if Key
= _lc
[I_PROP_TR_KEYS
] then
5852 with KeysForm
, vleObjectProperty
do
5853 begin // Выбор необходимых ключей:
5854 cbRedKey
.Checked
:= Pos('RK', Values
[Key
]) > 0;
5855 cbGreenKey
.Checked
:= Pos('GK', Values
[Key
]) > 0;
5856 cbBlueKey
.Checked
:= Pos('BK', Values
[Key
]) > 0;
5857 cbRedTeam
.Checked
:= Pos('RT', Values
[Key
]) > 0;
5858 cbBlueTeam
.Checked
:= Pos('BT', Values
[Key
]) > 0;
5860 if ShowModal() = mrOK
then
5863 if cbRedKey
.Checked
then
5865 if cbGreenKey
.Checked
then
5866 b
:= b
or KEY_GREEN
;
5867 if cbBlueKey
.Checked
then
5869 if cbRedTeam
.Checked
then
5870 b
:= b
or KEY_REDTEAM
;
5871 if cbBlueTeam
.Checked
then
5872 b
:= b
or KEY_BLUETEAM
;
5874 Values
[Key
] := KeyToStr(b
);
5875 vleObjectPropertyApply(Sender
);
5878 else if Key
= _lc
[I_PROP_TR_FX_TYPE
] then
5879 with ChooseTypeForm
, vleObjectProperty
do
5880 begin // Выбор типа эффекта:
5881 Caption
:= _lc
[I_CAP_FX_TYPE
];
5882 lbTypeSelect
.Items
.Clear();
5884 for b
:= EFFECT_NONE
to EFFECT_FIRE
do
5885 lbTypeSelect
.Items
.Add(EffectToStr(b
));
5887 lbTypeSelect
.ItemIndex
:= StrToEffect(Values
[Key
]);
5889 if ShowModal() = mrOK
then
5891 b
:= lbTypeSelect
.ItemIndex
;
5892 Values
[Key
] := EffectToStr(b
);
5893 vleObjectPropertyApply(Sender
);
5896 else if Key
= _lc
[I_PROP_TR_MONSTER_TYPE
] then
5897 with ChooseTypeForm
, vleObjectProperty
do
5898 begin // Выбор типа монстра:
5899 Caption
:= _lc
[I_CAP_MONSTER_TYPE
];
5900 lbTypeSelect
.Items
.Clear();
5902 for b
:= MONSTER_DEMON
to MONSTER_MAN
do
5903 lbTypeSelect
.Items
.Add(MonsterToStr(b
));
5905 lbTypeSelect
.ItemIndex
:= StrToMonster(Values
[Key
]) - MONSTER_DEMON
;
5907 if ShowModal() = mrOK
then
5909 b
:= lbTypeSelect
.ItemIndex
+ MONSTER_DEMON
;
5910 Values
[Key
] := MonsterToStr(b
);
5911 vleObjectPropertyApply(Sender
);
5914 else if Key
= _lc
[I_PROP_TR_ITEM_TYPE
] then
5915 with ChooseTypeForm
, vleObjectProperty
do
5916 begin // Выбор типа предмета:
5917 Caption
:= _lc
[I_CAP_ITEM_TYPE
];
5918 lbTypeSelect
.Items
.Clear();
5920 for b
:= ITEM_MEDKIT_SMALL
to ITEM_KEY_BLUE
do
5921 lbTypeSelect
.Items
.Add(ItemToStr(b
));
5922 lbTypeSelect
.Items
.Add(ItemToStr(ITEM_BOTTLE
));
5923 lbTypeSelect
.Items
.Add(ItemToStr(ITEM_HELMET
));
5924 lbTypeSelect
.Items
.Add(ItemToStr(ITEM_JETPACK
));
5925 lbTypeSelect
.Items
.Add(ItemToStr(ITEM_INVIS
));
5926 lbTypeSelect
.Items
.Add(ItemToStr(ITEM_WEAPON_FLAMETHROWER
));
5927 lbTypeSelect
.Items
.Add(ItemToStr(ITEM_AMMO_FUELCAN
));
5929 b
:= StrToItem(Values
[Key
]);
5930 if b
>= ITEM_BOTTLE
then
5932 lbTypeSelect
.ItemIndex
:= b
- ITEM_MEDKIT_SMALL
;
5934 if ShowModal() = mrOK
then
5936 b
:= lbTypeSelect
.ItemIndex
+ ITEM_MEDKIT_SMALL
;
5937 if b
>= ITEM_WEAPON_KASTET
then
5939 Values
[Key
] := ItemToStr(b
);
5940 vleObjectPropertyApply(Sender
);
5943 else if Key
= _lc
[I_PROP_TR_SHOT_TYPE
] then
5944 with ChooseTypeForm
, vleObjectProperty
do
5945 begin // Выбор типа предмета:
5946 Caption
:= _lc
[I_PROP_TR_SHOT_TYPE
];
5947 lbTypeSelect
.Items
.Clear();
5949 for b
:= TRIGGER_SHOT_PISTOL
to TRIGGER_SHOT_MAX
do
5950 lbTypeSelect
.Items
.Add(ShotToStr(b
));
5952 lbTypeSelect
.ItemIndex
:= StrToShot(Values
[Key
]);
5954 if ShowModal() = mrOK
then
5956 b
:= lbTypeSelect
.ItemIndex
;
5957 Values
[Key
] := ShotToStr(b
);
5958 vleObjectPropertyApply(Sender
);
5961 else if Key
= _lc
[I_PROP_TR_EFFECT_TYPE
] then
5962 with ChooseTypeForm
, vleObjectProperty
do
5963 begin // Выбор типа эффекта:
5964 Caption
:= _lc
[I_CAP_FX_TYPE
];
5965 lbTypeSelect
.Items
.Clear();
5967 lbTypeSelect
.Items
.Add(_lc
[I_PROP_TR_EFFECT_PARTICLE
]);
5968 lbTypeSelect
.Items
.Add(_lc
[I_PROP_TR_EFFECT_ANIMATION
]);
5969 if Values
[Key
] = _lc
[I_PROP_TR_EFFECT_ANIMATION
] then
5970 lbTypeSelect
.ItemIndex
:= 1
5972 lbTypeSelect
.ItemIndex
:= 0;
5974 if ShowModal() = mrOK
then
5976 b
:= lbTypeSelect
.ItemIndex
;
5978 Values
[Key
] := _lc
[I_PROP_TR_EFFECT_PARTICLE
]
5980 Values
[Key
] := _lc
[I_PROP_TR_EFFECT_ANIMATION
];
5981 vleObjectPropertyApply(Sender
);
5984 else if Key
= _lc
[I_PROP_TR_EFFECT_SUBTYPE
] then
5985 with ChooseTypeForm
, vleObjectProperty
do
5986 begin // Выбор подтипа эффекта:
5987 Caption
:= _lc
[I_CAP_FX_TYPE
];
5988 lbTypeSelect
.Items
.Clear();
5990 if Values
[_lc
[I_PROP_TR_EFFECT_TYPE
]] = _lc
[I_PROP_TR_EFFECT_ANIMATION
] then
5992 for b
:= EFFECT_TELEPORT
to EFFECT_FIRE
do
5993 lbTypeSelect
.Items
.Add(EffectToStr(b
));
5995 lbTypeSelect
.ItemIndex
:= StrToEffect(Values
[Key
]) - 1;
5998 lbTypeSelect
.Items
.Add(_lc
[I_PROP_TR_EFFECT_SLIQUID
]);
5999 lbTypeSelect
.Items
.Add(_lc
[I_PROP_TR_EFFECT_LLIQUID
]);
6000 lbTypeSelect
.Items
.Add(_lc
[I_PROP_TR_EFFECT_DLIQUID
]);
6001 lbTypeSelect
.Items
.Add(_lc
[I_PROP_TR_EFFECT_BLOOD
]);
6002 lbTypeSelect
.Items
.Add(_lc
[I_PROP_TR_EFFECT_SPARK
]);
6003 lbTypeSelect
.Items
.Add(_lc
[I_PROP_TR_EFFECT_BUBBLE
]);
6004 lbTypeSelect
.ItemIndex
:= TRIGGER_EFFECT_SLIQUID
;
6005 if Values
[Key
] = _lc
[I_PROP_TR_EFFECT_LLIQUID
] then
6006 lbTypeSelect
.ItemIndex
:= TRIGGER_EFFECT_LLIQUID
;
6007 if Values
[Key
] = _lc
[I_PROP_TR_EFFECT_DLIQUID
] then
6008 lbTypeSelect
.ItemIndex
:= TRIGGER_EFFECT_DLIQUID
;
6009 if Values
[Key
] = _lc
[I_PROP_TR_EFFECT_BLOOD
] then
6010 lbTypeSelect
.ItemIndex
:= TRIGGER_EFFECT_BLOOD
;
6011 if Values
[Key
] = _lc
[I_PROP_TR_EFFECT_SPARK
] then
6012 lbTypeSelect
.ItemIndex
:= TRIGGER_EFFECT_SPARK
;
6013 if Values
[Key
] = _lc
[I_PROP_TR_EFFECT_BUBBLE
] then
6014 lbTypeSelect
.ItemIndex
:= TRIGGER_EFFECT_BUBBLE
;
6017 if ShowModal() = mrOK
then
6019 b
:= lbTypeSelect
.ItemIndex
;
6021 if Values
[_lc
[I_PROP_TR_EFFECT_TYPE
]] = _lc
[I_PROP_TR_EFFECT_ANIMATION
] then
6022 Values
[Key
] := EffectToStr(b
+ 1)
6024 Values
[Key
] := _lc
[I_PROP_TR_EFFECT_SLIQUID
];
6025 if b
= TRIGGER_EFFECT_LLIQUID
then
6026 Values
[Key
] := _lc
[I_PROP_TR_EFFECT_LLIQUID
];
6027 if b
= TRIGGER_EFFECT_DLIQUID
then
6028 Values
[Key
] := _lc
[I_PROP_TR_EFFECT_DLIQUID
];
6029 if b
= TRIGGER_EFFECT_BLOOD
then
6030 Values
[Key
] := _lc
[I_PROP_TR_EFFECT_BLOOD
];
6031 if b
= TRIGGER_EFFECT_SPARK
then
6032 Values
[Key
] := _lc
[I_PROP_TR_EFFECT_SPARK
];
6033 if b
= TRIGGER_EFFECT_BUBBLE
then
6034 Values
[Key
] := _lc
[I_PROP_TR_EFFECT_BUBBLE
];
6037 vleObjectPropertyApply(Sender
);
6040 else if Key
= _lc
[I_PROP_TR_EFFECT_COLOR
] then
6041 with vleObjectProperty
do
6042 begin // Выбор цвета эффекта:
6043 ColorDialog
.Color
:= StrToIntDef(Values
[Key
], 0);
6044 if ColorDialog
.Execute
then
6046 Values
[Key
] := IntToStr(ColorDialog
.Color
);
6047 vleObjectPropertyApply(Sender
);
6050 else if Key
= _lc
[I_PROP_PANEL_TEX
] then
6051 begin // Смена текстуры:
6052 vleObjectProperty
.Values
[Key
] := SelectedTexture();
6053 vleObjectPropertyApply(Sender
);
6057 procedure TMainForm
.vleObjectPropertyApply(Sender
: TObject
);
6059 // hack to prevent empty ID in list
6060 RenderPanel
.SetFocus();
6061 bApplyProperty
.Click();
6062 vleObjectProperty
.SetFocus();
6065 procedure TMainForm
.aSaveMapExecute(Sender
: TObject
);
6067 FileName
, Section
, Res
: String;
6069 if OpenedMap
= '' then
6071 aSaveMapAsExecute(nil);
6075 g_ProcessResourceStr(OpenedMap
, FileName
, Section
, Res
);
6077 SaveMap(FileName
+':\'+Res
);
6080 procedure TMainForm
.aOpenMapExecute(Sender
: TObject
);
6082 OpenDialog
.Filter
:= _lc
[I_FILE_FILTER_ALL
];
6084 if OpenDialog
.Execute() then
6086 OpenMapFile(OpenDialog
.FileName
);
6087 OpenDialog
.InitialDir
:= ExtractFileDir(OpenDialog
.FileName
);
6091 procedure TMainForm
.OpenMapFile(FileName
: String);
6093 if (Pos('.ini', LowerCase(ExtractFileName(FileName
))) > 0) then
6097 pLoadProgress
.Left
:= (RenderPanel
.Width
div 2)-(pLoadProgress
.Width
div 2);
6098 pLoadProgress
.Top
:= (RenderPanel
.Height
div 2)-(pLoadProgress
.Height
div 2);
6099 pLoadProgress
.Show();
6104 LoadMapOld(FileName
);
6106 MainForm
.Caption
:= Format('%s - %s', [FormCaption
, ExtractFileName(FileName
)]);
6108 pLoadProgress
.Hide();
6109 MainForm
.FormResize(Self
);
6111 else // Карты из WAD:
6113 OpenMap(FileName
, '');
6117 procedure TMainForm
.FormActivate(Sender
: TObject
);
6122 MainForm
.ActiveControl
:= RenderPanel
;
6125 if gLanguage
= '' then
6127 lang
:= SelectLanguageForm
.ShowModal();
6129 1: gLanguage
:= LANGUAGE_ENGLISH
;
6130 else gLanguage
:= LANGUAGE_RUSSIAN
;
6133 config
:= TConfig
.CreateFile(EditorDir
+'Editor.cfg');
6134 config
.WriteStr('Editor', 'Language', gLanguage
);
6135 config
.SaveFile(EditorDir
+'Editor.cfg');
6139 //e_WriteLog('Read language file', MSG_NOTIFY);
6140 //g_Language_Load(EditorDir+'\data\'+gLanguage+LANGUAGE_FILE_NAME);
6141 g_Language_Set(gLanguage
);
6144 procedure TMainForm
.aDeleteMap(Sender
: TObject
);
6152 OpenDialog
.Filter
:= _lc
[I_FILE_FILTER_WAD
];
6154 if not OpenDialog
.Execute() then
6157 WAD
:= TWADEditor_1
.Create();
6159 if not WAD
.ReadFile(OpenDialog
.FileName
) then
6167 MapList
:= WAD
.GetResourcesList('');
6169 SelectMapForm
.Caption
:= _lc
[I_CAP_REMOVE
];
6170 SelectMapForm
.lbMapList
.Items
.Clear();
6172 if MapList
<> nil then
6173 for a
:= 0 to High(MapList
) do
6174 SelectMapForm
.lbMapList
.Items
.Add(win2utf(MapList
[a
]));
6176 if (SelectMapForm
.ShowModal() = mrOK
) then
6178 str
:= SelectMapForm
.lbMapList
.Items
[SelectMapForm
.lbMapList
.ItemIndex
];
6180 Move(str
[1], MapName
[0], Min(16, Length(str
)));
6182 if MessageBox(0, PChar(Format(_lc
[I_MSG_DELETE_MAP_PROMT
],
6183 [MapName
, OpenDialog
.FileName
])),
6184 PChar(_lc
[I_MSG_DELETE_MAP
]),
6185 MB_ICONQUESTION
or MB_YESNO
or
6186 MB_DEFBUTTON2
) <> mrYes
then
6189 WAD
.RemoveResource('', utf2win(MapName
));
6191 MessageBox(0, PChar(Format(_lc
[I_MSG_MAP_DELETED_PROMT
],
6193 PChar(_lc
[I_MSG_MAP_DELETED
]),
6194 MB_ICONINFORMATION
or MB_OK
or
6197 WAD
.SaveTo(OpenDialog
.FileName
);
6199 // Удалили текущую карту - сохранять по старому ее нельзя:
6200 if OpenedMap
= (OpenDialog
.FileName
+':\'+MapName
) then
6204 MainForm
.Caption
:= FormCaption
;
6211 procedure TMainForm
.vleObjectPropertyKeyDown(Sender
: TObject
;
6212 var Key
: Word; Shift
: TShiftState
);
6214 if Key
= VK_RETURN
then
6215 vleObjectPropertyApply(Sender
);
6218 procedure MovePanel(var ID
: DWORD
; MoveType
: Byte);
6223 if (ID
= 0) and (MoveType
= 0) then
6225 if (ID
= DWORD(High(gPanels
))) and (MoveType
<> 0) then
6227 if (ID
> DWORD(High(gPanels
))) then
6232 if MoveType
= 0 then // to Back
6234 if gTriggers
<> nil then
6235 for a
:= 0 to High(gTriggers
) do
6236 with gTriggers
[a
] do
6238 if TriggerType
= TRIGGER_NONE
then
6241 if TexturePanel
= _id
then
6244 if (TexturePanel
>= 0) and (TexturePanel
< _id
) then
6248 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
,
6249 TRIGGER_DOOR5
, TRIGGER_CLOSETRAP
, TRIGGER_TRAP
,
6250 TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
:
6251 if Data
.PanelID
= _id
then
6254 if (Data
.PanelID
>= 0) and (Data
.PanelID
< _id
) then
6258 if Data
.ShotPanelID
= _id
then
6259 Data
.ShotPanelID
:= 0
6261 if (Data
.ShotPanelID
>= 0) and (Data
.ShotPanelID
< _id
) then
6262 Inc(Data
.ShotPanelID
);
6266 tmp
:= gPanels
[_id
];
6268 for a
:= _id
downto 1 do
6269 gPanels
[a
] := gPanels
[a
-1];
6277 if gTriggers
<> nil then
6278 for a
:= 0 to High(gTriggers
) do
6279 with gTriggers
[a
] do
6281 if TriggerType
= TRIGGER_NONE
then
6284 if TexturePanel
= _id
then
6285 TexturePanel
:= High(gPanels
)
6287 if TexturePanel
> _id
then
6291 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
,
6292 TRIGGER_DOOR5
, TRIGGER_CLOSETRAP
, TRIGGER_TRAP
,
6293 TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
:
6294 if Data
.PanelID
= _id
then
6295 Data
.PanelID
:= High(gPanels
)
6297 if Data
.PanelID
> _id
then
6301 if Data
.ShotPanelID
= _id
then
6302 Data
.ShotPanelID
:= High(gPanels
)
6304 if Data
.ShotPanelID
> _id
then
6305 Dec(Data
.ShotPanelID
);
6309 tmp
:= gPanels
[_id
];
6311 for a
:= _id
to High(gPanels
)-1 do
6312 gPanels
[a
] := gPanels
[a
+1];
6314 gPanels
[High(gPanels
)] := tmp
;
6316 ID
:= High(gPanels
);
6320 procedure TMainForm
.aMoveToBack(Sender
: TObject
);
6324 if SelectedObjects
= nil then
6327 for a
:= 0 to High(SelectedObjects
) do
6328 with SelectedObjects
[a
] do
6329 if Live
and (ObjectType
= OBJECT_PANEL
) then
6331 SelectedObjects
[0] := SelectedObjects
[a
];
6332 SetLength(SelectedObjects
, 1);
6339 procedure TMainForm
.aMoveToFore(Sender
: TObject
);
6343 if SelectedObjects
= nil then
6346 for a
:= 0 to High(SelectedObjects
) do
6347 with SelectedObjects
[a
] do
6348 if Live
and (ObjectType
= OBJECT_PANEL
) then
6350 SelectedObjects
[0] := SelectedObjects
[a
];
6351 SetLength(SelectedObjects
, 1);
6358 procedure TMainForm
.aSaveMapAsExecute(Sender
: TObject
);
6362 SaveDialog
.Filter
:= _lc
[I_FILE_FILTER_WAD
];
6364 if not SaveDialog
.Execute() then
6367 SaveMapForm
.GetMaps(SaveDialog
.FileName
, True);
6369 if SaveMapForm
.ShowModal() <> mrOK
then
6372 SaveDialog
.InitialDir
:= ExtractFileDir(SaveDialog
.FileName
);
6373 OpenedMap
:= SaveDialog
.FileName
+':\'+SaveMapForm
.eMapName
.Text;
6374 OpenedWAD
:= SaveDialog
.FileName
;
6376 idx
:= RecentFiles
.IndexOf(OpenedMap
);
6377 // Такая карта уже недавно открывалась:
6379 RecentFiles
.Delete(idx
);
6380 RecentFiles
.Insert(0, OpenedMap
);
6385 gMapInfo
.FileName
:= SaveDialog
.FileName
;
6386 gMapInfo
.MapName
:= SaveMapForm
.eMapName
.Text;
6387 UpdateCaption(gMapInfo
.Name
, ExtractFileName(gMapInfo
.FileName
), gMapInfo
.MapName
);
6390 procedure TMainForm
.aSelectAllExecute(Sender
: TObject
);
6394 RemoveSelectFromObjects();
6396 case pcObjects
.ActivePageIndex
+1 of
6398 if gPanels
<> nil then
6399 for a
:= 0 to High(gPanels
) do
6400 if gPanels
[a
].PanelType
<> PANEL_NONE
then
6401 SelectObject(OBJECT_PANEL
, a
, True);
6403 if gItems
<> nil then
6404 for a
:= 0 to High(gItems
) do
6405 if gItems
[a
].ItemType
<> ITEM_NONE
then
6406 SelectObject(OBJECT_ITEM
, a
, True);
6408 if gMonsters
<> nil then
6409 for a
:= 0 to High(gMonsters
) do
6410 if gMonsters
[a
].MonsterType
<> MONSTER_NONE
then
6411 SelectObject(OBJECT_MONSTER
, a
, True);
6413 if gAreas
<> nil then
6414 for a
:= 0 to High(gAreas
) do
6415 if gAreas
[a
].AreaType
<> AREA_NONE
then
6416 SelectObject(OBJECT_AREA
, a
, True);
6418 if gTriggers
<> nil then
6419 for a
:= 0 to High(gTriggers
) do
6420 if gTriggers
[a
].TriggerType
<> TRIGGER_NONE
then
6421 SelectObject(OBJECT_TRIGGER
, a
, True);
6424 RecountSelectedObjects();
6427 procedure TMainForm
.tbGridOnClick(Sender
: TObject
);
6429 DotEnable
:= not DotEnable
;
6430 (Sender
as TToolButton
).Down
:= DotEnable
;
6433 procedure TMainForm
.OnIdle(Sender
: TObject
; var Done
: Boolean);
6435 // FIXME: this is a shitty hack
6436 if not gDataLoaded
then
6438 e_WriteLog('Init OpenGL', MSG_NOTIFY
);
6440 e_WriteLog('Loading data', MSG_NOTIFY
);
6441 LoadStdFont('STDTXT', 'STDFONT', gEditorFont
);
6442 e_WriteLog('Loading more data', MSG_NOTIFY
);
6444 e_WriteLog('Loading even more data', MSG_NOTIFY
);
6445 gDataLoaded
:= True;
6446 MainForm
.FormResize(nil);
6451 procedure TMainForm
.miMapPreviewClick(Sender
: TObject
);
6453 if PreviewMode
= 2 then
6456 if PreviewMode
= 0 then
6458 Splitter2
.Visible
:= False;
6459 Splitter1
.Visible
:= False;
6460 StatusBar
.Visible
:= False;
6461 PanelObjs
.Visible
:= False;
6462 PanelProps
.Visible
:= False;
6463 MainToolBar
.Visible
:= False;
6464 sbHorizontal
.Visible
:= False;
6465 sbVertical
.Visible
:= False;
6469 StatusBar
.Visible
:= True;
6470 PanelObjs
.Visible
:= True;
6471 PanelProps
.Visible
:= True;
6472 Splitter2
.Visible
:= True;
6473 Splitter1
.Visible
:= True;
6474 MainToolBar
.Visible
:= True;
6475 sbHorizontal
.Visible
:= True;
6476 sbVertical
.Visible
:= True;
6479 PreviewMode
:= PreviewMode
xor 1;
6480 (Sender
as TMenuItem
).Checked
:= PreviewMode
> 0;
6485 procedure TMainForm
.miLayer1Click(Sender
: TObject
);
6487 SwitchLayer(LAYER_BACK
);
6490 procedure TMainForm
.miLayer2Click(Sender
: TObject
);
6492 SwitchLayer(LAYER_WALLS
);
6495 procedure TMainForm
.miLayer3Click(Sender
: TObject
);
6497 SwitchLayer(LAYER_FOREGROUND
);
6500 procedure TMainForm
.miLayer4Click(Sender
: TObject
);
6502 SwitchLayer(LAYER_STEPS
);
6505 procedure TMainForm
.miLayer5Click(Sender
: TObject
);
6507 SwitchLayer(LAYER_WATER
);
6510 procedure TMainForm
.miLayer6Click(Sender
: TObject
);
6512 SwitchLayer(LAYER_ITEMS
);
6515 procedure TMainForm
.miLayer7Click(Sender
: TObject
);
6517 SwitchLayer(LAYER_MONSTERS
);
6520 procedure TMainForm
.miLayer8Click(Sender
: TObject
);
6522 SwitchLayer(LAYER_AREAS
);
6525 procedure TMainForm
.miLayer9Click(Sender
: TObject
);
6527 SwitchLayer(LAYER_TRIGGERS
);
6530 procedure TMainForm
.tbShowClick(Sender
: TObject
);
6536 for a
:= 0 to High(LayerEnabled
) do
6537 b
:= b
and LayerEnabled
[a
];
6541 ShowLayer(LAYER_BACK
, b
);
6542 ShowLayer(LAYER_WALLS
, b
);
6543 ShowLayer(LAYER_FOREGROUND
, b
);
6544 ShowLayer(LAYER_STEPS
, b
);
6545 ShowLayer(LAYER_WATER
, b
);
6546 ShowLayer(LAYER_ITEMS
, b
);
6547 ShowLayer(LAYER_MONSTERS
, b
);
6548 ShowLayer(LAYER_AREAS
, b
);
6549 ShowLayer(LAYER_TRIGGERS
, b
);
6552 procedure TMainForm
.miMiniMapClick(Sender
: TObject
);
6557 procedure TMainForm
.miSwitchGridClick(Sender
: TObject
);
6559 if DotStep
= DotStepOne
then
6560 DotStep
:= DotStepTwo
6562 DotStep
:= DotStepOne
;
6564 MousePos
.X
:= (MousePos
.X
div DotStep
) * DotStep
;
6565 MousePos
.Y
:= (MousePos
.Y
div DotStep
) * DotStep
;
6568 procedure TMainForm
.miShowEdgesClick(Sender
: TObject
);
6573 procedure TMainForm
.miSnapToGridClick(Sender
: TObject
);
6575 SnapToGrid
:= not SnapToGrid
;
6577 MousePos
.X
:= (MousePos
.X
div DotStep
) * DotStep
;
6578 MousePos
.Y
:= (MousePos
.Y
div DotStep
) * DotStep
;
6580 miSnapToGrid
.Checked
:= SnapToGrid
;
6583 procedure TMainForm
.minexttabClick(Sender
: TObject
);
6585 if pcObjects
.ActivePageIndex
< pcObjects
.PageCount
-1 then
6586 pcObjects
.ActivePageIndex
:= pcObjects
.ActivePageIndex
+1
6588 pcObjects
.ActivePageIndex
:= 0;
6591 procedure TMainForm
.miSaveMiniMapClick(Sender
: TObject
);
6593 SaveMiniMapForm
.ShowModal();
6596 procedure TMainForm
.bClearTextureClick(Sender
: TObject
);
6598 lbTextureList
.ItemIndex
:= -1;
6599 lTextureWidth
.Caption
:= '';
6600 lTextureHeight
.Caption
:= '';
6603 procedure TMainForm
.miPackMapClick(Sender
: TObject
);
6605 PackMapForm
.ShowModal();
6608 procedure TMainForm
.miMapTestSettingsClick(Sender
: TObject
);
6610 MapTestForm
.ShowModal();
6613 procedure TMainForm
.miTestMapClick(Sender
: TObject
);
6615 cmd
, mapWAD
, mapToRun
, tempWAD
: String;
6622 if OpenedMap
<> '' then
6624 // Указываем текущую карту для теста:
6625 g_ProcessResourceStr(OpenedMap
, @mapWAD
, nil, @mapToRun
);
6626 mapToRun
:= mapWAD
+ ':\' + mapToRun
;
6627 mapToRun
:= ExtractRelativePath(ExtractFilePath(TestD2dExe
) + 'maps/', mapToRun
);
6629 // Сохраняем временную карту:
6632 mapWAD
:= ExtractFilePath(TestD2dExe
) + Format('maps/temp%.4d.wad', [time
]);
6634 until not FileExists(mapWAD
);
6635 tempWAD
:= mapWAD
+ ':\' + TEST_MAP_NAME
;
6638 tempWAD
:= ExtractRelativePath(ExtractFilePath(TestD2dExe
) + 'maps/', tempWAD
);
6639 // Если карта не была открыта, указываем временную в качестве текущей:
6640 if mapToRun
= '' then
6641 mapToRun
:= tempWAD
;
6645 if TestOptionsTwoPlayers
then
6647 if TestOptionsTeamDamage
then
6649 if TestOptionsAllowExit
then
6651 if TestOptionsWeaponStay
then
6653 if TestOptionsMonstersDM
then
6656 // Составляем командную строку:
6657 cmd
:= '-map "' + mapToRun
+ '"';
6658 cmd
:= cmd
+ ' -testmap "' + tempWAD
+ '"';
6659 cmd
:= cmd
+ ' -gm ' + TestGameMode
;
6660 cmd
:= cmd
+ ' -limt ' + TestLimTime
;
6661 cmd
:= cmd
+ ' -lims ' + TestLimScore
;
6662 cmd
:= cmd
+ ' -opt ' + IntToStr(opt
);
6665 cmd
:= cmd
+ ' --close';
6667 cmd
:= cmd
+ ' --debug';
6670 proc
:= TProcessUTF8
.Create(nil);
6671 proc
.Executable
:= TestD2dExe
;
6672 proc
.Parameters
.Add(cmd
);
6681 Application
.Minimize();
6684 if (not res
) or (proc
.ExitCode
< 0) then
6686 MessageBox(0, 'FIXME',
6687 PChar(_lc
[I_MSG_EXEC_ERROR
]),
6688 MB_OK
or MB_ICONERROR
);
6692 SysUtils
.DeleteFile(mapWAD
);
6693 Application
.Restore();
6696 procedure TMainForm
.sbVerticalScroll(Sender
: TObject
;
6697 ScrollCode
: TScrollCode
; var ScrollPos
: Integer);
6699 MapOffset
.Y
:= -Normalize16(sbVertical
.Position
);
6702 procedure TMainForm
.sbHorizontalScroll(Sender
: TObject
;
6703 ScrollCode
: TScrollCode
; var ScrollPos
: Integer);
6705 MapOffset
.X
:= -Normalize16(sbHorizontal
.Position
);
6708 procedure TMainForm
.miOpenWadMapClick(Sender
: TObject
);
6710 if OpenedWAD
<> '' then
6712 OpenMap(OpenedWAD
, '');
6716 procedure TMainForm
.selectall1Click(Sender
: TObject
);
6720 RemoveSelectFromObjects();
6722 if gPanels
<> nil then
6723 for a
:= 0 to High(gPanels
) do
6724 if gPanels
[a
].PanelType
<> PANEL_NONE
then
6725 SelectObject(OBJECT_PANEL
, a
, True);
6727 if gItems
<> nil then
6728 for a
:= 0 to High(gItems
) do
6729 if gItems
[a
].ItemType
<> ITEM_NONE
then
6730 SelectObject(OBJECT_ITEM
, a
, True);
6732 if gMonsters
<> nil then
6733 for a
:= 0 to High(gMonsters
) do
6734 if gMonsters
[a
].MonsterType
<> MONSTER_NONE
then
6735 SelectObject(OBJECT_MONSTER
, a
, True);
6737 if gAreas
<> nil then
6738 for a
:= 0 to High(gAreas
) do
6739 if gAreas
[a
].AreaType
<> AREA_NONE
then
6740 SelectObject(OBJECT_AREA
, a
, True);
6742 if gTriggers
<> nil then
6743 for a
:= 0 to High(gTriggers
) do
6744 if gTriggers
[a
].TriggerType
<> TRIGGER_NONE
then
6745 SelectObject(OBJECT_TRIGGER
, a
, True);
6747 RecountSelectedObjects();
6750 procedure TMainForm
.Splitter1CanResize(Sender
: TObject
;
6751 var NewSize
: Integer; var Accept
: Boolean);
6753 Accept
:= (NewSize
> 140);
6756 procedure TMainForm
.Splitter2CanResize(Sender
: TObject
;
6757 var NewSize
: Integer; var Accept
: Boolean);
6759 Accept
:= (NewSize
> 110);
6762 procedure TMainForm
.vleObjectPropertyEnter(Sender
: TObject
);
6764 EditingProperties
:= True;
6767 procedure TMainForm
.vleObjectPropertyExit(Sender
: TObject
);
6769 EditingProperties
:= False;
6772 procedure TMainForm
.FormKeyUp(Sender
: TObject
; var Key
: Word;
6773 Shift
: TShiftState
);
6775 // Объекты передвигались:
6776 if MainForm
.ActiveControl
= RenderPanel
then
6778 if (Key
= VK_NUMPAD4
) or
6779 (Key
= VK_NUMPAD6
) or
6780 (Key
= VK_NUMPAD8
) or
6781 (Key
= VK_NUMPAD5
) or
6782 (Key
= Ord('V')) then
6785 // Быстрое превью карты:
6786 if Key
= Ord('E') then
6788 if PreviewMode
= 2 then
6791 RenderPanelMouseMove(Sender
, Shift
, RenderMousePos().X
, RenderMousePos().Y
);