/bin/dbxoodbc/Source/Demos/DbExplor/Optional.Libs/HexEdit/mpHexEditorEx.pas

http://github.com/sanelson/dbdesigner-fork · Pascal · 4001 lines · 3094 code · 340 blank · 567 comment · 316 complexity · e3f980de9fa61787c1dbaaecc21e0430 MD5 · raw file

  1. (*
  2. TMPHexEditorEx v 02-06-2006<br>
  3. @author((C) markus stephany, vcl[at]mirkes[dot]de, all rights reserved.)
  4. @abstract(TMPHexEditorEx, an enhanced TMPHexEditor: print and preview, ole drag and drop,
  5. ole clipboard handling, file backups...)
  6. @lastmod(02-06-2006)
  7. credits to :<br><br>
  8. - John Hamm, http://users.snapjax.com/john/<br><br>
  9. - Christophe Le Corfec for introducing the EBCDIC format and the nice idea about
  10. half byte insert/delete<br><br>
  11. - Philippe Chessa for his suggestions about AsText, AsHex and better support for
  12. the french keyboard layout<br><br>
  13. - Daniel Jensen for octal offset display and the INS-key recognition stuff<br><br>
  14. - Shmuel Zeigerman for introducing more flexible offset display formats<br><br>
  15. - Vaf, http://carradio.al.ru for reporting missing delver.inc and suggesting OnChange<br><br>
  16. - Eugene Tarasov for reporting that setting the BytesPerColumn value to 4 at design
  17. time didn't work<br><br>
  18. - FuseBurner for BytesPerUnit/RulerBytesPerUnit related suggestions<br><br>
  19. - Motzi for SyncView/ShowPositionIfNotFocused related suggestions<br><br>
  20. - Martin Hsiao for bcb compatibility and reporting some bugs when moving cursor beyond eof<br><br>
  21. - Miyu for delphi 7 defines<br><br>
  22. - Nils Hoyer for bcb testing and his help on creating a BCB6 package<br><br>
  23. - Skamnitsly S.V for reporting a bug when doubleclicking the ruler bar<br><br>
  24. - Pete Fraser for reporting problems with array properties under BCB<br><br>
  25. - Andrew Novikov for bug reports and suggestions<br><br>
  26. - Al for bug reports<br><br>
  27. - Dieter Köhler for reporting the delphi vcl related CanFocus bug<br><br>
  28. - Piotr Likus for reporting a cardinal&lt;-&gt;integer related bug in the Undo method<br><br>
  29. - Marc Girod for bug reports<br><br>
  30. - Gerd Schwartz for reporting a bug with printing headers/footers that contain long texts<br><br>
  31. - Bogdan Ureche for reporting an integer overflow when moving the cursor over a large selection<br><br>
  32. <h3>history:</h3>
  33. <p><ul>
  34. <li>v 02-06-2006: february 06, 2006<br><br>
  35. - changes in the base class (@link(TCustomMPHexEditor)) only <br><br></li>
  36. <li>v 05-23-2005: may 23, 2005<br><br>
  37. - changes in the base class (@link(TCustomMPHexEditor)) only <br><br></li>
  38. <li>v 12-29-2004: december 29, 2004<br><br>
  39. - initialized Result to '' in some string functions/methods to avoid
  40. non empty Result vars at function startup due to compiler
  41. optimizations (particularly on d4), e.g. printing did not work
  42. correctly under d4<br>
  43. - updated some of the sample projects (fixed the broken bcb6 sample,
  44. added printing to the hex viewer and the bcb6 editor sample) <br><br></li>
  45. <li>v 12-28-2004: december 28, 2004<br><br>
  46. - changes in the base class (@link(TCustomMPHexEditor)) only <br><br></li>
  47. <li>v 12-21-2004: december 21, 2004<br><br>
  48. - changes in the base class (@link(TCustomMPHexEditor))<br>
  49. - support for CF_HTML clipboard format<br><br></li>
  50. <li>v 11-12-2004: november 12, 2004<br><br>
  51. - changes in the base class (@link(TCustomMPHexEditor))<br>
  52. - ole drag and drop move operation is now disabled if the editor's
  53. ReadOnlyView property is set to True<br><br></li>
  54. <li>v 10-26-2004: october 26, 2004<br><br>
  55. - changes in the base class (@link(TCustomMPHexEditor))/unit (@link(mphexeditor)) only <br><br></li>
  56. <li>v 08-29-2004: august 29, 2004<br><br>
  57. - changes in the base class (@link(TCustomMPHexEditor))<br>
  58. - added pfIncludeRuler to @link(TMPHPrintFlag)<br><br></li>
  59. <li>v 08-14-2004: august 14, 2004<br><br>
  60. - changed printing (color handling, pfSelectionBold meaning)<br><br></li>
  61. <li>v 06-15-2004: june 15, 2004<br><br>
  62. - changes in the base class (@link(TCustomMPHexEditor)) and some more inherited
  63. published properties <br><br></li>
  64. <li>v 06-10-2004: june 10, 2004<br><br>
  65. - changes in the base class (@link(TCustomMPHexEditor)) only <br><br></li>
  66. <li>v 06-07-2004: june 07, 2004<br><br>
  67. - changes in the base class (@link(TCustomMPHexEditor)) only <br><br></li>
  68. <li>v 05-27-2004: may 27, 2004<br><br>
  69. - changes in the base class (@link(TCustomMPHexEditor)) only <br><br></li>
  70. <li>v 05-13-2004: may 13, 2004<br><br>
  71. - changes in the base class (@link(TCustomMPHexEditor)) only <br><br></li>
  72. <li>v 04-18-2004: april 18, 2004<br><br>
  73. - changes in the base class (@link(TCustomMPHexEditor)) only <br><br></li>
  74. <li>v 01-08-2004: january 08, 2004<br><br>
  75. - changes in the base class (@link(TCustomMPHexEditor)) only <br><br></li>
  76. <li>v 12-16-2003: december 16, 2003<br><br>
  77. - changes in the base class (@link(TCustomMPHexEditor)) only <br><br></li>
  78. <li>v 12-10-2003: december 10, 2003<br><br>
  79. - changes in the base class (@link(TCustomMPHexEditor)) only <br><br></li>
  80. <li>v 09-24-2003: september 24, 2003 <br><br>
  81. - modified the BCB6 package <br><br></li>
  82. <li>v 09-09-2003: september 09, 2003<br><br>
  83. - changed @link(UndoBeginUpdate) and @link(UndoEndUpdate) behaviour to automatically create an undo record
  84. on UndoBeginUpdate and check it on UndoEndUpdate, see also @link(CreateUndoOnUndoUpdate)<br>
  85. - added property @link(CreateUndoOnUndoUpdate) <br>
  86. - added defines for delphi7, renamed delver.inc to mpdelver.inc <br>
  87. - @link(PasteData) method added <br><br></li>
  88. <li>v 07-05-2003: july 05, 2003<br><br>
  89. - added support for pasting clipboard data in fixed filesize mode<br>
  90. - added RegEdit_HexData clipboard support<br><br></li>
  91. <li>v 05-25-2003-b: may 25, 2003<br><br>
  92. - fixed a bug (moving the cursor beyond eof)<br><br></li>
  93. <li>v 05-25-2003: may 25, 2003<br><br>
  94. - no ':' is printed when offset display is not used<br>
  95. - added hpp generating statements for bcb compatibility<br><br></li>
  96. <li>v 05-20-2003: may 20, 2003<br><br>
  97. - added unicode support in printing<br><br></li>
  98. <li>v 05-17-2003: may 17, 2003<br><br>
  99. - moved some property related functions to protected<br>
  100. - corrected bottom margin handling when printing<br>
  101. - corrected upper/lowercase hex chars in printing<br>
  102. - the current unit is selected now when doubleclicking data<br>
  103. - added flags pfCurrentViewOnly (just print the currently
  104. visible data) to @link(PrintOptions).Flags<br><br></li>
  105. <li>v 08-18-2002: august 18, 2002<br><br>
  106. - first release</li>
  107. </ul></p>
  108. *)
  109. {$IFDEF BCB}
  110. {$HPPEMIT 'DECLARE_DINTERFACE_TYPE(IDropTarget)'}
  111. {$HPPEMIT 'DECLARE_DINTERFACE_TYPE(IDropSource)'}
  112. {$HPPEMIT 'DECLARE_DINTERFACE_TYPE(IEnumFORMATETC)'}
  113. {$ENDIF}
  114. unit MPHexEditorEx;
  115. {$I MPDELVER.INC}
  116. interface
  117. uses
  118. Windows, Messages, SysUtils, Classes, Controls, Forms,
  119. MPHexEditor, ActiveX, Graphics, Printers,
  120. ShlObj, Menus;
  121. type
  122. //@exclude
  123. // is data dropped or pasted
  124. TMPHOLEOperation = (oleDrop, oleClipboard);
  125. // @exclude(available clipboard / IDataObject formats)
  126. TClipFormats = array of TClipFormat;
  127. // @exclude(ole drop handler class)
  128. TMPHDropTarget = class;
  129. // @exclude(persistent print options)
  130. TMPHPrintOptions = class;
  131. (* print option flags:<br><br>
  132. - pfSelectionOnly: only print data currently selected<br>
  133. - pfSelectionBold: render the current selection using either a bold font or inverted colors (if pfSelectionOnly isn't set)<br>
  134. - pfMonochrome: don't use colors, print/preview black on white<br>
  135. - pfUseBackgroundColor: fill the margin rect with the editor's background color (if pfMonochrome isn't set)<br>
  136. - pfCurrentViewOnly: just print the data currently displayed<br>
  137. - pfIncludeRuler: draw the ruler at every page's top<br>
  138. *)
  139. TMPHPrintFlag = (pfSelectionOnly, pfSelectionBold, pfMonochrome,
  140. pfUseBackgroundColor, pfCurrentViewOnly, pfIncludeRuler);
  141. // @exclude()
  142. TMPHPrintFlags = set of TMPHPrintFlag;
  143. // @exclude(print header/footer)
  144. TMPHPrintHeaders = array[0..1] of string;
  145. (* this event is called when @link(PropertiesAsString) is read or written. TMPHexEditorEx
  146. has a fixed list of properties that can be read/written using PropertiesAsString.
  147. you can exclude some of the properties by setting IsPublic to False.
  148. *)
  149. TMPHQueryPublicPropertyEvent = procedure(Sender: TObject; const PropertyName:
  150. string;
  151. var IsPublic: boolean) of object;
  152. // enhanced hex editor
  153. TMPHexEditorEx = class(TCustomMPHexEditor)
  154. private
  155. { Private-Deklarationen }
  156. FCreateBackups: boolean;
  157. FBackupFileExt: string;
  158. FOleDragDrop: boolean;
  159. FDropTarget: TMPHDropTarget;
  160. FOleFormat: array[TMPHOLEOperation] of TClipFormat;
  161. FOleDragging, FOleStartDrag: boolean;
  162. FOleDragX, FOleDragY: integer;
  163. FOleWasTarget: boolean;
  164. FPrintOptions: TMPHPrintOptions;
  165. FPrintPages: integer;
  166. FPrintFont: TFont;
  167. FUseEditorFontForPrinting: boolean;
  168. FClipboardAsHexText: boolean;
  169. FClipData: IDataObject;
  170. FFlushClipboardAtShutDown: boolean;
  171. FSupportsOtherClipFormats: boolean;
  172. FOffsetPopupMenu: TPopupMenu;
  173. FZoomOnWheel: boolean;
  174. FPaintUpdateCounter: integer;
  175. FOnQueryPublicProperty: TMPHQueryPublicPropertyEvent;
  176. FHasDoubleClicked: boolean;
  177. FBookmarksNoChange: boolean;
  178. FCreateUndoOnUndoUpdate: boolean;
  179. FModifiedNoUndo: boolean;
  180. procedure SetOleDragDrop(const Value: boolean);
  181. function OLEHasSupportedFormat(const dataObj: IDataObject;
  182. const Formats: array of TClipFormat; var Format: TClipFormat): boolean;
  183. function GetMyOLEFormats: TClipFormats;
  184. procedure WMDestroy(var Message: TWMDestroy); message WM_DESTROY;
  185. procedure SetPrintOptions(const Value: TMPHPrintOptions);
  186. function PrintToCanvas(ACanvas: TCanvas; const APage: integer;
  187. const AMargins: TRect): integer;
  188. function PrinterMarginRect: TRect;
  189. procedure SetPrintFont(const Value: TFont);
  190. procedure SetOffsetPopupMenu(const Value: TPopupMenu);
  191. function GetOffsetPopupMenu: TPopupMenu;
  192. function GetBookmarksAsString: string;
  193. procedure SetBookMarksAsString(Value: string);
  194. protected
  195. { Protected-Deklarationen }
  196. function CanCreateUndo(const aKind: TMPHUndoFlag; const aCount, aReplCount:
  197. integer): Boolean; override;
  198. {$IFDEF DELPHI6UP}
  199. // @exclude()
  200. function GetPropertiesAsString: string; virtual;
  201. // @exclude()
  202. procedure SetPropertiesAsString(const Value: string); virtual;
  203. // @exclude()
  204. function IsPropPublic(PropName: string): boolean; virtual;
  205. {$ENDIF}
  206. // @exclude(check if in offset col, if yes, popup offsetcontextmenu)
  207. procedure Notification(AComponent: TComponent; Operation: TOperation);
  208. override;
  209. {$IFDEF DELPHI6UP}
  210. // @exclude()
  211. procedure DoContextPopup(MousePos: TPoint; var Handled: boolean); override;
  212. {$ENDIF}
  213. // @exclude(parse control keys)
  214. procedure KeyDown(var Key: word; Shift: TShiftState); override;
  215. // @exclude(overwrite mouse wheel for zooming)
  216. function DoMouseWheelDown(Shift: TShiftState; MousePos: TPoint): boolean;
  217. override;
  218. // @exclude(overwrite mouse wheel for zooming)
  219. function DoMouseWheelUp(Shift: TShiftState; MousePos: TPoint): boolean;
  220. override;
  221. // @exclude(create backups in savefile)
  222. procedure PrepareOverwriteDiskFile; override;
  223. // @exclude(overwrite mouse handling for ole drag and drop)
  224. procedure MouseMove(Shift: TShiftState; X, Y: integer); override;
  225. // @exclude(overwrite mouse handling for ole drag and drop)
  226. procedure MouseUp(Button: TMouseButton; Shift: TShiftState; X, Y: integer);
  227. override;
  228. // @exclude(overwrite mouse handling for ole drag and drop)
  229. procedure MouseDown(Button: TMouseButton; Shift: TShiftState; X, Y:
  230. integer);
  231. override;
  232. // @exclude(reset drop target's HWND)
  233. procedure CreateWnd; override;
  234. // @exclude(supported dnd/clipboard data available?)
  235. function SupportsOLEData(const dataObj: IDataObject; const grfKeyState:
  236. longint; const pt: TPoint; var dwEffect: longint; const Operation:
  237. TMPHOLEOperation): HRESULT;
  238. // @exclude(insert ole-dropped data)
  239. function InsertOLEData(const dataObj: IDataObject; const grfKeyState:
  240. longint; const pt: TPoint; var dwEffect: longint; const Operation:
  241. TMPHOLEOperation): HRESULT;
  242. // @exclude(modify drageffect depending on key states and data format)
  243. function ModifyOLEDropEffect(const grfKeyState: longint; const pt: TPoint;
  244. var dwEffect: longint): HRESULT;
  245. // @exclude(paint handler)
  246. procedure Paint; override;
  247. // @exclude(doubleclick handler for unit selection)
  248. procedure DblClick; override;
  249. // @exclude(override to avoid much updates when using setbookmarksasstring);
  250. procedure BookmarkChanged; override;
  251. public
  252. { Public-Deklarationen }
  253. // @exclude(Init)
  254. constructor Create(AOwner: TComponent); override;
  255. // @exclude(Done)
  256. destructor Destroy; override;
  257. // see inherited @inherited
  258. procedure WriteBuffer(const Buffer; const Index, Count: Integer); override;
  259. (* if set to True (default is False), an undo record is automatically created on calling
  260. @link(UndoBeginUpdate) and on calling @link(UndoEndUpdate) the record is deleted if the
  261. data has not been changed between UndoBegin- and UndoEndUpdate *)
  262. property CreateUndoOnUndoUpdate: boolean read FCreateUndoOnUndoUpdate write
  263. FCreateUndoOnUndoUpdate;
  264. (* each call to BeginUpdate increments an internal counter that prevents from repainting
  265. (see also @link(EndUpdate))
  266. *)
  267. function BeginUpdate: integer;
  268. (* each call to EndUpdate decrements an internal counter that prevents from repainting.
  269. the return value is the value of this counter. if the counter is reset to zero,
  270. repainting is permitted again (see also @link(BeginUpdate))
  271. *)
  272. function EndUpdate: integer;
  273. (* each call to UndoBeginUpdate increments an internal counter that prevents using
  274. undo storage and also disables undo functionality (see also @link(UndoEndUpdate))
  275. *)
  276. function UndoBeginUpdate(const StrUndoDesc: string = ''): integer;
  277. reintroduce;
  278. (* each call to UndoEndUpdate decrements an internal counter that prevents using
  279. undo storage and also disables undo functionality. the return value is the value
  280. of this counter. if the counter is reset to zero, undo creation is permitted again
  281. (see also @link(UndoBeginUpdate))
  282. *)
  283. function UndoEndUpdate: integer; override;
  284. // create an undo for a range of bytes
  285. procedure CreateRangeUndo(const aStart, aCount: integer; sDesc: string);
  286. // is pasting from clipboard possible?
  287. function CanPaste: boolean;
  288. // is copying to clipboard possible?
  289. function CanCopy: boolean;
  290. // is cutting to clipboard possible?
  291. function CanCut: boolean;
  292. // copy selection to clipboard
  293. function CBCopy: boolean;
  294. // cut selection to clipboard
  295. function CBCut: boolean;
  296. // paste clipboard's contents over current selection
  297. function CBPaste: boolean;
  298. // do we own the clipboard data?
  299. function OwnsClipBoard: boolean;
  300. // flush or empty the clipboard (if we own the IDataObject)
  301. procedure ReleaseClipboard(const Flush: boolean);
  302. // save to file (overwrite)
  303. procedure Save;
  304. // @exclude(dump undo storage)
  305. function DumpUndoStorage(const FileName: string): boolean;
  306. (* creates a TMetaFile object and renders the specified page
  307. on its canvas. Freeing of the TMetaFile is up to the caller!
  308. *)
  309. function PrintPreview(const Page: integer): TMetaFile;
  310. (* print the given page to the default printer.
  311. Printer.BeginDoc, Printer.NewPage and Printer.EndDoc must be issued by the caller!
  312. *)
  313. procedure Print(const Page: integer);
  314. // get the number of pages to print
  315. function PrintNumPages: integer;
  316. // paste data (in clipboardmanner: check current selection and so on)
  317. procedure PasteData(P: Pointer; const ACount: integer; const UndoDesc: string
  318. = '');
  319. // get/set bookmarks as text (for storing in registry, ini-file)
  320. property BookMarksAsString: string read GetBookmarksAsString write
  321. SetBookMarksAsString;
  322. {$IFDEF DELPHI6UP}
  323. // get set properties as text (for storing in registry, ini-file);
  324. property PropertiesAsString: string read GetPropertiesAsString write
  325. SetPropertiesAsString;
  326. {$ENDIF}
  327. published
  328. { Published-Deklarationen }
  329. // create a backup on save ? (see also @link(BackupExtension))
  330. property CreateBackup: boolean read FCreateBackups write FCreateBackups
  331. default True;
  332. // add this extension to the file if making backups, see @link(CreateBackup)
  333. property BackupExtension: string read FBackupFileExt write FBackupFileExt;
  334. (* if set To True, OLE drag and drop will used automatically when dragging starts
  335. or supported OLE data has been dropped on the hex editor
  336. *)
  337. property OleDragDrop: boolean read FOleDragDrop write SetOleDragDrop default
  338. False;
  339. // if set to True, CF_TEXT on the clipboard will be treated as hex formatted text
  340. property ClipboardAsHexText: boolean read FClipboardAsHexText write
  341. FClipboardAsHexText default False;
  342. // flush or empty clipboard at shutdown
  343. property FlushClipboardAtShutDown: boolean read FFlushClipboardAtShutDown
  344. write FFlushClipboardAtShutDown default False;
  345. // do we support other formats than CF_MPHEXEDITOR and CF_HDROP?
  346. property SupportsOtherClipFormats: boolean read FSupportsOtherClipFormats
  347. write FSupportsOtherClipFormats default True;
  348. // print/preview options, see @link(TMPHPrintOptions)
  349. property PrintOptions: TMPHPrintOptions read FPrintOptions write
  350. SetPrintOptions;
  351. // print using this font
  352. property PrintFont: TFont read FPrintFont write SetPrintFont;
  353. // if set to True, the editor's font will be used for printing
  354. property UseEditorFontForPrinting: boolean read FUseEditorFontForPrinting
  355. write FUseEditorFontForPrinting default True;
  356. (* if this property is assigned to a TPopupMenu, it will be shown on right clicking
  357. the offset display pane. then the normal PopupMenu will open on right
  358. clicking the character and hex pane.
  359. *)
  360. property OffsetPopupMenu: TPopupMenu read GetOffsetPopupMenu write
  361. SetOffsetPopupMenu;
  362. // auto-zoom on mouse wheel?
  363. property ZoomOnWheel: boolean read FZoomOnWheel write FZoomOnWheel default
  364. True;
  365. (* this event is called when @link(PropertiesAsString) is read or written.
  366. (see @link(TMPHQueryPublicPropertyEvent))
  367. *)
  368. property OnQueryPublicProperty: TMPHQueryPublicPropertyEvent read
  369. FOnQueryPublicProperty write FOnQueryPublicProperty;
  370. // @exclude(inherited)
  371. property Align;
  372. // @exclude(inherited)
  373. property Anchors;
  374. // @exclude(inherited)
  375. property BiDiMode;
  376. // @exclude(inherited)
  377. property BorderStyle;
  378. // @exclude(inherited)
  379. property Constraints;
  380. // @exclude(inherited)
  381. property Ctl3D;
  382. // @exclude(inherited)
  383. property DragCursor;
  384. // @exclude(inherited)
  385. property DragKind;
  386. // @exclude(inherited)
  387. property DragMode;
  388. // @exclude(inherited)
  389. property Enabled;
  390. // @exclude(inherited)
  391. property Font;
  392. // @exclude(inherited)
  393. property ImeMode;
  394. // @exclude(inherited)
  395. property ImeName;
  396. // @exclude(inherited)
  397. property OnClick;
  398. // @exclude(inherited)
  399. property OnDblClick;
  400. // @exclude(inherited)
  401. property OnDragDrop;
  402. // @exclude(inherited)
  403. property OnDragOver;
  404. // @exclude(inherited)
  405. property OnEndDock;
  406. // @exclude(inherited)
  407. property OnEndDrag;
  408. // @exclude(inherited)
  409. property OnEnter;
  410. // @exclude(inherited)
  411. property OnExit;
  412. // @exclude(inherited)
  413. property OnKeyDown;
  414. // @exclude(inherited)
  415. property OnKeyPress;
  416. // @exclude(inherited)
  417. property OnKeyUp;
  418. // @exclude(inherited)
  419. property OnMouseDown;
  420. // @exclude(inherited)
  421. property OnMouseMove;
  422. // @exclude(inherited)
  423. property OnMouseUp;
  424. // @exclude(inherited)
  425. property OnMouseWheel;
  426. // @exclude(inherited)
  427. property OnMouseWheelDown;
  428. // @exclude(inherited)
  429. property OnMouseWheelUp;
  430. // @exclude(inherited)
  431. property OnStartDock;
  432. // @exclude(inherited)
  433. property OnStartDrag;
  434. // @exclude(inherited)
  435. property ParentBiDiMode;
  436. // @exclude(inherited)
  437. property ParentCtl3D;
  438. // @exclude(inherited)
  439. property ParentFont;
  440. // @exclude(inherited)
  441. property ParentShowHint;
  442. // @exclude(inherited)
  443. property PopupMenu;
  444. // @exclude(inherited)
  445. property ScrollBars;
  446. // @exclude(inherited)
  447. property ShowHint;
  448. // @exclude(inherited)
  449. property TabOrder;
  450. // @exclude(inherited)
  451. property TabStop;
  452. // @exclude(inherited)
  453. property Visible;
  454. // see inherited @inherited
  455. property BytesPerRow;
  456. // see inherited @inherited
  457. property BytesPerColumn;
  458. // see inherited @inherited
  459. property Translation;
  460. // see inherited @inherited
  461. property OffsetFormat;
  462. // see inherited @inherited
  463. property CaretKind;
  464. // see inherited @inherited
  465. property Colors;
  466. // see inherited @inherited
  467. property FocusFrame;
  468. // see inherited @inherited
  469. property SwapNibbles;
  470. // see inherited @inherited
  471. property MaskChar;
  472. // see inherited @inherited
  473. property NoSizeChange;
  474. // see inherited @inherited
  475. property AllowInsertMode;
  476. // see inherited @inherited
  477. property DrawGridLines;
  478. // see inherited @inherited
  479. property WantTabs;
  480. // see inherited @inherited
  481. property ReadOnlyView;
  482. // see inherited @inherited
  483. property HideSelection;
  484. // see inherited @inherited
  485. property GraySelectionIfNotFocused;
  486. // see inherited @inherited
  487. property GutterWidth;
  488. // see inherited @inherited
  489. property BookmarkBitmap;
  490. // see inherited @inherited
  491. property Version;
  492. // see inherited @inherited
  493. property MaxUndo;
  494. // see inherited @inherited
  495. property InsertMode;
  496. // see inherited @inherited
  497. property HexLowerCase;
  498. // see inherited @inherited
  499. property OnProgress;
  500. // see inherited @inherited
  501. property OnInvalidKey;
  502. // see inherited @inherited
  503. property OnTopLeftChanged;
  504. // see inherited @inherited
  505. property OnChange;
  506. // see inherited @inherited
  507. property DrawGutter3D;
  508. // see inherited @inherited
  509. property ShowRuler;
  510. // see inherited @inherited
  511. property BytesPerUnit;
  512. // see inherited @inherited
  513. property RulerBytesPerUnit;
  514. // see inherited @inherited
  515. property ShowPositionIfNotFocused;
  516. // see inherited @inherited
  517. property OnSelectionChanged;
  518. // see inherited @inherited
  519. property UnicodeChars;
  520. // see inherited @inherited
  521. property UnicodeBigEndian;
  522. // see inherited @inherited
  523. property OnDrawCell;
  524. // see inherited @inherited
  525. property OnBookmarkChanged;
  526. // see inherited @inherited
  527. property OnGetOffsetText;
  528. // see inherited @inherited
  529. property BytesPerBlock;
  530. // see inherited @inherited
  531. property SeparateBlocksInCharField;
  532. // see inherited @inherited
  533. property FindProgress;
  534. // see inherited @inherited
  535. property RulerNumberBase;
  536. end;
  537. // @exclude(ole drop target class)
  538. TMPHDropTarget = class(TInterfacedObject, IDropTarget)
  539. private
  540. FEditor: TMPHexEditorEx;
  541. FEditorHandle: THandle;
  542. FActive: boolean;
  543. procedure SetActive(const Value: boolean);
  544. public
  545. constructor Create(Editor: TMPHexEditorEx);
  546. procedure BeforeDestruction; override;
  547. function DragEnter(const dataObj: IDataObject; grfKeyState: longint; pt:
  548. TPoint; var dwEffect: longint): HResult; stdcall;
  549. function DragOver(grfKeyState: longint; pt: TPoint; var dwEffect: longint):
  550. HResult; stdcall;
  551. function DragLeave: HResult; stdcall;
  552. function Drop(const dataObj: IDataObject; grfKeyState: longint; pt: TPoint;
  553. var dwEffect: longint): HResult; stdcall;
  554. property Active: boolean read FActive write SetActive;
  555. end;
  556. // print / preview options
  557. TMPHPrintOptions = class(TPersistent)
  558. private
  559. FMargins: TRect;
  560. FHeaders: TMPHPrintHeaders;
  561. FFlags: TMPHPrintFlags;
  562. function GetHeader(const Index: integer): string;
  563. function GetMargin(const Index: integer): integer;
  564. procedure SetHeader(const Index: integer; const Value: string);
  565. procedure SetMargin(const Index, Value: integer);
  566. public
  567. // @exclude(Init)
  568. constructor Create;
  569. // @exclude()
  570. procedure Assign(Source: TPersistent); override;
  571. published
  572. // left margin in Millimeters
  573. property MarginLeft: integer index 1 read GetMargin write SetMargin;
  574. // top margin in Millimeters
  575. property MarginTop: integer index 2 read GetMargin write SetMargin;
  576. // right margin in Millimeters
  577. property MarginRight: integer index 3 read GetMargin write SetMargin;
  578. // bottom margin in Millimeters
  579. property MarginBottom: integer index 4 read GetMargin write SetMargin;
  580. (* this line will be rendered on top of the printed page, some characters have special meanings:<br><br>
  581. - the string may contain three parts separated by a "|" (pipe) character (left|center|right)<br>
  582. - each part knows some special variables:
  583. <ul>
  584. <li><b>%f</b>: substituted with the filename part of the editor's filename</li>
  585. <li><b>%F</b>: substituted with the expanded name of the editor's filename</li>
  586. <li><b>%p</b>: substituted with the number of the current page</li>
  587. <li><b>%P</b>: substituted with the number of pages</li>
  588. <li><b>%t</b>: substituted with the current time</li>
  589. <li><b>%d</b>: substituted with the current date</li>
  590. <li><b>%&gt;</b>: substituted with the long description of the editor's current @link(Translation)</li>
  591. <li><b>%&lt;</b>: substituted with the short description of the editor's current @link(Translation)</li>
  592. </ul>
  593. *)
  594. property PageHeader: string index 0 read GetHeader write SetHeader;
  595. // this line will be rendered on the bottom of the printed page (see @link(PageHeader))
  596. property PageFooter: string index 1 read GetHeader write SetHeader;
  597. (* printing flags:<br><br>
  598. - pfSelectionOnly: only print data currently selected<br>
  599. - pfSelectionBold: render the current selection using either a bold font or inverted colors (if pfSelectionOnly isn't set)<br>
  600. - pfMonochrome: don't use colors, print/preview black on white<br>
  601. - pfUseBackgroundColor: fill the margin rect with the editor's background color (if pfMonochrome isn't set)<br>
  602. - pfCurrentViewOnly: just print the data currently displayed
  603. *)
  604. property Flags: TMPHPrintFlags read FFlags write FFlags;
  605. end;
  606. // default print margins
  607. const
  608. MPH_DEF_PRINT_MARGINS: TRect = (Left: 20; Top: 15; Right: 25; Bottom: 25);
  609. implementation
  610. uses
  611. Consts, StdCtrls, ShellAPI, ComObj, TypInfo;
  612. resourcestring
  613. // error messages
  614. ERR_NOFILE = 'No filename specified';
  615. ERR_INVALID_PAGE = 'Invalid page index';
  616. ERR_PRINTING_FAILED = 'Printing failed';
  617. ERR_BACKUP_DELETE = 'Cannot delete previous backup %s. (%s)';
  618. ERR_BACKUP_CREATE = 'Cannot create backup %s. (%s)';
  619. ERR_INVALID_BOOKFMT = 'Invalid bookmark format';
  620. // additional undo descriptions
  621. UNDO_PASTECB = 'Paste from clipboard';
  622. UNDO_CUTCB = 'Cut to clipboard';
  623. UNDO_DROPPED = 'Data dropped';
  624. UNDO_MOVED = 'Data moved';
  625. // select clipb/ole format dialog strings
  626. SELECT_FORMAT_CAPTION = 'Select data format';
  627. SELECT_FORMAT_ASHEX = 'Hex text';
  628. // when data dropped to explorer, give it this filename; first %s filename w/o ext, (second %s original file ext)
  629. STR_SCRAPFILE = 'Dump of %s.bin';
  630. // native clipboard format name
  631. MPTH_CF = 'TMPHexeditorEx Clipboard Format';
  632. // predefined clipboard format names
  633. STR_CF_TEXT = 'Text';
  634. STR_CF_BITMAP = 'Bitmap Picture';
  635. STR_CF_METAFILEPICT = 'Metafile Picture';
  636. STR_CF_SYLK = 'Microsoft Symbolic Link (SYLK) data';
  637. STR_CF_DIF = 'Software Arts'' Data Interchange Format';
  638. STR_CF_TIFF = 'Tagged Image File Format (TIFF) Picture';
  639. STR_CF_OEMTEXT = 'OEM Text';
  640. STR_CF_DIB = 'Device Independent Bitmap Picture';
  641. STR_CF_PALETTE = 'Color Palete';
  642. STR_CF_PENDATA = 'Pen Data';
  643. STR_CF_RIFF = 'RIFF Audio Data';
  644. STR_CF_WAVE = 'Wave Audio';
  645. STR_CF_UNICODETEXT = 'Unicode Text';
  646. STR_CF_ENHMETAFILE = 'Enhanced Metafile Picture';
  647. STR_CF_HDROP = 'File List';
  648. STR_CF_LOCALE = 'Text Locale';
  649. type
  650. // my clipboard data struct
  651. PClipData = ^TClipData;
  652. TClipData = packed record
  653. Signature: DWORD;
  654. Version: DWORD;
  655. Size: integer;
  656. Data: array[0..0] of char;
  657. end;
  658. PRegEditHexData = ^TRegEditHexData;
  659. TRegEditHexData = packed record
  660. Size: integer;
  661. Data: array[0..0] of char;
  662. end;
  663. const
  664. // signature of own format clipboard data
  665. CLIP_SIG = $4854504D; // MPTH;
  666. // version of own format clipboard data
  667. CLIP_VER = $00010001;
  668. // initial file extension of backups
  669. BACKUP_EXT = '.bak';
  670. // not so predefined common/known clipboard format names
  671. CFSTR_RTF = 'Rich Text Format';
  672. CFSTR_LOGICALPERFORMEDDROPEFFECT = 'Logical Performed DropEffect';
  673. CFSTR_REGEDIT_HEXDATA = 'RegEdit_HexData';
  674. CFSTR_HTML = 'HTML Format';
  675. var
  676. // custom/ shell CF format
  677. CF_MPHEXEDITOR,
  678. CF_RTF,
  679. CF_FILECONTENTS,
  680. CF_PERFORMEDDROPEFFECT,
  681. CF_LOGICALPERFORMEDDROPEFFECT,
  682. CF_FILEDESCRIPTOR,
  683. CF_HTML,
  684. CF_REGEDIT_HEXDATA: TClipFormat;
  685. type
  686. // private idataobject format enumerator
  687. TFormatEnum = class
  688. private
  689. FFormats: array of TFormatETC;
  690. public
  691. constructor Create(const dataObject: IDataObject);
  692. destructor Destroy; override;
  693. function HasFormat(const cfFormat: TClipFormat): boolean;
  694. function GetFormatETC(const cfFormat: TClipFormat): TFormatETC;
  695. end;
  696. const
  697. // number of clip formats that we can provide
  698. MY_SUPPORTED_FORMATS = 4;
  699. type
  700. // ole "public" format enumerator for own data
  701. TMPHEnumFormatETC = class(TInterfacedObject, IEnumFormatETC)
  702. private
  703. FFormats: packed array[0..MY_SUPPORTED_FORMATS - 1] of TFormatETC;
  704. FIndex: integer;
  705. public
  706. constructor Create;
  707. function Next(celt: longint; out elt; pceltFetched: PLongint): HResult;
  708. stdcall;
  709. function Skip(celt: longint): HResult; stdcall;
  710. function Reset: HResult; stdcall;
  711. function Clone(out Enum: IEnumFormatEtc): HResult; stdcall;
  712. end;
  713. // ole drop source
  714. TMPHDropSource = class(TInterfacedObject, IDropSource)
  715. public
  716. function QueryContinueDrag(fEscapePressed: BOOL; grfKeyState: longint):
  717. HResult; stdcall;
  718. function GiveFeedback(dwEffect: longint): HResult; stdcall;
  719. end;
  720. // ole data container
  721. TMPHDataObject = class(TInterfacedObject, IDataObject)
  722. private
  723. FData: Pointer;
  724. FDataSize: integer;
  725. FFileName: ShortString;
  726. FHasDropEffect: boolean;
  727. FDropEffect: cardinal;
  728. FTextAsHex: boolean;
  729. FSwapNibbles: boolean;
  730. public
  731. constructor Create(Data: Pointer; DataSize: integer; ScrapFileName:
  732. ShortString; TextAsHex, SwapNibbles: boolean);
  733. constructor CreateFromStream(Stream: TStream; Position, DataSize: integer;
  734. ScrapFileName: ShortString; TextAsHex, SwapNibbles: boolean);
  735. procedure BeforeDestruction; override;
  736. function GetData(const formatetcIn: TFormatEtc; out medium: TStgMedium):
  737. HResult; stdcall;
  738. function GetDataHere(const formatetc: TFormatEtc; out medium: TStgMedium):
  739. HResult; stdcall;
  740. function QueryGetData(const formatetc: TFormatEtc): HResult; stdcall;
  741. function GetCanonicalFormatEtc(const formatetc: TFormatEtc; out
  742. formatetcOut: TFormatEtc): HResult; stdcall;
  743. function SetData(const formatetc: TFormatEtc; var medium: TStgMedium;
  744. fRelease: BOOL): HResult; stdcall;
  745. function EnumFormatEtc(dwDirection: longint; out enumFormatEtc:
  746. IEnumFormatEtc): HResult; stdcall;
  747. function DAdvise(const formatetc: TFormatEtc; advf: longint; const advSink:
  748. IAdviseSink; out dwConnection: longint): HResult; stdcall;
  749. function DUnadvise(dwConnection: longint): HResult; stdcall;
  750. function EnumDAdvise(out enumAdvise: IEnumStatData): HResult; stdcall;
  751. end;
  752. // draw hex on canvas
  753. TMPHCanvasPrinter = class(TObject)
  754. private
  755. FMargins: TRect;
  756. FHeaders,
  757. FPrintHeaders: TMPHPrintHeaders;
  758. FLinesPerPage: integer;
  759. FFlags: TMPHPrintFlags;
  760. FPages: integer;
  761. FEditor: TMPHexEditorEx;
  762. FCanvas: TCanvas;
  763. function GetLinesPerPage: integer;
  764. function BuildHeader(const S: string; const Page: integer): string;
  765. protected
  766. function DrawOrCalc(const JustCalc: boolean; const Page: integer): integer;
  767. public
  768. constructor Create(AEditor: TMPHexEditorEx; ACanvas: TCanvas; AFlags:
  769. TMPHPrintFlags; AMargins: TRect; AHeaders: TMPHPrintHeaders);
  770. procedure Draw(const Page: integer);
  771. property LinesPerPage: integer read GetLinesPerPage;
  772. property Pages: integer read FPages;
  773. end;
  774. var
  775. // most recent selected clip format
  776. LAST_USED_CF: integer = -1;
  777. // returns the stgmedium struct for a given idataobject/format specification
  778. function GetIDataObjectData(const dataObj: IDataObject; const Format:
  779. TClipFormat; out Medium: TStgMedium): HRESULT;
  780. var
  781. LobjEnum: TFormatEnum;
  782. begin
  783. LobjEnum := TFormatEnum.Create(dataObj);
  784. try
  785. if not LobjEnum.HasFormat(Format) then
  786. Result := E_FAIL
  787. else
  788. Result := dataObj.GetData(LobjEnum.GetFormatETC(Format), Medium);
  789. finally
  790. LobjEnum.Free;
  791. end;
  792. end;
  793. // cast/copy hglobal to data structure depending on the format
  794. function GetSomeData(const PData: Pointer; const HGlobal: THandle; Format:
  795. TClipFormat; const DataSize: integer; const UnicodeBigEndian: Boolean):
  796. string;
  797. var
  798. LWStrTemp: widestring;
  799. LRecBmpHeader: TBitmapFileheader;
  800. LRecPalette: TMaxLogPalette;
  801. LIntTemp: integer;
  802. LbmpTemp: TBitmap;
  803. LmefTemp: TMetaFile;
  804. LmstData: TMemoryStream;
  805. LIntLoop: integer;
  806. begin
  807. Result := '';
  808. // to use case..of (cf_rtf is not a constant)
  809. if (Format = CF_RTF) or (Format = CF_HTML) then
  810. Format := CF_TEXT;
  811. if Format = CF_MPHEXEDITOR then
  812. begin
  813. with PClipData(PData)^ do
  814. if (Signature = CLIP_SIG) and (Version = CLIP_VER) then
  815. SetString(Result, Data, Size)
  816. end
  817. else if Format = CF_REGEDIT_HEXDATA then
  818. begin
  819. with PRegEditHexData(PData)^ do
  820. SetString(Result, Data, Size);
  821. end
  822. else
  823. case Format of
  824. CF_TEXT,
  825. CF_OEMTEXT: Result := PChar(PData);
  826. CF_UNICODETEXT:
  827. begin
  828. LWStrTemp := PWideChar(PData);
  829. if UnicodeBigEndian then
  830. begin
  831. for LIntLoop := 1 to Length(LWstrTemp) do
  832. SwapWideChar(LWstrTemp[LIntLoop]);
  833. end;
  834. {$WARNINGS OFF}
  835. // don't convert, get wide data as is
  836. SetString(Result, PChar(LWStrTemp), Length(LWStrTemp) *
  837. (sizeof(widechar) div sizeof(char)));
  838. {$WARNINGS ON}
  839. end;
  840. CF_LOCALE:
  841. begin
  842. // locale id , word pointed to by the global handle
  843. SetLength(Result, sizeof(word));
  844. Move(PWord(PData)^, Result[1], sizeof(word));
  845. end;
  846. CF_DIB:
  847. begin
  848. // stored as bitmap without header, so prefix a bmp header
  849. FillChar(LRecBMPHeader, sizeof(LRecBMPHeader), #0);
  850. LRecBMPHeader.bfType := $4D42; // BM
  851. SetLength(Result, sizeof(LRecBMPHeader) + DataSize);
  852. Move(LRecBMPHeader, Result[1], sizeof(LRecBmpHeader));
  853. Move(PData^, Result[1 + sizeof(LRecBMPHeader)], DataSize);
  854. end;
  855. CF_PALETTE:
  856. begin
  857. // copy palette entries
  858. LIntTemp := 0;
  859. if (GetObject(HGlobal, sizeof(LIntTemp), @LIntTemp) <> 0) and (LIntTemp
  860. > 0) then
  861. begin
  862. with LRecPalette do
  863. begin
  864. palVersion := $0300;
  865. palNumEntries := LIntTemp;
  866. GetPaletteEntries(HGlobal, 0, LIntTemp, palPalEntry);
  867. end;
  868. SetLength(Result, sizeof(TLogPalette) + ((LintTemp - 1) *
  869. sizeof(TPaletteEntry)));
  870. Move(LRecPalette, Result[1], Length(Result));
  871. end;
  872. end;
  873. CF_BITMAP:
  874. begin
  875. // data not stored in global mem, but as a bitmap handle
  876. LbmpTemp := TBitmap.Create;
  877. try
  878. LbmpTemp.Handle := CopyImage(HGlobal, IMAGE_BITMAP, 0, 0,
  879. LR_COPYRETURNORG);
  880. LmstData := TMemoryStream.Create;
  881. try
  882. LbmpTemp.SaveToStream(LmstData);
  883. SetString(Result, PChar(LmstData.Memory), LmstData.Size);
  884. finally
  885. LmstData.Free;
  886. end;
  887. finally
  888. LbmpTemp.Free;
  889. end;
  890. end;
  891. CF_METAFILEPICT:
  892. begin
  893. // global mem contains mf struct
  894. LIntTemp := GetMetaFileBitsEx(PMetafilePict(PData)^.hMF, 0, nil);
  895. if LIntTemp > 0 then
  896. begin
  897. SetLength(Result, LIntTemp);
  898. GetMetaFileBitsEx(PMetafilePict(PData)^.hMF, LIntTemp, @Result[1]);
  899. end;
  900. end;
  901. CF_ENHMETAFILE:
  902. begin
  903. // emf handle
  904. LmefTemp := TMetaFile.Create;
  905. try
  906. LmefTemp.Handle := CopyEnhMetafile(HGlobal, nil);
  907. LmstData := TMemoryStream.Create;
  908. try
  909. LmefTemp.SaveToStream(LmstData);
  910. SetString(Result, PChar(LmstData.Memory), LmstData.Size);
  911. finally
  912. LmstData.Free;
  913. end;
  914. finally
  915. LmefTemp.Free;
  916. end;
  917. end;
  918. else
  919. // format not yet known
  920. SetString(Result, PChar(PData), DataSize);
  921. end;
  922. end;
  923. type
  924. // special dialog for format selection
  925. TFormatSelDialog = class(TForm)
  926. private
  927. LbtnOK: TButton;
  928. LbtnCancel: TButton;
  929. LlbxFormats: TListBox;
  930. LcbxTextAsHex: TCheckBox;
  931. procedure ListDoubleClick(Sender: TObject);
  932. procedure ListSelect(Sender: TObject);
  933. end;
  934. // select a format out of an array of available formats
  935. function SelectClipFormat(const Formats: array of TClipFormat; var Format:
  936. TClipFormat; var TextIsHexData: boolean): boolean;
  937. var
  938. LfrmDialog: TFormatSelDialog;
  939. LIntLoop: integer;
  940. LWrdCurrent: TClipFormat;
  941. LStrFormatName: string;
  942. LszBuffer: array[0..511] of char;
  943. begin
  944. Result := False;
  945. // create and show a dialog for clipboard format selection
  946. LfrmDialog := TFormatSelDialog.CreateNew(Application);
  947. with lfrmDialog do
  948. try
  949. BorderStyle := bsDialog;
  950. Width := Screen.Width div 4;
  951. Height := Screen.Height div 4;
  952. {$IFDEF DELPHI6UP}
  953. Position := poOwnerFormCenter;
  954. {$ELSE}
  955. Position := poScreenCenter;
  956. {$ENDIF}
  957. Caption := SELECT_FORMAT_CAPTION;
  958. LbtnOK := TButton.Create(LfrmDialog);
  959. LbtnCancel := TButton.Create(LfrmDialog);
  960. LcbxTextAsHex := TCheckBox.Create(LfrmDialog);
  961. LlbxFormats := TListBox.Create(LfrmDialog);
  962. try
  963. with lbtnOK do
  964. begin
  965. Parent := LfrmDialog;
  966. ModalResult := mrOk;
  967. Caption := SOKButton;
  968. Default := True;
  969. Width := (LfrmDialog.Width div 2) - 32;
  970. Top := LfrmDialog.ClientHeight - Height - 8;
  971. Left := 16;
  972. Enabled := False;
  973. end;
  974. with LbtnCancel do
  975. begin
  976. Parent := LfrmDialog;
  977. ModalResult := mrCancel;
  978. Cancel := True;
  979. Caption := SCancelButton;
  980. Width := (LfrmDialog.Width div 2) - 32;
  981. Top := LfrmDialog.ClientHeight - Height - 8;
  982. Left := LfrmDialog.ClientWidth - Width - 16;
  983. end;
  984. with LcbxTextAsHex do
  985. begin
  986. Parent := LfrmDialog;
  987. Enabled := False;
  988. Caption := SELECT_FORMAT_ASHEX;
  989. Top := LbtnCancel.Top - Height - 8;
  990. Left := LbtnOK.Left;
  991. Width := LfrmDialog.ClientWidth - Left;
  992. Checked := TextIsHexData;
  993. end;
  994. with LlbxFormats do
  995. begin
  996. Parent := LfrmDialog;
  997. Align := alTop;
  998. Height := LfrmDialog.ClientHeight - 16 - LbtnCancel.Height - 8 -
  999. LcbxTextAsHex.Height;
  1000. OnDblClick := ListDoubleClick;
  1001. OnClick := ListSelect;
  1002. for LIntLoop := Low(Formats) to High(Formats) do
  1003. begin
  1004. LWrdCurrent := Formats[LIntLoop];
  1005. case LWrdCurrent of
  1006. CF_TEXT: LStrFormatName := STR_CF_TEXT;
  1007. CF_BITMAP: LStrFormatName := STR_CF_BITMAP;
  1008. CF_METAFILEPICT: LStrFormatName := STR_CF_METAFILEPICT;
  1009. CF_SYLK: LStrFormatName := STR_CF_SYLK;
  1010. CF_DIF: LStrFormatName := STR_CF_DIF;
  1011. CF_TIFF: LStrFormatName := STR_CF_TIFF;
  1012. CF_OEMTEXT: LStrFormatName := STR_CF_OEMTEXT;
  1013. CF_DIB: LStrFormatName := STR_CF_DIB;
  1014. CF_PALETTE: LStrFormatName := STR_CF_PALETTE;
  1015. CF_PENDATA: LStrFormatName := STR_CF_PENDATA;
  1016. CF_RIFF: LStrFormatName := STR_CF_RIFF;
  1017. CF_WAVE: LStrFormatName := STR_CF_WAVE;
  1018. CF_UNICODETEXT: LStrFormatName := STR_CF_UNICODETEXT;
  1019. CF_ENHMETAFILE: LStrFormatName := STR_CF_ENHMETAFILE;
  1020. CF_HDROP: LStrFormatName := STR_CF_HDROP;
  1021. CF_LOCALE: LStrFormatName := STR_CF_LOCALE;
  1022. else
  1023. SetString(LStrFormatName, LszBuffer,
  1024. GetClipboardFormatName(LWrdCurrent, LszBuffer,
  1025. sizeof(LszBuffer)));
  1026. LStrFormatName := Trim(LStrFormatName);
  1027. end;
  1028. if LStrFormatName = '' then
  1029. LStrFormatName := '(' + IntToRadix(LWrdCurrent, 10) + ')';
  1030. Items.AddObject(LStrFormatName, Pointer(LWrdCurrent));
  1031. LbtnOK.Enabled := True;
  1032. ItemIndex := Items.IndexOfObject(Pointer(LAST_USED_CF));
  1033. if ItemIndex = -1 then
  1034. ItemIndex := 0;
  1035. end;
  1036. end;
  1037. // enable hextext checkbox depending on selected format
  1038. ListSelect(nil);
  1039. if (ShowModal = mrOk) and (LlbxFormats.ItemIndex > -1) then
  1040. begin
  1041. Format := TClipFormat(LlbxFormats.Items.Objects[LlbxFormats.ItemIndex]);
  1042. if Format in [CF_TEXT, CF_OEMTEXT] then
  1043. TextIsHexData := LcbxTextAsHex.Checked;
  1044. Result := True;
  1045. LAST_USED_CF := Format;
  1046. end;
  1047. finally
  1048. // not sure if they automatically get freed?
  1049. LbtnOK.Free;
  1050. LbtnCancel.Free;
  1051. LcbxTextAsHex.Free;
  1052. LlbxFormats.Free;
  1053. end;
  1054. finally
  1055. Free;
  1056. end;
  1057. end;
  1058. // query a data object's supported formats and check if we can "paste" them
  1059. function QueryOLEFormat(const SupportedFormats: array of TClipFormat; const
  1060. dataObj: IDataObject; var Format: TClipFormat; var TextIsHexData: boolean):
  1061. boolean;
  1062. var
  1063. LWrdFormats: array of TClipFormat;
  1064. LIntLoop: integer;
  1065. LobjEnum: TFormatEnum;
  1066. begin
  1067. Result := False;
  1068. LWrdFormats := nil;
  1069. LobjEnum := TFormatEnum.Create(dataObj);
  1070. try
  1071. // enum all available formats
  1072. if Length(SupportedFormats) > 0 then
  1073. begin
  1074. for LIntLoop := Low(SupportedFormats) to High(SupportedFormats) do
  1075. if LObjEnum.HasFormat(SupportedFormats[LIntLoop]) then
  1076. begin
  1077. SetLength(LWrdFormats, Succ(Length(LWrdFormats)));
  1078. LWrdFormats[Pred(Length(LWrdFormats))] := SupportedFormats[LIntLoop];
  1079. end;
  1080. case Length(LWrdFormats) of
  1081. 0: Exit;
  1082. 1:
  1083. begin
  1084. Format := LWrdFormats[0];
  1085. Result := True;
  1086. Exit;
  1087. end;
  1088. else
  1089. // show a dialog for data format selection
  1090. Result := SelectClipFormat(LWrdFormats, Format, TextIsHexData);
  1091. end;
  1092. end;
  1093. finally
  1094. LObjEnum.Free;
  1095. LWrdFormats := nil;
  1096. end;
  1097. end;
  1098. { TMPHexEditorEx }
  1099. // constructor
  1100. constructor TMPHexEditorEx.Create(AOwner: TComponent);
  1101. begin
  1102. inherited;
  1103. FModifiedNoUndo := False;
  1104. FCreateUndoOnUndoUpdate := False;
  1105. FBookmarksNoChange := False;
  1106. FHasDoubleClicked := False;
  1107. FPaintUpdateCounter := 0;
  1108. FClipData := nil;
  1109. FZoomOnWheel := True;
  1110. FCreateBackups := True;
  1111. FBackupFileExt := BACKUP_EXT;
  1112. FOleDragDrop := False;
  1113. FOleStartDrag := False;
  1114. FOleDragging := False;
  1115. FClipboardAsHexText := False;
  1116. FFlushClipboardAtShutDown := False;
  1117. FSupportsOtherClipFormats := True;
  1118. FPrintOptions := TMPHPrintOptions.Create;
  1119. FPrintFont := TFont.Create;
  1120. FPrintFont.Assign(Font);
  1121. FUseEditorFontForPrinting := True;
  1122. FOffsetPopupMenu := nil;
  1123. if not (csDesigning in ComponentState) then
  1124. FDropTarget := TMPHDropTarget.Create(self); // not in delphi ide
  1125. end;
  1126. // destructor
  1127. destructor TMPHexEditorEx.Destroy;
  1128. begin
  1129. // empty or flush clipboard
  1130. ReleaseClipboard(FFlushClipboardAtShutDown);
  1131. FPrintOptions.Free;
  1132. FPrintFont.Free;
  1133. if not (csDesigning in ComponentState) then
  1134. FDropTarget.Free;
  1135. inherited;
  1136. end;
  1137. // cb copy possible
  1138. function TMPHexEditorEx.CanCopy: boolean;
  1139. begin
  1140. Result := (DataSize > 0) and (SelCount > 0);
  1141. end;
  1142. // cb cut possible
  1143. function TMPHexEditorEx.CanCut: boolean;
  1144. begin
  1145. Result := CanCopy and not (ReadOnlyView or NoSizeChange);
  1146. end;
  1147. // cb paste possible
  1148. function TMPHexEditorEx.CanPaste: boolean;
  1149. var
  1150. LifData: IDataObject;
  1151. LIntEffect: integer;
  1152. begin
  1153. LIntEffect := DROPEFFECT_COPY;
  1154. Result := (not (ReadOnlyView (*or NoSizeChange*))) and
  1155. Succeeded(OLEGetClipboard(LifData)) and (SupportsOLEData(LifData, 0,
  1156. Point(0,
  1157. 0), LintEffect, oleClipboard) = S_OK);
  1158. if Result and NoSizeChange then
  1159. Result := DataSize > 0;
  1160. end;
  1161. // copy to clipboard
  1162. function TMPHexEditorEx.CBCopy: boolean;
  1163. begin
  1164. Result := CanCopy;
  1165. if Result then
  1166. begin
  1167. WaitCursor;
  1168. try
  1169. FClipData := TMPHDataObject.CreateFromStream(DataStorage, Min(SelStart,
  1170. SelEnd), SelCount, ExtractFileName(FileName), FClipboardAsHexText,
  1171. SwapNibbles);
  1172. OleCheck(OleSetClipboard(FClipData));
  1173. finally
  1174. OldCursor;
  1175. end;
  1176. end;
  1177. end;
  1178. // cut to clipboard
  1179. function TMPHexEditorEx.CBCut: boolean;
  1180. begin
  1181. Result := CanCut and CBCopy;
  1182. if Result then
  1183. begin
  1184. WaitCursor;
  1185. try
  1186. DeleteSelection(UNDO_CUTCB);
  1187. finally
  1188. OldCursor;
  1189. end;
  1190. end;
  1191. end;
  1192. // paste from clipboard
  1193. function TMPHexEditorEx.CBPaste: boolean;
  1194. var
  1195. LifData: IDataObject;
  1196. LIntEffect: integer;
  1197. begin
  1198. LIntEffect := DROPEFFECT_COPY;
  1199. Result := CanPaste and Succeeded(OLEGetClipboard(LifData)) and
  1200. Succeeded(InsertOLEData(LifData, 0, Point(0, 0), LIntEffect, oleClipboard));
  1201. end;
  1202. // create an undo for a range of bytes
  1203. procedure TMPHexEditorEx.CreateRangeUndo(const aStart, aCount: integer;
  1204. sDesc: string);
  1205. var
  1206. bMod: boolean;
  1207. begin
  1208. bMod := FModified;
  1209. try
  1210. if aCount < 1 then
  1211. CreateUndo(ufKindAllData, 0, 0, 0, sDesc)
  1212. else
  1213. CreateUndo(ufKindReplace, aStart, aCount, aCount, sDesc);
  1214. finally
  1215. FModified := bMod;
  1216. end;
  1217. end;
  1218. function TMPHexEditorEx.BeginUpdate: integer;
  1219. begin
  1220. Inc(FPaintUpdateCounter);
  1221. Result := FPaintUpdateCounter;
  1222. end;
  1223. function TMPHexEditorEx.EndUpdate: integer;
  1224. begin
  1225. Dec(FPaintUpdateCounter);
  1226. if FPaintUpdateCounter < 0 then
  1227. FPaintUpdateCounter := 0;
  1228. if FPaintUpdateCounter = 0 then
  1229. Invalidate;
  1230. Result := FPaintUpdateCounter;
  1231. end;
  1232. // mouse wheel overriding for zooming (font size) if CTRL/SHIFT is pressed,
  1233. // or bytes per line changing if CTRL pressed
  1234. function TMpHexEditorEx.DoMouseWheelDown(Shift: TShiftState; MousePos: TPoint):
  1235. boolean;
  1236. begin
  1237. if FZoomOnWheel and (Shift = [ssCtrl]) and (BytesPerRow > 1) then
  1238. begin
  1239. Result := True;
  1240. BytesPerRow := BytesPerRow - 1;
  1241. Invalidate;
  1242. end
  1243. else if FZoomOnWheel and (Shift = [ssShift, ssCtrl]) and (Font.Size > 2) then
  1244. begin
  1245. Result := True;
  1246. Font.Size := Font.Size - 1;
  1247. end
  1248. else
  1249. Result := inherited DoMouseWheelDown(Shift, MousePos);
  1250. end;
  1251. function TMpHexEditorEx.DoMouseWheelUp(Shift: TShiftState; MousePos: TPoint):
  1252. boolean;
  1253. begin
  1254. if FZoomOnWheel and (Shift = [ssCtrl]) and (BytesPerRow < 256) then
  1255. begin
  1256. Result := True;
  1257. BytesPerRow := BytesPerRow + 1;
  1258. Invalidate;
  1259. end
  1260. else if FZoomOnWheel and (Shift = [ssShift, ssCtrl]) then
  1261. begin
  1262. Result := True;
  1263. Font.Size := Font.Size + 1;
  1264. end
  1265. else
  1266. Result := inherited DoMouseWheelUp(Shift, MousePos);
  1267. end;
  1268. // overwrite key handling
  1269. procedure TMPHexEditorEx.KeyDown(var Key: word; Shift: TShiftState);
  1270. begin
  1271. inherited;
  1272. case Key of
  1273. // CTRL+A: select all
  1274. Ord('A'): if Shift = [ssCtrl] then
  1275. begin
  1276. SelectAll;
  1277. end;
  1278. // CTRL+C: copy to clipboard
  1279. Ord('C'): if (Shift = [ssCtrl]) and CanCopy then
  1280. begin
  1281. CBCopy;
  1282. end;
  1283. // CTRL+X: cut to clipboard
  1284. Ord('X'): if (Shift = [ssCtrl]) and CanCut then
  1285. begin
  1286. CBCut;
  1287. end;
  1288. // CTRL+V: paste from clipboard
  1289. Ord('V'): if (Shift = [ssCtrl]) and CanPaste then
  1290. begin
  1291. CBPaste;
  1292. end;
  1293. // CTRL+T/CTRL*SHIFT+Z: undo, redo
  1294. Ord('Z'):
  1295. begin
  1296. // undo
  1297. if (Shift = [ssCtrl]) and CanUndo then
  1298. begin
  1299. Undo;
  1300. end
  1301. // redo
  1302. else if (Shift = [ssShift, ssCtrl]) and CanRedo then
  1303. begin
  1304. Redo;
  1305. end
  1306. end;
  1307. end;
  1308. end;
  1309. // handle backup creation
  1310. procedure TMPHexEditorEx.PrepareOverwriteDiskFile;
  1311. var
  1312. LStrBackup: string;
  1313. begin
  1314. inherited;
  1315. if (FCreateBackups and Modified) and FileExists(FileName) then
  1316. begin
  1317. LStrBackup := FileName + FBackupFileExt;
  1318. if FileExists(LStrBackup) and not DeleteFile(LStrBackup) then
  1319. raise EMPHexEditor.CreateFmt(ERR_BACKUP_DELETE,
  1320. [LStrBackup, SysErrorMessage(GetLastError)]);
  1321. if (not MoveFile(PChar(FileName), PChar(LStrBackup))) then
  1322. raise EMPHexEditor.CreateFmt(ERR_BACKUP_CREATE,
  1323. [LStrBackup, SysErrorMessage(GetLastError)]);
  1324. end;
  1325. end;
  1326. // save to file (overwrite)
  1327. procedure TMPHexEditorEx.Save;
  1328. begin
  1329. if not HasFile then
  1330. raise EMPHexEditor.Create(ERR_NOFILE);
  1331. SaveToFile(FileName);
  1332. end;
  1333. // prepare ole dragging
  1334. procedure TMPHexEditorEx.MouseDown(Button: TMouseButton; Shift: TShiftState; X,
  1335. Y: integer);
  1336. begin
  1337. inherited;
  1338. if FOleDragDrop and (Button = mbLeft) and MouseOverSelection and (not
  1339. IsSelecting) then
  1340. begin
  1341. FOleStartDrag := True;
  1342. FOleDragging := False;
  1343. FOleDragX := X;
  1344. FOleDragY := Y;
  1345. end
  1346. end;
  1347. // check and eventually do ole dragging
  1348. procedure TMPHexEditorEx.MouseMove(Shift: TShiftState; X, Y: integer);
  1349. var
  1350. LHrsOperation: HRESULT;
  1351. LIntEffect: integer;
  1352. LobjData: TMPHDataObject;
  1353. begin
  1354. inherited;
  1355. if FOleDragDrop and (ssLeft in Shift) and (not FOleDragging) and FOleStartDrag
  1356. and MouseOverSelection and (not IsSelecting) and ((Abs(X - FOleDragX) >=
  1357. Mouse.DragThreshold) or (Abs(Y - FOleDragY) >= Mouse.DragThreshold)) then
  1358. begin
  1359. FOleStartDrag := False;
  1360. FOleDragging := True;
  1361. FoleWasTarget := False;
  1362. // start ole dragging
  1363. try
  1364. LobjData := TMPHDataObject.CreateFromStream(DataStorage, Min(SelStart,
  1365. SelEnd), SelCount, ExtractFileName(FileName), FClipboardAsHexText,
  1366. SwapNibbles);
  1367. if not ReadOnlyView then
  1368. LHrsOperation := DoDragDrop(LobjData, TMPHDropSource.Create,
  1369. DROPEFFECT_COPY or DROPEFFECT_MOVE, LIntEffect)
  1370. else
  1371. LHrsOperation := DoDragDrop(LobjData, TMPHDropSource.Create,
  1372. DROPEFFECT_COPY, LIntEffect);
  1373. // if feedback has given via idataobject.setdata
  1374. if LObjData.FHasDropEffect then
  1375. LIntEffect := LObjData.FDropEffect;
  1376. // unexcpected result
  1377. if (LHrsOperation <> DRAGDROP_S_CANCEL) and (LHrsOperation <>
  1378. DRAGDROP_S_DROP) then
  1379. OLECheck(LHrsOperation)
  1380. else if (LHrsOperation = DRAGDROP_S_DROP) and (LIntEffect =
  1381. DROPEFFECT_MOVE) then
  1382. begin
  1383. // dragged to an other window
  1384. if not FOleWasTarget then
  1385. DeleteSelection
  1386. else
  1387. // dragged to me, so on move, selection is already deleted, create a move undo
  1388. CombineUndo(2, UNDO_MOVED);
  1389. end;
  1390. finally
  1391. FOleDragging := False;
  1392. FOleWasTarget := False;
  1393. HideDragCell;
  1394. end;
  1395. end;
  1396. end;
  1397. // cancel dragging and flags
  1398. procedure TMPHexEditorEx.MouseUp(Button: TMouseButton; Shift: TShiftState; X, Y:
  1399. integer);
  1400. begin
  1401. if FHasDoubleClicked then
  1402. begin
  1403. MouseUpCanResetSel := False;
  1404. FHasDoubleClicked := False;
  1405. end;
  1406. inherited;
  1407. if FOleDragging then
  1408. begin
  1409. FOleDragging := False;
  1410. FOleStartDrag := False;
  1411. end;
  1412. end;
  1413. // don't allow ole dnd in ide or while loading
  1414. procedure TMPHexEditorEx.SetOleDragDrop(const Value: boolean);
  1415. begin
  1416. if Value <> FOleDragDrop then
  1417. begin
  1418. FOleDragDrop := Value;
  1419. if ComponentState * [csLoading, csDesigning] = [] then
  1420. FDropTarget.Active := Value;
  1421. end;
  1422. end;
  1423. // if ole dnd allowed, set new window handle in the drop target
  1424. procedure TMPHexEditorEx.CreateWnd;
  1425. begin
  1426. inherited;
  1427. if not (csDesigning in ComponentState) then
  1428. begin
  1429. FDropTarget.Active := FOleDragDrop;
  1430. end;
  1431. end;
  1432. // insert idataobject data
  1433. function TMPHexEditorEx.InsertOLEData(const dataObj: IDataObject; const
  1434. grfKeyState: longint; const pt: TPoint; var dwEffect: longint; const
  1435. Operation:
  1436. TMPHOLEOperation): HRESULT;
  1437. var
  1438. LRecStg: TStgMedium;
  1439. LStrData, LStrBin: string;
  1440. LIntData, LIntPos, LIntLoop: integer;
  1441. LszBuf: array[0..MAX_PATH] of char;
  1442. LfstFile: TFileStream;
  1443. LPtrLock: Pointer;
  1444. LIntGlobalSize: integer;
  1445. begin
  1446. Result := E_FAIL;
  1447. LStrData := '';
  1448. LIntData := 0;
  1449. WaitCursor;
  1450. try
  1451. // haupt-format?
  1452. if ((FOLEFormat[Operation] = CF_MPHEXEDITOR) or (FOLEFormat[Operation] =
  1453. CF_HDROP)) or QueryOLEFormat(GetMyOLEFormats, dataObj,
  1454. FOLEFormat[Operation], FClipboardAsHexText) then
  1455. begin
  1456. // je nach format daten konvertieren
  1457. case FOLEFormat[Operation] of
  1458. CF_HDROP: if Succeeded(GetIDataObjectData(dataObj,
  1459. FOLEFormat[Operation], LRecStg)) then
  1460. try
  1461. // link: -> put all filenames
  1462. // copy: -> copy contents of first file
  1463. LIntLoop := DragQueryFile(LRecStg.hGlobal, cardinal(-1), nil, 0);
  1464. if LintLoop > 0 then
  1465. begin
  1466. for LIntLoop := 0 to Pred(LIntLoop) do
  1467. begin
  1468. DragQueryFile(LRecStg.hGlobal, LIntLoop, LszBuf,
  1469. sizeof(LszBuf));
  1470. Result := S_OK;
  1471. if dwEffect = DROPEFFECT_LINK then
  1472. begin
  1473. LStrData := LStrData + StrPas(LszBuf) + #0;
  1474. LIntData := Length(LStrData);
  1475. end
  1476. else
  1477. begin
  1478. Result := E_FAIL;
  1479. LfstFile := TFileStream.Create(LszBuf, fmOpenRead or
  1480. fmShareDenyNone);
  1481. try
  1482. SetLength(LStrData, LfstFile.Size);
  1483. LfstFile.ReadBuffer(LStrData[1], LfstFile.Size);
  1484. Result := S_OK;
  1485. LIntData := Length(LStrData);
  1486. Break; // just 1st file
  1487. finally
  1488. LfstFile.Free;
  1489. end;
  1490. end;
  1491. end;
  1492. end;
  1493. finally
  1494. ReleaseStgMedium(LRecStg);
  1495. end;
  1496. else
  1497. // format other than CF_HDROP (=files dropped), retrieve data
  1498. if Succeeded(GetIDataObjectData(dataObj, FOLEFormat[Operation], LRecStg))
  1499. then
  1500. try
  1501. if LRecStg.tymed in [TYMED_HGLOBAL, TYMED_MFPICT] then
  1502. begin
  1503. LPtrLock := GlobalLock(LRecStg.hGlobal);
  1504. LIntGlobalSize := GlobalSize(LRecStg.hGlobal);
  1505. end
  1506. else
  1507. begin
  1508. LPtrLock := nil;
  1509. LIntGlobalSize := 0;
  1510. end;
  1511. try
  1512. LStrData := GetSomeData(LPtrLock, LRecStg.hGlobal,
  1513. FOLEFormat[Operation], LIntGlobalSize, UnicodeBigEndian);
  1514. if LStrData <> '' then
  1515. begin
  1516. LIntData := Length(LStrData);
  1517. if (FOLEFormat[Operation] in [CF_TEXT, CF_OEMTEXT]) and (Operation
  1518. = oleClipboard) and FClipBoardAsHexText then
  1519. begin
  1520. // convert hex text to data
  1521. SetLength(LStrBin, Length(LStrData));
  1522. ConvertHexToBin(@LStrData[1], @LStrBin[1], LIntData,
  1523. SwapNibbles, LIntData);
  1524. LStrData := Copy(LStrBin, 1, LIntData);
  1525. end;
  1526. Result := S_OK;
  1527. end;
  1528. finally
  1529. if Assigned(LPtrLock) then
  1530. GlobalUnlock(LRecStg.hGlobal);
  1531. end;
  1532. finally
  1533. ReleaseStgMedium(LRecStg);
  1534. end
  1535. end;
  1536. CheckUnit(LIntData);
  1537. if (LStrData <> '') and (LIntData > 0) then
  1538. begin
  1539. // insert the data
  1540. case Operation of
  1541. oleDrop:
  1542. begin
  1543. LIntPos := DropPosition;
  1544. if LIntPos < 0 then
  1545. Result := E_FAIL
  1546. else
  1547. begin
  1548. if FOleDragging and (dwEffect = DROPEFFECT_MOVE) then
  1549. begin
  1550. FFixedFileSizeOverride := True;
  1551. try
  1552. // delete selection if we have moved data within ourself
  1553. FOleWasTarget := True;
  1554. if LIntPos > Min(SelStart, SelEnd) then
  1555. Dec(LIntPos, SelCount);
  1556. DeleteSelection;
  1557. if LIntPos >= DataSize then
  1558. Appendbuffer(@LStrData[1], LIntData, UNDO_DROPPED)
  1559. else
  1560. InsertBuffer(@LStrData[1], LIntData, LIntPos,
  1561. UNDO_DROPPED);
  1562. finally
  1563. FFixedFileSizeOverride := False;
  1564. end;
  1565. end
  1566. else
  1567. begin
  1568. if LIntPos >= DataSize then
  1569. begin
  1570. if not NoSizeChange then
  1571. Appendbuffer(@LStrData[1], LIntData, UNDO_DROPPED)
  1572. end
  1573. else
  1574. begin
  1575. if not NoSizeChange then
  1576. begin
  1577. if IsSelected(LIntPos) then
  1578. ReplaceSelection(@LStrData[1], LIntData, UNDO_DROPPED)
  1579. else
  1580. InsertBuffer(@LStrData[1], LIntData, LIntPos,
  1581. UNDO_DROPPED)
  1582. end
  1583. else
  1584. begin
  1585. if (SelCount = 0) or (not IsSelected(LIntPos)) then
  1586. Replace(@LStrData[1], LIntPos, LIntData, LIntData,
  1587. UNDO_DROPPED)
  1588. else
  1589. ReplaceSelection(@LStrData[1], LIntData, UNDO_DROPPED)
  1590. end;
  1591. end;
  1592. end;
  1593. end;
  1594. end;
  1595. oleClipboard: PasteData(PChar(LStrData), LIntData, UNDO_PASTECB);
  1596. end;
  1597. end
  1598. else
  1599. Result := E_FAIL;
  1600. end;
  1601. finally
  1602. LStrData := '';
  1603. OldCursor;
  1604. end;
  1605. if Result <> S_OK then
  1606. dwEffect := DROPEFFECT_NONE;
  1607. end;
  1608. // do we support one of the provided idataobject formats?
  1609. function TMPHexEditorEx.SupportsOLEData(const dataObj: IDataObject; const
  1610. grfKeyState: integer; const pt: TPoint; var dwEffect: integer; const
  1611. Operation:
  1612. TMPHOLEOperation): HRESULT;
  1613. begin
  1614. Result := S_FALSE;
  1615. if (not ReadOnlyView) and OLEHasSupportedFormat(dataObj, GetMyOLEFormats,
  1616. FOLEFormat[Operation]) then
  1617. Result := S_OK;
  1618. if FOLEFormat[Operation] = CF_HDROP then
  1619. if dwEffect = DROPEFFECT_MOVE then
  1620. dwEffect := DROPEFFECT_LINK;
  1621. end;
  1622. function TMPHexEditorEx.OLEHasSupportedFormat(const dataObj: IDataObject; const
  1623. Formats: array of TClipFormat; var Format: TClipFormat): boolean;
  1624. var
  1625. LIntLoop: integer;
  1626. LObjEnum: TFormatEnum;
  1627. begin
  1628. Result := False;
  1629. LObjEnum := TFormatEnum.Create(dataObj);
  1630. try
  1631. if Length(Formats) > 0 then
  1632. for LIntLoop := Low(Formats) to High(Formats) do
  1633. if LObjEnum.HasFormat(Formats[LIntLoop]) then
  1634. begin
  1635. Format := Formats[LIntLoop];
  1636. Result := True;
  1637. Break;
  1638. end;
  1639. finally
  1640. LObjEnum.Free;
  1641. end;
  1642. end;
  1643. // modify effect (move/copy/link) depending on key state and data format
  1644. function TMPHexEditorEx.ModifyOLEDropEffect(const grfKeyState: integer; const
  1645. pt: TPoint; var dwEffect: integer): HRESULT;
  1646. begin
  1647. Result := S_OK;
  1648. if FOleDragging then
  1649. begin
  1650. if ReadOnlyView then
  1651. dwEffect := DROPEFFECT_COPY
  1652. else
  1653. begin
  1654. if Bool(grfKeyState and MK_CONTROL) then
  1655. dwEffect := DROPEFFECT_COPY
  1656. else
  1657. dwEffect := DROPEFFECT_MOVE;
  1658. end;
  1659. end
  1660. else
  1661. begin
  1662. if Bool(grfKeyState and MK_SHIFT) and (not ReadOnlyView) then
  1663. dwEffect := DROPEFFECT_MOVE
  1664. else
  1665. dwEffect := DROPEFFECT_COPY;
  1666. if FOLEFormat[oleDrop] = CF_HDROP then
  1667. if dwEffect = DROPEFFECT_MOVE then
  1668. dwEffect := DROPEFFECT_LINK;
  1669. end;
  1670. end;
  1671. // return a clipformat array with all supported formats
  1672. function TMPHexEditorEx.GetMyOLEFormats: TClipFormats;
  1673. begin
  1674. if FSupportsOtherClipFormats then
  1675. SetLength(Result, 17)
  1676. else
  1677. SetLength(Result, 2);
  1678. Result[0] := CF_MPHEXEDITOR;
  1679. Result[1] := CF_HDROP;
  1680. if FSupportsOtherClipFormats then
  1681. begin
  1682. Result[2] := CF_TEXT;
  1683. Result[3] := CF_RTF;
  1684. Result[4] := CF_UNICODETEXT;
  1685. Result[5] := CF_BITMAP;
  1686. Result[6] := CF_PALETTE;
  1687. Result[7] := CF_METAFILEPICT;
  1688. Result[8] := CF_TIFF;
  1689. Result[9] := CF_OEMTEXT;
  1690. Result[10] := CF_DIB;
  1691. Result[11] := CF_RIFF;
  1692. Result[12] := CF_WAVE;
  1693. Result[13] := CF_ENHMETAFILE;
  1694. Result[14] := CF_LOCALE;
  1695. Result[15] := CF_REGEDIT_HEXDATA;
  1696. Result[16] := CF_HTML;
  1697. end;
  1698. end;
  1699. // reset droptarget helper interface on window destruction
  1700. procedure TMPHexEditorEx.WMDestroy(var Message: TWMDestroy);
  1701. begin
  1702. inherited;
  1703. if ComponentState * [csLoading, csDesigning] = [] then
  1704. FDropTarget.Active := False;
  1705. end;
  1706. // internal
  1707. function TMPHexEditorEx.DumpUndoStorage(const FileName: string): boolean;
  1708. begin
  1709. Result := False;
  1710. if Assigned(UndoStorage) then
  1711. try
  1712. Result := True;
  1713. UndoStorage.SaveToFile(FileName);
  1714. except
  1715. Result := False;
  1716. end;
  1717. end;
  1718. // set new printing options
  1719. procedure TMPHexEditorEx.SetPrintOptions(const Value: TMPHPrintOptions);
  1720. begin
  1721. FPrintOptions.Assign(Value);
  1722. end;
  1723. // internal: draw the specified page to a canvas using the given margins and options
  1724. function TMPHexEditorEx.PrintToCanvas(ACanvas: TCanvas; const APage: integer;
  1725. const AMargins: TRect): integer;
  1726. var
  1727. LObjPrinter: TMPHCanvasPrinter;
  1728. LSetFlags: TMPHPrintFlags;
  1729. begin
  1730. if APage < 0 then
  1731. raise EMPHexEditor.Create(ERR_INVALID_PAGE);
  1732. WaitCursor;
  1733. LSetFlags := FPrintOptions.Flags;
  1734. try
  1735. if SelCount = 0 then
  1736. Exclude(LSetFlags, pfSelectionOnly);
  1737. LObjPrinter := TMPHCanvasPrinter.Create(self, ACanvas, LSetFlags, AMargins,
  1738. FPrintOptions.FHeaders);
  1739. try
  1740. Result := LObjPrinter.Pages;
  1741. if APage > Result then
  1742. raise EMPHexEditor.Create(ERR_INVALID_PAGE);
  1743. if APage > 0 then
  1744. if LObjPrinter.DrawOrCalc(False, APage) < 1 then
  1745. raise EMPHexEditor.Create(ERR_PRINTING_FAILED);
  1746. finally
  1747. LObjPrinter.Free;
  1748. end;
  1749. finally
  1750. OldCursor;
  1751. end;
  1752. end;
  1753. // create a metafile with the selected page as a print preview
  1754. function TMPHexEditorEx.PrintPreview(const Page: integer): TMetaFile;
  1755. var
  1756. LcnvMeta: TMetaFileCanvas;
  1757. LIntHeight, LIntWidth: integer;
  1758. begin
  1759. LIntWidth := GetDeviceCaps(Printer.Handle, HORZRES);
  1760. LIntHeight := GetDeviceCaps(Printer.Handle, VERTRES);
  1761. Result := TMetaFile.Create;
  1762. with Result do
  1763. begin
  1764. Width := LIntWidth;
  1765. Height := LIntHeight;
  1766. LcnvMeta := TMetaFileCanvas.Create(Result, 0);
  1767. with LcnvMeta do
  1768. try
  1769. if FUseEditorFontForPrinting then
  1770. Font.Assign(self.Font)
  1771. else
  1772. Font.Assign(self.FPrintFont);
  1773. SetMapMode(Handle, MM_ANISOTROPIC);
  1774. SetWindowExtEx(Handle, LIntWidth, LIntHeight, nil);
  1775. SetViewPortExtEx(Handle, LIntWidth, LIntHeight, nil);
  1776. Font.Size := Round(Font.Size * GetDeviceCaps(Printer.Handle, LOGPIXELSY) /
  1777. Screen.PixelsPerInch);
  1778. Brush.Style := bsSolid;
  1779. Brush.Color := clWhite;
  1780. FillRect(Rect(0, 0, LIntWidth, LIntHeight));
  1781. FPrintPages := PrintToCanvas(LcnvMeta, Page, PrinterMarginRect);
  1782. finally
  1783. Free;
  1784. end;
  1785. end;
  1786. end;
  1787. // print the given page
  1788. procedure TMPHexEditorEx.Print(const Page: integer);
  1789. var
  1790. LmtfTemp: TMetaFile;
  1791. begin
  1792. if Page < 1 then
  1793. raise EMPHexEditor.Create(ERR_INVALID_PAGE);
  1794. LmtfTemp := PrintPreview(Page);
  1795. with LmtfTemp do
  1796. try
  1797. Printer.Canvas.StretchDraw(Rect(0, 0, Printer.PageWidth,
  1798. Printer.PageHeight), LmtfTemp);
  1799. finally
  1800. Free;
  1801. end;
  1802. end;
  1803. // calculate margins from margins in print options
  1804. function TMPHexEditorEx.PrinterMarginRect: TRect;
  1805. var
  1806. LIntLogX, LIntLogY, LIntPhysWidth, LIntPhysHeight: integer;
  1807. begin
  1808. Result := FPrintOptions.FMargins;
  1809. LIntLogX := GetDeviceCaps(Printer.Handle, LOGPIXELSX);
  1810. // pixels per inch in x dir
  1811. LIntLogY := GetDeviceCaps(Printer.Handle, LOGPIXELSY);
  1812. // pixels per inch in Y dir
  1813. LIntPhysWidth := Printer.PageWidth;
  1814. LIntPhysHeight := Printer.PageHeight;
  1815. Result.Left := Round(Result.Left / 25.4 * LIntLogX);
  1816. Result.Top := Round(Result.Top / 25.4 * LIntLogY);
  1817. Result.Right := LIntPhysWidth - Round(Result.Right / 25.4 * LIntLogX);
  1818. Result.Bottom := LIntPhysHeight - Round(Result.Bottom / 25.4 * LIntLogY);
  1819. end;
  1820. // calculate page count
  1821. function TMPHexEditorEx.PrintNumPages: integer;
  1822. begin
  1823. PrintPreview(0).Free;
  1824. Result := FPrintPages;
  1825. end;
  1826. // empty or flush ole contents in clipboard that have been stored by this instance
  1827. procedure TMPHexEditorEx.ReleaseClipboard(const Flush: boolean);
  1828. begin
  1829. if OwnsClipboard then
  1830. begin
  1831. if Flush then
  1832. OleCheck(OleFlushClipboard)
  1833. else
  1834. OleSetClipboard(nil);
  1835. end;
  1836. end;
  1837. // is there data on the clipboard created by us?
  1838. function TMPHexEditorEx.OwnsClipBoard: boolean;
  1839. begin
  1840. Result := OleIsCurrentClipBoard(FClipData) = S_OK;
  1841. end;
  1842. procedure TMPHexEditorEx.SetPrintFont(const Value: TFont);
  1843. begin
  1844. FPrintFont.Assign(Value);
  1845. FUseEditorFontForPrinting := False;
  1846. end;
  1847. {$IFDEF DELPHI6UP}
  1848. procedure TMPHexEditorEx.DoContextPopup(MousePos: TPoint; var Handled: boolean);
  1849. begin
  1850. inherited;
  1851. if (not Handled) and (Assigned(FOffsetPopupMenu)) then
  1852. begin
  1853. // is mouse over offset col
  1854. with MousePos do
  1855. if ((X > -1) and (X < (ColWidths[0] + ColWidths[1]))) or ((Y > -1) and (Y
  1856. < (RowHeights[0] + RowHeights[1]))) then
  1857. begin
  1858. // in fixed range
  1859. if FOffsetPopupMenu.AutoPopup then
  1860. begin
  1861. Handled := True;
  1862. SendCancelMode(nil);
  1863. FOffsetPopupMenu.PopupComponent := Self;
  1864. MousePos := ClientToScreen(MousePos);
  1865. if InvalidPoint(MousePos) then
  1866. MousePos := ClientToScreen(Point(0, 0));
  1867. FOffsetPopupMenu.Popup(MousePos.X, MousePos.Y);
  1868. end;
  1869. end;
  1870. end
  1871. end;
  1872. {$ENDIF}
  1873. procedure TMPHexEditorEx.SetOffsetPopupMenu(const Value: TPopupMenu);
  1874. begin
  1875. FOffsetPopupMenu := Value;
  1876. if Assigned(Value) then
  1877. with Value do
  1878. begin
  1879. ParentBiDiModeChanged(self);
  1880. FreeNotification(self);
  1881. end;
  1882. end;
  1883. function TMPHexEditorEx.GetOffsetPopupMenu: TPopupMenu;
  1884. begin
  1885. Result := FOffsetPopupMenu;
  1886. end;
  1887. procedure TMPHexEditorEx.Notification(AComponent: TComponent; Operation:
  1888. TOperation);
  1889. begin
  1890. inherited;
  1891. if AComponent = FOffsetPopupMenu then
  1892. if Operation = opRemove then
  1893. OffsetPopupMenu := nil;
  1894. end;
  1895. function TMPHexEditorEx.CanCreateUndo(const aKind: TMPHUndoFlag;
  1896. const aCount, aReplCount: integer): Boolean;
  1897. begin
  1898. Result := inherited CanCreateUndo(aKind, aCount, aReplCount);
  1899. if Result and (UndoStorage.UpdateCount > 0) then
  1900. FModifiedNoUndo := True;
  1901. end;
  1902. function TMPHexEditorEx.GetBookmarksAsString: string;
  1903. procedure AddBM(const bm: TMPHBookmark);
  1904. begin
  1905. SetLength(Result, Length(Result)+sizeof(bm));
  1906. Move(bm, Result[Length(Result)-sizeof(bm)+1], sizeof(bm));
  1907. end;
  1908. var
  1909. LIntLoop: integer;
  1910. LBMSet: boolean;
  1911. begin
  1912. Result := '$';
  1913. LBMSet := False;
  1914. for LIntLoop := Low(TMPHBookMarks) to High(TMPHBookMarks) do
  1915. begin
  1916. AddBM(BookMark[LIntLoop]);
  1917. if (not LBMSet) and (BookMark[LIntLoop].mPosition <> -1) then
  1918. LBMSet := True;
  1919. end;
  1920. if not LBMSet then
  1921. Result := '';
  1922. end;
  1923. procedure TMPHexEditorEx.SetBookMarksAsString(Value: string);
  1924. var
  1925. LIntLoop, LIntCheck, LIntCheck1, LIntPos: integer;
  1926. LBoolChars: boolean;
  1927. LRecBook: TMPHBookMark;
  1928. procedure NewSetBookmarks;
  1929. procedure ExtractBM(const i: integer);
  1930. var
  1931. bm: TMPHBookmark;
  1932. begin
  1933. Move(Value[1], bm, sizeof(bm));
  1934. Delete(Value, 1, sizeof(bm));
  1935. BookMark[i] := bm;
  1936. end;
  1937. var
  1938. LIntLoop: integer;
  1939. begin
  1940. Delete(Value,1,1);
  1941. try
  1942. for LIntLoop := Low(TMPHBookMarks) to High(TMPHBookMarks) do
  1943. ExtractBM(LIntLoop);
  1944. except
  1945. raise EMPHexEditor.Create(ERR_INVALID_BOOKFMT);
  1946. end;
  1947. end;
  1948. begin
  1949. BeginUpdate;
  1950. FBookmarksNoChange := True;
  1951. try
  1952. // empty all bookmarks
  1953. LRecBook.mPosition := -1;
  1954. LRecBook.mInCharField := InCharField;
  1955. for LIntLoop := Low(TMPHBookMarks) to High(TMPHBookMarks) do
  1956. Bookmark[LIntLoop] := LRecBook;
  1957. if Value <> '' then
  1958. begin
  1959. if Value[1] = '$' then
  1960. NewSetBookmarks
  1961. else
  1962. try
  1963. // check sum
  1964. LIntCheck := RadixToInt(Copy(Value, 1, 8), 16);
  1965. Delete(Value, 1, 8);
  1966. // calc check sum
  1967. LIntCheck1 := 0;
  1968. for LIntLoop := 1 to Length(Value) do
  1969. LIntCheck1 := LIntCheck1 + Ord(Value[LIntLoop]);
  1970. if LIntCheck1 <> LIntCheck then
  1971. raise EMPHexEditor.Create(ERR_INVALID_BOOKFMT);
  1972. // set bookmarks
  1973. //for LIntLoop := Low(TMPHBookMarks) to High(TMPHBookMarks) do
  1974. while Value <> '' do
  1975. begin
  1976. LIntLoop := RadixToInt(Copy(Value, 1, 2), 16);
  1977. Delete(Value, 1, 2);
  1978. LIntPos := RadixToInt(Copy(Value, 1, 16), 16);
  1979. Delete(Value, 1, 16);
  1980. LBoolChars := boolean(RadixToInt(Copy(Value, 1, 2), 16));
  1981. Delete(Value, 1, 2);
  1982. LRecBook := Bookmark[LIntLoop];
  1983. if (LRecBook.mPosition <> LIntPos) or (LRecBook.mInCharField <>
  1984. LBoolChars) then
  1985. begin
  1986. LRecBook.mPosition := LIntPos;
  1987. LRecBook.mInCharField := LBoolChars;
  1988. Bookmark[LIntLoop] := LRecBook;
  1989. end;
  1990. end;
  1991. except
  1992. raise EMPHexEditor.Create(ERR_INVALID_BOOKFMT);
  1993. end;
  1994. end;
  1995. finally
  1996. EndUpdate;
  1997. FBookmarksNoChange := False;
  1998. BookmarkChanged;
  1999. end;
  2000. end;
  2001. {$IFDEF DELPHI6UP}
  2002. const
  2003. MPTH_PUBLIC_PROPS: array[0..66] of string = (
  2004. 'ShowRuler',
  2005. 'DrawGutter3D',
  2006. 'CreateBackup',
  2007. 'BackupExtension',
  2008. 'OleDragDrop',
  2009. 'ClipboardAsHexText',
  2010. 'FlushClipboardAtShutDown',
  2011. 'SupportsOtherClipFormats',
  2012. 'UseEditorFontForPrinting',
  2013. 'ZoomOnWheel',
  2014. 'BytesPerRow',
  2015. 'BytesPerColumn',
  2016. 'Translation',
  2017. 'OffsetFormat',
  2018. 'CaretKind',
  2019. 'FocusFrame',
  2020. 'SwapNibbles',
  2021. 'MaskChar',
  2022. 'NoSizeChange',
  2023. 'AllowInsertMode',
  2024. 'DrawGridLines',
  2025. 'ReadOnlyView',
  2026. 'HideSelection',
  2027. 'GraySelectionIfNotFocused',
  2028. 'GutterWidth',
  2029. 'MaxUndo',
  2030. 'InsertMode',
  2031. 'HexLowerCase',
  2032. 'Colors.Background',
  2033. 'Colors.ChangedBackground',
  2034. 'Colors.ChangedText',
  2035. 'Colors.CursorFrame',
  2036. 'Colors.NonFocusCursorFrame',
  2037. 'Colors.Offset',
  2038. 'Colors.OddColumn',
  2039. 'Colors.EvenColumn',
  2040. 'Colors.CurrentOffsetBackground',
  2041. 'Colors.OffsetBackGround',
  2042. 'Colors.CurrentOffset',
  2043. 'Colors.ActiveFieldBackground',
  2044. 'Colors.Grid',
  2045. 'PrintFont.Charset',
  2046. 'PrintFont.Color',
  2047. 'PrintFont.Name',
  2048. 'PrintFont.Size',
  2049. 'PrintFont.Style',
  2050. 'PrintOptions.MarginLeft',
  2051. 'PrintOptions.MarginTop',
  2052. 'PrintOptions.MarginRight',
  2053. 'PrintOptions.MarginBottom',
  2054. 'PrintOptions.PageHeader',
  2055. 'PrintOptions.PageFooter',
  2056. 'PrintOptions.Flags',
  2057. 'Font.Charset',
  2058. 'Font.Color',
  2059. 'Font.Name',
  2060. 'Font.Size',
  2061. 'Font.Style',
  2062. 'BytesPerUnit',
  2063. 'RulerBytesPerUnit',
  2064. 'ShowPositionIfNotFocused',
  2065. 'UnicodeChars',
  2066. 'UnicodeBigEndian',
  2067. 'BytesPerBlock',
  2068. 'SeparateBlocksInCharField',
  2069. 'FindProgress',
  2070. 'RulerNumberBase'
  2071. );
  2072. function TMPHexEditorEx.IsPropPublic(PropName: string): boolean;
  2073. var
  2074. LIntLoop: integer;
  2075. begin
  2076. Result := False;
  2077. for LIntLoop := Low(MPTH_PUBLIC_PROPS) to High(MPTH_PUBLIC_PROPS) do
  2078. if AnsiCompareText(PropName, MPTH_PUBLIC_PROPS[LIntLoop]) = 0 then
  2079. begin
  2080. Result := True;
  2081. Break;
  2082. end;
  2083. if Result and Assigned(FOnQueryPublicProperty) then
  2084. FOnQueryPublicProperty(self, PropName, Result);
  2085. end;
  2086. function TMPHexEditorEx.GetPropertiesAsString: string;
  2087. var
  2088. sl: TStrings;
  2089. procedure Recurse(Ref: TObject; const Prefix: string);
  2090. var
  2091. LPtrProps: PPropList;
  2092. LIntCount: integer;
  2093. begin
  2094. if Ref = nil then
  2095. Exit;
  2096. LIntCount := GetPropList(Ref, LPTrProps);
  2097. if LIntCount > 0 then
  2098. try
  2099. for LIntCount := 0 to Pred(LIntCount) do
  2100. with LPtrProps^[LIntCount]^ do
  2101. if PropType^^.Kind = tkClass then
  2102. Recurse(GetObjectProp(Ref, Name), Prefix + Name + '.')
  2103. else if IsPropPublic(Prefix + Name) then
  2104. sl.Add(Prefix + Name + '=' +string(GetPropValue(Ref, Name)));
  2105. finally
  2106. FreeMem(LPtrProps);
  2107. end;
  2108. end;
  2109. begin
  2110. sl := TStringList.Create;
  2111. try
  2112. Recurse(self, '');
  2113. Result := sl.Text;
  2114. finally
  2115. sl.Free;
  2116. end;
  2117. end;
  2118. {+}
  2119. type
  2120. TStringsPro = class(TStrings);
  2121. {+.}
  2122. procedure TMPHexEditorEx.SetPropertiesAsString(const Value: string);
  2123. var
  2124. LStrData: TStrings;
  2125. LIntLoop, LIntDot: integer;
  2126. LStrProp, LStrVal: string;
  2127. LObjProp: TObject;
  2128. {+}
  2129. function GetValueFromIndex(Index: Integer): string;
  2130. begin
  2131. if Index >= 0 then
  2132. Result := Copy(TStringsPro(LStrData).Get(Index), Length(LStrData.Names[Index]) + 2, MaxInt) else
  2133. Result := '';
  2134. end;
  2135. {+.}
  2136. begin
  2137. BeginUpdate;
  2138. try
  2139. LStrData := TStringList.Create;
  2140. with LStrData do
  2141. try
  2142. Text := Value;
  2143. if Count > 0 then
  2144. for LIntLoop := 0 to Pred(Count) do
  2145. begin
  2146. LStrProp := Names[LIntLoop];
  2147. if IsPropPublic(LStrProp) then
  2148. begin
  2149. {+}
  2150. //LStrVal := ValueFromIndex[LIntLoop];
  2151. LStrVal := GetValueFromIndex(LIntLoop);
  2152. {+.}
  2153. LObjProp := self;
  2154. repeat
  2155. LIntDot := Pos('.', LStrProp);
  2156. if LIntDot > 0 then
  2157. begin
  2158. LObjProp := GetObjectProp(LObjProp, Copy(LStrProp, 1, LIntDot -
  2159. 1));
  2160. System.Delete(LStrProp, 1, LIntDot);
  2161. end;
  2162. until LIntDot = 0;
  2163. if Assigned(LObjProp) then
  2164. SetPropValue(LObjProp, LStrProp, LStrVal);
  2165. end;
  2166. end;
  2167. finally
  2168. Free;
  2169. end;
  2170. finally
  2171. EndUpdate;
  2172. end;
  2173. end;
  2174. {$ENDIF}
  2175. procedure TMPHexEditorEx.Paint;
  2176. begin
  2177. //inherited;
  2178. if FPaintUpdateCounter < 1 then
  2179. inherited;
  2180. end;
  2181. procedure TMPHexEditorEx.DblClick;
  2182. var
  2183. LptMouse: TPoint;
  2184. LIntPos: Integer;
  2185. begin
  2186. // get the position where the mouse is
  2187. Windows.GetCursorPos(LptMouse);
  2188. LptMouse := ScreenToClient(LptMouse);
  2189. with CheckMouseCoord(LptMouse.X, LptMouse.Y) do
  2190. LIntPos := GetPosAtCursor(X, Y);
  2191. if (LIntPos > -1) and (LIntPos < DataSize) then
  2192. begin
  2193. NewSelection(LIntPos, LIntPos);
  2194. FHasDoubleClicked := True;
  2195. MouseUpCanResetSel := False;
  2196. end;
  2197. inherited;
  2198. end;
  2199. procedure TMPHexEditorEx.PasteData(P: Pointer; const ACount: integer;
  2200. const UndoDesc: string);
  2201. var
  2202. LgrcCoords: TGridCoord;
  2203. LIntPos: integer;
  2204. begin
  2205. // assure that we are positionned at the beginning of a unit
  2206. LIntPos := 0;
  2207. if SelCount = 0 then
  2208. begin
  2209. LIntPos := GetPosAtCursor(Col, Row);
  2210. if (LIntpos mod BytesPerUnit) <> 0 then
  2211. begin
  2212. while (LIntPos mod BytesPerUnit) <> 0 do
  2213. Dec(LIntPos);
  2214. LGrcCoords := GetCursorAtPos(LIntPos, InCharField);
  2215. with LGrcCoords do
  2216. begin
  2217. Col := X;
  2218. Row := Y;
  2219. end;
  2220. end;
  2221. end;
  2222. if (SelCount = 0) and NoSizeChange then
  2223. begin
  2224. SelStart := LIntPos;
  2225. SelEnd := Min(DataSize - 1, LIntPos + ACount - 1);
  2226. end;
  2227. ReplaceSelection(P, ACount, UndoDesc);
  2228. end;
  2229. procedure TMPHexEditorEx.BookmarkChanged;
  2230. begin
  2231. if not FBookmarksNoChange then
  2232. inherited;
  2233. end;
  2234. function TMPHexEditorEx.UndoBeginUpdate(const StrUndoDesc: string = ''):
  2235. integer;
  2236. begin
  2237. if (UndoStorage.UpdateCount = 0) and (FCreateUndoOnUndoUpdate or (StrUndoDesc
  2238. <> '')) then
  2239. begin
  2240. FCreateUndoOnUndoUpdate := True;
  2241. CreateRangeUndo(0, 0, StrUndoDesc);
  2242. FModifiedNoUndo := False;
  2243. end;
  2244. Result := inherited UndoBeginUpdate;
  2245. end;
  2246. function TMPHexEditorEx.UndoEndUpdate: integer;
  2247. begin
  2248. Result := inherited UndoEndUpdate;
  2249. if (Result = 0) and FCreateUndoOnUndoUpdate then
  2250. begin
  2251. if FModifiedNoUndo then
  2252. FModifiedNoUndo := False
  2253. else
  2254. begin
  2255. UndoStorage.RemoveLastUndo;
  2256. end;
  2257. end;
  2258. end;
  2259. procedure TMPHexEditorEx.WriteBuffer(const Buffer; const Index,
  2260. Count: Integer);
  2261. begin
  2262. inherited;
  2263. FModified := True;
  2264. if UndoStorage.UpdateCount > 0 then
  2265. FModifiedNoUndo := True;
  2266. end;
  2267. { TMPHDropTarget }
  2268. // constructor
  2269. constructor TMPHDropTarget.Create(Editor: TMPHexEditorEx);
  2270. begin
  2271. inherited Create;
  2272. FEditor := Editor;
  2273. FEditorHandle := 0;
  2274. FActive := False;
  2275. _AddRef; // don't free automatically because it's an object in TMPHexEditorEx
  2276. end;
  2277. // tinterfacedobject auto-destructor hook
  2278. procedure TMPHDropTarget.BeforeDestruction;
  2279. begin
  2280. Dec(FRefCount); // see create above
  2281. Active := False;
  2282. inherited;
  2283. end;
  2284. // do we support data format? if yes, set desired drop effect
  2285. function TMPHDropTarget.DragEnter(const dataObj: IDataObject; grfKeyState:
  2286. integer; pt: TPoint; var dwEffect: integer): HResult;
  2287. begin
  2288. Result := FEditor.SupportsOLEData(dataObj, grfKeyState, pt, dwEffect,
  2289. oleDrop);
  2290. if Result = S_OK then
  2291. begin
  2292. Result := FEditor.ModifyOLEDropEffect(grfKeyState, pt, dwEffect);
  2293. if Result = S_OK then
  2294. begin
  2295. pt := FEditor.ScreenToClient(pt);
  2296. FEditor.ShowDragCell(pt.X, pt.Y)
  2297. end;
  2298. end
  2299. else
  2300. dwEffect := DROPEFFECT_NONE;
  2301. end;
  2302. // dragged out of window
  2303. function TMPHDropTarget.DragLeave: HResult;
  2304. begin
  2305. Result := S_OK;
  2306. FEditor.HideDragCell;
  2307. end;
  2308. // dragging over window
  2309. function TMPHDropTarget.DragOver(grfKeyState: integer; pt: TPoint; var dwEffect:
  2310. integer): HResult;
  2311. begin
  2312. Result := FEditor.ModifyOLEDropEffect(grfKeyState, pt, dwEffect);
  2313. if Result = S_OK then
  2314. begin
  2315. pt := FEditor.ScreenToClient(pt);
  2316. FEditor.ShowDragCell(pt.X, pt.Y)
  2317. end
  2318. else
  2319. begin
  2320. dwEffect := DROPEFFECT_NONE;
  2321. FEditor.HideDragCell;
  2322. end;
  2323. end;
  2324. // dropped!
  2325. function TMPHDropTarget.Drop(const dataObj: IDataObject; grfKeyState: integer;
  2326. pt: TPoint; var dwEffect: integer): HResult;
  2327. begin
  2328. try
  2329. Result := FEditor.SupportsOLEData(dataObj, grfKeyState, pt, dwEffect,
  2330. oleDrop);
  2331. if Result = S_OK then
  2332. begin
  2333. Result := FEditor.ModifyOLEDropEffect(grfKeyState, pt, dwEffect);
  2334. if Result = S_OK then
  2335. try
  2336. Result := FEditor.InsertOLEData(dataObj, grfKeyState, pt, dwEffect,
  2337. oleDrop);
  2338. except
  2339. Result := E_FAIL;
  2340. ShowException(ExceptObject, ExceptAddr);
  2341. end;
  2342. end;
  2343. finally
  2344. FEditor.HideDragCell;
  2345. end;
  2346. end;
  2347. // retrieve window handle from associated hex editor and (de)activate drop target
  2348. procedure TMPHDropTarget.SetActive(const Value: boolean);
  2349. begin
  2350. if FActive <> Value then
  2351. begin
  2352. FActive := Value;
  2353. if not Value then
  2354. begin
  2355. OleCheck(RevokeDragDrop(FEditorHandle));
  2356. OleCheck(CoLockObjectExternal(self, False, True));
  2357. end
  2358. else
  2359. begin
  2360. FEditorHandle := FEditor.Handle;
  2361. OleCheck(RegisterDragDrop(FEditor.Handle, self));
  2362. OleCheck(CoLockObjectExternal(self, True, False));
  2363. end;
  2364. end;
  2365. end;
  2366. { TFormatEnum }
  2367. // constructor
  2368. constructor TFormatEnum.Create(const dataObject: IDataObject);
  2369. var
  2370. LRecFormat: TFormatETC;
  2371. LifEnum: IEnumFormatETC;
  2372. begin
  2373. FFormats := nil;
  2374. if Succeeded(dataObject.EnumFormatEtc(DATADIR_GET, LifEnum)) then
  2375. begin
  2376. while LifEnum.Next(1, LRecFormat, nil) = S_OK do
  2377. begin
  2378. SetLength(FFormats, Succ(Length(FFormats)));
  2379. FFormats[Pred(Length(FFormats))] := LRecFormat;
  2380. end;
  2381. end;
  2382. end;
  2383. // destructor
  2384. destructor TFormatEnum.Destroy;
  2385. begin
  2386. FFormats := nil;
  2387. inherited;
  2388. end;
  2389. // return the desired formatetc struct
  2390. function TFormatEnum.GetFormatETC(const cfFormat: TClipFormat): TFormatETC;
  2391. var
  2392. LBoolOK: boolean;
  2393. LIntLoop: integer;
  2394. begin
  2395. LBoolOK := False;
  2396. if Length(FFormats) > 0 then
  2397. for LIntLoop := 0 to Pred(Length(FFormats)) do
  2398. if FFormats[LIntLoop].cfFormat = cfFormat then
  2399. begin
  2400. LBoolOK := True;
  2401. Result := FFormats[LIntLoop];
  2402. Break;
  2403. end;
  2404. if not LBoolOK then
  2405. FillChar(Result, sizeof(Result), #$FF);
  2406. end;
  2407. // is the desired format available?
  2408. function TFormatEnum.HasFormat(const cfFormat: TClipFormat): boolean;
  2409. var
  2410. LIntLoop: integer;
  2411. begin
  2412. Result := False;
  2413. if Length(FFormats) > 0 then
  2414. for LIntLoop := 0 to Pred(Length(FFormats)) do
  2415. if FFormats[LIntLoop].cfFormat = cfFormat then
  2416. begin
  2417. Result := True;
  2418. Break;
  2419. end;
  2420. end;
  2421. { TMPHEnumFormatETC }
  2422. // constructor
  2423. constructor TMPHEnumFormatETC.Create;
  2424. begin
  2425. inherited Create;
  2426. FIndex := 0;
  2427. with FFormats[0] do
  2428. begin
  2429. cfFormat := CF_MPHEXEDITOR;
  2430. ptd := nil;
  2431. dwAspect := DVASPECT_CONTENT;
  2432. lindex := -1;
  2433. tymed := TYMED_HGLOBAL;
  2434. end;
  2435. with FFormats[1] do
  2436. begin
  2437. cfFormat := CF_TEXT;
  2438. ptd := nil;
  2439. dwAspect := DVASPECT_CONTENT;
  2440. lindex := -1;
  2441. tymed := TYMED_HGLOBAL;
  2442. end;
  2443. with FFormats[2] do
  2444. begin
  2445. cfFormat := CF_FILEDESCRIPTOR;
  2446. ptd := nil;
  2447. dwAspect := DVASPECT_CONTENT;
  2448. lindex := -1;
  2449. tymed := TYMED_HGLOBAL;
  2450. end;
  2451. with FFormats[3] do
  2452. begin
  2453. cfFormat := CF_FILECONTENTS;
  2454. ptd := nil;
  2455. dwAspect := DVASPECT_CONTENT;
  2456. lindex := -1;
  2457. tymed := TYMED_HGLOBAL;
  2458. end;
  2459. end;
  2460. // clone myself
  2461. function TMPHEnumFormatETC.Clone(out Enum: IEnumFormatEtc): HResult;
  2462. begin
  2463. Enum := TMPHEnumFormatETC.Create;
  2464. Result := S_OK;
  2465. end;
  2466. // iterate over all format records
  2467. function TMPHEnumFormatETC.Next(celt: integer; out elt; pceltFetched: PLongint):
  2468. HResult;
  2469. var
  2470. LIntLoop: integer;
  2471. LRecOut: packed array[0..MY_SUPPORTED_FORMATS - 1] of TFormatETC absolute elt;
  2472. begin
  2473. LIntLoop := 0;
  2474. while (LIntLoop < celt) and (FIndex < MY_SUPPORTED_FORMATS) do
  2475. begin
  2476. LRecOut[LIntLoop] := FFormats[FIndex];
  2477. Inc(FIndex);
  2478. Inc(LIntLoop);
  2479. end;
  2480. if pceltFetched <> nil then
  2481. pceltFetched^ := LIntLoop;
  2482. if LIntLoop = celt then
  2483. Result := S_OK
  2484. else
  2485. Result := S_FALSE;
  2486. end;
  2487. // reset iteration
  2488. function TMPHEnumFormatETC.Reset: HResult;
  2489. begin
  2490. FIndex := 0;
  2491. Result := S_OK;
  2492. end;
  2493. // skip entries
  2494. function TMPHEnumFormatETC.Skip(celt: integer): HResult;
  2495. begin
  2496. if (celt < MY_SUPPORTED_FORMATS - FIndex) then
  2497. begin
  2498. FIndex := FIndex + celt;
  2499. Result := S_OK;
  2500. end
  2501. else
  2502. Result := S_FALSE;
  2503. end;
  2504. { TMPHDropSource }
  2505. // use default ole dnd cursors
  2506. function TMPHDropSource.GiveFeedback(dwEffect: integer): HResult;
  2507. begin
  2508. case dwEffect and 7 of
  2509. DROPEFFECT_NONE,
  2510. DROPEFFECT_COPY,
  2511. DROPEFFECT_MOVE: Result := DRAGDROP_S_USEDEFAULTCURSORS;
  2512. else
  2513. Result := E_INVALIDARG;
  2514. end;
  2515. end;
  2516. // standard behaviour
  2517. function TMPHDropSource.QueryContinueDrag(fEscapePressed: BOOL; grfKeyState:
  2518. integer): HResult;
  2519. begin
  2520. if fEscapePressed then
  2521. Result := DRAGDROP_S_CANCEL
  2522. else if (grfKeyState and MK_LBUTTON) = 0 then
  2523. Result := DRAGDROP_S_DROP
  2524. else
  2525. Result := S_OK;
  2526. end;
  2527. { TMPHDataObject }
  2528. // constructor
  2529. constructor TMPHDataObject.Create(Data: Pointer; DataSize: integer;
  2530. ScrapFileName: ShortString; TextAsHex, SwapNibbles: boolean);
  2531. begin
  2532. inherited Create;
  2533. FData := nil;
  2534. FHasDropEffect := False;
  2535. FTextAsHex := TextAsHex;
  2536. FSwapNibbles := SwapNibbles;
  2537. if Assigned(Data) and (DataSize > 0) then
  2538. begin
  2539. FDataSize := DataSize;
  2540. FFileName := Format(STR_SCRAPFILE,
  2541. [ChangeFileExt(ExtractFileName(ScrapFileName), ''),
  2542. ExtractFileExt(ScrapFileName)]);
  2543. GetMem(FData, DataSize);
  2544. Move(Data^, FData^, FDataSize);
  2545. end;
  2546. end;
  2547. constructor TMPHDataObject.CreateFromStream(Stream: TStream; Position, DataSize:
  2548. integer; ScrapFileName: ShortString; TextAsHex, SwapNibbles: boolean);
  2549. begin
  2550. inherited Create;
  2551. FData := nil;
  2552. FHasDropEffect := False;
  2553. FTextAsHex := TextAsHex;
  2554. FSwapNibbles := SwapNibbles;
  2555. if Assigned(Stream) and (DataSize > 0) then
  2556. begin
  2557. FDataSize := DataSize;
  2558. FFileName := Format(STR_SCRAPFILE,
  2559. [ChangeFileExt(ExtractFileName(ScrapFileName), ''),
  2560. ExtractFileExt(ScrapFileName)]);
  2561. GetMem(FData, DataSize);
  2562. Stream.Position := Position;
  2563. Stream.ReadBuffer(FData^, FDataSize);
  2564. end;
  2565. end;
  2566. // destructor hook
  2567. procedure TMPHDataObject.BeforeDestruction;
  2568. begin
  2569. if Assigned(FData) and (FDataSize > 0) then
  2570. FreeMem(FData);
  2571. FData := nil;
  2572. FDataSize := 0;
  2573. inherited;
  2574. end;
  2575. // advise not supported
  2576. function TMPHDataObject.DAdvise(const formatetc: TFormatEtc; advf: integer; const
  2577. advSink: IAdviseSink; out dwConnection: integer): HResult;
  2578. begin
  2579. Result := OLE_E_ADVISENOTSUPPORTED;
  2580. end;
  2581. function TMPHDataObject.DUnadvise(dwConnection: integer): HResult;
  2582. begin
  2583. Result := OLE_E_ADVISENOTSUPPORTED;
  2584. end;
  2585. function TMPHDataObject.EnumDAdvise(out enumAdvise: IEnumStatData): HResult;
  2586. begin
  2587. Result := OLE_E_ADVISENOTSUPPORTED;
  2588. end;
  2589. // create a formetc enumerator, only for getdata
  2590. function TMPHDataObject.EnumFormatEtc(dwDirection: integer; out enumFormatEtc:
  2591. IEnumFormatEtc): HResult;
  2592. begin
  2593. enumFormatETC := nil;
  2594. if dwDirection = DATADIR_GET then
  2595. begin
  2596. enumFormatETC := TMPHEnumFormatETC.Create;
  2597. Result := S_OK;
  2598. end
  2599. else
  2600. Result := E_NOTIMPL;
  2601. end;
  2602. // always same format
  2603. function TMPHDataObject.GetCanonicalFormatEtc(const formatetc: TFormatEtc; out
  2604. formatetcOut: TFormatEtc): HResult;
  2605. begin
  2606. formatetcOut := formatetc;
  2607. formatetcOut.ptd := nil;
  2608. Result := DATA_S_SAMEFORMATETC;
  2609. end;
  2610. // render and return data depending on the desired format
  2611. function TMPHDataObject.GetData(const formatetcIn: TFormatEtc; out medium:
  2612. TStgMedium): HResult;
  2613. var
  2614. LIntDataSize: integer;
  2615. LPtrLocal: PClipData;
  2616. LRecSysTime: TSystemTime;
  2617. begin
  2618. FillChar(medium, sizeof(medium), #0);
  2619. Result := QueryGetData(formatetcIn);
  2620. if Result = S_OK then
  2621. begin
  2622. if formatetcIn.cfFormat = CF_MPHEXEDITOR then
  2623. LIntDataSize := sizeof(TClipData) - 1 + FDataSize
  2624. else if formatetcIn.cfFormat = CF_TEXT then
  2625. begin
  2626. if not FTextAsHex then
  2627. LIntDataSize := Min(FDataSize, StrLen(PChar(FData))) + 1
  2628. else
  2629. LIntDataSize := (FDataSize * 2) + 1;
  2630. end
  2631. else if formatetcIn.cfFormat = CF_FILEDESCRIPTOR then
  2632. LIntDataSize := sizeof(TFileGroupDescriptor)
  2633. else
  2634. LIntDataSize := FDataSize;
  2635. medium.hGlobal := GlobalAlloc(GMEM_SHARE or GMEM_MOVEABLE or GMEM_ZEROINIT,
  2636. LIntDataSize);
  2637. if medium.hGlobal = 0 then
  2638. Result := E_OUTOFMEMORY
  2639. else
  2640. begin
  2641. LPtrLocal := GlobalLock(medium.hGlobal);
  2642. try
  2643. try
  2644. medium.tymed := TYMED_HGLOBAL;
  2645. if formatetcIn.cfFormat = CF_TEXT then
  2646. begin
  2647. if FTextAsHex then
  2648. ConvertBinToHex(FData, PChar(LPtrLocal), FDataSize, FSwapNibbles)
  2649. else
  2650. Move(FData^, LPtrLocal^, LIntDataSize - 1);
  2651. PChar(LPtrLocal)[LIntDataSize - 1] := #0;
  2652. end
  2653. else if formatetcIn.cfFormat = CF_MPHEXEDITOR then
  2654. begin
  2655. LPtrLocal^.Signature := CLIP_SIG;
  2656. LPtrLocal^.Version := CLIP_VER;
  2657. LPtrLocal^.Size := FDataSize;
  2658. Move(FData^, LPtrLocal^.Data, FDataSize);
  2659. end
  2660. else if formatetcIn.cfFormat = CF_FILEDESCRIPTOR then
  2661. begin
  2662. with PFileGroupDescriptor(LPtrLocal)^ do
  2663. begin
  2664. cItems := 1;
  2665. with fgd[0] do
  2666. begin
  2667. dwFlags := FD_FILESIZE or FD_WRITESTIME; // or FD_PROGRESSUI;
  2668. nFileSizeLow := FDataSize;
  2669. nFileSizeHigh := 0;
  2670. GetSystemTime(LRecSysTime);
  2671. SystemTimeToFileTime(LRecSysTime, ftLastWriteTime);
  2672. Move(FFileName[1], cFileName, Min(Length(FFileName),
  2673. sizeof(cFileName) - 1));
  2674. end;
  2675. end;
  2676. end
  2677. else
  2678. begin
  2679. Move(FData^, LPtrLocal^, LIntDataSize);
  2680. end;
  2681. except
  2682. Result := E_OUTOFMEMORY;
  2683. GlobalFree(medium.hGlobal);
  2684. medium.hGlobal := 0;
  2685. end;
  2686. finally
  2687. GlobalUnlock(medium.hGlobal);
  2688. end;
  2689. end;
  2690. end;
  2691. end;
  2692. // what's this?
  2693. function TMPHDataObject.GetDataHere(const formatetc: TFormatEtc;
  2694. out medium: TStgMedium): HResult;
  2695. begin
  2696. Result := DV_E_FORMATETC;
  2697. end;
  2698. // do we support the desired format?
  2699. function TMPHDataObject.QueryGetData(const formatetc: TFormatEtc): HResult;
  2700. begin
  2701. Result := DV_E_FORMATETC;
  2702. with formatetc do
  2703. begin
  2704. if dwAspect <> DVASPECT_CONTENT then
  2705. Result := DV_E_DVASPECT
  2706. else if not Bool(tymed and TYMED_HGLOBAL)
  2707. {// multiple tymeds may be queried (e.g. from explorer, wordpad...)}then
  2708. Result := DV_E_TYMED
  2709. else if (lindex <> -1) and ((cfFormat <> CF_FILECONTENTS) and (Lindex <> 0))
  2710. then
  2711. Result := DV_E_LINDEX
  2712. else if (cfFormat = CF_MPHEXEDITOR) or (cfFormat = CF_TEXT) or
  2713. (cfFormat = CF_FILEDESCRIPTOR) or (cfFormat = CF_FILECONTENTS) then
  2714. Result := S_OK;
  2715. end;
  2716. end;
  2717. // check for dropeffect calls (dodragdrop not always return the real effect)
  2718. function TMPHDataObject.SetData(const formatetc: TFormatEtc; var medium:
  2719. TStgMedium; fRelease: BOOL): HResult;
  2720. var
  2721. LPtrEffect: PDWORD;
  2722. begin
  2723. Result := E_NOTIMPL;
  2724. if ((formatetc.cfFormat = CF_PERFORMEDDROPEFFECT) or (formatetc.cfFormat =
  2725. CF_LOGICALPERFORMEDDROPEFFECT)) and (medium.tymed = TYMED_HGLOBAL) then
  2726. begin
  2727. Result := S_OK;
  2728. // check drop effect
  2729. LPtrEffect := GlobalLock(medium.hGlobal);
  2730. try
  2731. FHasDropEffect := True;
  2732. FDropEffect := LPtrEffect^;
  2733. finally
  2734. GlobalUnLock(medium.hGlobal);
  2735. end;
  2736. end;
  2737. if fRelease then
  2738. ReleaseStgMedium(medium);
  2739. end;
  2740. { TMPHCanvasPrinter }
  2741. // init
  2742. constructor TMPHCanvasPrinter.Create(AEditor: TMPHexEditorEx; ACanvas: TCanvas;
  2743. AFlags: TMPHPrintFlags; AMargins: TRect; AHeaders: TMPHPrintHeaders);
  2744. begin
  2745. inherited Create;
  2746. FMargins := AMargins;
  2747. FCanvas := ACanvas;
  2748. FFlags := AFlags;
  2749. FEditor := AEditor;
  2750. FHeaders[0] := AHeaders[0];
  2751. FHeaders[1] := AHeaders[1];
  2752. GetLinesPerPage;
  2753. end;
  2754. // convert %s variables
  2755. function TMPHCanvasPrinter.BuildHeader(const S: string; const Page: integer):
  2756. string;
  2757. var
  2758. LIntLoop: integer;
  2759. begin
  2760. Result := '';
  2761. LIntLoop := 1;
  2762. while LIntLoop <= Length(S) do
  2763. begin
  2764. if (S[LIntLoop] = '%') and (LIntLoop < Length(S)) then
  2765. begin
  2766. Inc(LIntLoop);
  2767. case S[LIntLoop] of
  2768. 'f': Result := Result + ExtractFileName(FEditor.Filename);
  2769. 'F': Result := Result + FEditor.Filename;
  2770. 'p': Result := Result + IntToRadix(Page, 10);
  2771. 'P': Result := Result + IntToRadix(FPages, 10);
  2772. 't': Result := Result + TimeToStr(now);
  2773. 'd': Result := Result + DateToStr(now);
  2774. '>':
  2775. begin
  2776. if not FEditor.UnicodeChars then
  2777. Result := Result + MPHTranslationDesc
  2778. [FEditor.Translation]
  2779. else
  2780. begin
  2781. if not FEditor.UnicodeBigEndian then
  2782. Result := Result + MPH_UC
  2783. else
  2784. Result := Result + MPH_UC_BE
  2785. end;
  2786. end;
  2787. '<':
  2788. begin
  2789. if not FEditor.UnicodeChars then
  2790. Result := Result +
  2791. MPHTranslationDescShort
  2792. [FEditor.Translation]
  2793. else
  2794. begin
  2795. if not FEditor.UnicodeBigEndian then
  2796. Result := Result + MPH_UC_S
  2797. else
  2798. Result := Result + MPH_UC_BE_S
  2799. end;
  2800. end
  2801. else
  2802. Result := Result + '%' + S[LIntLoop];
  2803. end;
  2804. end
  2805. else
  2806. Result := Result + S[LIntLoop];
  2807. Inc(LIntLoop);
  2808. end;
  2809. end;
  2810. // calculate and draw page
  2811. procedure TMPHCanvasPrinter.Draw(const Page: integer);
  2812. begin
  2813. DrawOrCalc(False, Page);
  2814. end;
  2815. type
  2816. // text and color attributes per character
  2817. TCellAttribute = record
  2818. Back: TColor;
  2819. Fore: TColor;
  2820. Bold: boolean;
  2821. end;
  2822. TCellAttributes = array of TCellAttribute;
  2823. TTextWithAttr = record
  2824. Text: WideString;
  2825. Attributes: TCellAttributes;
  2826. end;
  2827. // calculate lines per page and/or draw page
  2828. function TMPHCanvasPrinter.DrawOrCalc(const JustCalc: boolean; const Page:
  2829. integer): integer;
  2830. // return one line of data
  2831. function GetOneLine(CurPosition, EndPosition: integer; const MinLen: integer):
  2832. TTextWithAttr;
  2833. // add spacer
  2834. procedure AddSpacer(UseDefAttr: boolean = False);
  2835. begin
  2836. Result.Text := Result.Text + ' ';
  2837. SetLength(Result.Attributes, Length(Result.Attributes) + 1);
  2838. if UseDefAttr or (Length(Result.Attributes) = 1) then
  2839. with Result.Attributes[Length(Result.Attributes) - 1] do
  2840. begin
  2841. Bold := False;
  2842. Fore := FEditor.Font.Color;
  2843. Back := FEditor.Colors.Background;
  2844. end
  2845. else
  2846. Result.Attributes[Length(Result.Attributes) - 1] :=
  2847. Result.Attributes[Length(Result.Attributes) - 2]
  2848. end;
  2849. // get hex representation of data (or empty if > datasize)
  2850. function GetByteHex(CurPosition, EndPosition: integer): string;
  2851. begin
  2852. if CurPosition > EndPosition then
  2853. Result := ' '
  2854. else
  2855. begin
  2856. if FEditor.HexLowerCase then
  2857. Result := LowerCase(IntToRadixLen(FEditor.Data[CurPosition], 16, 2))
  2858. else
  2859. Result := UpperCase(IntToRadixLen(FEditor.Data[CurPosition], 16, 2));
  2860. if FEditor.SwapNibbles and (Length(Result) = 2) then
  2861. Result := Result[2] + Result[1];
  2862. end;
  2863. end;
  2864. var
  2865. LIntLoop,
  2866. LIntLoopAttr: integer;
  2867. LStrPart: string;
  2868. LWChrText: WideChar;
  2869. lBold: boolean;
  2870. lOdd: boolean;
  2871. lFore,
  2872. lBack: TColor;
  2873. begin
  2874. Application.ProcessMessages;
  2875. LStrPart := FEditor.GetOffsetString(CurPosition);
  2876. if LStrPart <> '' then
  2877. begin
  2878. LStrPart := StringOfChar(' ', MinLen - Length(LStrPart)) + LStrPart;
  2879. if (not (pfUseBackgroundColor in FFlags)) or (pfMonochrome in FFlags) then
  2880. LStrPart := LStrPart + ':';
  2881. LStrPart := LStrPart + ' ';
  2882. end;
  2883. Result.Text := LStrPart;
  2884. SetLength(Result.Attributes, Length(Result.Text));
  2885. lBold := (FEditor.Row - FEditor.FixedRows) = (CurPosition div
  2886. FEditor.BytesPerRow);
  2887. for lIntLoop := 1 to Length(Result.Text) do
  2888. with Result.Attributes[lIntLoop - 1] do
  2889. begin
  2890. Bold := lBold;
  2891. if (lIntLoop = Length(Result.Text)) or (not (pfUseBackgroundColor in
  2892. FFlags)) then
  2893. begin
  2894. if (lIntLoop = Length(Result.Text)) and (pfUseBackgroundColor in
  2895. FFlags) then
  2896. Bold := False;
  2897. Fore := FEditor.Font.Color;
  2898. Back := FEditor.Colors.Background;
  2899. end
  2900. else
  2901. begin
  2902. if lBold then
  2903. begin
  2904. if (pfUseBackgroundColor in FFlags) and not (pfMonochrome in FFlags)
  2905. then
  2906. Bold := False;
  2907. Fore := FEditor.Colors.CurrentOffset;
  2908. Back := FEditor.Colors.CurrentOffsetBackground;
  2909. end
  2910. else
  2911. begin
  2912. Fore := FEditor.Colors.Offset;
  2913. Back := FEditor.Colors.OffsetBackground;
  2914. end;
  2915. end;
  2916. end;
  2917. LFore := FEditor.Colors.OddColumn;
  2918. if FEditor.InCharField then
  2919. LBack := FEditor.Colors.Background
  2920. else
  2921. lBack := FEditor.Colors.ActiveFieldBackground;
  2922. lOdd := True;
  2923. for LIntLoop := 1 to FEditor.BytesPerRow do
  2924. begin
  2925. LStrPart := GetByteHex(CurPosition - 1 + LIntLoop, EndPosition);
  2926. Result.Text := Result.Text + LStrPart;
  2927. LIntLoopAttr := Length(Result.Attributes);
  2928. SetLength(Result.Attributes, Length(Result.Attributes) +
  2929. Length(LStrPart));
  2930. for LIntLoopAttr := LIntLoopAttr to Pred(Length(Result.Attributes)) do
  2931. with Result.Attributes[LIntLoopAttr] do
  2932. begin
  2933. Bold := FEditor.IsSelected(CurPosition - 1 + LIntLoop);
  2934. Fore := LFore;
  2935. Back := LBack;
  2936. if FEditor.ByteChanged[CurPosition - 1 + LIntLoop] then
  2937. begin
  2938. Fore := FEditor.Colors.ChangedText;
  2939. Back := FEditor.Colors.ChangedBackGround;
  2940. end;
  2941. end;
  2942. if LIntLoop < FEditor.BytesPerRow then
  2943. begin
  2944. if (FEditor.BytesPerBlock > 1) and ((LIntLoop mod FEditor.BytesPerBlock)
  2945. = 0) then
  2946. AddSpacer;
  2947. if (LIntLoop mod FEditor.BytesPerColumn) = 0 then
  2948. begin
  2949. AddSpacer;
  2950. lOdd := not lOdd;
  2951. if lOdd then
  2952. begin
  2953. LFore := FEditor.Colors.OddColumn;
  2954. if FEditor.InCharField then
  2955. LBack := FEditor.Colors.Background
  2956. else
  2957. lBack := FEditor.Colors.ActiveFieldBackground;
  2958. end
  2959. else
  2960. begin
  2961. LFore := FEditor.Colors.EvenColumn;
  2962. if FEditor.InCharField then
  2963. LBack := FEditor.Colors.Background
  2964. else
  2965. lBack := FEditor.Colors.ActiveFieldBackground;
  2966. end;
  2967. end;
  2968. end;
  2969. end;
  2970. AddSpacer(True);
  2971. AddSpacer(True);
  2972. if not FEditor.UnicodeChars then
  2973. begin
  2974. for LIntLoop := 1 to FEditor.BytesPerRow do
  2975. begin
  2976. if (CurPosition + LIntLoop - 1) > EndPosition then
  2977. Result.Text := Result.Text + ' '
  2978. else
  2979. Result.Text := Result.Text +
  2980. FEditor.TranslateToAnsiChar(FEditor.Data[CurPosition + LIntLoop -
  2981. 1]);
  2982. SetLength(Result.Attributes, Length(Result.Attributes) + 1);
  2983. with Result.Attributes[Pred(Length(Result.Attributes))] do
  2984. begin
  2985. Bold := FEditor.IsSelected(CurPosition - 1 + LIntLoop);
  2986. if FEditor.ByteChanged[CurPosition - 1 + LIntLoop] then
  2987. begin
  2988. Fore := FEditor.Colors.ChangedText;
  2989. Back := FEditor.Colors.ChangedBackGround;
  2990. end
  2991. else
  2992. begin
  2993. Fore := FEditor.Font.Color;
  2994. if not FEditor.InCharField then
  2995. Back := FEditor.Colors.Background
  2996. else
  2997. Back := FEditor.Colors.ActiveFieldBackground;
  2998. end;
  2999. end;
  3000. if LIntLoop < FEditor.BytesPerRow then
  3001. begin
  3002. if (FEditor.BytesPerBlock > 1) and FEditor.SeparateBlocksInCharField and
  3003. ((LIntLoop mod FEditor.BytesPerBlock) = 0) then
  3004. AddSpacer;
  3005. if (FEditor.UsedRulerBytesPerUnit <> 1) and
  3006. ((LIntLoop mod FEditor.UsedRulerBytesPerUnit) = 0) then
  3007. AddSpacer;
  3008. end;
  3009. end;
  3010. end
  3011. else
  3012. for LIntLoop := 0 to Pred(FEditor.BytesPerRow) div 2 do
  3013. begin
  3014. if (CurPosition + (LIntLoop * 2)) > EndPosition then
  3015. Result.Text := Result.Text + ' '
  3016. else
  3017. begin
  3018. FEditor.ReadBuffer(LWChrText, CurPosition + (LIntLoop * 2), 2);
  3019. if FEditor.UnicodeBigEndian then
  3020. SwapWideChar(LWChrText);
  3021. if (LWChrText < #256) and (Char(LWChrText) in FEditor.MaskedChars)
  3022. then
  3023. LWChrText := WideChar(FEditor.MaskChar);
  3024. Result.Text := Result.Text + LWChrText;
  3025. end;
  3026. SetLength(Result.Attributes, Length(Result.Attributes) + 1);
  3027. with Result.Attributes[Pred(Length(Result.Attributes))] do
  3028. begin
  3029. Bold := FEditor.IsSelected(CurPosition + (LIntLoop * 2));
  3030. if FEditor.ByteChanged[CurPosition + (LIntLoop * 2)] or
  3031. FEditor.ByteChanged[(CurPosition + (LIntLoop * 2)) + 1] then
  3032. begin
  3033. Fore := FEditor.Colors.ChangedText;
  3034. Back := FEditor.Colors.ChangedBackGround;
  3035. end
  3036. else
  3037. begin
  3038. Fore := FEditor.Font.Color;
  3039. if not FEditor.InCharField then
  3040. Back := FEditor.Colors.Background
  3041. else
  3042. Back := FEditor.Colors.ActiveFieldBackground;
  3043. end;
  3044. end;
  3045. if LIntLoop < FEditor.BytesPerRow then
  3046. begin
  3047. if (FEditor.BytesPerBlock > 1) and FEditor.SeparateBlocksInCharField
  3048. and
  3049. (((LIntLoop + 1) mod (FEditor.BytesPerBlock div 2)) = 0) then
  3050. AddSpacer;
  3051. if (FEditor.UsedRulerBytesPerUnit <> 2) and (((LIntLoop * 2) mod
  3052. FEditor.UsedRulerBytesPerUnit) = 0) then
  3053. AddSpacer;
  3054. end;
  3055. end;
  3056. end;
  3057. // return ruler line
  3058. function GetRulerLine(MinLen: integer): TTextWithAttr;
  3059. // add spacer
  3060. procedure AddSpacer;
  3061. begin
  3062. Result.Text := Result.Text + ' ';
  3063. SetLength(Result.Attributes, Length(Result.Attributes) + 1);
  3064. with Result.Attributes[Length(Result.Attributes) - 1] do
  3065. begin
  3066. Bold := False;
  3067. Fore := FEditor.Colors.Offset;
  3068. Back := FEditor.Colors.OffsetBackground;
  3069. end
  3070. end;
  3071. var
  3072. LIntLoop: integer;
  3073. LStrPart: string;
  3074. lBold: boolean;
  3075. begin
  3076. Application.ProcessMessages;
  3077. if MinLen > 0 then
  3078. begin
  3079. LStrPart := StringOfChar(' ', MinLen);
  3080. if (not (pfUseBackgroundColor in FFlags)) or (pfMonochrome in FFlags) then
  3081. LStrPart := LStrPart + ' ';
  3082. LStrPart := LStrPart + ' ';
  3083. end;
  3084. Result.Text := LStrPart;
  3085. SetLength(Result.Attributes, Length(Result.Text));
  3086. for lIntLoop := 1 to Length(Result.Text) do
  3087. with Result.Attributes[lIntLoop - 1] do
  3088. begin
  3089. Fore := FEditor.Colors.Offset;
  3090. Back := FEditor.Colors.OffsetBackGround;
  3091. end;
  3092. for lIntLoop := 1 to Length(FEditor.FRulerString) do
  3093. begin
  3094. Result.Text := Result.Text + FEditor.FRulerString[lIntLoop];
  3095. SetLength(Result.Attributes, Succ(Length(Result.Attributes)));
  3096. lBold := (FEditor.Col - 1) = lIntLoop;
  3097. with Result.Attributes[Pred(Length(Result.Attributes))] do
  3098. begin
  3099. Bold := lBold;
  3100. if (lIntLoop = Length(Result.Text)) or (not (pfUseBackgroundColor in
  3101. FFlags)) then
  3102. begin
  3103. if (lIntLoop = Length(Result.Text)) and (pfUseBackgroundColor in
  3104. FFlags) then
  3105. Bold := False;
  3106. Fore := FEditor.Font.Color;
  3107. Back := FEditor.Colors.Background;
  3108. end
  3109. else
  3110. begin
  3111. if lBold then
  3112. begin
  3113. if (pfUseBackgroundColor in FFlags) and not (pfMonochrome in FFlags)
  3114. then
  3115. Bold := False;
  3116. Fore := FEditor.Colors.CurrentOffset;
  3117. Back := FEditor.Colors.CurrentOffsetBackground;
  3118. end
  3119. else
  3120. begin
  3121. Fore := FEditor.Colors.Offset;
  3122. Back := FEditor.Colors.OffsetBackground;
  3123. end;
  3124. end;
  3125. end;
  3126. if lIntLoop <> Length(FEditor.FRulerString) then
  3127. begin
  3128. if (FEditor.BytesPerBlock > 1) and ((LIntLoop mod (FEditor.BytesPerBlock *
  3129. 2)) = 0) then
  3130. AddSpacer;
  3131. if (LIntLoop mod (FEditor.BytesPerColumn * 2)) = 0 then
  3132. AddSpacer;
  3133. end;
  3134. end;
  3135. AddSpacer; AddSpacer;
  3136. for lIntLoop := 1 to Length(FEditor.FRulerCharString) do
  3137. begin
  3138. Result.Text := Result.Text + FEditor.FRulerCharString[lIntLoop];
  3139. SetLength(Result.Attributes, Succ(Length(Result.Attributes)));
  3140. lBold := (FEditor.Col - 2 - (FEditor.BytesPerRow * 2)) = lIntLoop;
  3141. with Result.Attributes[Pred(Length(Result.Attributes))] do
  3142. begin
  3143. Bold := lBold;
  3144. if (lIntLoop = Length(Result.Text)) or (not (pfUseBackgroundColor in
  3145. FFlags)) then
  3146. begin
  3147. if (lIntLoop = Length(Result.Text)) and (pfUseBackgroundColor in
  3148. FFlags) then
  3149. Bold := False;
  3150. Fore := FEditor.Font.Color;
  3151. Back := FEditor.Colors.Background;
  3152. end
  3153. else
  3154. begin
  3155. if lBold then
  3156. begin
  3157. if (pfUseBackgroundColor in FFlags) and not (pfMonochrome in FFlags)
  3158. then
  3159. Bold := False;
  3160. Fore := FEditor.Colors.CurrentOffset;
  3161. Back := FEditor.Colors.CurrentOffsetBackground;
  3162. end
  3163. else
  3164. begin
  3165. Fore := FEditor.Colors.Offset;
  3166. Back := FEditor.Colors.OffsetBackground;
  3167. end;
  3168. end;
  3169. end;
  3170. if lIntLoop <> Length(FEditor.FRulerCharString) then
  3171. begin
  3172. if not FEditor.UnicodeChars then
  3173. begin
  3174. if (FEditor.BytesPerBlock > 1) and FEditor.SeparateBlocksInCharField and
  3175. ((LIntLoop mod FEditor.BytesPerBlock) = 0) then
  3176. AddSpacer;
  3177. if (FEditor.UsedRulerBytesPerUnit <> 1) and
  3178. ((LIntLoop mod FEditor.UsedRulerBytesPerUnit) = 0) then
  3179. AddSpacer;
  3180. end
  3181. else
  3182. begin
  3183. if (FEditor.BytesPerBlock > 1) and FEditor.SeparateBlocksInCharField
  3184. and
  3185. ((LIntLoop mod (FEditor.BytesPerBlock div 2)) = 0) then
  3186. AddSpacer;
  3187. if (FEditor.UsedRulerBytesPerUnit <> 2) and ((((LIntLoop-1) * 2) mod
  3188. FEditor.UsedRulerBytesPerUnit) = 0) then
  3189. AddSpacer;
  3190. end;
  3191. end;
  3192. end;
  3193. end;
  3194. // render a header to the canvas
  3195. procedure DrawHeader(const LeftPos, Y, RightPos: integer; StrText: string);
  3196. var
  3197. LStrLeft, LStrCenter, LStrRight: string;
  3198. LIntPipe, LIntOldBKMode, LIntOldAlign: integer;
  3199. LIntRect: TRect;
  3200. begin
  3201. LStrLeft := '';
  3202. LStrCenter := '';
  3203. LStrRight := '';
  3204. LIntPipe := Pos('|', StrText);
  3205. if LIntPipe > 0 then
  3206. begin
  3207. LStrLeft := Copy(StrText, 1, LIntPipe - 1);
  3208. Delete(StrText, 1, LIntPipe);
  3209. LIntPipe := Pos('|', StrText);
  3210. if LIntPipe > 0 then
  3211. begin
  3212. LStrCenter := Copy(StrText, 1, LIntPipe - 1);
  3213. Delete(StrText, 1, LIntPipe);
  3214. if StrText <> '' then
  3215. LStrRight := StrText;
  3216. end
  3217. else
  3218. LStrCenter := StrText;
  3219. end
  3220. else
  3221. LStrLeft := StrText;
  3222. LIntOldAlign := GetTextAlign(FCanvas.Handle);
  3223. LIntOldBKMode := GetBKMode(FCanvas.Handle);
  3224. try
  3225. SetBKMode(FCanvas.Handle, TRANSPARENT);
  3226. LIntRect := Rect(LeftPos, Y, RightPos, Y+FCanvas.TextHeight('Yy'));
  3227. if LStrRight <> '' then
  3228. begin
  3229. SetTextAlign(FCanvas.Handle, TA_TOP or TA_RIGHT);
  3230. ExtTextOut(FCanvas.Handle, RightPos, Y, ETO_CLIPPED,
  3231. @LIntRect, PChar(LStrRight), Length(LStrRight), nil);
  3232. end;
  3233. if LStrCenter <> '' then
  3234. begin
  3235. SetTextAlign(FCanvas.Handle, TA_TOP or TA_CENTER);
  3236. ExtTextOut(FCanvas.Handle, LeftPos + ((RightPos - LeftPos) div 2), Y, ETO_CLIPPED,
  3237. @LIntRect, PChar(LStrCenter), Length(LStrCenter), nil);
  3238. end;
  3239. if LStrLeft <> '' then
  3240. begin
  3241. SetTextAlign(FCanvas.Handle, TA_TOP or TA_LEFT);
  3242. ExtTextOut(FCanvas.Handle, LeftPos, Y, ETO_CLIPPED,
  3243. @LIntRect, PChar(LStrLeft), Length(LStrLeft), nil);
  3244. end;
  3245. finally
  3246. SetTextAlign(FCanvas.Handle, LIntOldAlign);
  3247. SetBKMode(FCanvas.Handle, LIntOldBKMode);
  3248. end;
  3249. end;
  3250. var
  3251. LfntTemp: TFont;
  3252. LRecTextAttr: TTextWithAttr;
  3253. LIntWidth,
  3254. LIntHeight,
  3255. LIntDataPos,
  3256. LIntLeft,
  3257. LIntY,
  3258. LIntMaxY,
  3259. LIntDataEnd,
  3260. LIntDataStart: integer;
  3261. LclrFSave: TColor;
  3262. LclrBSave: TColor;
  3263. LfstSave: TFontStyles;
  3264. LIntLoop: integer;
  3265. LIntMinWidth: integer;
  3266. LRectOut: TRect;
  3267. begin
  3268. FPrintHeaders[0] := BuildHeader(FHeaders[0], Page);
  3269. FPrintHeaders[1] := BuildHeader(FHeaders[1], Page);
  3270. Result := -1;
  3271. if (not Assigned(FEditor)) or (FEditor.DataSize < 1) then
  3272. Exit;
  3273. LIntMinWidth := Length(FEditor.GetOffsetString(FEditor.DataSize));
  3274. if (not JustCalc) and (FLinesPerPage < 1) then
  3275. Exit;
  3276. if (pfSelectionOnly in FFlags) and (FEditor.SelCount > 0) then
  3277. begin
  3278. LIntDataEnd := FEditor.SelEnd;
  3279. LIntDataStart := FEditor.SelStart;
  3280. if LIntDataStart > LIntDataEnd then
  3281. begin
  3282. LIntDataStart := FEditor.SelEnd;
  3283. LIntDataEnd := FEditor.SelStart;
  3284. end;
  3285. end
  3286. else
  3287. begin
  3288. if (pfCurrentViewOnly in FFlags) then
  3289. begin
  3290. LIntDataStart := FEditor.DisplayStart;
  3291. LIntDataEnd := FEditor.DisplayEnd;
  3292. end
  3293. else
  3294. begin
  3295. LIntDataStart := 0;
  3296. LIntDataEnd := Pred(FEditor.DataSize);
  3297. end;
  3298. end;
  3299. if not (JustCalc) then
  3300. LIntDataStart := LIntDataStart + ((Page - 1) * (fLinesPerPage *
  3301. FEditor.BytesPerRow));
  3302. if LIntDataStart > LIntDataEnd then
  3303. Exit;
  3304. Result := 0;
  3305. // länge einer zeile berechnen
  3306. LRecTextAttr := GetOneLine(LIntDataStart, LIntDataEnd, LIntMinWidth);
  3307. LfntTemp := TFont.Create;
  3308. LfntTemp.Assign(FCanvas.Font);
  3309. try
  3310. if (pfMonochrome in FFlags) or (not (pfUseBackgroundColor in FFlags))
  3311. then
  3312. FCanvas.Brush.Color := clWhite
  3313. else
  3314. FCanvas.Brush.Color := FEditor.Colors.Background;
  3315. FCanvas.Brush.Style := bsSolid;
  3316. if (pfMonochrome in FFlags) then
  3317. FCanvas.Font.Color := clBlack
  3318. else
  3319. FCanvas.Font.Color := FEditor.Font.Color;
  3320. if not JustCalc then
  3321. FCanvas.FillRect(FMargins);
  3322. LIntWidth := FCanvas.TextWidth(LRecTextAttr.Text);
  3323. while (LIntWidth > (FMargins.Right - FMargins.Left)) and
  3324. (FCanvas.Font.Size
  3325. > 1) do
  3326. begin
  3327. FCanvas.Font.Size := FCanvas.Font.Size - 1;
  3328. LIntWidth := FCanvas.TextWidth(LRecTextAttr.Text);
  3329. end;
  3330. LIntHeight := FCanvas.TextHeight(LRecTextAttr.Text);
  3331. LIntDataPos := LIntDataStart;
  3332. LIntY := FMargins.Top;
  3333. LIntMaxY := FMargins.Bottom;
  3334. FPrintHeaders[0] := BuildHeader(FHeaders[0], Page);
  3335. FPrintHeaders[1] := BuildHeader(FHeaders[1], Page);
  3336. if FPrintHeaders[0] <> '' then
  3337. begin
  3338. if not JustCalc then
  3339. begin
  3340. DrawHeader(FMargins.Left, LIntY, FMargins.Right, FPrintHeaders[0]);
  3341. FCanvas.MoveTo(FMargins.Left, LIntY + LIntHeight);
  3342. FCanvas.LineTo(FMargins.Right, LIntY + LIntHeight);
  3343. end;
  3344. LIntY := LIntY + LIntHeight + LIntHeight;
  3345. end;
  3346. if FPrintHeaders[1] <> '' then
  3347. LIntMaxY := LIntMaxY - LIntHeight;
  3348. if (pfIncludeRuler in FFlags) and FEditor.ShowRuler then
  3349. begin
  3350. if not JustCalc then
  3351. begin
  3352. LRecTextAttr := GetRulerLine(LIntMinWidth);
  3353. LclrFSave := FCanvas.Font.Color;
  3354. LclrBSave := FCanvas.Brush.Color;
  3355. LfstSave := FCanvas.Font.Style;
  3356. LIntLeft := FMargins.Left;
  3357. for LIntLoop := 1 to Length(LRecTextAttr.Text) do
  3358. begin
  3359. if not (pfMonochrome in FFlags) then
  3360. begin
  3361. FCanvas.Font.Color := LRecTextAttr.Attributes[LIntLoop -
  3362. 1].Fore;
  3363. if pfUseBackgroundColor in FFlags then
  3364. FCanvas.Brush.Color := LRecTextAttr.Attributes[LIntLoop -
  3365. 1].Back
  3366. else
  3367. if LRecTextAttr.Attributes[LIntLoop - 1].Fore = clWhite then
  3368. FCanvas.Font.Color := clBlack;
  3369. end;
  3370. if FFlags * [pfSelectionBold, pfSelectionOnly] = [pfSelectionBold]
  3371. then
  3372. begin
  3373. if FFlags * [pfMonochrome, pfUseBackgroundColor] =
  3374. [pfUseBackGroundColor] then
  3375. begin
  3376. FCanvas.Font.Style := [];
  3377. if LRecTextAttr.Attributes[LIntLoop - 1].Bold then
  3378. begin
  3379. FCanvas.Font.Color := ColorToRGB(FCanvas.Font.Color) xor
  3380. $FFFFFF;
  3381. FCanvas.Brush.Color := ColorToRGB(FCanvas.Brush.Color) xor
  3382. $FFFFFF;
  3383. end;
  3384. end
  3385. else
  3386. begin
  3387. if LRecTextAttr.Attributes[LIntLoop - 1].Bold then
  3388. FCanvas.Font.Style := [fsBold]
  3389. else
  3390. FCanvas.Font.Style := [];
  3391. end;
  3392. end;
  3393. LRectOut := Rect(LIntLeft, LIntY, LIntLeft +
  3394. FCanvas.TextWidth('w'),
  3395. LIntY + LIntHeight);
  3396. if (not (pfUseBackgroundColor in FFlags)) or (pfMonochrome in
  3397. FFlags) then
  3398. LRectOut.Bottom := LIntY + (LIntHeight * 3 div 2);
  3399. ExtTextOutW(FCanvas.Handle, LIntLeft, LIntY, ETO_CLIPPED or
  3400. ETO_OPAQUE, @LRectOut, @LRecTextAttr.Text[LIntLoop],
  3401. 1, nil);
  3402. if (not (pfUseBackgroundColor in FFlags)) or (pfMonochrome in
  3403. FFlags) then
  3404. begin
  3405. FCanvas.MoveTo(LRectOut.Left, LIntY + LIntHeight + 1);
  3406. FCanvas.LineTo(LRectOut.Right + 1, LIntY + LIntHeight + 1);
  3407. end;
  3408. LIntLeft := LRectOut.Right;
  3409. end;
  3410. FCanvas.Font.Color := LclrFSave;
  3411. FCanvas.Brush.Color := LclrBSave;
  3412. FCanvas.Font.Style := LfstSave;
  3413. LRecTextAttr := GetOneLine(LIntDataStart, LIntDataEnd,
  3414. LIntMinWidth);
  3415. end;
  3416. if (not (pfUseBackgroundColor in FFlags)) or (pfMonochrome in FFlags)
  3417. then
  3418. LIntY := LIntY + (LIntHeight * 3 div 2)
  3419. else
  3420. LIntY := LIntY + LIntHeight;
  3421. end;
  3422. while (LIntHeight + LIntY) <= LIntMaxY do
  3423. begin
  3424. if not JustCalc then
  3425. begin
  3426. LclrFSave := FCanvas.Font.Color;
  3427. LclrBSave := FCanvas.Brush.Color;
  3428. LfstSave := FCanvas.Font.Style;
  3429. LIntLeft := FMargins.Left;
  3430. for LIntLoop := 1 to Length(LRecTextAttr.Text) do
  3431. begin
  3432. if not (pfMonochrome in FFlags) then
  3433. begin
  3434. FCanvas.Font.Color := LRecTextAttr.Attributes[LIntLoop -
  3435. 1].Fore;
  3436. if pfUseBackgroundColor in FFlags then
  3437. FCanvas.Brush.Color := LRecTextAttr.Attributes[LIntLoop -
  3438. 1].Back
  3439. else
  3440. if LRecTextAttr.Attributes[LIntLoop - 1].Fore = clWhite then
  3441. FCanvas.Font.Color := clBlack;
  3442. end;
  3443. if FFlags * [pfSelectionBold, pfSelectionOnly] = [pfSelectionBold]
  3444. then
  3445. begin
  3446. if FFlags * [pfMonochrome, pfUseBackgroundColor] =
  3447. [pfUseBackGroundColor] then
  3448. begin
  3449. FCanvas.Font.Style := [];
  3450. if LRecTextAttr.Attributes[LIntLoop - 1].Bold then
  3451. begin
  3452. FCanvas.Font.Color := ColorToRGB(FCanvas.Font.Color) xor
  3453. $FFFFFF;
  3454. FCanvas.Brush.Color := ColorToRGB(FCanvas.Brush.Color) xor
  3455. $FFFFFF;
  3456. end;
  3457. end
  3458. else
  3459. begin
  3460. if LRecTextAttr.Attributes[LIntLoop - 1].Bold then
  3461. FCanvas.Font.Style := [fsBold]
  3462. else
  3463. FCanvas.Font.Style := [];
  3464. end;
  3465. end;
  3466. LRectOut := Rect(LIntLeft, LIntY, LIntLeft +
  3467. FCanvas.TextWidth('w'),
  3468. LIntY + LIntHeight);
  3469. ExtTextOutW(FCanvas.Handle, LIntLeft, LIntY, ETO_CLIPPED or
  3470. ETO_OPAQUE, @LRectOut, @LRecTextAttr.Text[LIntLoop],
  3471. 1, nil);
  3472. LIntLeft := LRectOut.Right;
  3473. end;
  3474. FCanvas.Font.Color := LclrFSave;
  3475. FCanvas.Brush.Color := LclrBSave;
  3476. FCanvas.Font.Style := LfstSave;
  3477. end;
  3478. Inc(Result);
  3479. LIntDataPos := LIntDataPos + FEditor.BytesPerRow;
  3480. if LIntDataPos > LIntDataEnd then
  3481. begin
  3482. Break;
  3483. end;
  3484. if not JustCalc then
  3485. LRecTextAttr := GetOneLine(LIntDataPos, LIntDataEnd, LIntMinWidth);
  3486. LIntY := LIntY + LIntHeight;
  3487. end;
  3488. if FPrintHeaders[1] <> '' then
  3489. if not JustCalc then
  3490. begin
  3491. DrawHeader(FMargins.Left, FMargins.Bottom - LIntHeight,
  3492. FMargins.Right,
  3493. FPrintHeaders[1]);
  3494. FCanvas.MoveTo(FMargins.Left, FMargins.Bottom - LIntHeight);
  3495. FCanvas.LineTo(FMargins.Right, FMargins.Bottom - LIntHeight);
  3496. end;
  3497. finally
  3498. FCanvas.Font.Assign(LfntTemp);
  3499. LfntTemp.Free;
  3500. end;
  3501. end;
  3502. // count number of lines per page (as well as number of pages)
  3503. function TMPHCanvasPrinter.GetLinesPerPage: integer;
  3504. var
  3505. LIntSize: integer;
  3506. LSetTempFlags: TMPHPrintFlags;
  3507. begin
  3508. LSetTempFlags := FFlags;
  3509. Exclude(FFlags, pfSelectionOnly);
  3510. try
  3511. Result := DrawOrCalc(True, 1);
  3512. finally
  3513. FFlags := LSetTempFlags;
  3514. end;
  3515. FLinesPerPage := Result;
  3516. if pfSelectionOnly in FFlags then
  3517. LIntSize := Abs(FEditor.SelStart - FEditor.SelEnd)
  3518. else if pfCurrentViewOnly in FFlags then
  3519. begin
  3520. LIntSize := Abs(FEditor.DisplayEnd - FEditor.DisplayStart);
  3521. end
  3522. else
  3523. LIntSize := FEditor.DataSize;
  3524. while (LIntSize mod FEditor.BytesPerRow) <> 0 do
  3525. Inc(LIntSize);
  3526. LIntSize := LIntSize div FEditor.BytesPerRow;
  3527. while (LIntSize mod FLinesPerPage) <> 0 do
  3528. Inc(LIntSize);
  3529. FPages := LIntSize div FLinesPerPage;
  3530. end;
  3531. { TMPHPrintOptions }
  3532. // init
  3533. constructor TMPHPrintOptions.Create;
  3534. begin
  3535. inherited;
  3536. FMargins := MPH_DEF_PRINT_MARGINS;
  3537. FFlags := [pfMonochrome, pfSelectionBold];
  3538. end;
  3539. // copy props
  3540. procedure TMPHPrintOptions.Assign(Source: TPersistent);
  3541. begin
  3542. inherited;
  3543. if Source is TMPHPrintOptions then
  3544. with TMPHPrintOptions(Source) do
  3545. begin
  3546. self.FMargins := FMargins;
  3547. self.FHeaders := FHeaders;
  3548. self.FFlags := FFlags;
  3549. end;
  3550. end;
  3551. // header/footer
  3552. function TMPHPrintOptions.GetHeader(const Index: integer): string;
  3553. begin
  3554. Result := FHeaders[Index];
  3555. end;
  3556. // margin (mm)
  3557. function TMPHPrintOptions.GetMargin(const Index: integer): integer;
  3558. begin
  3559. case Index of
  3560. 1: Result := FMargins.Left;
  3561. 2: Result := FMargins.Top;
  3562. 3: Result := FMargins.Right;
  3563. else
  3564. Result := FMargins.Bottom;
  3565. end;
  3566. end;
  3567. // set haeder/footer
  3568. procedure TMPHPrintOptions.SetHeader(const Index: integer; const Value:
  3569. string);
  3570. begin
  3571. FHeaders[Index] := Value;
  3572. end;
  3573. // set margin (mm)
  3574. procedure TMPHPrintOptions.SetMargin(const Index, Value: integer);
  3575. begin
  3576. case Index of
  3577. 1: FMargins.Left := Value;
  3578. 2: FMargins.Top := Value;
  3579. 3: FMargins.Right := Value;
  3580. else
  3581. FMargins.Bottom := Value;
  3582. end;
  3583. end;
  3584. { TFormatSelDialog }
  3585. // ok on list doubleclick
  3586. procedure TFormatSelDialog.ListDoubleClick(Sender: TObject);
  3587. begin
  3588. ModalResult := mrOk;
  3589. end;
  3590. // enable checkbox if cf_text or cf_oemtext
  3591. procedure TFormatSelDialog.ListSelect(Sender: TObject);
  3592. begin
  3593. with LlbxFormats do
  3594. LcbxTextAsHex.Enabled := (ItemIndex > -1) and
  3595. (TClipFormat(Items.Objects[ItemIndex]) in [CF_TEXT, CF_OEMTEXT])
  3596. end;
  3597. initialization
  3598. // register clip formats
  3599. OleInitialize(nil);
  3600. CF_MPHEXEDITOR := RegisterClipboardFormat(PChar(MPTH_CF));
  3601. CF_REGEDIT_HEXDATA := RegisterClipboardFormat(CFSTR_REGEDIT_HEXDATA);
  3602. CF_RTF := RegisterClipboardFormat(CFSTR_RTF);
  3603. CF_HTML := RegisterClipboardFormat(CFSTR_HTML);
  3604. CF_FILECONTENTS := RegisterClipboardFormat(CFSTR_FILECONTENTS);
  3605. CF_FILEDESCRIPTOR := RegisterClipboardFormat(CFSTR_FILEDESCRIPTOR);
  3606. CF_PERFORMEDDROPEFFECT := RegisterClipboardFormat(CFSTR_PERFORMEDDROPEFFECT);
  3607. CF_LOGICALPERFORMEDDROPEFFECT :=
  3608. RegisterClipboardFormat(CFSTR_LOGICALPERFORMEDDROPEFFECT);
  3609. finalization
  3610. OleUninitialize;
  3611. end.