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
)
21 StatusBar
: TStatusBar
;
22 OpenDialog
: TOpenDialog
;
23 SaveDialog
: TSaveDialog
;
24 ColorDialog
: TColorDialog
;
28 ImageList
: TImageList
;
31 miAppleAbout
: TMenuItem
;
32 miAppleLine0
: TMenuItem
;
33 miApplePref
: TMenuItem
;
34 miAppleLine1
: TMenuItem
;
36 miMenuFile
: TMenuItem
;
39 miMacRecentSubMenu
: TMenuItem
;
40 miMacRecentEnd
: TMenuItem
;
41 miMacRecentClear
: TMenuItem
;
42 Separator1
: TMenuItem
;
44 miSaveMapAs
: TMenuItem
;
45 miOpenWadMap
: TMenuItem
;
47 miReopenMap
: TMenuItem
;
48 miSaveMiniMap
: TMenuItem
;
49 miDeleteMap
: TMenuItem
;
51 miWinRecentStart
: TMenuItem
;
52 miWinRecent
: TMenuItem
;
56 miMenuEdit
: TMenuItem
;
63 miSelectAll
: TMenuItem
;
65 miSnapToGrid
: TMenuItem
;
66 miSwitchGrid
: TMenuItem
;
67 Separator2
: TMenuItem
;
71 miMapOptions
: TMenuItem
;
74 miMenuView
: TMenuItem
;
85 miViewLine1
: TMenuItem
;
87 miShowEdges
: TMenuItem
;
88 miViewLine2
: TMenuItem
;
89 miMapPreview
: TMenuItem
;
91 miMenuService
: TMenuItem
;
92 miCheckMap
: TMenuItem
;
93 miOptimmization
: TMenuItem
;
96 miMenuWindow
: TMenuItem
;
97 miMacMinimize
: TMenuItem
;
100 miMenuHelp
: TMenuItem
;
103 miMenuHidden
: TMenuItem
;
104 minexttab
: TMenuItem
;
105 selectall1
: TMenuItem
;
108 ilToolbar
: TImageList
;
109 MainToolBar
: TToolBar
;
110 tbNewMap
: TToolButton
;
111 tbOpenMap
: TToolButton
;
112 tbSaveMap
: TToolButton
;
113 tbOpenWadMap
: TToolButton
;
114 tbLine1
: TToolButton
;
115 tbShowMap
: TToolButton
;
116 tbLine2
: TToolButton
;
119 miLayerP1
: TMenuItem
;
120 miLayerP2
: TMenuItem
;
121 miLayerP3
: TMenuItem
;
122 miLayerP4
: TMenuItem
;
123 miLayerP5
: TMenuItem
;
124 miLayerP6
: TMenuItem
;
125 miLayerP7
: TMenuItem
;
126 miLayerP8
: TMenuItem
;
127 miLayerP9
: TMenuItem
;
128 tbLine3
: TToolButton
;
129 tbGridOn
: TToolButton
;
131 tbLine4
: TToolButton
;
132 tbTestMap
: TToolButton
;
135 pLoadProgress
: TPanel
;
137 pbLoad
: TProgressBar
;
141 RenderPanel
: TOpenGLControl
;
142 sbHorizontal
: TScrollBar
;
143 sbVertical
: TScrollBar
;
145 // Object propertiy editor:
147 PanelPropApply
: TPanel
;
148 bApplyProperty
: TButton
;
149 vleObjectProperty
: TValueListEditor
;
153 pcObjects
: TPageControl
;
156 PanelPanelType
: TPanel
;
157 lbPanelType
: TListBox
;
158 lbTextureList
: TListBox
;
159 PanelTextures
: TPanel
;
161 lTextureWidth
: TLabel
;
163 lTextureHeight
: TLabel
;
164 cbPreview
: TCheckBox
;
165 bbAddTexture
: TBitBtn
;
166 bbRemoveTexture
: TBitBtn
;
167 bClearTexture
: TButton
;
170 lbItemList
: TListBox
;
174 tsMonsters
: TTabSheet
;
175 lbMonsterList
: TListBox
;
176 rbMonsterLeft
: TRadioButton
;
177 rbMonsterRight
: TRadioButton
;
180 lbAreasList
: TListBox
;
181 rbAreaLeft
: TRadioButton
;
182 rbAreaRight
: TRadioButton
;
184 tsTriggers
: TTabSheet
;
185 lbTriggersList
: TListBox
;
186 clbActivationType
: TCheckListBox
;
187 clbKeys
: TCheckListBox
;
189 procedure aAboutExecute(Sender
: TObject
);
190 procedure aCheckMapExecute(Sender
: TObject
);
191 procedure aMoveToFore(Sender
: TObject
);
192 procedure aMoveToBack(Sender
: TObject
);
193 procedure aCopyObjectExecute(Sender
: TObject
);
194 procedure aCutObjectExecute(Sender
: TObject
);
195 procedure aEditorOptionsExecute(Sender
: TObject
);
196 procedure aExitExecute(Sender
: TObject
);
197 procedure aMapOptionsExecute(Sender
: TObject
);
198 procedure aNewMapExecute(Sender
: TObject
);
199 procedure aOpenMapExecute(Sender
: TObject
);
200 procedure aOptimizeExecute(Sender
: TObject
);
201 procedure aPasteObjectExecute(Sender
: TObject
);
202 procedure aSelectAllExecute(Sender
: TObject
);
203 procedure aSaveMapExecute(Sender
: TObject
);
204 procedure aSaveMapAsExecute(Sender
: TObject
);
205 procedure aUndoExecute(Sender
: TObject
);
206 procedure aDeleteMap(Sender
: TObject
);
207 procedure bApplyPropertyClick(Sender
: TObject
);
208 procedure bbAddTextureClick(Sender
: TObject
);
209 procedure bbRemoveTextureClick(Sender
: TObject
);
210 procedure FormActivate(Sender
: TObject
);
211 procedure FormCloseQuery(Sender
: TObject
; var CanClose
: Boolean);
212 procedure FormCreate(Sender
: TObject
);
213 procedure FormDestroy(Sender
: TObject
);
214 procedure FormDropFiles(Sender
: TObject
; const FileNames
: array of String);
215 procedure FormKeyDown(Sender
: TObject
; var Key
: Word; Shift
: TShiftState
);
216 procedure FormResize(Sender
: TObject
);
217 procedure FormWindowStateChange(Sender
: TObject
);
218 procedure miRecentFileExecute(Sender
: TObject
);
219 procedure miMacRecentClearClick(Sender
: TObject
);
220 procedure miMacZoomClick(Sender
: TObject
);
221 procedure lbTextureListClick(Sender
: TObject
);
222 procedure lbTextureListDrawItem(Control
: TWinControl
; Index
: Integer;
223 ARect
: TRect
; State
: TOwnerDrawState
);
224 procedure miMacMinimizeClick(Sender
: TObject
);
225 procedure miReopenMapClick(Sender
: TObject
);
226 procedure RenderPanelMouseDown(Sender
: TObject
; Button
: TMouseButton
; Shift
: TShiftState
; X
, Y
: Integer);
227 procedure RenderPanelMouseMove(Sender
: TObject
; Shift
: TShiftState
; X
, Y
: Integer);
228 procedure RenderPanelMouseUp(Sender
: TObject
; Button
: TMouseButton
; Shift
: TShiftState
; X
, Y
: Integer);
229 procedure RenderPanelPaint(Sender
: TObject
);
230 procedure RenderPanelResize(Sender
: TObject
);
231 procedure Splitter1Moved(Sender
: TObject
);
232 procedure MapTestCheck(Sender
: TObject
);
233 procedure vleObjectPropertyEditButtonClick(Sender
: TObject
);
234 procedure vleObjectPropertyApply(Sender
: TObject
);
235 procedure vleObjectPropertyGetPickList(Sender
: TObject
; const KeyName
: String; Values
: TStrings
);
236 procedure vleObjectPropertyKeyDown(Sender
: TObject
; var Key
: Word;
238 procedure tbGridOnClick(Sender
: TObject
);
239 procedure miMapPreviewClick(Sender
: TObject
);
240 procedure miLayer1Click(Sender
: TObject
);
241 procedure miLayer2Click(Sender
: TObject
);
242 procedure miLayer3Click(Sender
: TObject
);
243 procedure miLayer4Click(Sender
: TObject
);
244 procedure miLayer5Click(Sender
: TObject
);
245 procedure miLayer6Click(Sender
: TObject
);
246 procedure miLayer7Click(Sender
: TObject
);
247 procedure miLayer8Click(Sender
: TObject
);
248 procedure miLayer9Click(Sender
: TObject
);
249 procedure tbShowClick(Sender
: TObject
);
250 procedure miSnapToGridClick(Sender
: TObject
);
251 procedure miMiniMapClick(Sender
: TObject
);
252 procedure miSwitchGridClick(Sender
: TObject
);
253 procedure miShowEdgesClick(Sender
: TObject
);
254 procedure minexttabClick(Sender
: TObject
);
255 procedure miSaveMiniMapClick(Sender
: TObject
);
256 procedure bClearTextureClick(Sender
: TObject
);
257 procedure miPackMapClick(Sender
: TObject
);
258 procedure miTestMapClick(Sender
: TObject
);
259 procedure sbVerticalScroll(Sender
: TObject
; ScrollCode
: TScrollCode
;
260 var ScrollPos
: Integer);
261 procedure sbHorizontalScroll(Sender
: TObject
; ScrollCode
: TScrollCode
;
262 var ScrollPos
: Integer);
263 procedure miOpenWadMapClick(Sender
: TObject
);
264 procedure selectall1Click(Sender
: TObject
);
265 procedure Splitter1CanResize(Sender
: TObject
; var NewSize
: Integer;
266 var Accept
: Boolean);
267 procedure Splitter2CanResize(Sender
: TObject
; var NewSize
: Integer;
268 var Accept
: Boolean);
269 procedure vleObjectPropertyEnter(Sender
: TObject
);
270 procedure vleObjectPropertyExit(Sender
: TObject
);
271 procedure FormKeyUp(Sender
: TObject
; var Key
: Word;
275 procedure OnIdle(Sender
: TObject
; var Done
: Boolean);
276 procedure RefillRecentMenu (menu
: TMenuItem
; start
: Integer; fmt
: AnsiString);
278 procedure RefreshRecentMenu();
279 procedure OpenMapFile(FileName
: String);
280 function RenderMousePos(): TPoint
;
281 procedure RecountSelectedObjects();
287 LAYER_FOREGROUND
= 2;
295 TEST_MAP_NAME
= '$$$_TEST_$$$';
296 LANGUAGE_FILE_NAME
= '_Editor.txt';
307 DotStepOne
, DotStepTwo
: Word;
309 DrawTexturePanel
: Boolean;
310 DrawPanelSize
: Boolean;
312 PreviewColor
: TColor
;
313 UseCheckerboard
: Boolean;
315 RecentCount
: Integer;
316 RecentFiles
: TStringList
;
317 slInvalidTextures
: TStringList
;
319 TestGameMode
: String;
321 TestLimScore
: String;
322 TestOptionsTwoPlayers
: Boolean;
323 TestOptionsTeamDamage
: Boolean;
324 TestOptionsAllowExit
: Boolean;
325 TestOptionsWeaponStay
: Boolean;
326 TestOptionsMonstersDM
: Boolean;
327 TestD2dExe
, TestD2DArgs
: String;
328 TestMapOnce
: Boolean;
330 LayerEnabled
: Array [LAYER_BACK
..LAYER_TRIGGERS
] of Boolean =
331 (True, True, True, True, True, True, True, True, True);
332 ContourEnabled
: Array [LAYER_BACK
..LAYER_TRIGGERS
] of Boolean =
333 (False, False, False, False, False, False, False, False, False);
334 PreviewMode
: Byte = 0;
340 procedure OpenMap(FileName
: String; mapN
: String);
341 function AddTexture(aWAD
, aSection
, aTex
: String; silent
: Boolean): Boolean;
342 procedure RemoveSelectFromObjects();
343 procedure ChangeShownProperty(Name
: String; NewValue
: String);
348 f_options
, e_graphics
, e_log
, GL
, Math
,
349 f_mapoptions
, g_basic
, f_about
, f_mapoptimization
,
350 f_mapcheck
, f_addresource_texture
, g_textures
,
351 f_activationtype
, f_keys
, wadreader
, fileutil
,
352 MAPREADER
, f_selectmap
, f_savemap
, WADEDITOR
, MAPDEF
,
353 g_map
, f_saveminimap
, f_addresource
, CONFIG
, f_packmap
,
354 f_addresource_sound
, f_choosetype
,
355 g_language
, ClipBrd
, g_resources
, g_options
;
358 UNDO_DELETE_PANEL
= 1;
359 UNDO_DELETE_ITEM
= 2;
360 UNDO_DELETE_AREA
= 3;
361 UNDO_DELETE_MONSTER
= 4;
362 UNDO_DELETE_TRIGGER
= 5;
366 UNDO_ADD_MONSTER
= 9;
367 UNDO_ADD_TRIGGER
= 10;
368 UNDO_MOVE_PANEL
= 11;
371 UNDO_MOVE_MONSTER
= 14;
372 UNDO_MOVE_TRIGGER
= 15;
373 UNDO_RESIZE_PANEL
= 16;
374 UNDO_RESIZE_TRIGGER
= 17;
376 MOUSEACTION_NONE
= 0;
377 MOUSEACTION_DRAWPANEL
= 1;
378 MOUSEACTION_DRAWTRIGGER
= 2;
379 MOUSEACTION_MOVEOBJ
= 3;
380 MOUSEACTION_RESIZE
= 4;
381 MOUSEACTION_MOVEMAP
= 5;
382 MOUSEACTION_DRAWPRESS
= 6;
383 MOUSEACTION_NOACTION
= 7;
386 RESIZETYPE_VERTICAL
= 1;
387 RESIZETYPE_HORIZONTAL
= 2;
396 SELECTFLAG_TELEPORT
= 1;
398 SELECTFLAG_TEXTURE
= 3;
400 SELECTFLAG_MONSTER
= 5;
401 SELECTFLAG_SPAWNPOINT
= 6;
402 SELECTFLAG_SHOTPANEL
= 7;
403 SELECTFLAG_SELECTED
= 8;
405 RECENT_FILES_MENU_START
= 12;
407 CLIPBOARD_SIG
= 'DF:ED';
411 case UndoType
: Byte of
412 UNDO_DELETE_PANEL
: (Panel
: ^TPanel
);
413 UNDO_DELETE_ITEM
: (Item
: TItem
);
414 UNDO_DELETE_AREA
: (Area
: TArea
);
415 UNDO_DELETE_MONSTER
: (Monster
: TMonster
);
416 UNDO_DELETE_TRIGGER
: (Trigger
: TTrigger
);
421 UNDO_ADD_TRIGGER
: (AddID
: DWORD
);
426 UNDO_MOVE_TRIGGER
: (MoveID
: DWORD
; dX
, dY
: Integer);
428 UNDO_RESIZE_TRIGGER
: (ResizeID
: DWORD
; dW
, dH
: Integer);
433 case ObjectType
: Byte of
434 OBJECT_PANEL
: (Panel
: ^TPanel
);
435 OBJECT_ITEM
: (Item
: TItem
);
436 OBJECT_AREA
: (Area
: TArea
);
437 OBJECT_MONSTER
: (Monster
: TMonster
);
438 OBJECT_TRIGGER
: (Trigger
: TTrigger
);
441 TCopyRecArray
= Array of TCopyRec
;
445 gDataLoaded
: Boolean = False;
446 ShowMap
: Boolean = False;
447 DrawRect
: PRect
= nil;
448 SnapToGrid
: Boolean = True;
450 MousePos
: Types
.TPoint
;
451 LastMovePoint
: Types
.TPoint
;
455 MouseLDownPos
: Types
.TPoint
;
456 MouseRDownPos
: Types
.TPoint
;
457 MouseMDownPos
: Types
.TPoint
;
459 SelectFlag
: Byte = SELECTFLAG_NONE
;
460 MouseAction
: Byte = MOUSEACTION_NONE
;
461 ResizeType
: Byte = RESIZETYPE_NONE
;
462 ResizeDirection
: Byte = RESIZEDIR_NONE
;
464 DrawPressRect
: Boolean = False;
465 EditingProperties
: Boolean = False;
467 UndoBuffer
: Array of Array of TUndoRec
= nil;
469 MapTestProcess
: TProcessUTF8
;
474 //----------------------------------------
475 //Далее идут вспомогательные процедуры
476 //----------------------------------------
478 function NameToBool(Name
: String): Boolean;
480 if Name
= BoolNames
[True] then
486 function NameToDir(Name
: String): TDirection
;
488 if Name
= DirNames
[D_LEFT
] then
494 function NameToDirAdv(Name
: String): Byte;
496 if Name
= DirNamesAdv
[1] then
499 if Name
= DirNamesAdv
[2] then
502 if Name
= DirNamesAdv
[3] then
508 function ActivateToStr(ActivateType
: Byte): String;
512 if ByteBool(ACTIVATE_PLAYERCOLLIDE
and ActivateType
) then
513 Result
:= Result
+ '+PC';
514 if ByteBool(ACTIVATE_MONSTERCOLLIDE
and ActivateType
) then
515 Result
:= Result
+ '+MC';
516 if ByteBool(ACTIVATE_PLAYERPRESS
and ActivateType
) then
517 Result
:= Result
+ '+PP';
518 if ByteBool(ACTIVATE_MONSTERPRESS
and ActivateType
) then
519 Result
:= Result
+ '+MP';
520 if ByteBool(ACTIVATE_SHOT
and ActivateType
) then
521 Result
:= Result
+ '+SH';
522 if ByteBool(ACTIVATE_NOMONSTER
and ActivateType
) then
523 Result
:= Result
+ '+NM';
525 if (Result
<> '') and (Result
[1] = '+') then
526 Delete(Result
, 1, 1);
529 function StrToActivate(Str
: String): Byte;
533 if Pos('PC', Str
) > 0 then
534 Result
:= ACTIVATE_PLAYERCOLLIDE
;
535 if Pos('MC', Str
) > 0 then
536 Result
:= Result
or ACTIVATE_MONSTERCOLLIDE
;
537 if Pos('PP', Str
) > 0 then
538 Result
:= Result
or ACTIVATE_PLAYERPRESS
;
539 if Pos('MP', Str
) > 0 then
540 Result
:= Result
or ACTIVATE_MONSTERPRESS
;
541 if Pos('SH', Str
) > 0 then
542 Result
:= Result
or ACTIVATE_SHOT
;
543 if Pos('NM', Str
) > 0 then
544 Result
:= Result
or ACTIVATE_NOMONSTER
;
547 function KeyToStr(Key
: Byte): String;
551 if ByteBool(KEY_RED
and Key
) then
552 Result
:= Result
+ '+RK';
553 if ByteBool(KEY_GREEN
and Key
) then
554 Result
:= Result
+ '+GK';
555 if ByteBool(KEY_BLUE
and Key
) then
556 Result
:= Result
+ '+BK';
557 if ByteBool(KEY_REDTEAM
and Key
) then
558 Result
:= Result
+ '+RT';
559 if ByteBool(KEY_BLUETEAM
and Key
) then
560 Result
:= Result
+ '+BT';
562 if (Result
<> '') and (Result
[1] = '+') then
563 Delete(Result
, 1, 1);
566 function StrToKey(Str
: String): Byte;
570 if Pos('RK', Str
) > 0 then
572 if Pos('GK', Str
) > 0 then
573 Result
:= Result
or KEY_GREEN
;
574 if Pos('BK', Str
) > 0 then
575 Result
:= Result
or KEY_BLUE
;
576 if Pos('RT', Str
) > 0 then
577 Result
:= Result
or KEY_REDTEAM
;
578 if Pos('BT', Str
) > 0 then
579 Result
:= Result
or KEY_BLUETEAM
;
582 function EffectToStr(Effect
: Byte): String;
584 if Effect
in [EFFECT_TELEPORT
..EFFECT_FIRE
] then
585 Result
:= EffectNames
[Effect
]
587 Result
:= EffectNames
[EFFECT_NONE
];
590 function StrToEffect(Str
: String): Byte;
594 Result
:= EFFECT_NONE
;
595 for i
:= EFFECT_TELEPORT
to EFFECT_FIRE
do
596 if EffectNames
[i
] = Str
then
603 function MonsterToStr(MonType
: Byte): String;
605 if MonType
in [MONSTER_DEMON
..MONSTER_MAN
] then
606 Result
:= MonsterNames
[MonType
]
608 Result
:= MonsterNames
[MONSTER_ZOMBY
];
611 function StrToMonster(Str
: String): Byte;
615 Result
:= MONSTER_ZOMBY
;
616 for i
:= MONSTER_DEMON
to MONSTER_MAN
do
617 if MonsterNames
[i
] = Str
then
624 function ItemToStr(ItemType
: Byte): String;
626 if ItemType
in [ITEM_MEDKIT_SMALL
..ITEM_MAX
] then
627 Result
:= ItemNames
[ItemType
]
629 Result
:= ItemNames
[ITEM_AMMO_BULLETS
];
632 function StrToItem(Str
: String): Byte;
636 Result
:= ITEM_AMMO_BULLETS
;
637 for i
:= ITEM_MEDKIT_SMALL
to ITEM_MAX
do
638 if ItemNames
[i
] = Str
then
645 function ShotToStr(ShotType
: Byte): String;
647 if ShotType
in [TRIGGER_SHOT_PISTOL
..TRIGGER_SHOT_MAX
] then
648 Result
:= ShotNames
[ShotType
]
650 Result
:= ShotNames
[TRIGGER_SHOT_PISTOL
];
653 function StrToShot(Str
: String): Byte;
657 Result
:= TRIGGER_SHOT_PISTOL
;
658 for i
:= TRIGGER_SHOT_PISTOL
to TRIGGER_SHOT_MAX
do
659 if ShotNames
[i
] = Str
then
666 function SelectedObjectCount(): Word;
672 if SelectedObjects
= nil then
675 for a
:= 0 to High(SelectedObjects
) do
676 if SelectedObjects
[a
].Live
then
677 Result
:= Result
+ 1;
680 function GetFirstSelected(): Integer;
686 if SelectedObjects
= nil then
689 for a
:= 0 to High(SelectedObjects
) do
690 if SelectedObjects
[a
].Live
then
697 function Normalize16(x
: Integer): Integer;
699 Result
:= (x
div 16) * 16;
702 procedure MoveMap(X
, Y
: Integer);
704 rx
, ry
, ScaleSz
: Integer;
706 with MainForm
.RenderPanel
do
708 ScaleSz
:= 16 div Scale
;
709 // Размер видимой части карты:
710 rx
:= Min(Normalize16(Width
), Normalize16(gMapInfo
.Width
)) div 2;
711 ry
:= Min(Normalize16(Height
), Normalize16(gMapInfo
.Height
)) div 2;
712 // Место клика на мини-карте:
713 MapOffset
.X
:= X
- (Width
- Max(gMapInfo
.Width
div ScaleSz
, 1) - 1);
714 MapOffset
.Y
:= Y
- 1;
715 // Это же место на "большой" карте:
716 MapOffset
.X
:= MapOffset
.X
* ScaleSz
;
717 MapOffset
.Y
:= MapOffset
.Y
* ScaleSz
;
718 // Левый верхний угол новой видимой части карты:
719 MapOffset
.X
:= MapOffset
.X
- rx
;
720 MapOffset
.Y
:= MapOffset
.Y
- ry
;
722 MapOffset
.X
:= EnsureRange(MapOffset
.X
, MainForm
.sbHorizontal
.Min
, MainForm
.sbHorizontal
.Max
);
723 MapOffset
.Y
:= EnsureRange(MapOffset
.Y
, MainForm
.sbVertical
.Min
, MainForm
.sbVertical
.Max
);
725 // MapOffset.X := Normalize16(MapOffset.X);
726 // MapOffset.Y := Normalize16(MapOffset.Y);
729 MainForm
.sbHorizontal
.Position
:= MapOffset
.X
;
730 MainForm
.sbVertical
.Position
:= MapOffset
.Y
;
732 MapOffset
.X
:= -MapOffset
.X
;
733 MapOffset
.Y
:= -MapOffset
.Y
;
738 function IsTexturedPanel(PanelType
: Word): Boolean;
740 Result
:= WordBool(PanelType
and (PANEL_WALL
or PANEL_BACK
or PANEL_FORE
or
741 PANEL_STEP
or PANEL_OPENDOOR
or PANEL_CLOSEDOOR
or
742 PANEL_WATER
or PANEL_ACID1
or PANEL_ACID2
));
745 procedure FillProperty();
750 MainForm
.vleObjectProperty
.Strings
.Clear();
751 MainForm
.RecountSelectedObjects();
753 // Отображаем свойства если выделен только один объект:
754 if SelectedObjectCount() <> 1 then
757 _id
:= GetFirstSelected();
758 if not SelectedObjects
[_id
].Live
then
761 with MainForm
.vleObjectProperty
do
762 with ItemProps
[InsertRow(MsgPropId
, IntToStr(SelectedObjects
[_id
].ID
), True)] do
764 EditStyle
:= esSimple
;
768 case SelectedObjects
[0].ObjectType
of
771 with MainForm
.vleObjectProperty
,
772 gPanels
[SelectedObjects
[_id
].ID
] do
774 with ItemProps
[InsertRow(MsgPropX
, IntToStr(X
), True)] do
776 EditStyle
:= esSimple
;
780 with ItemProps
[InsertRow(MsgPropY
, IntToStr(Y
), True)] do
782 EditStyle
:= esSimple
;
786 with ItemProps
[InsertRow(MsgPropWidth
, IntToStr(Width
), True)] do
788 EditStyle
:= esSimple
;
792 with ItemProps
[InsertRow(MsgPropHeight
, IntToStr(Height
), True)] do
794 EditStyle
:= esSimple
;
798 with ItemProps
[InsertRow(MsgPropPanelType
, GetPanelName(PanelType
), True)] do
800 EditStyle
:= esEllipsis
;
804 if IsTexturedPanel(PanelType
) then
805 begin // Может быть текстура
806 with ItemProps
[InsertRow(MsgPropPanelTex
, TextureName
, True)] do
808 EditStyle
:= esEllipsis
;
812 if TextureName
<> '' then
813 begin // Есть текстура
814 with ItemProps
[InsertRow(MsgPropPanelAlpha
, IntToStr(Alpha
), True)] do
816 EditStyle
:= esSimple
;
820 with ItemProps
[InsertRow(MsgPropPanelBlend
, BoolNames
[Blending
], True)] do
822 EditStyle
:= esPickList
;
832 with MainForm
.vleObjectProperty
,
833 gItems
[SelectedObjects
[_id
].ID
] do
835 with ItemProps
[InsertRow(MsgPropX
, IntToStr(X
), True)] do
837 EditStyle
:= esSimple
;
841 with ItemProps
[InsertRow(MsgPropY
, IntToStr(Y
), True)] do
843 EditStyle
:= esSimple
;
847 with ItemProps
[InsertRow(MsgPropDmOnly
, BoolNames
[OnlyDM
], True)] do
849 EditStyle
:= esPickList
;
853 with ItemProps
[InsertRow(MsgPropItemFalls
, BoolNames
[Fall
], True)] do
855 EditStyle
:= esPickList
;
863 with MainForm
.vleObjectProperty
,
864 gMonsters
[SelectedObjects
[_id
].ID
] do
866 with ItemProps
[InsertRow(MsgPropX
, IntToStr(X
), True)] do
868 EditStyle
:= esSimple
;
872 with ItemProps
[InsertRow(MsgPropY
, IntToStr(Y
), True)] do
874 EditStyle
:= esSimple
;
878 with ItemProps
[InsertRow(MsgPropDirection
, DirNames
[Direction
], True)] do
880 EditStyle
:= esPickList
;
888 with MainForm
.vleObjectProperty
,
889 gAreas
[SelectedObjects
[_id
].ID
] do
891 with ItemProps
[InsertRow(MsgPropX
, IntToStr(X
), True)] do
893 EditStyle
:= esSimple
;
897 with ItemProps
[InsertRow(MsgPropY
, IntToStr(Y
), True)] do
899 EditStyle
:= esSimple
;
903 with ItemProps
[InsertRow(MsgPropDirection
, DirNames
[Direction
], True)] do
905 EditStyle
:= esPickList
;
913 with MainForm
.vleObjectProperty
,
914 gTriggers
[SelectedObjects
[_id
].ID
] do
916 with ItemProps
[InsertRow(MsgPropTrType
, GetTriggerName(TriggerType
), True)] do
918 EditStyle
:= esSimple
;
922 with ItemProps
[InsertRow(MsgPropX
, IntToStr(X
), True)] do
924 EditStyle
:= esSimple
;
928 with ItemProps
[InsertRow(MsgPropY
, IntToStr(Y
), True)] do
930 EditStyle
:= esSimple
;
934 with ItemProps
[InsertRow(MsgPropWidth
, IntToStr(Width
), True)] do
936 EditStyle
:= esSimple
;
940 with ItemProps
[InsertRow(MsgPropHeight
, IntToStr(Height
), True)] do
942 EditStyle
:= esSimple
;
946 with ItemProps
[InsertRow(MsgPropTrEnabled
, BoolNames
[Enabled
], True)] do
948 EditStyle
:= esPickList
;
952 with ItemProps
[InsertRow(MsgPropTrTexturePanel
, IntToStr(TexturePanel
), True)] do
954 EditStyle
:= esEllipsis
;
958 with ItemProps
[InsertRow(MsgPropTrActivation
, ActivateToStr(ActivateType
), True)] do
960 EditStyle
:= esEllipsis
;
964 with ItemProps
[InsertRow(MsgPropTrKeys
, KeyToStr(Key
), True)] do
966 EditStyle
:= esEllipsis
;
973 str
:= win2utf(Data
.MapName
);
974 with ItemProps
[InsertRow(MsgPropTrNextMap
, str
, True)] do
976 EditStyle
:= esEllipsis
;
983 with ItemProps
[InsertRow(MsgPropTrTeleportTo
, Format('(%d:%d)', [Data
.TargetPoint
.X
, Data
.TargetPoint
.Y
]), True)] do
985 EditStyle
:= esEllipsis
;
989 with ItemProps
[InsertRow(MsgPropTrD2d
, BoolNames
[Data
.d2d_teleport
], True)] do
991 EditStyle
:= esPickList
;
995 with ItemProps
[InsertRow(MsgPropTrTeleportSilent
, BoolNames
[Data
.silent_teleport
], True)] do
997 EditStyle
:= esPickList
;
1001 with ItemProps
[InsertRow(MsgPropTrTeleportDir
, DirNamesAdv
[Data
.TlpDir
], True)] do
1003 EditStyle
:= esPickList
;
1008 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
,
1009 TRIGGER_DOOR
, TRIGGER_DOOR5
:
1011 with ItemProps
[InsertRow(MsgPropTrDoorPanel
, IntToStr(Data
.PanelID
), True)] do
1013 EditStyle
:= esEllipsis
;
1017 with ItemProps
[InsertRow(MsgPropTrSilent
, BoolNames
[Data
.NoSound
], True)] do
1019 EditStyle
:= esPickList
;
1023 with ItemProps
[InsertRow(MsgPropTrD2d
, BoolNames
[Data
.d2d_doors
], True)] do
1025 EditStyle
:= esPickList
;
1030 TRIGGER_CLOSETRAP
, TRIGGER_TRAP
:
1032 with ItemProps
[InsertRow(MsgPropTrTrapPanel
, IntToStr(Data
.PanelID
), True)] do
1034 EditStyle
:= esEllipsis
;
1038 with ItemProps
[InsertRow(MsgPropTrSilent
, BoolNames
[Data
.NoSound
], True)] do
1040 EditStyle
:= esPickList
;
1044 with ItemProps
[InsertRow(MsgPropTrD2d
, BoolNames
[Data
.d2d_doors
], True)] do
1046 EditStyle
:= esPickList
;
1051 TRIGGER_PRESS
, TRIGGER_ON
, TRIGGER_OFF
,
1054 with ItemProps
[InsertRow(MsgPropTrExArea
,
1055 Format('(%d:%d %d:%d)', [Data
.tX
, Data
.tY
, Data
.tWidth
, Data
.tHeight
]), True)] do
1057 EditStyle
:= esEllipsis
;
1061 with ItemProps
[InsertRow(MsgPropTrExDelay
, IntToStr(Data
.Wait
), True)] do
1063 EditStyle
:= esSimple
;
1067 with ItemProps
[InsertRow(MsgPropTrExCount
, IntToStr(Data
.Count
), True)] do
1069 EditStyle
:= esSimple
;
1073 with ItemProps
[InsertRow(MsgPropTrExMonster
, IntToStr(Data
.MonsterID
-1), True)] do
1075 EditStyle
:= esEllipsis
;
1079 if TriggerType
= TRIGGER_PRESS
then
1080 with ItemProps
[InsertRow(MsgPropTrExRandom
, BoolNames
[Data
.ExtRandom
], True)] do
1082 EditStyle
:= esPickList
;
1090 TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
:
1092 with ItemProps
[InsertRow(MsgPropTrLiftPanel
, IntToStr(Data
.PanelID
), True)] do
1094 EditStyle
:= esEllipsis
;
1098 with ItemProps
[InsertRow(MsgPropTrSilent
, BoolNames
[Data
.NoSound
], True)] do
1100 EditStyle
:= esPickList
;
1104 with ItemProps
[InsertRow(MsgPropTrD2d
, BoolNames
[Data
.d2d_doors
], True)] do
1106 EditStyle
:= esPickList
;
1113 with ItemProps
[InsertRow(MsgPropTrTextureOnce
, BoolNames
[Data
.ActivateOnce
], True)] do
1115 EditStyle
:= esPickList
;
1119 with ItemProps
[InsertRow(MsgPropTrTextureAnimOnce
, BoolNames
[Data
.AnimOnce
], True)] do
1121 EditStyle
:= esPickList
;
1128 str
:= win2utf(Data
.SoundName
);
1129 with ItemProps
[InsertRow(MsgPropTrSoundName
, str
, True)] do
1131 EditStyle
:= esEllipsis
;
1135 with ItemProps
[InsertRow(MsgPropTrSoundVolume
, IntToStr(Data
.Volume
), True)] do
1137 EditStyle
:= esSimple
;
1141 with ItemProps
[InsertRow(MsgPropTrSoundPan
, IntToStr(Data
.Pan
), True)] do
1143 EditStyle
:= esSimple
;
1147 with ItemProps
[InsertRow(MsgPropTrSoundCount
, IntToStr(Data
.PlayCount
), True)] do
1149 EditStyle
:= esSimple
;
1153 with ItemProps
[InsertRow(MsgPropTrSoundLocal
, BoolNames
[Data
.Local
], True)] do
1155 EditStyle
:= esPickList
;
1159 with ItemProps
[InsertRow(MsgPropTrSoundSwitch
, BoolNames
[Data
.SoundSwitch
], True)] do
1161 EditStyle
:= esPickList
;
1166 TRIGGER_SPAWNMONSTER
:
1168 with ItemProps
[InsertRow(MsgPropTrMonsterType
, MonsterToStr(Data
.MonType
), True)] do
1170 EditStyle
:= esEllipsis
;
1174 with ItemProps
[InsertRow(MsgPropTrSpawnTo
,
1175 Format('(%d:%d)', [Data
.MonPos
.X
, Data
.MonPos
.Y
]), True)] do
1177 EditStyle
:= esEllipsis
;
1181 with ItemProps
[InsertRow(MsgPropDirection
, DirNames
[TDirection(Data
.MonDir
)], True)] do
1183 EditStyle
:= esPickList
;
1187 with ItemProps
[InsertRow(MsgPropTrHealth
, IntToStr(Data
.MonHealth
), True)] do
1189 EditStyle
:= esSimple
;
1193 with ItemProps
[InsertRow(MsgPropTrMonsterActive
, BoolNames
[Data
.MonActive
], True)] do
1195 EditStyle
:= esPickList
;
1199 with ItemProps
[InsertRow(MsgPropTrCount
, IntToStr(Data
.MonCount
), True)] do
1201 EditStyle
:= esSimple
;
1205 with ItemProps
[InsertRow(MsgPropTrFxType
, EffectToStr(Data
.MonEffect
), True)] do
1207 EditStyle
:= esEllipsis
;
1211 with ItemProps
[InsertRow(MsgPropTrSpawnMax
, IntToStr(Data
.MonMax
), True)] do
1213 EditStyle
:= esSimple
;
1217 with ItemProps
[InsertRow(MsgPropTrSpawnDelay
, IntToStr(Data
.MonDelay
), True)] do
1219 EditStyle
:= esSimple
;
1223 case Data
.MonBehav
of
1224 1: str
:= MsgPropTrMonsterBehaviour1
;
1225 2: str
:= MsgPropTrMonsterBehaviour2
;
1226 3: str
:= MsgPropTrMonsterBehaviour3
;
1227 4: str
:= MsgPropTrMonsterBehaviour4
;
1228 5: str
:= MsgPropTrMonsterBehaviour5
;
1229 else str
:= MsgPropTrMonsterBehaviour0
;
1231 with ItemProps
[InsertRow(MsgPropTrMonsterBehaviour
, str
, True)] do
1233 EditStyle
:= esPickList
;
1240 with ItemProps
[InsertRow(MsgPropTrItemType
, ItemToStr(Data
.ItemType
), True)] do
1242 EditStyle
:= esEllipsis
;
1246 with ItemProps
[InsertRow(MsgPropTrSpawnTo
,
1247 Format('(%d:%d)', [Data
.ItemPos
.X
, Data
.ItemPos
.Y
]), True)] do
1249 EditStyle
:= esEllipsis
;
1253 with ItemProps
[InsertRow(MsgPropDmOnly
, BoolNames
[Data
.ItemOnlyDM
], True)] do
1255 EditStyle
:= esPickList
;
1259 with ItemProps
[InsertRow(MsgPropItemFalls
, BoolNames
[Data
.ItemFalls
], True)] do
1261 EditStyle
:= esPickList
;
1265 with ItemProps
[InsertRow(MsgPropTrCount
, IntToStr(Data
.ItemCount
), True)] do
1267 EditStyle
:= esSimple
;
1271 with ItemProps
[InsertRow(MsgPropTrFxType
, EffectToStr(Data
.ItemEffect
), True)] do
1273 EditStyle
:= esEllipsis
;
1277 with ItemProps
[InsertRow(MsgPropTrSpawnMax
, IntToStr(Data
.ItemMax
), True)] do
1279 EditStyle
:= esSimple
;
1283 with ItemProps
[InsertRow(MsgPropTrSpawnDelay
, IntToStr(Data
.ItemDelay
), True)] do
1285 EditStyle
:= esSimple
;
1292 str
:= win2utf(Data
.MusicName
);
1293 with ItemProps
[InsertRow(MsgPropTrMusicName
, str
, True)] do
1295 EditStyle
:= esEllipsis
;
1299 if Data
.MusicAction
= 1 then
1300 str
:= MsgPropTrMusicOn
1302 str
:= MsgPropTrMusicOff
;
1304 with ItemProps
[InsertRow(MsgPropTrMusicAct
, str
, True)] do
1306 EditStyle
:= esPickList
;
1313 with ItemProps
[InsertRow(MsgPropTrPushAngle
, IntToStr(Data
.PushAngle
), True)] do
1315 EditStyle
:= esSimple
;
1318 with ItemProps
[InsertRow(MsgPropTrPushForce
, IntToStr(Data
.PushForce
), True)] do
1320 EditStyle
:= esSimple
;
1323 with ItemProps
[InsertRow(MsgPropTrPushReset
, BoolNames
[Data
.ResetVel
], True)] do
1325 EditStyle
:= esPickList
;
1332 case Data
.ScoreAction
of
1333 1: str
:= MsgPropTrScoreAct1
;
1334 2: str
:= MsgPropTrScoreAct2
;
1335 3: str
:= MsgPropTrScoreAct3
;
1336 else str
:= MsgPropTrScoreAct0
;
1338 with ItemProps
[InsertRow(MsgPropTrScoreAct
, str
, True)] do
1340 EditStyle
:= esPickList
;
1343 with ItemProps
[InsertRow(MsgPropTrCount
, IntToStr(Data
.ScoreCount
), True)] do
1345 EditStyle
:= esSimple
;
1348 case Data
.ScoreTeam
of
1349 1: str
:= MsgPropTrScoreTeam1
;
1350 2: str
:= MsgPropTrScoreTeam2
;
1351 3: str
:= MsgPropTrScoreTeam3
;
1352 else str
:= MsgPropTrScoreTeam0
;
1354 with ItemProps
[InsertRow(MsgPropTrScoreTeam
, str
, True)] do
1356 EditStyle
:= esPickList
;
1359 with ItemProps
[InsertRow(MsgPropTrScoreCon
, BoolNames
[Data
.ScoreCon
], True)] do
1361 EditStyle
:= esPickList
;
1364 with ItemProps
[InsertRow(MsgPropTrScoreMsg
, BoolNames
[Data
.ScoreMsg
], True)] do
1366 EditStyle
:= esPickList
;
1373 case Data
.MessageKind
of
1374 1: str
:= MsgPropTrMessageKind1
;
1375 else str
:= MsgPropTrMessageKind0
;
1377 with ItemProps
[InsertRow(MsgPropTrMessageKind
, str
, True)] do
1379 EditStyle
:= esPickList
;
1382 case Data
.MessageSendTo
of
1383 1: str
:= MsgPropTrMessageTo1
;
1384 2: str
:= MsgPropTrMessageTo2
;
1385 3: str
:= MsgPropTrMessageTo3
;
1386 4: str
:= MsgPropTrMessageTo4
;
1387 5: str
:= MsgPropTrMessageTo5
;
1388 else str
:= MsgPropTrMessageTo0
;
1390 with ItemProps
[InsertRow(MsgPropTrMessageTo
, str
, True)] do
1392 EditStyle
:= esPickList
;
1395 str
:= win2utf(Data
.MessageText
);
1396 with ItemProps
[InsertRow(MsgPropTrMessageText
, str
, True)] do
1398 EditStyle
:= esSimple
;
1401 with ItemProps
[InsertRow(MsgPropTrMessageTime
, IntToStr(Data
.MessageTime
), True)] do
1403 EditStyle
:= esSimple
;
1410 with ItemProps
[InsertRow(MsgPropTrDamageValue
, IntToStr(Data
.DamageValue
), True)] do
1412 EditStyle
:= esSimple
;
1415 with ItemProps
[InsertRow(MsgPropTrInterval
, IntToStr(Data
.DamageInterval
), True)] do
1417 EditStyle
:= esSimple
;
1420 case Data
.DamageKind
of
1421 3: str
:= MsgPropTrDamageKind3
;
1422 4: str
:= MsgPropTrDamageKind4
;
1423 5: str
:= MsgPropTrDamageKind5
;
1424 6: str
:= MsgPropTrDamageKind6
;
1425 7: str
:= MsgPropTrDamageKind7
;
1426 8: str
:= MsgPropTrDamageKind8
;
1427 else str
:= MsgPropTrDamageKind0
;
1429 with ItemProps
[InsertRow(MsgPropTrDamageKind
, str
, True)] do
1431 EditStyle
:= esPickList
;
1438 with ItemProps
[InsertRow(MsgPropTrHealth
, IntToStr(Data
.HealValue
), True)] do
1440 EditStyle
:= esSimple
;
1443 with ItemProps
[InsertRow(MsgPropTrInterval
, IntToStr(Data
.HealInterval
), True)] do
1445 EditStyle
:= esSimple
;
1448 with ItemProps
[InsertRow(MsgPropTrHealthMax
, BoolNames
[Data
.HealMax
], True)] do
1450 EditStyle
:= esPickList
;
1453 with ItemProps
[InsertRow(MsgPropTrSilent
, BoolNames
[Data
.HealSilent
], True)] do
1455 EditStyle
:= esPickList
;
1462 with ItemProps
[InsertRow(MsgPropTrShotType
, ShotToStr(Data
.ShotType
), True)] do
1464 EditStyle
:= esEllipsis
;
1468 with ItemProps
[InsertRow(MsgPropTrShotSound
, BoolNames
[Data
.ShotSound
], True)] do
1470 EditStyle
:= esPickList
;
1474 with ItemProps
[InsertRow(MsgPropTrShotPanel
, IntToStr(Data
.ShotPanelID
), True)] do
1476 EditStyle
:= esEllipsis
;
1480 case Data
.ShotTarget
of
1481 1: str
:= MsgPropTrShotTo1
;
1482 2: str
:= MsgPropTrShotTo2
;
1483 3: str
:= MsgPropTrShotTo3
;
1484 4: str
:= MsgPropTrShotTo4
;
1485 5: str
:= MsgPropTrShotTo5
;
1486 6: str
:= MsgPropTrShotTo6
;
1487 else str
:= MsgPropTrShotTo0
;
1489 with ItemProps
[InsertRow(MsgPropTrShotTo
, str
, True)] do
1491 EditStyle
:= esPickList
;
1495 with ItemProps
[InsertRow(MsgPropTrShotSight
, IntToStr(Data
.ShotIntSight
), True)] do
1497 EditStyle
:= esSimple
;
1501 case Data
.ShotAim
of
1502 1: str
:= MsgPropTrShotAim1
;
1503 2: str
:= MsgPropTrShotAim2
;
1504 3: str
:= MsgPropTrShotAim3
;
1505 else str
:= MsgPropTrShotAim0
;
1507 with ItemProps
[InsertRow(MsgPropTrShotAim
, str
, True)] do
1509 EditStyle
:= esPickList
;
1513 with ItemProps
[InsertRow(MsgPropTrSpawnTo
,
1514 Format('(%d:%d)', [Data
.ShotPos
.X
, Data
.ShotPos
.Y
]), True)] do
1516 EditStyle
:= esEllipsis
;
1520 with ItemProps
[InsertRow(MsgPropTrShotAngle
, IntToStr(Data
.ShotAngle
), True)] do
1522 EditStyle
:= esSimple
;
1526 with ItemProps
[InsertRow(MsgPropTrExDelay
, IntToStr(Data
.ShotWait
), True)] do
1528 EditStyle
:= esSimple
;
1532 with ItemProps
[InsertRow(MsgPropTrShotAcc
, IntToStr(Data
.ShotAccuracy
), True)] do
1534 EditStyle
:= esSimple
;
1538 with ItemProps
[InsertRow(MsgPropTrShotAmmo
, IntToStr(Data
.ShotAmmo
), True)] do
1540 EditStyle
:= esSimple
;
1544 with ItemProps
[InsertRow(MsgPropTrShotReload
, IntToStr(Data
.ShotIntReload
), True)] do
1546 EditStyle
:= esSimple
;
1553 with ItemProps
[InsertRow(MsgPropTrCount
, IntToStr(Data
.FXCount
), True)] do
1555 EditStyle
:= esSimple
;
1559 if Data
.FXType
= 0 then
1560 str
:= MsgPropTrEffectParticle
1562 str
:= MsgPropTrEffectAnimation
;
1563 with ItemProps
[InsertRow(MsgPropTrEffectType
, str
, True)] do
1565 EditStyle
:= esEllipsis
;
1570 if Data
.FXType
= 0 then
1571 case Data
.FXSubType
of
1572 TRIGGER_EFFECT_SLIQUID
:
1573 str
:= MsgPropTrEffectSliquid
;
1574 TRIGGER_EFFECT_LLIQUID
:
1575 str
:= MsgPropTrEffectLliquid
;
1576 TRIGGER_EFFECT_DLIQUID
:
1577 str
:= MsgPropTrEffectDliquid
;
1578 TRIGGER_EFFECT_BLOOD
:
1579 str
:= MsgPropTrEffectBlood
;
1580 TRIGGER_EFFECT_SPARK
:
1581 str
:= MsgPropTrEffectSpark
;
1582 TRIGGER_EFFECT_BUBBLE
:
1583 str
:= MsgPropTrEffectBubble
;
1585 if Data
.FXType
= 1 then
1587 if (Data
.FXSubType
= 0) or (Data
.FXSubType
> EFFECT_FIRE
) then
1588 Data
.FXSubType
:= EFFECT_TELEPORT
;
1589 str
:= EffectToStr(Data
.FXSubType
);
1591 with ItemProps
[InsertRow(MsgPropTrEffectSubtype
, str
, True)] do
1593 EditStyle
:= esEllipsis
;
1597 with ItemProps
[InsertRow(MsgPropTrEffectColor
, IntToStr(Data
.FXColorR
or (Data
.FXColorG
shl 8) or (Data
.FXColorB
shl 16)), True)] do
1599 EditStyle
:= esEllipsis
;
1603 with ItemProps
[InsertRow(MsgPropTrEffectCenter
, BoolNames
[Data
.FXPos
= 0], True)] do
1605 EditStyle
:= esPickList
;
1609 with ItemProps
[InsertRow(MsgPropTrExDelay
, IntToStr(Data
.FXWait
), True)] do
1611 EditStyle
:= esSimple
;
1615 with ItemProps
[InsertRow(MsgPropTrEffectVelx
, IntToStr(Data
.FXVelX
), True)] do
1617 EditStyle
:= esSimple
;
1621 with ItemProps
[InsertRow(MsgPropTrEffectVely
, IntToStr(Data
.FXVelY
), True)] do
1623 EditStyle
:= esSimple
;
1627 with ItemProps
[InsertRow(MsgPropTrEffectSpl
, IntToStr(Data
.FXSpreadL
), True)] do
1629 EditStyle
:= esSimple
;
1633 with ItemProps
[InsertRow(MsgPropTrEffectSpr
, IntToStr(Data
.FXSpreadR
), True)] do
1635 EditStyle
:= esSimple
;
1639 with ItemProps
[InsertRow(MsgPropTrEffectSpu
, IntToStr(Data
.FXSpreadU
), True)] do
1641 EditStyle
:= esSimple
;
1645 with ItemProps
[InsertRow(MsgPropTrEffectSpd
, IntToStr(Data
.FXSpreadD
), True)] do
1647 EditStyle
:= esSimple
;
1651 end; //case TriggerType
1653 end; // OBJECT_TRIGGER:
1657 procedure ChangeShownProperty(Name
: String; NewValue
: String);
1661 if SelectedObjectCount() <> 1 then
1663 if not SelectedObjects
[GetFirstSelected()].Live
then
1666 // Есть ли такой ключ:
1667 if MainForm
.vleObjectProperty
.FindRow(Name
, row
) then
1669 MainForm
.vleObjectProperty
.Values
[Name
] := NewValue
;
1673 procedure SelectObject(fObjectType
: Byte; fID
: DWORD
; Multi
: Boolean);
1682 // Уже выделен - убираем:
1683 if SelectedObjects
<> nil then
1684 for a
:= 0 to High(SelectedObjects
) do
1685 with SelectedObjects
[a
] do
1686 if Live
and (ID
= fID
) and
1687 (ObjectType
= fObjectType
) then
1696 SetLength(SelectedObjects
, Length(SelectedObjects
)+1);
1698 with SelectedObjects
[High(SelectedObjects
)] do
1700 ObjectType
:= fObjectType
;
1707 SetLength(SelectedObjects
, 1);
1709 with SelectedObjects
[0] do
1711 ObjectType
:= fObjectType
;
1717 MainForm
.miCopy
.Enabled
:= True;
1718 MainForm
.miCut
.Enabled
:= True;
1720 if fObjectType
= OBJECT_PANEL
then
1722 MainForm
.miToFore
.Enabled
:= True;
1723 MainForm
.miToBack
.Enabled
:= True;
1727 procedure RemoveSelectFromObjects();
1729 SelectedObjects
:= nil;
1730 DrawPressRect
:= False;
1731 MouseLDown
:= False;
1732 MouseRDown
:= False;
1733 MouseAction
:= MOUSEACTION_NONE
;
1734 SelectFlag
:= SELECTFLAG_NONE
;
1735 ResizeType
:= RESIZETYPE_NONE
;
1736 ResizeDirection
:= RESIZEDIR_NONE
;
1738 MainForm
.vleObjectProperty
.Strings
.Clear();
1740 MainForm
.miCopy
.Enabled
:= False;
1741 MainForm
.miCut
.Enabled
:= False;
1742 MainForm
.miToFore
.Enabled
:= False;
1743 MainForm
.miToBack
.Enabled
:= False;
1746 procedure DeleteSelectedObjects();
1751 if SelectedObjects
= nil then
1757 for a
:= 0 to High(SelectedObjects
) do
1758 with SelectedObjects
[a
] do
1763 SetLength(UndoBuffer
, Length(UndoBuffer
)+1);
1764 i
:= High(UndoBuffer
);
1768 SetLength(UndoBuffer
[i
], Length(UndoBuffer
[i
])+1);
1769 ii
:= High(UndoBuffer
[i
]);
1774 UndoBuffer
[i
, ii
].UndoType
:= UNDO_DELETE_PANEL
;
1775 New(UndoBuffer
[i
, ii
].Panel
);
1776 UndoBuffer
[i
, ii
].Panel
^ := gPanels
[ID
];
1780 UndoBuffer
[i
, ii
].UndoType
:= UNDO_DELETE_ITEM
;
1781 UndoBuffer
[i
, ii
].Item
:= gItems
[ID
];
1785 UndoBuffer
[i
, ii
].UndoType
:= UNDO_DELETE_AREA
;
1786 UndoBuffer
[i
, ii
].Area
:= gAreas
[ID
];
1790 UndoBuffer
[i
, ii
].UndoType
:= UNDO_DELETE_TRIGGER
;
1791 UndoBuffer
[i
, ii
].Trigger
:= gTriggers
[ID
];
1795 RemoveObject(ID
, ObjectType
);
1798 RemoveSelectFromObjects();
1800 MainForm
.miUndo
.Enabled
:= UndoBuffer
<> nil;
1801 MainForm
.RecountSelectedObjects();
1804 procedure Undo_Add(ObjectType
: Byte; ID
: DWORD
; Group
: Boolean = False);
1808 if (not Group
) or (Length(UndoBuffer
) = 0) then
1809 SetLength(UndoBuffer
, Length(UndoBuffer
)+1);
1810 SetLength(UndoBuffer
[High(UndoBuffer
)], Length(UndoBuffer
[High(UndoBuffer
)])+1);
1811 i
:= High(UndoBuffer
);
1812 ii
:= High(UndoBuffer
[i
]);
1816 UndoBuffer
[i
, ii
].UndoType
:= UNDO_ADD_PANEL
;
1818 UndoBuffer
[i
, ii
].UndoType
:= UNDO_ADD_ITEM
;
1820 UndoBuffer
[i
, ii
].UndoType
:= UNDO_ADD_MONSTER
;
1822 UndoBuffer
[i
, ii
].UndoType
:= UNDO_ADD_AREA
;
1824 UndoBuffer
[i
, ii
].UndoType
:= UNDO_ADD_TRIGGER
;
1827 UndoBuffer
[i
, ii
].AddID
:= ID
;
1829 MainForm
.miUndo
.Enabled
:= UndoBuffer
<> nil;
1832 procedure FullClear();
1834 RemoveSelectFromObjects();
1836 LoadSky(gMapInfo
.SkyName
);
1838 slInvalidTextures
.Clear();
1839 MapCheckForm
.lbErrorList
.Clear();
1840 MapCheckForm
.mErrorDescription
.Clear();
1842 MainForm
.miUndo
.Enabled
:= False;
1843 MainForm
.sbHorizontal
.Position
:= 0;
1844 MainForm
.sbVertical
.Position
:= 0;
1845 MainForm
.FormResize(nil);
1846 MainForm
.Caption
:= FormCaption
;
1851 procedure ErrorMessageBox(str
: String);
1853 Application
.MessageBox(PChar(str
), PChar(MsgMsgError
),
1854 MB_ICONINFORMATION
or MB_OK
or MB_DEFBUTTON1
);
1857 function CheckProperty(): Boolean;
1863 _id
:= GetFirstSelected();
1865 if SelectedObjects
[_id
].ObjectType
= OBJECT_PANEL
then
1866 with gPanels
[SelectedObjects
[_id
].ID
] do
1868 if TextureWidth
<> 0 then
1869 if StrToIntDef(MainForm
.vleObjectProperty
.Values
[MsgPropWidth
], 1) mod TextureWidth
<> 0 then
1871 ErrorMessageBox(Format(MsgMsgWrongTexwidth
,
1876 if TextureHeight
<> 0 then
1877 if StrToIntDef(Trim(MainForm
.vleObjectProperty
.Values
[MsgPropHeight
]), 1) mod TextureHeight
<> 0 then
1879 ErrorMessageBox(Format(MsgMsgWrongTexheight
,
1884 if IsTexturedPanel(PanelType
) and (TextureName
<> '') then
1885 if not (StrToIntDef(MainForm
.vleObjectProperty
.Values
[MsgPropPanelAlpha
], -1) in [0..255]) then
1887 ErrorMessageBox(MsgMsgWrongAlpha
);
1892 if SelectedObjects
[_id
].ObjectType
in [OBJECT_PANEL
, OBJECT_TRIGGER
] then
1893 if (StrToIntDef(MainForm
.vleObjectProperty
.Values
[MsgPropWidth
], 0) <= 0) or
1894 (StrToIntDef(MainForm
.vleObjectProperty
.Values
[MsgPropHeight
], 0) <= 0) then
1896 ErrorMessageBox(MsgMsgWrongSize
);
1900 if (Trim(MainForm
.vleObjectProperty
.Values
[MsgPropX
]) = '') or
1901 (Trim(MainForm
.vleObjectProperty
.Values
[MsgPropY
]) = '') then
1903 ErrorMessageBox(MsgMsgWrongXy
);
1910 procedure SelectTexture(ID
: Integer);
1912 MainForm
.lbTextureList
.ItemIndex
:= ID
;
1913 MainForm
.lbTextureListClick(nil);
1916 function AddTexture(aWAD
, aSection
, aTex
: String; silent
: Boolean): Boolean;
1918 a
, FrameLen
: Integer;
1921 ResourceName
: String;
1922 FullResourceName
: String;
1923 SectionName
: String;
1925 Width
, Height
: Word;
1933 if aSection
= '..' then
1936 SectionName
:= aSection
;
1939 aWAD
:= MsgWadSpecialMap
;
1941 if aWAD
= MsgWadSpecialMap
then
1943 g_ProcessResourceStr(OpenedMap
, @fn
, nil, nil);
1945 ResourceName
:= ':'+SectionName
+'\'+aTex
;
1948 if aWAD
= MsgWadSpecialTexs
then
1949 begin // Спец. текстуры
1951 ResourceName
:= aTex
;
1954 begin // Внешний WAD
1955 FileName
:= WadsDir
+ DirectorySeparator
+ aWAD
;
1956 ResourceName
:= aWAD
+':'+SectionName
+'\'+aTex
;
1961 // Есть ли уже такая текстура:
1962 for a
:= 0 to MainForm
.lbTextureList
.Items
.Count
-1 do
1963 if ResourceName
= MainForm
.lbTextureList
.Items
[a
] then
1966 ErrorMessageBox(Format(MsgMsgTextureAlready
,
1971 // Название ресурса <= 64 символов:
1972 if Length(ResourceName
) > 64 then
1975 ErrorMessageBox(Format(MsgMsgResName64
,
1983 if aWAD
= MsgWadSpecialTexs
then
1985 a
:= MainForm
.lbTextureList
.Items
.Add(ResourceName
);
1992 FullResourceName
:= FileName
+':'+SectionName
+'\'+aTex
;
1994 if IsAnim(FullResourceName
) then
1995 begin // Аним. текстура
1996 GetFrame(FullResourceName
, Data
, FrameLen
, Width
, Height
);
1998 if not g_CreateTextureMemorySize(Data
, FrameLen
, ResourceName
, 0, 0, Width
, Height
, 1) then
2000 a
:= MainForm
.lbTextureList
.Items
.Add(ResourceName
);
2002 else // Обычная текстура
2004 if not g_CreateTextureWAD(ResourceName
, FullResourceName
) then
2006 a
:= MainForm
.lbTextureList
.Items
.Add(ResourceName
);
2008 if (not ok
) and (slInvalidTextures
.IndexOf(ResourceName
) = -1) then
2010 slInvalidTextures
.Add(ResourceName
);
2013 if (a
> -1) and (not silent
) then
2020 procedure UpdateCaption(sMap
, sFile
, sRes
: String);
2023 if (sFile
= '') and (sRes
= '') and (sMap
= '') then
2024 Caption
:= FormCaption
2027 Caption
:= Format('%s - %s:%s', [FormCaption
, sFile
, sRes
])
2029 if (sFile
<> '') and (sRes
<> '') then
2030 Caption
:= Format('%s - %s (%s:%s)', [FormCaption
, sMap
, sFile
, sRes
])
2032 Caption
:= Format('%s - %s', [FormCaption
, sMap
]);
2035 procedure OpenMap(FileName
: String; mapN
: String);
2040 SelectMapForm
.Caption
:= MsgCapOpen
;
2041 SelectMapForm
.GetMaps(FileName
);
2043 if (FileName
= OpenedWAD
) and
2044 (OpenedMap
<> '') then
2046 MapName
:= OpenedMap
;
2047 while (Pos(':\', MapName
) > 0) do
2048 Delete(MapName
, 1, Pos(':\', MapName
) + 1);
2050 idx
:= SelectMapForm
.lbMapList
.Items
.IndexOf(MapName
);
2051 SelectMapForm
.lbMapList
.ItemIndex
:= idx
;
2054 if SelectMapForm
.lbMapList
.Count
> 0 then
2055 SelectMapForm
.lbMapList
.ItemIndex
:= 0
2057 SelectMapForm
.lbMapList
.ItemIndex
:= -1;
2062 idx
:= SelectMapForm
.lbMapList
.Items
.IndexOf(mapN
);
2066 if (SelectMapForm
.ShowModal() = mrOK
) and
2067 (SelectMapForm
.lbMapList
.ItemIndex
<> -1) then
2068 idx
:= SelectMapForm
.lbMapList
.ItemIndex
2073 MapName
:= SelectMapForm
.lbMapList
.Items
[idx
];
2079 pLoadProgress
.Left
:= (RenderPanel
.Width
div 2)-(pLoadProgress
.Width
div 2);
2080 pLoadProgress
.Top
:= (RenderPanel
.Height
div 2)-(pLoadProgress
.Height
div 2);
2081 pLoadProgress
.Show();
2083 OpenedMap
:= FileName
+':\'+MapName
;
2084 OpenedWAD
:= FileName
;
2086 idx
:= RecentFiles
.IndexOf(OpenedMap
);
2087 // Такая карта уже недавно открывалась:
2089 RecentFiles
.Delete(idx
);
2090 RecentFiles
.Insert(0, OpenedMap
);
2091 RefreshRecentMenu();
2095 pLoadProgress
.Hide();
2098 lbTextureList
.Sorted
:= True;
2099 lbTextureList
.Sorted
:= False;
2101 UpdateCaption(gMapInfo
.Name
, ExtractFileName(FileName
), MapName
);
2105 procedure MoveSelectedObjects(Wall
, alt
: Boolean; dx
, dy
: Integer);
2110 if SelectedObjects
= nil then
2117 for a
:= 0 to High(SelectedObjects
) do
2118 if SelectedObjects
[a
].Live
then
2120 if ObjectCollideLevel(SelectedObjects
[a
].ID
, SelectedObjects
[a
].ObjectType
, dx
, 0) then
2123 if ObjectCollideLevel(SelectedObjects
[a
].ID
, SelectedObjects
[a
].ObjectType
, 0, dy
) then
2126 if (not okX
) or (not okY
) then
2132 for a
:= 0 to High(SelectedObjects
) do
2133 if SelectedObjects
[a
].Live
then
2136 MoveObject(SelectedObjects
[a
].ObjectType
, SelectedObjects
[a
].ID
, dx
, 0);
2139 MoveObject(SelectedObjects
[a
].ObjectType
, SelectedObjects
[a
].ID
, 0, dy
);
2141 if alt
and (SelectedObjects
[a
].ObjectType
= OBJECT_TRIGGER
) then
2143 if gTriggers
[SelectedObjects
[a
].ID
].TriggerType
in [TRIGGER_PRESS
,
2144 TRIGGER_ON
, TRIGGER_OFF
, TRIGGER_ONOFF
] then
2145 begin // Двигаем зону Расширителя
2147 gTriggers
[SelectedObjects
[a
].ID
].Data
.tX
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.tX
+dx
;
2149 gTriggers
[SelectedObjects
[a
].ID
].Data
.tY
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.tY
+dy
;
2152 if gTriggers
[SelectedObjects
[a
].ID
].TriggerType
in [TRIGGER_TELEPORT
] then
2153 begin // Двигаем точку назначения Телепорта
2155 gTriggers
[SelectedObjects
[a
].ID
].Data
.TargetPoint
.X
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.TargetPoint
.X
+dx
;
2157 gTriggers
[SelectedObjects
[a
].ID
].Data
.TargetPoint
.Y
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.TargetPoint
.Y
+dy
;
2160 if gTriggers
[SelectedObjects
[a
].ID
].TriggerType
in [TRIGGER_SPAWNMONSTER
] then
2161 begin // Двигаем точку создания монстра
2163 gTriggers
[SelectedObjects
[a
].ID
].Data
.MonPos
.X
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.MonPos
.X
+dx
;
2165 gTriggers
[SelectedObjects
[a
].ID
].Data
.MonPos
.Y
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.MonPos
.Y
+dy
;
2168 if gTriggers
[SelectedObjects
[a
].ID
].TriggerType
in [TRIGGER_SPAWNITEM
] then
2169 begin // Двигаем точку создания предмета
2171 gTriggers
[SelectedObjects
[a
].ID
].Data
.ItemPos
.X
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.ItemPos
.X
+dx
;
2173 gTriggers
[SelectedObjects
[a
].ID
].Data
.ItemPos
.Y
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.ItemPos
.Y
+dy
;
2176 if gTriggers
[SelectedObjects
[a
].ID
].TriggerType
in [TRIGGER_SHOT
] then
2177 begin // Двигаем точку создания выстрела
2179 gTriggers
[SelectedObjects
[a
].ID
].Data
.ShotPos
.X
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.ShotPos
.X
+dx
;
2181 gTriggers
[SelectedObjects
[a
].ID
].Data
.ShotPos
.Y
:= gTriggers
[SelectedObjects
[a
].ID
].Data
.ShotPos
.Y
+dy
;
2186 LastMovePoint
:= MousePos
;
2190 procedure ShowLayer(Layer
: Byte; show
: Boolean);
2192 LayerEnabled
[Layer
] := show
;
2197 MainForm
.miLayer1
.Checked
:= show
;
2198 MainForm
.miLayerP1
.Checked
:= show
;
2202 MainForm
.miLayer2
.Checked
:= show
;
2203 MainForm
.miLayerP2
.Checked
:= show
;
2207 MainForm
.miLayer3
.Checked
:= show
;
2208 MainForm
.miLayerP3
.Checked
:= show
;
2212 MainForm
.miLayer4
.Checked
:= show
;
2213 MainForm
.miLayerP4
.Checked
:= show
;
2217 MainForm
.miLayer5
.Checked
:= show
;
2218 MainForm
.miLayerP5
.Checked
:= show
;
2222 MainForm
.miLayer6
.Checked
:= show
;
2223 MainForm
.miLayerP6
.Checked
:= show
;
2227 MainForm
.miLayer7
.Checked
:= show
;
2228 MainForm
.miLayerP7
.Checked
:= show
;
2232 MainForm
.miLayer8
.Checked
:= show
;
2233 MainForm
.miLayerP8
.Checked
:= show
;
2237 MainForm
.miLayer9
.Checked
:= show
;
2238 MainForm
.miLayerP9
.Checked
:= show
;
2242 RemoveSelectFromObjects();
2245 procedure SwitchLayer(Layer
: Byte);
2247 ShowLayer(Layer
, not LayerEnabled
[Layer
]);
2250 procedure SwitchMap();
2252 ShowMap
:= not ShowMap
;
2253 MainForm
.tbShowMap
.Down
:= ShowMap
;
2254 MainForm
.miMiniMap
.Checked
:= ShowMap
;
2257 procedure ShowEdges();
2259 if drEdge
[3] < 255 then
2262 drEdge
[3] := gAlphaEdge
;
2263 MainForm
.miShowEdges
.Checked
:= drEdge
[3] <> 255;
2266 function SelectedTexture(): String;
2268 if MainForm
.lbTextureList
.ItemIndex
<> -1 then
2269 Result
:= MainForm
.lbTextureList
.Items
[MainForm
.lbTextureList
.ItemIndex
]
2274 function IsSpecialTextureSel(): Boolean;
2276 Result
:= (MainForm
.lbTextureList
.ItemIndex
<> -1) and
2277 IsSpecialTexture(MainForm
.lbTextureList
.Items
[MainForm
.lbTextureList
.ItemIndex
]);
2280 function CopyBufferToString(var CopyBuf
: TCopyRecArray
): String;
2285 procedure AddInt(x
: Integer);
2287 Res
:= Res
+ IntToStr(x
) + ' ';
2293 if Length(CopyBuf
) = 0 then
2296 Res
:= CLIPBOARD_SIG
+ ' ';
2298 for i
:= 0 to High(CopyBuf
) do
2300 if (CopyBuf
[i
].ObjectType
= OBJECT_PANEL
) and
2301 (CopyBuf
[i
].Panel
= nil) then
2305 AddInt(CopyBuf
[i
].ObjectType
);
2308 // Свойства объекта:
2309 case CopyBuf
[i
].ObjectType
of
2311 with CopyBuf
[i
].Panel
^ do
2318 Res
:= Res
+ '"' + TextureName
+ '" ';
2320 AddInt(IfThen(Blending
, 1, 0));
2324 with CopyBuf
[i
].Item
do
2329 AddInt(IfThen(OnlyDM
, 1, 0));
2330 AddInt(IfThen(Fall
, 1, 0));
2334 with CopyBuf
[i
].Monster
do
2336 AddInt(MonsterType
);
2339 AddInt(IfThen(Direction
= D_LEFT
, 1, 0));
2343 with CopyBuf
[i
].Area
do
2348 AddInt(IfThen(Direction
= D_LEFT
, 1, 0));
2352 with CopyBuf
[i
].Trigger
do
2354 AddInt(TriggerType
);
2359 AddInt(ActivateType
);
2361 AddInt(IfThen(Enabled
, 1, 0));
2362 AddInt(TexturePanel
);
2364 for j
:= 0 to 127 do
2365 AddInt(Data
.Default
[j
]);
2373 procedure StringToCopyBuffer(Str
: String; var CopyBuf
: TCopyRecArray
;
2378 function GetNext(): String;
2383 if Str
[1] = '"' then
2395 Result
:= Copy(Str
, 1, p
-1);
2411 Result
:= Copy(Str
, 1, p
-1);
2421 if GetNext() <> CLIPBOARD_SIG
then
2427 t
:= StrToIntDef(GetNext(), 0);
2429 if (t
< OBJECT_PANEL
) or (t
> OBJECT_TRIGGER
) or
2430 (GetNext() <> ';') then
2431 begin // Что-то не то => пропускаем:
2439 i
:= Length(CopyBuf
);
2440 SetLength(CopyBuf
, i
+ 1);
2442 CopyBuf
[i
].ObjectType
:= t
;
2443 CopyBuf
[i
].Panel
:= nil;
2445 // Свойства объекта:
2449 New(CopyBuf
[i
].Panel
);
2451 with CopyBuf
[i
].Panel
^ do
2453 PanelType
:= StrToIntDef(GetNext(), PANEL_WALL
);
2454 X
:= StrToIntDef(GetNext(), 0);
2455 Y
:= StrToIntDef(GetNext(), 0);
2456 pmin
.X
:= Min(X
, pmin
.X
);
2457 pmin
.Y
:= Min(Y
, pmin
.Y
);
2458 Width
:= StrToIntDef(GetNext(), 16);
2459 Height
:= StrToIntDef(GetNext(), 16);
2460 TextureName
:= GetNext();
2461 Alpha
:= StrToIntDef(GetNext(), 0);
2462 Blending
:= (GetNext() = '1');
2467 with CopyBuf
[i
].Item
do
2469 ItemType
:= StrToIntDef(GetNext(), ITEM_MEDKIT_SMALL
);
2470 X
:= StrToIntDef(GetNext(), 0);
2471 Y
:= StrToIntDef(GetNext(), 0);
2472 pmin
.X
:= Min(X
, pmin
.X
);
2473 pmin
.Y
:= Min(Y
, pmin
.Y
);
2474 OnlyDM
:= (GetNext() = '1');
2475 Fall
:= (GetNext() = '1');
2479 with CopyBuf
[i
].Monster
do
2481 MonsterType
:= StrToIntDef(GetNext(), MONSTER_DEMON
);
2482 X
:= StrToIntDef(GetNext(), 0);
2483 Y
:= StrToIntDef(GetNext(), 0);
2484 pmin
.X
:= Min(X
, pmin
.X
);
2485 pmin
.Y
:= Min(Y
, pmin
.Y
);
2487 if GetNext() = '1' then
2490 Direction
:= D_RIGHT
;
2494 with CopyBuf
[i
].Area
do
2496 AreaType
:= StrToIntDef(GetNext(), AREA_PLAYERPOINT1
);
2497 X
:= StrToIntDef(GetNext(), 0);
2498 Y
:= StrToIntDef(GetNext(), 0);
2499 pmin
.X
:= Min(X
, pmin
.X
);
2500 pmin
.Y
:= Min(Y
, pmin
.Y
);
2501 if GetNext() = '1' then
2504 Direction
:= D_RIGHT
;
2508 with CopyBuf
[i
].Trigger
do
2510 TriggerType
:= StrToIntDef(GetNext(), TRIGGER_EXIT
);
2511 X
:= StrToIntDef(GetNext(), 0);
2512 Y
:= StrToIntDef(GetNext(), 0);
2513 pmin
.X
:= Min(X
, pmin
.X
);
2514 pmin
.Y
:= Min(Y
, pmin
.Y
);
2515 Width
:= StrToIntDef(GetNext(), 16);
2516 Height
:= StrToIntDef(GetNext(), 16);
2517 ActivateType
:= StrToIntDef(GetNext(), 0);
2518 Key
:= StrToIntDef(GetNext(), 0);
2519 Enabled
:= (GetNext() = '1');
2520 TexturePanel
:= StrToIntDef(GetNext(), 0);
2522 for j
:= 0 to 127 do
2523 Data
.Default
[j
] := StrToIntDef(GetNext(), 0);
2528 pmin
.X
:= Min(Data
.TargetPoint
.X
, pmin
.X
);
2529 pmin
.Y
:= Min(Data
.TargetPoint
.Y
, pmin
.Y
);
2531 TRIGGER_PRESS
, TRIGGER_ON
, TRIGGER_OFF
, TRIGGER_ONOFF
:
2533 pmin
.X
:= Min(Data
.tX
, pmin
.X
);
2534 pmin
.Y
:= Min(Data
.tY
, pmin
.Y
);
2536 TRIGGER_SPAWNMONSTER
:
2538 pmin
.X
:= Min(Data
.MonPos
.X
, pmin
.X
);
2539 pmin
.Y
:= Min(Data
.MonPos
.Y
, pmin
.Y
);
2543 pmin
.X
:= Min(Data
.ItemPos
.X
, pmin
.X
);
2544 pmin
.Y
:= Min(Data
.ItemPos
.Y
, pmin
.Y
);
2548 pmin
.X
:= Min(Data
.ShotPos
.X
, pmin
.X
);
2549 pmin
.Y
:= Min(Data
.ShotPos
.Y
, pmin
.Y
);
2557 //----------------------------------------
2558 //Закончились вспомогательные процедуры
2559 //----------------------------------------
2561 procedure TMainForm
.miRecentFileExecute (Sender
: TObject
);
2565 s
:= RecentFiles
[(Sender
as TMenuItem
).Tag
];
2566 fn
:= g_ExtractWadName(s
);
2567 if FileExists(fn
) then
2568 OpenMap(fn
, g_ExtractFilePathName(s
))
2570 Application
.MessageBox('', 'File not available anymore', MB_OK
);
2571 // if Application.MessageBox(PChar(MsgMsgDelRecentPromt), PChar(MsgMsgDelRecent), MB_ICONQUESTION or MB_YESNO) = idYes then
2573 // RecentFiles.Delete(n);
2574 // RefreshRecentMenu();
2578 procedure TMainForm
.RefillRecentMenu (menu
: TMenuItem
; start
: Integer; fmt
: AnsiString);
2579 var i
: Integer; MI
: TMenuItem
; s
: AnsiString;
2581 Assert(menu
<> nil);
2583 Assert(start
<= menu
.Count
);
2585 // clear all the recent entries from menu
2587 while i
< menu
.Count
do
2589 MI
:= menu
.Items
[i
];
2590 if @MI
.OnClick
<> @TMainForm
.miRecentFileExecute
then
2599 // fill with a new ones
2600 for i
:= 0 to RecentFiles
.Count
-1 do
2602 MI
:= TMenuItem
.Create(menu
);
2603 s
:= RecentFiles
[i
];
2604 MI
.Caption
:= Format(fmt
, [i
+1, g_ExtractWadNameNoPath(s
), g_ExtractFilePathName(s
)]);
2605 MI
.OnClick
:= miRecentFileExecute
;
2607 menu
.Insert(start
+ i
, MI
); // transfers ownership
2611 procedure TMainForm
.RefreshRecentMenu();
2614 while RecentFiles
.Count
> RecentCount
do
2615 RecentFiles
.Delete(RecentFiles
.Count
- 1);
2617 if miMacRecentSubMenu
.Visible
then
2619 // Reconstruct OSX-like recent list
2620 RefillRecentMenu(miMacRecentSubMenu
, 0, '%1:s - %2:s');
2621 miMacRecentEnd
.Enabled
:= RecentFiles
.Count
<> 0;
2622 miMacRecentEnd
.Visible
:= RecentFiles
.Count
<> 0;
2625 if miWinRecentStart
.Visible
then
2627 // Reconstruct Windows-like recent list
2628 start
:= miMenuFile
.IndexOf(miWinRecent
);
2629 if start
< 0 then start
:= miMenuFile
.Count
else start
+= 1;
2630 RefillRecentMenu(miMenuFile
, start
, '%0:d %1:s:%2:s');
2631 miWinRecent
.Enabled
:= False;
2632 miWinRecent
.Visible
:= RecentFiles
.Count
= 0;
2636 procedure TMainForm
.miMacRecentClearClick(Sender
: TObject
);
2638 RecentFiles
.Clear();
2639 RefreshRecentMenu();
2642 procedure TMainForm
.aEditorOptionsExecute(Sender
: TObject
);
2644 OptionsForm
.ShowModal();
2647 procedure LoadStdFont(cfgres
, texture
: string; var FontID
: DWORD
);
2657 g_ReadResource(GameWad
, 'FONTS', cfgres
, cfgdata
, cfglen
);
2658 if cfgdata
<> nil then
2660 if not g_CreateTextureWAD('FONT_STD', GameWad
+ ':FONTS\' + texture
) then
2661 e_WriteLog('ERROR ERROR ERROR', MSG_WARNING
);
2663 config
:= TConfig
.CreateMem(cfgdata
, cfglen
);
2664 cwdt
:= Min(Max(config
.ReadInt('FontMap', 'CharWidth', 0), 0), 255);
2665 chgt
:= Min(Max(config
.ReadInt('FontMap', 'CharHeight', 0), 0), 255);
2666 spc
:= Min(Max(config
.ReadInt('FontMap', 'Kerning', 0), -128), 127);
2668 if g_GetTexture('FONT_STD', ID
) then
2669 e_TextureFontBuild(ID
, FontID
, cwdt
, chgt
, spc
- 2);
2676 e_WriteLog('Could not load FONT_STD', MSG_WARNING
)
2680 procedure TMainForm
.FormCreate(Sender
: TObject
);
2689 miApple
.Enabled
:= True;
2690 miApple
.Visible
:= True;
2691 miMacRecentSubMenu
.Enabled
:= True;
2692 miMacRecentSubMenu
.Visible
:= True;
2693 miWinRecentStart
.Enabled
:= False;
2694 miWinRecentStart
.Visible
:= False;
2695 miWinRecent
.Enabled
:= False;
2696 miWinRecent
.Visible
:= False;
2697 miLine2
.Enabled
:= False;
2698 miLine2
.Visible
:= False;
2699 miExit
.Enabled
:= False;
2700 miExit
.Visible
:= False;
2701 miOptions
.Enabled
:= False;
2702 miOptions
.Visible
:= False;
2703 miMenuWindow
.Enabled
:= True;
2704 miMenuWindow
.Visible
:= True;
2705 miAbout
.Enabled
:= False;
2706 miAbout
.Visible
:= False;
2708 miApple
.Enabled
:= False;
2709 miApple
.Visible
:= False;
2710 miMacRecentSubMenu
.Enabled
:= False;
2711 miMacRecentSubMenu
.Visible
:= False;
2712 miWinRecentStart
.Enabled
:= True;
2713 miWinRecentStart
.Visible
:= True;
2714 miWinRecent
.Enabled
:= True;
2715 miWinRecent
.Visible
:= True;
2716 miLine2
.Enabled
:= True;
2717 miLine2
.Visible
:= True;
2718 miExit
.Enabled
:= True;
2719 miExit
.Visible
:= True;
2720 miOptions
.Enabled
:= True;
2721 miOptions
.Visible
:= True;
2722 miMenuWindow
.Enabled
:= False;
2723 miMenuWindow
.Visible
:= False;
2724 miAbout
.Enabled
:= True;
2725 miAbout
.Visible
:= True;
2728 miNewMap
.ShortCut
:= ShortCut(VK_N
, [ssModifier
]);
2729 miOpenMap
.ShortCut
:= ShortCut(VK_O
, [ssModifier
]);
2730 miSaveMap
.ShortCut
:= ShortCut(VK_S
, [ssModifier
]);
2732 miSaveMapAs
.ShortCut
:= ShortCut(VK_S
, [ssModifier
, ssShift
]);
2733 miReopenMap
.ShortCut
:= ShortCut(VK_F5
, [ssModifier
]);
2735 miUndo
.ShortCut
:= ShortCut(VK_Z
, [ssModifier
]);
2736 miCopy
.ShortCut
:= ShortCut(VK_C
, [ssModifier
]);
2737 miCut
.ShortCut
:= ShortCut(VK_X
, [ssModifier
]);
2738 miPaste
.ShortCut
:= ShortCut(VK_V
, [ssModifier
]);
2739 miSelectAll
.ShortCut
:= ShortCut(VK_A
, [ssModifier
]);
2740 miToFore
.ShortCut
:= ShortCut(VK_LCL_CLOSE_BRACKET
, [ssModifier
]);
2741 miToBack
.ShortCut
:= ShortCut(VK_LCL_OPEN_BRACKET
, [ssModifier
]);
2743 miMapOptions
.Shortcut
:= ShortCut(VK_P
, [ssModifier
, ssAlt
]);
2744 selectall1
.Shortcut
:= ShortCut(VK_A
, [ssModifier
, ssAlt
]);
2747 e_WriteLog('Doom 2D: Forever Editor version ' + EDITOR_VERSION
, MSG_NOTIFY
);
2748 e_WriteLog('Build date: ' + EDITOR_BUILDDATE
+ ' ' + EDITOR_BUILDTIME
, MSG_NOTIFY
);
2749 e_WriteLog('Build hash: ' + g_GetBuildHash(), MSG_NOTIFY
);
2750 e_WriteLog('Build by: ' + g_GetBuilderName(), MSG_NOTIFY
);
2752 slInvalidTextures
:= TStringList
.Create
;
2754 ShowLayer(LAYER_BACK
, True);
2755 ShowLayer(LAYER_WALLS
, True);
2756 ShowLayer(LAYER_FOREGROUND
, True);
2757 ShowLayer(LAYER_STEPS
, True);
2758 ShowLayer(LAYER_WATER
, True);
2759 ShowLayer(LAYER_ITEMS
, True);
2760 ShowLayer(LAYER_MONSTERS
, True);
2761 ShowLayer(LAYER_AREAS
, True);
2762 ShowLayer(LAYER_TRIGGERS
, True);
2766 FormCaption
:= MainForm
.Caption
;
2770 config
:= TConfig
.CreateFile(CfgFileName
);
2772 if config
.ReadInt('Editor', 'XPos', -1) = -1 then
2773 Position
:= poDesktopCenter
2775 Left
:= config
.ReadInt('Editor', 'XPos', Left
);
2776 Top
:= config
.ReadInt('Editor', 'YPos', Top
);
2777 Width
:= config
.ReadInt('Editor', 'Width', Width
);
2778 Height
:= config
.ReadInt('Editor', 'Height', Height
);
2780 if config
.ReadBool('Editor', 'Maximize', False) then
2781 WindowState
:= wsMaximized
;
2782 ShowMap
:= config
.ReadBool('Editor', 'Minimap', False);
2783 PanelProps
.Width
:= config
.ReadInt('Editor', 'PanelProps', PanelProps
.ClientWidth
);
2784 Splitter1
.Left
:= PanelProps
.Left
;
2785 PanelObjs
.Height
:= config
.ReadInt('Editor', 'PanelObjs', PanelObjs
.ClientHeight
);
2786 Splitter2
.Top
:= PanelObjs
.Top
;
2787 StatusBar
.Top
:= PanelObjs
.BoundsRect
.Bottom
;
2788 DotEnable
:= config
.ReadBool('Editor', 'DotEnable', True);
2789 DotColor
:= config
.ReadInt('Editor', 'DotColor', $FFFFFF);
2790 DotStepOne
:= config
.ReadInt('Editor', 'DotStepOne', 16);
2791 DotStepTwo
:= config
.ReadInt('Editor', 'DotStepTwo', 8);
2792 DotStep
:= config
.ReadInt('Editor', 'DotStep', DotStepOne
);
2793 DrawTexturePanel
:= config
.ReadBool('Editor', 'DrawTexturePanel', True);
2794 DrawPanelSize
:= config
.ReadBool('Editor', 'DrawPanelSize', True);
2795 BackColor
:= config
.ReadInt('Editor', 'BackColor', $7F6040);
2796 PreviewColor
:= config
.ReadInt('Editor', 'PreviewColor', $00FF00);
2797 UseCheckerboard
:= config
.ReadBool('Editor', 'UseCheckerboard', True);
2798 gColorEdge
:= config
.ReadInt('Editor', 'EdgeColor', COLOR_EDGE
);
2799 gAlphaEdge
:= config
.ReadInt('Editor', 'EdgeAlpha', ALPHA_EDGE
);
2800 if gAlphaEdge
= 255 then
2801 gAlphaEdge
:= ALPHA_EDGE
;
2802 drEdge
[0] := GetRValue(gColorEdge
);
2803 drEdge
[1] := GetGValue(gColorEdge
);
2804 drEdge
[2] := GetBValue(gColorEdge
);
2805 if not config
.ReadBool('Editor', 'EdgeShow', True) then
2808 drEdge
[3] := gAlphaEdge
;
2809 gAlphaTriggerLine
:= config
.ReadInt('Editor', 'LineAlpha', ALPHA_LINE
);
2810 if gAlphaTriggerLine
= 255 then
2811 gAlphaTriggerLine
:= ALPHA_LINE
;
2812 gAlphaTriggerArea
:= config
.ReadInt('Editor', 'TriggerAlpha', ALPHA_AREA
);
2813 if gAlphaTriggerArea
= 255 then
2814 gAlphaTriggerArea
:= ALPHA_AREA
;
2815 gAlphaMonsterRect
:= config
.ReadInt('Editor', 'MonsterRectAlpha', 0);
2816 gAlphaAreaRect
:= config
.ReadInt('Editor', 'AreaRectAlpha', 0);
2817 Scale
:= Max(config
.ReadInt('Editor', 'Scale', 1), 1);
2818 DotSize
:= Max(config
.ReadInt('Editor', 'DotSize', 1), 1);
2819 OpenDialog
.InitialDir
:= config
.ReadStr('Editor', 'LastOpenDir', MapsDir
);
2820 SaveDialog
.InitialDir
:= config
.ReadStr('Editor', 'LastSaveDir', MapsDir
);
2822 s
:= config
.ReadStr('Editor', 'Language', '');
2825 Compress
:= config
.ReadBool('Editor', 'Compress', True);
2826 Backup
:= config
.ReadBool('Editor', 'Backup', True);
2828 TestGameMode
:= config
.ReadStr('TestRun', 'GameMode', 'DM');
2829 TestLimTime
:= config
.ReadStr('TestRun', 'LimTime', '0');
2830 TestLimScore
:= config
.ReadStr('TestRun', 'LimScore', '0');
2831 TestOptionsTwoPlayers
:= config
.ReadBool('TestRun', 'TwoPlayers', False);
2832 TestOptionsTeamDamage
:= config
.ReadBool('TestRun', 'TeamDamage', False);
2833 TestOptionsAllowExit
:= config
.ReadBool('TestRun', 'AllowExit', True);
2834 TestOptionsWeaponStay
:= config
.ReadBool('TestRun', 'WeaponStay', False);
2835 TestOptionsMonstersDM
:= config
.ReadBool('TestRun', 'MonstersDM', False);
2836 TestMapOnce
:= config
.ReadBool('TestRun', 'MapOnce', False);
2837 {$IF DEFINED(DARWIN)}
2838 TestD2dExe
:= config
.ReadStr('TestRun', 'ExeDrawin', GameExeFile
);
2839 {$ELSEIF DEFINED(WINDOWS)}
2840 TestD2dExe
:= config
.ReadStr('TestRun', 'ExeWindows', GameExeFile
);
2842 TestD2dExe
:= config
.ReadStr('TestRun', 'ExeUnix', GameExeFile
);
2844 TestD2DArgs
:= config
.ReadStr('TestRun', 'Args', '');
2846 RecentCount
:= config
.ReadInt('Editor', 'RecentCount', 5);
2847 if RecentCount
> 10 then
2849 if RecentCount
< 2 then
2852 RecentFiles
:= TStringList
.Create();
2853 for i
:= 0 to RecentCount
-1 do
2856 s
:= config
.ReadStr('RecentFilesWin', IntToStr(i
), '');
2858 s
:= config
.ReadStr('RecentFilesUnix', IntToStr(i
), '');
2863 RefreshRecentMenu();
2867 tbShowMap
.Down
:= ShowMap
;
2868 tbGridOn
.Down
:= DotEnable
;
2869 pcObjects
.ActivePageIndex
:= 0;
2870 Application
.Title
:= MsgEditorTitle
;
2872 Application
.OnIdle
:= OnIdle
;
2875 procedure PrintBlack(X
, Y
: Integer; Text: string; FontID
: DWORD
);
2877 // NOTE: all the font printing routines assume CP1251
2878 e_TextureFontPrintEx(X
, Y
, Text, FontID
, 0, 0, 0, 1.0);
2881 procedure TMainForm
.Draw();
2886 Width
, Height
: Word;
2889 aX
, aY
, aX2
, aY2
, XX
, ScaleSz
: Integer;
2898 e_Clear(GL_COLOR_BUFFER_BIT
,
2899 GetRValue(BackColor
)/255,
2900 GetGValue(BackColor
)/255,
2901 GetBValue(BackColor
)/255);
2905 ObjCount
:= SelectedObjectCount();
2907 // Обводим выделенные объекты красной рамкой:
2908 if ObjCount
> 0 then
2910 for a
:= 0 to High(SelectedObjects
) do
2911 if SelectedObjects
[a
].Live
then
2913 Rect
:= ObjectGetRect(SelectedObjects
[a
].ObjectType
, SelectedObjects
[a
].ID
);
2917 e_DrawQuad(X
+MapOffset
.X
, Y
+MapOffset
.Y
,
2918 X
+MapOffset
.X
+Width
-1, Y
+MapOffset
.Y
+Height
-1,
2921 // Рисуем точки изменения размеров:
2922 if (ObjCount
= 1) and
2923 (SelectedObjects
[GetFirstSelected
].ObjectType
in [OBJECT_PANEL
, OBJECT_TRIGGER
]) then
2925 e_DrawPoint(5, X
+MapOffset
.X
, Y
+MapOffset
.Y
+(Height
div 2), 255, 255, 255);
2926 e_DrawPoint(5, X
+MapOffset
.X
+Width
-1, Y
+MapOffset
.Y
+(Height
div 2), 255, 255, 255);
2927 e_DrawPoint(5, X
+MapOffset
.X
+(Width
div 2), Y
+MapOffset
.Y
, 255, 255, 255);
2928 e_DrawPoint(5, X
+MapOffset
.X
+(Width
div 2), Y
+MapOffset
.Y
+Height
-1, 255, 255, 255);
2930 e_DrawPoint(3, X
+MapOffset
.X
, Y
+MapOffset
.Y
+(Height
div 2), 255, 0, 0);
2931 e_DrawPoint(3, X
+MapOffset
.X
+Width
-1, Y
+MapOffset
.Y
+(Height
div 2), 255, 0, 0);
2932 e_DrawPoint(3, X
+MapOffset
.X
+(Width
div 2), Y
+MapOffset
.Y
, 255, 0, 0);
2933 e_DrawPoint(3, X
+MapOffset
.X
+(Width
div 2), Y
+MapOffset
.Y
+Height
-1, 255, 0, 0);
2940 if DotEnable
and (PreviewMode
= 0) then
2947 x
:= MapOffset
.X
mod DotStep
;
2948 y
:= MapOffset
.Y
mod DotStep
;
2950 while x
< RenderPanel
.Width
do
2952 while y
< RenderPanel
.Height
do
2954 e_DrawPoint(DotSize
, x
+ a
, y
+ a
,
2955 GetRValue(DotColor
),
2956 GetGValue(DotColor
),
2957 GetBValue(DotColor
));
2961 y
:= MapOffset
.Y
mod DotStep
;
2966 if (lbTextureList
.ItemIndex
<> -1) and (cbPreview
.Checked
) and
2967 (not IsSpecialTextureSel()) and (PreviewMode
= 0) then
2969 if not g_GetTexture(SelectedTexture(), ID
) then
2970 g_GetTexture('NOTEXTURE', ID
);
2971 g_GetTextureSizeByID(ID
, Width
, Height
);
2972 if UseCheckerboard
then
2974 if g_GetTexture('PREVIEW', PID
) then
2975 e_DrawFill(PID
, RenderPanel
.Width
-Width
, RenderPanel
.Height
-Height
, Width
div 16 + 1, Height
div 16 + 1, 0, True, False);
2977 e_DrawFillQuad(RenderPanel
.Width
-Width
-2, RenderPanel
.Height
-Height
-2,
2978 RenderPanel
.Width
-1, RenderPanel
.Height
-1,
2979 GetRValue(PreviewColor
), GetGValue(PreviewColor
), GetBValue(PreviewColor
), 0);
2980 e_Draw(ID
, RenderPanel
.Width
-Width
, RenderPanel
.Height
-Height
, 0, True, False);
2983 // Подсказка при выборе точки Телепорта:
2984 if SelectFlag
= SELECTFLAG_TELEPORT
then
2986 with gTriggers
[SelectedObjects
[GetFirstSelected()].ID
] do
2987 if Data
.d2d_teleport
then
2988 e_DrawLine(2, MousePos
.X
-16, MousePos
.Y
-1,
2989 MousePos
.X
+16, MousePos
.Y
-1,
2992 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+AreaSize
[AREA_DMPOINT
].Width
-1,
2993 MousePos
.Y
+AreaSize
[AREA_DMPOINT
].Height
-1, 255, 255, 255);
2995 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 192, 192, 192, 127);
2996 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 255, 255, 255);
2997 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, utf8to1251(MsgHintTeleport
), gEditorFont
);
3000 // Подсказка при выборе точки появления:
3001 if SelectFlag
= SELECTFLAG_SPAWNPOINT
then
3003 e_DrawLine(2, MousePos
.X
-16, MousePos
.Y
-1,
3004 MousePos
.X
+16, MousePos
.Y
-1,
3006 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 192, 192, 192, 127);
3007 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 255, 255, 255);
3008 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, utf8to1251(MsgHintSpawn
), gEditorFont
);
3011 // Подсказка при выборе панели двери:
3012 if SelectFlag
= SELECTFLAG_DOOR
then
3014 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 192, 192, 192, 127);
3015 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 255, 255, 255);
3016 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, utf8to1251(MsgHintPanelDoor
), gEditorFont
);
3019 // Подсказка при выборе панели с текстурой:
3020 if SelectFlag
= SELECTFLAG_TEXTURE
then
3022 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+196, MousePos
.Y
+18, 192, 192, 192, 127);
3023 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+196, MousePos
.Y
+18, 255, 255, 255);
3024 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, utf8to1251(MsgHintPanelTexture
), gEditorFont
);
3027 // Подсказка при выборе панели индикации выстрела:
3028 if SelectFlag
= SELECTFLAG_SHOTPANEL
then
3030 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+316, MousePos
.Y
+18, 192, 192, 192, 127);
3031 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+316, MousePos
.Y
+18, 255, 255, 255);
3032 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, utf8to1251(MsgHintPanelShot
), gEditorFont
);
3035 // Подсказка при выборе панели лифта:
3036 if SelectFlag
= SELECTFLAG_LIFT
then
3038 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 192, 192, 192, 127);
3039 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+180, MousePos
.Y
+18, 255, 255, 255);
3040 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, utf8to1251(MsgHintPanelLift
), gEditorFont
);
3043 // Подсказка при выборе монстра:
3044 if SelectFlag
= SELECTFLAG_MONSTER
then
3046 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+120, MousePos
.Y
+18, 192, 192, 192, 127);
3047 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+120, MousePos
.Y
+18, 255, 255, 255);
3048 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, utf8to1251(MsgHintMonster
), gEditorFont
);
3051 // Подсказка при выборе области воздействия:
3052 if DrawPressRect
then
3054 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+204, MousePos
.Y
+18, 192, 192, 192, 127);
3055 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+204, MousePos
.Y
+18, 255, 255, 255);
3056 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, utf8to1251(MsgHintExtArea
), gEditorFont
);
3059 // Рисуем текстуры, если чертим панель:
3060 if (MouseAction
= MOUSEACTION_DRAWPANEL
) and (DrawTexturePanel
) and
3061 (lbTextureList
.ItemIndex
<> -1) and (DrawRect
<> nil) and
3062 (lbPanelType
.ItemIndex
in [0..8]) and not IsSpecialTextureSel() then
3064 if not g_GetTexture(SelectedTexture(), ID
) then
3065 g_GetTexture('NOTEXTURE', ID
);
3066 g_GetTextureSizeByID(ID
, Width
, Height
);
3068 if (Abs(Right
-Left
) >= Width
) and (Abs(Bottom
-Top
) >= Height
) then
3069 e_DrawFill(ID
, Min(Left
, Right
), Min(Top
, Bottom
), Abs(Right
-Left
) div Width
,
3070 Abs(Bottom
-Top
) div Height
, 64, True, False);
3073 // Прямоугольник выделения:
3074 if DrawRect
<> nil then
3076 e_DrawQuad(Left
, Top
, Right
-1, Bottom
-1, 255, 255, 255);
3078 // Чертим мышью панель/триггер или меняем мышью их размер:
3079 if (((MouseAction
in [MOUSEACTION_DRAWPANEL
, MOUSEACTION_DRAWTRIGGER
]) and
3080 not(ssCtrl
in GetKeyShiftState())) or (MouseAction
= MOUSEACTION_RESIZE
)) and
3081 (DrawPanelSize
) then
3083 e_DrawFillQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+88, MousePos
.Y
+33, 192, 192, 192, 127);
3084 e_DrawQuad(MousePos
.X
, MousePos
.Y
, MousePos
.X
+88, MousePos
.Y
+33, 255, 255, 255);
3086 if MouseAction
in [MOUSEACTION_DRAWPANEL
, MOUSEACTION_DRAWTRIGGER
] then
3087 begin // Чертим новый
3088 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, Format(utf8to1251(MsgHintWidth
),
3089 [Abs(MousePos
.X
-MouseLDownPos
.X
)]), gEditorFont
);
3090 PrintBlack(MousePos
.X
+2, MousePos
.Y
+16, Format(utf8to1251(MsgHintHeight
),
3091 [Abs(MousePos
.Y
-MouseLDownPos
.Y
)]), gEditorFont
);
3093 else // Растягиваем существующий
3094 if SelectedObjects
[GetFirstSelected
].ObjectType
in [OBJECT_PANEL
, OBJECT_TRIGGER
] then
3096 if SelectedObjects
[GetFirstSelected
].ObjectType
= OBJECT_PANEL
then
3098 Width
:= gPanels
[SelectedObjects
[GetFirstSelected
].ID
].Width
;
3099 Height
:= gPanels
[SelectedObjects
[GetFirstSelected
].ID
].Height
;
3103 Width
:= gTriggers
[SelectedObjects
[GetFirstSelected
].ID
].Width
;
3104 Height
:= gTriggers
[SelectedObjects
[GetFirstSelected
].ID
].Height
;
3107 PrintBlack(MousePos
.X
+2, MousePos
.Y
+2, Format(utf8to1251(MsgHintWidth
), [Width
]),
3109 PrintBlack(MousePos
.X
+2, MousePos
.Y
+16, Format(utf8to1251(MsgHintHeight
), [Height
]),
3114 // Ближайшая к курсору мыши точка на сетке:
3115 e_DrawPoint(3, MousePos
.X
, MousePos
.Y
, 0, 0, 255);
3120 // Сколько пикселов карты в 1 пикселе мини-карты:
3121 ScaleSz
:= 16 div Scale
;
3122 // Размеры мини-карты:
3123 aX
:= max(gMapInfo
.Width
div ScaleSz
, 1);
3124 aY
:= max(gMapInfo
.Height
div ScaleSz
, 1);
3125 // X-координата на RenderPanel нулевой x-координаты карты:
3126 XX
:= RenderPanel
.Width
- aX
- 1;
3128 e_DrawFillQuad(XX
-1, 0, RenderPanel
.Width
-1, aY
+1, 0, 0, 0, 0);
3129 e_DrawQuad(XX
-1, 0, RenderPanel
.Width
-1, aY
+1, 197, 197, 197);
3131 if gPanels
<> nil then
3134 for a
:= 0 to High(gPanels
) do
3136 if PanelType
<> 0 then
3138 // Левый верхний угол:
3139 aX
:= XX
+ (X
div ScaleSz
);
3140 aY
:= 1 + (Y
div ScaleSz
);
3142 aX2
:= max(Width
div ScaleSz
, 1);
3143 aY2
:= max(Height
div ScaleSz
, 1);
3144 // Правый нижний угол:
3145 aX2
:= aX
+ aX2
- 1;
3146 aY2
:= aY
+ aY2
- 1;
3149 PANEL_WALL
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 208, 208, 208, 0);
3150 PANEL_WATER
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 0, 0, 192, 0);
3151 PANEL_ACID1
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 0, 176, 0, 0);
3152 PANEL_ACID2
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 176, 0, 0, 0);
3153 PANEL_STEP
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 128, 128, 128, 0);
3154 PANEL_LIFTUP
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 116, 72, 36, 0);
3155 PANEL_LIFTDOWN
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 116, 124, 96, 0);
3156 PANEL_LIFTLEFT
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 200, 80, 4, 0);
3157 PANEL_LIFTRIGHT
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 252, 140, 56, 0);
3158 PANEL_OPENDOOR
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 100, 220, 92, 0);
3159 PANEL_CLOSEDOOR
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 212, 184, 64, 0);
3160 PANEL_BLOCKMON
: e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 192, 0, 192, 0);
3164 // Рисуем красным выделенные панели:
3165 if SelectedObjects
<> nil then
3166 for b
:= 0 to High(SelectedObjects
) do
3167 with SelectedObjects
[b
] do
3168 if Live
and (ObjectType
= OBJECT_PANEL
) then
3169 with gPanels
[SelectedObjects
[b
].ID
] do
3170 if PanelType
and not(PANEL_BACK
or PANEL_FORE
) <> 0 then
3172 // Левый верхний угол:
3173 aX
:= XX
+ (X
div ScaleSz
);
3174 aY
:= 1 + (Y
div ScaleSz
);
3176 aX2
:= max(Width
div ScaleSz
, 1);
3177 aY2
:= max(Height
div ScaleSz
, 1);
3178 // Правый нижний угол:
3179 aX2
:= aX
+ aX2
- 1;
3180 aY2
:= aY
+ aY2
- 1;
3182 e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 255, 0, 0, 0)
3186 if (gMapInfo
.Width
> RenderPanel
.Width
) or
3187 (gMapInfo
.Height
> RenderPanel
.Height
) then
3189 // Окно, показывающее текущее положение экрана на карте:
3191 x
:= max(min(RenderPanel
.Width
, gMapInfo
.Width
) div ScaleSz
, 1);
3192 y
:= max(min(RenderPanel
.Height
, gMapInfo
.Height
) div ScaleSz
, 1);
3193 // Левый верхний угол:
3194 aX
:= XX
+ ((-MapOffset
.X
) div ScaleSz
);
3195 aY
:= 1 + ((-MapOffset
.Y
) div ScaleSz
);
3196 // Правый нижний угол:
3200 e_DrawFillQuad(aX
, aY
, aX2
, aY2
, 127, 192, 127, 127, B_BLEND
);
3201 e_DrawQuad(aX
, aY
, aX2
, aY2
, 255, 0, 0);
3206 RenderPanel
.SwapBuffers();
3209 procedure TMainForm
.FormResize(Sender
: TObject
);
3211 e_SetViewPort(0, 0, RenderPanel
.Width
, RenderPanel
.Height
);
3213 sbHorizontal
.Min
:= Min(gMapInfo
.Width
- RenderPanel
.Width
, -RenderPanel
.Width
div 2);
3214 sbHorizontal
.Max
:= Max(0, gMapInfo
.Width
- RenderPanel
.Width
div 2);
3215 sbVertical
.Min
:= Min(gMapInfo
.Height
- RenderPanel
.Height
, -RenderPanel
.Height
div 2);
3216 sbVertical
.Max
:= Max(0, gMapInfo
.Height
- RenderPanel
.Height
div 2);
3218 MapOffset
.X
:= -sbHorizontal
.Position
;
3219 MapOffset
.Y
:= -sbVertical
.Position
;
3222 procedure TMainForm
.FormWindowStateChange(Sender
: TObject
);
3228 // deactivate all menus when main window minimized
3229 e
:= self
.WindowState
<> wsMinimized
;
3230 miMenuFile
.Enabled
:= e
;
3231 miMenuEdit
.Enabled
:= e
;
3232 miMenuView
.Enabled
:= e
;
3233 miMenuService
.Enabled
:= e
;
3234 miMenuWindow
.Enabled
:= e
;
3235 miMenuHelp
.Enabled
:= e
;
3236 miMenuHidden
.Enabled
:= e
;
3240 procedure SelectNextObject(X
, Y
: Integer; ObjectType
: Byte; ID
: DWORD
);
3245 j_max
:= 0; // shut up compiler
3249 res
:= (gPanels
<> nil) and
3250 PanelInShownLayer(gPanels
[ID
].PanelType
) and
3251 g_CollidePoint(X
, Y
, gPanels
[ID
].X
, gPanels
[ID
].Y
,
3253 gPanels
[ID
].Height
);
3254 j_max
:= Length(gPanels
) - 1;
3259 res
:= (gItems
<> nil) and
3260 LayerEnabled
[LAYER_ITEMS
] and
3261 g_CollidePoint(X
, Y
, gItems
[ID
].X
, gItems
[ID
].Y
,
3262 ItemSize
[gItems
[ID
].ItemType
][0],
3263 ItemSize
[gItems
[ID
].ItemType
][1]);
3264 j_max
:= Length(gItems
) - 1;
3269 res
:= (gMonsters
<> nil) and
3270 LayerEnabled
[LAYER_MONSTERS
] and
3271 g_CollidePoint(X
, Y
, gMonsters
[ID
].X
, gMonsters
[ID
].Y
,
3272 MonsterSize
[gMonsters
[ID
].MonsterType
].Width
,
3273 MonsterSize
[gMonsters
[ID
].MonsterType
].Height
);
3274 j_max
:= Length(gMonsters
) - 1;
3279 res
:= (gAreas
<> nil) and
3280 LayerEnabled
[LAYER_AREAS
] and
3281 g_CollidePoint(X
, Y
, gAreas
[ID
].X
, gAreas
[ID
].Y
,
3282 AreaSize
[gAreas
[ID
].AreaType
].Width
,
3283 AreaSize
[gAreas
[ID
].AreaType
].Height
);
3284 j_max
:= Length(gAreas
) - 1;
3289 res
:= (gTriggers
<> nil) and
3290 LayerEnabled
[LAYER_TRIGGERS
] and
3291 g_CollidePoint(X
, Y
, gTriggers
[ID
].X
, gTriggers
[ID
].Y
,
3292 gTriggers
[ID
].Width
,
3293 gTriggers
[ID
].Height
);
3294 j_max
:= Length(gTriggers
) - 1;
3304 // Перебор ID: от ID-1 до 0; потом от High до ID+1:
3313 if j
= Integer(ID
) then
3318 res
:= PanelInShownLayer(gPanels
[j
].PanelType
) and
3319 g_CollidePoint(X
, Y
, gPanels
[j
].X
, gPanels
[j
].Y
,
3323 res
:= (gItems
[j
].ItemType
<> ITEM_NONE
) and
3324 g_CollidePoint(X
, Y
, gItems
[j
].X
, gItems
[j
].Y
,
3325 ItemSize
[gItems
[j
].ItemType
][0],
3326 ItemSize
[gItems
[j
].ItemType
][1]);
3328 res
:= (gMonsters
[j
].MonsterType
<> MONSTER_NONE
) and
3329 g_CollidePoint(X
, Y
, gMonsters
[j
].X
, gMonsters
[j
].Y
,
3330 MonsterSize
[gMonsters
[j
].MonsterType
].Width
,
3331 MonsterSize
[gMonsters
[j
].MonsterType
].Height
);
3333 res
:= (gAreas
[j
].AreaType
<> AREA_NONE
) and
3334 g_CollidePoint(X
, Y
, gAreas
[j
].X
, gAreas
[j
].Y
,
3335 AreaSize
[gAreas
[j
].AreaType
].Width
,
3336 AreaSize
[gAreas
[j
].AreaType
].Height
);
3338 res
:= (gTriggers
[j
].TriggerType
<> TRIGGER_NONE
) and
3339 g_CollidePoint(X
, Y
, gTriggers
[j
].X
, gTriggers
[j
].Y
,
3341 gTriggers
[j
].Height
);
3348 SetLength(SelectedObjects
, 1);
3350 SelectedObjects
[0].ObjectType
:= ObjectType
;
3351 SelectedObjects
[0].ID
:= j
;
3352 SelectedObjects
[0].Live
:= True;
3360 procedure TMainForm
.RenderPanelMouseDown(Sender
: TObject
;
3361 Button
: TMouseButton
; Shift
: TShiftState
; X
, Y
: Integer);
3365 c1
, c2
, c3
, c4
: Boolean;
3371 MainForm
.ActiveControl
:= RenderPanel
;
3372 RenderPanel
.SetFocus();
3374 RenderPanelMouseMove(RenderPanel
, Shift
, X
, Y
);
3376 if Button
= mbLeft
then // Left Mouse Button
3378 // Двигаем карту с помощью мыши и мини-карты:
3380 g_CollidePoint(X
, Y
,
3381 RenderPanel
.Width
-max(gMapInfo
.Width
div (16 div Scale
), 1)-1,
3383 max(gMapInfo
.Width
div (16 div Scale
), 1),
3384 max(gMapInfo
.Height
div (16 div Scale
), 1) ) then
3387 MouseAction
:= MOUSEACTION_MOVEMAP
;
3389 else // Ставим предмет/монстра/область:
3390 if (pcObjects
.ActivePageIndex
in [1, 2, 3]) and
3391 (not (ssShift
in Shift
)) then
3393 case pcObjects
.ActivePageIndex
of
3395 if lbItemList
.ItemIndex
= -1 then
3396 ErrorMessageBox(MsgMsgChooseItem
)
3399 item
.ItemType
:= lbItemList
.ItemIndex
+ ITEM_MEDKIT_SMALL
;
3400 if item
.ItemType
>= ITEM_WEAPON_KASTET
then
3401 item
.ItemType
:= item
.ItemType
+ 2;
3402 item
.X
:= MousePos
.X
-MapOffset
.X
;
3403 item
.Y
:= MousePos
.Y
-MapOffset
.Y
;
3405 if not (ssCtrl
in Shift
) then
3407 item
.X
:= item
.X
- (ItemSize
[item
.ItemType
][0] div 2);
3408 item
.Y
:= item
.Y
- ItemSize
[item
.ItemType
][1];
3411 item
.OnlyDM
:= cbOnlyDM
.Checked
;
3412 item
.Fall
:= cbFall
.Checked
;
3413 Undo_Add(OBJECT_ITEM
, AddItem(item
));
3416 if lbMonsterList
.ItemIndex
= -1 then
3417 ErrorMessageBox(MsgMsgChooseMonster
)
3420 monster
.MonsterType
:= lbMonsterList
.ItemIndex
+ MONSTER_DEMON
;
3421 monster
.X
:= MousePos
.X
-MapOffset
.X
;
3422 monster
.Y
:= MousePos
.Y
-MapOffset
.Y
;
3424 if not (ssCtrl
in Shift
) then
3426 monster
.X
:= monster
.X
- (MonsterSize
[monster
.MonsterType
].Width
div 2);
3427 monster
.Y
:= monster
.Y
- MonsterSize
[monster
.MonsterType
].Height
;
3430 if rbMonsterLeft
.Checked
then
3431 monster
.Direction
:= D_LEFT
3433 monster
.Direction
:= D_RIGHT
;
3434 Undo_Add(OBJECT_MONSTER
, AddMonster(monster
));
3437 if lbAreasList
.ItemIndex
= -1 then
3438 ErrorMessageBox(MsgMsgChooseArea
)
3440 if (lbAreasList
.ItemIndex
+ 1) <> AREA_DOMFLAG
then
3442 area
.AreaType
:= lbAreasList
.ItemIndex
+ AREA_PLAYERPOINT1
;
3443 area
.X
:= MousePos
.X
-MapOffset
.X
;
3444 area
.Y
:= MousePos
.Y
-MapOffset
.Y
;
3446 if not (ssCtrl
in Shift
) then
3448 area
.X
:= area
.X
- (AreaSize
[area
.AreaType
].Width
div 2);
3449 area
.Y
:= area
.Y
- AreaSize
[area
.AreaType
].Height
;
3452 if rbAreaLeft
.Checked
then
3453 area
.Direction
:= D_LEFT
3455 area
.Direction
:= D_RIGHT
;
3456 Undo_Add(OBJECT_AREA
, AddArea(area
));
3462 i
:= GetFirstSelected();
3464 // Выбираем объект под текущим:
3465 if (SelectedObjects
<> nil) and
3466 (ssShift
in Shift
) and (i
>= 0) and
3467 (SelectedObjects
[i
].Live
) then
3469 if SelectedObjectCount() = 1 then
3470 SelectNextObject(X
-MapOffset
.X
, Y
-MapOffset
.Y
,
3471 SelectedObjects
[i
].ObjectType
,
3472 SelectedObjects
[i
].ID
);
3476 // Рисуем область триггера "Расширитель":
3477 if DrawPressRect
and (i
>= 0) and
3478 (SelectedObjects
[i
].ObjectType
= OBJECT_TRIGGER
) and
3479 (gTriggers
[SelectedObjects
[i
].ID
].TriggerType
in
3480 [TRIGGER_PRESS
, TRIGGER_ON
, TRIGGER_OFF
, TRIGGER_ONOFF
]) then
3481 MouseAction
:= MOUSEACTION_DRAWPRESS
3482 else // Рисуем панель:
3483 if pcObjects
.ActivePageIndex
= 0 then
3485 if (lbPanelType
.ItemIndex
>= 0) then
3486 MouseAction
:= MOUSEACTION_DRAWPANEL
3488 else // Рисуем триггер:
3489 if (lbTriggersList
.ItemIndex
>= 0) then
3491 MouseAction
:= MOUSEACTION_DRAWTRIGGER
;
3495 end; // if Button = mbLeft
3497 if Button
= mbRight
then // Right Mouse Button
3499 // Клик по мини-карте:
3501 g_CollidePoint(X
, Y
,
3502 RenderPanel
.Width
-max(gMapInfo
.Width
div (16 div Scale
), 1)-1,
3504 max(gMapInfo
.Width
div (16 div Scale
), 1),
3505 max(gMapInfo
.Height
div (16 div Scale
), 1) ) then
3507 MouseAction
:= MOUSEACTION_NOACTION
;
3509 else // Нужно что-то выбрать мышью:
3510 if SelectFlag
<> SELECTFLAG_NONE
then
3513 SELECTFLAG_TELEPORT
:
3514 // Точку назначения телепортации:
3515 with gTriggers
[SelectedObjects
[
3516 GetFirstSelected() ].ID
].Data
.TargetPoint
do
3518 X
:= MousePos
.X
-MapOffset
.X
;
3519 Y
:= MousePos
.Y
-MapOffset
.Y
;
3522 SELECTFLAG_SPAWNPOINT
:
3523 // Точку создания монстра:
3524 with gTriggers
[SelectedObjects
[GetFirstSelected()].ID
] do
3525 if TriggerType
= TRIGGER_SPAWNMONSTER
then
3527 Data
.MonPos
.X
:= MousePos
.X
-MapOffset
.X
;
3528 Data
.MonPos
.Y
:= MousePos
.Y
-MapOffset
.Y
;
3530 else if TriggerType
= TRIGGER_SPAWNITEM
then
3531 begin // Точка создания предмета:
3532 Data
.ItemPos
.X
:= MousePos
.X
-MapOffset
.X
;
3533 Data
.ItemPos
.Y
:= MousePos
.Y
-MapOffset
.Y
;
3535 else if TriggerType
= TRIGGER_SHOT
then
3536 begin // Точка создания выстрела:
3537 Data
.ShotPos
.X
:= MousePos
.X
-MapOffset
.X
;
3538 Data
.ShotPos
.Y
:= MousePos
.Y
-MapOffset
.Y
;
3544 IDArray
:= ObjectInRect(X
-MapOffset
.X
,
3546 2, 2, OBJECT_PANEL
, True);
3547 if IDArray
<> nil then
3549 for i
:= 0 to High(IDArray
) do
3550 if (gPanels
[IDArray
[i
]].PanelType
= PANEL_OPENDOOR
) or
3551 (gPanels
[IDArray
[i
]].PanelType
= PANEL_CLOSEDOOR
) then
3553 gTriggers
[SelectedObjects
[
3554 GetFirstSelected() ].ID
].Data
.PanelID
:= IDArray
[i
];
3559 gTriggers
[SelectedObjects
[
3560 GetFirstSelected() ].ID
].Data
.PanelID
:= -1;
3564 // Панель с текстурой:
3566 IDArray
:= ObjectInRect(X
-MapOffset
.X
,
3568 2, 2, OBJECT_PANEL
, True);
3569 if IDArray
<> nil then
3571 for i
:= 0 to High(IDArray
) do
3572 if ((gPanels
[IDArray
[i
]].PanelType
in
3573 [PANEL_WALL
, PANEL_BACK
, PANEL_FORE
,
3574 PANEL_WATER
, PANEL_ACID1
, PANEL_ACID2
,
3576 (gPanels
[IDArray
[i
]].PanelType
= PANEL_OPENDOOR
) or
3577 (gPanels
[IDArray
[i
]].PanelType
= PANEL_CLOSEDOOR
)) and
3578 (gPanels
[IDArray
[i
]].TextureName
<> '') then
3580 gTriggers
[SelectedObjects
[
3581 GetFirstSelected() ].ID
].TexturePanel
:= IDArray
[i
];
3586 gTriggers
[SelectedObjects
[
3587 GetFirstSelected() ].ID
].TexturePanel
:= -1;
3593 IDArray
:= ObjectInRect(X
-MapOffset
.X
,
3595 2, 2, OBJECT_PANEL
, True);
3596 if IDArray
<> nil then
3598 for i
:= 0 to High(IDArray
) do
3599 if (gPanels
[IDArray
[i
]].PanelType
= PANEL_LIFTUP
) or
3600 (gPanels
[IDArray
[i
]].PanelType
= PANEL_LIFTDOWN
) or
3601 (gPanels
[IDArray
[i
]].PanelType
= PANEL_LIFTLEFT
) or
3602 (gPanels
[IDArray
[i
]].PanelType
= PANEL_LIFTRIGHT
) then
3604 gTriggers
[SelectedObjects
[
3605 GetFirstSelected() ].ID
].Data
.PanelID
:= IDArray
[i
];
3610 gTriggers
[SelectedObjects
[
3611 GetFirstSelected() ].ID
].Data
.PanelID
:= -1;
3617 IDArray
:= ObjectInRect(X
-MapOffset
.X
,
3619 2, 2, OBJECT_MONSTER
, False);
3620 if IDArray
<> nil then
3621 gTriggers
[SelectedObjects
[
3622 GetFirstSelected() ].ID
].Data
.MonsterID
:= IDArray
[0]+1
3624 gTriggers
[SelectedObjects
[
3625 GetFirstSelected() ].ID
].Data
.MonsterID
:= 0;
3628 SELECTFLAG_SHOTPANEL
:
3629 // Панель индикации выстрела:
3631 if gTriggers
[SelectedObjects
[
3632 GetFirstSelected() ].ID
].TriggerType
= TRIGGER_SHOT
then
3634 IDArray
:= ObjectInRect(X
-MapOffset
.X
,
3636 2, 2, OBJECT_PANEL
, True);
3637 if IDArray
<> nil then
3639 for i
:= 0 to High(IDArray
) do
3640 if ((gPanels
[IDArray
[i
]].PanelType
in
3641 [PANEL_WALL
, PANEL_BACK
, PANEL_FORE
,
3642 PANEL_WATER
, PANEL_ACID1
, PANEL_ACID2
,
3644 (gPanels
[IDArray
[i
]].PanelType
= PANEL_OPENDOOR
) or
3645 (gPanels
[IDArray
[i
]].PanelType
= PANEL_CLOSEDOOR
)) and
3646 (gPanels
[IDArray
[i
]].TextureName
<> '') then
3648 gTriggers
[SelectedObjects
[
3649 GetFirstSelected() ].ID
].Data
.ShotPanelID
:= IDArray
[i
];
3654 gTriggers
[SelectedObjects
[
3655 GetFirstSelected() ].ID
].Data
.ShotPanelID
:= -1;
3660 SelectFlag
:= SELECTFLAG_SELECTED
;
3662 else // if SelectFlag <> SELECTFLAG_NONE...
3664 // Что уже выбрано и не нажат Ctrl:
3665 if (SelectedObjects
<> nil) and
3666 (not (ssCtrl
in Shift
)) then
3667 for i
:= 0 to High(SelectedObjects
) do
3668 with SelectedObjects
[i
] do
3671 if (ObjectType
in [OBJECT_PANEL
, OBJECT_TRIGGER
]) and
3672 (SelectedObjectCount() = 1) then
3674 Rect
:= ObjectGetRect(ObjectType
, ID
);
3676 c1
:= g_Collide(X
-MapOffset
.X
-1, Y
-MapOffset
.Y
-1, 2, 2,
3677 Rect
.X
-2, Rect
.Y
+(Rect
.Height
div 2)-2, 4, 4);
3678 c2
:= g_Collide(X
-MapOffset
.X
-1, Y
-MapOffset
.Y
-1, 2, 2,
3679 Rect
.X
+Rect
.Width
-3, Rect
.Y
+(Rect
.Height
div 2)-2, 4, 4);
3680 c3
:= g_Collide(X
-MapOffset
.X
-1, Y
-MapOffset
.Y
-1, 2, 2,
3681 Rect
.X
+(Rect
.Width
div 2)-2, Rect
.Y
-2, 4, 4);
3682 c4
:= g_Collide(X
-MapOffset
.X
-1, Y
-MapOffset
.Y
-1, 2, 2,
3683 Rect
.X
+(Rect
.Width
div 2)-2, Rect
.Y
+Rect
.Height
-3, 4, 4);
3685 // Меняем размер панели или триггера:
3686 if c1
or c2
or c3
or c4
then
3688 MouseAction
:= MOUSEACTION_RESIZE
;
3689 LastMovePoint
:= MousePos
;
3693 ResizeType
:= RESIZETYPE_HORIZONTAL
;
3695 ResizeDirection
:= RESIZEDIR_LEFT
3697 ResizeDirection
:= RESIZEDIR_RIGHT
;
3698 RenderPanel
.Cursor
:= crSizeWE
;
3702 ResizeType
:= RESIZETYPE_VERTICAL
;
3704 ResizeDirection
:= RESIZEDIR_UP
3706 ResizeDirection
:= RESIZEDIR_DOWN
;
3707 RenderPanel
.Cursor
:= crSizeNS
;
3714 // Перемещаем панель или триггер:
3715 if ObjectCollide(ObjectType
, ID
,
3717 Y
-MapOffset
.Y
-1, 2, 2) then
3719 MouseAction
:= MOUSEACTION_MOVEOBJ
;
3720 LastMovePoint
:= MousePos
;
3726 end; // if Button = mbRight
3728 if Button
= mbMiddle
then // Middle Mouse Button
3730 SetCapture(RenderPanel
.Handle
);
3731 RenderPanel
.Cursor
:= crSize
;
3734 MouseMDown
:= Button
= mbMiddle
;
3736 MouseMDownPos
:= Mouse
.CursorPos
;
3738 MouseRDown
:= Button
= mbRight
;
3740 MouseRDownPos
:= MousePos
;
3742 MouseLDown
:= Button
= mbLeft
;
3744 MouseLDownPos
:= MousePos
;
3747 procedure TMainForm
.RenderPanelMouseUp(Sender
: TObject
;
3748 Button
: TMouseButton
; Shift
: TShiftState
; X
, Y
: Integer);
3753 rSelectRect
: Boolean;
3754 wWidth
, wHeight
: Word;
3757 procedure SelectObjects(ObjectType
: Byte);
3762 IDArray
:= ObjectInRect(rRect
.X
, rRect
.Y
,
3763 rRect
.Width
, rRect
.Height
,
3764 ObjectType
, rSelectRect
);
3766 if IDArray
<> nil then
3767 for i
:= 0 to High(IDArray
) do
3768 SelectObject(ObjectType
, IDArray
[i
], (ssCtrl
in Shift
) or rSelectRect
);
3771 if Button
= mbLeft
then
3772 MouseLDown
:= False;
3773 if Button
= mbRight
then
3774 MouseRDown
:= False;
3775 if Button
= mbMiddle
then
3776 MouseMDown
:= False;
3779 ResizeType
:= RESIZETYPE_NONE
;
3782 if Button
= mbLeft
then // Left Mouse Button
3784 if MouseAction
<> MOUSEACTION_NONE
then
3785 begin // Было действие мышью
3786 // Мышь сдвинулась во время удержания клавиши,
3787 // либо активирован режим быстрого рисования:
3788 if ((MousePos
.X
<> MouseLDownPos
.X
) and
3789 (MousePos
.Y
<> MouseLDownPos
.Y
)) or
3790 ((MouseAction
in [MOUSEACTION_DRAWPANEL
, MOUSEACTION_DRAWTRIGGER
]) and
3791 (ssCtrl
in Shift
)) then
3794 MOUSEACTION_DRAWPANEL
:
3796 // Фон или передний план без текстуры - ошибка:
3797 if (lbPanelType
.ItemIndex
in [1, 2]) and
3798 (lbTextureList
.ItemIndex
= -1) then
3799 ErrorMessageBox(MsgMsgChooseTexture
)
3800 else // Назначаем параметры панели:
3802 case lbPanelType
.ItemIndex
of
3803 0: Panel
.PanelType
:= PANEL_WALL
;
3804 1: Panel
.PanelType
:= PANEL_BACK
;
3805 2: Panel
.PanelType
:= PANEL_FORE
;
3806 3: Panel
.PanelType
:= PANEL_OPENDOOR
;
3807 4: Panel
.PanelType
:= PANEL_CLOSEDOOR
;
3808 5: Panel
.PanelType
:= PANEL_STEP
;
3809 6: Panel
.PanelType
:= PANEL_WATER
;
3810 7: Panel
.PanelType
:= PANEL_ACID1
;
3811 8: Panel
.PanelType
:= PANEL_ACID2
;
3812 9: Panel
.PanelType
:= PANEL_LIFTUP
;
3813 10: Panel
.PanelType
:= PANEL_LIFTDOWN
;
3814 11: Panel
.PanelType
:= PANEL_LIFTLEFT
;
3815 12: Panel
.PanelType
:= PANEL_LIFTRIGHT
;
3816 13: Panel
.PanelType
:= PANEL_BLOCKMON
;
3819 Panel
.X
:= Min(MousePos
.X
-MapOffset
.X
, MouseLDownPos
.X
-MapOffset
.X
);
3820 Panel
.Y
:= Min(MousePos
.Y
-MapOffset
.Y
, MouseLDownPos
.Y
-MapOffset
.Y
);
3821 if ssCtrl
in Shift
then
3825 if (lbTextureList
.ItemIndex
<> -1) and
3826 (not IsSpecialTextureSel()) then
3828 if not g_GetTexture(SelectedTexture(), TextureID
) then
3829 g_GetTexture('NOTEXTURE', TextureID
);
3830 g_GetTextureSizeByID(TextureID
, wWidth
, wHeight
);
3832 Panel
.Width
:= wWidth
;
3833 Panel
.Height
:= wHeight
;
3837 Panel
.Width
:= Abs(MousePos
.X
-MouseLDownPos
.X
);
3838 Panel
.Height
:= Abs(MousePos
.Y
-MouseLDownPos
.Y
);
3841 // Лифты, блокМон или отсутствие текстуры - пустая текстура:
3842 if (lbPanelType
.ItemIndex
in [9, 10, 11, 12, 13]) or
3843 (lbTextureList
.ItemIndex
= -1) then
3845 Panel
.TextureHeight
:= 1;
3846 Panel
.TextureWidth
:= 1;
3847 Panel
.TextureName
:= '';
3848 Panel
.TextureID
:= TEXTURE_SPECIAL_NONE
;
3850 else // Есть текстура:
3852 Panel
.TextureName
:= SelectedTexture();
3854 // Обычная текстура:
3855 if not IsSpecialTextureSel() then
3857 g_GetTextureSizeByName(Panel
.TextureName
,
3858 Panel
.TextureWidth
, Panel
.TextureHeight
);
3859 g_GetTexture(Panel
.TextureName
, Panel
.TextureID
);
3861 else // Спец.текстура:
3863 Panel
.TextureHeight
:= 1;
3864 Panel
.TextureWidth
:= 1;
3865 Panel
.TextureID
:= SpecialTextureID(SelectedTexture());
3870 Panel
.Blending
:= False;
3872 Undo_Add(OBJECT_PANEL
, AddPanel(Panel
));
3876 // Рисовали триггер:
3877 MOUSEACTION_DRAWTRIGGER
:
3879 trigger
.X
:= Min(MousePos
.X
-MapOffset
.X
, MouseLDownPos
.X
-MapOffset
.X
);
3880 trigger
.Y
:= Min(MousePos
.Y
-MapOffset
.Y
, MouseLDownPos
.Y
-MapOffset
.Y
);
3881 if ssCtrl
in Shift
then
3885 trigger
.Width
:= wWidth
;
3886 trigger
.Height
:= wHeight
;
3890 trigger
.Width
:= Abs(MousePos
.X
-MouseLDownPos
.X
);
3891 trigger
.Height
:= Abs(MousePos
.Y
-MouseLDownPos
.Y
);
3894 trigger
.Enabled
:= True;
3895 trigger
.TriggerType
:= lbTriggersList
.ItemIndex
+1;
3896 trigger
.TexturePanel
:= -1;
3899 trigger
.ActivateType
:= 0;
3901 if clbActivationType
.Checked
[0] then
3902 trigger
.ActivateType
:= Trigger
.ActivateType
or ACTIVATE_PLAYERCOLLIDE
;
3903 if clbActivationType
.Checked
[1] then
3904 trigger
.ActivateType
:= Trigger
.ActivateType
or ACTIVATE_MONSTERCOLLIDE
;
3905 if clbActivationType
.Checked
[2] then
3906 trigger
.ActivateType
:= Trigger
.ActivateType
or ACTIVATE_PLAYERPRESS
;
3907 if clbActivationType
.Checked
[3] then
3908 trigger
.ActivateType
:= Trigger
.ActivateType
or ACTIVATE_MONSTERPRESS
;
3909 if clbActivationType
.Checked
[4] then
3910 trigger
.ActivateType
:= Trigger
.ActivateType
or ACTIVATE_SHOT
;
3911 if clbActivationType
.Checked
[5] then
3912 trigger
.ActivateType
:= Trigger
.ActivateType
or ACTIVATE_NOMONSTER
;
3914 // Необходимые для активации ключи:
3917 if clbKeys
.Checked
[0] then
3918 trigger
.Key
:= Trigger
.Key
or KEY_RED
;
3919 if clbKeys
.Checked
[1] then
3920 trigger
.Key
:= Trigger
.Key
or KEY_GREEN
;
3921 if clbKeys
.Checked
[2] then
3922 trigger
.Key
:= Trigger
.Key
or KEY_BLUE
;
3923 if clbKeys
.Checked
[3] then
3924 trigger
.Key
:= Trigger
.Key
or KEY_REDTEAM
;
3925 if clbKeys
.Checked
[4] then
3926 trigger
.Key
:= Trigger
.Key
or KEY_BLUETEAM
;
3928 // Параметры триггера:
3929 FillByte(trigger
.Data
.Default
[0], 128, 0);
3931 case trigger
.TriggerType
of
3932 // Переключаемая панель:
3933 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
,
3934 TRIGGER_DOOR5
, TRIGGER_CLOSETRAP
, TRIGGER_TRAP
,
3935 TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
:
3937 Trigger
.Data
.PanelID
:= -1;
3943 trigger
.Data
.TargetPoint
.X
:= trigger
.X
-64;
3944 trigger
.Data
.TargetPoint
.Y
:= trigger
.Y
-64;
3945 trigger
.Data
.d2d_teleport
:= True;
3946 trigger
.Data
.TlpDir
:= 0;
3949 // Изменение других триггеров:
3950 TRIGGER_PRESS
, TRIGGER_ON
, TRIGGER_OFF
,
3953 trigger
.Data
.Count
:= 1;
3959 trigger
.Data
.Volume
:= 255;
3960 trigger
.Data
.Pan
:= 127;
3961 trigger
.Data
.PlayCount
:= 1;
3962 trigger
.Data
.Local
:= True;
3963 trigger
.Data
.SoundSwitch
:= False;
3969 trigger
.Data
.MusicAction
:= 1;
3972 // Создание монстра:
3973 TRIGGER_SPAWNMONSTER
:
3975 trigger
.Data
.MonType
:= MONSTER_ZOMBY
;
3976 trigger
.Data
.MonPos
.X
:= trigger
.X
-64;
3977 trigger
.Data
.MonPos
.Y
:= trigger
.Y
-64;
3978 trigger
.Data
.MonHealth
:= 0;
3979 trigger
.Data
.MonActive
:= False;
3980 trigger
.Data
.MonCount
:= 1;
3983 // Создание предмета:
3986 trigger
.Data
.ItemType
:= ITEM_AMMO_BULLETS
;
3987 trigger
.Data
.ItemPos
.X
:= trigger
.X
-64;
3988 trigger
.Data
.ItemPos
.Y
:= trigger
.Y
-64;
3989 trigger
.Data
.ItemOnlyDM
:= False;
3990 trigger
.Data
.ItemFalls
:= False;
3991 trigger
.Data
.ItemCount
:= 1;
3992 trigger
.Data
.ItemMax
:= 0;
3993 trigger
.Data
.ItemDelay
:= 0;
3999 trigger
.Data
.PushAngle
:= 90;
4000 trigger
.Data
.PushForce
:= 10;
4001 trigger
.Data
.ResetVel
:= True;
4006 trigger
.Data
.ScoreCount
:= 1;
4007 trigger
.Data
.ScoreCon
:= True;
4008 trigger
.Data
.ScoreMsg
:= True;
4013 trigger
.Data
.MessageKind
:= 0;
4014 trigger
.Data
.MessageSendTo
:= 0;
4015 trigger
.Data
.MessageText
:= '';
4016 trigger
.Data
.MessageTime
:= 144;
4021 trigger
.Data
.DamageValue
:= 5;
4022 trigger
.Data
.DamageInterval
:= 12;
4027 trigger
.Data
.HealValue
:= 5;
4028 trigger
.Data
.HealInterval
:= 36;
4033 trigger
.Data
.ShotType
:= TRIGGER_SHOT_BULLET
;
4034 trigger
.Data
.ShotSound
:= True;
4035 trigger
.Data
.ShotPanelID
:= -1;
4036 trigger
.Data
.ShotTarget
:= 0;
4037 trigger
.Data
.ShotIntSight
:= 0;
4038 trigger
.Data
.ShotAim
:= TRIGGER_SHOT_AIM_DEFAULT
;
4039 trigger
.Data
.ShotPos
.X
:= trigger
.X
-64;
4040 trigger
.Data
.ShotPos
.Y
:= trigger
.Y
-64;
4041 trigger
.Data
.ShotAngle
:= 0;
4042 trigger
.Data
.ShotWait
:= 18;
4043 trigger
.Data
.ShotAccuracy
:= 0;
4044 trigger
.Data
.ShotAmmo
:= 0;
4045 trigger
.Data
.ShotIntReload
:= 0;
4050 trigger
.Data
.FXCount
:= 1;
4051 trigger
.Data
.FXType
:= TRIGGER_EFFECT_PARTICLE
;
4052 trigger
.Data
.FXSubType
:= TRIGGER_EFFECT_SLIQUID
;
4053 trigger
.Data
.FXColorR
:= 0;
4054 trigger
.Data
.FXColorG
:= 0;
4055 trigger
.Data
.FXColorB
:= 255;
4056 trigger
.Data
.FXPos
:= TRIGGER_EFFECT_POS_CENTER
;
4057 trigger
.Data
.FXWait
:= 1;
4058 trigger
.Data
.FXVelX
:= 0;
4059 trigger
.Data
.FXVelY
:= -20;
4060 trigger
.Data
.FXSpreadL
:= 5;
4061 trigger
.Data
.FXSpreadR
:= 5;
4062 trigger
.Data
.FXSpreadU
:= 4;
4063 trigger
.Data
.FXSpreadD
:= 0;
4067 Undo_Add(OBJECT_TRIGGER
, AddTrigger(trigger
));
4070 // Рисовали область триггера "Расширитель":
4071 MOUSEACTION_DRAWPRESS
:
4072 with gTriggers
[SelectedObjects
[GetFirstSelected
].ID
] do
4074 Data
.tX
:= Min(MousePos
.X
-MapOffset
.X
, MouseLDownPos
.X
-MapOffset
.X
);
4075 Data
.tY
:= Min(MousePos
.Y
-MapOffset
.Y
, MouseLDownPos
.Y
-MapOffset
.Y
);
4076 Data
.tWidth
:= Abs(MousePos
.X
-MouseLDownPos
.X
);
4077 Data
.tHeight
:= Abs(MousePos
.Y
-MouseLDownPos
.Y
);
4079 DrawPressRect
:= False;
4083 MouseAction
:= MOUSEACTION_NONE
;
4085 end // if Button = mbLeft...
4086 else if Button
= mbRight
then // Right Mouse Button:
4088 if MouseAction
= MOUSEACTION_NOACTION
then
4090 MouseAction
:= MOUSEACTION_NONE
;
4094 // Объект передвинут или изменен в размере:
4095 if MouseAction
in [MOUSEACTION_MOVEOBJ
, MOUSEACTION_RESIZE
] then
4097 RenderPanel
.Cursor
:= crDefault
;
4098 MouseAction
:= MOUSEACTION_NONE
;
4103 // Еще не все выбрали:
4104 if SelectFlag
<> SELECTFLAG_NONE
then
4106 if SelectFlag
= SELECTFLAG_SELECTED
then
4107 SelectFlag
:= SELECTFLAG_NONE
;
4112 // Мышь сдвинулась во время удержания клавиши:
4113 if (MousePos
.X
<> MouseRDownPos
.X
) and
4114 (MousePos
.Y
<> MouseRDownPos
.Y
) then
4116 rSelectRect
:= True;
4118 rRect
.X
:= Min(MousePos
.X
, MouseRDownPos
.X
)-MapOffset
.X
;
4119 rRect
.Y
:= Min(MousePos
.Y
, MouseRDownPos
.Y
)-MapOffset
.Y
;
4120 rRect
.Width
:= Abs(MousePos
.X
-MouseRDownPos
.X
);
4121 rRect
.Height
:= Abs(MousePos
.Y
-MouseRDownPos
.Y
);
4123 else // Мышь не сдвинулась - нет прямоугольника:
4125 rSelectRect
:= False;
4127 rRect
.X
:= X
-MapOffset
.X
-1;
4128 rRect
.Y
:= Y
-MapOffset
.Y
-1;
4133 // Если зажат Ctrl - выделять еще, иначе только один выделенный объект:
4134 if not (ssCtrl
in Shift
) then
4135 RemoveSelectFromObjects();
4137 // Выделяем всё в выбранном прямоугольнике:
4138 if (ssCtrl
in Shift
) and (ssAlt
in Shift
) then
4140 SelectObjects(OBJECT_PANEL
);
4141 SelectObjects(OBJECT_ITEM
);
4142 SelectObjects(OBJECT_MONSTER
);
4143 SelectObjects(OBJECT_AREA
);
4144 SelectObjects(OBJECT_TRIGGER
);
4147 SelectObjects(pcObjects
.ActivePageIndex
+1);
4152 else // Middle Mouse Button
4154 RenderPanel
.Cursor
:= crDefault
;
4159 procedure TMainForm
.RenderPanelPaint(Sender
: TObject
);
4164 function TMainForm
.RenderMousePos(): Types
.TPoint
;
4166 Result
:= RenderPanel
.ScreenToClient(Mouse
.CursorPos
);
4169 procedure TMainForm
.RecountSelectedObjects();
4171 if SelectedObjectCount() = 0 then
4172 StatusBar
.Panels
[0].Text := ''
4174 StatusBar
.Panels
[0].Text := Format(MsgCapStatSelected
, [SelectedObjectCount()]);
4177 procedure TMainForm
.RenderPanelMouseMove(Sender
: TObject
;
4178 Shift
: TShiftState
; X
, Y
: Integer);
4181 dWidth
, dHeight
: Integer;
4184 wWidth
, wHeight
: Word;
4186 _id
:= GetFirstSelected();
4189 // Рисуем панель с текстурой, сетка - размеры текстуры:
4190 if (MouseAction
= MOUSEACTION_DRAWPANEL
) and
4191 (lbPanelType
.ItemIndex
in [0..8]) and
4192 (lbTextureList
.ItemIndex
<> -1) and
4193 (not IsSpecialTextureSel()) then
4195 sX
:= StrToIntDef(lTextureWidth
.Caption
, DotStep
);
4196 sY
:= StrToIntDef(lTextureHeight
.Caption
, DotStep
);
4199 // Меняем размер панели с текстурой, сетка - размеры текстуры:
4200 if (MouseAction
= MOUSEACTION_RESIZE
) and
4201 ( (SelectedObjects
[_id
].ObjectType
= OBJECT_PANEL
) and
4202 IsTexturedPanel(gPanels
[SelectedObjects
[_id
].ID
].PanelType
) and
4203 (gPanels
[SelectedObjects
[_id
].ID
].TextureName
<> '') and
4204 (not IsSpecialTexture(gPanels
[SelectedObjects
[_id
].ID
].TextureName
)) ) then
4206 sX
:= gPanels
[SelectedObjects
[_id
].ID
].TextureWidth
;
4207 sY
:= gPanels
[SelectedObjects
[_id
].ID
].TextureHeight
;
4210 // Выравнивание по сетке:
4216 else // Нет выравнивания по сетке:
4222 // Новая позиция мыши:
4224 begin // Зажата левая кнопка мыши
4225 MousePos
.X
:= (Round((X
-MouseLDownPos
.X
)/sX
)*sX
)+MouseLDownPos
.X
;
4226 MousePos
.Y
:= (Round((Y
-MouseLDownPos
.Y
)/sY
)*sY
)+MouseLDownPos
.Y
;
4230 begin // Зажата правая кнопка мыши
4231 MousePos
.X
:= (Round((X
-MouseRDownPos
.X
)/sX
)*sX
)+MouseRDownPos
.X
;
4232 MousePos
.Y
:= (Round((Y
-MouseRDownPos
.Y
)/sY
)*sY
)+MouseRDownPos
.Y
;
4235 begin // Кнопки мыши не зажаты
4236 MousePos
.X
:= Round((-MapOffset
.X
+ X
) / sX
) * sX
+ MapOffset
.X
;
4237 MousePos
.Y
:= Round((-MapOffset
.Y
+ Y
) / sY
) * sY
+ MapOffset
.Y
;
4240 // Зажата только правая кнопка мыши:
4241 if (not MouseLDown
) and (MouseRDown
) and (not MouseMDown
) then
4243 // Рисуем прямоугольник выделения:
4244 if MouseAction
= MOUSEACTION_NONE
then
4246 if DrawRect
= nil then
4248 DrawRect
.Top
:= MouseRDownPos
.y
;
4249 DrawRect
.Left
:= MouseRDownPos
.x
;
4250 DrawRect
.Bottom
:= MousePos
.y
;
4251 DrawRect
.Right
:= MousePos
.x
;
4254 // Двигаем выделенные объекты:
4255 if MouseAction
= MOUSEACTION_MOVEOBJ
then
4257 MoveSelectedObjects(ssShift
in Shift
, ssCtrl
in Shift
,
4258 MousePos
.X
-LastMovePoint
.X
,
4259 MousePos
.Y
-LastMovePoint
.Y
);
4262 // Меняем размер выделенного объекта:
4263 if MouseAction
= MOUSEACTION_RESIZE
then
4265 if (SelectedObjectCount
= 1) and
4266 (SelectedObjects
[GetFirstSelected
].Live
) then
4268 dWidth
:= MousePos
.X
-LastMovePoint
.X
;
4269 dHeight
:= MousePos
.Y
-LastMovePoint
.Y
;
4272 RESIZETYPE_VERTICAL
: dWidth
:= 0;
4273 RESIZETYPE_HORIZONTAL
: dHeight
:= 0;
4276 case ResizeDirection
of
4277 RESIZEDIR_UP
: dHeight
:= -dHeight
;
4278 RESIZEDIR_LEFT
: dWidth
:= -dWidth
;
4281 if ResizeObject(SelectedObjects
[GetFirstSelected
].ObjectType
,
4282 SelectedObjects
[GetFirstSelected
].ID
,
4283 dWidth
, dHeight
, ResizeDirection
) then
4284 LastMovePoint
:= MousePos
;
4289 // Зажата только левая кнопка мыши:
4290 if (not MouseRDown
) and (MouseLDown
) and (not MouseMDown
) then
4292 // Рисуем прямоугольник планирования панели:
4293 if MouseAction
in [MOUSEACTION_DRAWPANEL
,
4294 MOUSEACTION_DRAWTRIGGER
,
4295 MOUSEACTION_DRAWPRESS
] then
4297 if DrawRect
= nil then
4299 if ssCtrl
in Shift
then
4303 if (lbTextureList
.ItemIndex
<> -1) and (not IsSpecialTextureSel()) and
4304 (MouseAction
= MOUSEACTION_DRAWPANEL
) then
4306 if not g_GetTexture(SelectedTexture(), TextureID
) then
4307 g_GetTexture('NOTEXTURE', TextureID
);
4308 g_GetTextureSizeByID(TextureID
, wWidth
, wHeight
);
4310 DrawRect
.Top
:= MouseLDownPos
.y
;
4311 DrawRect
.Left
:= MouseLDownPos
.x
;
4312 DrawRect
.Bottom
:= DrawRect
.Top
+ wHeight
;
4313 DrawRect
.Right
:= DrawRect
.Left
+ wWidth
;
4317 DrawRect
.Top
:= MouseLDownPos
.y
;
4318 DrawRect
.Left
:= MouseLDownPos
.x
;
4319 DrawRect
.Bottom
:= MousePos
.y
;
4320 DrawRect
.Right
:= MousePos
.x
;
4323 else // Двигаем карту:
4324 if MouseAction
= MOUSEACTION_MOVEMAP
then
4330 // Only Middle Mouse Button is pressed
4331 if (not MouseLDown
) and (not MouseRDown
) and (MouseMDown
) then
4333 MapOffset
.X
:= -EnsureRange(-MapOffset
.X
+ MouseMDownPos
.X
- Mouse
.CursorPos
.X
,
4334 sbHorizontal
.Min
, sbHorizontal
.Max
);
4335 sbHorizontal
.Position
:= -MapOffset
.X
;
4336 MapOffset
.Y
:= -EnsureRange(-MapOffset
.Y
+ MouseMDownPos
.Y
- Mouse
.CursorPos
.Y
,
4337 sbVertical
.Min
, sbVertical
.Max
);
4338 sbVertical
.Position
:= -MapOffset
.Y
;
4339 MouseMDownPos
:= Mouse
.CursorPos
;
4342 // Клавиши мыши не зажаты:
4343 if (not MouseRDown
) and (not MouseLDown
) then
4346 // Строка состояния - координаты мыши:
4347 StatusBar
.Panels
[1].Text := Format('(%d:%d)',
4348 [MousePos
.X
-MapOffset
.X
, MousePos
.Y
-MapOffset
.Y
]);
4350 RenderPanel
.Invalidate
;
4353 procedure TMainForm
.FormCloseQuery(Sender
: TObject
; var CanClose
: Boolean);
4355 CanClose
:= Application
.MessageBox(PChar(MsgMsgExitPromt
),
4357 MB_ICONQUESTION
or MB_YESNO
or
4358 MB_DEFBUTTON1
) = idYes
;
4361 procedure TMainForm
.aExitExecute(Sender
: TObject
);
4366 procedure TMainForm
.FormDestroy(Sender
: TObject
);
4372 config
:= TConfig
.CreateFile(CfgFileName
);
4374 if WindowState
<> wsMaximized
then
4376 config
.WriteInt('Editor', 'XPos', Left
);
4377 config
.WriteInt('Editor', 'YPos', Top
);
4378 config
.WriteInt('Editor', 'Width', Width
);
4379 config
.WriteInt('Editor', 'Height', Height
);
4383 config
.WriteInt('Editor', 'XPos', RestoredLeft
);
4384 config
.WriteInt('Editor', 'YPos', RestoredTop
);
4385 config
.WriteInt('Editor', 'Width', RestoredWidth
);
4386 config
.WriteInt('Editor', 'Height', RestoredHeight
);
4388 config
.WriteBool('Editor', 'Maximize', WindowState
= wsMaximized
);
4389 config
.WriteBool('Editor', 'Minimap', ShowMap
);
4390 config
.WriteInt('Editor', 'PanelProps', PanelProps
.ClientWidth
);
4391 config
.WriteInt('Editor', 'PanelObjs', PanelObjs
.ClientHeight
);
4392 config
.WriteBool('Editor', 'DotEnable', DotEnable
);
4393 config
.WriteInt('Editor', 'DotStep', DotStep
);
4394 config
.WriteStr('Editor', 'LastOpenDir', OpenDialog
.InitialDir
);
4395 config
.WriteStr('Editor', 'LastSaveDir', SaveDialog
.InitialDir
);
4396 config
.WriteStr('Editor', 'Language', gLanguage
);
4397 config
.WriteBool('Editor', 'EdgeShow', drEdge
[3] < 255);
4398 config
.WriteInt('Editor', 'EdgeColor', gColorEdge
);
4399 config
.WriteInt('Editor', 'EdgeAlpha', gAlphaEdge
);
4400 config
.WriteInt('Editor', 'LineAlpha', gAlphaTriggerLine
);
4401 config
.WriteInt('Editor', 'TriggerAlpha', gAlphaTriggerArea
);
4402 config
.WriteInt('Editor', 'MonsterRectAlpha', gAlphaMonsterRect
);
4403 config
.WriteInt('Editor', 'AreaRectAlpha', gAlphaAreaRect
);
4405 for i
:= 0 to RecentCount
- 1 do
4407 if i
< RecentFiles
.Count
then s
:= RecentFiles
[i
] else s
:= '';
4409 config
.WriteStr('RecentFilesWin', IntToStr(i
), s
);
4411 config
.WriteStr('RecentFilesUnix', IntToStr(i
), s
);
4416 config
.SaveFile(CfgFileName
);
4419 slInvalidTextures
.Free
;
4422 procedure TMainForm
.FormDropFiles(Sender
: TObject
;
4423 const FileNames
: array of String);
4425 if Length(FileNames
) <> 1 then
4428 OpenMapFile(FileNames
[0]);
4431 procedure TMainForm
.RenderPanelResize(Sender
: TObject
);
4433 if MainForm
.Visible
then
4437 procedure TMainForm
.Splitter1Moved(Sender
: TObject
);
4442 procedure TMainForm
.MapTestCheck(Sender
: TObject
);
4444 if MapTestProcess
<> nil then
4446 if MapTestProcess
.Running
= false then
4448 if MapTestProcess
.ExitCode
<> 0 then
4449 Application
.MessageBox(PChar(MsgMsgExecError
), 'FIXME', MB_OK
or MB_ICONERROR
);
4450 SysUtils
.DeleteFile(MapTestFile
);
4452 FreeAndNil(MapTestProcess
);
4453 tbTestMap
.Enabled
:= True;
4458 procedure TMainForm
.aMapOptionsExecute(Sender
: TObject
);
4462 MapOptionsForm
.ShowModal();
4464 ResName
:= OpenedMap
;
4465 while (Pos(':\', ResName
) > 0) do
4466 Delete(ResName
, 1, Pos(':\', ResName
) + 1);
4468 UpdateCaption(gMapInfo
.Name
, ExtractFileName(OpenedWAD
), ResName
);
4471 procedure TMainForm
.aAboutExecute(Sender
: TObject
);
4473 AboutForm
.ShowModal();
4476 procedure TMainForm
.FormKeyDown(Sender
: TObject
; var Key
: Word; Shift
: TShiftState
);
4482 if (not EditingProperties
) then
4484 if ssCtrl
in Shift
then
4487 '1': ContourEnabled
[LAYER_BACK
] := not ContourEnabled
[LAYER_BACK
];
4488 '2': ContourEnabled
[LAYER_WALLS
] := not ContourEnabled
[LAYER_WALLS
];
4489 '3': ContourEnabled
[LAYER_FOREGROUND
] := not ContourEnabled
[LAYER_FOREGROUND
];
4490 '4': ContourEnabled
[LAYER_STEPS
] := not ContourEnabled
[LAYER_STEPS
];
4491 '5': ContourEnabled
[LAYER_WATER
] := not ContourEnabled
[LAYER_WATER
];
4492 '6': ContourEnabled
[LAYER_ITEMS
] := not ContourEnabled
[LAYER_ITEMS
];
4493 '7': ContourEnabled
[LAYER_MONSTERS
] := not ContourEnabled
[LAYER_MONSTERS
];
4494 '8': ContourEnabled
[LAYER_AREAS
] := not ContourEnabled
[LAYER_AREAS
];
4495 '9': ContourEnabled
[LAYER_TRIGGERS
] := not ContourEnabled
[LAYER_TRIGGERS
];
4499 for i
:= Low(ContourEnabled
) to High(ContourEnabled
) do
4500 if ContourEnabled
[i
] then
4502 for i
:= Low(ContourEnabled
) to High(ContourEnabled
) do
4503 ContourEnabled
[i
] := not ok
4510 '1': SwitchLayer(LAYER_BACK
);
4511 '2': SwitchLayer(LAYER_WALLS
);
4512 '3': SwitchLayer(LAYER_FOREGROUND
);
4513 '4': SwitchLayer(LAYER_STEPS
);
4514 '5': SwitchLayer(LAYER_WATER
);
4515 '6': SwitchLayer(LAYER_ITEMS
);
4516 '7': SwitchLayer(LAYER_MONSTERS
);
4517 '8': SwitchLayer(LAYER_AREAS
);
4518 '9': SwitchLayer(LAYER_TRIGGERS
);
4519 '0': tbShowClick(tbShow
);
4523 if Key
= Ord('I') then
4524 begin // Поворот монстров и областей:
4525 if (SelectedObjects
<> nil) then
4527 for i
:= 0 to High(SelectedObjects
) do
4528 if (SelectedObjects
[i
].Live
) then
4530 if (SelectedObjects
[i
].ObjectType
= OBJECT_MONSTER
) then
4532 g_ChangeDir(gMonsters
[SelectedObjects
[i
].ID
].Direction
);
4535 if (SelectedObjects
[i
].ObjectType
= OBJECT_AREA
) then
4537 g_ChangeDir(gAreas
[SelectedObjects
[i
].ID
].Direction
);
4543 if pcObjects
.ActivePage
= tsMonsters
then
4545 if rbMonsterLeft
.Checked
then
4546 rbMonsterRight
.Checked
:= True
4548 rbMonsterLeft
.Checked
:= True;
4550 if pcObjects
.ActivePage
= tsAreas
then
4552 if rbAreaLeft
.Checked
then
4553 rbAreaRight
.Checked
:= True
4555 rbAreaLeft
.Checked
:= True;
4560 if not (ssCtrl
in Shift
) then
4562 // Быстрое превью карты:
4563 if Key
= Ord('E') then
4565 if PreviewMode
= 0 then
4569 // Вертикальный скролл карты:
4572 if Key
= Ord('W') then
4575 if ssShift
in Shift
then Position
:= EnsureRange(Position
- DotStep
* 4, Min
, Max
)
4576 else Position
:= EnsureRange(Position
- DotStep
, Min
, Max
);
4577 MapOffset
.Y
:= -Position
;
4580 if (MouseLDown
or MouseRDown
) then
4582 if DrawRect
<> nil then
4584 Inc(MouseLDownPos
.y
, dy
);
4585 Inc(MouseRDownPos
.y
, dy
);
4587 Inc(LastMovePoint
.Y
, dy
);
4588 RenderPanelMouseMove(Sender
, Shift
, RenderMousePos().X
, RenderMousePos().Y
);
4592 if Key
= Ord('S') then
4595 if ssShift
in Shift
then Position
:= EnsureRange(Position
+ DotStep
* 4, Min
, Max
)
4596 else Position
:= EnsureRange(Position
+ DotStep
, Min
, Max
);
4597 MapOffset
.Y
:= -Position
;
4600 if (MouseLDown
or MouseRDown
) then
4602 if DrawRect
<> nil then
4604 Inc(MouseLDownPos
.y
, dy
);
4605 Inc(MouseRDownPos
.y
, dy
);
4607 Inc(LastMovePoint
.Y
, dy
);
4608 RenderPanelMouseMove(Sender
, Shift
, RenderMousePos().X
, RenderMousePos().Y
);
4613 // Горизонтальный скролл карты:
4614 with sbHorizontal
do
4616 if Key
= Ord('A') then
4619 if ssShift
in Shift
then Position
:= EnsureRange(Position
- DotStep
* 4, Min
, Max
)
4620 else Position
:= EnsureRange(Position
- DotStep
, Min
, Max
);
4621 MapOffset
.X
:= -Position
;
4624 if (MouseLDown
or MouseRDown
) then
4626 if DrawRect
<> nil then
4628 Inc(MouseLDownPos
.x
, dx
);
4629 Inc(MouseRDownPos
.x
, dx
);
4631 Inc(LastMovePoint
.X
, dx
);
4632 RenderPanelMouseMove(Sender
, Shift
, RenderMousePos().X
, RenderMousePos().Y
);
4636 if Key
= Ord('D') then
4639 if ssShift
in Shift
then Position
:= EnsureRange(Position
+ DotStep
* 4, Min
, Max
)
4640 else Position
:= EnsureRange(Position
+ DotStep
, Min
, Max
);
4641 MapOffset
.X
:= -Position
;
4644 if (MouseLDown
or MouseRDown
) then
4646 if DrawRect
<> nil then
4648 Inc(MouseLDownPos
.x
, dx
);
4649 Inc(MouseRDownPos
.x
, dx
);
4651 Inc(LastMovePoint
.X
, dx
);
4652 RenderPanelMouseMove(Sender
, Shift
, RenderMousePos().X
, RenderMousePos().Y
);
4657 else // ssCtrl in Shift
4659 if ssShift
in Shift
then
4661 // Вставка по абсолютному смещению:
4662 if Key
= Ord('V') then
4663 aPasteObjectExecute(Sender
);
4665 RenderPanelMouseMove(Sender
, Shift
, RenderMousePos().X
, RenderMousePos().Y
);
4669 // Удалить выделенные объекты:
4670 if (Key
= VK_DELETE
) and (SelectedObjects
<> nil) and
4671 RenderPanel
.Focused() then
4672 DeleteSelectedObjects();
4675 if (Key
= VK_ESCAPE
) and (SelectedObjects
<> nil) then
4676 RemoveSelectFromObjects();
4678 // Передвинуть объекты:
4679 if MainForm
.ActiveControl
= RenderPanel
then
4684 if Key
= VK_NUMPAD4
then
4685 dx
:= IfThen(ssAlt
in Shift
, -1, -DotStep
);
4686 if Key
= VK_NUMPAD6
then
4687 dx
:= IfThen(ssAlt
in Shift
, 1, DotStep
);
4688 if Key
= VK_NUMPAD8
then
4689 dy
:= IfThen(ssAlt
in Shift
, -1, -DotStep
);
4690 if Key
= VK_NUMPAD5
then
4691 dy
:= IfThen(ssAlt
in Shift
, 1, DotStep
);
4693 if (dx
<> 0) or (dy
<> 0) then
4695 MoveSelectedObjects(ssShift
in Shift
, ssCtrl
in Shift
, dx
, dy
);
4700 if ssCtrl
in Shift
then
4702 // Выбор панели с текстурой для триггера
4703 if Key
= Ord('T') then
4705 DrawPressRect
:= False;
4706 if SelectFlag
= SELECTFLAG_TEXTURE
then
4708 SelectFlag
:= SELECTFLAG_NONE
;
4711 vleObjectProperty
.FindRow(MsgPropTrTexturePanel
, i
);
4713 SelectFlag
:= SELECTFLAG_TEXTURE
;
4716 if Key
= Ord('D') then
4718 SelectFlag
:= SELECTFLAG_NONE
;
4719 if DrawPressRect
then
4721 DrawPressRect
:= False;
4726 // Выбор области воздействия, в зависимости от типа триггера
4727 vleObjectProperty
.FindRow(MsgPropTrExArea
, i
);
4730 DrawPressRect
:= True;
4733 vleObjectProperty
.FindRow(MsgPropTrDoorPanel
, i
);
4735 vleObjectProperty
.FindRow(MsgPropTrTrapPanel
, i
);
4738 SelectFlag
:= SELECTFLAG_DOOR
;
4741 vleObjectProperty
.FindRow(MsgPropTrLiftPanel
, i
);
4744 SelectFlag
:= SELECTFLAG_LIFT
;
4747 vleObjectProperty
.FindRow(MsgPropTrTeleportTo
, i
);
4750 SelectFlag
:= SELECTFLAG_TELEPORT
;
4753 vleObjectProperty
.FindRow(MsgPropTrSpawnTo
, i
);
4756 SelectFlag
:= SELECTFLAG_SPAWNPOINT
;
4760 // Выбор основного параметра, в зависимости от типа триггера
4761 vleObjectProperty
.FindRow(MsgPropTrNextMap
, i
);
4764 g_ProcessResourceStr(OpenedMap
, @FileName
, nil, nil);
4765 SelectMapForm
.Caption
:= MsgCapSelect
;
4766 SelectMapForm
.GetMaps(FileName
);
4768 if SelectMapForm
.ShowModal() = mrOK
then
4770 vleObjectProperty
.Cells
[1, i
] := SelectMapForm
.lbMapList
.Items
[SelectMapForm
.lbMapList
.ItemIndex
];
4771 bApplyProperty
.Click();
4775 vleObjectProperty
.FindRow(MsgPropTrSoundName
, i
);
4777 vleObjectProperty
.FindRow(MsgPropTrMusicName
, i
);
4780 AddSoundForm
.OKFunction
:= nil;
4781 AddSoundForm
.lbResourcesList
.MultiSelect
:= False;
4782 AddSoundForm
.SetResource
:= vleObjectProperty
.Cells
[1, i
];
4784 if (AddSoundForm
.ShowModal() = mrOk
) then
4786 vleObjectProperty
.Cells
[1, i
] := AddSoundForm
.ResourceName
;
4787 bApplyProperty
.Click();
4791 vleObjectProperty
.FindRow(MsgPropTrPushAngle
, i
);
4793 vleObjectProperty
.FindRow(MsgPropTrMessageText
, i
);
4796 vleObjectProperty
.Row
:= i
;
4797 vleObjectProperty
.SetFocus();
4804 procedure TMainForm
.aOptimizeExecute(Sender
: TObject
);
4806 RemoveSelectFromObjects();
4807 MapOptimizationForm
.ShowModal();
4810 procedure TMainForm
.aCheckMapExecute(Sender
: TObject
);
4812 MapCheckForm
.ShowModal();
4815 procedure TMainForm
.bbAddTextureClick(Sender
: TObject
);
4817 AddTextureForm
.lbResourcesList
.MultiSelect
:= True;
4818 AddTextureForm
.ShowModal();
4821 procedure TMainForm
.lbTextureListClick(Sender
: TObject
);
4824 TextureWidth
, TextureHeight
: Word;
4829 if (lbTextureList
.ItemIndex
<> -1) and
4830 (not IsSpecialTextureSel()) then
4832 if g_GetTexture(SelectedTexture(), TextureID
) then
4834 g_GetTextureSizeByID(TextureID
, TextureWidth
, TextureHeight
);
4836 lTextureWidth
.Caption
:= IntToStr(TextureWidth
);
4837 lTextureHeight
.Caption
:= IntToStr(TextureHeight
);
4840 lTextureWidth
.Caption
:= MsgNotAccessible
;
4841 lTextureHeight
.Caption
:= MsgNotAccessible
;
4846 lTextureWidth
.Caption
:= '';
4847 lTextureHeight
.Caption
:= '';
4851 procedure TMainForm
.lbTextureListDrawItem(Control
: TWinControl
; Index
: Integer;
4852 ARect
: TRect
; State
: TOwnerDrawState
);
4854 with Control
as TListBox
do
4856 if LCLType
.odSelected
in State
then
4858 Canvas
.Brush
.Color
:= clHighlight
;
4859 Canvas
.Font
.Color
:= clHighlightText
;
4861 if (Items
<> nil) and (Index
>= 0) then
4862 if slInvalidTextures
.IndexOf(Items
[Index
]) > -1 then
4864 Canvas
.Brush
.Color
:= clRed
;
4865 Canvas
.Font
.Color
:= clWhite
;
4867 Canvas
.FillRect(ARect
);
4868 Canvas
.TextRect(ARect
, ARect
.Left
, ARect
.Top
, Items
[Index
]);
4872 procedure TMainForm
.miMacMinimizeClick(Sender
: TObject
);
4874 self
.WindowState
:= wsMinimized
;
4875 self
.FormWindowStateChange(Sender
);
4878 procedure TMainForm
.miMacZoomClick(Sender
: TObject
);
4880 if self
.WindowState
= wsMaximized
then
4881 self
.WindowState
:= wsNormal
4883 self
.WindowState
:= wsMaximized
;
4884 self
.FormWindowStateChange(Sender
);
4887 procedure TMainForm
.miReopenMapClick(Sender
: TObject
);
4889 FileName
, Resource
: String;
4891 if OpenedMap
= '' then
4894 if Application
.MessageBox(PChar(MsgMsgReopenMapPromt
),
4895 PChar(MsgMenuFileReopen
), MB_ICONQUESTION
or MB_YESNO
) <> idYes
then
4898 g_ProcessResourceStr(OpenedMap
, @FileName
, nil, @Resource
);
4899 OpenMap(FileName
, Resource
);
4902 procedure TMainForm
.vleObjectPropertyGetPickList(Sender
: TObject
;
4903 const KeyName
: String; Values
: TStrings
);
4905 if vleObjectProperty
.ItemProps
[KeyName
].EditStyle
= esPickList
then
4907 if KeyName
= MsgPropDirection
then
4909 Values
.Add(DirNames
[D_LEFT
]);
4910 Values
.Add(DirNames
[D_RIGHT
]);
4912 else if KeyName
= MsgPropTrTeleportDir
then
4914 Values
.Add(DirNamesAdv
[0]);
4915 Values
.Add(DirNamesAdv
[1]);
4916 Values
.Add(DirNamesAdv
[2]);
4917 Values
.Add(DirNamesAdv
[3]);
4919 else if KeyName
= MsgPropTrMusicAct
then
4921 Values
.Add(MsgPropTrMusicOn
);
4922 Values
.Add(MsgPropTrMusicOff
);
4924 else if KeyName
= MsgPropTrMonsterBehaviour
then
4926 Values
.Add(MsgPropTrMonsterBehaviour0
);
4927 Values
.Add(MsgPropTrMonsterBehaviour1
);
4928 Values
.Add(MsgPropTrMonsterBehaviour2
);
4929 Values
.Add(MsgPropTrMonsterBehaviour3
);
4930 Values
.Add(MsgPropTrMonsterBehaviour4
);
4931 Values
.Add(MsgPropTrMonsterBehaviour5
);
4933 else if KeyName
= MsgPropTrScoreAct
then
4935 Values
.Add(MsgPropTrScoreAct0
);
4936 Values
.Add(MsgPropTrScoreAct1
);
4937 Values
.Add(MsgPropTrScoreAct2
);
4938 Values
.Add(MsgPropTrScoreAct3
);
4940 else if KeyName
= MsgPropTrScoreTeam
then
4942 Values
.Add(MsgPropTrScoreTeam0
);
4943 Values
.Add(MsgPropTrScoreTeam1
);
4944 Values
.Add(MsgPropTrScoreTeam2
);
4945 Values
.Add(MsgPropTrScoreTeam3
);
4947 else if KeyName
= MsgPropTrMessageKind
then
4949 Values
.Add(MsgPropTrMessageKind0
);
4950 Values
.Add(MsgPropTrMessageKind1
);
4952 else if KeyName
= MsgPropTrMessageTo
then
4954 Values
.Add(MsgPropTrMessageTo0
);
4955 Values
.Add(MsgPropTrMessageTo1
);
4956 Values
.Add(MsgPropTrMessageTo2
);
4957 Values
.Add(MsgPropTrMessageTo3
);
4958 Values
.Add(MsgPropTrMessageTo4
);
4959 Values
.Add(MsgPropTrMessageTo5
);
4961 else if KeyName
= MsgPropTrShotTo
then
4963 Values
.Add(MsgPropTrShotTo0
);
4964 Values
.Add(MsgPropTrShotTo1
);
4965 Values
.Add(MsgPropTrShotTo2
);
4966 Values
.Add(MsgPropTrShotTo3
);
4967 Values
.Add(MsgPropTrShotTo4
);
4968 Values
.Add(MsgPropTrShotTo5
);
4969 Values
.Add(MsgPropTrShotTo6
);
4971 else if KeyName
= MsgPropTrShotAim
then
4973 Values
.Add(MsgPropTrShotAim0
);
4974 Values
.Add(MsgPropTrShotAim1
);
4975 Values
.Add(MsgPropTrShotAim2
);
4976 Values
.Add(MsgPropTrShotAim3
);
4978 else if KeyName
= MsgPropTrDamageKind
then
4980 Values
.Add(MsgPropTrDamageKind0
);
4981 Values
.Add(MsgPropTrDamageKind3
);
4982 Values
.Add(MsgPropTrDamageKind4
);
4983 Values
.Add(MsgPropTrDamageKind5
);
4984 Values
.Add(MsgPropTrDamageKind6
);
4985 Values
.Add(MsgPropTrDamageKind7
);
4986 Values
.Add(MsgPropTrDamageKind8
);
4988 else if (KeyName
= MsgPropPanelBlend
) or
4989 (KeyName
= MsgPropDmOnly
) or
4990 (KeyName
= MsgPropItemFalls
) or
4991 (KeyName
= MsgPropTrEnabled
) or
4992 (KeyName
= MsgPropTrD2d
) or
4993 (KeyName
= MsgPropTrSilent
) or
4994 (KeyName
= MsgPropTrTeleportSilent
) or
4995 (KeyName
= MsgPropTrExRandom
) or
4996 (KeyName
= MsgPropTrTextureOnce
) or
4997 (KeyName
= MsgPropTrTextureAnimOnce
) or
4998 (KeyName
= MsgPropTrSoundLocal
) or
4999 (KeyName
= MsgPropTrSoundSwitch
) or
5000 (KeyName
= MsgPropTrMonsterActive
) or
5001 (KeyName
= MsgPropTrPushReset
) or
5002 (KeyName
= MsgPropTrScoreCon
) or
5003 (KeyName
= MsgPropTrScoreMsg
) or
5004 (KeyName
= MsgPropTrHealthMax
) or
5005 (KeyName
= MsgPropTrShotSound
) or
5006 (KeyName
= MsgPropTrEffectCenter
) then
5008 Values
.Add(BoolNames
[True]);
5009 Values
.Add(BoolNames
[False]);
5014 procedure TMainForm
.bApplyPropertyClick(Sender
: TObject
);
5016 _id
, a
, r
, c
: Integer;
5026 if SelectedObjectCount() <> 1 then
5028 if not SelectedObjects
[GetFirstSelected()].Live
then
5032 if not CheckProperty() then
5038 _id
:= GetFirstSelected();
5040 r
:= vleObjectProperty
.Row
;
5041 c
:= vleObjectProperty
.Col
;
5043 case SelectedObjects
[_id
].ObjectType
of
5046 with gPanels
[SelectedObjects
[_id
].ID
] do
5048 X
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropX
]));
5049 Y
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropY
]));
5050 Width
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropWidth
]));
5051 Height
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropHeight
]));
5053 PanelType
:= GetPanelType(vleObjectProperty
.Values
[MsgPropPanelType
]);
5055 // Сброс ссылки на триггеры смены текстуры:
5056 if not WordBool(PanelType
and (PANEL_WALL
or PANEL_FORE
or PANEL_BACK
)) then
5057 if gTriggers
<> nil then
5058 for a
:= 0 to High(gTriggers
) do
5060 if (gTriggers
[a
].TriggerType
<> 0) and
5061 (gTriggers
[a
].TexturePanel
= Integer(SelectedObjects
[_id
].ID
)) then
5062 gTriggers
[a
].TexturePanel
:= -1;
5063 if (gTriggers
[a
].TriggerType
= TRIGGER_SHOT
) and
5064 (gTriggers
[a
].Data
.ShotPanelID
= Integer(SelectedObjects
[_id
].ID
)) then
5065 gTriggers
[a
].Data
.ShotPanelID
:= -1;
5068 // Сброс ссылки на триггеры лифта:
5069 if not WordBool(PanelType
and (PANEL_LIFTUP
or PANEL_LIFTDOWN
or PANEL_LIFTLEFT
or PANEL_LIFTRIGHT
)) then
5070 if gTriggers
<> nil then
5071 for a
:= 0 to High(gTriggers
) do
5072 if (gTriggers
[a
].TriggerType
in [TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
]) and
5073 (gTriggers
[a
].Data
.PanelID
= Integer(SelectedObjects
[_id
].ID
)) then
5074 gTriggers
[a
].Data
.PanelID
:= -1;
5076 // Сброс ссылки на триггеры двери:
5077 if not WordBool(PanelType
and (PANEL_OPENDOOR
or PANEL_CLOSEDOOR
)) then
5078 if gTriggers
<> nil then
5079 for a
:= 0 to High(gTriggers
) do
5080 if (gTriggers
[a
].TriggerType
in [TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
,
5081 TRIGGER_DOOR5
, TRIGGER_CLOSETRAP
, TRIGGER_TRAP
]) and
5082 (gTriggers
[a
].Data
.PanelID
= Integer(SelectedObjects
[_id
].ID
)) then
5083 gTriggers
[a
].Data
.PanelID
:= -1;
5085 if IsTexturedPanel(PanelType
) then
5086 begin // Может быть текстура
5087 if TextureName
<> '' then
5088 begin // Была текстура
5089 Alpha
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropPanelAlpha
]));
5090 Blending
:= NameToBool(vleObjectProperty
.Values
[MsgPropPanelBlend
]);
5099 TextureName
:= vleObjectProperty
.Values
[MsgPropPanelTex
];
5101 if TextureName
<> '' then
5102 begin // Есть текстура
5103 // Обычная текстура:
5104 if not IsSpecialTexture(TextureName
) then
5106 g_GetTextureSizeByName(TextureName
,
5107 TextureWidth
, TextureHeight
);
5109 // Проверка кратности размеров панели:
5111 if TextureWidth
<> 0 then
5112 if gPanels
[SelectedObjects
[_id
].ID
].Width
mod TextureWidth
<> 0 then
5114 ErrorMessageBox(Format(MsgMsgWrongTexwidth
,
5118 if Res
and (TextureHeight
<> 0) then
5119 if gPanels
[SelectedObjects
[_id
].ID
].Height
mod TextureHeight
<> 0 then
5121 ErrorMessageBox(Format(MsgMsgWrongTexheight
,
5128 if not g_GetTexture(TextureName
, TextureID
) then
5129 // Не удалось загрузить текстуру, рисуем NOTEXTURE
5130 if g_GetTexture('NOTEXTURE', NoTextureID
) then
5132 TextureID
:= TEXTURE_SPECIAL_NOTEXTURE
;
5133 g_GetTextureSizeByID(NoTextureID
, NW
, NH
);
5135 TextureHeight
:= NH
;
5138 TextureID
:= TEXTURE_SPECIAL_NONE
;
5148 TextureID
:= TEXTURE_SPECIAL_NONE
;
5151 else // Спец.текстура
5155 TextureID
:= SpecialTextureID(TextureName
);
5158 else // Нет текстуры
5162 TextureID
:= TEXTURE_SPECIAL_NONE
;
5165 else // Не может быть текстуры
5172 TextureID
:= TEXTURE_SPECIAL_NONE
;
5179 with gItems
[SelectedObjects
[_id
].ID
] do
5181 X
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropX
]));
5182 Y
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropY
]));
5183 OnlyDM
:= NameToBool(vleObjectProperty
.Values
[MsgPropDmOnly
]);
5184 Fall
:= NameToBool(vleObjectProperty
.Values
[MsgPropItemFalls
]);
5190 with gMonsters
[SelectedObjects
[_id
].ID
] do
5192 X
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropX
]));
5193 Y
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropY
]));
5194 Direction
:= NameToDir(vleObjectProperty
.Values
[MsgPropDirection
]);
5200 with gAreas
[SelectedObjects
[_id
].ID
] do
5202 X
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropX
]));
5203 Y
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropY
]));
5204 Direction
:= NameToDir(vleObjectProperty
.Values
[MsgPropDirection
]);
5210 with gTriggers
[SelectedObjects
[_id
].ID
] do
5212 X
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropX
]));
5213 Y
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropY
]));
5214 Width
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropWidth
]));
5215 Height
:= StrToInt(Trim(vleObjectProperty
.Values
[MsgPropHeight
]));
5216 Enabled
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrEnabled
]);
5217 ActivateType
:= StrToActivate(vleObjectProperty
.Values
[MsgPropTrActivation
]);
5218 Key
:= StrToKey(vleObjectProperty
.Values
[MsgPropTrKeys
]);
5223 s
:= utf2win(vleObjectProperty
.Values
[MsgPropTrNextMap
]);
5224 FillByte(Data
.MapName
[0], 16, 0);
5226 Move(s
[1], Data
.MapName
[0], Min(Length(s
), 16));
5231 Data
.ActivateOnce
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrTextureOnce
]);
5232 Data
.AnimOnce
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrTextureAnimOnce
]);
5235 TRIGGER_PRESS
, TRIGGER_ON
, TRIGGER_OFF
, TRIGGER_ONOFF
:
5237 Data
.Wait
:= Min(StrToIntDef(vleObjectProperty
.Values
[MsgPropTrExDelay
], 0), 65535);
5238 Data
.Count
:= Min(StrToIntDef(vleObjectProperty
.Values
[MsgPropTrExCount
], 0), 65535);
5239 if Data
.Count
< 1 then
5241 if TriggerType
= TRIGGER_PRESS
then
5242 Data
.ExtRandom
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrExRandom
]);
5245 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
, TRIGGER_DOOR5
,
5246 TRIGGER_CLOSETRAP
, TRIGGER_TRAP
, TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
,
5249 Data
.NoSound
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrSilent
]);
5250 Data
.d2d_doors
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrD2d
]);
5255 Data
.d2d_teleport
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrD2d
]);
5256 Data
.silent_teleport
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrTeleportSilent
]);
5257 Data
.TlpDir
:= NameToDirAdv(vleObjectProperty
.Values
[MsgPropTrTeleportDir
]);
5262 s
:= utf2win(vleObjectProperty
.Values
[MsgPropTrSoundName
]);
5263 FillByte(Data
.SoundName
[0], 64, 0);
5265 Move(s
[1], Data
.SoundName
[0], Min(Length(s
), 64));
5267 Data
.Volume
:= Min(StrToIntDef(vleObjectProperty
.Values
[MsgPropTrSoundVolume
], 0), 255);
5268 Data
.Pan
:= Min(StrToIntDef(vleObjectProperty
.Values
[MsgPropTrSoundPan
], 0), 255);
5269 Data
.PlayCount
:= Min(StrToIntDef(vleObjectProperty
.Values
[MsgPropTrSoundCount
], 0), 255);
5270 Data
.Local
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrSoundLocal
]);
5271 Data
.SoundSwitch
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrSoundSwitch
]);
5274 TRIGGER_SPAWNMONSTER
:
5276 Data
.MonType
:= StrToMonster(vleObjectProperty
.Values
[MsgPropTrMonsterType
]);
5277 Data
.MonDir
:= Byte(NameToDir(vleObjectProperty
.Values
[MsgPropDirection
]));
5278 Data
.MonHealth
:= Min(StrToIntDef(vleObjectProperty
.Values
[MsgPropTrHealth
], 0), 1000000);
5279 if Data
.MonHealth
< 0 then
5280 Data
.MonHealth
:= 0;
5281 Data
.MonActive
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrMonsterActive
]);
5282 Data
.MonCount
:= Min(StrToIntDef(vleObjectProperty
.Values
[MsgPropTrCount
], 0), 64);
5283 if Data
.MonCount
< 1 then
5285 Data
.MonEffect
:= StrToEffect(vleObjectProperty
.Values
[MsgPropTrFxType
]);
5286 Data
.MonMax
:= Min(StrToIntDef(vleObjectProperty
.Values
[MsgPropTrSpawnMax
], 0), 65535);
5287 Data
.MonDelay
:= Min(StrToIntDef(vleObjectProperty
.Values
[MsgPropTrSpawnDelay
], 0), 65535);
5289 if vleObjectProperty
.Values
[MsgPropTrMonsterBehaviour
] = MsgPropTrMonsterBehaviour1
then
5291 if vleObjectProperty
.Values
[MsgPropTrMonsterBehaviour
] = MsgPropTrMonsterBehaviour2
then
5293 if vleObjectProperty
.Values
[MsgPropTrMonsterBehaviour
] = MsgPropTrMonsterBehaviour3
then
5295 if vleObjectProperty
.Values
[MsgPropTrMonsterBehaviour
] = MsgPropTrMonsterBehaviour4
then
5297 if vleObjectProperty
.Values
[MsgPropTrMonsterBehaviour
] = MsgPropTrMonsterBehaviour5
then
5303 Data
.ItemType
:= StrToItem(vleObjectProperty
.Values
[MsgPropTrItemType
]);
5304 Data
.ItemOnlyDM
:= NameToBool(vleObjectProperty
.Values
[MsgPropDmOnly
]);
5305 Data
.ItemFalls
:= NameToBool(vleObjectProperty
.Values
[MsgPropItemFalls
]);
5306 Data
.ItemCount
:= Min(StrToIntDef(vleObjectProperty
.Values
[MsgPropTrCount
], 0), 64);
5307 if Data
.ItemCount
< 1 then
5308 Data
.ItemCount
:= 1;
5309 Data
.ItemEffect
:= StrToEffect(vleObjectProperty
.Values
[MsgPropTrFxType
]);
5310 Data
.ItemMax
:= Min(StrToIntDef(vleObjectProperty
.Values
[MsgPropTrSpawnMax
], 0), 65535);
5311 Data
.ItemDelay
:= Min(StrToIntDef(vleObjectProperty
.Values
[MsgPropTrSpawnDelay
], 0), 65535);
5316 s
:= utf2win(vleObjectProperty
.Values
[MsgPropTrMusicName
]);
5317 FillByte(Data
.MusicName
[0], 64, 0);
5319 Move(s
[1], Data
.MusicName
[0], Min(Length(s
), 64));
5321 if vleObjectProperty
.Values
[MsgPropTrMusicAct
] = MsgPropTrMusicOn
then
5322 Data
.MusicAction
:= 1
5324 Data
.MusicAction
:= 0;
5329 Data
.PushAngle
:= Min(
5330 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrPushAngle
], 0), 360);
5331 Data
.PushForce
:= Min(
5332 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrPushForce
], 0), 255);
5333 Data
.ResetVel
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrPushReset
]);
5338 Data
.ScoreAction
:= 0;
5339 if vleObjectProperty
.Values
[MsgPropTrScoreAct
] = MsgPropTrScoreAct1
then
5340 Data
.ScoreAction
:= 1
5341 else if vleObjectProperty
.Values
[MsgPropTrScoreAct
] = MsgPropTrScoreAct2
then
5342 Data
.ScoreAction
:= 2
5343 else if vleObjectProperty
.Values
[MsgPropTrScoreAct
] = MsgPropTrScoreAct3
then
5344 Data
.ScoreAction
:= 3;
5345 Data
.ScoreCount
:= Min(Max(
5346 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrCount
], 0), 0), 255);
5347 Data
.ScoreTeam
:= 0;
5348 if vleObjectProperty
.Values
[MsgPropTrScoreTeam
] = MsgPropTrScoreTeam1
then
5350 else if vleObjectProperty
.Values
[MsgPropTrScoreTeam
] = MsgPropTrScoreTeam2
then
5352 else if vleObjectProperty
.Values
[MsgPropTrScoreTeam
] = MsgPropTrScoreTeam3
then
5353 Data
.ScoreTeam
:= 3;
5354 Data
.ScoreCon
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrScoreCon
]);
5355 Data
.ScoreMsg
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrScoreMsg
]);
5360 Data
.MessageKind
:= 0;
5361 if vleObjectProperty
.Values
[MsgPropTrMessageKind
] = MsgPropTrMessageKind1
then
5362 Data
.MessageKind
:= 1;
5364 Data
.MessageSendTo
:= 0;
5365 if vleObjectProperty
.Values
[MsgPropTrMessageTo
] = MsgPropTrMessageTo1
then
5366 Data
.MessageSendTo
:= 1
5367 else if vleObjectProperty
.Values
[MsgPropTrMessageTo
] = MsgPropTrMessageTo2
then
5368 Data
.MessageSendTo
:= 2
5369 else if vleObjectProperty
.Values
[MsgPropTrMessageTo
] = MsgPropTrMessageTo3
then
5370 Data
.MessageSendTo
:= 3
5371 else if vleObjectProperty
.Values
[MsgPropTrMessageTo
] = MsgPropTrMessageTo4
then
5372 Data
.MessageSendTo
:= 4
5373 else if vleObjectProperty
.Values
[MsgPropTrMessageTo
] = MsgPropTrMessageTo5
then
5374 Data
.MessageSendTo
:= 5;
5376 s
:= utf2win(vleObjectProperty
.Values
[MsgPropTrMessageText
]);
5377 FillByte(Data
.MessageText
[0], 100, 0);
5379 Move(s
[1], Data
.MessageText
[0], Min(Length(s
), 100));
5381 Data
.MessageTime
:= Min(Max(
5382 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrMessageTime
], 0), 0), 65535);
5387 Data
.DamageValue
:= Min(Max(
5388 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrDamageValue
], 0), 0), 65535);
5389 Data
.DamageInterval
:= Min(Max(
5390 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrInterval
], 0), 0), 65535);
5391 s
:= vleObjectProperty
.Values
[MsgPropTrDamageKind
];
5392 if s
= MsgPropTrDamageKind3
then
5393 Data
.DamageKind
:= 3
5394 else if s
= MsgPropTrDamageKind4
then
5395 Data
.DamageKind
:= 4
5396 else if s
= MsgPropTrDamageKind5
then
5397 Data
.DamageKind
:= 5
5398 else if s
= MsgPropTrDamageKind6
then
5399 Data
.DamageKind
:= 6
5400 else if s
= MsgPropTrDamageKind7
then
5401 Data
.DamageKind
:= 7
5402 else if s
= MsgPropTrDamageKind8
then
5403 Data
.DamageKind
:= 8
5405 Data
.DamageKind
:= 0;
5410 Data
.HealValue
:= Min(Max(
5411 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrHealth
], 0), 0), 65535);
5412 Data
.HealInterval
:= Min(Max(
5413 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrInterval
], 0), 0), 65535);
5414 Data
.HealMax
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrHealthMax
]);
5415 Data
.HealSilent
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrSilent
]);
5420 Data
.ShotType
:= StrToShot(vleObjectProperty
.Values
[MsgPropTrShotType
]);
5421 Data
.ShotSound
:= NameToBool(vleObjectProperty
.Values
[MsgPropTrShotSound
]);
5422 Data
.ShotTarget
:= 0;
5423 if vleObjectProperty
.Values
[MsgPropTrShotTo
] = MsgPropTrShotTo1
then
5424 Data
.ShotTarget
:= 1
5425 else if vleObjectProperty
.Values
[MsgPropTrShotTo
] = MsgPropTrShotTo2
then
5426 Data
.ShotTarget
:= 2
5427 else if vleObjectProperty
.Values
[MsgPropTrShotTo
] = MsgPropTrShotTo3
then
5428 Data
.ShotTarget
:= 3
5429 else if vleObjectProperty
.Values
[MsgPropTrShotTo
] = MsgPropTrShotTo4
then
5430 Data
.ShotTarget
:= 4
5431 else if vleObjectProperty
.Values
[MsgPropTrShotTo
] = MsgPropTrShotTo5
then
5432 Data
.ShotTarget
:= 5
5433 else if vleObjectProperty
.Values
[MsgPropTrShotTo
] = MsgPropTrShotTo6
then
5434 Data
.ShotTarget
:= 6;
5435 Data
.ShotIntSight
:= Min(Max(
5436 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrShotSight
], 0), 0), 65535);
5438 if vleObjectProperty
.Values
[MsgPropTrShotAim
] = MsgPropTrShotAim1
then
5440 else if vleObjectProperty
.Values
[MsgPropTrShotAim
] = MsgPropTrShotAim2
then
5442 else if vleObjectProperty
.Values
[MsgPropTrShotAim
] = MsgPropTrShotAim3
then
5444 Data
.ShotAngle
:= Min(
5445 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrShotAngle
], 0), 360);
5446 Data
.ShotWait
:= Min(Max(
5447 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrExDelay
], 0), 0), 65535);
5448 Data
.ShotAccuracy
:= Min(Max(
5449 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrShotAcc
], 0), 0), 65535);
5450 Data
.ShotAmmo
:= Min(Max(
5451 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrShotAmmo
], 0), 0), 65535);
5452 Data
.ShotIntReload
:= Min(Max(
5453 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrShotReload
], 0), 0), 65535);
5458 Data
.FXCount
:= Min(Max(
5459 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrCount
], 0), 0), 255);
5460 if vleObjectProperty
.Values
[MsgPropTrEffectType
] = MsgPropTrEffectParticle
then
5462 Data
.FXType
:= TRIGGER_EFFECT_PARTICLE
;
5463 Data
.FXSubType
:= TRIGGER_EFFECT_SLIQUID
;
5464 if vleObjectProperty
.Values
[MsgPropTrEffectSubtype
] = MsgPropTrEffectSliquid
then
5465 Data
.FXSubType
:= TRIGGER_EFFECT_SLIQUID
5466 else if vleObjectProperty
.Values
[MsgPropTrEffectSubtype
] = MsgPropTrEffectLliquid
then
5467 Data
.FXSubType
:= TRIGGER_EFFECT_LLIQUID
5468 else if vleObjectProperty
.Values
[MsgPropTrEffectSubtype
] = MsgPropTrEffectDliquid
then
5469 Data
.FXSubType
:= TRIGGER_EFFECT_DLIQUID
5470 else if vleObjectProperty
.Values
[MsgPropTrEffectSubtype
] = MsgPropTrEffectBlood
then
5471 Data
.FXSubType
:= TRIGGER_EFFECT_BLOOD
5472 else if vleObjectProperty
.Values
[MsgPropTrEffectSubtype
] = MsgPropTrEffectSpark
then
5473 Data
.FXSubType
:= TRIGGER_EFFECT_SPARK
5474 else if vleObjectProperty
.Values
[MsgPropTrEffectSubtype
] = MsgPropTrEffectBubble
then
5475 Data
.FXSubType
:= TRIGGER_EFFECT_BUBBLE
;
5478 Data
.FXType
:= TRIGGER_EFFECT_ANIMATION
;
5479 Data
.FXSubType
:= StrToEffect(vleObjectProperty
.Values
[MsgPropTrEffectSubtype
]);
5482 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrEffectColor
], 0), 0), $FFFFFF);
5483 Data
.FXColorR
:= a
and $FF;
5484 Data
.FXColorG
:= (a
shr 8) and $FF;
5485 Data
.FXColorB
:= (a
shr 16) and $FF;
5486 if NameToBool(vleObjectProperty
.Values
[MsgPropTrEffectCenter
]) then
5490 Data
.FXWait
:= Min(Max(
5491 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrExDelay
], 0), 0), 65535);
5492 Data
.FXVelX
:= Min(Max(
5493 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrEffectVelx
], 0), -128), 127);
5494 Data
.FXVelY
:= Min(Max(
5495 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrEffectVely
], 0), -128), 127);
5496 Data
.FXSpreadL
:= Min(Max(
5497 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrEffectSpl
], 0), 0), 255);
5498 Data
.FXSpreadR
:= Min(Max(
5499 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrEffectSpr
], 0), 0), 255);
5500 Data
.FXSpreadU
:= Min(Max(
5501 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrEffectSpu
], 0), 0), 255);
5502 Data
.FXSpreadD
:= Min(Max(
5503 StrToIntDef(vleObjectProperty
.Values
[MsgPropTrEffectSpd
], 0), 0), 255);
5512 vleObjectProperty
.Row
:= r
;
5513 vleObjectProperty
.Col
:= c
;
5516 procedure TMainForm
.bbRemoveTextureClick(Sender
: TObject
);
5520 i
:= lbTextureList
.ItemIndex
;
5524 if Application
.MessageBox(PChar(Format(MsgMsgDelTexturePromt
,
5525 [SelectedTexture()])),
5526 PChar(MsgMsgDelTexture
),
5527 MB_ICONQUESTION
or MB_YESNO
or
5528 MB_DEFBUTTON1
) <> idYes
then
5531 if gPanels
<> nil then
5532 for a
:= 0 to High(gPanels
) do
5533 if (gPanels
[a
].PanelType
<> 0) and
5534 (gPanels
[a
].TextureName
= SelectedTexture()) then
5536 ErrorMessageBox(MsgMsgDelTextureCant
);
5540 g_DeleteTexture(SelectedTexture());
5541 i
:= slInvalidTextures
.IndexOf(lbTextureList
.Items
[i
]);
5543 slInvalidTextures
.Delete(i
);
5544 if lbTextureList
.ItemIndex
> -1 then
5545 lbTextureList
.Items
.Delete(lbTextureList
.ItemIndex
)
5548 procedure TMainForm
.aNewMapExecute(Sender
: TObject
);
5550 if Application
.MessageBox(PChar(MsgMsgClearMapPromt
), PChar(MsgMsgClearMap
), MB_ICONQUESTION
or MB_YESNO
or MB_DEFBUTTON1
) = mrYes
then
5554 procedure TMainForm
.aUndoExecute(Sender
: TObject
);
5558 if UndoBuffer
= nil then
5560 if UndoBuffer
[High(UndoBuffer
)] = nil then
5563 for a
:= 0 to High(UndoBuffer
[High(UndoBuffer
)]) do
5564 with UndoBuffer
[High(UndoBuffer
)][a
] do
5572 UNDO_DELETE_ITEM
: AddItem(Item
);
5573 UNDO_DELETE_AREA
: AddArea(Area
);
5574 UNDO_DELETE_MONSTER
: AddMonster(Monster
);
5575 UNDO_DELETE_TRIGGER
: AddTrigger(Trigger
);
5576 UNDO_ADD_PANEL
: RemoveObject(AddID
, OBJECT_PANEL
);
5577 UNDO_ADD_ITEM
: RemoveObject(AddID
, OBJECT_ITEM
);
5578 UNDO_ADD_AREA
: RemoveObject(AddID
, OBJECT_AREA
);
5579 UNDO_ADD_MONSTER
: RemoveObject(AddID
, OBJECT_MONSTER
);
5580 UNDO_ADD_TRIGGER
: RemoveObject(AddID
, OBJECT_TRIGGER
);
5584 SetLength(UndoBuffer
, Length(UndoBuffer
)-1);
5586 RemoveSelectFromObjects();
5588 miUndo
.Enabled
:= UndoBuffer
<> nil;
5592 procedure TMainForm
.aCopyObjectExecute(Sender
: TObject
);
5595 CopyBuffer
: TCopyRecArray
;
5599 function CB_Compare(I1
, I2
: TCopyRec
): Integer;
5601 Result
:= Integer(I1
.ObjectType
) - Integer(I2
.ObjectType
);
5603 if Result
= 0 then // Одного типа
5604 Result
:= Integer(I1
.ID
) - Integer(I2
.ID
);
5607 procedure QuickSortCopyBuffer(L
, R
: Integer);
5615 P
:= CopyBuffer
[(L
+ R
) shr 1];
5618 while CB_Compare(CopyBuffer
[I
], P
) < 0 do
5620 while CB_Compare(CopyBuffer
[J
], P
) > 0 do
5626 CopyBuffer
[I
] := CopyBuffer
[J
];
5634 QuickSortCopyBuffer(L
, J
);
5641 if SelectedObjects
= nil then
5647 // Копируем объекты:
5648 for a
:= 0 to High(SelectedObjects
) do
5649 if SelectedObjects
[a
].Live
then
5650 with SelectedObjects
[a
] do
5652 SetLength(CopyBuffer
, Length(CopyBuffer
)+1);
5653 b
:= High(CopyBuffer
);
5654 CopyBuffer
[b
].ID
:= ID
;
5655 CopyBuffer
[b
].Panel
:= nil;
5660 CopyBuffer
[b
].ObjectType
:= OBJECT_PANEL
;
5661 New(CopyBuffer
[b
].Panel
);
5662 CopyBuffer
[b
].Panel
^ := gPanels
[ID
];
5667 CopyBuffer
[b
].ObjectType
:= OBJECT_ITEM
;
5668 CopyBuffer
[b
].Item
:= gItems
[ID
];
5673 CopyBuffer
[b
].ObjectType
:= OBJECT_MONSTER
;
5674 CopyBuffer
[b
].Monster
:= gMonsters
[ID
];
5679 CopyBuffer
[b
].ObjectType
:= OBJECT_AREA
;
5680 CopyBuffer
[b
].Area
:= gAreas
[ID
];
5685 CopyBuffer
[b
].ObjectType
:= OBJECT_TRIGGER
;
5686 CopyBuffer
[b
].Trigger
:= gTriggers
[ID
];
5691 // Сортировка по ID:
5692 if CopyBuffer
<> nil then
5694 QuickSortCopyBuffer(0, b
);
5697 // Постановка ссылок триггеров:
5698 for a
:= 0 to Length(CopyBuffer
)-1 do
5699 if CopyBuffer
[a
].ObjectType
= OBJECT_TRIGGER
then
5701 case CopyBuffer
[a
].Trigger
.TriggerType
of
5702 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
,
5703 TRIGGER_DOOR5
, TRIGGER_CLOSETRAP
, TRIGGER_TRAP
,
5704 TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
:
5705 if CopyBuffer
[a
].Trigger
.Data
.PanelID
<> -1 then
5709 for b
:= 0 to Length(CopyBuffer
)-1 do
5710 if (CopyBuffer
[b
].ObjectType
= OBJECT_PANEL
) and
5711 (Integer(CopyBuffer
[b
].ID
) = CopyBuffer
[a
].Trigger
.Data
.PanelID
) then
5713 CopyBuffer
[a
].Trigger
.Data
.PanelID
:= b
;
5718 // Этих панелей нет среди копируемых:
5720 CopyBuffer
[a
].Trigger
.Data
.PanelID
:= -1;
5723 TRIGGER_PRESS
, TRIGGER_ON
,
5724 TRIGGER_OFF
, TRIGGER_ONOFF
:
5725 if CopyBuffer
[a
].Trigger
.Data
.MonsterID
<> 0 then
5729 for b
:= 0 to Length(CopyBuffer
)-1 do
5730 if (CopyBuffer
[b
].ObjectType
= OBJECT_MONSTER
) and
5731 (Integer(CopyBuffer
[b
].ID
) = CopyBuffer
[a
].Trigger
.Data
.MonsterID
-1) then
5733 CopyBuffer
[a
].Trigger
.Data
.MonsterID
:= b
+1;
5738 // Этих монстров нет среди копируемых:
5740 CopyBuffer
[a
].Trigger
.Data
.MonsterID
:= 0;
5744 if CopyBuffer
[a
].Trigger
.Data
.ShotPanelID
<> -1 then
5748 for b
:= 0 to Length(CopyBuffer
)-1 do
5749 if (CopyBuffer
[b
].ObjectType
= OBJECT_PANEL
) and
5750 (Integer(CopyBuffer
[b
].ID
) = CopyBuffer
[a
].Trigger
.Data
.ShotPanelID
) then
5752 CopyBuffer
[a
].Trigger
.Data
.ShotPanelID
:= b
;
5757 // Этих панелей нет среди копируемых:
5759 CopyBuffer
[a
].Trigger
.Data
.ShotPanelID
:= -1;
5763 if CopyBuffer
[a
].Trigger
.TexturePanel
<> -1 then
5767 for b
:= 0 to Length(CopyBuffer
)-1 do
5768 if (CopyBuffer
[b
].ObjectType
= OBJECT_PANEL
) and
5769 (Integer(CopyBuffer
[b
].ID
) = CopyBuffer
[a
].Trigger
.TexturePanel
) then
5771 CopyBuffer
[a
].Trigger
.TexturePanel
:= b
;
5776 // Этих панелей нет среди копируемых:
5778 CopyBuffer
[a
].Trigger
.TexturePanel
:= -1;
5783 str
:= CopyBufferToString(CopyBuffer
);
5784 ClipBoard
.AsText
:= str
;
5786 for a
:= 0 to Length(CopyBuffer
)-1 do
5787 if (CopyBuffer
[a
].ObjectType
= OBJECT_PANEL
) and
5788 (CopyBuffer
[a
].Panel
<> nil) then
5789 Dispose(CopyBuffer
[a
].Panel
);
5794 procedure TMainForm
.aPasteObjectExecute(Sender
: TObject
);
5797 CopyBuffer
: TCopyRecArray
;
5799 swad
, ssec
, sres
: String;
5802 xadj
, yadj
: LongInt;
5807 pmin
.X
:= High(pmin
.X
);
5808 pmin
.Y
:= High(pmin
.Y
);
5810 StringToCopyBuffer(ClipBoard
.AsText
, CopyBuffer
, pmin
);
5811 if CopyBuffer
= nil then
5814 rel
:= not(ssShift
in GetKeyShiftState());
5815 h
:= High(CopyBuffer
);
5816 RemoveSelectFromObjects();
5820 xadj
:= Floor((-pmin
.X
- MapOffset
.X
+ 32) / DotStep
) * DotStep
;
5821 yadj
:= Floor((-pmin
.Y
- MapOffset
.Y
+ 32) / DotStep
) * DotStep
;
5830 with CopyBuffer
[a
] do
5834 if Panel
<> nil then
5842 Panel
^.TextureID
:= TEXTURE_SPECIAL_NONE
;
5843 Panel
^.TextureWidth
:= 1;
5844 Panel
^.TextureHeight
:= 1;
5846 if (Panel
^.PanelType
= PANEL_LIFTUP
) or
5847 (Panel
^.PanelType
= PANEL_LIFTDOWN
) or
5848 (Panel
^.PanelType
= PANEL_LIFTLEFT
) or
5849 (Panel
^.PanelType
= PANEL_LIFTRIGHT
) or
5850 (Panel
^.PanelType
= PANEL_BLOCKMON
) or
5851 (Panel
^.TextureName
= '') then
5852 begin // Нет или не может быть текстуры:
5854 else // Есть текстура:
5856 // Обычная текстура:
5857 if not IsSpecialTexture(Panel
^.TextureName
) then
5859 res
:= g_GetTexture(Panel
^.TextureName
, Panel
^.TextureID
);
5863 g_ProcessResourceStr(Panel
^.TextureName
, swad
, ssec
, sres
);
5864 AddTexture(swad
, ssec
, sres
, True);
5865 res
:= g_GetTexture(Panel
^.TextureName
, Panel
^.TextureID
);
5869 g_GetTextureSizeByName(Panel
^.TextureName
,
5870 Panel
^.TextureWidth
, Panel
^.TextureHeight
)
5872 if g_GetTexture('NOTEXTURE', NoTextureID
) then
5874 Panel
^.TextureID
:= TEXTURE_SPECIAL_NOTEXTURE
;
5875 g_GetTextureSizeByID(NoTextureID
, Panel
^.TextureWidth
, Panel
^.TextureHeight
);
5878 else // Спец.текстура:
5880 Panel
^.TextureID
:= SpecialTextureID(Panel
^.TextureName
);
5881 with MainForm
.lbTextureList
.Items
do
5882 if IndexOf(Panel
^.TextureName
) = -1 then
5883 Add(Panel
^.TextureName
);
5887 ID
:= AddPanel(Panel
^);
5889 Undo_Add(OBJECT_PANEL
, ID
, a
> 0);
5890 SelectObject(OBJECT_PANEL
, ID
, True);
5901 ID
:= AddItem(Item
);
5902 Undo_Add(OBJECT_ITEM
, ID
, a
> 0);
5903 SelectObject(OBJECT_ITEM
, ID
, True);
5914 ID
:= AddMonster(Monster
);
5915 Undo_Add(OBJECT_MONSTER
, ID
, a
> 0);
5916 SelectObject(OBJECT_MONSTER
, ID
, True);
5927 ID
:= AddArea(Area
);
5928 Undo_Add(OBJECT_AREA
, ID
, a
> 0);
5929 SelectObject(OBJECT_AREA
, ID
, True);
5943 Data
.TargetPoint
.X
+= xadj
;
5944 Data
.TargetPoint
.Y
+= yadj
;
5946 TRIGGER_PRESS
, TRIGGER_ON
, TRIGGER_OFF
, TRIGGER_ONOFF
:
5951 TRIGGER_SPAWNMONSTER
:
5953 Data
.MonPos
.X
+= xadj
;
5954 Data
.MonPos
.Y
+= yadj
;
5958 Data
.ItemPos
.X
+= xadj
;
5959 Data
.ItemPos
.Y
+= yadj
;
5963 Data
.ShotPos
.X
+= xadj
;
5964 Data
.ShotPos
.Y
+= yadj
;
5969 ID
:= AddTrigger(Trigger
);
5970 Undo_Add(OBJECT_TRIGGER
, ID
, a
> 0);
5971 SelectObject(OBJECT_TRIGGER
, ID
, True);
5976 // Переставляем ссылки триггеров:
5977 for a
:= 0 to High(CopyBuffer
) do
5978 if CopyBuffer
[a
].ObjectType
= OBJECT_TRIGGER
then
5980 case CopyBuffer
[a
].Trigger
.TriggerType
of
5981 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
,
5982 TRIGGER_DOOR5
, TRIGGER_CLOSETRAP
, TRIGGER_TRAP
,
5983 TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
:
5984 if CopyBuffer
[a
].Trigger
.Data
.PanelID
<> -1 then
5985 gTriggers
[CopyBuffer
[a
].ID
].Data
.PanelID
:=
5986 CopyBuffer
[CopyBuffer
[a
].Trigger
.Data
.PanelID
].ID
;
5988 TRIGGER_PRESS
, TRIGGER_ON
,
5989 TRIGGER_OFF
, TRIGGER_ONOFF
:
5990 if CopyBuffer
[a
].Trigger
.Data
.MonsterID
<> 0 then
5991 gTriggers
[CopyBuffer
[a
].ID
].Data
.MonsterID
:=
5992 CopyBuffer
[CopyBuffer
[a
].Trigger
.Data
.MonsterID
-1].ID
+1;
5995 if CopyBuffer
[a
].Trigger
.Data
.ShotPanelID
<> -1 then
5996 gTriggers
[CopyBuffer
[a
].ID
].Data
.ShotPanelID
:=
5997 CopyBuffer
[CopyBuffer
[a
].Trigger
.Data
.ShotPanelID
].ID
;
6000 if CopyBuffer
[a
].Trigger
.TexturePanel
<> -1 then
6001 gTriggers
[CopyBuffer
[a
].ID
].TexturePanel
:=
6002 CopyBuffer
[CopyBuffer
[a
].Trigger
.TexturePanel
].ID
;
6011 procedure TMainForm
.aCutObjectExecute(Sender
: TObject
);
6014 DeleteSelectedObjects();
6017 procedure TMainForm
.vleObjectPropertyEditButtonClick(Sender
: TObject
);
6019 Key
, FileName
: String;
6022 Key
:= vleObjectProperty
.Keys
[vleObjectProperty
.Row
];
6024 if Key
= MsgPropPanelType
then
6026 with ChooseTypeForm
, vleObjectProperty
do
6027 begin // Выбор типа панели:
6028 Caption
:= MsgPropPanelType
;
6029 lbTypeSelect
.Items
.Clear();
6031 for b
:= 0 to High(PANELNAMES
) do
6033 lbTypeSelect
.Items
.Add(PANELNAMES
[b
]);
6034 if Values
[Key
] = PANELNAMES
[b
] then
6035 lbTypeSelect
.ItemIndex
:= b
;
6038 if ShowModal() = mrOK
then
6040 b
:= lbTypeSelect
.ItemIndex
;
6041 Values
[Key
] := PANELNAMES
[b
];
6042 vleObjectPropertyApply(Sender
);
6046 else if Key
= MsgPropTrTeleportTo
then
6047 SelectFlag
:= SELECTFLAG_TELEPORT
6048 else if Key
= MsgPropTrSpawnTo
then
6049 SelectFlag
:= SELECTFLAG_SPAWNPOINT
6050 else if (Key
= MsgPropTrDoorPanel
) or
6051 (Key
= MsgPropTrTrapPanel
) then
6052 SelectFlag
:= SELECTFLAG_DOOR
6053 else if Key
= MsgPropTrTexturePanel
then
6055 DrawPressRect
:= False;
6056 SelectFlag
:= SELECTFLAG_TEXTURE
;
6058 else if Key
= MsgPropTrShotPanel
then
6059 SelectFlag
:= SELECTFLAG_SHOTPANEL
6060 else if Key
= MsgPropTrLiftPanel
then
6061 SelectFlag
:= SELECTFLAG_LIFT
6062 else if key
= MsgPropTrExMonster
then
6063 SelectFlag
:= SELECTFLAG_MONSTER
6064 else if Key
= MsgPropTrExArea
then
6066 SelectFlag
:= SELECTFLAG_NONE
;
6067 DrawPressRect
:= True;
6069 else if Key
= MsgPropTrNextMap
then
6070 begin // Выбор следующей карты:
6071 g_ProcessResourceStr(OpenedMap
, @FileName
, nil, nil);
6072 SelectMapForm
.Caption
:= MsgCapSelect
;
6073 SelectMapForm
.GetMaps(FileName
);
6075 if SelectMapForm
.ShowModal() = mrOK
then
6077 vleObjectProperty
.Values
[Key
] := SelectMapForm
.lbMapList
.Items
[SelectMapForm
.lbMapList
.ItemIndex
];
6078 vleObjectPropertyApply(Sender
);
6081 else if (Key
= MsgPropTrSoundName
) or
6082 (Key
= MsgPropTrMusicName
) then
6083 begin // Выбор файла звука/музыки:
6084 AddSoundForm
.OKFunction
:= nil;
6085 AddSoundForm
.lbResourcesList
.MultiSelect
:= False;
6086 AddSoundForm
.SetResource
:= vleObjectProperty
.Values
[Key
];
6088 if (AddSoundForm
.ShowModal() = mrOk
) then
6090 vleObjectProperty
.Values
[Key
] := AddSoundForm
.ResourceName
;
6091 vleObjectPropertyApply(Sender
);
6094 else if Key
= MsgPropTrActivation
then
6095 with ActivationTypeForm
, vleObjectProperty
do
6096 begin // Выбор типов активации:
6097 cbPlayerCollide
.Checked
:= Pos('PC', Values
[Key
]) > 0;
6098 cbMonsterCollide
.Checked
:= Pos('MC', Values
[Key
]) > 0;
6099 cbPlayerPress
.Checked
:= Pos('PP', Values
[Key
]) > 0;
6100 cbMonsterPress
.Checked
:= Pos('MP', Values
[Key
]) > 0;
6101 cbShot
.Checked
:= Pos('SH', Values
[Key
]) > 0;
6102 cbNoMonster
.Checked
:= Pos('NM', Values
[Key
]) > 0;
6104 if ShowModal() = mrOK
then
6107 if cbPlayerCollide
.Checked
then
6108 b
:= ACTIVATE_PLAYERCOLLIDE
;
6109 if cbMonsterCollide
.Checked
then
6110 b
:= b
or ACTIVATE_MONSTERCOLLIDE
;
6111 if cbPlayerPress
.Checked
then
6112 b
:= b
or ACTIVATE_PLAYERPRESS
;
6113 if cbMonsterPress
.Checked
then
6114 b
:= b
or ACTIVATE_MONSTERPRESS
;
6115 if cbShot
.Checked
then
6116 b
:= b
or ACTIVATE_SHOT
;
6117 if cbNoMonster
.Checked
then
6118 b
:= b
or ACTIVATE_NOMONSTER
;
6120 Values
[Key
] := ActivateToStr(b
);
6121 vleObjectPropertyApply(Sender
);
6124 else if Key
= MsgPropTrKeys
then
6125 with KeysForm
, vleObjectProperty
do
6126 begin // Выбор необходимых ключей:
6127 cbRedKey
.Checked
:= Pos('RK', Values
[Key
]) > 0;
6128 cbGreenKey
.Checked
:= Pos('GK', Values
[Key
]) > 0;
6129 cbBlueKey
.Checked
:= Pos('BK', Values
[Key
]) > 0;
6130 cbRedTeam
.Checked
:= Pos('RT', Values
[Key
]) > 0;
6131 cbBlueTeam
.Checked
:= Pos('BT', Values
[Key
]) > 0;
6133 if ShowModal() = mrOK
then
6136 if cbRedKey
.Checked
then
6138 if cbGreenKey
.Checked
then
6139 b
:= b
or KEY_GREEN
;
6140 if cbBlueKey
.Checked
then
6142 if cbRedTeam
.Checked
then
6143 b
:= b
or KEY_REDTEAM
;
6144 if cbBlueTeam
.Checked
then
6145 b
:= b
or KEY_BLUETEAM
;
6147 Values
[Key
] := KeyToStr(b
);
6148 vleObjectPropertyApply(Sender
);
6151 else if Key
= MsgPropTrFxType
then
6152 with ChooseTypeForm
, vleObjectProperty
do
6153 begin // Выбор типа эффекта:
6154 Caption
:= MsgCapFxType
;
6155 lbTypeSelect
.Items
.Clear();
6157 for b
:= EFFECT_NONE
to EFFECT_FIRE
do
6158 lbTypeSelect
.Items
.Add(EffectToStr(b
));
6160 lbTypeSelect
.ItemIndex
:= StrToEffect(Values
[Key
]);
6162 if ShowModal() = mrOK
then
6164 b
:= lbTypeSelect
.ItemIndex
;
6165 Values
[Key
] := EffectToStr(b
);
6166 vleObjectPropertyApply(Sender
);
6169 else if Key
= MsgPropTrMonsterType
then
6170 with ChooseTypeForm
, vleObjectProperty
do
6171 begin // Выбор типа монстра:
6172 Caption
:= MsgCapMonsterType
;
6173 lbTypeSelect
.Items
.Clear();
6175 for b
:= MONSTER_DEMON
to MONSTER_MAN
do
6176 lbTypeSelect
.Items
.Add(MonsterToStr(b
));
6178 lbTypeSelect
.ItemIndex
:= StrToMonster(Values
[Key
]) - MONSTER_DEMON
;
6180 if ShowModal() = mrOK
then
6182 b
:= lbTypeSelect
.ItemIndex
+ MONSTER_DEMON
;
6183 Values
[Key
] := MonsterToStr(b
);
6184 vleObjectPropertyApply(Sender
);
6187 else if Key
= MsgPropTrItemType
then
6188 with ChooseTypeForm
, vleObjectProperty
do
6189 begin // Выбор типа предмета:
6190 Caption
:= MsgCapItemType
;
6191 lbTypeSelect
.Items
.Clear();
6193 for b
:= ITEM_MEDKIT_SMALL
to ITEM_KEY_BLUE
do
6194 lbTypeSelect
.Items
.Add(ItemToStr(b
));
6195 lbTypeSelect
.Items
.Add(ItemToStr(ITEM_BOTTLE
));
6196 lbTypeSelect
.Items
.Add(ItemToStr(ITEM_HELMET
));
6197 lbTypeSelect
.Items
.Add(ItemToStr(ITEM_JETPACK
));
6198 lbTypeSelect
.Items
.Add(ItemToStr(ITEM_INVIS
));
6199 lbTypeSelect
.Items
.Add(ItemToStr(ITEM_WEAPON_FLAMETHROWER
));
6200 lbTypeSelect
.Items
.Add(ItemToStr(ITEM_AMMO_FUELCAN
));
6202 b
:= StrToItem(Values
[Key
]);
6203 if b
>= ITEM_BOTTLE
then
6205 lbTypeSelect
.ItemIndex
:= b
- ITEM_MEDKIT_SMALL
;
6207 if ShowModal() = mrOK
then
6209 b
:= lbTypeSelect
.ItemIndex
+ ITEM_MEDKIT_SMALL
;
6210 if b
>= ITEM_WEAPON_KASTET
then
6212 Values
[Key
] := ItemToStr(b
);
6213 vleObjectPropertyApply(Sender
);
6216 else if Key
= MsgPropTrShotType
then
6217 with ChooseTypeForm
, vleObjectProperty
do
6218 begin // Выбор типа предмета:
6219 Caption
:= MsgPropTrShotType
;
6220 lbTypeSelect
.Items
.Clear();
6222 for b
:= TRIGGER_SHOT_PISTOL
to TRIGGER_SHOT_MAX
do
6223 lbTypeSelect
.Items
.Add(ShotToStr(b
));
6225 lbTypeSelect
.ItemIndex
:= StrToShot(Values
[Key
]);
6227 if ShowModal() = mrOK
then
6229 b
:= lbTypeSelect
.ItemIndex
;
6230 Values
[Key
] := ShotToStr(b
);
6231 vleObjectPropertyApply(Sender
);
6234 else if Key
= MsgPropTrEffectType
then
6235 with ChooseTypeForm
, vleObjectProperty
do
6236 begin // Выбор типа эффекта:
6237 Caption
:= MsgCapFxType
;
6238 lbTypeSelect
.Items
.Clear();
6240 lbTypeSelect
.Items
.Add(MsgPropTrEffectParticle
);
6241 lbTypeSelect
.Items
.Add(MsgPropTrEffectAnimation
);
6242 if Values
[Key
] = MsgPropTrEffectAnimation
then
6243 lbTypeSelect
.ItemIndex
:= 1
6245 lbTypeSelect
.ItemIndex
:= 0;
6247 if ShowModal() = mrOK
then
6249 b
:= lbTypeSelect
.ItemIndex
;
6251 Values
[Key
] := MsgPropTrEffectParticle
6253 Values
[Key
] := MsgPropTrEffectAnimation
;
6254 vleObjectPropertyApply(Sender
);
6257 else if Key
= MsgPropTrEffectSubtype
then
6258 with ChooseTypeForm
, vleObjectProperty
do
6259 begin // Выбор подтипа эффекта:
6260 Caption
:= MsgCapFxType
;
6261 lbTypeSelect
.Items
.Clear();
6263 if Values
[MsgPropTrEffectType
] = MsgPropTrEffectAnimation
then
6265 for b
:= EFFECT_TELEPORT
to EFFECT_FIRE
do
6266 lbTypeSelect
.Items
.Add(EffectToStr(b
));
6268 lbTypeSelect
.ItemIndex
:= StrToEffect(Values
[Key
]) - 1;
6271 lbTypeSelect
.Items
.Add(MsgPropTrEffectSliquid
);
6272 lbTypeSelect
.Items
.Add(MsgPropTrEffectLliquid
);
6273 lbTypeSelect
.Items
.Add(MsgPropTrEffectDliquid
);
6274 lbTypeSelect
.Items
.Add(MsgPropTrEffectBlood
);
6275 lbTypeSelect
.Items
.Add(MsgPropTrEffectSpark
);
6276 lbTypeSelect
.Items
.Add(MsgPropTrEffectBubble
);
6277 lbTypeSelect
.ItemIndex
:= TRIGGER_EFFECT_SLIQUID
;
6278 if Values
[Key
] = MsgPropTrEffectLliquid
then
6279 lbTypeSelect
.ItemIndex
:= TRIGGER_EFFECT_LLIQUID
;
6280 if Values
[Key
] = MsgPropTrEffectDliquid
then
6281 lbTypeSelect
.ItemIndex
:= TRIGGER_EFFECT_DLIQUID
;
6282 if Values
[Key
] = MsgPropTrEffectBlood
then
6283 lbTypeSelect
.ItemIndex
:= TRIGGER_EFFECT_BLOOD
;
6284 if Values
[Key
] = MsgPropTrEffectSpark
then
6285 lbTypeSelect
.ItemIndex
:= TRIGGER_EFFECT_SPARK
;
6286 if Values
[Key
] = MsgPropTrEffectBubble
then
6287 lbTypeSelect
.ItemIndex
:= TRIGGER_EFFECT_BUBBLE
;
6290 if ShowModal() = mrOK
then
6292 b
:= lbTypeSelect
.ItemIndex
;
6294 if Values
[MsgPropTrEffectType
] = MsgPropTrEffectAnimation
then
6295 Values
[Key
] := EffectToStr(b
+ 1)
6297 Values
[Key
] := MsgPropTrEffectSliquid
;
6298 if b
= TRIGGER_EFFECT_LLIQUID
then
6299 Values
[Key
] := MsgPropTrEffectLliquid
;
6300 if b
= TRIGGER_EFFECT_DLIQUID
then
6301 Values
[Key
] := MsgPropTrEffectDliquid
;
6302 if b
= TRIGGER_EFFECT_BLOOD
then
6303 Values
[Key
] := MsgPropTrEffectBlood
;
6304 if b
= TRIGGER_EFFECT_SPARK
then
6305 Values
[Key
] := MsgPropTrEffectSpark
;
6306 if b
= TRIGGER_EFFECT_BUBBLE
then
6307 Values
[Key
] := MsgPropTrEffectBubble
;
6310 vleObjectPropertyApply(Sender
);
6313 else if Key
= MsgPropTrEffectColor
then
6314 with vleObjectProperty
do
6315 begin // Выбор цвета эффекта:
6316 ColorDialog
.Color
:= StrToIntDef(Values
[Key
], 0);
6317 if ColorDialog
.Execute
then
6319 Values
[Key
] := IntToStr(ColorDialog
.Color
);
6320 vleObjectPropertyApply(Sender
);
6323 else if Key
= MsgPropPanelTex
then
6324 begin // Смена текстуры:
6325 vleObjectProperty
.Values
[Key
] := SelectedTexture();
6326 vleObjectPropertyApply(Sender
);
6330 procedure TMainForm
.vleObjectPropertyApply(Sender
: TObject
);
6332 // hack to prevent empty ID in list
6333 RenderPanel
.SetFocus();
6334 bApplyProperty
.Click();
6335 vleObjectProperty
.SetFocus();
6338 procedure TMainForm
.aSaveMapExecute(Sender
: TObject
);
6340 FileName
, Section
, Res
: String;
6342 if OpenedMap
= '' then
6344 aSaveMapAsExecute(nil);
6348 g_ProcessResourceStr(OpenedMap
, FileName
, Section
, Res
);
6350 SaveMap(FileName
+':\'+Res
);
6353 procedure TMainForm
.aOpenMapExecute(Sender
: TObject
);
6355 OpenDialog
.Filter
:= MsgFileFilterAll
;
6357 if OpenDialog
.Execute() then
6359 OpenMapFile(OpenDialog
.FileName
);
6360 OpenDialog
.InitialDir
:= ExtractFileDir(OpenDialog
.FileName
);
6364 procedure TMainForm
.OpenMapFile(FileName
: String);
6366 if (Pos('.ini', LowerCase(ExtractFileName(FileName
))) > 0) then
6370 pLoadProgress
.Left
:= (RenderPanel
.Width
div 2)-(pLoadProgress
.Width
div 2);
6371 pLoadProgress
.Top
:= (RenderPanel
.Height
div 2)-(pLoadProgress
.Height
div 2);
6372 pLoadProgress
.Show();
6377 LoadMapOld(FileName
);
6379 MainForm
.Caption
:= Format('%s - %s', [FormCaption
, ExtractFileName(FileName
)]);
6381 pLoadProgress
.Hide();
6382 MainForm
.FormResize(Self
);
6384 else // Карты из WAD:
6386 OpenMap(FileName
, '');
6390 procedure TMainForm
.FormActivate(Sender
: TObject
);
6392 MainForm
.ActiveControl
:= RenderPanel
;
6395 procedure TMainForm
.aDeleteMap(Sender
: TObject
);
6401 OpenDialog
.Filter
:= MsgFileFilterWad
;
6403 if not OpenDialog
.Execute() then
6406 FileName
:= OpenDialog
.FileName
;
6407 SelectMapForm
.Caption
:= MsgCapRemove
;
6408 SelectMapForm
.lbMapList
.Items
.Clear();
6409 SelectMapForm
.GetMaps(FileName
);
6411 if SelectMapForm
.ShowModal() <> mrOK
then
6414 MapName
:= SelectMapForm
.lbMapList
.Items
[SelectMapForm
.lbMapList
.ItemIndex
];
6415 if Application
.MessageBox(PChar(Format(MsgMsgDeleteMapPromt
, [MapName
, OpenDialog
.FileName
])), PChar(MsgMsgDeleteMap
), MB_ICONQUESTION
or MB_YESNO
or MB_DEFBUTTON2
) <> mrYes
then
6418 g_DeleteResource(FileName
, '', MapName
, res
);
6421 Application
.MessageBox(PChar('Cant delete map res=' + IntToStr(res
)), PChar('Map not deleted!'), MB_ICONINFORMATION
or MB_OK
or MB_DEFBUTTON1
);
6425 Application
.MessageBox(
6426 PChar(Format(MsgMsgMapDeletedPromt
, [MapName
])),
6427 PChar(MsgMsgMapDeleted
),
6428 MB_ICONINFORMATION
or MB_OK
or MB_DEFBUTTON1
6431 // Удалили текущую карту - сохранять по старому ее нельзя:
6432 if OpenedMap
= (FileName
+ ':\' + MapName
) then
6436 MainForm
.Caption
:= FormCaption
6440 procedure TMainForm
.vleObjectPropertyKeyDown(Sender
: TObject
;
6441 var Key
: Word; Shift
: TShiftState
);
6443 if Key
= VK_RETURN
then
6444 vleObjectPropertyApply(Sender
);
6447 procedure MovePanel(var ID
: DWORD
; MoveType
: Byte);
6452 if (ID
= 0) and (MoveType
= 0) then
6454 if (ID
= DWORD(High(gPanels
))) and (MoveType
<> 0) then
6456 if (ID
> DWORD(High(gPanels
))) then
6461 if MoveType
= 0 then // to Back
6463 if gTriggers
<> nil then
6464 for a
:= 0 to High(gTriggers
) do
6465 with gTriggers
[a
] do
6467 if TriggerType
= TRIGGER_NONE
then
6470 if TexturePanel
= _id
then
6473 if (TexturePanel
>= 0) and (TexturePanel
< _id
) then
6477 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
,
6478 TRIGGER_DOOR5
, TRIGGER_CLOSETRAP
, TRIGGER_TRAP
,
6479 TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
:
6480 if Data
.PanelID
= _id
then
6483 if (Data
.PanelID
>= 0) and (Data
.PanelID
< _id
) then
6487 if Data
.ShotPanelID
= _id
then
6488 Data
.ShotPanelID
:= 0
6490 if (Data
.ShotPanelID
>= 0) and (Data
.ShotPanelID
< _id
) then
6491 Inc(Data
.ShotPanelID
);
6495 tmp
:= gPanels
[_id
];
6497 for a
:= _id
downto 1 do
6498 gPanels
[a
] := gPanels
[a
-1];
6506 if gTriggers
<> nil then
6507 for a
:= 0 to High(gTriggers
) do
6508 with gTriggers
[a
] do
6510 if TriggerType
= TRIGGER_NONE
then
6513 if TexturePanel
= _id
then
6514 TexturePanel
:= High(gPanels
)
6516 if TexturePanel
> _id
then
6520 TRIGGER_OPENDOOR
, TRIGGER_CLOSEDOOR
, TRIGGER_DOOR
,
6521 TRIGGER_DOOR5
, TRIGGER_CLOSETRAP
, TRIGGER_TRAP
,
6522 TRIGGER_LIFTUP
, TRIGGER_LIFTDOWN
, TRIGGER_LIFT
:
6523 if Data
.PanelID
= _id
then
6524 Data
.PanelID
:= High(gPanels
)
6526 if Data
.PanelID
> _id
then
6530 if Data
.ShotPanelID
= _id
then
6531 Data
.ShotPanelID
:= High(gPanels
)
6533 if Data
.ShotPanelID
> _id
then
6534 Dec(Data
.ShotPanelID
);
6538 tmp
:= gPanels
[_id
];
6540 for a
:= _id
to High(gPanels
)-1 do
6541 gPanels
[a
] := gPanels
[a
+1];
6543 gPanels
[High(gPanels
)] := tmp
;
6545 ID
:= High(gPanels
);
6549 procedure TMainForm
.aMoveToBack(Sender
: TObject
);
6553 if SelectedObjects
= nil then
6556 for a
:= 0 to High(SelectedObjects
) do
6557 with SelectedObjects
[a
] do
6558 if Live
and (ObjectType
= OBJECT_PANEL
) then
6560 SelectedObjects
[0] := SelectedObjects
[a
];
6561 SetLength(SelectedObjects
, 1);
6568 procedure TMainForm
.aMoveToFore(Sender
: TObject
);
6572 if SelectedObjects
= nil then
6575 for a
:= 0 to High(SelectedObjects
) do
6576 with SelectedObjects
[a
] do
6577 if Live
and (ObjectType
= OBJECT_PANEL
) then
6579 SelectedObjects
[0] := SelectedObjects
[a
];
6580 SetLength(SelectedObjects
, 1);
6587 procedure TMainForm
.aSaveMapAsExecute(Sender
: TObject
);
6591 SaveDialog
.Filter
:= MsgFileFilterWad
;
6593 if not SaveDialog
.Execute() then
6596 SaveMapForm
.GetMaps(SaveDialog
.FileName
, True);
6598 if SaveMapForm
.ShowModal() <> mrOK
then
6601 SaveDialog
.InitialDir
:= ExtractFileDir(SaveDialog
.FileName
);
6602 OpenedMap
:= SaveDialog
.FileName
+':\'+SaveMapForm
.eMapName
.Text;
6603 OpenedWAD
:= SaveDialog
.FileName
;
6605 idx
:= RecentFiles
.IndexOf(OpenedMap
);
6606 // Такая карта уже недавно открывалась:
6608 RecentFiles
.Delete(idx
);
6609 RecentFiles
.Insert(0, OpenedMap
);
6614 gMapInfo
.FileName
:= SaveDialog
.FileName
;
6615 gMapInfo
.MapName
:= SaveMapForm
.eMapName
.Text;
6616 UpdateCaption(gMapInfo
.Name
, ExtractFileName(gMapInfo
.FileName
), gMapInfo
.MapName
);
6619 procedure TMainForm
.aSelectAllExecute(Sender
: TObject
);
6623 RemoveSelectFromObjects();
6625 case pcObjects
.ActivePageIndex
+1 of
6627 if gPanels
<> nil then
6628 for a
:= 0 to High(gPanels
) do
6629 if gPanels
[a
].PanelType
<> PANEL_NONE
then
6630 SelectObject(OBJECT_PANEL
, a
, True);
6632 if gItems
<> nil then
6633 for a
:= 0 to High(gItems
) do
6634 if gItems
[a
].ItemType
<> ITEM_NONE
then
6635 SelectObject(OBJECT_ITEM
, a
, True);
6637 if gMonsters
<> nil then
6638 for a
:= 0 to High(gMonsters
) do
6639 if gMonsters
[a
].MonsterType
<> MONSTER_NONE
then
6640 SelectObject(OBJECT_MONSTER
, a
, True);
6642 if gAreas
<> nil then
6643 for a
:= 0 to High(gAreas
) do
6644 if gAreas
[a
].AreaType
<> AREA_NONE
then
6645 SelectObject(OBJECT_AREA
, a
, True);
6647 if gTriggers
<> nil then
6648 for a
:= 0 to High(gTriggers
) do
6649 if gTriggers
[a
].TriggerType
<> TRIGGER_NONE
then
6650 SelectObject(OBJECT_TRIGGER
, a
, True);
6653 RecountSelectedObjects();
6656 procedure TMainForm
.tbGridOnClick(Sender
: TObject
);
6658 DotEnable
:= not DotEnable
;
6659 (Sender
as TToolButton
).Down
:= DotEnable
;
6662 procedure TMainForm
.OnIdle(Sender
: TObject
; var Done
: Boolean);
6665 // FIXME: this is a shitty hack
6666 if not gDataLoaded
then
6668 e_WriteLog('Init OpenGL', MSG_NOTIFY
);
6670 e_WriteLog('Loading data', MSG_NOTIFY
);
6671 LoadStdFont('STDTXT', 'STDFONT', gEditorFont
);
6672 e_WriteLog('Loading more data', MSG_NOTIFY
);
6674 e_WriteLog('Loading even more data', MSG_NOTIFY
);
6675 gDataLoaded
:= True;
6676 MainForm
.FormResize(nil);
6679 if StartMap
<> '' then
6687 procedure TMainForm
.miMapPreviewClick(Sender
: TObject
);
6689 if PreviewMode
= 2 then
6692 if PreviewMode
= 0 then
6694 Splitter2
.Visible
:= False;
6695 Splitter1
.Visible
:= False;
6696 StatusBar
.Visible
:= False;
6697 PanelObjs
.Visible
:= False;
6698 PanelProps
.Visible
:= False;
6699 MainToolBar
.Visible
:= False;
6700 sbHorizontal
.Visible
:= False;
6701 sbVertical
.Visible
:= False;
6705 StatusBar
.Visible
:= True;
6706 PanelObjs
.Visible
:= True;
6707 PanelProps
.Visible
:= True;
6708 Splitter2
.Visible
:= True;
6709 Splitter1
.Visible
:= True;
6710 MainToolBar
.Visible
:= True;
6711 sbHorizontal
.Visible
:= True;
6712 sbVertical
.Visible
:= True;
6715 PreviewMode
:= PreviewMode
xor 1;
6716 (Sender
as TMenuItem
).Checked
:= PreviewMode
> 0;
6721 procedure TMainForm
.miLayer1Click(Sender
: TObject
);
6723 SwitchLayer(LAYER_BACK
);
6726 procedure TMainForm
.miLayer2Click(Sender
: TObject
);
6728 SwitchLayer(LAYER_WALLS
);
6731 procedure TMainForm
.miLayer3Click(Sender
: TObject
);
6733 SwitchLayer(LAYER_FOREGROUND
);
6736 procedure TMainForm
.miLayer4Click(Sender
: TObject
);
6738 SwitchLayer(LAYER_STEPS
);
6741 procedure TMainForm
.miLayer5Click(Sender
: TObject
);
6743 SwitchLayer(LAYER_WATER
);
6746 procedure TMainForm
.miLayer6Click(Sender
: TObject
);
6748 SwitchLayer(LAYER_ITEMS
);
6751 procedure TMainForm
.miLayer7Click(Sender
: TObject
);
6753 SwitchLayer(LAYER_MONSTERS
);
6756 procedure TMainForm
.miLayer8Click(Sender
: TObject
);
6758 SwitchLayer(LAYER_AREAS
);
6761 procedure TMainForm
.miLayer9Click(Sender
: TObject
);
6763 SwitchLayer(LAYER_TRIGGERS
);
6766 procedure TMainForm
.tbShowClick(Sender
: TObject
);
6772 for a
:= 0 to High(LayerEnabled
) do
6773 b
:= b
and LayerEnabled
[a
];
6777 ShowLayer(LAYER_BACK
, b
);
6778 ShowLayer(LAYER_WALLS
, b
);
6779 ShowLayer(LAYER_FOREGROUND
, b
);
6780 ShowLayer(LAYER_STEPS
, b
);
6781 ShowLayer(LAYER_WATER
, b
);
6782 ShowLayer(LAYER_ITEMS
, b
);
6783 ShowLayer(LAYER_MONSTERS
, b
);
6784 ShowLayer(LAYER_AREAS
, b
);
6785 ShowLayer(LAYER_TRIGGERS
, b
);
6788 procedure TMainForm
.miMiniMapClick(Sender
: TObject
);
6793 procedure TMainForm
.miSwitchGridClick(Sender
: TObject
);
6795 if DotStep
= DotStepOne
then
6796 DotStep
:= DotStepTwo
6798 DotStep
:= DotStepOne
;
6800 MousePos
.X
:= (MousePos
.X
div DotStep
) * DotStep
;
6801 MousePos
.Y
:= (MousePos
.Y
div DotStep
) * DotStep
;
6804 procedure TMainForm
.miShowEdgesClick(Sender
: TObject
);
6809 procedure TMainForm
.miSnapToGridClick(Sender
: TObject
);
6811 SnapToGrid
:= not SnapToGrid
;
6813 MousePos
.X
:= (MousePos
.X
div DotStep
) * DotStep
;
6814 MousePos
.Y
:= (MousePos
.Y
div DotStep
) * DotStep
;
6816 miSnapToGrid
.Checked
:= SnapToGrid
;
6819 procedure TMainForm
.minexttabClick(Sender
: TObject
);
6821 if pcObjects
.ActivePageIndex
< pcObjects
.PageCount
-1 then
6822 pcObjects
.ActivePageIndex
:= pcObjects
.ActivePageIndex
+1
6824 pcObjects
.ActivePageIndex
:= 0;
6827 procedure TMainForm
.miSaveMiniMapClick(Sender
: TObject
);
6829 SaveMiniMapForm
.ShowModal();
6832 procedure TMainForm
.bClearTextureClick(Sender
: TObject
);
6834 lbTextureList
.ItemIndex
:= -1;
6835 lTextureWidth
.Caption
:= '';
6836 lTextureHeight
.Caption
:= '';
6839 procedure TMainForm
.miPackMapClick(Sender
: TObject
);
6841 PackMapForm
.ShowModal();
6844 type SSArray
= array of String;
6846 function ParseString (Str
: AnsiString): SSArray
;
6847 function GetStr (var Str
: AnsiString): AnsiString;
6851 if Str
[1] = '"' then
6852 for b
:= 1 to Length(Str
) do
6853 if (b
= Length(Str
)) or (Str
[b
+ 1] = '"') then
6855 Result
:= Copy(Str
, 2, b
- 1);
6856 Delete(Str
, 1, b
+ 1);
6860 for a
:= 1 to Length(Str
) do
6861 if (a
= Length(Str
)) or (Str
[a
+ 1] = ' ') then
6863 Result
:= Copy(Str
, 1, a
);
6864 Delete(Str
, 1, a
+ 1);
6874 SetLength(Result
, Length(Result
)+1);
6875 Result
[High(Result
)] := GetStr(Str
);
6879 procedure TMainForm
.miTestMapClick(Sender
: TObject
);
6881 newWAD
, oldWAD
, tempMap
, ext
: String;
6888 // Ignore while map testing in progress
6889 if MapTestProcess
<> nil then
6892 // Сохраняем временную карту:
6895 newWAD
:= Format('%s/temp%.4d', [MapsDir
, time
]);
6897 until not FileExists(newWAD
);
6898 if OpenedMap
<> '' then
6900 oldWad
:= g_ExtractWadName(OpenedMap
);
6901 newWad
:= newWad
+ ExtractFileExt(oldWad
);
6902 if CopyFile(oldWad
, newWad
) = false then
6903 e_WriteLog('MapTest: unable to copy [' + oldWad
+ '] to [' + newWad
+ ']', MSG_WARNING
)
6907 newWad
:= newWad
+ '.wad'
6909 tempMap
:= newWAD
+ ':\' + TEST_MAP_NAME
;
6914 if TestOptionsTwoPlayers
then
6916 if TestOptionsTeamDamage
then
6918 if TestOptionsAllowExit
then
6920 if TestOptionsWeaponStay
then
6922 if TestOptionsMonstersDM
then
6926 proc
:= TProcessUTF8
.Create(nil);
6927 proc
.Executable
:= TestD2dExe
;
6929 // TODO: get real executable name from Info.plist
6930 if LowerCase(ExtractFileExt(TestD2dExe
)) = '.app' then
6931 proc
.Executable
:= TestD2dExe
+ DirectorySeparator
+ 'Contents' + DirectorySeparator
+ 'MacOS' + DirectorySeparator
+ 'Doom2DF';
6933 proc
.Parameters
.Add('-map');
6934 proc
.Parameters
.Add(tempMap
);
6935 proc
.Parameters
.Add('-gm');
6936 proc
.Parameters
.Add(TestGameMode
);
6937 proc
.Parameters
.Add('-limt');
6938 proc
.Parameters
.Add(TestLimTime
);
6939 proc
.Parameters
.Add('-lims');
6940 proc
.Parameters
.Add(TestLimScore
);
6941 proc
.Parameters
.Add('-opt');
6942 proc
.Parameters
.Add(IntToStr(opt
));
6943 proc
.Parameters
.Add('--debug');
6945 proc
.Parameters
.Add('--close');
6947 args
:= ParseString(TestD2DArgs
);
6948 for i
:= 0 to High(args
) do
6949 proc
.Parameters
.Add(args
[i
]);
6959 tbTestMap
.Enabled
:= False;
6960 MapTestFile
:= newWAD
;
6961 MapTestProcess
:= proc
;
6965 Application
.MessageBox(PChar(MsgMsgExecError
), 'FIXME', MB_OK
or MB_ICONERROR
);
6966 SysUtils
.DeleteFile(newWAD
);
6971 procedure TMainForm
.sbVerticalScroll(Sender
: TObject
;
6972 ScrollCode
: TScrollCode
; var ScrollPos
: Integer);
6974 MapOffset
.Y
:= -sbVertical
.Position
;
6975 RenderPanel
.Invalidate
;
6978 procedure TMainForm
.sbHorizontalScroll(Sender
: TObject
;
6979 ScrollCode
: TScrollCode
; var ScrollPos
: Integer);
6981 MapOffset
.X
:= -sbHorizontal
.Position
;
6982 RenderPanel
.Invalidate
;
6985 procedure TMainForm
.miOpenWadMapClick(Sender
: TObject
);
6987 if OpenedWAD
<> '' then
6989 OpenMap(OpenedWAD
, '');
6993 procedure TMainForm
.selectall1Click(Sender
: TObject
);
6997 RemoveSelectFromObjects();
6999 if gPanels
<> nil then
7000 for a
:= 0 to High(gPanels
) do
7001 if gPanels
[a
].PanelType
<> PANEL_NONE
then
7002 SelectObject(OBJECT_PANEL
, a
, True);
7004 if gItems
<> nil then
7005 for a
:= 0 to High(gItems
) do
7006 if gItems
[a
].ItemType
<> ITEM_NONE
then
7007 SelectObject(OBJECT_ITEM
, a
, True);
7009 if gMonsters
<> nil then
7010 for a
:= 0 to High(gMonsters
) do
7011 if gMonsters
[a
].MonsterType
<> MONSTER_NONE
then
7012 SelectObject(OBJECT_MONSTER
, a
, True);
7014 if gAreas
<> nil then
7015 for a
:= 0 to High(gAreas
) do
7016 if gAreas
[a
].AreaType
<> AREA_NONE
then
7017 SelectObject(OBJECT_AREA
, a
, True);
7019 if gTriggers
<> nil then
7020 for a
:= 0 to High(gTriggers
) do
7021 if gTriggers
[a
].TriggerType
<> TRIGGER_NONE
then
7022 SelectObject(OBJECT_TRIGGER
, a
, True);
7024 RecountSelectedObjects();
7027 procedure TMainForm
.Splitter1CanResize(Sender
: TObject
;
7028 var NewSize
: Integer; var Accept
: Boolean);
7030 Accept
:= (NewSize
> 140);
7033 procedure TMainForm
.Splitter2CanResize(Sender
: TObject
;
7034 var NewSize
: Integer; var Accept
: Boolean);
7036 Accept
:= (NewSize
> 110);
7039 procedure TMainForm
.vleObjectPropertyEnter(Sender
: TObject
);
7041 EditingProperties
:= True;
7044 procedure TMainForm
.vleObjectPropertyExit(Sender
: TObject
);
7046 EditingProperties
:= False;
7049 procedure TMainForm
.FormKeyUp(Sender
: TObject
; var Key
: Word; Shift
: TShiftState
);
7051 // Объекты передвигались:
7052 if MainForm
.ActiveControl
= RenderPanel
then
7054 if (Key
= VK_NUMPAD4
) or
7055 (Key
= VK_NUMPAD6
) or
7056 (Key
= VK_NUMPAD8
) or
7057 (Key
= VK_NUMPAD5
) or
7058 (Key
= Ord('V')) then
7061 // Быстрое превью карты:
7062 if Key
= Ord('E') then
7064 if PreviewMode
= 2 then
7067 RenderPanelMouseMove(Sender
, Shift
, RenderMousePos().X
, RenderMousePos().Y
);