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

http://github.com/sanelson/dbdesigner-fork · Pascal · 8853 lines · 6740 code · 836 blank · 1277 comment · 732 complexity · e8dbdc67a837e897f7e36da3b4ef897b MD5 · raw file

  1. (*
  2. TMPHexEditor v 02-06-2006<br>
  3. @author((C) markus stephany, vcl[at]mirkes[dot]de, all rights reserved.)
  4. @abstract(TMPHexEditor displays and edits binary files in hexadecimal notation)
  5. @lastmod(02-06-2006)
  6. credits to :<br><br>
  7. - John Hamm, http://users.snapjax.com/john/<br><br>
  8. - Christophe Le Corfec for introducing the EBCDIC format and the nice idea about
  9. half byte insert/delete<br><br>
  10. - Philippe Chessa for his suggestions about AsText, AsHex and better support for
  11. the french keyboard layout<br><br>
  12. - Daniel Jensen for octal offset display and the INS-key recognition stuff<br><br>
  13. - Shmuel Zeigerman for introducing more flexible offset display formats<br><br>
  14. - Vaf, http://carradio.al.ru for reporting missing delver.inc and suggesting OnChange<br><br>
  15. - Eugene Tarasov for reporting that setting the BytesPerColumn value to 4 at design
  16. time didn't work<br><br>
  17. - FuseBurner for BytesPerUnit/RulerBytesPerUnit related suggestions<br><br>
  18. - Motzi for SyncView/ShowPositionIfNotFocused related suggestions<br><br>
  19. - Martin Hsiao for bcb compatibility and reporting some bugs when moving cursor beyond eof<br><br>
  20. - Miyu for delphi 7 defines<br><br>
  21. - Nils Hoyer for bcb testing and his help on creating a BCB6 package<br><br>
  22. - Skamnitsly S.V for reporting a bug when doubleclicking the ruler bar<br><br>
  23. - Pete Fraser for reporting problems with array properties under BCB<br><br>
  24. - Andrew Novikov for bug reports and suggestions<br><br>
  25. - Al for bug reports<br><br>
  26. - Dieter Köhler for reporting the delphi vcl related CanFocus bug<br><br>
  27. - Piotr Likus for reporting a cardinal&lt;-&gt;integer related bug in the Undo method<br><br>
  28. - Marc Girod for bug reports<br><br>
  29. - Gerd Schwartz for reporting a bug with printing headers/footers that contain long texts
  30. (MPHexEditorEx only)<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. - changed key handling (VK_INSERT, no action if a control key is pressed)<br>
  36. - fixed an access violation in CursorOverSelection when moving the cursor over
  37. a large selection<br>
  38. - added conditional defines for delphi 8 and delphi 2005 in MPDELVER.INC</br>
  39. - removed FastPointer property, use the GetFastPointer function instead (it checks boundaries)<br><br></li>
  40. <li>v 05-23-2005: may 23, 2005<br><br>
  41. - fixed an access violation in the undo storage code when reallocating
  42. memory during storing of undo data<br>
  43. - the secondary focus frame on the hex pane now is painted around the
  44. whole actual data value (4 digits if unicode, 2 digits otherwise)<br>
  45. - added procedure @link(CenterCursorPosition)<br>
  46. - in @link(InsertBuffer) and @link(Replace) now the position parameter
  47. is checked<br><br></li>
  48. <li>v 12-29-2004: december 29, 2004<br><br>
  49. - initialized Result to '' in some string functions/methods to avoid
  50. non empty Result vars at function startup due to compiler
  51. optimizations (particularly on d4), e.g. printing did not work
  52. correctly under d4<br>
  53. - updated some of the sample projects (fixed the broken bcb6 sample,
  54. added printing to the hex viewer and the bcb6 editor sample) <br><br></li>
  55. <li>v 12-28-2004: december 28, 2004<br><br>
  56. - changed the progress event calling part in @link(Find) and
  57. @link(FindWithWildcard) to avoid a division by zero error when working
  58. on files &lt; 500 bytes<br><br></li>
  59. <li>v 12-21-2004: december 21, 2004<br><br>
  60. - changed @link(PrepareFindReplaceData) to avoid an exception when
  61. the string parameter is empty<br><br></li>
  62. <li>v 11-12-2004: november 12, 2004<br><br>
  63. - changed mouse selection in insert mode, now it's more text
  64. editor-like<br>
  65. - @link(Undo) and @link(Redo) disabled when @link(ReadonlyView)
  66. is True<br>
  67. - some small other modifications<br>
  68. <br><br></li>
  69. <li>v 10-26-2004: october 26, 2004<br><br>
  70. - fixed a typecasting bug in the Undo method (integer overflow)<br>
  71. - added some utility functions for unsigned int64 arithmetics (@link(AddU64), @link(TryAddU64),
  72. @link(SubtractU64), @link(TrySubtractU64), @link(MultiplyU64), @link(TryMultiplyU64),
  73. @link(DivideU64), @link(TryDivideU64), @link(ModuloU64), @link(TryModuloU64))
  74. <br><br></li>
  75. <li>v 08-29-2004: august 29, 2004<br><br>
  76. - Added @link(ActiveFieldBackground) color property<br><br></li>
  77. <li>v 08-14-2004: august 14, 2004<br><br>
  78. - the caret was not set properly when switching from
  79. hex to char pane if no data was in the editor <br>
  80. - Added @link(MaskedChars) property<br><br></li>
  81. <li>v 06-15-2004: june 15, 2004<br><br>
  82. - Added @link(DrawDataPosition) and @link(IsDrawDataSelected) properties <br>
  83. - changes in drawing/invalidating to avoid unnecessary painting <br>
  84. - OnMouseDown is now called also if offset pane or ruler are clicked <br>
  85. - if @link(BytesPerUnit) is changed, the selection is reset
  86. if (SelCount mod BytesPerUnit) &lt;&gt; 0 <br>
  87. - if @link(CaretKind) is ckAuto, the caret is a bottom line if
  88. @link(ReadOnlyView) is True<br><br></li>
  89. <li>v 06-10-2004: june 10, 2004<br><br>
  90. - added @link(RulerNumberBase) property <br>
  91. - overwritten CanFocus method due to vcl bug (see
  92. <a href="http://info.borland.com/devsupport/delphi/fixes/delphi4/vcl.html">
  93. http://info.borland.com/devsupport/delphi/fixes/delphi4/vcl.html</a>,
  94. ref 279<br><br></li>
  95. <li>v 06-07-2004: june 07, 2004<br><br>
  96. - fixed a crash ("Grid index out of range") when switching from
  97. unicode <br>
  98. - @link(SyncView) modified to be able to synchronize the view
  99. of editors with different data sizes/layouts, also with offset <br>
  100. - on changing TopRow/LeftCol the caret is repositionned <br>
  101. - overwritten mouse wheel handling to allow page scrolling <br>
  102. - manual handling of MaskChar property streaming due to bug reports
  103. ("Invalid Property Value")<br><br></li>
  104. <li>v 05-30-2004: may 30, 2004<br><br>
  105. - fixed broken CanOpenFile routine (files were always marked read-only)<br><br></li>
  106. <li>v 05-27-2004: may 27, 2004<br><br>
  107. - added @link(IsMaxOffset) property <br>
  108. - the control gets focused when the mouse is clicked even when
  109. the mouse is over the selection<br><br></li>
  110. <li>v 05-13-2004: may 13, 2004<br><br>
  111. - @link(OnDrawCell) is now also called for the top left cell<br>
  112. - setting @link(UnicodeChars) to False now correctly sets
  113. @link(BytesPerUnit) to 1 <br><br></li>
  114. <li>v 04-18-2004: april 18, 2004<br><br>
  115. - parameters aBuffer and bBuffer were interchanged in the
  116. CopyMemory call in @link(TranslateBufferFromAnsi)<br>
  117. - @link(GetOffsetString) can now be called in @link(OnGetOffsetText)
  118. without crashing (infinite recursion = stack overflow) <br><br></li>
  119. <li>v 01-08-2004: january 08, 2004<br><br>
  120. - added some explicit pointer typecasts for {$T+} compatibility<br>
  121. - removed FindTable and FindTableI properties under BCB (doesn't
  122. compile) <br><br></li>
  123. <li>v 12-16-2003: december 16, 2003<br><br>
  124. - Setting the @link(DataSize) property is now undoable<br>
  125. - Added the public @link(SetDataSizeFillByte) property to be able to control
  126. what byte is used to enlarge the data<br>
  127. - Now checking @link(NoSizeChange) before allowing to set @link(DataSize)<br>
  128. - CreateUndo is no longer a function, but a procedure. Now an
  129. exception is raised when no undo record can be created <br><br></li>
  130. <li>v 12-10-2003: december 10, 2003<br><br>
  131. - Renamed OnLoadSaveProgress to @link(OnProgress)<br>
  132. - Added property @link(FindProgress)<br>
  133. - Added custom find methods (@link(OnFind), @link(OnWildcardFind)<br>
  134. - @link(Find) and @link(FindWithWildcard) speeded up by using
  135. precompiled character tables<br>
  136. - @link(Find) and @link(FindWithWildcard) now also fire the @link(OnProgress) event
  137. if @link(FindProgress) is set to true<br>
  138. - fixed a bug in mouse handling (weird selection or line offsets when
  139. doublecklicking ruler bar/offset panel) <br>
  140. - modified selectioncode to better support double byte selection (unicode) <br><br></li>
  141. <li>v 09-24-2003: september 24, 2003<br><br>
  142. - modified the BCB6 package<br><br></li>
  143. <li>v 09-09-2003: september 09, 2003<br><br>
  144. - changed some constants, classes and types from MPTH... to MPH...<br>
  145. - changed MPHCustTransFieldFrom/To to @link(MPHCustomCharConv)<br>
  146. - @link(BytesPerBlock) and @link(SeparateBlocksInCharField) properties added<br>
  147. - @link(DataSize) property is writeable now<br>
  148. - Page down key now also reaches the last row<br>
  149. - added @link(OnGetOffsetText) property<br>
  150. - added @link(AddSelectionUndo) procedure<br>
  151. - added defines for delphi7, renamed delver.inc to mpdelver.inc<br>
  152. - added wildcard search (@link(FindWithWildcard))<br>
  153. - added @link(SeekToEOF)<br>
  154. - changed keyboard handling, so OnKeyDown should work better<br>
  155. - added @link(GotoBookmark) method to set cursor to a bookmarked position<br>
  156. - added @link(OnBookmarkChanged) property<br>
  157. - support for unsigned int64 radix conversions<br>
  158. - @link(Replace) method added<br><br></li>
  159. <li>v 07-05-2003: july 05, 2003<br><br>
  160. - better handling of odd sized files when BytesPerUnit &lt;&gt; 1<br>
  161. - added support for pasting clipboard data in fixed filesize mode in @link(TMPHexEditorEx)<br>
  162. - added RegEdit_HexData clipboard support in @link(TMPHexEditorEx)<br><br></li>
  163. <li>v 05-25-2003-b: may 25, 2003<br><br>
  164. - fixed a bug (moving the cursor beyond eof)<br><br></li>
  165. <li>v 05-25-2003: may 25, 2003<br><br>
  166. - added some kind of ownerdraw (see @link(OnDrawCell))<br><br></li>
  167. <li>v 05-20-2003: may 20, 2003<br><br>
  168. - renamed, added and changed some methods, classes and properties<br>
  169. - fixed some bugs in the ruler display (e.g. when BytesPerRow is
  170. changed)<br>
  171. - fixed some bugs when BytesPerUnit &lt;&gt; 1<br>
  172. - added some unicode support (@link(UnicodeChars) and
  173. @link(UnicodeBigEndian))<br>
  174. - fixed some half byte (nibble) related bugs<br><br></li>
  175. <li>v 05-17-2003: may 17, 2003<br><br>
  176. - added @link(DisplayStart) and @link(DisplayEnd) functions to retrieve
  177. the data bounds currently displayed<br>
  178. - added @link(BytesPerUnit) and @link(RulerBytesPerUnit) properties to
  179. treat words/dwords/qwords as a unit<br>
  180. - added @link(SyncView) procedure and @link(OnSelectionChanged)
  181. property to synchronize position and selection with another
  182. editor<br>
  183. - added @link(ShowPositionIfNotFocused) property to show the current
  184. position if the editor is not focused<br><br></li>
  185. <li>v 10-25-2002: october 25, 2002<br><br>
  186. - corrected the BytesPerColumn default value<br><br></li>
  187. <li>v 08-18-2002: august 18, 2002<br><br>
  188. - modified painting and selection<br>
  189. - implemented an additional ruler bar at the top<br>
  190. - new properties: @link(ShowRuler), @link(DrawGutter3D)<br><br></li>
  191. <li>v 08-12-2002: august 12, 2002<br><br>
  192. - modified Changed calls to get correct Modified property in
  193. OnChange handler<br><br></li>
  194. <li>v 08-09-2002: august 09, 2002<br><br>
  195. - included missing include file delver.inc<br>
  196. - added OnChange event<br><br></li>
  197. <li>v 07-21-2002: july 21, 2002<br><br>
  198. too many changes to mention here (completely rewritten, basic and advanced versions
  199. TMPHexEditor and TMPHexEditorEx), plz read the documentation included with this
  200. package for more information</li>
  201. </ul></p>
  202. *)
  203. unit MPHexEditor;
  204. {$R *.res}
  205. {.$DEFINE TINYHEXER}// don't define this!
  206. {$DEFINE FASTACCESS} // if this is defined, direct access to the stream memory is given
  207. (* define this if you want to have the old savetostream behaviour
  208. (clear target stream before copying data).
  209. if it is undef'd, do not clear the target stream
  210. (just copy the editor data to the stream) *)
  211. {.$DEFINE OLD_STREAM_OUT}
  212. {$I MPDELVER.INC}
  213. interface
  214. uses
  215. Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms,
  216. Grids;
  217. type
  218. // @exclude
  219. TGridCoord = Grids.TGridCoord;
  220. // character conversion type
  221. TMPHCharConvType = (cctFromAnsi, cctToAnsi);
  222. // character conversion table
  223. TMPHCharConvTable = array[0..255] of Char;
  224. // character conversion data storage
  225. TMPHCharConv = array[TMPHCharConvType] of TMPHCharConvTable;
  226. const
  227. // block size in file i/o
  228. MPH_FILEIO_BLOCKSIZE = $F000;
  229. // this message is posted to the hex editor when it should update the caret position
  230. CM_INTUPDATECARET = CM_BASE + $100;
  231. // this message is posted when an OnSelectionChange event is to be fired
  232. CM_SELECTIONCHANGED = CM_BASE + $101;
  233. (* translation tables from/to ms windows ansi (~ MS Latin-1) *)
  234. // macintosh..ms ansi conversion
  235. MPH_CCONV_MAC: TMPHCharConv = (
  236. //ansi to mac
  237. (#$00, #$01, #$02, #$03, #$04, #$05, #$06, #$07, #$08, #$09, #$0A, #$0B,
  238. #$0C, #$0D, #$0E, #$0F,
  239. #$10, #$11, #$12, #$13, #$14, #$15, #$16, #$17, #$18, #$19, #$1A, #$1B,
  240. #$1C, #$1D, #$1E, #$1F,
  241. #$20, #$21, #$22, #$23, #$24, #$25, #$26, #$27, #$28, #$29, #$2A, #$2B,
  242. #$2C, #$2D, #$2E, #$2F,
  243. #$30, #$31, #$32, #$33, #$34, #$35, #$36, #$37, #$38, #$39, #$3A, #$3B,
  244. #$3C, #$3D, #$3E, #$3F,
  245. #$40, #$41, #$42, #$43, #$44, #$45, #$46, #$47, #$48, #$49, #$4A, #$4B,
  246. #$4C, #$4D, #$4E, #$4F,
  247. #$50, #$51, #$52, #$53, #$54, #$55, #$56, #$57, #$58, #$59, #$5A, #$5B,
  248. #$5C, #$5D, #$5E, #$5F,
  249. #$60, #$61, #$62, #$63, #$64, #$65, #$66, #$67, #$68, #$69, #$6A, #$6B,
  250. #$6C, #$6D, #$6E, #$6F,
  251. #$70, #$71, #$72, #$73, #$74, #$75, #$76, #$77, #$78, #$79, #$7A, #$7B,
  252. #$7C, #$7D, #$7E, #$7F,
  253. #$C4, #$C5, #$AB, #$C9, #$D1, #$F7, #$DC, #$E1, #$E0, #$E2, #$E4, #$E3,
  254. #$AC, #$B0, #$AA, #$F8,
  255. #$D5, #$CE, #$C3, #$CF, #$D3, #$D4, #$D2, #$DB, #$DA, #$DD, #$F6, #$F5,
  256. #$FA, #$F9, #$FB, #$FC,
  257. #$A0, #$C1, #$A2, #$A3, #$DF, #$B4, #$B6, #$A4, #$C6, #$A9, #$BB, #$C7,
  258. #$C2, #$AD, #$A8, #$FF,
  259. #$A1, #$B1, #$B2, #$B3, #$A5, #$B5, #$A6, #$B7, #$B8, #$B9, #$BC, #$C8,
  260. #$BA, #$BD, #$CA, #$C0,
  261. #$CB, #$E7, #$E5, #$CC, #$80, #$81, #$AE, #$82, #$E9, #$83, #$E6, #$E8,
  262. #$ED, #$EA, #$EB, #$EC,
  263. #$D0, #$84, #$F1, #$EE, #$EF, #$CD, #$85, #$D7, #$AF, #$F4, #$F2, #$F3,
  264. #$86, #$D9, #$DE, #$A7,
  265. #$88, #$87, #$89, #$8B, #$8A, #$8C, #$BE, #$8D, #$8F, #$8E, #$90, #$91,
  266. #$93, #$92, #$94, #$95,
  267. #$F0, #$96, #$98, #$97, #$99, #$9B, #$9A, #$D6, #$BF, #$9D, #$9C, #$9E,
  268. #$9F, #$FD, #$FE, #$D8
  269. ),
  270. // mac to ansi
  271. (#$00, #$01, #$02, #$03, #$04, #$05, #$06, #$07, #$08, #$09, #$0A, #$0B,
  272. #$0C, #$0D, #$0E, #$0F,
  273. #$10, #$11, #$12, #$13, #$14, #$15, #$16, #$17, #$18, #$19, #$1A, #$1B,
  274. #$1C, #$1D, #$1E, #$1F,
  275. #$20, #$21, #$22, #$23, #$24, #$25, #$26, #$27, #$28, #$29, #$2A, #$2B,
  276. #$2C, #$2D, #$2E, #$2F,
  277. #$30, #$31, #$32, #$33, #$34, #$35, #$36, #$37, #$38, #$39, #$3A, #$3B,
  278. #$3C, #$3D, #$3E, #$3F,
  279. #$40, #$41, #$42, #$43, #$44, #$45, #$46, #$47, #$48, #$49, #$4A, #$4B,
  280. #$4C, #$4D, #$4E, #$4F,
  281. #$50, #$51, #$52, #$53, #$54, #$55, #$56, #$57, #$58, #$59, #$5A, #$5B,
  282. #$5C, #$5D, #$5E, #$5F,
  283. #$60, #$61, #$62, #$63, #$64, #$65, #$66, #$67, #$68, #$69, #$6A, #$6B,
  284. #$6C, #$6D, #$6E, #$6F,
  285. #$70, #$71, #$72, #$73, #$74, #$75, #$76, #$77, #$78, #$79, #$7A, #$7B,
  286. #$7C, #$7D, #$7E, #$7F,
  287. #$C4, #$C5, #$C7, #$C9, #$D1, #$D6, #$DC, #$E1, #$E0, #$E2, #$E4, #$E3,
  288. #$E5, #$E7, #$E9, #$E8,
  289. #$EA, #$EB, #$ED, #$EC, #$EE, #$EF, #$F1, #$F3, #$F2, #$F4, #$F6, #$F5,
  290. #$FA, #$F9, #$FB, #$FC,
  291. #$A0, #$B0, #$A2, #$A3, #$A7, #$B4, #$B6, #$DF, #$AE, #$A9, #$8E, #$82,
  292. #$8C, #$AD, #$C6, #$D8,
  293. #$8D, #$B1, #$B2, #$B3, #$A5, #$B5, #$A6, #$B7, #$B8, #$B9, #$BC, #$AA,
  294. #$BA, #$BD, #$E6, #$F8,
  295. #$BF, #$A1, #$AC, #$92, #$80, #$81, #$A8, #$AB, #$BB, #$83, #$BE, #$C0,
  296. #$C3, #$D5, #$91, #$93,
  297. #$D0, #$84, #$96, #$94, #$95, #$90, #$F7, #$D7, #$FF, #$DD, #$98, #$97,
  298. #$86, #$99, #$DE, #$A4,
  299. #$88, #$87, #$89, #$8B, #$8A, #$C2, #$CA, #$C1, #$CB, #$C8, #$CD, #$CE,
  300. #$CF, #$CC, #$D3, #$D4,
  301. #$F0, #$D2, #$DA, #$DB, #$D9, #$9B, #$9A, #$85, #$8F, #$9D, #$9C, #$9E,
  302. #$9F, #$FD, #$FE, #$AF
  303. )
  304. );
  305. // ebcdic cp38..ms ansi conversion
  306. MPH_CCONV_BCD38: TMPHCharConv = (
  307. //ansi to bcd (taken from recode 3.5)
  308. (#$00, #$01, #$02, #$03, #$37, #$2D, #$2E, #$2F, #$16, #$05, #$25, #$0B,
  309. #$0C, #$0D, #$0E, #$0F,
  310. #$10, #$11, #$12, #$13, #$3C, #$3D, #$32, #$26, #$18, #$19, #$3F, #$27,
  311. #$1C, #$1D, #$1E, #$1F,
  312. #$40, #$4F, #$7F, #$7B, #$5B, #$6C, #$50, #$7D, #$4D, #$5D, #$5C, #$4E,
  313. #$6B, #$60, #$4B, #$61,
  314. #$F0, #$F1, #$F2, #$F3, #$F4, #$F5, #$F6, #$F7, #$F8, #$F9, #$7A, #$5E,
  315. #$4C, #$7E, #$6E, #$6F,
  316. #$7C, #$C1, #$C2, #$C3, #$C4, #$C5, #$C6, #$C7, #$C8, #$C9, #$D1, #$D2,
  317. #$D3, #$D4, #$D5, #$D6,
  318. #$D7, #$D8, #$D9, #$E2, #$E3, #$E4, #$E5, #$E6, #$E7, #$E8, #$E9, #$4A,
  319. #$E0, #$5A, #$5F, #$6D,
  320. #$79, #$81, #$82, #$83, #$84, #$85, #$86, #$87, #$88, #$89, #$91, #$92,
  321. #$93, #$94, #$95, #$96,
  322. #$97, #$98, #$99, #$A2, #$A3, #$A4, #$A5, #$A6, #$A7, #$A8, #$A9, #$C0,
  323. #$20, #$D0, #$A1, #$07,
  324. #$80, #$22, #$62, #$63, #$64, #$65, #$66, #$67, #$68, #$69, #$8A, #$8B,
  325. #$8C, #$8D, #$8E, #$8F,
  326. #$90, #$77, #$2C, #$0A, #$3B, #$3E, #$1A, #$70, #$71, #$72, #$9A, #$9B,
  327. #$9C, #$9D, #$9E, #$9F,
  328. #$A0, #$15, #$73, #$74, #$75, #$76, #$6A, #$78, #$09, #$3A, #$AA, #$AB,
  329. #$AC, #$AD, #$AE, #$AF,
  330. #$B0, #$B1, #$B2, #$B3, #$B4, #$B5, #$B6, #$B7, #$B8, #$B9, #$BA, #$BB,
  331. #$BC, #$BD, #$BE, #$BF,
  332. #$23, #$41, #$42, #$43, #$44, #$45, #$46, #$47, #$48, #$49, #$CA, #$CB,
  333. #$CC, #$CD, #$CE, #$CF,
  334. #$1B, #$24, #$06, #$14, #$28, #$2B, #$21, #$17, #$51, #$52, #$DA, #$DB,
  335. #$DC, #$DD, #$DE, #$DF,
  336. #$2A, #$E1, #$53, #$54, #$55, #$56, #$57, #$58, #$59, #$29, #$EA, #$EB,
  337. #$EC, #$ED, #$EE, #$EF,
  338. #$30, #$31, #$08, #$33, #$34, #$35, #$36, #$04, #$38, #$39, #$FA, #$FB,
  339. #$FC, #$FD, #$FE, #$FF
  340. ),
  341. // bcd to ansi (taken from recode 3.5)
  342. (#$00, #$01, #$02, #$03, #$F7, #$09, #$D2, #$7F, #$F2, #$A8, #$93, #$0B,
  343. #$0C, #$0D, #$0E, #$0F,
  344. #$10, #$11, #$12, #$13, #$D3, #$A1, #$08, #$D7, #$18, #$19, #$96, #$D0,
  345. #$1C, #$1D, #$1E, #$1F,
  346. #$7C, #$D6, #$81, #$C0, #$D1, #$0A, #$17, #$1B, #$D4, #$E9, #$E0, #$D5,
  347. #$92, #$05, #$06, #$07,
  348. #$F0, #$F1, #$16, #$F3, #$F4, #$F5, #$F6, #$04, #$F8, #$F9, #$A9, #$94,
  349. #$14, #$15, #$95, #$1A,
  350. #$20, #$C1, #$C2, #$C3, #$C4, #$C5, #$C6, #$C7, #$C8, #$C9, #$5B, #$2E,
  351. #$3C, #$28, #$2B, #$21,
  352. #$26, #$D8, #$D9, #$E2, #$E3, #$E4, #$E5, #$E6, #$E7, #$E8, #$5D, #$24,
  353. #$2A, #$29, #$3B, #$5E,
  354. #$2D, #$2F, #$82, #$83, #$84, #$85, #$86, #$87, #$88, #$89, #$A6, #$2C,
  355. #$25, #$5F, #$3E, #$3F,
  356. #$97, #$98, #$99, #$A2, #$A3, #$A4, #$A5, #$91, #$A7, #$60, #$3A, #$23,
  357. #$40, #$27, #$3D, #$22,
  358. #$80, #$61, #$62, #$63, #$64, #$65, #$66, #$67, #$68, #$69, #$8A, #$8B,
  359. #$8C, #$8D, #$8E, #$8F,
  360. #$90, #$6A, #$6B, #$6C, #$6D, #$6E, #$6F, #$70, #$71, #$72, #$9A, #$9B,
  361. #$9C, #$9D, #$9E, #$9F,
  362. #$A0, #$7E, #$73, #$74, #$75, #$76, #$77, #$78, #$79, #$7A, #$AA, #$AB,
  363. #$AC, #$AD, #$AE, #$AF,
  364. #$B0, #$B1, #$B2, #$B3, #$B4, #$B5, #$B6, #$B7, #$B8, #$B9, #$BA, #$BB,
  365. #$BC, #$BD, #$BE, #$BF,
  366. #$7B, #$41, #$42, #$43, #$44, #$45, #$46, #$47, #$48, #$49, #$CA, #$CB,
  367. #$CC, #$CD, #$CE, #$CF,
  368. #$7D, #$4A, #$4B, #$4C, #$4D, #$4E, #$4F, #$50, #$51, #$52, #$DA, #$DB,
  369. #$DC, #$DD, #$DE, #$DF,
  370. #$5C, #$E1, #$53, #$54, #$55, #$56, #$57, #$58, #$59, #$5A, #$EA, #$EB,
  371. #$EC, #$ED, #$EE, #$EF,
  372. #$30, #$31, #$32, #$33, #$34, #$35, #$36, #$37, #$38, #$39, #$FA, #$FB,
  373. #$FC, #$FD, #$FE, #$FF
  374. )
  375. );
  376. type
  377. // custom Exception class
  378. EMPHexEditor = class(Exception);
  379. (* bookmark record:<br>
  380. defined by pressing SHIFT+CTRL+[0..9], goto bookmark by pressing CTRL+[0..9]<br><br>
  381. - mPosition: file position<br>
  382. - mInCharField: cursor in character pane (True) or hex number pane
  383. *)
  384. TMPHBookmark = record
  385. mPosition: integer;
  386. mInCharField: boolean;
  387. end;
  388. // array of bookmarks, representing keys 0..9
  389. TMPHBookmarks = array[0..9] of TMPHBookmark;
  390. (* look of the editor's caret:<br>
  391. - ckFull: full block<br>
  392. - ckLeft: left line<br>
  393. - ckBottom: bottom line<br>
  394. - ckAuto: left line if @link(InsertMode), full block if overwrite,
  395. bottom line if ReadOnlyView
  396. *)
  397. TMPHCaretKind = (ckFull,
  398. ckLeft,
  399. ckBottom,
  400. ckAuto
  401. );
  402. (* how to show a file's content in the character pane of the editor:<br>
  403. - tkAsIs: leave as is (current windows code page)<br>
  404. - tkDos8: current dos codepage<br>
  405. - tkASCII: 7 bit ascii<br>
  406. - tkMac: macintosh charset (translation always from/to ms cp 1252 (ms latin1)!!<br>
  407. - tkBCD: ibm ebcdic codepage 38 (translation always from/to ms cp 1252 (ms latin1)!!<br>
  408. - tkCustom: custom codepage stored in @link(MPHCustomCharConv)
  409. *)
  410. TMPHTranslationKind = (tkAsIs,
  411. tkDos8,
  412. tkASCII,
  413. tkMac,
  414. tkBCD
  415. , tkCustom
  416. );
  417. (* action indicator used in @link(OnProgress) event handler:<br>
  418. - pkLoad: loading data<br>
  419. - pkSave: saving data<br>
  420. - pkFind: finding
  421. *)
  422. TMPHProgressKind = (pkLoad,
  423. pkSave, pkFind
  424. );
  425. (* progress event handler, used in @link(OnProgress)<br><br>
  426. - ProgressType: am i loading or saving? (see @link(TMPHProgressKind))<br>
  427. - aName: name of file to be load from/saved to<br>
  428. - Percent: current progress (0..100)<br>
  429. - Cancel: if set to true, the load/save procedure will abort (no meaning in Find* methods) <br>
  430. *)
  431. TMPHProgressEvent = procedure(Sender: TObject;
  432. const ProgressType: TMPHProgressKind;
  433. const aName: TFileName;
  434. const Percent: byte;
  435. var Cancel: boolean) of object;
  436. (* retrieve the "line number" to display by the application<br><br>
  437. - Number: the number to convert to text
  438. - OffsetText: the resulting text output
  439. *)
  440. TMPHGetOffsetTextEvent = procedure(Sender: TObject;
  441. const Number: int64;
  442. var OffsetText: string) of object;
  443. (* handler for custom search routines<br><br>
  444. - Pattern: the data to find
  445. - PatLength: length of the data to find
  446. - SearchFrom: first search position
  447. - SearchUntil: last search position
  448. - IgnoreCase: case sensitive?
  449. - Wilcard: Wildcard character (only used by FindWithWildcard)
  450. - FoundPos: result, set to -1 if data was not found
  451. *)
  452. TMPHFindEvent = procedure(Sender: TObject;
  453. const Pattern: PChar; const PatLength: integer;
  454. const SearchFrom, SearchUntil: integer;
  455. const IgnoreCase: boolean;
  456. const Wildcard: Char;
  457. var FoundPos: Integer) of object;
  458. // precompiled converted character table types for faster data search
  459. PMPHFindTable = ^TMPHFindTable;
  460. TMPHFindTable = array[#0..#255] of Char;
  461. //@exclude
  462. // flags internally used in the undo storage
  463. TMPHUndoFlag = (
  464. // kind of undo storage
  465. ufKindBytesChanged,
  466. ufKindByteRemoved,
  467. ufKindInsertBuffer,
  468. ufKindReplace,
  469. ufKindAppendBuffer,
  470. ufKindNibbleInsert,
  471. ufKindNibbleDelete,
  472. ufKindConvert,
  473. ufKindSelection, // store a selection
  474. ufKindCombined,
  475. ufKindAllData, // store current data and size for complete undo
  476. // additional information
  477. ufFlagByte1Changed,
  478. ufFlagByte2Changed,
  479. ufFlagModified,
  480. ufFlag2ndByteCol,
  481. ufFlagInCharField,
  482. ufFlagHasSelection,
  483. ufFlagInsertMode,
  484. ufFlagIsUnicode,
  485. ufFlagIsUnicodeBigEndian,
  486. ufFlagHasDescription
  487. );
  488. //@exclude
  489. // set of undo flags
  490. TMPHUndoFlags = set of TMPHUndoFlag;
  491. type
  492. // persistent color storage (contains the colors in hex editors)
  493. TMPHColors = class(TPersistent)
  494. private
  495. FParent: TControl;
  496. FOffset: TColor;
  497. FOddColumn: TColor;
  498. FEvenColumn: TColor;
  499. FCursorFrame: TColor;
  500. FNonFocusCursorFrame: TColor;
  501. FBackground: TColor;
  502. FChangedText: TColor;
  503. FChangedBackground: TColor;
  504. FCurrentOffsetBackground: TColor;
  505. FOffsetBackGround: TColor;
  506. FActiveFieldBackground: TColor;
  507. FCurrentOffset: TColor;
  508. FGrid: TColor;
  509. procedure SetOffsetBackGround(const Value: TColor);
  510. procedure SetCurrentOffset(const Value: TColor);
  511. procedure SetParent(const Value: TControl);
  512. procedure SetGrid(const Value: TColor);
  513. procedure SetBackground(const Value: TColor);
  514. procedure SetChangedBackground(const Value: TColor);
  515. procedure SetChangedText(const Value: TColor);
  516. procedure SetCursorFrame(const Value: TColor);
  517. procedure SetEvenColumn(const Value: TColor);
  518. procedure SetOddColumn(const Value: TColor);
  519. procedure SetOffset(const Value: TColor);
  520. procedure SetActiveFieldBackground(const Value: TColor);
  521. procedure SetCurrentOffsetBackground(const Value: TColor);
  522. procedure SetNonFocusCursorFrame(const Value: TColor);
  523. public
  524. // @exclude(constructor)
  525. constructor Create(Parent: TControl);
  526. // @exclude()
  527. procedure Assign(Source: TPersistent); override;
  528. // parent hex editor control
  529. property Parent: TControl read FParent write SetParent;
  530. published
  531. // background color
  532. property Background: TColor read FBackground write SetBackground;
  533. // background color of modified bytes (in overwrite mode)
  534. property ChangedBackground: TColor read FChangedBackground write
  535. SetChangedBackground;
  536. // foreground color of modified bytes (in overwrite mode)
  537. property ChangedText: TColor read FChangedText write SetChangedText;
  538. // color of the cursor and position frame in the second pane
  539. property CursorFrame: TColor read FCursorFrame write SetCursorFrame;
  540. // foreground color of the line offsets
  541. property Offset: TColor read FOffset write SetOffset;
  542. // foreground color of odd columns
  543. property OddColumn: TColor read FOddColumn write SetOddColumn;
  544. // foreground color of even columns
  545. property EvenColumn: TColor read FEvenColumn write SetEvenColumn;
  546. // background color of the current line in the offset pane (gutter)
  547. property CurrentOffsetBackground: TColor read FCurrentOffsetBackground write
  548. SetCurrentOffsetBackground;
  549. // background color of the offset pane (gutter)
  550. property OffsetBackGround: TColor read FOffsetBackGround write
  551. SetOffsetBackGround;
  552. // foreground color of the current line in the offset pane (gutter)
  553. property CurrentOffset: TColor read FCurrentOffset write SetCurrentOffset;
  554. // pen color of the grid
  555. property Grid: TColor read FGrid write SetGrid;
  556. // color of a cursor frame in a non-focused editor
  557. property NonFocusCursorFrame: TColor read FNonFocusCursorFrame write
  558. SetNonFocusCursorFrame;
  559. // background color of the active field (hex/chars)
  560. property ActiveFieldBackground: TColor read FActiveFieldBackground write
  561. SetActiveFieldBackground;
  562. end;
  563. // @exclude(stream class for internal storage/undo)
  564. TMPHMemoryStream = class(TMemoryStream)
  565. private
  566. procedure CheckBounds(const AMax: Integer);
  567. function PointerAt(const APosition: Integer): Pointer;
  568. protected
  569. public
  570. {$IFDEF FASTACCESS}
  571. function GetAddress(const Index, Count: integer): PByte;
  572. {$ENDIF}
  573. procedure ReadBufferAt(var Buffer; const APosition, ACount: Integer);
  574. procedure WriteBufferAt(const Buffer; const APosition, ACount: Integer);
  575. procedure Move(const AFromPos, AToPos, ACount: Integer);
  576. procedure TranslateToAnsi(const FromTranslation: TMPHTranslationKind; const
  577. APosition, ACount: integer);
  578. procedure TranslateFromAnsi(const ToTranslation: TMPHTranslationKind; const
  579. APosition, ACount: integer);
  580. function GetAsHex(const APosition, ACount: integer; const SwapNibbles:
  581. Boolean): string;
  582. end;
  583. //@exclude
  584. // undo storage implementation
  585. TMPHUndoStorage = class;
  586. //@exclude
  587. // offset format flags
  588. TMPHOffsetFormatFlag = (offCalcWidth,
  589. // calculate minwidth depending on data size (width field = '-')
  590. offCalcRow,
  591. // calculate _BytesPerUnit depending on bytes per row (=real line numbers)
  592. offCalcColumn, // " bytes per column (= column numbers)
  593. offBytesPerUnit // use BytesPerUnit property
  594. );
  595. //@exclude
  596. // set of the above flags
  597. TMPHOffsetFormatFlags = set of TMPHOffsetFormatFlag;
  598. //@exclude
  599. // offset format record
  600. TMPHOffsetFormat = record
  601. Format: string; // format as string
  602. Prefix,
  603. Suffix: string; // splitted format
  604. MinWidth: integer; // min length of value (zero padded on the left)
  605. Flags: // auto calculation flags
  606. TMPHOffsetFormatFlags;
  607. Radix, // radix (base) of display (2..16)
  608. _BytesPerUnit: byte; // length of one unit (1 Byte...BytesPerRow Bytes)
  609. end;
  610. (* owner draw event type. parameters:<br><br>
  611. - Sender: the hex editor<br>
  612. - ACanvas: the editor's canvas<br>
  613. - ACol, ARow: the position to be drawn<br>
  614. - AWideText: the text to be drawn<br>
  615. - ARect: the cell rectangle<br>
  616. - ADefaultDraw: if set to True (default), default drawing isperformed after the event handler returns.
  617. if set to false, the event handler must do all cell painting.
  618. *)
  619. TMPHDrawCellEvent = procedure(Sender: TObject; ACanvas: TCanvas; ACol, ARow:
  620. Integer; var AWideText: WideString; ARect: TRect; var ADefaultDraw: Boolean)
  621. of object;
  622. // protected ancestor of the hex editor components
  623. TCustomMPHexEditor = class(TCustomGrid)
  624. private
  625. FIntLastHexCol: integer;
  626. FFindTable,
  627. FFindTableI: TMPHFindTable;
  628. FIsMaxOffset: boolean;
  629. FFindProgress: boolean;
  630. FBlockSize: Integer;
  631. FSepCharBlocks: boolean;
  632. FOnGetOffsetText: TMPHGetOffsetTextEvent;
  633. FFixedFileSize: boolean;
  634. FCharWidth,
  635. FCharHeight: integer;
  636. FBookmarkImageList: TImageList;
  637. FInsertModeOn: boolean;
  638. FCaretBitmap: TBitmap;
  639. FColors: TMPHColors;
  640. FBytesPerRow: integer;
  641. FOffSetDisplayWidth: integer;
  642. FBytesPerRowDup: integer;
  643. FDataStorage: TMPHMemoryStream;
  644. FSwapNibbles: integer;
  645. FFocusFrame: boolean;
  646. FIsFileReadonly: boolean;
  647. FBytesPerCol: integer;
  648. FPosInCharField,
  649. FLastPosInCharField: boolean;
  650. FFileName: string;
  651. FModifiedBytes: TBits;
  652. FBookmarks: TMPHBookmarks;
  653. FSelStart,
  654. FSelPosition,
  655. FSelEnd: integer;
  656. FSelBeginPosition: integer;
  657. FTranslation: TMPHTranslationKind;
  658. FCaretKind: TMPHCaretKind;
  659. FReplaceUnprintableCharsBy: char;
  660. FAllowInsertMode: boolean;
  661. FWantTabs: boolean;
  662. FReadOnlyView: boolean;
  663. FHideSelection: boolean;
  664. FGraySelOnLostFocus: boolean;
  665. FOnProgress: TMPHProgressEvent;
  666. FMouseDownCol,
  667. FMouseDownRow: integer;
  668. FShowDrag: boolean;
  669. FDropCol,
  670. FDropRow: integer;
  671. FOnInvalidKey,
  672. FOnTopLeftChanged: TNotifyEvent;
  673. FDrawGridLines: boolean;
  674. FDrawGutter3D: boolean;
  675. FGutterWidth: integer;
  676. FOffsetFormat: TMPHOffsetFormat;
  677. FSelectionPossible: boolean;
  678. FBookmarkBitmap: TBitmap;
  679. FCursorList: array of integer;
  680. FHasCustomBMP: boolean;
  681. FStreamFileName: string;
  682. FHasFile: boolean;
  683. FMaxUndo: integer;
  684. FHexChars: array[0..15] of char;
  685. FHexLowerCase: boolean;
  686. FOnChange: TNotifyEvent;
  687. FShowRuler: boolean;
  688. FBytesPerUnit: Integer;
  689. FRulerBytesPerUnit: Integer;
  690. FOnSelectionChanged: TNotifyEvent;
  691. FSelectionChangedCount: Integer;
  692. FShowPositionIfNotFocused: Boolean;
  693. FOffsetHandler: Boolean;
  694. FUsedRulerBytesPerUnit: Integer;
  695. FIsSelecting: boolean;
  696. FMouseUpCanResetSel: boolean;
  697. FUndoStorage: TMPHUndoStorage;
  698. FUnicodeCharacters: Boolean;
  699. FUnicodeBigEndian: Boolean;
  700. FMaskedChars: TSysCharSet;
  701. FDrawDataPosition: integer;
  702. FOnDrawCell: TMPHDrawCellEvent;
  703. FOnBookmarkChanged: TNotifyEvent;
  704. FIsDrawDataSelected: boolean;
  705. FOnWildcardFind: TMPHFindEvent;
  706. FOnFind: TMPHFindEvent;
  707. {$IFDEF FASTACCESS}
  708. FSetDataSizeFillByte: Byte;
  709. {$ENDIF}
  710. FRulerNumberBase: byte;
  711. property Color;
  712. function IsInsertModePossible: boolean;
  713. function IsFileSizeFixed: boolean;
  714. procedure InternalErase(const KeyWasBackspace: boolean; const UndoDesc:
  715. string = '');
  716. procedure SetReadOnlyView(const Value: boolean);
  717. procedure SetCaretKind(const Value: TMPHCaretKind);
  718. procedure SetFocusFrame(const Value: boolean);
  719. procedure SetBytesPerColumn(const Value: integer);
  720. procedure SetSwapNibbles(const Value: boolean);
  721. function GetSwapNibbles: boolean;
  722. function GetBytesPerColumn: integer;
  723. procedure SetOffsetDisplayWidth;
  724. procedure SetColors(const Value: TMPHColors);
  725. procedure SetReadOnlyFile(const Value: boolean);
  726. procedure SetTranslation(const Value: TMPHTranslationKind);
  727. procedure SetModified(const Value: boolean);
  728. procedure SetChanged(DataPos: integer; const Value: boolean);
  729. procedure SetFixedFileSize(const Value: boolean);
  730. procedure SetAllowInsertMode(const Value: boolean);
  731. function GetInsertMode: boolean;
  732. procedure SetWantTabs(const Value: boolean);
  733. procedure SetHideSelection(const Value: boolean);
  734. procedure SetGraySelectionIfNotFocused(const Value: boolean);
  735. function CalcColCount: integer;
  736. function GetLastCharCol: integer;
  737. function GetPropColCount: integer;
  738. function GetPropRowCount: integer;
  739. function GetMouseOverSelection: boolean;
  740. function CursorOverSelection(const X, Y: integer): boolean;
  741. function MouseOverFixed(const X, Y: integer): boolean;
  742. procedure AdjustBookmarks(const From, Offset: integer);
  743. procedure IntSetCaretPos(const X, Y, ACol: integer);
  744. procedure TruncMaxPosition(var DataPos: integer);
  745. procedure SetSelection(DataPos, StartPos, EndPos: integer);
  746. function GetCurrentValue: integer;
  747. procedure SetInsertMode(const Value: boolean);
  748. function GetModified: boolean;
  749. //function GetDataPointer: Pointer;
  750. procedure SetBytesPerRow(const Value: integer);
  751. procedure SetMaskChar(const Value: char);
  752. procedure SetAsText(const Value: string);
  753. procedure SetAsHex(const Value: string);
  754. function GetAsText: string;
  755. function GetAsHex: string;
  756. procedure WMTimer(var Msg: TWMTimer); message WM_TIMER;
  757. // show or hide caret depending on row/col in view
  758. procedure CheckSetCaret;
  759. // get the row according to the given buffer position
  760. function GetRow(const DataPos: integer): integer;
  761. // invalid key pressed (in ebcdic)
  762. procedure WrongKey;
  763. // create an inverting caret bitmap
  764. procedure CreateCaretGlyph;
  765. // get start of selection
  766. function GetSelStart: integer;
  767. // get end of selection
  768. function GetSelEnd: integer;
  769. // get selection count
  770. function GetSelCount: integer;
  771. // set selection start
  772. procedure SetSelStart(aValue: integer);
  773. // set selection end
  774. procedure SetSelEnd(aValue: integer);
  775. // position the caret in the given field
  776. procedure SetInCharField(const Value: boolean);
  777. // is the caret in the char field ?
  778. function GetInCharField: boolean;
  779. // insert a buffer (internal)
  780. procedure InternalInsertBuffer(Buffer: PChar; const Size, Position:
  781. integer);
  782. // append some data (int)
  783. procedure InternalAppendBuffer(Buffer: PChar; const Size: integer);
  784. // store the caret properties
  785. procedure InternalGetCurSel(var StartPos, EndPos, ACol, ARow: integer);
  786. // delete data
  787. procedure InternalDelete(StartPos, EndPos, ACol, ARow: integer);
  788. // delete one half byte
  789. function InternalDeleteNibble(const Pos: integer;
  790. const HighNibble: boolean): boolean;
  791. // insert half byte
  792. function InternalInsertNibble(const Pos: integer; const HighNibble:
  793. boolean): boolean;
  794. // used by nibble functions
  795. function CreateShift4BitStream(const StartPos: integer; var FName:
  796. TFileName): TFileStream;
  797. // convert a given amount of data from ansi to something different and vice versa
  798. procedure InternalConvertRange(const aFrom, aTo: integer; const aTransFrom,
  799. aTransTo: TMPHTranslationKind);
  800. // move data in buffer to a different position
  801. procedure MoveFileMem(const aFrom, aTo, aCount: integer);
  802. function GetBookmark(Index: byte): TMPHBookmark;
  803. procedure SetBookmark(Index: byte; const Value: TMPHBookmark);
  804. procedure SetBookmarkVals(const Index: byte; const Position: integer; const
  805. InCharField: boolean);
  806. procedure SetDrawGridLines(const Value: boolean);
  807. procedure SetGutterWidth(const Value: integer);
  808. // images have changed
  809. procedure BookmarkBitmapChanged(Sender: TObject);
  810. procedure SetBookmarkBitmap(const Value: TBitmap);
  811. function GetVersion: string;
  812. procedure SetVersion(const Value: string);
  813. // free alloc'd memory of one of the storage streams;
  814. procedure FreeStorage(FreeUndo: boolean = False);
  815. function GetCanUndo: boolean;
  816. function GetCanRedo: boolean;
  817. function GetUndoDescription: string;
  818. function GetOffsetFormat: string;
  819. procedure SetOffsetFormat(const Value: string);
  820. // generate offset format
  821. procedure GenerateOffsetFormat(Value: string);
  822. procedure SetHexLowerCase(const Value: boolean);
  823. procedure SetDrawGutter3D(const Value: boolean);
  824. procedure SetShowRuler(const Value: boolean);
  825. procedure SetBytesPerUnit(const Value: integer);
  826. procedure SetRulerString;
  827. procedure CheckSelectUnit(var AStart, AEnd: Integer);
  828. procedure SetRulerBytesPerUnit(const Value: integer);
  829. procedure SetShowPositionIfNotFocused(const Value: Boolean);
  830. function GetDataAt(Index: integer): Byte;
  831. procedure SetDataAt(Index: integer; const Value: Byte);
  832. procedure SetUnicodeCharacters(const Value: Boolean);
  833. procedure SetUnicodeBigEndian(const Value: Boolean);
  834. function GetPositionAtCursor(const ACol, ARow: integer): integer;
  835. function GetIsCharFieldCol(const ACol: integer): Boolean;
  836. procedure SetDataSize(const Value: integer);
  837. procedure SetBlockSize(const Value: Integer);
  838. procedure SetSepCharBlocks(const Value: boolean);
  839. procedure SetFindProgress(const Value: boolean);
  840. procedure SetRulerNumberBase(const Value: byte);
  841. procedure SetMaskedChars(const Value: TSysCharSet);
  842. {+}
  843. {.$IFDEF BCB}
  844. protected
  845. // bcb seems to need overwritten abstract methods for dynamically created
  846. // controls (this method is never called in TMPHexEditor/Ex)
  847. procedure DrawCell(ACol, ARow: Longint; ARect: TRect; AState: TGridDrawState); override;
  848. {.$ENDIF}
  849. {+.}
  850. protected
  851. // @exclude()
  852. FRulerString: string;
  853. // @exclude()
  854. FRulerCharString: string;
  855. // @exclude(used by TMPHexEditorEx for internal drag 'n' drop)
  856. FFixedFileSizeOverride: boolean;
  857. // @exclude(used by TMPHexEditorEx for internal undo changing)
  858. FModified: boolean;
  859. // @exclude(overwrite mouse wheel for zooming)
  860. function DoMouseWheelDown(Shift: TShiftState; MousePos: TPoint): boolean;
  861. override;
  862. // @exclude(overwrite mouse wheel for zooming)
  863. function DoMouseWheelUp(Shift: TShiftState; MousePos: TPoint): boolean;
  864. override;
  865. // @exclude(actually used bytes per unit)
  866. property UsedRulerBytesPerUnit: Integer read FUsedRulerBytesPerUnit;
  867. // @exclude(True: cells are currently to be selected)
  868. property IsSelecting: boolean read FIsSelecting;
  869. // @exclude(True: MouseUp resets selection)
  870. property MouseUpCanResetSel: boolean read FMouseUpCanResetSel write
  871. FMouseUpCanResetSel;
  872. // @exclude(memory stream which contains the undo/redo data)
  873. property UndoStorage: TMPHUndoStorage read FUndoStorage;
  874. // @exclude(stream that contains the data)
  875. property DataStorage: TMPHMemoryStream read FDataStorage;
  876. // @exclude(fire OnSelectionChange)
  877. procedure SelectionChanged; virtual;
  878. // @exclude(set a new selection)
  879. procedure NewSelection(SelFrom, SelTo: integer);
  880. // @exclude(get the current mouse position)
  881. function CheckMouseCoord(var X, Y: integer): TGridCoord;
  882. // @exclude(assure the value is a multiple of FBytesPerUnit)
  883. procedure CheckUnit(var AValue: Integer);
  884. // call changed on every undo creation for OnChange event
  885. procedure Changed; virtual;
  886. // returns the drop file position after a drag'n'drop operation
  887. function DropPosition: integer;
  888. // copy a stream to a second one and fire the OnProgress handler
  889. procedure Stream2Stream(strFrom, strTo: TStream; const Operation:
  890. TMPHProgressKind; const Count: integer = -1);
  891. (* allows descendants to take special action if contents are to be saved
  892. to the file from where the data was load *)
  893. procedure PrepareOverwriteDiskFile; virtual;
  894. // store the current Cursor and set it to crHourGlass (see also @link(OldCursor))
  895. procedure WaitCursor;
  896. // reset the Cursor to the previous value (see also @link(WaitCursor))
  897. procedure OldCursor;
  898. // @exclude(override paint)
  899. procedure Paint; override;
  900. // @exclude(view changed)
  901. procedure TopLeftChanged; override;
  902. // adjust cell widths/heigths depending on font, offset format, bytes per row/column...
  903. procedure AdjustMetrics;
  904. // get the size of the contained data
  905. function GetDataSize: integer;
  906. // @exclude(calculate the grid sizes)
  907. procedure CalcSizes;
  908. // @exclude(select one cell)
  909. function SelectCell(ACol, ARow: longint): boolean; override;
  910. // @exclude(get the data position depending on col and row)
  911. function GetPosAtCursor(const aCol, aRow: integer): integer;
  912. // @exclude(vice versa)
  913. function GetCursorAtPos(const aPos: integer; const aChars: boolean):
  914. TGridCoord;
  915. // @exclude(get the column of the other field (hex<->char))
  916. function GetOtherFieldCol(const aCol: integer): integer;
  917. // @exclude(get the column of the other field (hex<->char))
  918. function GetOtherFieldColCheck(const aCol: integer): integer;
  919. // @exclude(can the cell be selected ?)
  920. function CheckSelectCell(aCol, aRow: integer): boolean;
  921. // @exclude(char message handler)
  922. procedure WMChar(var Msg: TWMChar); message WM_CHAR;
  923. // @exclude(posted message to update the caret position)
  924. procedure CMINTUPDATECARET(var Msg: TMessage); message CM_INTUPDATECARET;
  925. // @exclude(posted message to fire an OnSelectionChanged event)
  926. procedure CMSelectionChanged(var Msg: TMessage); message
  927. CM_SELECTIONCHANGED;
  928. // @exclude(for shortcuts)
  929. procedure WMGetDlgCode(var Msg: TWMGetDlgCode); message WM_GETDLGCODE;
  930. // @exclude(readjust grid sizes after font has changed)
  931. procedure CMFontChanged(var Message: TMessage); message CM_FONTCHANGED;
  932. // @exclude(change a byte at the given position)
  933. procedure IntChangeByte(const aOldByte, aNewByte: byte;
  934. aPos, aCol, aRow: integer; const UndoDesc: string = '');
  935. // @exclude(change two bytes at the given position)
  936. procedure IntChangeWideChar(const aOldChar, aNewChar: WideChar; aPos, aCol,
  937. aRow: integer; const UndoDesc: string = '');
  938. // @exclude(keydown handler)
  939. procedure KeyDown(var Key: word; Shift: TShiftState); override;
  940. // @exclude(keyup handler)
  941. //procedure KeyUp(var Key: word; Shift: TShiftState); override;
  942. // @exclude(has this byte been modified ?)
  943. function HasChanged(aPos: integer): boolean;
  944. // @exclude(redraw some lines)
  945. procedure RedrawPos(aFrom, aTo: integer);
  946. // @exclude(make a selection)
  947. procedure Select(const aCurCol, aCurRow, aNewCol, aNewRow: integer);
  948. // @exclude(mouse down handler)
  949. procedure MouseDown(Button: TMouseButton; Shift: TShiftState; X, Y:
  950. integer); override;
  951. // @exclude(mouse move handler)
  952. procedure MouseMove(Shift: TShiftState; X, Y: integer); override;
  953. // @exclude(mouse up handler)
  954. procedure MouseUp(Button: TMouseButton; Shift: TShiftState; X, Y: integer);
  955. override;
  956. // @exclude(is undo record creation possible?)
  957. function CanCreateUndo(const aKind: TMPHUndoFlag; const aCount, aReplCount:
  958. integer): Boolean; virtual;
  959. // @exclude(add an undo to the undo buffer)
  960. procedure CreateUndo(const aKind: TMPHUndoFlag; const aPos, aCount,
  961. aReplCount: integer; const sDesc: string = '');
  962. // @exclude(after loading)
  963. procedure Loaded; override;
  964. // @exclude(override CreateWnd)
  965. procedure CreateWnd; override;
  966. // @exclude(wm_setfocus handler)
  967. procedure WMSetFocus(var Msg: TWMSetFocus); message WM_SETFOCUS;
  968. // @exclude(wm_killfocus handler)
  969. procedure WMKillFocus(var Msg: TWMKillFocus); message WM_KILLFOCUS;
  970. // @exclude(wm_vscroll handler)
  971. procedure WMVScroll(var Msg: TWMVScroll); message WM_VSCROLL;
  972. // @exclude(wm_hscroll handler)
  973. procedure WMHScroll(var Msg: TWMHScroll); message WM_HSCROLL;
  974. // @exclude(resize the control)
  975. procedure Resize; override;
  976. // @exclude(store bitmap ? (its set to true, if a custom bitmap has been stored in BookmarkBitmap))
  977. function HasCustomBookmarkBitmap: boolean;
  978. // number of bytes to show in each row
  979. property BytesPerRow: integer read FBytesPerRow write SetBytesPerRow;
  980. // if set to True, the find* routines also fire OnProgress events (default is False)
  981. property FindProgress: boolean read FFindProgress write SetFindProgress
  982. default False;
  983. // number of bytes to show in each column
  984. property BytesPerColumn: integer read GetBytesPerColumn write
  985. SetBytesPerColumn default 2;
  986. (* translation kind of the data (used to show characters on and to handle key presses in the char pane),
  987. (see also @link(TMPHTranslationKind))
  988. *)
  989. property Translation: TMPHTranslationKind read FTranslation write
  990. SetTranslation;
  991. (* offset display ("line numbers") format, in the form<br>
  992. [r|c|&lt;HEXNUM&gt;%][-|&lt;HEXNUM&gt;!]&lt;HEXNUM&gt;:[Prefix]|[Suffix]<br>
  993. (&lt;HEXNUM&gt; means a number in hexadecimal format (without prefix/suffix))<br><br>
  994. - first field (up to the percent sign):<br>
  995. <ul>
  996. <li>sets the "bytes per unit field" of the offset display format</li>
  997. <li>if it's set to 1, each row offset displays the data position in bytes</li>
  998. <li>if it's set to 2, each row offset displays the data position in words</li>
  999. <li>if it's set to 4, each row offset displays the data position in dwords</li>
  1000. <li>if it's set to "r", each row offset displays the current row number (1st row=0,
  1001. see also @link(BytesPerRow))</li>
  1002. <li>if it's set to "c", each row offset displays the current column number (1st column=0,
  1003. see also @link(BytesPerColumn))</li>
  1004. <li>if this field is omitted, bytes per unit is set to the value of the
  1005. @link(RulerBytesPerUnit) property</li>
  1006. </ul><br>
  1007. - second field (up to the exclamation mark):<br>
  1008. <ul>
  1009. <li>sets the minimum width of the number part, if the number is shorter, it will be padded
  1010. by '0' chars at the left</li>
  1011. <li>if this field reads -!, the the minimum width is automatically set to the longest number
  1012. that can appear in the editor (the data's size)</li>
  1013. <li>if this field is omitted, the minimum width is set to 1</li>
  1014. </ul><br>
  1015. - third field (up to the colon):<br>
  1016. <ul>
  1017. <li>sets the radix (base) of the offset format in hex notation</li>
  1018. <li>set this to '10' (without quotes) for hexadecimal offset display, set it to '08' for
  1019. octal and to '0a' for decimal offset display</li>
  1020. <li>this field cannot be omitted, but the whole format string my be blank to avoid the display of
  1021. offset identifiers</li>
  1022. </ul></br>
  1023. - fourth field (up to the pipe ('|') char):<br>
  1024. <ul>
  1025. <li>the prefix that is put in front of the "number" string (e.g. '0x' or '$' to show that numbers are in hex format)
  1026. </li><li>this field may be omitted (but not the pipe char!)</li>
  1027. </ul><br>
  1028. - fifth (and last) field:<br>
  1029. <ul>
  1030. <li>the suffix to put after the "number string" (e.g. 'h' to show hex numbers)</li>
  1031. <li>this field may be omitted</li></ul>
  1032. *)
  1033. property OffsetFormat: string read GetOffsetFormat write SetOffsetFormat;
  1034. (* if this handler is assigned, the @link(OffsetFormat) is not used to
  1035. create "line numbers", but the application tells the editor how to format the offset text
  1036. *)
  1037. property OnGetOffsetText: TMPHGetOffsetTextEvent read FOnGetOffsetText write
  1038. FOnGetOffsetText;
  1039. (* how many bytes form one block in a row? blocks are separated by a one character wide blank.
  1040. -1 means no block separation (see also @link(SeparateBlocksInCharField)) *)
  1041. property BytesPerBlock: Integer read FBlockSize write SetBlockSize default
  1042. -1;
  1043. (* if @link(BytesPerBlock) is used, this property tells the editor whether it should
  1044. separate blocks of bytes in the character pane too or not *)
  1045. property SeparateBlocksInCharField: boolean read FSepCharBlocks write
  1046. SetSepCharBlocks default True;
  1047. // look of the editor's caret (see @link(TMPHCaretKind))
  1048. property CaretKind: TMPHCaretKind read FCaretKind write SetCaretKind default
  1049. ckAuto;
  1050. // colors to display (see @link(TMPHColors))
  1051. property Colors: TMPHColors read FColors write SetColors;
  1052. (* if FocusFrame is set to True, the current caret position will be displayed in the
  1053. second field (hex - characters) as a dotted focus frame, if set to False, it will
  1054. be shown as an ordinary rectangle
  1055. *)
  1056. property FocusFrame: boolean read FFocusFrame write SetFocusFrame;
  1057. (* if SwapNibbles is set to True, the hex pane will show all bytes in the order
  1058. lower 4 bits-higher 4 bits (i.e. the value 192 dec = C0 hex will be drawn as
  1059. 0C). if set to False, hex values will be displayed in usual order. this
  1060. setting also affects hex data input and hex-string conversions
  1061. *)
  1062. property SwapNibbles: boolean read GetSwapNibbles write SetSwapNibbles
  1063. default False;
  1064. // replace @link(MaskedChars) with the following character in the character pane
  1065. property MaskChar: char read FReplaceUnprintableCharsBy write SetMaskChar
  1066. stored False;
  1067. (* if set to True, the data size is readonly, e.g. no data may be appended, deleted
  1068. or inserted, just overwriting is allowed. this also affects @link(InsertMode).
  1069. *)
  1070. property NoSizeChange: boolean read FFixedFileSize write SetFixedFileSize
  1071. default False;
  1072. (* if set to False, switching between overwrite and insert mode is not allowed
  1073. (see also @link(InsertMode) and @link(NoSizeChange))
  1074. *)
  1075. property AllowInsertMode: boolean read FAllowInsertMode write
  1076. SetAllowInsertMode default True;
  1077. (* if set to True, the Tab key is used to switch the caret between hex and character pane.
  1078. if set to False, the Tab key can be used to switch between controls. then the
  1079. combination CTRL+T is used to switch the panes
  1080. *)
  1081. property WantTabs: boolean read FWantTabs write SetWantTabs default True;
  1082. // if set to True, the data can not be edited, just cursor movement is allowed ("Hex Viewer" mode)
  1083. property ReadOnlyView: boolean read FReadOnlyView write SetReadOnlyView
  1084. default False;
  1085. // hide the current selection when the hex editor looses focus (see also @link(GraySelectionIfNotFocused))
  1086. property HideSelection: boolean read FHideSelection write SetHideSelection
  1087. default False;
  1088. (* if set to True and @link(HideSelection) is False, then the current selection will be
  1089. grayed when the hex editor looses focus (the values from the @link(Colors) property will
  1090. be converted to grayscale colors)
  1091. *)
  1092. property GraySelectionIfNotFocused: boolean read FGraySelOnLostFocus write
  1093. SetGraySelectionIfNotFocused default False;
  1094. (* this event is called in @link(LoadFromFile), @link(SaveToFile), @link(Find) and
  1095. @link(FindWithWildcard) routines, so a progress indicator may be updated
  1096. (see also @link(TMPHProgressEvent), @link(FindProgress))
  1097. *)
  1098. property OnProgress: TMPHProgressEvent read FOnProgress write
  1099. FOnProgress;
  1100. (* this event is fired if an invalid character has been typed (like non-hex characters
  1101. in the hex pane)
  1102. *)
  1103. property OnInvalidKey: TNotifyEvent read FOnInvalidKey write FOnInvalidKey;
  1104. // this event is fired if the first visible row or column have been changed (e.g. on scrolling)
  1105. property OnTopLeftChanged: TNotifyEvent read FOnTopLeftChanged write
  1106. FOnTopLeftChanged;
  1107. // returns the current selection in hex format ('00010203...') as string, uses @link(SwapNibbles)
  1108. function GetSelectionAsHex: string;
  1109. (* replace the current selection by a string containing data in hex format ('00 01 02 03' or similar),
  1110. uses @link(SwapNibbles)
  1111. *)
  1112. procedure SetSelectionAsHex(const s: string);
  1113. // returns a string containing the currently selected data
  1114. function GetSelectionAsText: string;
  1115. // replaces the currently selected data with the string's contents
  1116. procedure SetSelectionAsText(const s: string);
  1117. // if set to True, a grid is drawn
  1118. property DrawGridLines: boolean read FDrawGridLines write SetDrawGridLines;
  1119. // width of the offset display gutter, if set to -1, automatically adjust the gutter's width
  1120. property GutterWidth: integer read FGutterWidth write SetGutterWidth default
  1121. -1;
  1122. (* bitmap containing 20 10x10 pixels pictures for bokkmarks (they are displayed in the offset
  1123. gutter), the first ten pictures represent the bookmarks 0(10)..9, if they are set in the
  1124. hexpane, the last 10 pics are shown if bookmarks are set in the character pane (see also
  1125. @link(TMPHBookMark))
  1126. *)
  1127. property BookmarkBitmap: TBitmap read FBookmarkBitmap write SetBookmarkBitmap
  1128. stored HasCustomBookmarkBitmap;
  1129. // current version of the hex editor component (returns the build data), readonly
  1130. property Version: string read GetVersion write SetVersion stored True;
  1131. // maximum memory that is used for undo storage (in bytes, approximately)
  1132. property MaxUndo: integer read FMaxUndo write FMaxUndo default 1024 * 1024;
  1133. (* insert mode (typed characters are inserted at the current position) or
  1134. overwrite mode (typed characters replace values at the current position), see also
  1135. @link(AllowInsertMode), @link(NoSizeChange) and @link(ReadOnlyView)
  1136. *)
  1137. property InsertMode: boolean read GetInsertMode write SetInsertMode default
  1138. False;
  1139. // if set to True, hex data and hex offsets are displayed in lower case
  1140. property HexLowerCase: boolean read FHexLowerCase write SetHexLowerCase
  1141. default False;
  1142. // this event is called on every data change (load/empty/undo/redo)
  1143. property OnChange: TNotifyEvent read FOnChange write FOnChange;
  1144. // if set to True, a 3d line is drawn at the right of the offset gutter
  1145. property DrawGutter3D: boolean read FDrawGutter3D write SetDrawGutter3D
  1146. default True;
  1147. // if set to True, a ruler is shown above the first row
  1148. property ShowRuler: boolean read FShowRuler write SetShowRuler default
  1149. False;
  1150. (* number base (i.e. radix) for the ruler display (2-16), tells the component
  1151. which number format to use when drawing the ruler
  1152. *)
  1153. property RulerNumberBase: byte read FRulerNumberBase write SetRulerNumberBase
  1154. default 16;
  1155. (* setting this property changes the way how mouse/keyboard selection
  1156. works:<br>
  1157. e.g. if set to two, two bytes will be treated as a unit, that means you
  1158. cannot select a single byte, only two, four, six... bytes can be selected.
  1159. also drag/drop and clipboard pasting is affected (data size
  1160. is always a multiple of BytesPerUnit). See also @link(RulerBytesPerUnit)
  1161. *)
  1162. property BytesPerUnit: integer read FBytesPerUnit write SetBytesPerUnit
  1163. default 1;
  1164. (* setting this property affects the offset/ruler drawing:<br>
  1165. e.g. if set to two, two bytes will be treated as a unit, that means the
  1166. offset and ruler values will step by one each two bytes.
  1167. if this property is set to -1, it will use the value of the
  1168. @link(BytesPerUnit) property
  1169. *)
  1170. property RulerBytesPerUnit: integer read FRulerBytesPerUnit write
  1171. SetRulerBytesPerUnit default -1;
  1172. // mark the current position even if the editor is not focused
  1173. property ShowPositionIfNotFocused: Boolean read FShowPositionIfNotFocused
  1174. write SetShowPositionIfNotFocused default False;
  1175. (* if set to True, the character pane displays unicode characters
  1176. and the @link(BytesPerUnit) property is set to 2. @link(Translation) is
  1177. set to tkAsIs. @link(BytesPerRow) and @link(BytesPerColumn) must be a
  1178. multiple of two to be able to use the unicode mode.
  1179. see also @link(UnicodeBigEndian)
  1180. *)
  1181. property UnicodeChars: Boolean read FUnicodeCharacters write
  1182. SetUnicodeCharacters default False;
  1183. (* if set to True, big endian unicode mode is used if @link(UnicodeChars) is
  1184. enabled
  1185. *)
  1186. property UnicodeBigEndian: Boolean read FUnicodeBigEndian write
  1187. SetUnicodeBigEndian default False;
  1188. // this event is fired when the selection/caret position has changed
  1189. property OnSelectionChanged: TNotifyEvent read FOnSelectionChanged write
  1190. FOnSelectionChanged;
  1191. // use this event to implement owner drawing. see also @link(TMPHDrawCellEvent)
  1192. property OnDrawCell: TMPHDrawCellEvent read FOnDrawCell write FOnDrawCell;
  1193. // fire OnBookmarkChanged
  1194. procedure BookmarkChanged; virtual;
  1195. procedure DoSetCellWidth(const Index: integer; Value: integer);
  1196. procedure DefineProperties(Filer: TFiler); override;
  1197. procedure ReadMaskChar(Reader: TReader);
  1198. procedure ReadMaskChar_I(Reader: TReader);
  1199. procedure WriteMaskChar_I(Writer: TWriter);
  1200. public
  1201. { Public-Deklarationen }
  1202. {$IFDEF FASTACCESS}
  1203. // return the memory address at the given stream position after checking bounaries
  1204. function GetFastPointer(const Index, Count: integer): PByte;
  1205. {$ENDIF}
  1206. //@exclude()
  1207. constructor Create(aOwner: TComponent); override;
  1208. //@exclude()
  1209. destructor Destroy; override;
  1210. // these characters are masked in the character pane using @link(MaskChar)
  1211. property MaskedChars: TSysCharSet read FMaskedChars write SetMaskedChars;
  1212. (* during OnDrawCell event handlers, this property tells the data position currently
  1213. being drawn (-1, if offset or ruler are drawn)
  1214. *)
  1215. property DrawDataPosition: integer read FDrawDataPosition;
  1216. (* during OnDrawCell event handlers, this property tells whether the cell is
  1217. to be drawn in selected style (only valid if DrawDataPosition <> -1)
  1218. *)
  1219. property IsDrawDataSelected: boolean read FIsDrawDataSelected;
  1220. // @exclude(use TMPHexEditor.ReadBuffer!)
  1221. function GetMemory(const Index: Integer): char;
  1222. (* @exclude(see http://info.borland.com/devsupport/delphi/fixes/delphi4/vcl.html,
  1223. ref 279)
  1224. *)
  1225. function CanFocus: Boolean;
  1226. {$IFDEF DELPHi5UP} override;
  1227. {$ENDIF}
  1228. // @exclude(use TMPHexEditor.WriteBuffer!)
  1229. procedure SetMemory(const Index: integer; const Value: char);
  1230. (* this property is valid only in the @link(OnGetOffsetText) event. if True,
  1231. the component asks for the string of the highest possible offset, if False,
  1232. a row's offset text is queried
  1233. *)
  1234. property IsMaxOffset: boolean read FIsMaxOffset;
  1235. // seek behind the last position if @link(InsertMode) = True, goto last position otherwise
  1236. procedure SeekToEOF;
  1237. (* synchronize another TCustomMPHexEditor view (top, left, selection),
  1238. the optional SyncOffset parameter may be used for a different viewpoint
  1239. *)
  1240. procedure SyncView(Source: TCustomMPHexEditor; SyncOffset: integer = 0);
  1241. // return the offset of the first displayed data
  1242. function DisplayStart: integer;
  1243. // return the offset of the last displayed data
  1244. function DisplayEnd: integer;
  1245. // is the given position part of the selection?
  1246. function IsSelected(const APosition: integer): boolean;
  1247. // calculate a data position from a col/row pair
  1248. property PositionAtCursor[const ACol, ARow: integer]: integer read
  1249. GetPositionAtCursor;
  1250. // is the given col in the hex or the character pane?
  1251. property IsCharFieldCol[const ACol: integer]: Boolean read
  1252. GetIsCharFieldCol;
  1253. {$IFDEF FASTACCESS}
  1254. // this byte value is used to fill the data when setting @link(DataSize)
  1255. // enlarges the stream
  1256. property SetDataSizeFillByte: Byte read FSetDataSizeFillByte write
  1257. FSetDataSizeFillByte;
  1258. {$ENDIF}
  1259. // has data been load from/saved to a file (or is the filename valid)
  1260. property HasFile: boolean read FHasFile write FHasFile;
  1261. (* each call to UndoBeginUpdate increments an internal counter that prevents using
  1262. undo storage and also disables undo functionality (see also @link(UndoEndUpdate))
  1263. *)
  1264. function UndoBeginUpdate: integer; virtual;
  1265. (* each call to UndoEndUpdate decrements an internal counter that prevents using
  1266. undo storage and also disables undo functionality. the return value is the value
  1267. of this counter. if the counter is reset to zero, undo creation is permitted again
  1268. (see also @link(UndoBeginUpdate))
  1269. *)
  1270. function UndoEndUpdate: integer; virtual;
  1271. // remove selection state from all data
  1272. procedure ResetSelection(const aDraw: boolean);
  1273. // see @link(GetSelectionAsHex) and @link(SetSelectionAsHex)
  1274. property SelectionAsHex: string read GetSelectionAsHex write
  1275. SetSelectionAsHex;
  1276. // see @link(GetSelectionAsText) and @link(SetSelectionAsText)
  1277. property SelectionAsText: string read GetSelectionAsText write
  1278. SetSelectionAsText;
  1279. {$IFNDEF BCB}
  1280. (* precompiled character comparison table for custom find routines, see also
  1281. @link(FindTableI), @link(OnFind), @link(OnWildcardFind), case sensitive, not
  1282. public under BCB!
  1283. *)
  1284. property FindTable: TMPHFindTable read FFindTable;
  1285. (* precompiled character comparison table for custom find routines, see also
  1286. @link(FindTable), @link(OnFind), @link(OnWildcardFind), case insensitive, not
  1287. public under BCB!
  1288. *)
  1289. property FindTableI: TMPHFindTable read FFindTableI;
  1290. {$ENDIF}
  1291. // implement your custom @link(Find) routine by assigning a method to this handler,
  1292. // see also @link(OnWildcardFind)
  1293. property OnFind: TMPHFindEvent read FOnFind write FOnFind;
  1294. // implement your custom @link(FindWithWildcard) routine by assigning a method
  1295. // to this handler, see also @link(OnFind)
  1296. property OnWildcardFind: TMPHFindEvent read FOnWildcardFind
  1297. write FOnWildcardFind;
  1298. (* returns the given position as it would be drawn in the offset gutter,
  1299. see also @link(OffsetFormat)
  1300. *)
  1301. function GetOffsetString(const Position: cardinal): string; virtual;
  1302. (* returns the given position as it would be drawn in the offset gutter, exception:
  1303. if @link(OffsetFormat) is set to an empty string, returns the hexadecimal representation
  1304. of the Position value (see also @link(GetOffsetString))
  1305. *)
  1306. function GetAnyOffsetString(const Position: integer): string; virtual;
  1307. // returns the height of one row in pixels
  1308. function RowHeight: integer;
  1309. // free the undo storage (discard all possible undo steps)
  1310. procedure ResetUndo;
  1311. // set the current position (like TStream.Seek)
  1312. function Seek(const aOffset, aOrigin: integer): integer;
  1313. (* searches for text or data in the data buffer, returns the find position (-1, if data have not been found):<br><br>
  1314. - aBuffer: data to search for<br>
  1315. - aCount: size of data in aBuffer<br>
  1316. - aStart: start search at this position<br>
  1317. - aEnd: searches up to this position<br>
  1318. - IgnoreCase: if True, lowercase and uppercase characters are treated as if they were equal<br>
  1319. - SearchText: if True, the current @link(Translation) is taken into account when searching textual data<br><br>
  1320. NOTE: call @link(PrepareFindReplaceData) before the first Find call
  1321. *)
  1322. function Find(aBuffer: PChar; aCount: integer; const aStart, aEnd: integer;
  1323. const IgnoreCase: boolean): integer;
  1324. (* searches for text or data in the data buffer using a wildcard character
  1325. returns the find position (-1, if data have not been found):<br><br>
  1326. - aBuffer: data to search for<br>
  1327. - aCount: size of data in aBuffer<br>
  1328. - aStart: start search at this position<br>
  1329. - aEnd: searches up to this position<br>
  1330. - IgnoreCase: if True, lowercase and uppercase characters are treated as if they were equal<br>
  1331. - SearchText: if True, the current @link(Translation) is taken into account when searching textual data<br>
  1332. - Wildcard: this character is a placeholder for any character<br><br>
  1333. NOTE: call @link(PrepareFindReplaceData) before the first FindWithWildcard call
  1334. *)
  1335. function FindWithWildcard(aBuffer: PChar; aCount: integer; const aStart,
  1336. aEnd: integer;
  1337. const IgnoreCase: boolean; const Wildcard: char): integer;
  1338. (* convert a buffer for @link(Find)/@link(FindWithWildcard)/replace operation depending on
  1339. unicode mode. sets the string to lower case if IgnoreCase is True. if in unicode mode,
  1340. creates a unicode string.
  1341. *)
  1342. (*
  1343. store a selection as undo record, so you can restore the selection start and end by using
  1344. @link(Undo). this can be useful e.g. to show position of replaced data
  1345. *)
  1346. procedure AddSelectionUndo(const AStart, ACount: integer);
  1347. function PrepareFindReplaceData(StrData: string; const IgnoreCase, IsText:
  1348. boolean): string;
  1349. // read data into a buffer
  1350. procedure ReadBuffer(var Buffer; const Index, Count: Integer);
  1351. // write a buffer to the file data
  1352. procedure WriteBuffer(const Buffer; const Index, Count: Integer); virtual;
  1353. // delete the currently selected data
  1354. procedure DeleteSelection(const UndoDesc: string = '');
  1355. // load the contents of a stream into the data buffer
  1356. procedure LoadFromStream(Strm: TStream);
  1357. // load the contents of a file into the data buffer
  1358. procedure LoadFromFile(const Filename: string);
  1359. // save the contents of the data buffer into a stream
  1360. procedure SaveToStream(Strm: TStream);
  1361. // save the contents of the data buffer to a file
  1362. procedure SaveToFile(const Filename: string; const aUnModify: boolean =
  1363. True);
  1364. // save a range of bytes to a stream
  1365. procedure SaveRangeToStream(Strm: TStream; const APosition, ACount:
  1366. integer);
  1367. // undo the last modification, multiple undos are possible
  1368. function Undo: boolean;
  1369. // discard the last undo action (only one single redo is possible)
  1370. function Redo: boolean;
  1371. // empty the data buffer and set the filename (e.g. "Untitled")
  1372. procedure CreateEmptyFile(const TempName: string);
  1373. (* returns a buffer containing parts of the data buffer's contents. the buffer is allocated
  1374. in this routine and must be freed by the caller
  1375. *)
  1376. function BufferFromFile(const aPos: integer; var aCount: integer): PChar;
  1377. // insert some data at the specified position into the data buffer
  1378. procedure InsertBuffer(aBuffer: PChar; const aSize, aPos: integer; const
  1379. UndoDesc: string = ''; const MoveCursor: Boolean = True);
  1380. // append some data at the end of the data buffer
  1381. procedure AppendBuffer(aBuffer: PChar; const aSize: integer; const UndoDesc:
  1382. string = ''; const MoveCursor: Boolean = True);
  1383. // replace the currently selected data with some other data
  1384. procedure ReplaceSelection(aBuffer: PChar; aSize: integer; const UndoDesc:
  1385. string = ''; const MoveCursor: Boolean = True);
  1386. // replace some amount of data
  1387. function Replace(aBuffer: PChar; aPosition, aOldCount, aNewCount: integer;
  1388. const UndoDesc:
  1389. string = ''; const MoveCursor: Boolean = False): integer;
  1390. // get the current data position (depending on the cursor/caret)
  1391. function GetCursorPos: integer;
  1392. // delete 4 bits (=half byte = nibble) from the data buffer (see also @link(InsertNibble))
  1393. function DeleteNibble(const aPos: integer; const HighNibble: boolean; const
  1394. UndoDesc: string = ''): boolean;
  1395. // insert 4 bits (0000) into the data buffer (see also @link(DeleteNibble))
  1396. function InsertNibble(const aPos: integer; const HighNibble: boolean; const
  1397. UndoDesc: string = ''): boolean;
  1398. // convert a part of the data buffer's content from one character table to a different one
  1399. procedure ConvertRange(const aFrom, aTo: integer; const aTransFrom,
  1400. aTransTo: TMPHTranslationKind; const UndoDesc: string = '');
  1401. (* returns the data position of the top left cell and also whether the caret is in the
  1402. character pane, see also @link(SetTopLeftPosition)
  1403. *)
  1404. function GetTopLeftPosition(var oInCharField: boolean): integer;
  1405. (* set top left cell to the given data position and also whether the caret is in the
  1406. character pane (see also @link(GetTopLeftPosition))
  1407. *)
  1408. procedure SetTopLeftPosition(const aPosition: integer; const aInCharField:
  1409. boolean);
  1410. (* show a drop position marker on the cell at the given mouse cursor position
  1411. (see also @link(HideDragCell))
  1412. *)
  1413. function ShowDragCell(const X, Y: integer): integer;
  1414. // hide the drop position marker (see also @link(ShowDragCell))
  1415. procedure HideDragCell;
  1416. // combine two or more changes, so @link(Undo) will discard the at once
  1417. procedure CombineUndo(const aCount: integer; const sDesc: string = '');
  1418. (* translate a byte from the current @link(Translation) to the Windows Codepage
  1419. (see also @link(TranslateFromAnsiChar))
  1420. *)
  1421. function TranslateToAnsiChar(const aByte: byte): char;
  1422. (* translate a byte from Windows Codepage to the current @link(Translation)
  1423. (see also @link(TranslateToAnsiChar))
  1424. *)
  1425. function TranslateFromAnsiChar(const aByte: byte): char;
  1426. // retrieve or set the selection start
  1427. property SelStart: integer read GetSelStart write SetSelStart;
  1428. // retrieve or set the selection end
  1429. property SelEnd: integer read GetSelEnd write SetSelEnd;
  1430. // retrieve the size of the selected data
  1431. property SelCount: integer read GetSelCount;
  1432. // is @link(Undo) possible?
  1433. property CanUndo: boolean read GetCanUndo;
  1434. // is @link(Redo) possible?
  1435. property CanRedo: boolean read GetCanRedo;
  1436. // is the caret in the character or the hex pane ?
  1437. property InCharField: boolean read GetInCharField write SetInCharField;
  1438. // description of the next @link(Undo) action
  1439. property UndoDescription: string read GetUndoDescription;
  1440. // if True, the currently loaded file cannot be overwritten
  1441. property ReadOnlyFile: boolean read FIsFileReadonly write SetReadOnlyFile;
  1442. // if True, changes have been made to the data buffer content
  1443. property Modified: boolean read GetModified write SetModified;
  1444. // retrieves or stores the amount of data in the data buffer
  1445. // when enlarging the data stream, the @link(SetDataSizeFillByte) property
  1446. // tells which value to use to fill the new data
  1447. property DataSize: integer read GetDataSize write SetDataSize;
  1448. // array to the data buffer's content
  1449. property Data[Index: integer]: Byte read GetDataAt write SetDataAt;
  1450. // retrieve or set the data as string
  1451. property AsText: string read GetAsText write SetAsText;
  1452. // retrieve or set the data as hex formatted string (00 01 02 03...)
  1453. property AsHex: string read GetAsHex write SetAsHex;
  1454. // name of the file that has been loaded into the data buffer
  1455. property Filename: string read FFileName;
  1456. // retrieve or set bookmarks programmatically (see also @link(TMPHBookmark))
  1457. property Bookmark[Index: byte]: TMPHBookmark read GetBookmark write
  1458. SetBookmark;
  1459. // has the byte at the given position been modified ? (only in overwrite mode)
  1460. property ByteChanged[index: integer]: boolean read HasChanged write
  1461. SetChanged;
  1462. // retrieves the number of columns (grid columns)
  1463. property ColCountRO: integer read GetPropColCount;
  1464. // retrieves the number of rows (grid rows)
  1465. property RowCountRO: integer read GetPropRowCount;
  1466. // returns True if the mouse cursor is positionned over selected data
  1467. property MouseOverSelection: boolean read GetMouseOverSelection;
  1468. // get the data value at the current caret position, returns -1 if an error occured
  1469. property CurrentValue: integer read GetCurrentValue;
  1470. // pointer to the whole data buffer's contents
  1471. //property DataPointer: Pointer read GetDataPointer;
  1472. // select all data
  1473. procedure SelectAll;
  1474. // retrieves the number of visible columns
  1475. property VisibleColCount;
  1476. // retrieves the number of visible rows
  1477. property VisibleRowCount;
  1478. // the control's canvas
  1479. property Canvas;
  1480. // current column (grid column)
  1481. property Col;
  1482. // first visible column
  1483. property LeftCol;
  1484. // current row (grid row)
  1485. property Row;
  1486. // first visible row (grid row)
  1487. property TopRow;
  1488. // this event is fired when a bookmark is added/modifed/removed
  1489. property OnBookmarkChanged: TNotifyEvent read FOnBookmarkChanged write
  1490. FOnBookmarkChanged;
  1491. // call this procedure to navigate to a bookmarked position
  1492. function GotoBookmark(const Index: integer): boolean;
  1493. // call this function if the external offset formatting changed (see @link(OnGetOffsetText))
  1494. procedure UpdateGetOffsetText;
  1495. // center the current position vertically
  1496. procedure CenterCursorPosition;
  1497. end;
  1498. // published hex editor component
  1499. TMPHexEditor = class(TCustomMPHexEditor)
  1500. published
  1501. // @exclude(inherited)
  1502. property Align;
  1503. // @exclude(inherited)
  1504. property Anchors;
  1505. // @exclude(inherited)
  1506. property BiDiMode;
  1507. // @exclude(inherited)
  1508. property BorderStyle;
  1509. // @exclude(inherited)
  1510. property Constraints;
  1511. // @exclude(inherited)
  1512. property Ctl3D;
  1513. // @exclude(inherited)
  1514. property DragCursor;
  1515. // @exclude(inherited)
  1516. property DragKind;
  1517. // @exclude(inherited)
  1518. property DragMode;
  1519. // @exclude(inherited)
  1520. property Enabled;
  1521. // @exclude(inherited)
  1522. property Font;
  1523. // @exclude(inherited)
  1524. property ImeMode;
  1525. // @exclude(inherited)
  1526. property ImeName;
  1527. // @exclude(inherited)
  1528. property OnClick;
  1529. // @exclude(inherited)
  1530. property OnDblClick;
  1531. // @exclude(inherited)
  1532. property OnDragDrop;
  1533. // @exclude(inherited)
  1534. property OnDragOver;
  1535. // @exclude(inherited)
  1536. property OnEndDock;
  1537. // @exclude(inherited)
  1538. property OnEndDrag;
  1539. // @exclude(inherited)
  1540. property OnEnter;
  1541. // @exclude(inherited)
  1542. property OnExit;
  1543. // @exclude(inherited)
  1544. property OnKeyDown;
  1545. // @exclude(inherited)
  1546. property OnKeyPress;
  1547. // @exclude(inherited)
  1548. property OnKeyUp;
  1549. // @exclude(inherited)
  1550. property OnMouseDown;
  1551. // @exclude(inherited)
  1552. property OnMouseMove;
  1553. // @exclude(inherited)
  1554. property OnMouseUp;
  1555. // @exclude(inherited)
  1556. property OnMouseWheel;
  1557. // @exclude(inherited)
  1558. property OnMouseWheelDown;
  1559. // @exclude(inherited)
  1560. property OnMouseWheelUp;
  1561. // @exclude(inherited)
  1562. property OnStartDock;
  1563. // @exclude(inherited)
  1564. property OnStartDrag;
  1565. // @exclude(inherited)
  1566. property ParentBiDiMode;
  1567. // @exclude(inherited)
  1568. property ParentCtl3D;
  1569. // @exclude(inherited)
  1570. property ParentFont;
  1571. // @exclude(inherited)
  1572. property ParentShowHint;
  1573. // @exclude(inherited)
  1574. property PopupMenu;
  1575. // @exclude(inherited)
  1576. property ScrollBars;
  1577. // @exclude(inherited)
  1578. property ShowHint;
  1579. // @exclude(inherited)
  1580. property TabOrder;
  1581. // @exclude(inherited)
  1582. property TabStop;
  1583. // @exclude(inherited)
  1584. property Visible;
  1585. // see inherited @inherited
  1586. property BytesPerRow;
  1587. // see inherited @inherited
  1588. property BytesPerColumn;
  1589. // see inherited @inherited
  1590. property Translation;
  1591. // see inherited @inherited
  1592. property OffsetFormat;
  1593. // see inherited @inherited
  1594. property CaretKind;
  1595. // see inherited @inherited
  1596. property Colors;
  1597. // see inherited @inherited
  1598. property FocusFrame;
  1599. // see inherited @inherited
  1600. property SwapNibbles;
  1601. // see inherited @inherited
  1602. property MaskChar;
  1603. // see inherited @inherited
  1604. property NoSizeChange;
  1605. // see inherited @inherited
  1606. property AllowInsertMode;
  1607. // see inherited @inherited
  1608. property DrawGridLines;
  1609. // see inherited @inherited
  1610. property WantTabs;
  1611. // see inherited @inherited
  1612. property ReadOnlyView;
  1613. // see inherited @inherited
  1614. property HideSelection;
  1615. // see inherited @inherited
  1616. property GraySelectionIfNotFocused;
  1617. // see inherited @inherited
  1618. property GutterWidth;
  1619. // see inherited @inherited
  1620. property BookmarkBitmap;
  1621. // see inherited @inherited
  1622. property Version;
  1623. // see inherited @inherited
  1624. property MaxUndo;
  1625. // see inherited @inherited
  1626. property InsertMode;
  1627. // see inherited @inherited
  1628. property HexLowerCase;
  1629. // see inherited @inherited
  1630. property OnProgress;
  1631. // see inherited @inherited
  1632. property OnInvalidKey;
  1633. // see inherited @inherited
  1634. property OnTopLeftChanged;
  1635. // see inherited @inherited
  1636. property OnChange;
  1637. // see inherited @inherited
  1638. property DrawGutter3D;
  1639. // see inherited @inherited
  1640. property ShowRuler;
  1641. // see inherited @inherited
  1642. property BytesPerUnit;
  1643. // see inherited @inherited
  1644. property RulerBytesPerUnit;
  1645. // see inherited @inherited
  1646. property ShowPositionIfNotFocused;
  1647. // see inherited @inherited
  1648. property OnSelectionChanged;
  1649. // see inherited @inherited
  1650. property UnicodeChars;
  1651. // see inherited @inherited
  1652. property UnicodeBigEndian;
  1653. // see inherited @inherited
  1654. property OnDrawCell;
  1655. // see inherited @inherited
  1656. property OnBookmarkChanged;
  1657. // see inherited @inherited
  1658. property OnGetOffsetText;
  1659. // see inherited @inherited
  1660. property BytesPerBlock;
  1661. // see inherited @inherited
  1662. property SeparateBlocksInCharField;
  1663. // see inherited @inherited
  1664. property FindProgress;
  1665. // see inherited @inherited
  1666. property RulerNumberBase;
  1667. end;
  1668. // @exclude(undo storage record)
  1669. PMPHUndoRec = ^TMPHUndoRec;
  1670. // @exclude(undo storage record)
  1671. TMPHUndoRec = packed record
  1672. DataLen: integer;
  1673. Flags: TMPHUndoFlags;
  1674. CurPos: integer;
  1675. Pos, Count, ReplCount: cardinal;
  1676. CurTranslation: TMPHTranslationKind;
  1677. CurBPU: Integer;
  1678. Buffer: byte;
  1679. end;
  1680. // @exclude(implements undo/redo)
  1681. TMPHUndoStorage = class(TMemoryStream)
  1682. private
  1683. FCount,
  1684. FUpdateCount: integer;
  1685. FEditor: TCustomMPHexEditor;
  1686. FDescription: string;
  1687. FRedoPointer,
  1688. FLastUndo: PMPHUndoRec;
  1689. FLastUndoSize: integer;
  1690. FLastUndoDesc: string;
  1691. procedure SetCount(const Value: integer);
  1692. procedure ResetRedo;
  1693. procedure CreateRedo(const Rec: TMPHUndoRec);
  1694. function GetUndoKind(const Flags: TMPHUndoFlags): TMPHUndoFlag;
  1695. procedure AddSelection(const APos, ACount: integer);
  1696. function ReadUndoRecord(var aUR: TMPHUndoRec; var SDescription: string):
  1697. TMPHUndoFlag;
  1698. function GetLastUndoKind: TMPHUndoFlag;
  1699. public
  1700. constructor Create(AEditor: TCustomMPHexEditor);
  1701. destructor Destroy; override;
  1702. procedure SetSize(NewSize: longint); override;
  1703. procedure CreateUndo(aKind: TMPHUndoFlag; APosition, ACount, AReplaceCount:
  1704. integer; const SDescription: string = '');
  1705. function CanUndo: boolean;
  1706. function CanRedo: boolean;
  1707. function Redo: boolean;
  1708. function Undo: boolean;
  1709. function BeginUpdate: integer;
  1710. function EndUpdate: integer;
  1711. procedure Reset(AResetRedo: boolean = True);
  1712. procedure RemoveLastUndo;
  1713. property Count: integer read FCount write SetCount;
  1714. property UpdateCount: integer read FUpdateCount;
  1715. property Description: string read FDescription;
  1716. property UndoKind: TMPHUndoFlag read GetLastUndoKind;
  1717. end;
  1718. resourcestring
  1719. // long descriptive names of character translations
  1720. // tkAsIs
  1721. MPH_TK_ASIS = 'Windows';
  1722. // tkDos8
  1723. MPH_TK_DOS8 = 'Dos 8 bits';
  1724. // tkASCII
  1725. MPH_TK_ASCII7 = 'ASCII 7 bits';
  1726. // tkMac
  1727. MPH_TK_MAC = 'Macintosh';
  1728. // tkBCD
  1729. MPH_TK_BCD38 = 'EBCDIC codepage 38';
  1730. // unicode
  1731. MPH_UC = 'Unicode little endian';
  1732. // unicode be
  1733. MPH_UC_BE = 'Unicode big endian';
  1734. // short names (e.g. for status bars) of character translations
  1735. // tkAsIs
  1736. MPH_TK_ASIS_S = 'WIN';
  1737. // tkDos8
  1738. MPH_TK_DOS8_S = 'DOS';
  1739. // tkASCII
  1740. MPH_TK_ASCII7_S = 'ASC';
  1741. // tkMac
  1742. MPH_TK_MAC_S = 'MAC';
  1743. // tkBCD
  1744. MPH_TK_BCD38_S = 'BCD';
  1745. // tkCustom
  1746. MPH_TK_CUSTOM_S = 'Cust';
  1747. // tkCustom
  1748. MPH_TK_CUSTOM = 'Custom translation';
  1749. // unicode
  1750. MPH_UC_S = 'UCLE';
  1751. // unicode be
  1752. MPH_UC_BE_S = 'UCBE';
  1753. const
  1754. // long descriptions of the different translations (e.g. for menues)
  1755. MPHTranslationDesc: array[TMPHTranslationKind] of string = (MPH_TK_ASIS,
  1756. MPH_TK_DOS8, MPH_TK_ASCII7, MPH_TK_MAC,
  1757. MPH_TK_BCD38,
  1758. MPH_TK_CUSTOM);
  1759. // short descriptions of the different translations (e.g. for status bars)
  1760. MPHTranslationDescShort: array[TMPHTranslationKind] of string =
  1761. (MPH_TK_ASIS_S, MPH_TK_DOS8_S, MPH_TK_ASCII7_S, MPH_TK_MAC_S,
  1762. MPH_TK_BCD38_S, MPH_TK_CUSTOM_S);
  1763. // public utility functions
  1764. (* translate a hexadecimal data representation ("a000 cc45 d3 42"...) to binary data
  1765. (see @link(SwapNibbles) for the meaning of the SwapNibbles value)
  1766. *)
  1767. function ConvertHexToBin(aFrom, aTo: PChar; const aCount: integer; const
  1768. SwapNibbles: boolean; var BytesTranslated: integer): PChar;
  1769. (* translate binary data to its hex representation (see @link(ConvertHexToBin)),
  1770. (see @link(SwapNibbles) for the meaning of the SwapNibbles value)
  1771. *)
  1772. function ConvertBinToHex(aFrom, aTo: PChar; const aCount: integer; const
  1773. SwapNibbles: boolean): PChar;
  1774. // convert X and Y into a TGridCoord record
  1775. function GridCoord(aX, aY: longint): TGridCoord;
  1776. // check whether the given key (VK_...) is currently down
  1777. function IsKeyDown(aKey: integer): boolean;
  1778. // get a unique filename in the temporary directory
  1779. function GetTempName: string;
  1780. (* translate an integer to a radix (base) coded string, e.g.<br>
  1781. - IntToRadix(100,16) converts into a hexadecimal (number) string<br>
  1782. - IntToRadix(100,2) converts into a string consisting only of 0 and 1<br>
  1783. - IntToRadix(100,8) means IntToOctal<br>
  1784. <br>
  1785. hint: Radix must be in the range of 2..16*)
  1786. function IntToRadix(Value: integer; Radix: byte): string;
  1787. function IntToRadix64(Value: int64; Radix: byte): string;
  1788. // translate an integer to a radix coded string and left fill with 0 (see also @link(IntToRadix))
  1789. function IntToRadixLen(Value: integer; Radix, Len: byte): string;
  1790. function IntToRadixLen64(Value: int64; Radix, Len: byte): string;
  1791. // translate an integer to an octal string (see also @link(IntToRadix))
  1792. function IntToOctal(const Value: integer): string;
  1793. (* translate a radix coded number string into an integer, e.g.<br>
  1794. - RadixToInt('0f', 16) => 15<br>
  1795. - RadixToInt('755', 8) => 493
  1796. *)
  1797. function RadixToInt(Value: string; Radix: byte): integer;
  1798. function RadixToInt64(Value: string; Radix: byte): int64;
  1799. (* 64 bit unsigned integer arithmetics *)
  1800. // division of two unsigned int64 values, may raise an exception on error
  1801. function DivideU64(const Dividend, Divisor: int64): int64;
  1802. // division of two unsigned int64 values, returns false if an error occurred
  1803. function TryDivideU64(const Dividend, Divisor: int64;
  1804. var Val: int64): boolean;
  1805. // modulo of two unsigned int64 values, may raise an exception on error
  1806. function ModuloU64(const Dividend, Divisor: int64): int64;
  1807. // modulo of two unsigned int64 values, returns false if an error occurred
  1808. function TryModuloU64(const Dividend, Divisor: int64;
  1809. var Val: int64): boolean;
  1810. // multiplication of two unsigned int64 values, may raise an exception on error
  1811. function MultiplyU64(const Multiplier, Multiplicator: int64): int64;
  1812. // multiplication of two unsigned int64 values, returns false if an error occurred
  1813. function TryMultiplyU64(const Multiplier, Multiplicator: int64;
  1814. var Val: int64): boolean;
  1815. // addition of two unsigned int64 values, may raise an exception on error
  1816. function AddU64(const Addend1, Addend2: int64): int64;
  1817. // addition of two unsigned int64 values, returns false if an error occurred
  1818. function TryAddU64(const Addend1, Addend2: int64;
  1819. var Val: int64): boolean;
  1820. // subtraction of two unsigned int64 values, may raise an exception on error
  1821. function SubtractU64(const Minuend, Subtrahend: int64): int64;
  1822. // subtraction of two unsigned int64 values, returns false if an error occurred
  1823. function TrySubtractU64(const Minuend, Subtrahend: int64;
  1824. var Val: int64): boolean;
  1825. (* try to find the correct radix (based on prefix/suffix) and return the number, known
  1826. prefixes/suffixes are:<br>
  1827. 0x&lt;number&gt;, 0X&lt;number&gt;, $&lt;number&gt;, &lt;number&gt;h, &lt;number&gt;H: radix 16<br>
  1828. o&lt;number&gt;, O&lt;number&gt;, 0&lt;number&gt;, &lt;number&gt;o, &lt;number&gt;O: radix 8<br>
  1829. %&lt;number&gt;, &lt;number&gt;%: radix 2<br>
  1830. otherwise: radix 10
  1831. *)
  1832. function CheckRadixToInt(Value: string): integer;
  1833. function CheckRadixToInt64(Value: string): int64;
  1834. // translate an number string built on radix 8 into an integer (see also @link(RadixToInt))
  1835. function OctalToInt(const Value: string): integer;
  1836. // swap lo and high byte of a widechar
  1837. procedure SwapWideChar(var WChar: WideChar);
  1838. // @exclude(fade a color to a gray value)
  1839. function FadeToGray(aColor: TColor): TColor;
  1840. (* translate data from Ansi to a different character set (see also @link(TMPHTranslationKind))<br>
  1841. - TType: translate to this character set<br>
  1842. - aBuffer: pointer to source data<br>
  1843. - bBuffer: pointer to target data, must be allocated (may equal to aBuffer)<br>
  1844. - aCount: number of bytes to translate
  1845. *)
  1846. procedure TranslateBufferFromAnsi(const TType: TMPHTranslationKind; aBuffer,
  1847. bBuffer: PChar; const aCount: integer);
  1848. // translate data from a different character set to Ansi (see also @link(TranslateBufferFromAnsi))
  1849. procedure TranslateBufferToAnsi(const TType: TMPHTranslationKind; aBuffer,
  1850. bBuffer: PChar; const aCount: integer);
  1851. // compatibility
  1852. {$IFNDEF DELPHI6UP}
  1853. procedure RaiseLastOSError;
  1854. {$ENDIF}
  1855. // returns the lower of the two numbers
  1856. function Min(a1, a2: integer): integer;
  1857. // returns the higer of the two numbers
  1858. function Max(a1, a2: integer): integer;
  1859. var
  1860. (* translation tables for tkCustom *)
  1861. // this character conversion is used in translations from tkAsIs to tkCustom (see @link(TMPHTranslationKind))
  1862. MPHCustomCharConv: TMPHCharConv;
  1863. const
  1864. (* standard offset formats *)
  1865. // standard offset format: hex, auto min width, prefixed by 0x
  1866. MPHOffsetHex = '-!10:0x|';
  1867. // standard offset format: decimal
  1868. MPHOffsetDec = 'a:|';
  1869. // standard offset format: octal, suffixed by a small "o"
  1870. MPHOffsetOct = '0!8:o|';
  1871. implementation
  1872. uses
  1873. Consts, {$IFDEF DELPHI6UP}RTLConsts, {$ENDIF}ImgList, StdCtrls, SysConst;
  1874. const
  1875. MPH_VERSION = 'February 06, 2006; © markus stephany, vcl[at]mirkes[dot]de';
  1876. resourcestring
  1877. // undo descriptions
  1878. UNDO_BYTESCHANGED = 'Change byte(s)';
  1879. UNDO_REMOVED = 'Remove data';
  1880. UNDO_INSERT = 'Insert buffer';
  1881. UNDO_REPLACE = 'Replace';
  1882. UNDO_APPEND = 'Append buffer';
  1883. UNDO_INSNIBBLE = 'Insert nibble';
  1884. UNDO_DELNIBBLE = 'Delete nibble';
  1885. UNDO_CONVERT = 'Convert';
  1886. UNDO_SELECTION = 'Cursor movement';
  1887. UNDO_COMBINED = 'Multiple modification';
  1888. UNDO_ALLDATA = 'All data saved';
  1889. UNDO_NOUNDO = 'No undo';
  1890. // error messages
  1891. ERR_FILE_OPEN_FAILED = 'Cannot open %s.'#13#10'(%s.)';
  1892. ERR_FILE_READONLY = 'Cannot save readonly file %s.';
  1893. ERR_INVALID_BOOKMARK = 'Invalid bookmark index';
  1894. ERR_INVALID_SELSTART = 'Invalid selection start';
  1895. ERR_INVALID_SELEND = 'Invalid selection end';
  1896. ERR_INVALID_BYTESPERLINE = 'Invalid bytes per line argument';
  1897. ERR_INVALID_BUFFERFROMFILE = 'Invalid buffer from file argument';
  1898. ERR_INVALID_BYTESPERCOL = 'Invalid bytes per column argument';
  1899. ERR_INVALID_BOOKMARKBMP = 'Invalid bookmark bitmap (must be 10 x 200 px)';
  1900. ERR_CANCELLED = 'Operation cancelled';
  1901. ERR_MISSING_FORMATCHAR = 'Missing char in offset format: %s';
  1902. ERR_INVALID_FORMATRADIX =
  1903. 'Invalid radix in offset format (%xh), allowed: 02h..10h';
  1904. ERR_INVALID_RADIXCHAR =
  1905. 'Invalid character %s, cannot convert using radix %xh';
  1906. ERR_INVALID_BPU = 'Invalid bytes per unit value %d, allowed: 1,2,4,8';
  1907. ERR_INVALID_BPU_U = 'BytesPerUnit must be set to 2 in unicode mode';
  1908. ERR_INVALID_RBPU =
  1909. 'Invalid ruler bytes per unit value %d, allowed: -1,1,2,4,8';
  1910. ERR_DATA_BOUNDS = 'Data position/length out of data bounds';
  1911. ERR_NO_TRANSLATION_IN_UNICODE_MODE =
  1912. 'Translations cannot be used in unicode mode';
  1913. ERR_ODD_FILESIZE_UNICODE = 'Cannot use unicode mode with odd-sized files';
  1914. ERR_FIXED_FILESIZE = 'Cannot change fixed filesize';
  1915. ERR_NOUNDO = 'Cannot update undo storage';
  1916. // new, empty file
  1917. UNNAMED_FILE = 'Untitled';
  1918. const
  1919. // fixed cols/rows
  1920. GRID_FIXED = 2;
  1921. // available undo descriptions
  1922. STRS_UNDODESC: array[ufKindBytesChanged..ufKindAllData] of string =
  1923. (UNDO_BYTESCHANGED, UNDO_REMOVED, UNDO_INSERT, UNDO_REPLACE, UNDO_APPEND,
  1924. UNDO_INSNIBBLE, UNDO_DELNIBBLE, UNDO_CONVERT, UNDO_SELECTION, UNDO_COMBINED,
  1925. UNDO_ALLDATA);
  1926. // valid hex characters
  1927. HEX_LOWER = '0123456789abcdef';
  1928. HEX_UPPER = '0123456789ABCDEF';
  1929. HEX_ALLCHARS = HEX_LOWER + HEX_UPPER;
  1930. {$IFNDEF DELPHI6UP}
  1931. procedure RaiseLastOSError;
  1932. begin
  1933. RaiseLastWin32Error;
  1934. end;
  1935. {$ENDIF}
  1936. // invert the given color
  1937. function Invert(Color: TColor): TColor;
  1938. begin
  1939. Result := ColorToRGB(Color) xor $00FFFFFF;
  1940. end;
  1941. // translate the buffer from ANSI to the given translation mode
  1942. procedure TranslateBufferFromAnsi(const TType: TMPHTranslationKind; aBuffer,
  1943. bBuffer: PChar; const aCount: integer);
  1944. var
  1945. LIntLoop: integer;
  1946. begin
  1947. case TType of
  1948. // changed 04/18/04: bBuffer and aBuffer were interchanged!
  1949. tkAsIs: Move(aBuffer^, bBuffer^, aCount);
  1950. tkDOS8,
  1951. tkASCII: CharToOEMBuff(aBuffer, bBuffer, aCount);
  1952. tkMAC: if aCount > 0 then
  1953. for LIntLoop := 0 to Pred(aCount) do
  1954. bBuffer[LIntLoop] :=
  1955. MPH_CCONV_MAC[cctFromAnsi][Ord(aBuffer[LIntLoop])];
  1956. tkBCD: if aCount > 0 then
  1957. for LIntLoop := 0 to Pred(aCount) do
  1958. bBuffer[LIntLoop] :=
  1959. MPH_CCONV_BCD38[cctFromAnsi][Ord(aBuffer[LIntLoop])];
  1960. tkCustom: if aCount > 0 then
  1961. for LIntLoop := 0 to Pred(aCount) do
  1962. bBuffer[LIntLoop] :=
  1963. MPHCustomCharConv[cctFromAnsi][Ord(aBuffer[LIntLoop])];
  1964. end;
  1965. end;
  1966. // translate the buffer to ANSI from the given translation mode
  1967. procedure TranslateBufferToAnsi(const TType: TMPHTranslationKind; aBuffer,
  1968. bBuffer: PChar; const aCount: integer);
  1969. var
  1970. LIntLoop: integer;
  1971. begin
  1972. case TType of
  1973. tkAsIs: Move(aBuffer^, bBuffer^, aCount);
  1974. tkDOS8,
  1975. tkASCII: OEMToCharBuff(aBuffer, bBuffer, aCount);
  1976. tkMAC: if aCount > 0 then
  1977. for LIntLoop := 0 to Pred(aCount) do
  1978. bBuffer[LIntLoop] := MPH_CCONV_MAC[cctToAnsi][Ord(aBuffer[LIntLoop])];
  1979. tkBCD: if aCount > 0 then
  1980. for LIntLoop := 0 to Pred(aCount) do
  1981. bBuffer[LIntLoop] :=
  1982. MPH_CCONV_BCD38[cctToAnsi][Ord(aBuffer[LIntLoop])];
  1983. tkCustom: if aCount > 0 then
  1984. for LIntLoop := 0 to Pred(aCount) do
  1985. bBuffer[LIntLoop] :=
  1986. MPHCustomCharConv[cctToAnsi][Ord(aBuffer[LIntLoop])];
  1987. end;
  1988. end;
  1989. // ansi to oem
  1990. function OEM2Char(aByte: byte): char;
  1991. var
  1992. LszBuf: array[0..1] of char;
  1993. begin
  1994. LszBuf[0] := char(aByte);
  1995. LszBuf[1] := #0;
  1996. OEMToChar(LSzBuf, LSzBuf);
  1997. Result := LSzBuf[0];
  1998. end;
  1999. // oem to ansi
  2000. function Char2OEM(aByte: byte): char;
  2001. var
  2002. LszBuf: array[0..1] of char;
  2003. begin
  2004. LszBuf[0] := char(aByte);
  2005. LszBuf[1] := #0;
  2006. CharToOEM(LSzBuf, LSzBuf);
  2007. Result := LSzBuf[0];
  2008. end;
  2009. (* helper functions *)
  2010. // get a temporary file name
  2011. function GetTempName: string;
  2012. var
  2013. LStrTemp: string;
  2014. begin
  2015. SetLength(LStrTemp, MAX_PATH + 1);
  2016. SetLength(LStrTemp, GetTempPath(MAX_PATH, @LStrTemp[1]));
  2017. LStrTemp := Trim(LStrTemp);
  2018. {$IFDEF DELPHI6UP}
  2019. LstrTemp := IncludeTrailingPathDelimiter(LstrTemp);
  2020. {$ELSE}
  2021. if LStrTemp[Length(LStrTemp)] <> '\' then
  2022. LStrTemp := LStrTemp + '\';
  2023. {$ENDIF}
  2024. repeat
  2025. Result := LStrTemp + IntToHex(GetTickCount, 8) + '.MPHT';
  2026. until GetFileAttributes(PChar(Result)) = $FFFFFFFF;
  2027. end;
  2028. // can the file be opened for reading (possibly read only) ?
  2029. function CanOpenFile(const aName: TFileName; var ReadOnly: boolean): boolean;
  2030. var
  2031. LHdlFile: THandle;
  2032. begin
  2033. Result := False;
  2034. ReadOnly := True;
  2035. LHdlFile := FileOpen(aName, fmOpenRead or fmShareDenyNone);
  2036. if LHdlFile <> INVALID_HANDLE_VALUE then
  2037. begin
  2038. FileClose(LHdlFile);
  2039. Result := True;
  2040. try
  2041. LHdlFile := FileOpen(aName, fmOpenReadWrite);
  2042. if LHdlFile <> INVALID_HANDLE_VALUE then
  2043. begin
  2044. FileClose(LHdlFile);
  2045. ReadOnly := False;
  2046. end;
  2047. except
  2048. Result := True;
  2049. ReadOnly := True;
  2050. end;
  2051. end;
  2052. end;
  2053. // is that key pressed ?
  2054. function IsKeyDown(aKey: integer): boolean;
  2055. begin
  2056. Result := (GetKeyState(aKey) and (not 1)) <> 0;
  2057. end;
  2058. // return the lesser value
  2059. function Min(a1, a2: integer): integer;
  2060. begin
  2061. if a1 < a2 then
  2062. Result := a1
  2063. else
  2064. Result := a2;
  2065. end;
  2066. // return the bigger value
  2067. function Max(a1, a2: integer): integer;
  2068. begin
  2069. if a1 > a2 then
  2070. Result := a1
  2071. else
  2072. Result := a2;
  2073. end;
  2074. // cast x,y to grid coord
  2075. function GridCoord(aX, aY: longint): TGridCoord;
  2076. begin
  2077. Result.x := aX;
  2078. Result.y := aY;
  2079. end;
  2080. // convert '00 01 02...' to binary data
  2081. function ConvertHexToBin(aFrom, aTo: PChar; const aCount: integer;
  2082. const SwapNibbles: boolean; var BytesTranslated: integer): PChar;
  2083. var
  2084. LBoolHi: boolean;
  2085. LIntLoop: integer;
  2086. LBytCurrent: byte;
  2087. LChrCurrent: char;
  2088. begin
  2089. Result := aTo;
  2090. BytesTranslated := 0;
  2091. LBoolHi := True;
  2092. LBytCurrent := 0;
  2093. for LIntLoop := 0 to Pred(aCount) do
  2094. if Pos(aFrom[LIntLoop], HEX_ALLCHARS) <> 0 then
  2095. begin
  2096. LChrCurrent := UpCase(aFrom[LIntLoop]);
  2097. if LBoolHi then
  2098. LBytCurrent := ((Pos(LChrCurrent, HEX_UPPER) - 1) * 16)
  2099. else
  2100. LBytCurrent := LBytCurrent or ((Pos(LChrCurrent, HEX_UPPER) - 1));
  2101. LBoolHi := not LBoolHi;
  2102. if LBoolHi then
  2103. begin
  2104. if SwapNibbles then
  2105. aTo[BytesTranslated] := char(((LBytCurrent and 15) * 16) or
  2106. ((LBytCurrent and $F0) shr 4))
  2107. else
  2108. aTo[BytesTranslated] := char(LBytCurrent);
  2109. Inc(BytesTranslated);
  2110. end;
  2111. end;
  2112. end;
  2113. // convert binary data to '00 01 02...'
  2114. function ConvertBinToHex(aFrom, aTo: PChar; const aCount: integer;
  2115. const SwapNibbles: boolean): PChar;
  2116. var
  2117. LIntLoop: integer;
  2118. LByteCurrent: byte;
  2119. LIntLoop2: integer;
  2120. begin
  2121. Result := aTo;
  2122. LIntLoop2 := 0;
  2123. for LIntLoop := 0 to Pred(aCount) do
  2124. begin
  2125. LByteCurrent := Ord(aFrom[LIntLoop]);
  2126. if SwapNibbles then
  2127. begin
  2128. aTo[LIntLoop2] := UpCase(HEX_UPPER[(LByteCurrent and 15) + 1]);
  2129. aTo[LIntLoop2 + 1] := UpCase(HEX_UPPER[(LByteCurrent shr 4) + 1])
  2130. end
  2131. else
  2132. begin
  2133. aTo[LIntLoop2 + 1] := UpCase(HEX_UPPER[(LByteCurrent and 15) + 1]);
  2134. aTo[LIntLoop2] := UpCase(HEX_UPPER[(LByteCurrent shr 4) + 1])
  2135. end;
  2136. Inc(LIntLoop2, 2);
  2137. end;
  2138. aTO[LIntLoop2] := #0;
  2139. end;
  2140. // translate an integer to a radix coded string
  2141. function IntToRadix(Value: integer; Radix: byte): string;
  2142. begin
  2143. Result := IntToRadixLen(Value, Radix, 0);
  2144. end;
  2145. function IntToRadix64(Value: int64; Radix: byte): string;
  2146. begin
  2147. Result := IntToRadixLen64(Value, Radix, 0);
  2148. end;
  2149. // translate an integer to a radix coded string and left fill with 0
  2150. function IntToRadixLen(Value: integer; Radix, Len: byte): string;
  2151. var
  2152. LCrdTemp: cardinal absolute Value;
  2153. begin
  2154. Result := '';
  2155. repeat
  2156. Result := HEX_UPPER[(LCrdTemp mod Radix) + 1] + Result;
  2157. LCrdTemp := LCrdTemp div Radix;
  2158. until LCrdTemp = 0;
  2159. while Length(Result) < Len do
  2160. Result := '0' + Result;
  2161. end;
  2162. // unsigned 64 bit integer routines (division and modulo)
  2163. // this code is derived from assembler code written by
  2164. // Norbert Juffa, found on "the assembly gems page"
  2165. // (http://www.df.lth.se/~john_e/)
  2166. procedure _UModDiv64;
  2167. begin
  2168. asm
  2169. // divisor > 2^32-1 ?
  2170. test ecx, ecx
  2171. // yes, divisor > 32^32-1
  2172. jnz @big_divisor
  2173. // only one division needed ? (ecx = 0)
  2174. cmp edx, ebx
  2175. // yes, one division sufficient
  2176. jb @one_div
  2177. // save dividend-lo in ecx
  2178. mov ecx, eax
  2179. // get dividend-hi
  2180. mov eax, edx
  2181. // zero extend it into edx:eax
  2182. xor edx, edx
  2183. // quotient-hi in eax
  2184. div ebx
  2185. // ecx = quotient-hi, eax =dividend-lo
  2186. xchg eax, ecx
  2187. @one_div:
  2188. // eax = quotient-lo
  2189. div ebx
  2190. //ebx = remainder-lo
  2191. mov ebx, edx
  2192. //edx = quotient-hi(quotient in edx:eax)
  2193. mov edx, ecx
  2194. // ecx = remainder-hi (rem. in ecx:ebx)
  2195. xor ecx, ecx
  2196. jmp @cleanup;
  2197. @big_divisor:
  2198. // save dividend
  2199. push edx
  2200. push eax
  2201. // divisor now in edi:ebx and ecx:esi
  2202. mov esi, ebx
  2203. mov edi, ecx
  2204. // shift both divisor and and dividend right by 1 bit
  2205. shr edx, 1
  2206. rcr eax, 1
  2207. ror edi, 1
  2208. rcr ebx, 1
  2209. // ecx = number of remaining shifts
  2210. bsr ecx, ecx
  2211. // scale down divisor and dividend such that divisor less than 2^32 (i.e. fits in ebx)
  2212. shrd ebx, edi, CL
  2213. shrd eax, edx, CL
  2214. shr edx, CL
  2215. // restore original divisor (edi:esi)
  2216. rol edi, 1
  2217. // compute quotient
  2218. div ebx
  2219. // get dividend lo-word
  2220. pop ebx
  2221. // save quotient
  2222. mov ecx, eax
  2223. // quotient * divisor hi-word (low only)
  2224. imul edi, eax
  2225. // quotient * divisor lo-word
  2226. mul esi
  2227. // edx:eax = quotient * divisor
  2228. add edx, edi
  2229. // dividend-lo - (quot.*divisor)-lo
  2230. sub ebx, eax
  2231. // get quotient
  2232. mov eax, ecx
  2233. // restore dividend hi-word
  2234. pop ecx
  2235. // subtract divisor * quot. from dividend
  2236. sbb ecx, edx
  2237. // 0 if remainder > 0, else FFFFFFFFh
  2238. sbb edx, edx
  2239. // nothing to add
  2240. and esi, edx
  2241. // back if remainder positive
  2242. and edi, edx
  2243. // correct remaider and quotient if necessary
  2244. add ebx, esi
  2245. adc ecx, edi
  2246. add eax, edx
  2247. // clear hi-word of quot (eax<=FFFFFFFFh)
  2248. xor edx, edx
  2249. @cleanup:
  2250. end;
  2251. end;
  2252. {$WARNINGS OFF}
  2253. function UDiv64(I1, I2: Int64): int64;
  2254. begin
  2255. asm
  2256. // save registers
  2257. push ebp
  2258. push ebx
  2259. push esi
  2260. push edi
  2261. // load I2 into ebx/ecx
  2262. mov ebx, [ebp+$08];
  2263. mov ecx, [ebp+$0c];
  2264. // load I1 into eax/edx
  2265. mov eax, [ebp+$10];
  2266. mov edx, [ebp+$14];
  2267. call _UModDiv64
  2268. // store result (division result is in eax:edx)
  2269. mov [ebp-$08], eax;
  2270. mov [ebp-$04], edx;
  2271. // restore registers
  2272. pop edi
  2273. pop esi
  2274. pop ebx
  2275. pop ebp
  2276. end;
  2277. end;
  2278. function UMod64(I1, I2: Int64): int64;
  2279. begin
  2280. asm
  2281. // save registers
  2282. push ebp
  2283. push ebx
  2284. push esi
  2285. push edi
  2286. // load I2 into ebx/ecx
  2287. mov ebx, [ebp+$08];
  2288. mov ecx, [ebp+$0c];
  2289. // load I1 into eax/edx
  2290. mov eax, [ebp+$10];
  2291. mov edx, [ebp+$14];
  2292. call _UModDiv64
  2293. // store result (division remainder is in ebx:ecx)
  2294. mov [ebp-$08], ebx;
  2295. mov [ebp-$04], ecx;
  2296. // restore registers
  2297. pop edi
  2298. pop esi
  2299. pop ebx
  2300. pop ebp
  2301. end;
  2302. end;
  2303. {$WARNINGS ON}
  2304. (* 64 bit unsigned integer arithmetics *)
  2305. function DivideU64(const Dividend, Divisor: int64): int64;
  2306. begin
  2307. Result := UDiv64(Dividend, Divisor);
  2308. end;
  2309. function TryDivideU64(const Dividend, Divisor: int64;
  2310. var Val: int64): boolean;
  2311. begin
  2312. Result := True;
  2313. try
  2314. Val := UDiv64(Dividend, Divisor);
  2315. except
  2316. Result := False;
  2317. end;
  2318. end;
  2319. function ModuloU64(const Dividend, Divisor: int64): int64;
  2320. begin
  2321. Result := UMod64(Dividend, Divisor);
  2322. end;
  2323. function TryModuloU64(const Dividend, Divisor: int64;
  2324. var Val: int64): boolean;
  2325. begin
  2326. Result := True;
  2327. try
  2328. Val := UMod64(Dividend, Divisor);
  2329. except
  2330. Result := False;
  2331. end;
  2332. end;
  2333. // unsigned 64 bit integer routines (multiplication, addition, substraction)
  2334. // this code is derived from assembler code found in the online book
  2335. // "Art of Assembly Programming" maintained by Randall Hyde
  2336. // (http://webster.cs.ucr.edu/)
  2337. function TryMultiplyU64(const Multiplier, Multiplicator: int64;
  2338. var Val: int64): boolean;
  2339. asm
  2340. // save registers
  2341. push ebx
  2342. push esi
  2343. mov byte ptr result, 1
  2344. // store val pointer
  2345. mov esi, eax
  2346. // multiply lo dword of multiplier * lo dword of multiplicator
  2347. mov eax, dword ptr Multiplier
  2348. mul dword ptr Multiplicator
  2349. // save lo dword
  2350. mov dword [esi], eax
  2351. // save hi dword of partial product
  2352. mov ecx, edx
  2353. // multiply lo dword of multiplier * hi dword of multiplicator
  2354. mov eax, dword ptr Multiplier
  2355. mul dword ptr Multiplicator+4
  2356. // add to the partial product (including carry)
  2357. add eax, ecx
  2358. adc edx, 0
  2359. // save partial product
  2360. mov ebx, eax
  2361. mov ecx, edx
  2362. // multiply hi dword of multiplier * lo dword of multiplicator
  2363. mov eax, dword ptr Multiplier+4
  2364. mul dword ptr Multiplicator
  2365. // add the partial product
  2366. add eax, ebx
  2367. // save the partial product
  2368. mov dword ptr [esi+4], eax
  2369. // add in the carry flag
  2370. adc ecx, edx
  2371. // save carry
  2372. pushfd
  2373. // multiply hi dword of multiplier * hi dword of multiplicator
  2374. mov eax, dword ptr Multiplier+4
  2375. mul dword ptr Multiplicator+4
  2376. // load carry
  2377. popfd
  2378. // add partial product + carry
  2379. adc eax, ecx
  2380. adc edx, 0
  2381. // check overflow
  2382. test eax, eax
  2383. jnz @over
  2384. test edx, edx
  2385. jz @finish
  2386. @over:
  2387. // overflow
  2388. mov byte ptr result, 0
  2389. @finish:
  2390. // restore register
  2391. pop esi
  2392. pop ebx
  2393. end;
  2394. function MultiplyU64(const Multiplier, Multiplicator: int64): int64;
  2395. begin
  2396. if not TryMultiplyU64(Multiplier, Multiplicator, Result) then
  2397. raise EIntOverflow.Create(SIntOverflow);
  2398. end;
  2399. function TryAddU64(const Addend1, Addend2: int64;
  2400. var Val: int64): boolean;
  2401. asm
  2402. mov byte ptr result, 1
  2403. // store val pointer
  2404. mov edx, eax
  2405. // add lo dwords
  2406. mov eax, dword ptr Addend1
  2407. add eax, dword ptr Addend2
  2408. // store lo dword
  2409. mov dword ptr [edx], eax
  2410. // add hi dwords + carry
  2411. mov eax, dword ptr Addend1+4
  2412. adc eax, dword ptr Addend2+4
  2413. // store hi dword
  2414. mov dword ptr [edx+4], eax
  2415. // check carry
  2416. jnc @finish
  2417. mov byte ptr result, 0
  2418. @finish:
  2419. end;
  2420. function AddU64(const Addend1, Addend2: int64): int64;
  2421. begin
  2422. if not TryAddU64(Addend1, Addend2, Result) then
  2423. raise EIntOverflow.Create(SIntOverflow);
  2424. end;
  2425. function TrySubtractU64(const Minuend, Subtrahend: int64;
  2426. var Val: int64): boolean;
  2427. asm
  2428. mov byte ptr result, 1
  2429. // store val pointer
  2430. mov edx, eax
  2431. // subtract lo dwords
  2432. mov eax, dword ptr Minuend
  2433. sub eax, dword ptr Subtrahend
  2434. // store lo dword
  2435. mov dword ptr [edx], eax
  2436. // subtract hi dwords - carry
  2437. mov eax, dword ptr Minuend+4
  2438. sbb eax, dword ptr Subtrahend+4
  2439. // store hi dword
  2440. mov dword ptr [edx+4], eax
  2441. // check carry
  2442. jnc @finish
  2443. mov byte ptr result, 0
  2444. @finish:
  2445. end;
  2446. function SubtractU64(const Minuend, Subtrahend: int64): int64;
  2447. begin
  2448. if not TrySubtractU64(Minuend, Subtrahend, Result) then
  2449. raise EIntOverflow.Create(SIntOverflow);
  2450. end;
  2451. function IntToRadixLen64(Value: int64; Radix, Len: byte): string;
  2452. begin
  2453. Result := '';
  2454. repeat
  2455. Result := HEX_UPPER[UMod64(Value, Radix) + 1] + Result;
  2456. Value := UDiv64(Value, Radix);
  2457. until Value = 0;
  2458. while Length(Result) < Len do
  2459. Result := '0' + Result;
  2460. end;
  2461. // translate an integer value to an octal string
  2462. function IntToOctal(const Value: integer): string;
  2463. begin
  2464. Result := IntToRadix(Value, 8);
  2465. end;
  2466. // translate a radix coded string into an integer
  2467. function RadixToInt(Value: string; Radix: byte): integer;
  2468. var
  2469. LCrdTemp: cardinal absolute Result;
  2470. begin
  2471. LCrdTemp := 0;
  2472. Value := UpperCase(Value);
  2473. while Value <> '' do
  2474. begin
  2475. if not (Pos(Value[1], HEX_UPPER) in [1..Radix]) then
  2476. raise EMPHexEditor.CreateFmt(ERR_INVALID_RADIXCHAR, [Value[1], Radix]);
  2477. LCrdTemp := LCrdTemp * Radix + cardinal(Pos(Value[1], HEX_UPPER) - 1);
  2478. Delete(Value, 1, 1);
  2479. end;
  2480. end;
  2481. function RadixToInt64(Value: string; Radix: byte): int64;
  2482. begin
  2483. Result := 0;
  2484. Value := UpperCase(Value);
  2485. while Value <> '' do
  2486. begin
  2487. if not (Pos(Value[1], HEX_UPPER) in [1..Radix]) then
  2488. raise EMPHexEditor.CreateFmt(ERR_INVALID_RADIXCHAR, [Value[1], Radix]);
  2489. Result := Result * Radix + cardinal(Pos(Value[1], HEX_UPPER) - 1);
  2490. Delete(Value, 1, 1);
  2491. end;
  2492. end;
  2493. (* try to find the correct radix (based on prefix/suffix) and return the number, known
  2494. prefixes/suffixes are:<br>
  2495. 0x<number>, 0X<number>, $<number>, <number>h, <number>H: radix 16<br>
  2496. o<number>, O<number>, <number>o, <number>O: radix 8<br>
  2497. %<number>, <number>%: radix 2<br>
  2498. otherwise: radix 10
  2499. *)
  2500. function CheckRadixToInt(Value: string): integer;
  2501. begin
  2502. // hex
  2503. if UpperCase(Copy(Value, 1, 2)) = '0X' then
  2504. Result := RadixToInt(Copy(Value, 3, MaxInt), 16)
  2505. else if Copy(Value, 1, 1) = '$' then
  2506. Result := RadixToInt(Copy(Value, 2, MaxInt), 16)
  2507. else if UpperCase(Copy(Value, Length(Value), 1)) = 'H' then
  2508. Result := RadixToInt(Copy(Value, 1, Length(Value) - 1), 16)
  2509. else {// octal} if UpperCase(Copy(Value, Length(Value), 1)) = 'O' then
  2510. Result := RadixToInt(Copy(Value, 1, Length(Value) - 1), 8)
  2511. else if UpperCase(Copy(Value, 1, 1)) = 'O' then
  2512. Result := RadixToInt(Copy(Value, 2, MaxInt), 8)
  2513. (* removed, is ambigous else if (Copy(Value, 1, 1) = '0') and (AllCharsIn(['0'..'7'])) then
  2514. Result := RadixToInt(Value, 8)*)
  2515. else {// binary} if UpperCase(Copy(Value, Length(Value), 1)) = '%' then
  2516. Result := RadixToInt(Copy(Value, 1, Length(Value) - 1), 2)
  2517. else if UpperCase(Copy(Value, 1, 1)) = '%' then
  2518. Result := RadixToInt(Copy(Value, 2, MaxInt), 2)
  2519. else
  2520. // decimal
  2521. Result := StrToInt(Value);
  2522. end;
  2523. function CheckRadixToInt64(Value: string): int64;
  2524. begin
  2525. // hex
  2526. if UpperCase(Copy(Value, 1, 2)) = '0X' then
  2527. Result := RadixToInt64(Copy(Value, 3, MaxInt), 16)
  2528. else if Copy(Value, 1, 1) = '$' then
  2529. Result := RadixToInt64(Copy(Value, 2, MaxInt), 16)
  2530. else if UpperCase(Copy(Value, Length(Value), 1)) = 'H' then
  2531. Result := RadixToInt64(Copy(Value, 1, Length(Value) - 1), 16)
  2532. else {// octal} if UpperCase(Copy(Value, Length(Value), 1)) = 'O' then
  2533. Result := RadixToInt64(Copy(Value, 1, Length(Value) - 1), 8)
  2534. else if UpperCase(Copy(Value, 1, 1)) = 'O' then
  2535. Result := RadixToInt64(Copy(Value, 2, MaxInt), 8)
  2536. (* removed, is ambigous else if (Copy(Value, 1, 1) = '0') and (AllCharsIn(['0'..'7'])) then
  2537. Result := RadixToInt(Value, 8)*)
  2538. else {// binary} if UpperCase(Copy(Value, Length(Value), 1)) = '%' then
  2539. Result := RadixToInt64(Copy(Value, 1, Length(Value) - 1), 2)
  2540. else if UpperCase(Copy(Value, 1, 1)) = '%' then
  2541. Result := RadixToInt64(Copy(Value, 2, MaxInt), 2)
  2542. else
  2543. // decimal
  2544. Result := StrToInt64(Value)
  2545. end;
  2546. // translate an octal to an integer
  2547. function OctalToInt(const Value: string): integer;
  2548. begin
  2549. Result := RadixToInt(Value, 8);
  2550. end;
  2551. // swap lo and high byte of a widechar
  2552. procedure SwapWideChar(var WChar: WideChar);
  2553. var
  2554. LWrdChar: word absolute WChar;
  2555. begin
  2556. LWrdChar := Swap(LWrdChar);
  2557. end;
  2558. // fade a color to a gray value
  2559. function FadeToGray(aColor: TColor): TColor;
  2560. var
  2561. LBytGray: byte;
  2562. begin
  2563. aColor := ColorToRGB(aColor);
  2564. LBytGray := HiByte(GetRValue(aColor) * 74 + GetGValue(aColor) * 146 +
  2565. GetBValue(aColor) * 36);
  2566. Result := RGB(LBytGray, LBytGray, LBytGray);
  2567. end;
  2568. (* TCustomMPHexEditor *)
  2569. constructor TCustomMPHexEditor.Create(aOwner: TComponent);
  2570. var
  2571. LIntLoop: integer;
  2572. begin
  2573. inherited Create(aOwner);
  2574. {$IFDEF FASTACCESS}
  2575. FSetDataSizeFillByte := 0;
  2576. {$ENDIF}
  2577. FMaskedChars := [#0..#31];
  2578. FRulerNumberBase := 16;
  2579. FOffsetHandler := False;
  2580. FOnFind := nil;
  2581. FOnWildcardFind := nil;
  2582. FFindProgress := False;
  2583. FBlockSize := -1;
  2584. FSepCharBlocks := True;
  2585. FUnicodeCharacters := False;
  2586. FUnicodeBigEndian := False;
  2587. FSelectionChangedCount := 0;
  2588. FBytesPerUnit := 1;
  2589. FRulerBytesPerUnit := -1;
  2590. FUsedRulerBytesPerUnit := 1;
  2591. FShowPositionIfNotFocused := False;
  2592. FShowRuler := False;
  2593. FDrawGutter3D := True;
  2594. FHexLowerCase := True;
  2595. SetHexLowerCase(False);
  2596. DoubleBuffered := True;
  2597. FBookmarkBitmap := TBitmap.Create;
  2598. FCursorList := nil;
  2599. FHasCustomBMP := False;
  2600. FStreamFileName := '';
  2601. FHasFile := False;
  2602. FMaxUndo := 1024 * 1024;
  2603. FPosInCharField := False;
  2604. FLastPosInCharField := True;
  2605. FGutterWidth := -1;
  2606. GenerateOffsetFormat(MPHOffsetHex);
  2607. FSelectionPossible := True;
  2608. FBookmarkImageList := TImageList.Create(self);
  2609. FBookmarkImageList.DrawingStyle := dsTransparent;
  2610. FBookmarkImageList.BkColor := clBlack;
  2611. FBookmarkImageList.Width := 10;
  2612. FBookmarkImageList.Height := 10;
  2613. Options := [goThumbTracking];
  2614. DesignOptionsBoost := [];
  2615. DefaultDrawing := False;
  2616. FSaveCellExtents := False;
  2617. FColors := TMPHColors.Create(Self);
  2618. FDrawGridLines := False;
  2619. ParentColor := False;
  2620. FDataStorage := TMPHMemoryStream.Create;
  2621. FUndoStorage := TMPHUndoStorage.Create(self);
  2622. Color := FColors.Background;
  2623. FCharWidth := -1;
  2624. FOffSetDisplayWidth := -1;
  2625. FBytesPerRow := 16;
  2626. FCaretKind := ckAuto;
  2627. FFocusFrame := True;
  2628. FSwapNibbles := 0;
  2629. FFileName := '---';
  2630. Font.Name := 'Courier New';
  2631. Font.Size := 11;
  2632. BorderStyle := bsSingle;
  2633. FBytesPerCol := 4;
  2634. CTL3D := False;
  2635. Cursor := crIBeam;
  2636. FModifiedBytes := TBits.Create;
  2637. for LIntLoop := Low(FBookmarks) to High(FBookmarks) do
  2638. FBookmarks[LIntLoop].mPosition := -1;
  2639. SetSelection(-1, -1, -1);
  2640. FIsSelecting := False;
  2641. ResetUndo;
  2642. DefaultColWidth := 0;
  2643. DefaultRowHeight := 0;
  2644. RowHeights[0] := 0;
  2645. RowHeights[1] := 0;
  2646. ColCount := CalcColCount;
  2647. RowCount := GRID_FIXED + 1;
  2648. FTranslation := tkAsIs;
  2649. FModified := False;
  2650. FIsFileReadonly := True;
  2651. FBytesPerRowDup := 2 * FBytesPerRow;
  2652. FIntLastHexCol := (GRID_FIXED + FBytesPerRowDup - 1);
  2653. FReplaceUnprintableCharsBy := '.';
  2654. FCaretBitmap := TBitmap.Create;
  2655. FFixedFileSize := False;
  2656. FFixedFileSizeOverride := False;
  2657. FAllowInsertMode := True;
  2658. FInsertModeOn := False;
  2659. FWantTabs := True;
  2660. FReadOnlyView := False;
  2661. FHideSelection := False;
  2662. FGraySelOnLostFocus := False;
  2663. FOnProgress := nil;
  2664. FShowDrag := False;
  2665. FSelBeginPosition := -1;
  2666. FBookmarkBitmap.OnChange := BookmarkBitmapChanged;
  2667. FBookmarkBitmap.LoadFromResourceName(HINSTANCE, 'BOOKMARKICONS');
  2668. SetRulerString;
  2669. {$IFDEF DELPHI7UP}
  2670. ControlStyle := ControlStyle + [csNeedsBorderPaint];
  2671. {$ENDIF}
  2672. end;
  2673. destructor TCustomMPHexEditor.Destroy;
  2674. begin
  2675. FCursorList := nil;
  2676. FBookmarkBitmap.OnChange := nil;
  2677. FreeStorage;
  2678. FreeStorage(True);
  2679. FUndoStorage.Free;
  2680. FDataStorage.Free;
  2681. FModifiedBytes.Free;
  2682. FColors.Free;
  2683. FCaretBitmap.Free;
  2684. FBookmarkImageList.Free;
  2685. FBookmarkBitmap.Free;
  2686. inherited Destroy;
  2687. end;
  2688. procedure TCustomMPHexEditor.AdjustMetrics;
  2689. var
  2690. LIntLoop: integer;
  2691. LIntChWidth: integer;
  2692. begin
  2693. Canvas.Font.Assign(Font);
  2694. FCharWidth := Canvas.TextWidth('w');
  2695. SetOffsetDisplayWidth;
  2696. DoSetCellWidth(1, 6);
  2697. for LIntLoop := 0 to FBytesPerRowDup do
  2698. begin
  2699. if LIntLoop = Pred(FBytesPerRowDup) then
  2700. LIntChWidth := FCharWidth * 2
  2701. else
  2702. begin
  2703. LIntChWidth := FCharWidth;
  2704. if (((LIntLoop + GRID_FIXED) mod FBytesPerCol) = 1) then
  2705. Inc(LIntChWidth, FCharWidth);
  2706. if (FBlockSize > 1) and (((LIntLoop + GRID_FIXED) mod (FBlockSize * 2)) =
  2707. 1) then
  2708. Inc(LIntChWidth, FCharWidth);
  2709. end;
  2710. DoSetCellWidth(LIntLoop + GRID_FIXED, LIntChWidth);
  2711. end;
  2712. if FUnicodeCharacters then
  2713. LIntLoop := Pred(FBytesPerRow div 2)
  2714. else
  2715. LIntLoop := Pred(FBytesPerRow);
  2716. for LIntLoop := 0 to LIntLoop do
  2717. //FBytesPerRowDup + 1 to (FBytesPerRow * 3) - 1 do
  2718. begin
  2719. if (FUsedRulerBytesPerUnit > 1) and ((LIntLoop mod FUsedRulerBytesPerUnit)
  2720. = Pred(FUsedRulerBytesPerUnit)) and (not FUnicodeCharacters) then
  2721. LIntChWidth := (FCharWidth * 3 div 2) + 1
  2722. else
  2723. LIntChWidth := FCharWidth + 1;
  2724. if not FUnicodeCharacters then
  2725. begin
  2726. if (FBlockSize > 1) and FSepCharBlocks and ((LIntLoop mod FBlockSize) =
  2727. Pred(FBlockSize)) then
  2728. Inc(LIntChWidth, FCharWidth);
  2729. end
  2730. else
  2731. begin
  2732. if (FBlockSize > 1) and FSepCharBlocks and ((LIntLoop mod (FBlockSize div
  2733. 2)) = Pred(FBlockSize div 2)) then
  2734. Inc(LIntChWidth, FCharWidth);
  2735. end;
  2736. DoSetCellWidth(LIntLoop + GRID_FIXED + FBytesPerRowDup + 1, LIntChWidth);
  2737. end;
  2738. DoSetCellWidth(GetLastCharCol, (FCharWidth * 2) + 1);
  2739. FCharHeight := Canvas.TextHeight('yY') + 2;
  2740. DefaultRowHeight := FCharHeight;
  2741. RowHeights[1] := 0;
  2742. if FShowRuler then
  2743. RowHeights[0] := DefaultRowHeight + 3
  2744. else
  2745. RowHeights[0] := 0;
  2746. CheckSetCaret;
  2747. end;
  2748. function TCustomMPHexEditor.GetDataSize: integer;
  2749. begin
  2750. Result := FDataStorage.Size;
  2751. end;
  2752. procedure TCustomMPHexEditor.CreateEmptyFile;
  2753. begin
  2754. FreeStorage;
  2755. if TempName = '' then
  2756. FFileName := UNNAMED_FILE
  2757. else
  2758. FFileName := TempName;
  2759. ResetUndo;
  2760. ResetSelection(False);
  2761. FModifiedBytes.Size := 0;
  2762. CalcSizes;
  2763. FModified := False;
  2764. FIsFileReadonly := True;
  2765. FHasFile := False;
  2766. MoveColRow(GRID_FIXED, GRID_FIXED, True, True);
  2767. Changed;
  2768. end;
  2769. procedure TCustomMPHexEditor.SaveToStream(Strm: TStream);
  2770. begin
  2771. WaitCursor;
  2772. try
  2773. FDataStorage.Position := 0;
  2774. Stream2Stream(FDataStorage, Strm, pkSave);
  2775. finally
  2776. Invalidate;
  2777. OldCursor;
  2778. end;
  2779. end;
  2780. procedure TCustomMPHexEditor.SaveRangeToStream(Strm: TStream; const APosition,
  2781. ACount: integer);
  2782. begin
  2783. WaitCursor;
  2784. try
  2785. FDataStorage.Position := APosition;
  2786. Stream2Stream(FDataStorage, Strm, pkSave, ACount);
  2787. finally
  2788. Invalidate;
  2789. OldCursor;
  2790. end;
  2791. end;
  2792. procedure TCustomMPHexEditor.SaveToFile(const Filename: string;
  2793. const aUnModify: boolean = True);
  2794. var
  2795. LfstFile: TFileStream;
  2796. begin
  2797. if (FFileName = FileName) then
  2798. PrepareOverwriteDiskFile;
  2799. LfstFile := TFileStream.Create(FileName, fmCreate);
  2800. try
  2801. FStreamFileName := FileName;
  2802. SaveToStream(LfstFile);
  2803. FHasFile := True;
  2804. if aUnModify then
  2805. begin
  2806. FModifiedBytes.Size := 0;
  2807. FModified := False;
  2808. FIsFileReadonly := False;
  2809. FFileName := Filename;
  2810. FDataStorage.Position := 0;
  2811. ResetUndo;
  2812. end;
  2813. finally
  2814. FStreamFileName := '';
  2815. LfstFile.Free;
  2816. end;
  2817. end;
  2818. procedure TCustomMPHexEditor.LoadFromStream(Strm: TStream);
  2819. begin
  2820. try
  2821. FreeStorage;
  2822. CalcSizes;
  2823. WaitCursor;
  2824. try
  2825. try
  2826. Strm.Position := 0;
  2827. FDataStorage.Size := Strm.Size;
  2828. FDataStorage.Position := 0;
  2829. Stream2Stream(Strm, FDataStorage, pkLoad);
  2830. //FDataStorage.CopyFrom(Strm, Strm.Size - Strm.Position);
  2831. FDataStorage.Position := 0;
  2832. finally
  2833. with FUndoStorage do
  2834. if UpdateCount < 1 then
  2835. Reset;
  2836. FModifiedBytes.Size := 0;
  2837. CalcSizes;
  2838. FModified := False;
  2839. FIsSelecting := False;
  2840. MoveColRow(GRID_FIXED, GRID_FIXED, True, True);
  2841. Changed;
  2842. end;
  2843. finally
  2844. OldCursor;
  2845. end;
  2846. except
  2847. FreeStorage;
  2848. FreeStorage(True);
  2849. FHasFile := False;
  2850. raise;
  2851. end;
  2852. end;
  2853. procedure TCustomMPHexEditor.LoadFromFile(const Filename: string);
  2854. var
  2855. LfstFile: TFileStream;
  2856. begin
  2857. if CanOpenFile(FileName, FIsFileReadonly) then
  2858. begin
  2859. LfstFile := TFileStream.Create(FileName, fmOpenRead or fmShareDenyNone);
  2860. try
  2861. FStreamFileName := FileName;
  2862. try
  2863. LoadFromStream(LfstFile);
  2864. except
  2865. FHasFile := False;
  2866. raise;
  2867. end;
  2868. FFileName := FileName;
  2869. FHasFile := True;
  2870. finally
  2871. FStreamFileName := '';
  2872. LfstFile.Free;
  2873. end;
  2874. end
  2875. else
  2876. raise EFOpenError.CreateFmt(ERR_FILE_OPEN_FAILED, [FileName,
  2877. SysErrorMessage(GetLastError)]);
  2878. end;
  2879. procedure TCustomMPHexEditor.CalcSizes;
  2880. var
  2881. LIntRows: integer;
  2882. begin
  2883. if FModifiedBytes.Size > DataSize then
  2884. FModifiedBytes.Size := DataSize;
  2885. if DataSize < 1 then
  2886. begin
  2887. RowCount := GRID_FIXED + 1;
  2888. ColCount := CalcColCount;
  2889. FixedCols := GRID_FIXED;
  2890. end
  2891. else
  2892. begin
  2893. LIntRows := (DataSize + (FBytesPerRow - 1)) div FBytesPerRow;
  2894. if ((DataSize mod FBytesPerRow) = 0) and InsertMode then
  2895. INC(LIntRows);
  2896. RowCount := LIntRows + GRID_FIXED;
  2897. ColCount := CalcColCount;
  2898. FixedCols := GRID_FIXED;
  2899. end;
  2900. FixedRows := GRID_FIXED;
  2901. AdjustMetrics;
  2902. end;
  2903. function TCustomMPHexEditor.TranslateFromAnsiChar(const aByte: byte): char;
  2904. begin
  2905. case FTranslation of
  2906. tkAsIs: Result := char(aByte);
  2907. tkDos8,
  2908. tkASCII:
  2909. begin
  2910. if ((FTranslation = tkDos8) or (aByte < 128)) and (aByte > 31) then
  2911. Result := Char2Oem(aByte)
  2912. else
  2913. Result := #0;
  2914. end;
  2915. tkMac: Result := MPH_CCONV_MAC[cctFromAnsi][aByte];
  2916. tkBCD: Result := MPH_CCONV_BCD38[cctFromAnsi][aByte];
  2917. tkCustom: Result := MPHCustomCharConv[cctFromAnsi][aByte];
  2918. else
  2919. Result := #0;
  2920. end;
  2921. if Result in FMaskedChars then
  2922. Result := #0;
  2923. end;
  2924. function TCustomMPHexEditor.TranslateToAnsiChar(const aByte: byte): char;
  2925. begin
  2926. case FTranslation of
  2927. tkAsIs: Result := char(aByte);
  2928. tkDos8,
  2929. tkASCII:
  2930. begin
  2931. Result := Oem2Char(aByte);
  2932. if ((FTranslation = tkASCII) and (aByte > 127)) then
  2933. Result := FReplaceUnprintableCharsBy;
  2934. end;
  2935. tkMac: Result := MPH_CCONV_MAC[cctToAnsi][aByte];
  2936. tkBCD: Result := MPH_CCONV_BCD38[cctToAnsi][aByte];
  2937. tkCustom: Result := MPHCustomCharConv[cctToAnsi][aByte];
  2938. else
  2939. Result := FReplaceUnprintableCharsBy;
  2940. end;
  2941. if (FReplaceUnprintableCharsBy <> #0) and (Result in FMaskedChars) then
  2942. Result := FReplaceUnprintableCharsBy;
  2943. end;
  2944. // get the position of the drag marker
  2945. function TCustomMPHexEditor.DropPosition: integer;
  2946. var
  2947. LBoolInCharField: boolean;
  2948. begin
  2949. Result := -1;
  2950. LBoolInCharField := FPosInCharField;
  2951. try
  2952. if FShowDrag then
  2953. begin
  2954. Result := GetPosAtCursor(FDropCol, FDropRow);
  2955. CheckUnit(Result);
  2956. end;
  2957. finally
  2958. FPosInCharField := LBoolInCharField;
  2959. end;
  2960. end;
  2961. procedure TCustomMPHexEditor.Stream2Stream(strFrom, strTo: TStream;
  2962. const Operation: TMPHProgressKind; const Count: integer = -1);
  2963. var
  2964. LBytProgress, LBytLastProgress: byte;
  2965. LIntRemain, LIntRead, LIntCount: integer;
  2966. LBoolCancel: boolean;
  2967. LStrFile: string;
  2968. LBytBuffer: array[0..MPH_FILEIO_BLOCKSIZE - 1] of byte;
  2969. begin
  2970. LIntCount := Count;
  2971. if LIntCount = -1 then
  2972. LIntCount := strFrom.Size - strFrom.Position;
  2973. LIntRemain := LIntCount;
  2974. LBoolCancel := False;
  2975. LBytLastProgress := 255;
  2976. LStrFile := FStreamFileName;
  2977. if LStrFile = '' then
  2978. LStrFile := FFileName;
  2979. while LIntRemain > 0 do
  2980. begin
  2981. LBytProgress := Round(((LIntCount - LIntRemain) / LIntCount) * 100);
  2982. if (LBytProgress <> LBytLastProgress) or (LIntRemain <=
  2983. MPH_FILEIO_BLOCKSIZE) then
  2984. begin
  2985. if LIntRemain <= MPH_FILEIO_BLOCKSIZE then
  2986. LBytLastProgress := 100
  2987. else
  2988. LBytLastProgress := LBytProgress;
  2989. if Assigned(FOnProgress) then
  2990. begin
  2991. FOnProgress(self, Operation, LStrFile, LBytLastProgress,
  2992. LBoolCancel);
  2993. if LBoolCancel then
  2994. raise EMPHexEditor.Create(ERR_CANCELLED);
  2995. end
  2996. end;
  2997. LIntRead := Min(LIntRemain, MPH_FILEIO_BLOCKSIZE);
  2998. strFrom.ReadBuffer(LBytBuffer, LIntRead);
  2999. strTo.WriteBuffer(LBytBuffer, LIntRead);
  3000. Dec(LIntRemain, LIntRead);
  3001. end;
  3002. end;
  3003. function TCustomMPHexEditor.SelectCell(ACol, ARow: longint): boolean;
  3004. var
  3005. LIntCurRow: integer;
  3006. LRctCellRect: TRect;
  3007. LIntOtherFieldCol: integer;
  3008. LIntNewPosition, LIntPrevPosition: integer;
  3009. begin
  3010. LIntCurRow := Row;
  3011. if DataSize > 0 then
  3012. Result := CheckSelectCell(aCol, aRow)
  3013. else
  3014. begin
  3015. if not ((aCol = GRID_FIXED) or (aCol = Max(GetOtherFieldColCheck(GRID_FIXED)
  3016. , GRID_FIXED)) and (aRow = GRID_FIXED)) then
  3017. Result := False
  3018. else
  3019. begin
  3020. LRctCellRect := CellRect(aCol, aRow);
  3021. if LRctCellRect.Left + LRctCellRect.Bottom = 0 then
  3022. IntSetCaretPos(-50, -50, -1)
  3023. else
  3024. IntSetCaretPos(LRctCellRect.Left, LRctCellRect.Top, aCol);
  3025. Result := True;
  3026. Exit;
  3027. end;
  3028. end;
  3029. if Result then
  3030. begin
  3031. //cursor an alter stelle löschen
  3032. if (aCol <> Col) or (aRow <> Row) then
  3033. begin
  3034. LIntOtherFieldCol := GetOtherFieldColCheck(Col);
  3035. if not FPosInCharField then
  3036. LRctCellRect := CellRect(LIntOtherFieldCol, LIntCurRow)
  3037. else
  3038. begin
  3039. if FUnicodeCharacters then
  3040. LRctCellRect := BoxRect(LIntOtherFieldCol, LIntCurRow,
  3041. LIntOtherFieldCol+3, LIntCurRow)
  3042. else
  3043. LRctCellRect := BoxRect(LIntOtherFieldCol, LIntCurRow,
  3044. LIntOtherFieldCol+1, LIntCurRow)
  3045. end;
  3046. InvalidateRect(Handle, @LRctCellRect, False);
  3047. if FShowRuler and (aCol <> Col) then
  3048. begin
  3049. LRctCellRect := CellRect(LIntOtherFieldCol, 0);
  3050. InvalidateRect(Handle, @LRctCellRect, False);
  3051. LRctCellRect := CellRect(Col, 0);
  3052. InvalidateRect(Handle, @LRctCellRect, False);
  3053. end;
  3054. // cursor an neuer stelle setzen
  3055. LIntOtherFieldCol := GetOtherFieldColCheck(aCol);
  3056. if not FPosInCharField then
  3057. LRctCellRect := CellRect(LIntOtherFieldCol, aRow)
  3058. else
  3059. begin
  3060. if FUnicodeCharacters then
  3061. LRctCellRect := BoxRect(LIntOtherFieldCol, aRow,
  3062. LIntOtherFieldCol+3, aRow)
  3063. else
  3064. LRctCellRect := BoxRect(LIntOtherFieldCol, aRow,
  3065. LIntOtherFieldCol+1, aRow)
  3066. end;
  3067. InvalidateRect(Handle, @LRctCellRect, False);
  3068. if FShowRuler and (aCol <> Col) then
  3069. begin
  3070. LRctCellRect := CellRect(LIntOtherFieldCol, 0);
  3071. InvalidateRect(Handle, @LRctCellRect, False);
  3072. LRctCellRect := CellRect(aCol, 0);
  3073. InvalidateRect(Handle, @LRctCellRect, False);
  3074. end;
  3075. if LIntCurRow <> aRow then
  3076. begin
  3077. LRctCellRect := CellRect(0, LIntCurRow);
  3078. InvalidateRect(Handle, @LRctCellRect, False);
  3079. LRctCellRect := CellRect(0, aRow);
  3080. InvalidateRect(Handle, @LRctCellRect, False);
  3081. end;
  3082. end;
  3083. if FIsSelecting then
  3084. begin
  3085. LIntNewPosition := GetPosAtCursor(aCol, aRow);
  3086. LIntPrevPosition := GetPosAtCursor(Col, Row);
  3087. if FSelBeginPosition = -1 then
  3088. FSelBeginPosition := LIntPrevPosition;
  3089. if not InsertMode then
  3090. begin
  3091. CheckSelectUnit(FSelBeginPosition, LIntNewPosition);
  3092. NewSelection(FSelBeginPosition, LIntNewPosition);
  3093. end
  3094. else
  3095. begin
  3096. if FSelBeginPosition > LIntNewPosition then
  3097. begin
  3098. CheckUnit(FSelBeginPosition);
  3099. CheckUnit(LIntNewPosition);
  3100. if FSelBeginPosition = LIntNewPosition then
  3101. begin
  3102. ResetSelection(True);
  3103. FSelBeginPosition := LIntNewPosition;
  3104. FIsSelecting := True;
  3105. end
  3106. else
  3107. begin
  3108. NewSelection(FSelBeginPosition - FBytesPerUnit, LIntNewPosition);
  3109. end;
  3110. end
  3111. else if FSelBeginPosition < LIntNewPosition then
  3112. begin
  3113. CheckUnit(FSelBeginPosition);
  3114. CheckUnit(LIntNewPosition);
  3115. if FSelBeginPosition = LIntNewPosition then
  3116. begin
  3117. ResetSelection(True);
  3118. FSelBeginPosition := LIntNewPosition;
  3119. FIsSelecting := True;
  3120. end
  3121. else
  3122. begin
  3123. NewSelection(FSelBeginPosition, LIntNewPosition - FBytesPerUnit);
  3124. end;
  3125. end
  3126. else
  3127. begin
  3128. ResetSelection(True);
  3129. FSelBeginPosition := LIntNewPosition;
  3130. FIsSelecting := True;
  3131. end
  3132. end;
  3133. end
  3134. else
  3135. ResetSelection(True);
  3136. // caret neu setzen
  3137. //CheckSetCaret;
  3138. LRctCellRect := CellRect(aCol, aRow);
  3139. if LRctCellRect.Left + LRctCellRect.Bottom = 0 then
  3140. IntSetCaretPos(-50, -50, -1)
  3141. else
  3142. IntSetCaretPos(LRctCellRect.Left, LRctCellRect.Top, aCol);
  3143. SelectionChanged;
  3144. end;
  3145. end;
  3146. // Obtient la position dans le fichier à partir de la position du curseur
  3147. function TCustomMPHexEditor.GetPosAtCursor(const aCol, aRow: integer): integer;
  3148. begin
  3149. FPosInCharField := (aCol > (GRID_FIXED + FBytesPerRowDup));
  3150. if FPosInCharField then
  3151. begin
  3152. Result := aCol - ((GRID_FIXED + 1) + FBytesPerRowDup);
  3153. if FUnicodeCharacters then
  3154. Result := Result * 2;
  3155. end
  3156. else
  3157. Result := (aCol - GRID_FIXED) div 2;
  3158. Result := Result + ((aRow - GRID_FIXED) * FBytesPerRow);
  3159. if Result < 0 then
  3160. Result := 0;
  3161. end;
  3162. function TCustomMPHexEditor.GetRow(const DataPos: integer): integer;
  3163. begin
  3164. Result := (DataPos div FBytesPerRow) + GRID_FIXED;
  3165. end;
  3166. function TCustomMPHexEditor.GetCursorAtPos(const aPos: integer;
  3167. const aChars: boolean): TGridCoord;
  3168. var
  3169. LIntCol: integer;
  3170. begin
  3171. if aPos < 0 then
  3172. begin
  3173. Result.y := GRID_FIXED;
  3174. Result.x := GRID_FIXED;
  3175. Exit;
  3176. end;
  3177. Result.y := GetRow(aPos);
  3178. LIntCol := aPos mod FBytesPerRow;
  3179. if aChars then
  3180. begin
  3181. if FUnicodeCharacters then
  3182. Result.x := (LIntCol div 2) + (GRID_FIXED + 1) + FBytesPerRowDup
  3183. else
  3184. Result.x := LIntCol + (GRID_FIXED + 1 + FBytesPerRowDup)
  3185. end
  3186. else
  3187. Result.x := (LIntCol * 2) + GRID_FIXED;
  3188. end;
  3189. function TCustomMPHexEditor.GetOtherFieldCol(const aCol: integer): integer;
  3190. var
  3191. LIntCol: integer;
  3192. begin
  3193. FPosInCharField := (aCol > (GRID_FIXED + FBytesPerRowDup));
  3194. if FPosInCharField then
  3195. begin
  3196. LIntCol := (aCol - (GRID_FIXED + 1 + FBytesPerRowDup));
  3197. if FUnicodeCharacters then
  3198. Result := (LIntCol * 4) + GRID_FIXED
  3199. else
  3200. Result := (LIntCol * 2) + GRID_FIXED;
  3201. end
  3202. else
  3203. begin
  3204. if FUnicodeCharacters then
  3205. LIntCol := ((aCol - GRID_FIXED) div 4)
  3206. else
  3207. LIntCol := ((aCol - GRID_FIXED) div 2);
  3208. Result := LIntCol + (GRID_FIXED + 1 + FBytesPerRowDup);
  3209. end;
  3210. end;
  3211. function TCustomMPHexEditor.GetOtherFieldColCheck(const aCol: integer): integer;
  3212. var
  3213. LIntCol: integer;
  3214. begin
  3215. if aCol > (GRID_FIXED + FBytesPerRowDup) then
  3216. begin
  3217. LIntCol := (aCol - (GRID_FIXED + 1 + FBytesPerRowDup));
  3218. if FUnicodeCharacters then
  3219. Result := (LIntCol * 4) + GRID_FIXED
  3220. else
  3221. Result := (LIntCol * 2) + GRID_FIXED;
  3222. end
  3223. else
  3224. begin
  3225. if FUnicodeCharacters then
  3226. LIntCol := ((aCol - GRID_FIXED) div 4)
  3227. else
  3228. LIntCol := ((aCol - GRID_FIXED) div 2);
  3229. Result := LIntCol + (GRID_FIXED + 1 + FBytesPerRowDup);
  3230. end;
  3231. end;
  3232. function TCustomMPHexEditor.CheckSelectCell(aCol, aRow: integer): boolean;
  3233. var
  3234. LgrcEndCoords: TGridCoord;
  3235. LIntPos: integer;
  3236. begin
  3237. Result := inherited SelectCell(aCol, aRow);
  3238. if not FSelectionPossible then
  3239. Exit;
  3240. try
  3241. FSelectionPossible := False;
  3242. if Result then
  3243. begin
  3244. // überprüfen, ob linke maustaste oder shift gedrückt, sonst selection zurücksetzen
  3245. if not (IsKeyDown(VK_SHIFT) or IsKeyDown(VK_LBUTTON)) then
  3246. ResetSelection(True);
  3247. // überprüfen, ob außerhalb der DateiGröße
  3248. LIntPos := GetPosAtCursor(aCol, aRow);
  3249. if (LIntPos >= DataSize) and not (InsertMode and (LIntPos = DataSize) and
  3250. (FPosInCharField or ((aCol mod 2) = 0))) then
  3251. begin
  3252. if (not InsertMode) then
  3253. LgrcEndCoords := GetCursorAtPos(DataSize - 1, InCharField)
  3254. else
  3255. LgrcEndCoords := GetCursorAtPos(DataSize, InCharField);
  3256. MoveColRow(LgrcEndCoords.x, LgrcEndCoords.y, True, True);
  3257. Result := False;
  3258. end
  3259. else if aCol = (GRID_FIXED + FBytesPerRowDup) then
  3260. begin
  3261. Result := False;
  3262. if IsKeyDown(VK_LBUTTON) then
  3263. begin
  3264. aCol := aCol - 1;
  3265. aCol := Max(GRID_FIXED, aCol);
  3266. MoveColRow(aCol, aRow, True, True);
  3267. Exit;
  3268. end;
  3269. end;
  3270. end;
  3271. finally
  3272. FSelectionPossible := True;
  3273. end;
  3274. end;
  3275. procedure TCustomMPHexEditor.WMChar(var Msg: TWMChar);
  3276. var
  3277. LIntPos: integer;
  3278. LChrChar: char;
  3279. LBytOldData, LBytNewData: byte;
  3280. LArrNewData: packed array[0..7] of byte;
  3281. LWChrNewData: WideChar absolute LArrNewData;
  3282. LgrcPosition: TGridCoord;
  3283. LWChrOldData: WideChar;
  3284. LWrdKey: Word;
  3285. begin
  3286. LChrChar := char(Msg.CharCode);
  3287. if Assigned(OnKeyPress) then
  3288. OnKeyPress(Self, LChrChar);
  3289. if FReadOnlyView or (LChrChar in FMaskedChars) then
  3290. Exit;
  3291. LIntPos := GetPosAtCursor(Col, Row);
  3292. if (LIntPos >= DataSize) and not InsertMode then
  3293. Exit;
  3294. if not FPosInCharField then
  3295. begin
  3296. // hex-eingabe, nur 0..9 , a..f erlaubt
  3297. if Pos(LChrChar, HEX_ALLCHARS) <> 0 then
  3298. begin
  3299. LChrChar := UpCase(LChrChar);
  3300. if not InsertMode then
  3301. ResetSelection(True);
  3302. LgrcPosition := GetCursorAtPos(LIntPos, FPosInCharField);
  3303. // Obtient la valeur du byte dans le fichier (OldByte)
  3304. if DataSize > LIntPos then
  3305. LBytOldData := Data[LIntPos]
  3306. else
  3307. LBytOldData := 0;
  3308. if (LgrcPosition.x = (Col - FSwapNibbles)) or (SelCount <> 0) then
  3309. LBytNewData := LBytOldData and 15 + ((Pos(LChrChar, HEX_UPPER) - 1) * 16)
  3310. else
  3311. LBytNewData := (LBytOldData and $F0) + (Pos(LChrChar, HEX_UPPER) - 1);
  3312. FillChar(LArrNewData, sizeof(LArrNewData), #0);
  3313. if InsertMode and ((((Col - GRID_FIXED) mod (FBytesPerUnit * 2)) = 0) or
  3314. (SelCount > 0)) then
  3315. begin
  3316. if FSwapNibbles = 0 then
  3317. LBytNewData := LBytNewData and $F0
  3318. else
  3319. LBytNewData := LBytNewData and $0F;
  3320. LArrNewData[0] := LBytNewData;
  3321. if DataSize = 0 then
  3322. AppendBuffer(PChar(@LArrNewData), FBytesPerUnit, '', False)
  3323. else if SelCount = 0 then
  3324. begin
  3325. InsertBuffer(PChar(@LArrNewData), FBytesPerUnit, LIntPos, '', False);
  3326. end
  3327. else
  3328. ReplaceSelection(PChar(@LArrNewData), FBytesPerUnit, '', False);
  3329. end
  3330. else
  3331. begin
  3332. if LIntPos >= DataSize then
  3333. Exit;
  3334. IntChangeByte(LBytOldData, LBytNewData, LIntPos, Col, Row);
  3335. end;
  3336. FIsSelecting := False;
  3337. LWrdKey := VK_RIGHT;
  3338. KeyDown(LWrdKey, []);
  3339. end
  3340. else
  3341. WrongKey
  3342. end
  3343. else
  3344. begin
  3345. // zeichen-eingabe, alle zeichen erlaubt
  3346. LChrChar := TranslateFromAnsiChar(Ord(LChrChar));
  3347. if (LChrChar in FMaskedChars) then
  3348. begin
  3349. WrongKey;
  3350. Exit;
  3351. end;
  3352. if not InsertMode then
  3353. ResetSelection(True);
  3354. LgrcPosition := GetCursorAtPos(LIntPos, FPosInCharField);
  3355. FillChar(LArrNewData, sizeof(LArrNewData), #0);
  3356. if not FUnicodeCharacters then
  3357. LArrNewData[0] := Ord(LChrChar)
  3358. else
  3359. begin
  3360. LWChrNewData := StringToWideChar(LChrChar, @LWChrNewData, 2)^;
  3361. if FUnicodeBigEndian then
  3362. SwapWideChar(LWChrNewData);
  3363. end;
  3364. if (DataSize = 0) or (DataSize = LIntPos) then
  3365. LBytOldData := 0
  3366. else
  3367. LBytOldData := Data[LIntPos];
  3368. if FUnicodeCharacters then
  3369. begin
  3370. if (DataSize = 0) or (DataSize = LIntPos) or (DataSize = (LIntPos + 1))
  3371. then
  3372. LWChrOldData := #0
  3373. else
  3374. ReadBuffer(LWChrOldData, LIntPos, 2);
  3375. end;
  3376. if InsertMode then
  3377. begin
  3378. if SelCount > 0 then
  3379. ReplaceSelection(PChar(@LArrNewData), FBytesPerUnit, '', False)
  3380. else
  3381. begin
  3382. if LIntPos = DataSize then
  3383. AppendBuffer(PChar(@LArrNewData), FBytesPerUnit)
  3384. else
  3385. begin
  3386. if (LIntPos mod FBytesPerUnit) = 0 then
  3387. InsertBuffer(PChar(@LArrNewData), FBytesPerUnit, LIntPos, '', False)
  3388. else
  3389. IntChangeByte(LBytOldData, LArrNewData[0], LIntPos, Col, Row)
  3390. end;
  3391. FIsSelecting := False;
  3392. end;
  3393. end
  3394. else
  3395. begin
  3396. if FUnicodeCharacters then
  3397. IntChangeWideChar(LWChrOldData, LWChrNewData, LIntPos, Col, Row)
  3398. else
  3399. IntChangeByte(LBytOldData, Ord(LChrChar), LIntPos, Col, Row);
  3400. end;
  3401. LWrdKey := VK_RIGHT;
  3402. KeyDown(LWrdKey, []);
  3403. end;
  3404. end;
  3405. {-------------------------------------------------------------------------------}
  3406. // *** procedure TCustomMPHexEditor.IntChangeByte***
  3407. // Change la valeur du byte
  3408. // Renseigne la structure Undo
  3409. {-------------------------------------------------------------------------------}
  3410. procedure TCustomMPHexEditor.IntChangeByte(const aOldByte, aNewByte: byte; aPos,
  3411. aCol, aRow: integer; const UndoDesc: string = '');
  3412. var
  3413. LRctBoxRect: TRect;
  3414. LIntOtherFieldCol: integer;
  3415. begin
  3416. if aOldByte = aNewByte then
  3417. Exit;
  3418. CreateUndo(ufKindBytesChanged, aPos, 1, 0, UndoDesc);
  3419. // Ecrit dans le fichier
  3420. Data[aPos] := aNewByte;
  3421. if not InsertMode then
  3422. FModifiedBytes.Bits[aPos] := True;
  3423. aCol := GetCursorAtPos(aPos, False).X;
  3424. LIntOtherFieldCol := GetOtherFieldColCheck(aCol);
  3425. LRctBoxRect := BoxRect(aCol, aRow, aCol + 1, aRow);
  3426. InvalidateRect(Handle, @LRctBoxRect, False);
  3427. LRctBoxRect := BoxRect(LIntOtherFieldCol, aRow, LIntOtherFieldCol, aRow);
  3428. InvalidateRect(Handle, @LRctBoxRect, False);
  3429. Changed;
  3430. end;
  3431. procedure TCustomMPHexEditor.IntChangeWideChar(const aOldChar, aNewChar:
  3432. WideChar; aPos, aCol, aRow: integer; const UndoDesc: string);
  3433. var
  3434. LRctBoxRect: TRect;
  3435. LIntOtherFieldCol: integer;
  3436. LBArrOld: packed array[0..1] of Byte absolute aOldChar;
  3437. LBArrNew: packed array[0..1] of Byte absolute aNewChar;
  3438. begin
  3439. if aOldChar = aNewChar then
  3440. Exit;
  3441. CreateUndo(ufKindBytesChanged, aPos, 2, 0, UndoDesc);
  3442. // Ecrit dans le fichier
  3443. WriteBuffer(aNewChar, aPos, 2);
  3444. if not InsertMode then
  3445. begin
  3446. FModifiedBytes.Bits[aPos] := LBArrOld[0] <> LBArrNew[0];
  3447. FModifiedBytes.Bits[aPos + 1] := LBArrOld[1] <> LBArrNew[1];
  3448. end;
  3449. aCol := GetCursorAtPos(aPos, False).X;
  3450. LIntOtherFieldCol := GetOtherFieldColCheck(aCol);
  3451. LRctBoxRect := BoxRect(aCol, aRow, aCol + 3, aRow);
  3452. InvalidateRect(Handle, @LRctBoxRect, False);
  3453. LRctBoxRect := BoxRect(LIntOtherFieldCol, aRow, LIntOtherFieldCol, aRow);
  3454. InvalidateRect(Handle, @LRctBoxRect, False);
  3455. Changed;
  3456. end;
  3457. procedure TCustomMPHexEditor.KeyDown(var Key: word; Shift: TShiftState);
  3458. var
  3459. LIntCol: integer;
  3460. LgrcPosition: TGridCoord;
  3461. LIntRow: integer;
  3462. begin
  3463. if Assigned(OnKeyDown) then
  3464. OnKeyDown(self, Key, Shift);
  3465. // reset selection if no shift key is pressed (except of SHIFT-Key)
  3466. if not ((Shift <> []) or (KEY = VK_SHIFT)) then
  3467. if not InsertMode then
  3468. ResetSelection(True);
  3469. case Key of
  3470. VK_PRIOR:
  3471. begin
  3472. if ssCtrl in Shift then
  3473. begin
  3474. // go to the first visible line
  3475. LIntRow := TopRow;
  3476. LIntCol := Col;
  3477. if LIntRow > -1 then
  3478. begin
  3479. MoveColRow(LIntCol, LIntRow, True, True);
  3480. end;
  3481. end
  3482. else
  3483. begin
  3484. // scroll up one page
  3485. LIntRow := Max(GRID_FIXED, Row - VisibleRowCount + 1);
  3486. TopRow := Max(GRID_FIXED, TopRow - VisibleRowCount + 1);
  3487. LIntCol := Col;
  3488. if LIntRow > -1 then
  3489. begin
  3490. MoveColRow(LIntCol, LIntRow, True, True);
  3491. end;
  3492. end;
  3493. end;
  3494. VK_NEXT:
  3495. begin
  3496. if ssCtrl in Shift then
  3497. begin
  3498. // go to the Last visible line
  3499. LIntRow := Min(RowCount - 1, TopRow + VisibleRowCount - 1);
  3500. LIntCol := Col;
  3501. if LIntRow > 0 then
  3502. begin
  3503. MoveColRow(LIntCol, LIntRow, True, True);
  3504. end;
  3505. end
  3506. else
  3507. begin
  3508. // scroll down one page
  3509. LIntRow := Min(RowCount - 1, Row + VisibleRowCount - 1);
  3510. TopRow := Min(Max(GRID_FIXED, RowCount - VisibleRowCount),
  3511. TopRow + VisibleRowCount - 1);
  3512. LIntCol := Col;
  3513. if LIntRow > 0 then
  3514. begin
  3515. MoveColRow(LIntCol, LIntRow, True, True);
  3516. end;
  3517. end;
  3518. end;
  3519. VK_HOME:
  3520. begin
  3521. InCharField;
  3522. if (ssCtrl in Shift) then
  3523. begin // strg+pos1
  3524. if not FPosInCharField then
  3525. MoveColRow(GRID_FIXED, GRID_FIXED, True, True)
  3526. else
  3527. MoveColRow(Max(GRID_FIXED, GetOtherFieldCol(GRID_FIXED)),
  3528. GRID_FIXED, True, True);
  3529. end
  3530. else
  3531. begin // normaler zeilenstart
  3532. if not FPosInCharField then
  3533. MoveColRow(GRID_FIXED, Row, True, True)
  3534. else
  3535. MoveColRow(Max(GRID_FIXED, GetOtherFieldCol(GRID_FIXED)),
  3536. Row, True, True);
  3537. end;
  3538. end;
  3539. VK_END:
  3540. begin
  3541. InCharField;
  3542. if (ssCtrl in Shift) then
  3543. begin // strg+end
  3544. if (not InsertMode) then
  3545. LgrcPosition := GetCursorAtPos(DataSize - 1, FPosInCharField)
  3546. else
  3547. LgrcPosition := GetCursorAtPos(DataSize, FPosInCharField);
  3548. MoveColRow(LgrcPosition.x, LgrcPosition.y, True, True)
  3549. end
  3550. else
  3551. begin // normales zeilenende
  3552. if not FPosInCharField then
  3553. begin
  3554. LIntCol := GetPosAtCursor(GRID_FIXED, Row + 1) - 1;
  3555. TruncMaxPosition(LIntCol);
  3556. LgrcPosition := GetCursorAtPos(LIntCol, FPosInCharField);
  3557. MoveColRow(LgrcPosition.x + 1, LgrcPosition.y, True, True)
  3558. end
  3559. else
  3560. begin
  3561. LIntCol := GetPosAtCursor(GRID_FIXED, Row + 1) - 1;
  3562. TruncMaxPosition(LIntCol);
  3563. LgrcPosition := GetCursorAtPos(LIntCol, True);
  3564. MoveColRow(LgrcPosition.x, LgrcPosition.y, True, True);
  3565. end
  3566. end;
  3567. end;
  3568. VK_LEFT, VK_BACK:
  3569. if (InsertMode and (not FReadOnlyView)) and (Key = VK_BACK) then
  3570. begin
  3571. if SelCount > 0 then
  3572. DeleteSelection
  3573. else
  3574. InternalErase(True);
  3575. end
  3576. else if (not (ssCTRL in Shift)) then
  3577. begin
  3578. if FIsSelecting or (FUnicodeCharacters and FPosInCharField) then
  3579. LIntCol := GetPosAtCursor(Col, Row) - FBytesPerUnit
  3580. else
  3581. LIntCol := GetPosAtCursor(Col, Row) - 1;
  3582. if FPosInCharField then
  3583. begin
  3584. if LIntCol < 0 then
  3585. LIntCol := 0;
  3586. LgrcPosition := GetCursorAtPos(LIntCol, FPosInCharField);
  3587. MoveColRow(LgrcPosition.x, LgrcPosition.y, True, True);
  3588. end
  3589. else
  3590. begin
  3591. if FIsSelecting then
  3592. begin
  3593. CheckUnit(LIntCol);
  3594. LgrcPosition := GetCursorAtPos(LIntCol, FPosInCharField);
  3595. MoveColRow(LgrcPosition.x, LgrcPosition.y, True, True);
  3596. end
  3597. else
  3598. begin
  3599. LIntCol := LIntCol + 1;
  3600. LgrcPosition := GetCursorAtPos(LIntCol, False);
  3601. if LgrcPosition.x < Col then
  3602. MoveColRow(Col - 1, Row, True, True)
  3603. else
  3604. begin
  3605. LIntCol := LIntCol - 1;
  3606. if LIntCol >= 0 then
  3607. begin
  3608. LgrcPosition := GetCursorAtPos(LIntCol, FPosInCharField);
  3609. MoveColRow(LgrcPosition.x + 1, LgrcPosition.y, True, True);
  3610. end;
  3611. end
  3612. end;
  3613. end;
  3614. end
  3615. else
  3616. begin
  3617. if Key = VK_LEFT then
  3618. begin
  3619. LIntCol := GRID_FIXED;
  3620. MoveColRow(LIntCol, Row, True, True);
  3621. end;
  3622. end;
  3623. VK_RIGHT:
  3624. begin
  3625. if (not (ssCTRL in Shift)) then
  3626. begin
  3627. if FIsSelecting or (FUnicodeCharacters and FPosInCharField) then
  3628. LIntCol := GetPosAtCursor(Col, Row) + FBytesPerUnit
  3629. else
  3630. LIntCol := GetPosAtCursor(Col, Row) + 1;
  3631. if FPosInCharField then
  3632. begin
  3633. TruncMaxPosition(LIntCol);
  3634. LgrcPosition := GetCursorAtPos(LIntCol, FPosInCharField);
  3635. MoveColRow(LgrcPosition.x, LgrcPosition.y, True, True);
  3636. end
  3637. else
  3638. begin
  3639. if FIsSelecting then
  3640. begin
  3641. CheckUnit(LIntCol);
  3642. TruncMaxPosition(LIntCol);
  3643. LgrcPosition := GetCursorAtPos(LIntCol, FPosInCharField);
  3644. MoveColRow(LgrcPosition.x, LgrcPosition.y, True, True);
  3645. end
  3646. else
  3647. begin
  3648. LIntCol := LIntCol - 1;
  3649. LgrcPosition := GetCursorAtPos(LIntCol, False);
  3650. if (LgrcPosition.x = Col) and not (LIntCol = DataSize) then
  3651. MoveColRow(Col + 1, Row, True, True)
  3652. else
  3653. begin
  3654. LIntCol := LIntCol + 1;
  3655. if (LIntCol < DataSize) or ((LIntCol = DataSize) and InsertMode)
  3656. then
  3657. begin
  3658. LgrcPosition := GetCursorAtPos(LIntCol, FPosInCharField);
  3659. MoveColRow(LgrcPosition.x, LgrcPosition.y, True, True);
  3660. end;
  3661. end
  3662. end;
  3663. end;
  3664. end
  3665. else
  3666. begin
  3667. LIntCol := GetLastCharCol;
  3668. MoveColRow(LIntCol, Row, True, True);
  3669. end;
  3670. end;
  3671. VK_DOWN:
  3672. begin
  3673. if (not (ssCTRL in Shift)) then
  3674. begin
  3675. LIntRow := Row + 1;
  3676. LIntCol := Col;
  3677. if LIntRow < RowCount then
  3678. begin
  3679. MoveColRow(LIntCol, LIntRow, True, True);
  3680. end
  3681. end;
  3682. end;
  3683. VK_UP:
  3684. begin
  3685. if (not (ssCTRL in Shift)) then
  3686. begin
  3687. LIntRow := Row - 1;
  3688. LIntCol := Col;
  3689. if LIntRow >= GRID_FIXED then
  3690. begin
  3691. MoveColRow(LIntCol, LIntRow, True, True);
  3692. end
  3693. end;
  3694. end;
  3695. Word('T'): if (ssCtrl in Shift) then
  3696. begin
  3697. Col := Max(GRID_FIXED, GetOtherFieldCol(Col));
  3698. end;
  3699. VK_TAB: if ((Shift = []) or (Shift = [ssShift])) then
  3700. begin // tab-taste
  3701. Col := Max(GRID_FIXED, GetOtherFieldCol(Col));
  3702. end;
  3703. Word('0')..Word('9'): if ssCtrl in Shift then
  3704. begin
  3705. if ssShift in Shift then
  3706. begin
  3707. LIntRow := GetPosAtCursor(Col, Row);
  3708. SetBookmarkVals(Key - Ord('0'), LIntRow, FPosInCharField);
  3709. end
  3710. else
  3711. begin
  3712. GotoBookmark(Key - Ord('0'));
  3713. end;
  3714. end;
  3715. VK_SHIFT: if (Shift = [ssShift]) or (Shift = [ssShift, ssCtrl]) then
  3716. begin // selektion starten
  3717. FIsSelecting := True;
  3718. end;
  3719. VK_DELETE: if (not FReadOnlyView) then
  3720. begin
  3721. if (SelCount > 0) and (InsertMode or (Shift = [ssCtrl])) then
  3722. DeleteSelection
  3723. else if InsertMode or (Shift = [ssCtrl]) then
  3724. InternalErase(False)
  3725. end;
  3726. VK_INSERT: if (Shift = []) then InsertMode := not InsertMode;
  3727. end;
  3728. end;
  3729. function TCustomMPHexEditor.HasChanged(aPos: integer): boolean;
  3730. begin
  3731. Result := False;
  3732. if InsertMode then
  3733. Exit;
  3734. if FModifiedBytes.Size > aPos then
  3735. Result := FModifiedBytes.Bits[aPos];
  3736. end;
  3737. function TCustomMPHexEditor.IsSelected(const APosition: integer): boolean;
  3738. begin
  3739. Result := False;
  3740. if (FSelPosition <> -1) and (APosition >= FSelStart) and (APosition <= FSelEnd)
  3741. then
  3742. begin
  3743. Result := True
  3744. end;
  3745. end;
  3746. procedure TCustomMPHexEditor.NewSelection(SelFrom, SelTo: integer);
  3747. var
  3748. LIntSelStart, LIntSelEnd, LIntSelPos: integer;
  3749. LIntOldStart, LIntNewStart, LIntOldEnd, LIntNewEnd: integer;
  3750. begin
  3751. CheckSelectUnit(SelFrom, SelTo);
  3752. LIntSelEnd := FSelEnd;
  3753. LIntSelStart := FSelStart;
  3754. LIntSelPos := FSelPosition;
  3755. SetSelection(SelFrom, Min(SelFrom, SelTo), Max(SelFrom, SelTo));
  3756. if (LIntSelPos = -1) then
  3757. RedrawPos(Min(FSelStart, FSelEnd), Max(FSelStart, FSelEnd))
  3758. else
  3759. begin
  3760. // den bereich neu zeichnen, der neu selektiert ist, sowie den, der nicht mehr selektiert ist
  3761. // hinzugekommene selektion berechnen
  3762. LIntNewStart := Min(SelFrom, SelTo);
  3763. LIntOldStart := Min(LIntSelEnd, LIntSelStart);
  3764. LIntNewEnd := Max(SelFrom, SelTo);
  3765. LIntOldEnd := Max(LIntSelEnd, LIntSelStart);
  3766. RedrawPos(Min(LIntNewStart, LIntOldStart), Max(LIntNewStart, LIntOldStart));
  3767. RedrawPos(Min(LIntOldEnd, LIntNewEnd), Max(LIntOldEnd, LIntNewEnd));
  3768. end;
  3769. SelectionChanged;
  3770. end;
  3771. function TCustomMPHexEditor.GetOffsetFormat: string;
  3772. begin
  3773. Result := FOffsetFormat.Format;
  3774. end;
  3775. procedure TCustomMPHexEditor.SetOffsetFormat(const Value: string);
  3776. begin
  3777. if Value <> FOffsetFormat.Format then
  3778. try
  3779. GenerateOffsetFormat(Value);
  3780. SetOffsetDisplayWidth;
  3781. Invalidate;
  3782. except
  3783. GenerateOffsetFormat(FOffsetFormat.Format);
  3784. raise;
  3785. end;
  3786. end;
  3787. procedure TCustomMPHexEditor.SetHexLowerCase(const Value: boolean);
  3788. begin
  3789. if FHexLowerCase <> Value then
  3790. begin
  3791. FHexLowerCase := Value;
  3792. if Value then
  3793. Move(HEX_LOWER[1], FHexChars, sizeof(FHexChars))
  3794. else
  3795. Move(HEX_UPPER[1], FHexChars, sizeof(FHexChars));
  3796. Invalidate;
  3797. end;
  3798. end;
  3799. procedure TCustomMPHexEditor.GenerateOffsetFormat(Value: string);
  3800. var
  3801. LIntTemp: integer;
  3802. LStrTemp: string;
  3803. begin
  3804. with FOffsetFormat do
  3805. begin
  3806. Flags := [];
  3807. LStrTemp := Value;
  3808. // aufbau: [r|c|<HEXNUM>%][-|<HEXNUM>!]<HEXNUM>:[Prefix]|[Suffix]
  3809. if LStrTemp <> '' then
  3810. begin
  3811. // bytes per unit
  3812. if UpperCase(Copy(LStrTemp, 1, 2)) = 'R%' then
  3813. begin
  3814. Flags := Flags + [offCalcRow];
  3815. Delete(LStrTemp, 1, 2);
  3816. _BytesPerUnit := BytesPerRow;
  3817. end
  3818. else if UpperCase(Copy(LStrTemp, 1, 2)) = 'C%' then
  3819. begin
  3820. Flags := Flags + [offCalcColumn];
  3821. Delete(LStrTemp, 1, 2);
  3822. _BytesPerUnit := BytesPerColumn;
  3823. end
  3824. else
  3825. begin
  3826. LIntTemp := 1;
  3827. while (LIntTemp <= Length(LStrTemp)) and
  3828. (LStrTemp[LIntTemp] in ['0'..'9', 'A'..'F', 'a'..'f']) do
  3829. Inc(LIntTemp);
  3830. if Copy(LStrTemp, LIntTemp, 1) = '%' then
  3831. begin
  3832. // width field
  3833. if LIntTemp = 1 then
  3834. begin
  3835. Flags := Flags + [offBytesPerUnit];
  3836. _BytesPerUnit := FUsedRulerBytesPerUnit;
  3837. Delete(LStrTemp, 1, 1)
  3838. end
  3839. else
  3840. begin
  3841. _BytesPerUnit := RadixToInt(Copy(LStrTemp, 1, LIntTemp - 1), 16);
  3842. // StrToInt('$'+Copy(LStrTemp, 1, LIntTemp-1));
  3843. Delete(LStrTemp, 1, LIntTemp);
  3844. end;
  3845. end
  3846. else
  3847. begin
  3848. Flags := Flags + [offBytesPerUnit];
  3849. _BytesPerUnit := FUsedRulerBytesPerUnit;
  3850. end;
  3851. end;
  3852. if not (_BytesPerUnit in [1, 2, 4, 8]) then
  3853. raise EMPHexEditor.CreateFmt(ERR_INVALID_BPU, [_BytesPerUnit]);
  3854. // auto calc width
  3855. if Copy(LStrTemp, 1, 2) = '-!' then
  3856. begin
  3857. Flags := Flags + [offCalcWidth];
  3858. Delete(LStrTemp, 1, 2);
  3859. MinWidth := 1;
  3860. end
  3861. else
  3862. begin
  3863. // width ?
  3864. LIntTemp := 1;
  3865. while (LIntTemp <= Length(LStrTemp)) and
  3866. (LStrTemp[LIntTemp] in ['0'..'9', 'A'..'F', 'a'..'f']) do
  3867. Inc(LIntTemp);
  3868. if Copy(LStrTemp, LIntTemp, 1) = '!' then
  3869. begin
  3870. // width field
  3871. if LIntTemp = 1 then
  3872. begin
  3873. MinWidth := 1;
  3874. Delete(LStrTemp, 1, 1)
  3875. end
  3876. else
  3877. begin
  3878. MinWidth := RadixToInt(Copy(LStrTemp, 1, LIntTemp - 1), 16);
  3879. // StrToInt('$'+Copy(LStrTemp, 1, LIntTemp-1));
  3880. Delete(LStrTemp, 1, LIntTemp);
  3881. end;
  3882. end
  3883. else
  3884. MinWidth := 1;
  3885. end;
  3886. // radix
  3887. LIntTemp := 1;
  3888. while (LIntTemp <= Length(LStrTemp)) and (LStrTemp[LIntTemp] in ['0'..'9',
  3889. 'A'..'F', 'a'..'f']) do
  3890. Inc(LIntTemp);
  3891. if LIntTemp = 1 then
  3892. raise EMPHexEditor.CreateFmt(ERR_MISSING_FORMATCHAR, ['number radix']);
  3893. if Copy(LStrTemp, LIntTemp, 1) <> ':' then
  3894. raise EMPHexEditor.CreateFmt(ERR_MISSING_FORMATCHAR, [':']);
  3895. Radix := RadixToInt(Copy(LStrTemp, 1, LIntTemp - 1), 16);
  3896. if not (Radix in [2..16]) then
  3897. raise EMPHexEditor.CreateFmt(ERR_INVALID_FORMATRADIX, [Radix]);
  3898. Delete(LStrTemp, 1, LIntTemp);
  3899. // prefix, suffix
  3900. LIntTemp := Pos('|', LStrTemp);
  3901. if LIntTemp = 0 then
  3902. raise EMPHexEditor.CreateFmt(ERR_MISSING_FORMATCHAR, ['|']);
  3903. Prefix := Copy(LStrTemp, 1, LIntTemp - 1);
  3904. Suffix := Copy(LStrTemp, LIntTemp + 1, MaxInt);
  3905. end;
  3906. Format := Value;
  3907. end;
  3908. end;
  3909. procedure TCustomMPHexEditor.Select(const aCurCol, aCurRow, aNewCol, aNewRow:
  3910. integer);
  3911. var
  3912. LIntOldStart,
  3913. //LIntOldEnd,
  3914. LIntNewStart,
  3915. LIntNewEnd: integer;
  3916. begin
  3917. //LIntOldEnd := FSelEnd;
  3918. //LIntOldStart := FSelStart;
  3919. LIntNewStart := GetPosAtCursor(aNewCol, aNewRow);
  3920. if FSelPosition = -1 then
  3921. begin
  3922. LIntOldStart := LIntNewStart;
  3923. //LIntOldEnd := LIntNewStart;
  3924. LIntNewEnd := GetPosAtCursor(aCurCol, aCurRow);
  3925. NewSelection(LIntNewEnd, LIntOldStart); // abcd
  3926. //SetSelection(LIntNewEnd, Min(LIntOldStart, LIntNewEnd), Max(LIntNewEnd,
  3927. //LIntOldEnd));
  3928. //RedrawPos(FSelStart, FSelEnd)
  3929. end
  3930. else
  3931. //begin
  3932. NewSelection(FSelPosition, LIntNewStart); // abcd
  3933. (*// testen, ob neue selection /\ liegt als fSelPO
  3934. // wenn ja, dann start = sel, ende = selpo
  3935. if LIntNewStart < FSelPosition then
  3936. begin
  3937. NewSelection(FSelPosition, LIntNewStart);// abcd
  3938. //SetSelection(FSelPosition, LIntNewStart, FSelPosition);
  3939. //RedrawPos(Min(FSelStart, LIntOldStart), Max(FSelStart, LIntOldStart));
  3940. //RedrawPos(Min(FSelEnd, LIntOldEnd), Max(FSelEnd, LIntOldEnd));
  3941. end
  3942. else
  3943. begin
  3944. NewSelection(FSelPosition, LIntNewStart); //abcd
  3945. //SetSelection(FSelPosition, FSelPosition, LIntNewStart);
  3946. //RedrawPos(Min(FSelStart, LIntOldStart), Max(FSelStart, LIntOldStart));
  3947. //RedrawPos(Min(FSelEnd, LIntOldEnd), Max(FSelEnd, LIntOldEnd));
  3948. end;
  3949. end;*)
  3950. end;
  3951. procedure TCustomMPHexEditor.RedrawPos(aFrom, aTo: integer);
  3952. var
  3953. LRctBox: TRect;
  3954. begin
  3955. aFrom := GetRow(aFrom);
  3956. aTo := GetRow(aTo);
  3957. LRctBox := BoxRect(GRID_FIXED, aFrom, GetLastCharCol, aTo);
  3958. InvalidateRect(Handle, @LRctBox, False);
  3959. end;
  3960. procedure TCustomMPHexEditor.ResetSelection(const aDraw: boolean);
  3961. var
  3962. LIntOldStart,
  3963. LIntOldEnd: integer;
  3964. begin
  3965. FIsSelecting := False;
  3966. LIntOldStart := FSelStart;
  3967. LIntOldEnd := FSelEnd;
  3968. SetSelection(-1, -1, -1);
  3969. FSelBeginPosition := -1;
  3970. if aDraw and ((LIntOldStart > -1) or (LIntOldStart > -1)) then
  3971. RedrawPos(LIntOldStart, LIntOldEnd);
  3972. end;
  3973. procedure TCustomMPHexEditor.MouseDown(Button: TMouseButton; Shift: TShiftState;
  3974. X, Y: integer);
  3975. var
  3976. LgrcDummy: TGridCoord;
  3977. lboolInherited: boolean;
  3978. begin
  3979. FIsSelecting := False;
  3980. FMouseUpCanResetSel := False;
  3981. if Button = mbLeft then
  3982. LgrcDummy := CheckMouseCoord(X, Y);
  3983. // do not change selection when clicking ruler or offset panel.
  3984. if (not MouseOverSelection) and (not MouseOverFixed(x, y)) then
  3985. begin
  3986. lBoolInherited := True;
  3987. inherited MouseDown(Button, Shift, x, y);
  3988. end
  3989. else
  3990. begin
  3991. lboolInherited := False;
  3992. // but set focus if possible (05/27/2004)
  3993. if not (csDesigning in ComponentState) and
  3994. (CanFocus or (GetParentForm(Self) = nil)) then
  3995. SetFocus;
  3996. end;
  3997. if (GetParentForm(self) <> nil) then
  3998. if (GetParentForm(self).ActiveControl = self) then
  3999. if GetParentForm(self) <> Screen.ActiveForm then
  4000. if HandleAllocated then
  4001. Windows.SetFocus(self.Handle);
  4002. if (Button = mbLeft) and (not MouseOverSelection) and
  4003. (LgrcDummy.X >= GRID_FIXED) and (LgrcDummy.Y >= GRID_FIXED) then
  4004. begin
  4005. ResetSelection(True);
  4006. if not (ssDouble in Shift) then
  4007. FIsSelecting := True;
  4008. end;
  4009. if (Button = mbLeft) and MouseOverSelection then
  4010. begin
  4011. FMouseDownCol := x;
  4012. FMouseDownRow := y;
  4013. FMouseUpCanResetSel := True;
  4014. end;
  4015. if (not lBoolInherited) and (Assigned(OnMouseDown)) and Focused then
  4016. OnMouseDown(self, Button, Shift, X, Y);
  4017. end;
  4018. procedure TCustomMPHexEditor.InternalGetCurSel(var StartPos, EndPos, ACol, ARow:
  4019. integer);
  4020. begin
  4021. if FSelPosition = -1 then
  4022. begin
  4023. StartPos := GetPosAtCursor(Col, Row);
  4024. EndPos := StartPos + 1;
  4025. aCol := Col;
  4026. aRow := Row;
  4027. end
  4028. else
  4029. begin
  4030. StartPos := FSelStart;
  4031. EndPos := FSelEnd + 1;
  4032. with GetCursorAtPos(FSelStart, InCharField) do
  4033. begin
  4034. aCOL := X;
  4035. aROW := Y;
  4036. end;
  4037. end;
  4038. if FModifiedBytes.Size > StartPos then
  4039. FModifiedBytes.Size := StartPos;
  4040. end;
  4041. function TCustomMPHexEditor.CreateShift4BitStream(const StartPos: integer; var
  4042. FName: TFileName): TFileStream;
  4043. var
  4044. LByt1,
  4045. LByt2: byte;
  4046. LBytBuffer: array[0..511] of byte;
  4047. LIntLoop,
  4048. LIntRead: integer;
  4049. begin
  4050. Result := nil;
  4051. if StartPos >= DataSize then
  4052. Exit;
  4053. FName := GetTempName;
  4054. Result := TFileStream.Create(FName, fmCreate);
  4055. Result.Position := 0;
  4056. FDataStorage.Position := StartPos;
  4057. LByt1 := 0;
  4058. while FDataStorage.Position < DataSize do
  4059. begin
  4060. FillChar(LBytBuffer[0], 512, 0);
  4061. LIntRead := FDataStorage.Read(LBytBuffer[0], 512);
  4062. for LIntLoop := 0 to Pred(LIntRead) do
  4063. begin
  4064. LByt2 := LBytBuffer[LIntLoop] and 15;
  4065. LBytBuffer[LIntLoop] := (LBytBuffer[LIntLoop] shr 4) or (LByt1 shl 4);
  4066. LByt1 := LByt2;
  4067. end;
  4068. Result.WriteBuffer(LBytBuffer[0], LIntRead);
  4069. end;
  4070. Result.Position := 0;
  4071. end;
  4072. function TCustomMPHexEditor.InternalInsertNibble(const Pos: integer; const
  4073. HighNibble: boolean): boolean;
  4074. var
  4075. LfstNibbleStream: TFileStream;
  4076. LStrFName: TFileName;
  4077. LIntOldSize: integer;
  4078. LByteFirst,
  4079. LByteLast: byte;
  4080. begin
  4081. Result := False;
  4082. if DataSize = 0 then
  4083. Exit;
  4084. LIntOldSize := FDataStorage.Size;
  4085. WaitCursor;
  4086. try
  4087. // nun zuerst alle restlichen bits verschieben
  4088. LByteFirst := Data[Pos];
  4089. LByteLast := Data[Pred(DataSize)];
  4090. LfstNibbleStream := CreateShift4BitStream(Pos, LStrFName);
  4091. with LfstNibbleStream do
  4092. try
  4093. FDataStorage.Position := Pos;
  4094. FDataStorage.CopyFrom(LfstNibbleStream, LfstNibbleStream.Size);
  4095. finally
  4096. Free;
  4097. DeleteFile(LStrFName);
  4098. end;
  4099. if HighNibble then
  4100. LByteFirst := LByteFirst shr 4
  4101. else
  4102. LByteFirst := LByteFirst and 240;
  4103. Data[Pos] := LByteFirst;
  4104. FDataStorage.Size := LIntOldSize + 1;
  4105. Data[Pred(DataSize)] := LByteLast shl 4;
  4106. Result := True;
  4107. finally
  4108. OldCursor;
  4109. end;
  4110. end;
  4111. function TCustomMPHexEditor.InsertNibble(const aPos: integer; const HighNibble:
  4112. boolean; const UndoDesc: string = ''): boolean;
  4113. const
  4114. L_BytAppend: byte = 0;
  4115. begin
  4116. Result := False;
  4117. if DataSize < 1 then
  4118. begin
  4119. ResetSelection(False);
  4120. AppendBuffer(PChar(@L_BytAppend), 1);
  4121. Result := True;
  4122. Exit;
  4123. end;
  4124. if (aPos >= DataSize) or (aPos < 0) then
  4125. Exit;
  4126. CreateUndo(ufKindNibbleInsert, aPos, 0, 0, UndoDesc);
  4127. ResetSelection(False);
  4128. Result := InternalInsertNibble(aPos, HighNibble);
  4129. if Result and (FModifiedBytes.Size >= (aPos)) then
  4130. FModifiedBytes.Size := aPos;
  4131. CalcSizes;
  4132. Changed;
  4133. end;
  4134. function TCustomMPHexEditor.InternalDeleteNibble(const Pos: integer; const
  4135. HighNibble: boolean): boolean;
  4136. var
  4137. LfstNibbleStream: TFileStream;
  4138. LStrFName: TFileName;
  4139. LIntOldSize: integer;
  4140. LByt1: byte;
  4141. begin
  4142. Result := False;
  4143. if DataSize = 0 then
  4144. Exit;
  4145. LIntOldSize := FDataStorage.Size;
  4146. WaitCursor;
  4147. try
  4148. // nun zuerst alle restlichen bits verschieben
  4149. LByt1 := Data[Pos];
  4150. LfstNibbleStream := CreateShift4BitStream(Pos, LStrFName);
  4151. with LfstNibbleStream do
  4152. try
  4153. FDataStorage.Position := Pos;
  4154. Position := 1;
  4155. FDataStorage.CopyFrom(LfstNibbleStream, LfstNibbleStream.Size - 1);
  4156. finally
  4157. Free;
  4158. DeleteFile(LStrFName);
  4159. end;
  4160. if not HighNibble then
  4161. Data[Pos] := (LByt1 and 240) or (Data[Pos] and 15);
  4162. Result := True;
  4163. FDataStorage.Size := LIntOldSize;
  4164. Data[Pred(DataSize)] := Data[Pred(DataSize)] shl 4;
  4165. finally
  4166. OldCursor;
  4167. end;
  4168. end;
  4169. function TCustomMPHexEditor.DeleteNibble(const aPos: integer; const HighNibble:
  4170. boolean; const UndoDesc: string = ''): boolean;
  4171. begin
  4172. Result := False;
  4173. if (aPos >= DataSize) or (aPos < 0) then
  4174. Exit;
  4175. CreateUndo(ufKindNibbleDelete, aPos, 0, 0, UndoDesc);
  4176. ResetSelection(False);
  4177. Result := InternalDeleteNibble(aPos, HighNibble);
  4178. if Result and (FModifiedBytes.Size >= (aPos)) then
  4179. FModifiedBytes.Size := aPos;
  4180. CalcSizes;
  4181. Changed;
  4182. end;
  4183. procedure TCustomMPHexEditor.InternalConvertRange(const aFrom, aTo: integer;
  4184. const aTransFrom, aTransTo: TMPHTranslationKind);
  4185. var
  4186. LIntSize: integer;
  4187. begin
  4188. LIntSize := (aTo - aFrom) + 1;
  4189. WaitCursor;
  4190. try
  4191. FDataStorage.TranslateToAnsi(aTransFrom, aFrom, LIntSize);
  4192. FDataStorage.TranslateFromAnsi(aTransTo, aFrom, LIntSize);
  4193. finally
  4194. OldCursor;
  4195. end;
  4196. end;
  4197. procedure TCustomMPHexEditor.ConvertRange(const aFrom, aTo: integer; const
  4198. aTransFrom, aTransTo: TMPHTranslationKind; const UndoDesc: string = '');
  4199. begin
  4200. if aFrom > aTo then
  4201. Exit;
  4202. if aTransFrom = aTransTo then
  4203. Exit;
  4204. if (aTo >= DataSize) or (aFrom < 0) then
  4205. Exit;
  4206. CreateUndo(ufKindConvert, aFrom, (aTo - aFrom) + 1, 0, UndoDesc);
  4207. InternalConvertRange(aFrom, aTo, aTransFrom, aTransTo);
  4208. Invalidate;
  4209. Changed;
  4210. end;
  4211. procedure TCustomMPHexEditor.InternalDelete(StartPos, EndPos, ACol, ARow:
  4212. integer);
  4213. var
  4214. LgrdEndPos: TGridCoord;
  4215. LIntNewCol: integer;
  4216. begin
  4217. if EndPos <= (DataSize - 1) then
  4218. MoveFileMem(EndPos, StartPos, DataSize - EndPos);
  4219. FDataStorage.Size := DataSize - (EndPos - StartPos);
  4220. EndPos := GetPosAtCursor(aCol, aRow);
  4221. if DataSize < 1 then
  4222. begin
  4223. LIntNewCol := GRID_FIXED;
  4224. if FPosInCharField then
  4225. LIntNewCol := Max(GRID_FIXED, GetOtherFieldColCheck(LIntNewCol));
  4226. MoveColRow(LIntNewCol, GRID_FIXED, True, True)
  4227. end
  4228. else if EndPos >= DataSize then
  4229. begin
  4230. if InsertMode then
  4231. LgrdEndPos := GetCursorAtPos(DataSize, FPosInCharField)
  4232. else
  4233. LgrdEndPos := GetCursorAtPos(DataSize - 1, FPosInCharField);
  4234. MoveColRow(LgrdEndPos.x, LgrdEndPos.y, True, True);
  4235. end
  4236. else if ACol > -1 then
  4237. MoveColRow(aCol, aRow, True, True);
  4238. CalcSizes;
  4239. ResetSelection(False);
  4240. Invalidate;
  4241. end;
  4242. procedure TCustomMPHexEditor.DeleteSelection(const UndoDesc: string = '');
  4243. var
  4244. LIntSelStart,
  4245. LIntSelEnd,
  4246. LIntCol,
  4247. LIntRow: integer;
  4248. begin
  4249. InternalGetCurSel(LIntSelStart, LIntSelEnd, LIntCol, LIntRow);
  4250. CreateUndo(ufKindByteRemoved, LIntSelStart, LIntSelEnd - LIntSelStart,
  4251. 0, UndoDesc);
  4252. InternalDelete(LIntSelStart, LIntSelEnd, LIntCol, LIntRow);
  4253. Changed;
  4254. end;
  4255. procedure TCustomMPHexEditor.CreateUndo(const aKind: TMPHUndoFlag; const aPos,
  4256. aCount, aReplCount: integer; const sDesc: string = '');
  4257. begin
  4258. if CanCreateUndo(aKind, aCount, aReplCount) then
  4259. begin
  4260. if FUndoStorage.UpdateCount = 0 then
  4261. FUndoStorage.CreateUndo(aKind, aPos, aCount, aReplCount, sDesc);
  4262. FModified := True;
  4263. //Changed;
  4264. end
  4265. else
  4266. raise EMPHexEditor.Create(ERR_NOUNDO);
  4267. end;
  4268. procedure TCustomMPHexEditor.ResetUndo;
  4269. begin
  4270. FUndoStorage.Reset;
  4271. end;
  4272. function TCustomMPHexEditor.GetCanUndo: boolean;
  4273. begin
  4274. Result := (not FReadOnlyView) and FUndoStorage.CanUndo;
  4275. end;
  4276. function TCustomMPHexEditor.GetCanRedo: boolean;
  4277. begin
  4278. Result := (not FReadOnlyView) and FUndoStorage.CanRedo;
  4279. end;
  4280. function TCustomMPHexEditor.GetUndoDescription: string;
  4281. begin
  4282. if not (csDestroying in ComponentState) then
  4283. begin
  4284. with FUndoStorage do
  4285. if CanUndo then
  4286. Result := Description
  4287. else
  4288. Result := UNDO_NOUNDO;
  4289. end
  4290. else
  4291. Result := UNDO_NOUNDO;
  4292. end;
  4293. function TCustomMPHexEditor.GetSelStart: integer;
  4294. begin
  4295. if FSelPosition = -1 then
  4296. begin
  4297. Result := GetPosAtCursor(Col, Row);
  4298. end
  4299. else
  4300. Result := FSelPosition;
  4301. end;
  4302. function TCustomMPHexEditor.GetSelEnd: integer;
  4303. begin
  4304. if FSelPosition = -1 then
  4305. Result := GetPosAtCursor(Col, Row)
  4306. else
  4307. begin
  4308. Result := FSelEnd;
  4309. if FSelPosition = FSelEnd then
  4310. Result := FSelStart;
  4311. end;
  4312. end;
  4313. procedure TCustomMPHexEditor.SetSelStart(aValue: integer);
  4314. begin
  4315. if (aValue < 0) or (aValue >= DataSize) then
  4316. raise EMPHexEditor.Create(ERR_INVALID_SELSTART)
  4317. else
  4318. begin
  4319. ResetSelection(True);
  4320. with GetCursorAtPos(aValue, InCharField) do
  4321. MoveColRow(X, Y, True, True);
  4322. end;
  4323. end;
  4324. procedure TCustomMPHexEditor.SetSelEnd(aValue: integer);
  4325. begin
  4326. if (aValue < -1) or (aValue >= DataSize) then
  4327. raise EMPHexEditor.Create(ERR_INVALID_SELEND)
  4328. else
  4329. begin
  4330. ResetSelection(True);
  4331. if aValue > -1 then
  4332. begin
  4333. with GetCursorAtPos(aValue, InCharField) do
  4334. Select(Col, Row, X, Y);
  4335. SelectionChanged;
  4336. end;
  4337. end;
  4338. end;
  4339. procedure TCustomMPHexEditor.SetInCharField(const Value: boolean);
  4340. begin
  4341. if (DataSize < 1) then
  4342. Exit;
  4343. if InCharField <> Value then
  4344. MoveColRow(GetOtherFieldCol(Col), Row, True, True);
  4345. end;
  4346. function TCustomMPHexEditor.GetInCharField: boolean;
  4347. begin
  4348. Result := False;
  4349. if DataSize < 1 then
  4350. Exit;
  4351. GetPosAtCursor(Col, Row);
  4352. Result := FPosInCharField;
  4353. end;
  4354. procedure TCustomMPHexEditor.Loaded;
  4355. begin
  4356. inherited;
  4357. CreateEmptyFile(UNNAMED_FILE);
  4358. end;
  4359. procedure TCustomMPHexEditor.CreateWnd;
  4360. begin
  4361. inherited;
  4362. if (csDesigning in ComponentState) or (FFileName = '---') then
  4363. CreateEmptyFile(UNNAMED_FILE);
  4364. end;
  4365. procedure TCustomMPHexEditor.WMSetFocus(var Msg: TWMSetFocus);
  4366. begin
  4367. inherited;
  4368. CreateCaretGlyph;
  4369. CheckSetCaret;
  4370. Invalidate;
  4371. end;
  4372. procedure TCustomMPHexEditor.WMKillFocus(var Msg: TWMKillFocus);
  4373. begin
  4374. inherited;
  4375. HideCaret(Handle);
  4376. DestroyCaret();
  4377. FIsSelecting := False;
  4378. Invalidate;
  4379. end;
  4380. procedure TCustomMPHexEditor.CMINTUPDATECARET(var Msg: TMessage);
  4381. begin
  4382. if Msg.WParam = 7 then
  4383. begin
  4384. CheckSetCaret;
  4385. end;
  4386. end;
  4387. procedure TCustomMPHexEditor.SetTranslation(const Value: TMPHTranslationKind);
  4388. begin
  4389. if FTranslation <> Value then
  4390. begin
  4391. if (Value <> tkAsIs) and FUnicodeCharacters then
  4392. raise EMPHexEditor.Create(ERR_NO_TRANSLATION_IN_UNICODE_MODE);
  4393. FTranslation := Value;
  4394. Invalidate;
  4395. end;
  4396. end;
  4397. procedure TCustomMPHexEditor.SetModified(const Value: boolean);
  4398. begin
  4399. FModified := Value;
  4400. if not Value then
  4401. begin
  4402. ResetUndo;
  4403. FModifiedBytes.Size := 0;
  4404. Invalidate;
  4405. end;
  4406. end;
  4407. procedure TCustomMPHexEditor.SetBytesPerRow(const Value: integer);
  4408. var
  4409. LIntPos,
  4410. LIntSelPos,
  4411. LIntSelStart,
  4412. LIntSelEnd: integer;
  4413. LBoolInCharField,
  4414. LBool2ndCol: boolean;
  4415. begin
  4416. if ((Value < 1) or (Value > 256)) or
  4417. (FUnicodeCharacters and ((Value mod 2) <> 0)) then
  4418. raise EMPHexEditor.Create(ERR_INVALID_BYTESPERLINE)
  4419. else if FBytesPerRow <> Value then
  4420. begin
  4421. with FOffsetFormat do
  4422. if offCalcRow in Flags then
  4423. _BytesPerUnit := Value;
  4424. LIntSelPos := FSelPosition;
  4425. LIntSelStart := FSelStart;
  4426. LIntSelEnd := FSelEnd;
  4427. LIntPos := GetPosAtCursor(Col, Row);
  4428. LBoolInCharField := FPosInCharField;
  4429. LBool2ndCol := GetCursorAtPos(LIntPos, LBoolInCharField).x <> Col;
  4430. FBytesPerRow := Value;
  4431. FBytesPerRowDup := Value * 2;
  4432. FIntLastHexCol := (GRID_FIXED + FBytesPerRowDup - 1);
  4433. SetRulerString;
  4434. CalcSizes;
  4435. if (LIntPos >= DataSize) or (InsertMode and (LIntPos > DataSize)) then
  4436. LIntPos := DataSize - 1;
  4437. with GetCursorAtPos(LIntPos, LBoolInCharField) do
  4438. begin
  4439. if LBool2ndCol then
  4440. Inc(x);
  4441. MoveColRow(x, y, True, True);
  4442. end;
  4443. SetSelection(LIntSelPos, LIntSelStart, LIntSelEnd);
  4444. end;
  4445. end;
  4446. procedure TCustomMPHexEditor.InternalAppendBuffer(Buffer: PChar; const Size:
  4447. integer);
  4448. var
  4449. LIntSize: integer;
  4450. begin
  4451. if DataSize = 0 then
  4452. begin
  4453. FDataStorage.Position := 0;
  4454. FModifiedBytes.Size := 0;
  4455. end;
  4456. LIntSize := DataSize;
  4457. FDataStorage.Size := LIntSize + Size;
  4458. WriteBuffer(Buffer^, LIntSize, Size);
  4459. CalcSizes;
  4460. end;
  4461. procedure TCustomMPHexEditor.InternalInsertBuffer(Buffer: PChar; const Size,
  4462. Position: integer);
  4463. var
  4464. LIntSize: integer;
  4465. begin
  4466. if DataSize = 0 then
  4467. begin
  4468. FDataStorage.Position := 0;
  4469. FModifiedBytes.Size := 0;
  4470. end;
  4471. LIntSize := DataSize;
  4472. FDataStorage.Size := LIntSize + Size;
  4473. if Position < LIntSize then
  4474. // nur, wenn nicht hinter streamende, dann platz schaffen
  4475. MoveFileMem(Position, Position + Size, DataSize - Position - Size); //+ 1);
  4476. if Buffer <> nil then
  4477. WriteBuffer(Buffer^, Position, Size);
  4478. CalcSizes;
  4479. end;
  4480. procedure TCustomMPHexEditor.InsertBuffer(aBuffer: PChar; const aSize, aPos:
  4481. integer; const UndoDesc: string = ''; const MoveCursor: Boolean = True);
  4482. begin
  4483. FDataStorage.CheckBounds(aPos);
  4484. CreateUndo(ufKindInsertBuffer, aPos, aSize, 0, UndoDesc);
  4485. InternalInsertBuffer(aBuffer, aSize, aPos);
  4486. if FModifiedBytes.Size >= (aPos) then
  4487. FModifiedBytes.Size := aPos;
  4488. if Enabled then
  4489. begin
  4490. SetSelection(aPos, aPos, aPos + aSize - 1);
  4491. if MoveCursor then
  4492. begin
  4493. with GetCursorAtPos(FSelEnd, InCharField) do
  4494. MoveColRow(x, y, True, True);
  4495. SetSelection(aPos, aPos, aPos + aSize - 1);
  4496. end;
  4497. Invalidate;
  4498. end;
  4499. Changed;
  4500. end;
  4501. procedure TCustomMPHexEditor.AppendBuffer(aBuffer: PChar; const aSize: integer;
  4502. const UndoDesc: string = ''; const MoveCursor: Boolean = True);
  4503. var
  4504. LIntSize: integer;
  4505. begin
  4506. if (not Assigned(aBuffer)) or (aSize = 0) then
  4507. Exit;
  4508. CreateUndo(ufKindAppendBuffer, DataSize, aSize, 0, UndoDesc);
  4509. if FModifiedBytes.Size >= (DataSize) then
  4510. FModifiedBytes.Size := DataSize;
  4511. LIntSize := DataSize;
  4512. InternalAppendBuffer(aBuffer, aSize);
  4513. if MoveCursor then
  4514. with GetCursorAtPos(LIntSize, InCharField) do
  4515. MoveColRow(x, y, True, True);
  4516. SetSelection(LIntSize, LIntSize, LIntSize + aSize - 1);
  4517. Invalidate;
  4518. Changed;
  4519. end;
  4520. procedure TCustomMPHexEditor.ReplaceSelection(aBuffer: PChar; aSize: integer;
  4521. const UndoDesc: string = ''; const MoveCursor: Boolean = True);
  4522. var
  4523. LIntStart,
  4524. LIntEnd,
  4525. LIntCol,
  4526. LIntRow: integer;
  4527. LBoolInCharField: boolean;
  4528. begin
  4529. // auswahl berechnen
  4530. LBoolInCharField := GetInCharField;
  4531. if FSelPosition = -1 then
  4532. InsertBuffer(aBuffer, aSize, SelStart, UndoDesc, MoveCursor)
  4533. else
  4534. begin
  4535. if IsFileSizeFixed then
  4536. begin
  4537. if aSize > SelCount then
  4538. aSize := SelCount
  4539. else if SelCount > aSize then
  4540. begin
  4541. SelStart := Min(SelStart, SelEnd);
  4542. SelEnd := SelStart + aSize - 1;
  4543. end;
  4544. end;
  4545. CreateUndo(ufKindReplace, FSelStart, aSize, SelCount, UndoDesc);
  4546. // zuerst aktuelle auswahl löschen
  4547. InternalGetCurSel(LIntStart, LIntEnd, LIntCol, LIntRow);
  4548. InternalDelete(LIntStart, LIntEnd, LIntCol, LIntRow);
  4549. InternalInsertBuffer(aBuffer, aSize, LIntStart);
  4550. if FModifiedBytes.Size >= LIntStart then
  4551. FModifiedBytes.Size := Max(0, LIntStart);
  4552. if MoveCursor then
  4553. begin
  4554. with GetCursorAtPos(LIntStart + aSize - 1, LBoolInCharField) do
  4555. MoveColRow(x, y, True, True);
  4556. SetSelection(LIntStart + aSize - 1, LIntStart, LIntStart + aSize - 1);
  4557. end;
  4558. Invalidate;
  4559. Changed;
  4560. end;
  4561. end;
  4562. procedure TCustomMPHexEditor.SetChanged(DataPos: integer; const Value: boolean);
  4563. begin
  4564. if InsertMode then
  4565. FModifiedBytes.Size := 0;
  4566. if not Value then
  4567. if FModifiedBytes.Size <= DataPos then
  4568. Exit;
  4569. FModifiedBytes[DataPos] := Value;
  4570. end;
  4571. procedure TCustomMPHexEditor.MoveFileMem(const aFrom, aTo, aCount: integer);
  4572. begin
  4573. FDataStorage.Move(aFrom, aTo, aCount);
  4574. end;
  4575. function TCustomMPHexEditor.GetCursorPos: integer;
  4576. begin
  4577. Result := GetPosAtCursor(Col, Row);
  4578. if Result < 0 then
  4579. Result := 0;
  4580. if Result > Max(0, DataSize - 1) then
  4581. Result := Max(0, DataSize - 1)
  4582. end;
  4583. function TCustomMPHexEditor.GetSelCount: integer;
  4584. begin
  4585. if FSelPosition = -1 then
  4586. Result := 0
  4587. else
  4588. Result := Max(FSelStart, FSelEnd) - Min(FSelStart, FSelEnd) + 1;
  4589. end;
  4590. procedure TCustomMPHexEditor.SetReadOnlyFile(const Value: boolean);
  4591. begin
  4592. if Value and (not FIsFileReadonly) then
  4593. begin
  4594. FIsFileReadonly := True;
  4595. end;
  4596. end;
  4597. function TCustomMPHexEditor.BufferFromFile(const aPos: integer; var aCount:
  4598. integer): PChar;
  4599. begin
  4600. if (aPos < 0) or (aPos >= DataSize) then
  4601. raise EMPHexEditor.Create(ERR_INVALID_BUFFERFROMFILE)
  4602. else
  4603. begin
  4604. if (aPos + aCount) > DataSize then
  4605. aCount := (DataSize - aPos) + 1;
  4606. GetMem(Result, aCount);
  4607. try
  4608. FDataStorage.ReadBufferAt(Result^, aPos, aCount);
  4609. except
  4610. try
  4611. FreeMem(Result);
  4612. except
  4613. end;
  4614. Result := nil;
  4615. aCount := 0;
  4616. end;
  4617. end;
  4618. end;
  4619. procedure TCustomMPHexEditor.WMVScroll(var Msg: TWMVScroll);
  4620. begin
  4621. inherited;
  4622. CheckSetCaret;
  4623. end;
  4624. procedure TCustomMPHexEditor.WMHScroll(var Msg: TWMHScroll);
  4625. begin
  4626. inherited;
  4627. CheckSetCaret;
  4628. end;
  4629. procedure TCustomMPHexEditor.CreateCaretGlyph;
  4630. begin
  4631. DestroyCaret();
  4632. FCaretBitmap.Width := FCharWidth;
  4633. FCaretBitmap.Height := FCharHeight - 2;
  4634. FCaretBitmap.Canvas.Brush.Color := clBlack;
  4635. FCaretBitmap.Canvas.FillRect(Rect(0, 0, FCharWidth, FCharHeight - 2));
  4636. FCaretBitmap.Canvas.Brush.Color := clWhite;
  4637. case FCaretKind of
  4638. ckFull: FCaretBitmap.Canvas.FillRect(Rect(0, 0, FCharWidth, FCharHeight -
  4639. 2));
  4640. ckLeft: FCaretBitmap.Canvas.FillRect(Rect(0, 0, 2, FCharHeight - 2));
  4641. ckBottom: FCaretBitmap.Canvas.FillRect(Rect(0, FCharHeight - 4, FCharWidth,
  4642. FCharHeight - 2));
  4643. ckAuto:
  4644. begin
  4645. if FReadOnlyView then
  4646. FCaretBitmap.Canvas.FillRect(Rect(0, FCharHeight - 4, FCharWidth,
  4647. FCharHeight - 2))
  4648. else
  4649. begin
  4650. if FInsertModeOn then
  4651. FCaretBitmap.Canvas.FillRect(Rect(0, 0, 2, FCharHeight - 2))
  4652. else
  4653. FCaretBitmap.Canvas.FillRect(Rect(0, 0, FCharWidth, FCharHeight -
  4654. 2));
  4655. end;
  4656. end;
  4657. end;
  4658. CreateCaret(Handle, FCaretBitmap.Handle, 0, 0);
  4659. ShowCaret(Handle);
  4660. end;
  4661. procedure TCustomMPHexEditor.SetBytesPerColumn(const Value: integer);
  4662. begin
  4663. if ((Value < 1) or (Value > 256)) or
  4664. (FUnicodeCharacters and ((Value mod 2) <> 0)) then
  4665. raise EMPHexEditor.Create(ERR_INVALID_BYTESPERCOL)
  4666. else if FBytesPerCol <> (Value * 2) then
  4667. begin
  4668. with FOffsetFormat do
  4669. if offCalcColumn in Flags then
  4670. _BytesPerUnit := Value;
  4671. FBytesPerCol := Value * 2;
  4672. AdjustMetrics;
  4673. SetRulerString;
  4674. Invalidate;
  4675. end;
  4676. end;
  4677. function TCustomMPHexEditor.GetBytesPerColumn: integer;
  4678. begin
  4679. Result := FBytesPerCol div 2;
  4680. end;
  4681. function TCustomMPHexEditor.PrepareFindReplaceData(StrData: string; const
  4682. IgnoreCase, IsText: boolean): string;
  4683. var
  4684. LWStrTemp: WideString;
  4685. LIntLoop: Integer;
  4686. lChrTbl: Char;
  4687. begin
  4688. if Length(StrData) = 0 then
  4689. Result := ''
  4690. else
  4691. begin
  4692. if IgnoreCase then
  4693. StrData := AnsiLowerCase(StrData);
  4694. if IsText and (FTranslation <> tkAsIs) then
  4695. begin
  4696. UniqueString(StrData);
  4697. TranslateBufferFromAnsi(FTranslation, @StrData[1], @StrData[1],
  4698. Length(StrData));
  4699. end;
  4700. if (not IsText) or (not FUnicodeCharacters) then
  4701. Result := StrData
  4702. else
  4703. begin
  4704. // create a unicode string
  4705. LWStrTemp := StrData;
  4706. if FUnicodeBigEndian then
  4707. for LIntLoop := 1 to Length(LWStrTemp) do
  4708. SwapWideChar(LWStrTemp[LIntLoop]);
  4709. SetLength(Result, Length(LWStrTemp) * 2);
  4710. Move(LWStrTemp[1], Result[1], Length(Result));
  4711. end;
  4712. // create compare tables
  4713. for LChrTbl := #0 to #255 do
  4714. begin
  4715. FFindTable[LChrTbl] := LChrTbl;
  4716. FFindTableI[LChrTbl] := LChrTbl;
  4717. if FTranslation <> tkAsIs then
  4718. TranslateBufferToAnsi(FTranslation, @FFindTableI[LChrTbl],
  4719. @FFindTableI[LChrTbl], 1);
  4720. CharLowerBuff(@FFindTableI[LChrTbl], 1);
  4721. if FTranslation <> tkAsIs then
  4722. TranslateBufferFromAnsi(FTranslation, @FFindTableI[LChrTbl],
  4723. @FFindTableI[LChrTbl], 1);
  4724. end;
  4725. end;
  4726. end;
  4727. function TCustomMPHexEditor.Find(aBuffer: PChar; aCount: integer; const aStart,
  4728. aEnd: integer; const IgnoreCase: boolean): integer;
  4729. var
  4730. LBoolDummy: Boolean;
  4731. LChrCurrent: char;
  4732. LIntCurPos,
  4733. LIntLoop,
  4734. LIntFound,
  4735. LIntEnd: integer;
  4736. cLoop,
  4737. cInc: Cardinal;
  4738. LPTblFind: PMPHFindTable;
  4739. begin
  4740. if Assigned(FOnFind) then
  4741. FOnFind(self, aBuffer, aCount, aStart, aEnd, IgnoreCase, #0, Result)
  4742. else
  4743. begin
  4744. Result := -1;
  4745. LIntEnd := aEnd;
  4746. cLoop := 0;
  4747. if LIntEnd >= DataSize then
  4748. LIntEnd := DataSize - 1;
  4749. if aCount < 1 then
  4750. Exit;
  4751. if aStart + aCount > (LIntEnd + 1) then
  4752. Exit; // will never be found, if search-part is smaller than searched data
  4753. if IgnoreCase then
  4754. LPTblFind := @FFindTableI
  4755. else
  4756. LPTblFind := @FFindTable;
  4757. cInc := DataSize div 500;
  4758. WaitCursor;
  4759. try
  4760. LIntCurPos := aStart;
  4761. LIntLoop := 0;
  4762. LIntFound := LIntCurPos + 1;
  4763. repeat
  4764. if FFindProgress and Assigned(FOnProgress) then
  4765. begin
  4766. Inc(cLoop);
  4767. // changed in 12-28-2004 to avoid edivbyzero
  4768. if (cInc = 0) or ((cLoop mod cInc) = 0) then
  4769. FOnProgress(self, pkFind, FFileName, Round((LIntCurpos / DataSize) *
  4770. 100), LBoolDummy);
  4771. end;
  4772. if LIntCurPos > LIntEnd then
  4773. Exit;
  4774. LChrCurrent := LPTblFind^[char(Data[LIntCurPos])];
  4775. if (LChrCurrent = aBuffer[LIntLoop]) then
  4776. begin
  4777. if LIntLoop = (aCount - 1) then
  4778. begin
  4779. Result := LIntCurPos - aCount + 1;
  4780. Exit;
  4781. end
  4782. else
  4783. begin
  4784. if LIntLoop = 0 then
  4785. LIntFound := LIntCurPos + 1;
  4786. Inc(LIntCurPos);
  4787. Inc(LIntLoop);
  4788. end;
  4789. end
  4790. else
  4791. begin
  4792. LIntCurPos := LIntFound;
  4793. LIntLoop := 0;
  4794. LIntFound := LIntCurPos + 1;
  4795. end;
  4796. until False;
  4797. finally
  4798. OldCursor;
  4799. end;
  4800. end;
  4801. end;
  4802. procedure TCustomMPHexEditor.AddSelectionUndo(const AStart,
  4803. ACount: integer);
  4804. begin
  4805. CreateUndo(ufKindSelection, AStart, aCount, 0, '');
  4806. end;
  4807. function TCustomMPHexEditor.FindWithWildcard(aBuffer: PChar;
  4808. aCount: integer; const aStart, aEnd: integer; const IgnoreCase: boolean;
  4809. const Wildcard: char): integer;
  4810. var
  4811. LBoolDummy: boolean;
  4812. LChrCurrent: char;
  4813. LIntCurPos,
  4814. LIntLoop,
  4815. LIntFound,
  4816. LIntEnd: integer;
  4817. bFound: boolean;
  4818. cLoop,
  4819. cInc: cardinal;
  4820. LPTblFind: PMPHFindTable;
  4821. begin
  4822. if Assigned(FOnWildcardFind) then
  4823. FOnWildcardFind(self, aBuffer, aCount, aStart, aEnd, IgnoreCase, Wildcard,
  4824. Result)
  4825. else
  4826. begin
  4827. Result := -1;
  4828. LIntEnd := aEnd;
  4829. cLoop := 0;
  4830. if LIntEnd >= DataSize then
  4831. LIntEnd := DataSize - 1;
  4832. if aCount < 1 then
  4833. Exit;
  4834. if aStart + aCount > (LIntEnd + 1) then
  4835. Exit; // will never be found, if search-part is smaller than searched data
  4836. if IgnoreCase then
  4837. LPTblFind := @FFindTableI
  4838. else
  4839. LPTblFind := @FFindTable;
  4840. cInc := DataSize div 500;
  4841. WaitCursor;
  4842. try
  4843. LIntCurPos := aStart;
  4844. LIntLoop := 0;
  4845. LIntFound := LIntCurPos + 1;
  4846. repeat
  4847. if FFindProgress and Assigned(FOnProgress) then
  4848. begin
  4849. Inc(cLoop);
  4850. // changed in 12-28-2004 to avoid edivbyzero
  4851. if (cInc = 0) or ((cLoop mod cInc) = 0) then
  4852. FOnProgress(self, pkFind, FFileName, Round((LIntCurpos / DataSize) *
  4853. 100), LBoolDummy);
  4854. end;
  4855. if LIntCurPos > LIntEnd then
  4856. Exit;
  4857. bFound := aBuffer[LIntLoop] = WildCard;
  4858. if not bFound then
  4859. begin
  4860. LChrCurrent := LPTblFind^[char(Data[LIntCurPos])];
  4861. bFound := (LChrCurrent = aBuffer[LIntLoop]);
  4862. end;
  4863. if bFound then
  4864. begin
  4865. if LIntLoop = (aCount - 1) then
  4866. begin
  4867. Result := LIntCurPos - aCount + 1;
  4868. Exit;
  4869. end
  4870. else
  4871. begin
  4872. if LIntLoop = 0 then
  4873. LIntFound := LIntCurPos + 1;
  4874. Inc(LIntCurPos);
  4875. Inc(LIntLoop);
  4876. end;
  4877. end
  4878. else
  4879. begin
  4880. LIntCurPos := LIntFound;
  4881. LIntLoop := 0;
  4882. LIntFound := LIntCurPos + 1;
  4883. end;
  4884. until False;
  4885. finally
  4886. OldCursor;
  4887. end;
  4888. end;
  4889. end;
  4890. procedure TCustomMPHexEditor.SetOffsetDisplayWidth;
  4891. var
  4892. s: string;
  4893. begin
  4894. if Assigned(FOnGetOffsetText) and (not FOffsetHandler) then
  4895. begin
  4896. FOffsetHandler := True;
  4897. try
  4898. FIsMaxOffset := True;
  4899. FOnGetOffsetText(self, (RowCount - 3) * FBytesPerRow, s);
  4900. finally
  4901. FOffsetHandler := False;
  4902. end;
  4903. FOffsetDisplayWidth := Length(s) + 1;
  4904. end
  4905. else
  4906. begin
  4907. with FOffsetFormat do
  4908. if offCalcWidth in Flags then
  4909. MinWidth := Length(IntToRadix(((RowCount - 3) * FBytesPerRow) div
  4910. _BytesPerUnit, Radix));
  4911. FOffSetDisplayWidth := Length(GetOffsetString((RowCount - 3) * FBytesPerRow))
  4912. + 1;
  4913. end;
  4914. if FGutterWidth = -1 then
  4915. DoSetCellWidth(0, FOffSetDisplayWidth * FCharWidth + 20 + 1)
  4916. else
  4917. DoSetCellWidth(0, FGutterWidth);
  4918. end;
  4919. function TCustomMPHexEditor.Seek(const aOffset, aOrigin: integer): integer;
  4920. var
  4921. LIntPos: integer;
  4922. begin
  4923. Result := -1;
  4924. LIntPos := GetCursorPos;
  4925. case aOrigin of
  4926. soFromBeginning: LIntPos := aOffset;
  4927. soFromCurrent: LIntPos := GetCursorPos + aOffset;
  4928. soFromEnd: LIntPos := DataSize + aOffset - 1;
  4929. end;
  4930. if DataSize < 1 then
  4931. Exit;
  4932. LIntPos := Min(Max(0, LIntPos), DataSize - 1);
  4933. SelStart := LIntPos;
  4934. Result := LIntPos;
  4935. end;
  4936. procedure TCustomMPHexEditor.SetSwapNibbles(const Value: boolean);
  4937. begin
  4938. if integer(Value) <> FSwapNibbles then
  4939. begin
  4940. FSwapNibbles := integer(Value);
  4941. Invalidate;
  4942. end;
  4943. end;
  4944. function TCustomMPHexEditor.GetSwapNibbles: boolean;
  4945. begin
  4946. Result := boolean(FSwapNibbles);
  4947. end;
  4948. procedure TCustomMPHexEditor.SetColors(const Value: TMPHColors);
  4949. begin
  4950. FColors.Assign(Value);
  4951. end;
  4952. procedure TCustomMPHexEditor.SetCaretKind(const Value: TMPHCaretKind);
  4953. begin
  4954. if FCaretKind <> Value then
  4955. begin
  4956. FCaretKind := Value;
  4957. if Focused then
  4958. begin
  4959. CreateCaretGlyph;
  4960. IntSetCaretPos(-50, -50, -1);
  4961. Invalidate;
  4962. end;
  4963. end;
  4964. end;
  4965. procedure TCustomMPHexEditor.SetFocusFrame(const Value: boolean);
  4966. begin
  4967. if FFocusFrame <> Value then
  4968. begin
  4969. FFocusFrame := Value;
  4970. Invalidate;
  4971. end;
  4972. end;
  4973. procedure TCustomMPHexEditor.SetMaskChar(const Value: char);
  4974. begin
  4975. if FReplaceUnprintableCharsBy <> Value then
  4976. begin
  4977. FReplaceUnprintableCharsBy := Value;
  4978. Invalidate;
  4979. end;
  4980. end;
  4981. procedure TCustomMPHexEditor.SetAsText(const Value: string);
  4982. var
  4983. LpszBuffer: PChar;
  4984. begin
  4985. if DataSize > 0 then
  4986. begin
  4987. // alles selektieren
  4988. SelStart := 0;
  4989. SelEnd := DataSize - 1;
  4990. end;
  4991. // do translation (thanks to philippe chessa) dec 17 98
  4992. GetMem(LpszBuffer, Length(Value));
  4993. try
  4994. Move(Value[1], LpszBuffer^, Length(Value));
  4995. TranslateBufferFromANSI(FTranslation, @Value[1], LpszBuffer, Length(Value));
  4996. ReplaceSelection(LpszBuffer, Length(Value));
  4997. finally
  4998. FreeMem(LpszBuffer);
  4999. end;
  5000. end;
  5001. procedure TCustomMPHexEditor.SetAsHex(const Value: string);
  5002. var
  5003. LpszBuffer: PChar;
  5004. LIntAmount: integer;
  5005. begin
  5006. if DataSize > 0 then
  5007. begin
  5008. // alles selektieren
  5009. SelStart := 0;
  5010. SelEnd := DataSize - 1;
  5011. end;
  5012. GetMem(LpszBuffer, Length(Value));
  5013. try
  5014. ConvertHexToBin(@Value[1], LpszBuffer, Length(Value), SwapNibbles,
  5015. LIntAmount);
  5016. ReplaceSelection(LpszBuffer, LIntAmount);
  5017. finally
  5018. FreeMem(LpszBuffer);
  5019. end;
  5020. end;
  5021. function TCustomMPHexEditor.GetAsText: string;
  5022. begin
  5023. if DataSize < 1 then
  5024. Result := ''
  5025. else
  5026. begin
  5027. SetLength(Result, DataSize);
  5028. ReadBuffer(Result[1], 0, DataSize);
  5029. end;
  5030. end;
  5031. function TCustomMPHexEditor.GetAsHex: string;
  5032. begin
  5033. Result := FDataStorage.GetAsHex(0, DataSize, SwapNibbles)
  5034. end;
  5035. function TCustomMPHexEditor.GetSelectionAsHex: string;
  5036. begin
  5037. if (DataSize < 1) or (SelCount < 1) then
  5038. Result := ''
  5039. else
  5040. Result := FDataStorage.GetAsHex(Min(SelStart, SelEnd), SelCount,
  5041. SwapNibbles);
  5042. end;
  5043. function TCustomMPHexEditor.GetInsertMode: boolean;
  5044. begin
  5045. Result := FInsertModeOn and IsInsertModePossible;
  5046. end;
  5047. procedure TCustomMPHexEditor.SetAllowInsertMode(const Value: boolean);
  5048. begin
  5049. if not Value then
  5050. begin
  5051. if FInsertModeOn then
  5052. InsertMode := False;
  5053. end;
  5054. FAllowInsertMode := Value;
  5055. end;
  5056. procedure TCustomMPHexEditor.SetFixedFileSize(const Value: boolean);
  5057. begin
  5058. if Value <> FFixedFileSize then
  5059. begin
  5060. if Value then
  5061. InsertMode := False;
  5062. FFixedFileSize := Value;
  5063. end;
  5064. end;
  5065. procedure TCustomMPHexEditor.InternalErase(const KeyWasBackspace: boolean; const
  5066. UndoDesc: string = '');
  5067. var
  5068. LIntPos: integer;
  5069. LIntSavePos: integer;
  5070. LIntCount: integer;
  5071. begin
  5072. LIntPos := GetCursorPos div FBytesPerUnit * FBytesPerUnit;
  5073. LIntCount := FBytesPerUnit;
  5074. LIntSavePos := LIntPos;
  5075. if KeyWasBackspace then
  5076. begin // Delete previous byte(s)
  5077. if InsertMode and (SelCount = 0) then
  5078. begin
  5079. LIntPos := GetPosAtCursor(Col, Row);
  5080. if (LIntPos = DataSize) and ((DataSize mod FBytesPerUnit) <> 0) then
  5081. LIntCount := 1
  5082. else
  5083. begin
  5084. LIntPos := LIntPos div FBytesPerUnit * FBytesPerUnit;
  5085. LIntCount := FBytesPerUnit;
  5086. end;
  5087. end;
  5088. if LIntPos = 0 then
  5089. Exit; // Can't delete at offset -1
  5090. CreateUndo(ufKindByteRemoved, LIntPos - LIntCount, LIntCount,
  5091. 0, UndoDesc);
  5092. InternalDelete(LIntPos - LIntCount, LIntPos, Col, Row);
  5093. if LIntSavePos = LIntPos then
  5094. Seek(LIntPos - LIntCount, soFromBeginning) // Move caret
  5095. else
  5096. begin
  5097. if (Col + 1) <= GetLastCharCol then
  5098. Col := Col + 1;
  5099. end;
  5100. Changed;
  5101. end
  5102. else
  5103. begin // Delete next byte
  5104. if LIntPos >= DataSize then
  5105. Exit; // Cant delete at EOF
  5106. while (LIntPos + LIntCount) > DataSize do
  5107. Dec(LIntCount);
  5108. CreateUndo(ufKindByteRemoved, LIntPos, LIntCount, 0, UndoDesc);
  5109. InternalDelete(LIntPos, LIntPos + LIntCount, Col, Row);
  5110. Changed;
  5111. end;
  5112. end;
  5113. procedure TCustomMPHexEditor.WMGetDlgCode(var Msg: TWMGetDlgCode);
  5114. begin
  5115. inherited;
  5116. Msg.Result := Msg.Result or DLGC_WANTARROWS or DLGC_WANTCHARS or
  5117. DLGC_WANTALLKEYS;
  5118. if FWantTabs then
  5119. Msg.Result := Msg.Result or DLGC_WANTTAB
  5120. else
  5121. Msg.Result := Msg.Result and not DLGC_WANTTAB;
  5122. end;
  5123. procedure TCustomMPHexEditor.CMFontChanged(var Message: TMessage);
  5124. begin
  5125. inherited;
  5126. if HandleAllocated then
  5127. begin
  5128. AdjustMetrics;
  5129. if Focused then
  5130. begin
  5131. CreateCaretGlyph;
  5132. end;
  5133. end;
  5134. end;
  5135. procedure TCustomMPHexEditor.SetWantTabs(const Value: boolean);
  5136. begin
  5137. FWantTabs := Value;
  5138. end;
  5139. procedure TCustomMPHexEditor.SetReadOnlyView(const Value: boolean);
  5140. begin
  5141. FReadOnlyView := Value;
  5142. if (FCaretKind = ckAuto) and Focused then
  5143. CreateCaretGlyph;
  5144. end;
  5145. procedure TCustomMPHexEditor.SetHideSelection(const Value: boolean);
  5146. begin
  5147. if FHideSelection <> Value then
  5148. begin
  5149. FHideSelection := Value;
  5150. if (not Focused) and (GetSelCount > 0) then
  5151. Invalidate;
  5152. end;
  5153. end;
  5154. procedure TCustomMPHexEditor.SetGraySelectionIfNotFocused(const Value: boolean);
  5155. begin
  5156. if FGraySelOnLostFocus <> Value then
  5157. begin
  5158. FGraySelOnLostFocus := Value;
  5159. if (not Focused) and (GetSelCount > 0) and (not FHideSelection) then
  5160. Invalidate;
  5161. end;
  5162. end;
  5163. function TCustomMPHexEditor.CalcColCount: integer;
  5164. begin
  5165. if FUnicodeCharacters then
  5166. Result := (FBytesPerRow * 2) + (FBytesPerRow div 2) + 1 + GRID_FIXED
  5167. else
  5168. Result := FBytesPerRow * 3 + 1 + GRID_FIXED;
  5169. end;
  5170. function TCustomMPHexEditor.GetLastCharCol: integer;
  5171. begin
  5172. Result := ColCount - 1;
  5173. end;
  5174. function TCustomMPHexEditor.GetTopLeftPosition(var oInCharField: boolean):
  5175. integer;
  5176. begin
  5177. Result := GetPosAtCursor(Max(LeftCol, GRID_FIXED), TopRow);
  5178. oInCharField := InCharField;
  5179. end;
  5180. procedure TCustomMPHexEditor.SetTopLeftPosition(const aPosition: integer; const
  5181. aInCharField: boolean);
  5182. begin
  5183. with GetCursorAtPos(aPosition, aInCharField) do
  5184. begin
  5185. TopRow := y;
  5186. LeftCol := x;
  5187. end;
  5188. end;
  5189. function TCustomMPHexEditor.GetPropColCount: integer;
  5190. begin
  5191. Result := inherited ColCount;
  5192. end;
  5193. function TCustomMPHexEditor.GetPropRowCount: integer;
  5194. begin
  5195. Result := inherited RowCount;
  5196. end;
  5197. function TCustomMPHexEditor.ShowDragCell(const X, Y: integer): integer;
  5198. var
  5199. LRctCell: TRect;
  5200. LIntDragPos,
  5201. LIntMouseX,
  5202. LIntMouseY: integer;
  5203. begin
  5204. with MouseCoord(X, Y) do
  5205. begin
  5206. LIntMouseX := X;
  5207. LIntMouseY := Y;
  5208. if X < GRID_FIXED then
  5209. X := GRID_FIXED;
  5210. if Y >= RowCount then
  5211. Y := RowCount - 1;
  5212. if Y < GRID_FIXED then
  5213. Y := GRID_FIXED;
  5214. LIntDragPos := GetPosAtCursor(X, Y)
  5215. end;
  5216. if LIntDragPos < 0 then
  5217. LIntDragPos := 0;
  5218. if LIntDragPos > DataSize then
  5219. LIntDragPos := DataSize;
  5220. if IsSelected(LIntDragPos) then
  5221. LIntDragPos := Min(SelStart, SelEnd);
  5222. CheckUnit(LIntDragPos);
  5223. Result := LIntDragPos;
  5224. FShowDrag := True;
  5225. if (LIntMouseY <= TopRow) and (LIntMouseY > GRID_FIXED) then
  5226. begin
  5227. // nach oben scrollen
  5228. TopRow := TopRow - 1;
  5229. end
  5230. else if (LIntMouseY >= (TopRow + VisibleRowCount - 1)) and (LIntMouseY <
  5231. Pred(RowCount)) then
  5232. begin
  5233. // nach unten scrollen
  5234. TopRow := TopRow + 1;
  5235. end;
  5236. if (LIntMouseX <= LeftCol) and (LIntMouseX > GRID_FIXED) then
  5237. begin
  5238. // nach links scrollen
  5239. LeftCol := LeftCol - 1;
  5240. end
  5241. else if (LIntMouseX >= (LeftCol + VisibleColCount - 1)) and
  5242. (LIntMouseX < GetLastCharCol) then
  5243. begin
  5244. // nach unten scrollen
  5245. LeftCol := LeftCol + 1;
  5246. end;
  5247. with GetCursorAtPos(LIntDragPos, FPosInCharField) do
  5248. begin
  5249. if (x = FDropCol) and (y = FDropRow) then
  5250. Exit;
  5251. LRctCell := CellRect(FDropCol, FDropRow);
  5252. FDropCol := x;
  5253. FDropRow := y;
  5254. InvalidateRect(Handle, @LRctCell, True);
  5255. LRctCell := CellRect(X, Y);
  5256. InvalidateRect(Handle, @LRctCell, True);
  5257. end;
  5258. end;
  5259. procedure TCustomMPHexEditor.HideDragCell;
  5260. begin
  5261. FShowDrag := False;
  5262. Invalidate;
  5263. end;
  5264. procedure TCustomMPHexEditor.CombineUndo(const aCount: integer; const sDesc:
  5265. string = '');
  5266. begin
  5267. CreateUndo(ufKindCombined, 0, aCount, 0, sDesc);
  5268. end;
  5269. function TCustomMPHexEditor.GetMouseOverSelection: boolean;
  5270. var
  5271. LPntMouse: TPoint;
  5272. begin
  5273. Windows.GetCursorPos(LPntMouse);
  5274. LPntMouse := ScreenToClient(LPntMouse);
  5275. Result := CursorOverSelection(LPntMouse.x, LPntMouse.y);
  5276. end;
  5277. function TCustomMPHexEditor.CursorOverSelection(const X, Y: integer): boolean;
  5278. var
  5279. LIntPos: integer;
  5280. LBoolInCharField: boolean;
  5281. begin
  5282. Result := False;
  5283. if (SelCount = 0) or (DataSize = 0) then
  5284. Exit;
  5285. LBoolInCharField := FPosInCharField;
  5286. with MouseCoord(x, y) do
  5287. begin
  5288. if (x < GRID_FIXED) or (y < GRID_FIXED) then
  5289. Exit;
  5290. LIntPos := GetPosAtCursor(X, Y);
  5291. FPosInCharField := (LBoolInCharField);
  5292. if (LIntPos < 0) or (LIntPos >= DataSize) then
  5293. Exit;
  5294. end;
  5295. Result := IsSelected(LIntPos);
  5296. end;
  5297. function TCustomMPHexEditor.MouseOverFixed(const X, Y: integer): boolean;
  5298. begin
  5299. with MouseCoord(x, y) do
  5300. Result := (x < GRID_FIXED) or (y < GRID_FIXED);
  5301. end;
  5302. procedure TCustomMPHexEditor.MouseMove(Shift: TShiftState; X, Y: integer);
  5303. var
  5304. LgrcCoords: TGridCoord;
  5305. begin
  5306. if Shift = [ssLeft] then
  5307. LgrcCoords := CheckMouseCoord(X, Y);
  5308. inherited MouseMove(Shift, x, y);
  5309. if FMouseUpCanResetSel then
  5310. begin
  5311. FMouseUpCanResetSel := (LgrcCoords.x = FMouseDownCol) and
  5312. (LgrcCoords.y = FMouseDownRow);
  5313. end;
  5314. if (Shift = []) and (CursorOverSelection(X, Y) or MouseOverFixed(X, Y)) then
  5315. Cursor := crArrow
  5316. else
  5317. Cursor := crIBeam;
  5318. end;
  5319. procedure TCustomMPHexEditor.WMTimer(var Msg: TWMTimer);
  5320. var
  5321. LPtMouse: TPoint;
  5322. LgrcCoord: TGridCoord;
  5323. begin
  5324. if FGridState <> gsSelecting then
  5325. Exit;
  5326. Windows.GetCursorPos(LPtMouse);
  5327. LPtMouse := ScreenToClient(LPtMouse);
  5328. LgrcCoord := CheckMouseCoord(LPtMouse.X, LPtMouse.Y);
  5329. if (LGrcCoord.X <> -1) and (LGrcCoord.Y <> -1) then
  5330. inherited;
  5331. end;
  5332. function TCustomMPHexEditor.CheckMouseCoord(var X, Y: integer): TGridCoord;
  5333. var
  5334. LRctCell: TRect;
  5335. begin
  5336. Result := MouseCoord(X, Y);
  5337. if FInsertModeOn then
  5338. begin
  5339. // use the following cell if the mouse is over the second half of the cell
  5340. LRctCell := CellRect(Result.X, Result.Y);
  5341. if (LRctCell.Left + (FCharWidth div 2)) <= X then
  5342. begin
  5343. if not (Result.X in [GetLastCharCol, FBytesPerRowDup + GRID_FIXED - 1])
  5344. then
  5345. begin
  5346. X := LRctCell.Right + 1;
  5347. Inc(Result.X);
  5348. LRctCell := CellRect(Result.X, Result.Y);
  5349. end;
  5350. end;
  5351. if (Result.X = GetLastCharCol) then
  5352. begin
  5353. if (X - LRctCell.Left) > (FCharWidth div 2) then
  5354. begin
  5355. Y := Y + RowHeight;
  5356. Result.Y := Result.Y + 1;
  5357. Result.X := FBytesPerRowDup + 1 + GRID_FIXED;
  5358. X := CellRect(Result.X, Result.Y - 1).Left;
  5359. //Dec(X, FCharWidth * FBytesPerRow);
  5360. end;
  5361. end
  5362. else if Result.X = (FBytesPerRowDup + GRID_FIXED - 1) then
  5363. begin
  5364. if (X - LRctCell.Left) > (FCharWidth div 2) then
  5365. begin
  5366. Y := Y + RowHeight;
  5367. Result.Y := Result.Y + 1;
  5368. Result.X := GRID_FIXED;
  5369. X := CellRect(Result.X, Result.Y - 1).Left;
  5370. //Dec(X, FCharWidth * FBytesPerRow);
  5371. end;
  5372. end;
  5373. end;
  5374. end;
  5375. procedure TCustomMPHexEditor.MouseUp(Button: TMouseButton; Shift: TShiftState;
  5376. X, Y: integer);
  5377. begin
  5378. CheckMouseCoord(X, Y);
  5379. inherited;
  5380. if FMouseUpCanResetSel then
  5381. begin
  5382. FMouseUpCanResetSel := False;
  5383. ResetSelection(True);
  5384. with MouseCoord(x, y) do
  5385. MoveColRow(x, y, True, True);
  5386. end;
  5387. if FShowDrag then
  5388. HideDragCell;
  5389. end;
  5390. procedure TCustomMPHexEditor.AdjustBookmarks(const From, Offset: integer);
  5391. var
  5392. LIntLoop: integer;
  5393. LBoolChanged: boolean;
  5394. begin
  5395. LBoolChanged := False;
  5396. if From >= 0 then
  5397. for LIntLoop := 0 to 9 do
  5398. with FBookmarks[LIntLoop] do
  5399. if mPosition >= From then
  5400. begin
  5401. LBoolChanged := True;
  5402. Inc(mPosition, Offset);
  5403. if mPosition > DataSize then
  5404. mPosition := -1;
  5405. end;
  5406. if LBoolChanged then
  5407. BookMarkChanged;
  5408. end;
  5409. procedure TCustomMPHexEditor.IntSetCaretPos(const X, Y, aCol: integer);
  5410. begin
  5411. if Focused then
  5412. begin
  5413. if aCol <> -1 then
  5414. begin
  5415. FPosInCharField := (aCol > (GRID_FIXED + FBytesPerRowDup));
  5416. if FLastPosInCharField <> FPosInCharField then
  5417. begin
  5418. FLastPosInCharField := FPosInCharField;
  5419. Invalidate;
  5420. end;
  5421. end;
  5422. SetCaretPos(X, Y);
  5423. end;
  5424. end;
  5425. procedure TCustomMPHexEditor.TruncMaxPosition(var DataPos: integer);
  5426. begin
  5427. if DataPos >= DataSize then
  5428. begin
  5429. DataPos := DataSize - 1;
  5430. if InsertMode then
  5431. DataPos := DataSize;
  5432. end;
  5433. end;
  5434. function TCustomMPHexEditor.GetCurrentValue: integer;
  5435. var
  5436. LIntPos: integer;
  5437. begin
  5438. Result := -1;
  5439. LIntPos := GetPosAtCursor(Col, Row);
  5440. if (LIntPos >= DataSize) or (LIntPos < 0) then
  5441. Exit;
  5442. Result := Data[LIntPos]
  5443. end;
  5444. procedure TCustomMPHexEditor.SetInsertMode(const Value: boolean);
  5445. var
  5446. LIntPos: integer;
  5447. begin
  5448. if Value = FInsertModeOn then
  5449. Exit;
  5450. if IsInsertModePossible then
  5451. begin
  5452. FInsertModeOn := Value;
  5453. if (FCaretKind = ckAuto) and Focused then
  5454. CreateCaretGlyph;
  5455. if DataSize < 1 then
  5456. Exit;
  5457. if not FInsertModeOn then
  5458. begin
  5459. if ((DataSize mod FBytesPerRow) = 0) and (DataSize > 0) then
  5460. RowCount := RowCount - 1;
  5461. LIntPos := GetPosAtCursor(Col, Row);
  5462. if LIntPos = DataSize then
  5463. SelStart := DataSize - 1;
  5464. end
  5465. else
  5466. begin
  5467. if ((DataSize mod FBytesPerRow) = 0) and (DataSize > 0) then
  5468. RowCount := RowCount + 1;
  5469. end;
  5470. FModifiedBytes.Size := 0;
  5471. Invalidate;
  5472. end;
  5473. end;
  5474. function TCustomMPHexEditor.GetModified: boolean;
  5475. begin
  5476. Result := FModified and ((DataSize > 0) or FileExists(FileName));
  5477. end;
  5478. procedure TCustomMPHexEditor.SetSelection(DataPos, StartPos, EndPos:
  5479. integer);
  5480. begin
  5481. //CheckSelectUnit(StartPos, EndPos);
  5482. FSelEnd := Max(-1, Min(EndPos, DataSize - 1));
  5483. FSelPosition := Max(-1, Min(DataPos, DataSize - 1));
  5484. FSelStart := Max(-1, Min(StartPos, DataSize - 1));
  5485. end;
  5486. procedure TCustomMPHexEditor.Resize;
  5487. begin
  5488. PostMessage(Handle, CM_INTUPDATECARET, 7, 7);
  5489. inherited;
  5490. end;
  5491. procedure TCustomMPHexEditor.WrongKey;
  5492. begin
  5493. if Assigned(FOnInvalidKey) then
  5494. FOnInvalidKey(self);
  5495. end;
  5496. procedure TCustomMPHexEditor.TopLeftChanged;
  5497. begin
  5498. CheckSetCaret;
  5499. if Assigned(FOnTopLeftChanged) then
  5500. FOnTopLeftChanged(self);
  5501. end;
  5502. function TCustomMPHexEditor.GetOffsetString(const Position: cardinal): string;
  5503. begin
  5504. Result := '';
  5505. if Assigned(FOnGetOffsetText) and (not FOffsetHandler) then
  5506. begin
  5507. FOffsetHandler := True;
  5508. try
  5509. FIsMaxOffset := False;
  5510. FOnGetOffsetText(self, Position, Result)
  5511. finally
  5512. FOffsetHandler := False;
  5513. end;
  5514. end
  5515. else
  5516. begin
  5517. with FOffsetFormat do
  5518. begin
  5519. if Format <> '' then
  5520. begin
  5521. if (MinWidth <> 0) or (Position <> 0) then
  5522. begin
  5523. if FHexLowercase then
  5524. Result := LowerCase(IntToRadixLen(Position div _BytesPerUnit, Radix,
  5525. MinWidth))
  5526. else
  5527. Result := Uppercase(IntToRadixLen(Position div _BytesPerUnit, Radix,
  5528. MinWidth));
  5529. end;
  5530. Result := Prefix + Result + Suffix;
  5531. end;
  5532. end;
  5533. end;
  5534. end;
  5535. function TCustomMPHexEditor.GetAnyOffsetString(const Position: integer): string;
  5536. begin
  5537. if FOffsetFormat.Format = '' then
  5538. Result := IntToRadix(Position, 16)
  5539. else
  5540. Result := GetOffsetString(Position);
  5541. end;
  5542. function TCustomMPHexEditor.RowHeight: integer;
  5543. begin
  5544. Result := DefaultRowHeight;
  5545. end;
  5546. function TCustomMPHexEditor.GetBookmark(Index: byte): TMPHBookmark;
  5547. begin
  5548. if Index > 9 then
  5549. raise EMPHexEditor.Create(ERR_INVALID_BOOKMARK);
  5550. Result := FBookmarks[Index];
  5551. end;
  5552. procedure TCustomMPHexEditor.SetBookmark(Index: byte; const Value:
  5553. TMPHBookmark);
  5554. begin
  5555. SetBookmarkVals(Index, Value.mPosition, Value.mInCharField);
  5556. end;
  5557. procedure TCustomMPHexEditor.SetBookmarkVals(const Index: byte; const Position:
  5558. integer; const InCharField: boolean);
  5559. begin
  5560. if Index > 9 then
  5561. raise EMPHexEditor.Create(ERR_INVALID_BOOKMARK);
  5562. if (FBookmarks[Index].mPosition <> Position) or
  5563. (FBookmarks[Index].mInCharField <> InCharField) then
  5564. begin
  5565. FBookmarks[Index].mPosition := Position;
  5566. FBookmarks[Index].mInCharField := InCharField;
  5567. Invalidate;
  5568. end
  5569. else
  5570. begin
  5571. FBookmarks[Index].mPosition := -1;
  5572. FBookmarks[Index].mInCharField := InCharField;
  5573. Invalidate;
  5574. end;
  5575. BookmarkChanged;
  5576. end;
  5577. {.$DEFINE TESTCOLOR}// check for unneeded drawings
  5578. type
  5579. TestColor = TColor;
  5580. procedure TCustomMPHexEditor.Paint;
  5581. type
  5582. TKindOfCell = (kocData, kocRuler, kocOffset, kocEmpty);
  5583. var
  5584. DrawInfo: TGridDrawInfo;
  5585. LIntCurCol, LIntCurRow: longint;
  5586. LRctWhere: TRect;
  5587. LBoolOddCol: boolean;
  5588. LBoolChanged: boolean;
  5589. LIntDataPos, LIntDataSize: integer;
  5590. LWStrOutput: WideString;
  5591. LColTextColor, LColTextBackColor, LColBackColor: TColor;
  5592. LIntPenWidthSave: integer;
  5593. LrecSize: TSize;
  5594. LBoolDraw: Boolean;
  5595. LBoolFocused: boolean;
  5596. LRect2: TRect;
  5597. LIntLastCol: integer;
  5598. // get the width of a wide text
  5599. function GetTextWidthW: Integer;
  5600. begin
  5601. GetTextExtentPoint32W(Canvas.Handle, PWideChar(LWStrOutput),
  5602. Length(LWStrOutput), LrecSize);
  5603. Result := LRecSize.cx;
  5604. end;
  5605. // render an offset/ruler/fixed cell
  5606. procedure _TextOut;
  5607. begin
  5608. with Canvas, LRctWhere do
  5609. begin
  5610. Brush.Color := TestColor(LColBackColor);
  5611. Font.Color := LColTextColor;
  5612. SetBKColor(Handle, ColorToRGB(TestColor(LColTextBackColor)));
  5613. LRect2 := LRctWhere; //Rect(Left, Top, Left + FCharWidth, Bottom);
  5614. LRect2.Right := Left + FCharWidth;
  5615. //SetTextColor(Handle, ColorToRGB(LColTextColor));
  5616. LBoolDraw := True;
  5617. if Assigned(FOnDrawCell) then
  5618. begin
  5619. if LIntCurCol = 0 then
  5620. FOnDrawCell(self, Canvas, LIntCurCol, LIntCurRow, LWStrOutput,
  5621. LRctWhere, LBoolDraw)
  5622. else
  5623. FOnDrawCell(self, Canvas, LIntCurCol, LIntCurRow, LWStrOutput, LRect2,
  5624. LBoolDraw)
  5625. end;
  5626. if LBoolDraw then
  5627. begin
  5628. FillRect(LRctWhere);
  5629. if LIntCurCol = 0 then
  5630. ExtTextOutW(Handle, Right - GetTextWidthW - 4, Top,
  5631. ETO_CLIPPED or ETO_OPAQUE, @LRctWhere, PWideChar(LWStrOutput),
  5632. Length(LWStrOutput), nil)
  5633. else
  5634. ExtTextOutW(Handle, Left, Top,
  5635. ETO_CLIPPED or ETO_OPAQUE, @LRect2, PWideChar(LWStrOutput),
  5636. Length(LWStrOutput), nil);
  5637. end
  5638. else
  5639. LBoolDraw := True;
  5640. end;
  5641. end;
  5642. // render a data cell
  5643. procedure _TextOutData;
  5644. begin
  5645. with Canvas, LRctWhere do
  5646. begin
  5647. Brush.Color := TestColor(LColBackColor);
  5648. Font.Color := LColTextColor;
  5649. SetBKColor(Handle, ColorToRGB(TestColor(LColTextBackColor)));
  5650. LRect2 := LRctWhere; //Rect(Left, Top, Left + FCharWidth, Bottom);
  5651. LRect2.Right := Left + FCharWidth;
  5652. //SetTextColor(Handle, ColorToRGB(LColTextColor));
  5653. LBoolDraw := True;
  5654. if Assigned(FOnDrawCell) then
  5655. begin
  5656. FOnDrawCell(self, Canvas, LIntCurCol, LIntCurRow, LWStrOutput, LRect2,
  5657. LBoolDraw)
  5658. end;
  5659. if LBoolDraw then
  5660. begin
  5661. FillRect(LRctWhere);
  5662. ExtTextOutW(Handle, Left, Top,
  5663. ETO_CLIPPED or ETO_OPAQUE, @LRect2, PWideChar(LWStrOutput),
  5664. Length(LWStrOutput), nil);
  5665. end
  5666. else
  5667. LBoolDraw := True;
  5668. if FShowDrag and (LIntCurCol = FDropCol) and (LIntCurRow = FDropRow) then
  5669. begin
  5670. LIntPenWidthSave := Pen.Width;
  5671. try
  5672. Pen.Width := 2;
  5673. Pen.Color := LColTextColor;
  5674. MoveTo(Left + 1, Top + 1);
  5675. LineTo(Left + 1, Bottom - 1)
  5676. finally
  5677. Pen.Width := LIntPenWidthSave;
  5678. end;
  5679. end
  5680. end;
  5681. end;
  5682. // draw an offset cell
  5683. procedure DrawOffsetCell;
  5684. var
  5685. LIntLoop: integer;
  5686. begin
  5687. if (LIntCurRow = Row) then
  5688. begin
  5689. LColBackColor := FColors.CurrentOffsetBackground;
  5690. LColTextColor := FColors.CurrentOffset;
  5691. end
  5692. else
  5693. begin
  5694. LColBackColor := FColors.OffsetBackground;
  5695. LColTextColor := Colors.Offset;
  5696. end;
  5697. LColTextBackColor := LColBackColor;
  5698. (* text ausgeben *)
  5699. LWStrOutput := GetOffsetString((LIntCurRow - GRID_FIXED) * FBytesPerRow);
  5700. _TextOut;
  5701. (* auf bookmark prüfen *)
  5702. for LIntLoop := 0 to 9 do
  5703. with FBookmarks[lIntLoop] do
  5704. if (mPosition > -1) and ((mPosition div FBytesPerRow) = (LIntCurRow -
  5705. GRID_FIXED)) then
  5706. with LRctWhere do
  5707. FBookmarkImageList.Draw(Canvas, Left + 3, ((Bottom - Top - 10) div 2)
  5708. + Top, lIntLoop + (10 * integer(mInCharField)));
  5709. end;
  5710. // draw a ruler cell
  5711. procedure DrawRulerCell;
  5712. begin
  5713. if LIntCurCol <> (GRID_FIXED + FBytesPerRowDup) then
  5714. begin
  5715. if LIntCurCol > (GRID_FIXED + FBytesPerRowDup) then
  5716. begin
  5717. LIntDataPos := (LIntCurCol - (GRID_FIXED + FBytesPerRowDup + 1));
  5718. LWStrOutput := FRulerCharString[LIntDataPos + 1];
  5719. end
  5720. else
  5721. LWStrOutput := FRulerString[LIntCurCol - GRID_FIXED + 1];
  5722. end
  5723. else
  5724. LWStrOutput := ' ';
  5725. LColBackColor := FColors.OffsetBackGround;
  5726. if Col = LIntCurCol then
  5727. begin
  5728. LColTextBackColor := FColors.CurrentOffsetBackGround;
  5729. LColTextColor := FColors.CurrentOffset;
  5730. end
  5731. else
  5732. begin
  5733. LColTextBackColor := FColors.OffsetBackGround;
  5734. LColTextColor := FColors.Offset;
  5735. end;
  5736. _TextOut;
  5737. end;
  5738. // draw a hex/char cell
  5739. procedure DrawDataCell(const bIsCharCell, bIsCurrentField: boolean);
  5740. begin
  5741. (*// caret setzen
  5742. if (LIntCurRow = Row) and (LIntCurCol = Col) then
  5743. IntSetCaretPos(LRctWhere.Left, LRctWhere.Top);*)
  5744. LIntDataPos := GetPosAtCursor(LIntCurCol, LIntCurRow);
  5745. FDrawDataPosition := LIntDataPos;
  5746. if bIsCurrentField and (LIntCurCol < LIntLastCol) and
  5747. (LIntCurCol <> FIntLastHexCol) then
  5748. LColBackColor := FColors.FActiveFieldBackground
  5749. else
  5750. LColBackColor := FColors.FBackground;
  5751. // nicht zeichnen, falls keine daten
  5752. if (LIntDataPos < LIntDataSize) then
  5753. begin
  5754. if not bIsCharCell then
  5755. begin // partie hexadecimale
  5756. if ((LIntCurCol - GRID_FIXED) mod 2) = FSwapNibbles then
  5757. LWStrOutput := FHexChars[Data[LIntDataPos] shr 4]
  5758. else
  5759. LWStrOutput := FHexChars[Data[LIntDataPos] and 15]
  5760. end
  5761. else
  5762. begin
  5763. if FUnicodeCharacters then
  5764. begin
  5765. SetLength(LWStrOutput, 1);
  5766. LWStrOutput[1] := #0;
  5767. ReadBuffer(LWStrOutput[1], LIntDataPos, Min(2, LIntDataSize -
  5768. LIntDataPos));
  5769. if FUnicodeBigEndian then
  5770. SwapWideChar(LWStrOutput[1]);
  5771. if (LWStrOutput[1] < #256) and (Char(LWStrOutput[1]) in FMaskedChars)
  5772. then
  5773. LWStrOutput[1] := WideChar(FReplaceUnprintableCharsBy);
  5774. end
  5775. else
  5776. LWStrOutput := TranslateToAnsiChar(Data[LIntDataPos]);
  5777. end;
  5778. // testen ob byte geändert
  5779. LBoolChanged := (HasChanged(LIntDataPos)) or ((FUnicodeCharacters and
  5780. bIsCharCell) and HasChanged(LIntDataPos + 1));
  5781. LBoolOddCol := (((LIntCurCol - GRID_FIXED) div FBytesPerCol) mod 2) = 0;
  5782. if LBoolChanged then
  5783. begin
  5784. LColTextColor := FColors.FChangedText;
  5785. LColTextBackColor := FColors.FChangedBackground;
  5786. end
  5787. else
  5788. begin
  5789. if bIsCurrentField then
  5790. LColTextBackColor := FColors.FActiveFieldBackground
  5791. else
  5792. LColTextBackColor := FColors.FBackground;
  5793. if not FPosInCharField then
  5794. begin
  5795. if LBoolOddCol then
  5796. LColTextColor := Colors.FOddColumn
  5797. else
  5798. LColTextColor := Colors.FEvenColumn;
  5799. end
  5800. else
  5801. LColTextColor := Font.Color;
  5802. end;
  5803. if (FSelPosition <> -1) and IsSelected(LIntDataPos) then
  5804. begin
  5805. FIsDrawDataSelected := True;
  5806. if (not FHideSelection) or LBoolFocused then
  5807. begin
  5808. if (LIntCurCol < LIntLastCol) and (LIntCurCol <> FIntLastHexCol)
  5809. and (LIntDataPos <> Max(FSelStart, FSelEnd)) then
  5810. LColBackColor := Invert(LColBackColor);
  5811. LColTextBackColor := Invert(LColTextBackColor);
  5812. LColTextColor := Invert(LColTextColor);
  5813. if FGraySelOnLostFocus and (not LBoolFocused) then
  5814. begin
  5815. LColTextBackColor := FadeToGray(LColTextBackColor);
  5816. LColTextColor := FadeToGray(LColTextColor);
  5817. end;
  5818. end;
  5819. end
  5820. else
  5821. FIsDrawDataSelected := False
  5822. ;
  5823. _TextOutData
  5824. end;
  5825. // focus frame auf der anderen seite
  5826. if LBoolFocused then
  5827. begin
  5828. if not FPosInCharField then
  5829. begin
  5830. if (LIntCurRow = Row) then
  5831. begin
  5832. if not FUnicodeCharacters then
  5833. begin
  5834. if GetOtherFieldColCheck(Col) = (LIntCurCol - 1) then
  5835. with LRctWhere do
  5836. if FFocusFrame then
  5837. Canvas.DrawFocusRect(Rect(
  5838. CellRect(LIntCurCol-1, LIntCurRow).Left,
  5839. Top, Left + FCharWidth, Bottom - 1))
  5840. else
  5841. begin
  5842. Canvas.Pen.Color := FColors.CursorFrame;
  5843. Canvas.Brush.Style := bsClear;
  5844. Canvas.Rectangle(CellRect(LIntCurCol-3, LIntCurRow).Left, Top,
  5845. Left + FCharWidth, Bottom - 1);
  5846. end;
  5847. end
  5848. else if GetOtherFieldColCheck(Col) = (LIntCurCol - 3) then
  5849. with LRctWhere do
  5850. if FFocusFrame then
  5851. Canvas.DrawFocusRect(Rect(
  5852. CellRect(LIntCurCol-3, LIntCurRow).Left, Top,
  5853. Left + FCharWidth, Bottom - 1))
  5854. else
  5855. begin
  5856. Canvas.Pen.Color := FColors.CursorFrame;
  5857. Canvas.Brush.Style := bsClear;
  5858. Canvas.Rectangle(CellRect(LIntCurCol-3, LIntCurRow).Left, Top,
  5859. Left + FCharWidth, Bottom - 1);
  5860. end;
  5861. end;
  5862. end
  5863. else
  5864. begin
  5865. if (LIntCurRow = Row) and (GetOtherFieldColCheck(Col) = LIntCurCol) then
  5866. begin
  5867. with LRctWhere do
  5868. if FFocusFrame then
  5869. Canvas.DrawFocusRect(Rect(Left, Top, Left + FCharWidth, Bottom -
  5870. 1))
  5871. else
  5872. begin
  5873. Canvas.Pen.Color := FColors.CursorFrame;
  5874. Canvas.Brush.Style := bsClear;
  5875. Canvas.Rectangle(Left, Top, Left + FCharWidth, Bottom - 1);
  5876. end;
  5877. end;
  5878. end;
  5879. end
  5880. else
  5881. begin
  5882. // possibly draw a mark at the current position when not focused
  5883. if FShowPositionIfNotFocused and (LIntCurRow = Row) and (Col = LIntCurCol)
  5884. then
  5885. begin
  5886. with LRctWhere do
  5887. if FFocusFrame then
  5888. Canvas.DrawFocusRect(Rect(Left, Top, Left + FCharWidth, Bottom - 1))
  5889. else
  5890. begin
  5891. Canvas.Pen.Color := FColors.NonFocusCursorFrame;
  5892. Canvas.Brush.Style := bsClear;
  5893. Canvas.Rectangle(Left, Top, Left + FCharWidth, Bottom - 1);
  5894. end;
  5895. end;
  5896. end;
  5897. if FDrawGridLines and (LIntCurCol = LIntLastCol) then
  5898. with Canvas, LRctWhere do
  5899. begin
  5900. Pen.Color := FColors.FGrid;
  5901. MoveTo(Right - 1, Top);
  5902. LineTo(Right - 1, Bottom - 1);
  5903. end;
  5904. end;
  5905. // draw
  5906. procedure DrawCells(ACol, ARow: longint; StartX, StartY, StopX, StopY:
  5907. integer;
  5908. Kind: TKindOfCell);
  5909. begin
  5910. LIntCurRow := ARow;
  5911. LRctWhere.Top := StartY;
  5912. while (LRctWhere.Top < StopY) and (LIntCurRow < RowCount) do
  5913. begin
  5914. LIntCurCol := ACol;
  5915. LRctWhere.Left := StartX;
  5916. LRctWhere.Bottom := LRctWhere.Top + RowHeights[LIntCurRow];
  5917. while (LRctWhere.Left < StopX) and (LIntCurCol <= LIntLastCol) do
  5918. begin
  5919. LRctWhere.Right := LRctWhere.Left + ColWidths[LIntCurCol];
  5920. if (LRctWhere.Right > LRctWhere.Left) (*and RectVisible(Canvas.Handle,
  5921. LRctWhere) slows down, removed *)then
  5922. begin
  5923. case Kind of
  5924. kocData:
  5925. begin
  5926. if LIntCurCol < (GRID_FIXED + FBytesPerRowDup) then
  5927. DrawDataCell(False, not FLastPosInCharField)
  5928. else if LIntCurCol > (GRID_FIXED + FBytesPerRowDup) then
  5929. DrawDataCell(True, FLastPosInCharField)
  5930. else if FDrawGridLines then
  5931. with Canvas do
  5932. begin
  5933. Pen.Color := FColors.FGrid;
  5934. MoveTo(LRctWhere.Left, LRctWhere.Top);
  5935. LineTo(LRctWhere.Left, LRctWhere.Bottom - 1);
  5936. end;
  5937. if FDrawGridLines then
  5938. with Canvas do
  5939. begin
  5940. Pen.Color := FColors.FGrid;
  5941. MoveTo(LRctWhere.Left, LRctWhere.Bottom - 1);
  5942. LineTo(LRctWhere.Right, LRctWhere.Bottom - 1);
  5943. end;
  5944. end;
  5945. kocEmpty:
  5946. begin
  5947. FDrawDataPosition := -1;
  5948. LColTextBackColor := FColors.OffsetBackGround;
  5949. LColTextColor := FColors.Offset;
  5950. LWStrOutput := '';
  5951. _TextOut;
  5952. end;
  5953. kocRuler:
  5954. begin
  5955. FDrawDataPosition := -1;
  5956. DrawRulerCell;
  5957. end;
  5958. kocOffset:
  5959. begin
  5960. FDrawDataPosition := -1;
  5961. if LIntCurCol = 1 then
  5962. begin
  5963. if FDrawGridLines then
  5964. with Canvas do
  5965. begin
  5966. Pen.Color := FColors.FGrid;
  5967. MoveTo(LRctWhere.Left, LRctWhere.Bottom - 1);
  5968. LineTo(LRctWhere.Right, LRctWhere.Bottom - 1);
  5969. end;
  5970. end
  5971. else
  5972. DrawOffsetCell;
  5973. end;
  5974. end;
  5975. end;
  5976. LRctWhere.Left := LRctWhere.Right;
  5977. Inc(LIntCurCol);
  5978. end;
  5979. LRctWhere.Top := LRctWhere.Bottom;
  5980. Inc(LIntCurRow);
  5981. end;
  5982. end;
  5983. var
  5984. LIntTop: integer;
  5985. begin
  5986. {$IFDEF DELPHI6UP}
  5987. if UseRightToLeftAlignment then
  5988. ChangeGridOrientation(True);
  5989. {$ENDIF}
  5990. CalcDrawInfo(DrawInfo);
  5991. LBoolFocused := Focused;
  5992. LIntDataSize := DataSize;
  5993. LIntLastCol := GetLastCharCol;
  5994. with DrawInfo do
  5995. begin
  5996. if FShowRuler then
  5997. begin
  5998. // oben links, fixed
  5999. DrawCells(0, 0, 0, 0, Horz.FixedBoundary, Vert.FixedBoundary, kocEmpty);
  6000. // oben, fixed
  6001. DrawCells(LeftCol, 0, Horz.FixedBoundary, 0, Horz.GridBoundary,
  6002. Vert.FixedBoundary, kocRuler);
  6003. end;
  6004. // links, fixed
  6005. DrawCells(0, TopRow, 0, Vert.FixedBoundary, Horz.FixedBoundary,
  6006. Vert.GridBoundary, kocOffset);
  6007. // daten
  6008. DrawCells(LeftCol, TopRow, Horz.FixedBoundary, Vert.FixedBoundary,
  6009. Horz.GridBoundary, Vert.GridBoundary, kocData);
  6010. // paint unoccupied space on the right
  6011. if Horz.GridBoundary < Horz.GridExtent then
  6012. begin
  6013. Canvas.Brush.Color := TestColor(Color);
  6014. Canvas.FillRect(Rect(Horz.GridBoundary, 0, Horz.GridExtent,
  6015. Vert.GridBoundary));
  6016. // fixed (ruler)
  6017. Canvas.Brush.Color := TestColor(FColors.OffsetBackGround);
  6018. Canvas.FillRect(Rect(Horz.GridBoundary, 0, Horz.GridExtent, RowHeights[0]
  6019. + RowHeights[1]));
  6020. end;
  6021. // paint unoccupied space on bottom
  6022. if Vert.GridBoundary < Vert.GridExtent then
  6023. begin
  6024. // hex + chars
  6025. Canvas.Brush.Color := TestColor(Color);
  6026. Canvas.FillRect(Rect(ColWidths[0] + 1, Vert.GridBoundary, Horz.GridExtent,
  6027. Vert.GridExtent));
  6028. // fixed (position gutter)
  6029. Canvas.Brush.Color := TestColor(FColors.OffsetBackGround);
  6030. Canvas.FillRect(Rect(0, Vert.GridBoundary, ColWidths[0],
  6031. Vert.GridExtent));
  6032. end;
  6033. LIntTop := RowHeights[0] + RowHeights[1];
  6034. // draw bevel on the right of the offset gutter
  6035. if (ColWidths[0] <> 0) then
  6036. begin
  6037. if FDrawGutter3D then
  6038. begin
  6039. Canvas.MoveTo(ColWidths[0], LIntTop);
  6040. Canvas.Pen.Color := TestColor(clBtnShadow);
  6041. Canvas.LineTo(ColWidths[0], Vert.GridExtent);
  6042. Canvas.MoveTo(ColWidths[0] - 1, LIntTop);
  6043. Canvas.Pen.Color := TestColor(clBtnHighlight);
  6044. Canvas.LineTo(ColWidths[0] - 1, Vert.GridExtent);
  6045. end
  6046. else if FDrawGridLines then
  6047. begin
  6048. Canvas.MoveTo(ColWidths[0] - 1, LIntTop);
  6049. Canvas.Pen.Color := TestColor(FColors.Grid);
  6050. Canvas.LineTo(ColWidths[0] - 1, Vert.GridExtent);
  6051. end;
  6052. end;
  6053. if (FShowRuler) then
  6054. begin
  6055. if FDrawGutter3D then
  6056. begin
  6057. Canvas.MoveTo(ColWidths[0] - 1, LIntTop - 1);
  6058. Canvas.Pen.Color := TestColor(clBtnShadow);
  6059. Canvas.LineTo(Horz.GridExtent, LIntTop - 1);
  6060. Canvas.MoveTo(ColWidths[0] - 1, LIntTop - 2);
  6061. Canvas.Pen.Color := TestColor(clBtnHighlight);
  6062. Canvas.LineTo(Horz.GridExtent, LIntTop - 2);
  6063. end
  6064. else if FDrawGridLines then
  6065. begin
  6066. Canvas.MoveTo(ColWidths[0] - 1, LIntTop - 1);
  6067. Canvas.Pen.Color := TestColor(FColors.Grid);
  6068. Canvas.LineTo(Horz.GridExtent, LIntTop - 1);
  6069. end;
  6070. end;
  6071. end;
  6072. {$IFDEF DELPHI6UP}
  6073. if UseRightToLeftAlignment then
  6074. ChangeGridOrientation(False);
  6075. {$ENDIF}
  6076. end;
  6077. procedure TCustomMPHexEditor.SetSelectionAsHex(const s: string);
  6078. var
  6079. LStrData: string;
  6080. LIntAmount: integer;
  6081. begin
  6082. if s <> '' then
  6083. begin
  6084. SetLength(LStrData, Length(s));
  6085. ConvertHexToBin(@s[1], @LStrData[1], Length(s), SwapNibbles, LIntAmount);
  6086. SetLength(LStrData, LIntAmount);
  6087. SetSelectionAsText(LStrData);
  6088. end;
  6089. end;
  6090. function TCustomMPHexEditor.GetSelectionAsText: string;
  6091. begin
  6092. if (DataSize < 1) or (SelCount < 1) then
  6093. Result := ''
  6094. else
  6095. begin
  6096. SetLength(Result, SelCount);
  6097. FDataStorage.ReadBufferAt(Result[1], Min(SelStart, SelEnd), SelCount);
  6098. end;
  6099. end;
  6100. procedure TCustomMPHexEditor.SetSelectionAsText(const s: string);
  6101. begin
  6102. if s <> '' then
  6103. ReplaceSelection(@s[1], Length(s));
  6104. end;
  6105. procedure TCustomMPHexEditor.SetDrawGridLines(const Value: boolean);
  6106. begin
  6107. if Value <> FDrawGridLines then
  6108. begin
  6109. FDrawGridLines := Value;
  6110. Invalidate;
  6111. end;
  6112. end;
  6113. function TCustomMPHexEditor.UndoBeginUpdate: integer;
  6114. begin
  6115. Result := FUndoStorage.BeginUpdate;
  6116. end;
  6117. function TCustomMPHexEditor.UndoEndUpdate: integer;
  6118. begin
  6119. Result := FUndoStorage.EndUpdate;
  6120. end;
  6121. function TCustomMPHexEditor.Undo: boolean;
  6122. begin
  6123. Result := FUndoStorage.Undo;
  6124. end;
  6125. function TCustomMPHexEditor.Redo: boolean;
  6126. begin
  6127. Result := FUndoStorage.Redo;
  6128. end;
  6129. procedure TCustomMPHexEditor.SetGutterWidth(const Value: integer);
  6130. begin
  6131. if FGutterWidth <> Value then
  6132. begin
  6133. FGutterWidth := Value;
  6134. SetOffsetDisplayWidth;
  6135. Invalidate;
  6136. end;
  6137. end;
  6138. procedure TCustomMPHexEditor.BookmarkBitmapChanged(Sender: TObject);
  6139. var
  6140. LRctBox: TRect;
  6141. begin
  6142. // spalte 1 invalidieren
  6143. FBookmarkImageList.Clear;
  6144. FBookmarkImageList.AddMasked(FBookmarkBitmap, FBookmarkBitmap.Canvas.Pixels[0,
  6145. 0]);
  6146. if HandleAllocated then
  6147. begin
  6148. LRctBox := BoxRect(0, TopRow, 0, TopRow + VisibleRowCount);
  6149. InvalidateRect(Handle, @LRctBox, False);
  6150. end;
  6151. end;
  6152. procedure TCustomMPHexEditor.SetBookmarkBitmap(const Value: TBitmap);
  6153. begin
  6154. if Value = nil then
  6155. FBookmarkBitmap.LoadFromResourceName(HINSTANCE, 'BOOKMARKICONS')
  6156. else
  6157. begin
  6158. if (Value.Width <> 200) or (Value.Height <> 10) then
  6159. raise EMPHexEditor.Create(ERR_INVALID_BOOKMARKBMP);
  6160. FBookmarkBitmap.Assign(Value);
  6161. end;
  6162. FHasCustomBMP := Value <> nil;
  6163. end;
  6164. procedure TCustomMPHexEditor.SelectAll;
  6165. var
  6166. LgrcPosition: TGridCoord;
  6167. begin
  6168. if DataSize > 0 then
  6169. begin
  6170. // position auf ende stzen
  6171. if (not InsertMode) then
  6172. LgrcPosition := GetCursorAtPos(DataSize - 1, InCharField)
  6173. else
  6174. LgrcPosition := GetCursorAtPos(DataSize, InCharField);
  6175. MoveColRow(LgrcPosition.x, LgrcPosition.y, True, True);
  6176. // alles wählen
  6177. NewSelection(0, Pred(DataSize));
  6178. end;
  6179. end;
  6180. function TCustomMPHexEditor.GetVersion: string;
  6181. begin
  6182. Result := MPH_VERSION;
  6183. end;
  6184. procedure TCustomMPHexEditor.SetVersion(const Value: string);
  6185. begin
  6186. // readonly property
  6187. end;
  6188. procedure TCustomMPHexEditor.FreeStorage(FreeUndo: boolean = False);
  6189. begin
  6190. if not FreeUndo then
  6191. FDataStorage.Size := 0
  6192. else
  6193. FUndoStorage.Size := 0;
  6194. end;
  6195. procedure TCustomMPHexEditor.OldCursor;
  6196. begin
  6197. if Length(FCursorList) > 0 then
  6198. begin
  6199. Cursor := FCursorList[Pred(Length(FCursorList))];
  6200. SetLength(FCursorList, PRed(Length(FCursorList)));
  6201. end;
  6202. end;
  6203. procedure TCustomMPHexEditor.WaitCursor;
  6204. begin
  6205. SetLength(FCursorList, Succ(Length(FCursorList)));
  6206. FCursorList[Pred(Length(FCursorList))] := Cursor;
  6207. Cursor := crHourGlass;
  6208. end;
  6209. function TCustomMPHexEditor.HasCustomBookmarkBitmap: boolean;
  6210. begin
  6211. Result := FHasCustomBMP;
  6212. end;
  6213. procedure TCustomMPHexEditor.PrepareOverwriteDiskFile;
  6214. begin
  6215. if FIsFileReadonly then
  6216. raise EFOpenError.CreateFmt(ERR_FILE_READONLY, [FileName]);
  6217. end;
  6218. procedure TCustomMPHexEditor.Changed;
  6219. begin
  6220. if Assigned(FOnChange) then
  6221. FOnChange(self);
  6222. SelectionChanged;
  6223. end;
  6224. procedure TCustomMPHexEditor.SetDrawGutter3D(const Value: boolean);
  6225. begin
  6226. if FDrawGutter3D <> Value then
  6227. begin
  6228. FDrawGutter3D := Value;
  6229. Repaint;
  6230. end;
  6231. end;
  6232. procedure TCustomMPHexEditor.SetShowRuler(const Value: boolean);
  6233. begin
  6234. if (FShowRuler <> Value) or (csLoading in ComponentState) then
  6235. begin
  6236. FShowRuler := Value;
  6237. AdjustMetrics;
  6238. end;
  6239. end;
  6240. function TCustomMPHexEditor.DisplayEnd: integer;
  6241. begin
  6242. if DataSize < 1 then
  6243. Result := -1
  6244. else
  6245. Result := Min((DataSize - 1), (DisplayStart - 1) + (VisibleRowCount *
  6246. BytesPerRow));
  6247. end;
  6248. function TCustomMPHexEditor.DisplayStart: integer;
  6249. begin
  6250. if DataSize < 1 then
  6251. Result := -1
  6252. else
  6253. Result := GetPosAtCursor(GRID_FIXED, TopRow);
  6254. end;
  6255. procedure TCustomMPHexEditor.SetBytesPerUnit(const Value: integer);
  6256. begin
  6257. if FBytesPerUnit <> Value then
  6258. begin
  6259. if FUnicodeCharacters and (Value <> 2) then
  6260. raise EMPHexEditor.Create(ERR_INVALID_BPU_U);
  6261. if not (Value in [1, 2, 4, 8]) then
  6262. raise EMPHexEditor.CreateFmt(ERR_INVALID_BPU, [Value]);
  6263. FBytesPerUnit := Value;
  6264. if FRulerBytesPerUnit = -1 then
  6265. FUsedRulerBytesPerUnit := Value;
  6266. with FOffsetFormat do
  6267. if offBytesPerUnit in Flags then
  6268. _BytesPerUnit := FUsedRulerBytesPerUnit;
  6269. AdjustMetrics;
  6270. SetRulerString;
  6271. if (SelCount mod FBytesPerUnit) <> 0 then
  6272. ResetSelection(False);
  6273. Invalidate;
  6274. end;
  6275. end;
  6276. procedure TCustomMPHexEditor.SetRulerString;
  6277. var
  6278. intLoop, intLen: Integer;
  6279. sLoop: string;
  6280. begin
  6281. FRulerString := '';
  6282. intLen := 2 * FUsedRulerBytesPerUnit;
  6283. for intLoop := 0 to Pred(FBytesPerRow div FUsedRulerBytesPerUnit) do
  6284. begin
  6285. sLoop := IntToRadixLen(intLoop, FRulerNumberBase, intLen);
  6286. if Length(sLoop) > intLen then
  6287. Delete(sLoop, 1, Length(sLoop) - intLen);
  6288. FRulerString := FRulerString + sLoop;
  6289. end;
  6290. if FHexLowerCase then
  6291. FRulerString := LowerCase(FRulerString)
  6292. else
  6293. FRulerString := UpperCase(FRulerString);
  6294. FRulerCharString := '';
  6295. if FUnicodeCharacters then
  6296. intLen := FUsedRulerBytesPerUnit div 2
  6297. else
  6298. intLen := FUsedRulerBytesPerUnit;
  6299. for intLoop := 0 to Pred(FBytesPerRow div FUsedRulerBytesPerUnit) do
  6300. begin
  6301. sLoop := IntToRadix(intLoop, FRulerNumberBase);
  6302. if Length(sLoop) > intLen then
  6303. Delete(sLoop, 1, Length(sLoop) - intLen)
  6304. else
  6305. while Length(sLoop) < intLen do
  6306. sLoop := ' ' + sLoop;
  6307. FRulerCharString := FRulerCharString + sLoop;
  6308. end;
  6309. if FHexLowerCase then
  6310. FRulerCharString := LowerCase(FRulerCharString)
  6311. else
  6312. FRulerCharString := UpperCase(FRulerCharString);
  6313. end;
  6314. procedure TCustomMPHexEditor.CheckSelectUnit(var AStart, AEnd: Integer);
  6315. begin
  6316. // assure that the selection covers a whole unit
  6317. if AStart <= AEnd then
  6318. begin
  6319. CheckUnit(AStart);
  6320. CheckUnit(AEnd);
  6321. Inc(AEnd, FBytesPerUnit - 1);
  6322. if (AEnd >= DataSize) then
  6323. AEnd := Pred(DataSize);
  6324. end
  6325. else
  6326. begin
  6327. CheckUnit(AEnd);
  6328. CheckUnit(AStart);
  6329. Inc(AStart, FBytesPerUnit - 1);
  6330. if (AStart >= DataSize) then
  6331. AStart := Pred(DataSize);
  6332. end;
  6333. end;
  6334. // make sure the value is a multiple of FBytesPerUnit
  6335. procedure TCustomMPHexEditor.CheckUnit(var AValue: Integer);
  6336. begin
  6337. AValue := AValue div FBytesPerUnit * FBytesPerUnit;
  6338. end;
  6339. procedure TCustomMPHexEditor.SelectionChanged;
  6340. begin
  6341. if FSelectionChangedCount = 0 then
  6342. PostMessage(Handle, CM_SELECTIONCHANGED, 0, 0);
  6343. Inc(FSelectionChangedCount);
  6344. end;
  6345. procedure TCustomMPHexEditor.SyncView(Source: TCustomMPHexEditor;
  6346. SyncOffset: integer = 0);
  6347. var
  6348. curPos, SelS, SelE: integer;
  6349. coord: TGridCoord;
  6350. begin
  6351. if (Source.BytesPerRow = BytesPerRow) and (Source.BytesPerColumn =
  6352. BytesPerColumn) and (Source.BytesPerUnit = BytesPerUnit) and (Source.DataSize
  6353. = DataSize) and (SyncOffset = 0) then
  6354. begin
  6355. TopRow := Source.TopRow;
  6356. LeftCol := Source.LeftCol;
  6357. MoveColRow(Source.Col, Source.Row, True, False);
  6358. end
  6359. else
  6360. begin
  6361. // get the current view
  6362. curPos := Source.GetCursorPos;
  6363. coord := Source.GetCursorAtPos(curPos, Source.InCharField);
  6364. with Source.CellRect(coord.X, coord.Y) do
  6365. if Left + Bottom = 0 then
  6366. begin
  6367. curPos := Source.GetPositionAtCursor(Source.LeftCol, Source.TopRow) +
  6368. SyncOffset;
  6369. if curPos >= DataSize then
  6370. curPos := Pred(DataSize);
  6371. if curPos < 0 then
  6372. curPos := 0;
  6373. coord := GetCursorAtPos(curPos, Source.InCharField);
  6374. LeftCol := coord.X;
  6375. TopRow := coord.Y;
  6376. end
  6377. else
  6378. begin
  6379. // use this value if visible, left/top otherwise (when wheeling or scrolling)
  6380. curPos := curPos + SyncOffset;
  6381. if curPos >= DataSize then
  6382. curPos := Pred(DataSize);
  6383. if curPos < 0 then
  6384. curPos := 0;
  6385. coord := GetCursorAtPos(curPos, Source.InCharField);
  6386. MoveColRow(coord.X, coord.Y, True, True);
  6387. end;
  6388. end;
  6389. if (Source.SelCount = 0) then
  6390. begin
  6391. if (SelCount <> 0) then
  6392. ResetSelection(True)
  6393. end
  6394. else
  6395. begin
  6396. SelS := Source.FSelStart + SyncOffset;
  6397. SelE := Source.FSelEnd + SyncOffset;
  6398. if SelE >= DataSize then
  6399. SelE := DataSize - 1;
  6400. if SelS >= DataSize then
  6401. SelS := DataSize - 1;
  6402. if SelE < 0 then
  6403. SelE := 0;
  6404. if SelS < 0 then
  6405. SelS := 0;
  6406. NewSelection(SelS, SelE);
  6407. end;
  6408. end;
  6409. procedure TCustomMPHexEditor.CMSelectionChanged(var Msg: TMessage);
  6410. begin
  6411. if (FSelectionChangedCount <> 0) and Assigned(FOnSelectionChanged) then
  6412. try
  6413. FOnSelectionChanged(self);
  6414. finally
  6415. FSelectionChangedCount := 0;
  6416. end;
  6417. end;
  6418. procedure TCustomMPHexEditor.SetRulerBytesPerUnit(const Value: integer);
  6419. begin
  6420. if FRulerBytesPerUnit <> Value then
  6421. begin
  6422. if (not (Value in [1, 2, 4, 8])) and (Value <> -1) then
  6423. raise EMPHexEditor.CreateFmt(ERR_INVALID_RBPU, [Value]);
  6424. FRulerBytesPerUnit := Value;
  6425. if Value = -1 then
  6426. FUsedRulerBytesPerUnit := FBytesPerUnit
  6427. else
  6428. FUsedRulerBytesPerUnit := Value;
  6429. with FOffsetFormat do
  6430. if offBytesPerUnit in Flags then
  6431. _BytesPerUnit := FUsedRulerBytesPerUnit;
  6432. AdjustMetrics;
  6433. SetRulerString;
  6434. Invalidate;
  6435. end;
  6436. end;
  6437. procedure TCustomMPHexEditor.SetShowPositionIfNotFocused(const Value: Boolean);
  6438. begin
  6439. if FShowPositionIfNotFocused <> Value then
  6440. begin
  6441. FShowPositionIfNotFocused := Value;
  6442. Invalidate;
  6443. end;
  6444. end;
  6445. function TCustomMPHexEditor.GetDataAt(Index: integer): Byte;
  6446. begin
  6447. {$IFDEF FASTACCESS}
  6448. {$R-}
  6449. Result := GetFastPointer(Index,1)^;
  6450. {$ELSE}
  6451. ReadBuffer(Result, Index, sizeof(Result));
  6452. {$ENDIF}
  6453. end;
  6454. procedure TCustomMPHexEditor.SetDataAt(Index: integer; const Value: Byte);
  6455. begin
  6456. {$IFDEF FASTACCESS}
  6457. GetFastPointer(Index, 1)^ := Value;
  6458. {$ELSE}
  6459. WriteBuffer(Value, Index, sizeof(Value));
  6460. {$ENDIF}
  6461. end;
  6462. procedure TCustomMPHexEditor.ReadBuffer(var Buffer; const Index, Count:
  6463. Integer);
  6464. begin
  6465. {$IFDEF FASTACCESS}
  6466. Move(GetFastPointer(Index, Count)^, Buffer, Count);
  6467. {$ELSE}
  6468. FDataStorage.ReadBufferAt(Buffer, Index, Count);
  6469. {$ENDIF}
  6470. end;
  6471. procedure TCustomMPHexEditor.WriteBuffer(const Buffer; const Index, Count:
  6472. Integer);
  6473. begin
  6474. {$IFDEF FASTACCESS}
  6475. Move(Buffer, GetFastPointer(Index,Count)^, Count);
  6476. {$ELSE}
  6477. FDataStorage.WriteBufferAt(Buffer, Index, Count);
  6478. {$ENDIF}
  6479. end;
  6480. // fire OnBookmarkChanged
  6481. procedure TCustomMPHexEditor.BookmarkChanged;
  6482. begin
  6483. if Assigned(FOnBookmarkChanged) then
  6484. FOnBookmarkChanged(self);
  6485. end;
  6486. procedure TCustomMPHexEditor.DoSetCellWidth(const Index: integer;
  6487. Value: integer);
  6488. begin
  6489. ColWidths[Index] := Value;
  6490. end;
  6491. // legacy, do not use
  6492. function TCustomMPHexEditor.GetMemory(const Index: Integer): char;
  6493. begin
  6494. Result := Char(Data[Index])
  6495. end;
  6496. // legacy, do not use
  6497. procedure TCustomMPHexEditor.SetMemory(const Index: integer; const Value: char);
  6498. begin
  6499. Data[Index] := Ord(Value);
  6500. end;
  6501. procedure TCustomMPHexEditor.SetUnicodeCharacters(const Value: Boolean);
  6502. begin
  6503. if FUnicodeCharacters <> Value then
  6504. begin
  6505. if Value then
  6506. begin
  6507. if (BytesPerRow mod 2) <> 0 then
  6508. raise EMPHexEditor.Create(ERR_INVALID_BYTESPERLINE);
  6509. if (BytesPerColumn mod 2) <> 0 then
  6510. raise EMPHexEditor.Create(ERR_INVALID_BYTESPERCOL);
  6511. if (DataSize mod 2) <> 0 then
  6512. raise EMPHexEditor.Create(ERR_ODD_FILESIZE_UNICODE);
  6513. FTranslation := tkAsIs;
  6514. end;
  6515. FUnicodeCharacters := Value;
  6516. ColCount := CalcColCount;
  6517. if Value then
  6518. BytesPerUnit := 2
  6519. else
  6520. BytesPerUnit := 1;
  6521. CalcSizes;
  6522. SetRulerString;
  6523. Invalidate;
  6524. end;
  6525. end;
  6526. procedure TCustomMPHexEditor.SetUnicodeBigEndian(const Value: Boolean);
  6527. begin
  6528. if FUnicodeBigEndian <> Value then
  6529. begin
  6530. FUnicodeBigEndian := Value;
  6531. if FUnicodeCharacters then
  6532. Invalidate;
  6533. end;
  6534. end;
  6535. function TCustomMPHexEditor.GetPositionAtCursor(const ACol,
  6536. ARow: integer): integer;
  6537. var
  6538. LBoolInCharField: Boolean;
  6539. begin
  6540. LBoolInCharField := FPosInCharField;
  6541. try
  6542. Result := GetPosAtCursor(ACol, ARow);
  6543. finally
  6544. FPosInCharField := (LBoolInCharField);
  6545. end;
  6546. end;
  6547. function TCustomMPHexEditor.GetIsCharFieldCol(
  6548. const ACol: integer): Boolean;
  6549. begin
  6550. Result := ACol > (GRID_FIXED + FBytesPerRowDup);
  6551. end;
  6552. function TCustomMPHexEditor.IsFileSizeFixed: boolean;
  6553. begin
  6554. if FFixedFileSizeOverride then
  6555. Result := False
  6556. else
  6557. Result := FFixedFileSize;
  6558. end;
  6559. function TCustomMPHexEditor.IsInsertModePossible: boolean;
  6560. begin
  6561. Result := (not IsFileSizeFixed) and FAllowInsertMode and (not FReadOnlyView)
  6562. end;
  6563. function TCustomMPHexEditor.Replace(aBuffer: PChar; aPosition, aOldCount,
  6564. aNewCount: integer;
  6565. const UndoDesc: string = ''; const MoveCursor: Boolean = False): integer;
  6566. var
  6567. LBoolInCharField: boolean;
  6568. LIntSize: integer;
  6569. begin
  6570. FDataStorage.CheckBounds((Abs(aPosition) + Abs(aOldCount)) - 1);
  6571. LIntSize := DataSize;
  6572. // auswahl berechnen
  6573. LBoolInCharField := GetInCharField;
  6574. if LIntSize - APosition < aOldCount then
  6575. begin
  6576. if aNewCount = aOldCount then
  6577. aNewCount := LIntSize - APosition;
  6578. aOldCount := LIntSize - APosition;
  6579. end;
  6580. if IsFileSizeFixed then
  6581. begin
  6582. if aOldCount < aNewCount then
  6583. aNewCount := aOldCount
  6584. else
  6585. aOldCount := aNewCount;
  6586. end;
  6587. CreateUndo(ufKindReplace, APosition, aNewCount, aOldCount, UndoDesc);
  6588. if (not MoveCursor) and (FUndoStorage.FUpdateCount = 0) then
  6589. FUndoStorage.AddSelection(APosition, aOldCount);
  6590. if aOldCount = aNewCount then
  6591. WriteBuffer(aBuffer^, APosition, aOldCount)
  6592. else
  6593. if aOldCount > aNewCount then
  6594. begin
  6595. InternalDelete(APosition, APosition + (aOldCount - aNewCount), Col, Row);
  6596. WriteBuffer(aBuffer^, APosition, aNewCount)
  6597. end
  6598. else
  6599. begin
  6600. InternalInsertBuffer(nil, aNewCount-aOldCount, APosition);
  6601. WriteBuffer(aBuffer^, APosition, aNewCount)
  6602. end;
  6603. Result := aNewCount;
  6604. if FModifiedBytes.Size >= APosition then
  6605. FModifiedBytes.Size := Max(0, APosition);
  6606. if MoveCursor then
  6607. begin
  6608. with GetCursorAtPos(APosition, LBoolInCharField) do
  6609. MoveColRow(x, y, True, True);
  6610. end;
  6611. Invalidate;
  6612. Changed;
  6613. end;
  6614. function TCustomMPHexEditor.GotoBookmark(const Index: integer): boolean;
  6615. var
  6616. LIntRow: integer;
  6617. LgrcPosition: TGridCoord;
  6618. begin
  6619. Result := False;
  6620. if FBookmarks[Index].mPosition > -1 then
  6621. begin
  6622. ResetSelection(True);
  6623. LIntRow := FBookmarks[Index].mPosition;
  6624. if (LIntRow < DataSize) or ((LIntRow = DataSize) and InsertMode) then
  6625. begin
  6626. LgrcPosition := GetCursorAtPos(LIntRow, FBookmarks[Index].mInCharField);
  6627. MoveColRow(LgrcPosition.x, LgrcPosition.y, True, True);
  6628. Result := True;
  6629. end
  6630. else
  6631. SetBookmarkVals(Index, -1, False);
  6632. end;
  6633. end;
  6634. procedure TCustomMPHexEditor.UpdateGetOffsetText;
  6635. begin
  6636. SetOffsetDisplayWidth;
  6637. Invalidate;
  6638. CheckSetCaret;
  6639. end;
  6640. {$IFDEF FASTACCESS}
  6641. function TCustomMPHexEditor.GetFastPointer(const Index, Count: integer): PByte;
  6642. begin
  6643. Result := FDataStorage.GetAddress(Index, Count);
  6644. end;
  6645. {$ENDIF}
  6646. procedure TCustomMPHexEditor.SeekToEOF;
  6647. var
  6648. LgrcPosition: TGridCoord;
  6649. begin
  6650. InCharField;
  6651. if (not InsertMode) then
  6652. LgrcPosition := GetCursorAtPos(DataSize - 1, FPosInCharField)
  6653. else
  6654. LgrcPosition := GetCursorAtPos(DataSize, FPosInCharField);
  6655. MoveColRow(LgrcPosition.x, LgrcPosition.y, True, True)
  6656. end;
  6657. function TCustomMPHexEditor.CanCreateUndo(const aKind: TMPHUndoFlag; const
  6658. aCount,
  6659. aReplCount: integer): Boolean;
  6660. begin
  6661. Result := False;
  6662. if DataSize > 0 then
  6663. Result := True;
  6664. if not Result then
  6665. if aKind in [ufKindInsertBuffer, ufKindAppendBuffer, ufKindAllData] then
  6666. Result := True;
  6667. // check for NoSizeChange
  6668. if IsFileSizeFixed and Result then
  6669. if (aKind in [ufKindByteRemoved, ufKindInsertBuffer, ufKindAppendBuffer,
  6670. ufKindNibbleInsert,
  6671. ufKindNibbleDelete]) or
  6672. ((aKind = ufKindReplace) and (aCount <> aReplCount)) then
  6673. Result := False;
  6674. if (not Result) and ((aKind = ufKindCombined) and (FUndoStorage.Count >=
  6675. aCount)) then
  6676. Result := True;
  6677. end;
  6678. procedure TCustomMPHexEditor.SetDataSize(const Value: integer);
  6679. var
  6680. iPos: Integer;
  6681. iSize: integer;
  6682. begin
  6683. iSize := DataSize;
  6684. if Value <> iSize then
  6685. begin
  6686. iPos := GetCursorPos;
  6687. // new in 12-16-2003: don't allow change of datasize if nosizechange
  6688. // and (new datasize <> 0 and old datasize <> 0)
  6689. if (Value <> 0) and (iSize <> 0) and IsFileSizeFixed then
  6690. raise EMPHexEditor.Create(ERR_FIXED_FILESIZE);
  6691. FFixedFileSizeOverride := True;
  6692. try
  6693. // new in 12-16-2003: generate undo
  6694. if Value < iSize then
  6695. // create a 'bytes deleted' undo
  6696. CreateUndo(ufKindByteRemoved, Value, DataSize - Value, 0)
  6697. else
  6698. // create a 'append buffer' undo
  6699. CreateUndo(ufKindAppendBuffer, DataSize, Value - DataSize, 0);
  6700. FDataStorage.Size := Value;
  6701. {$IFDEF FASTACCESS}
  6702. if Value > iSize then
  6703. // fill the new data block
  6704. FillChar(GetFastPointer(iSize, Value-iSize)^, Value - iSize, FSetDataSizeFillByte);
  6705. {$ENDIF}
  6706. FModified := True;
  6707. CalcSizes;
  6708. if iPos > DataSize then
  6709. begin
  6710. ResetSelection(True);
  6711. if (DataSize = 0) and (not InsertMode) then
  6712. begin
  6713. with GetCursorAtPos(0, InCharField) do
  6714. MoveColRow(X, Y, True, True);
  6715. end
  6716. else
  6717. SeekToEOF;
  6718. end;
  6719. finally
  6720. FFixedFileSizeOverride := False;
  6721. end;
  6722. end;
  6723. end;
  6724. procedure TCustomMPHexEditor.SetBlockSize(const Value: Integer);
  6725. begin
  6726. if FBlockSize <> Value then
  6727. begin
  6728. FBlockSize := Value;
  6729. AdjustMetrics;
  6730. end;
  6731. end;
  6732. procedure TCustomMPHexEditor.SetSepCharBlocks(const Value: boolean);
  6733. begin
  6734. if FSepCharBlocks <> Value then
  6735. begin
  6736. FSepCharBlocks := Value;
  6737. if Value and (FBlockSize > 1) then
  6738. AdjustMetrics;
  6739. end;
  6740. end;
  6741. procedure TCustomMPHexEditor.SetFindProgress(const Value: boolean);
  6742. begin
  6743. FFindProgress := Value;
  6744. end;
  6745. procedure TCustomMPHexEditor.DefineProperties(Filer: TFiler);
  6746. begin
  6747. inherited;
  6748. Filer.DefineProperty('MaskChar', ReadMaskChar, nil, False);
  6749. Filer.DefineProperty('MaskChar_AsInteger', ReadMaskChar_I, WriteMaskChar_I,
  6750. FReplaceUnprintableCharsBy <> '.');
  6751. end;
  6752. procedure TCustomMPHexEditor.ReadMaskChar(Reader: TReader);
  6753. var
  6754. s: string;
  6755. begin
  6756. s := Reader.ReadString;
  6757. if Length(s) <> 1 then
  6758. FReplaceUnprintableCharsBy := '.'
  6759. else
  6760. try
  6761. FReplaceUnprintableCharsBy := s[1];
  6762. except
  6763. FReplaceUnprintableCharsBy := '.';
  6764. end;
  6765. end;
  6766. procedure TCustomMPHexEditor.ReadMaskChar_I(Reader: TReader);
  6767. begin
  6768. try
  6769. Byte(FReplaceUnprintableCharsBy) := Reader.ReadInteger;
  6770. except
  6771. FReplaceUnprintableCharsBy := '.';
  6772. end;
  6773. end;
  6774. procedure TCustomMPHexEditor.WriteMaskChar_I(Writer: TWriter);
  6775. begin
  6776. Writer.WriteInteger(Byte(FReplaceUnprintableCharsBy));
  6777. end;
  6778. function TCustomMPHexEditor.DoMouseWheelDown(Shift: TShiftState;
  6779. MousePos: TPoint): boolean;
  6780. begin
  6781. if Shift <> [] then
  6782. Result := inherited DoMouseWheelDown(Shift, MousePos)
  6783. else
  6784. begin
  6785. // scroll down one page
  6786. TopRow := Min(Max(GRID_FIXED, RowCount - VisibleRowCount),
  6787. TopRow + VisibleRowCount - 1);
  6788. CheckSetCaret;
  6789. Result := True;
  6790. end;
  6791. end;
  6792. function TCustomMPHexEditor.DoMouseWheelUp(Shift: TShiftState;
  6793. MousePos: TPoint): boolean;
  6794. begin
  6795. if Shift <> [] then
  6796. Result := inherited DoMouseWheelUp(Shift, MousePos)
  6797. else
  6798. begin
  6799. // scroll up one page
  6800. TopRow := Max(GRID_FIXED, TopRow - VisibleRowCount + 1);
  6801. CheckSetCaret;
  6802. Result := True;
  6803. end;
  6804. end;
  6805. procedure TCustomMPHexEditor.CheckSetCaret;
  6806. begin
  6807. with CellRect(Col, Row) do
  6808. begin
  6809. if Left + Bottom = 0 then
  6810. IntSetCaretPos(-50, -50, -1)
  6811. else
  6812. IntSetCaretPos(Left, Top, Col);
  6813. end;
  6814. end;
  6815. function TCustomMPHexEditor.CanFocus: Boolean;
  6816. var
  6817. Form: TCustomForm;
  6818. begin
  6819. Result := {$IFDEF DELPHI5UP}inherited CanFocus{$ELSE}True{$ENDIF};
  6820. if Result and not (csDesigning in ComponentState) then
  6821. begin
  6822. Form := GetParentForm(Self);
  6823. Result := (not Assigned(Form)) or (Form.Enabled and Form.Visible);
  6824. end;
  6825. end;
  6826. procedure TCustomMPHexEditor.SetRulerNumberBase(const Value: byte);
  6827. begin
  6828. if FRulerNumberBase <> Value then
  6829. begin
  6830. // force number that can be represented using '0'-'9','A'-'F'
  6831. if not (Value in [2..16]) then
  6832. FRulerNumberBase := 16
  6833. else
  6834. FRulerNumberBase := Value;
  6835. SetRulerString;
  6836. if FShowRuler then
  6837. Invalidate;
  6838. end;
  6839. end;
  6840. procedure TCustomMPHexEditor.SetMaskedChars(const Value: TSysCharSet);
  6841. begin
  6842. if FMaskedChars <> Value then
  6843. begin
  6844. FMaskedChars := Value;
  6845. Invalidate;
  6846. end;
  6847. end;
  6848. procedure TCustomMPHexEditor.CenterCursorPosition;
  6849. var
  6850. iPos: integer;
  6851. begin
  6852. iPos := GetCursorPos;
  6853. iPos := (iPos div FBytesPerRow) + GRID_FIXED;
  6854. TopRow := Max(GRID_FIXED, Min(iPos - (VisibleRowCount div 2), RowCount-VisibleRowCount));
  6855. end;
  6856. {+}
  6857. {.$IFDEF BCB}
  6858. procedure TCustomMPHexEditor.DrawCell(ACol, ARow: Integer; ARect: TRect; AState: TGridDrawState);
  6859. begin
  6860. { empty }
  6861. end;
  6862. {.$ENDIF}
  6863. {+.}
  6864. { TMPHColors }
  6865. procedure TMPHColors.Assign(Source: TPersistent);
  6866. begin
  6867. if Source is TMPHColors then
  6868. begin
  6869. Background := TMPHColors(Source).Background;
  6870. ChangedText := TMPHColors(Source).ChangedText;
  6871. CursorFrame := TMPHColors(Source).CursorFrame;
  6872. NonFocusCursorFrame := TMPHColors(Source).NonFocusCursorFrame;
  6873. Offset := TMPHColors(Source).Offset;
  6874. OddColumn := TMPHColors(Source).OddColumn;
  6875. EvenColumn := TMPHColors(Source).EvenColumn;
  6876. ChangedBackground := TMPHColors(Source).ChangedBackground;
  6877. CurrentOffsetBackground := TMPHColors(Source).CurrentOffsetBackground;
  6878. CurrentOffset := TMPHColors(Source).CurrentOffset;
  6879. OffsetBackground := TMPHColors(Source).OffsetBackground;
  6880. ActiveFieldBackground := TMPHColors(Source).ActiveFieldBackground;
  6881. Grid := TMPHColors(Source).Grid;
  6882. end;
  6883. end;
  6884. constructor TMPHColors.Create(Parent: TControl);
  6885. begin
  6886. inherited Create;
  6887. FBackground := clWindow;
  6888. FActiveFieldBackground := clWindow;
  6889. FChangedText := clMaroon;
  6890. FCursorFrame := clNavy;
  6891. FNonFocusCursorFrame := clAqua;
  6892. FOffset := clBlack;
  6893. FOddColumn := clBlue;
  6894. FEvenColumn := clNavy;
  6895. FChangedBackground := $00A8FFFF;
  6896. FCurrentOffsetBackground := clBtnShadow;
  6897. FCurrentOffset := clBtnHighLight;
  6898. FOffsetBackground := clBtnFace;
  6899. FGrid := clBtnFace;
  6900. FParent := Parent;
  6901. end;
  6902. procedure TMPHColors.SetBackground(const Value: TColor);
  6903. begin
  6904. if FBackground <> Value then
  6905. begin
  6906. FBackground := Value;
  6907. if Assigned(fParent) then
  6908. begin
  6909. TCustomMPHexEditor(FParent).Color := Value;
  6910. fParent.Invalidate;
  6911. end;
  6912. end;
  6913. end;
  6914. procedure TMPHColors.SetChangedBackground(const Value: TColor);
  6915. begin
  6916. if FChangedBackground <> Value then
  6917. begin
  6918. FChangedBackground := Value;
  6919. if Assigned(fParent) then
  6920. fParent.Invalidate;
  6921. end;
  6922. end;
  6923. procedure TMPHColors.SetCurrentOffsetBackground(const Value: TColor);
  6924. begin
  6925. if FCurrentOffsetBackground <> Value then
  6926. begin
  6927. FCurrentOffsetBackground := Value;
  6928. if Assigned(fParent) then
  6929. fParent.Invalidate;
  6930. end;
  6931. end;
  6932. procedure TMPHColors.SetNonFocusCursorFrame(const Value: TColor);
  6933. begin
  6934. if FNonFocusCursorFrame <> Value then
  6935. begin
  6936. FNonFocusCursorFrame := Value;
  6937. if Assigned(fParent) then
  6938. fParent.Invalidate;
  6939. end;
  6940. end;
  6941. procedure TMPHColors.SetChangedText(const Value: TColor);
  6942. begin
  6943. if FChangedText <> Value then
  6944. begin
  6945. FChangedText := Value;
  6946. if Assigned(fParent) then
  6947. fParent.Invalidate;
  6948. end;
  6949. end;
  6950. procedure TMPHColors.SetCursorFrame(const Value: TColor);
  6951. begin
  6952. if FCursorFrame <> Value then
  6953. begin
  6954. FCursorFrame := Value;
  6955. if Assigned(fParent) then
  6956. fParent.Invalidate;
  6957. end;
  6958. end;
  6959. procedure TMPHColors.SetEvenColumn(const Value: TColor);
  6960. begin
  6961. if FEvenColumn <> Value then
  6962. begin
  6963. FEvenColumn := Value;
  6964. if Assigned(fParent) then
  6965. fParent.Invalidate;
  6966. end;
  6967. end;
  6968. procedure TMPHColors.SetOddColumn(const Value: TColor);
  6969. begin
  6970. if FOddColumn <> Value then
  6971. begin
  6972. FOddColumn := Value;
  6973. if Assigned(fParent) then
  6974. fParent.Invalidate;
  6975. end;
  6976. end;
  6977. procedure TMPHColors.SetOffset(const Value: TColor);
  6978. begin
  6979. if FOffset <> Value then
  6980. begin
  6981. FOffset := Value;
  6982. if Assigned(fParent) then
  6983. fParent.Invalidate;
  6984. end;
  6985. end;
  6986. procedure TMPHColors.SetOffsetBackGround(const Value: TColor);
  6987. begin
  6988. if FOffsetBackGround <> Value then
  6989. begin
  6990. FOffsetBackGround := Value;
  6991. if Assigned(fParent) then
  6992. fParent.Invalidate;
  6993. end;
  6994. end;
  6995. procedure TMPHColors.SetCurrentOffset(const Value: TColor);
  6996. begin
  6997. if FCurrentOffset <> Value then
  6998. begin
  6999. FCurrentOffset := Value;
  7000. if Assigned(fParent) then
  7001. fParent.Invalidate;
  7002. end;
  7003. end;
  7004. procedure TMPHColors.SetParent(const Value: TControl);
  7005. begin
  7006. FParent := Value;
  7007. Assign(self);
  7008. end;
  7009. procedure TMPHColors.SetGrid(const Value: TColor);
  7010. begin
  7011. if FGrid <> Value then
  7012. begin
  7013. FGrid := Value;
  7014. if Assigned(fParent) then
  7015. fParent.Invalidate;
  7016. end;
  7017. end;
  7018. procedure TMPHColors.SetActiveFieldBackground(const Value: TColor);
  7019. begin
  7020. if FActiveFieldBackground <> Value then
  7021. begin
  7022. FActiveFieldBackground := Value;
  7023. if Assigned(fParent) then
  7024. fParent.Invalidate;
  7025. end;
  7026. end;
  7027. { TMPHUndoStorage }
  7028. type
  7029. // undo storage
  7030. PUndoSelRec = ^TUndoSelRec;
  7031. TUndoSelRec = packed record
  7032. SelStart,
  7033. SelEnd,
  7034. SelPos: integer;
  7035. end;
  7036. constructor TMPHUndoStorage.Create(AEditor: TCustomMPHexEditor);
  7037. begin
  7038. inherited Create;
  7039. FEditor := AEditor;
  7040. FRedoPointer := nil;
  7041. FLastUndo := nil;
  7042. FLastUndoSize := 0;
  7043. Reset;
  7044. end;
  7045. destructor TMPHUndoStorage.Destroy;
  7046. begin
  7047. Reset;
  7048. inherited;
  7049. end;
  7050. function TMPHUndoStorage.BeginUpdate: integer;
  7051. begin
  7052. Inc(FUpdateCount);
  7053. Result := FUpdateCount;
  7054. end;
  7055. function TMPHUndoStorage.CanUndo: boolean;
  7056. begin
  7057. Result := (FCount > 0) and (FUpdateCount < 1) and (Size > 0);
  7058. end;
  7059. procedure TMPHUndoStorage.CreateUndo(aKind: TMPHUndoFlag; APosition, ACount,
  7060. AReplaceCount: integer; const SDescription: string);
  7061. var
  7062. urPos: integer;
  7063. function PUndoRec: PMPHUndoRec;
  7064. begin
  7065. Result := PMPHUndoRec(@(PChar(Memory)[urPos]))
  7066. end;
  7067. //LPurUndoRec: PMPHUndoRec;
  7068. procedure NewFillBuffer(ASize: integer);
  7069. var
  7070. i: integer;
  7071. begin
  7072. i := Position;
  7073. urPos := i;
  7074. (*if FEditor.FSelPosition > -1 then
  7075. ASize := ASize+sizeof(TUndoSelRec);*)
  7076. Size := Size + sizeof(TMPHUndoRec) + ASize;
  7077. FillChar(PUndoRec^, SizeOf(TMPHUndoRec) + ASize, 0);
  7078. with PUndoRec^ do
  7079. begin
  7080. Flags := [aKind];
  7081. CurPos := FEditor.GetPosAtCursor(FEditor.Col, FEditor.Row);
  7082. if not FEditor.FPosInCharField then
  7083. with FEditor.GetCursorAtPos(CurPos, FEditor.FPosInCharField) do
  7084. if (FEditor.Col - x) <> 0 then
  7085. Include(Flags, ufFlag2ndByteCol);
  7086. if FEditor.FPosInCharField then
  7087. Include(Flags, ufFlagInCharField);
  7088. if FEditor.FInsertModeOn then
  7089. Include(Flags, ufFlagInsertMode);
  7090. Pos := aPosition;
  7091. Count := aCount;
  7092. ReplCount := aReplaceCount;
  7093. CurTranslation := FEditor.FTranslation;
  7094. if FEditor.UnicodeChars then
  7095. Include(Flags, ufFlagIsUnicode);
  7096. if FEditor.UnicodeBigEndian then
  7097. Include(Flags, ufFlagIsUnicodeBigEndian);
  7098. CurBPU := FEditor.BytesPerUnit;
  7099. if FEditor.FModified then
  7100. Include(Flags, ufFlagModified);
  7101. if FEditor.FSelPosition > -1 then
  7102. Include(Flags, ufFlagHasSelection);
  7103. if SDescription <> '' then
  7104. Include(Flags, ufFlagHasDescription);
  7105. end;
  7106. end;
  7107. procedure DeleteOldestUndoRec;
  7108. var
  7109. LintRecSize: integer;
  7110. begin
  7111. begin
  7112. if Size < 4 then
  7113. begin
  7114. Size := 0;
  7115. FCount := 0;
  7116. end
  7117. else
  7118. begin
  7119. Seek(0, soFromBeginning);
  7120. Read(LIntRecSize, sizeof(integer));
  7121. if LIntRecSize < sizeof(TMPHUndoRec) then
  7122. begin
  7123. Size := 0;
  7124. FCount := 0;
  7125. end
  7126. else
  7127. begin
  7128. Move(PChar(Memory)[LIntRecSize], Memory^, Size - LIntRecSize);
  7129. Size := Size - LIntRecSize;
  7130. Dec(FCount);
  7131. end;
  7132. end;
  7133. end;
  7134. end;
  7135. procedure UpdateUndoRecord(Length: integer = 0);
  7136. var
  7137. LRecSelection: TUndoSelRec;
  7138. i: integer;
  7139. begin
  7140. PUndoRec^.DataLen := SizeOf(TMPHUndoRec) + Length + 4;
  7141. if ufFlagHasSelection in PUndoRec^.Flags then
  7142. Inc(PUndoRec^.DataLen, sizeof(TUndoSelRec));
  7143. if ufFlagHasDescription in PUndoRec^.Flags then
  7144. Inc(PUndoRec^.DataLen, system.Length(SDescription) + sizeof(i));
  7145. Position := Size;
  7146. if ufFlagHasDescription in PUndoRec^.Flags then
  7147. begin
  7148. write(Sdescription[1], system.Length(SDescription));
  7149. i := system.Length(sDescription);
  7150. write(i, sizeof(i));
  7151. Length := Length + i + sizeof(i);
  7152. end;
  7153. if ufFlagHasSelection in PUndoRec^.Flags then
  7154. begin
  7155. with LRecSelection do
  7156. begin
  7157. SelStart := FEditor.FSelStart;
  7158. SelEnd := FEditor.FSelEnd;
  7159. SelPos := FEditor.FSelPosition;
  7160. end;
  7161. Write(LRecSelection, sizeof(LRecSelection));
  7162. Length := Length + sizeof(LRecSelection);
  7163. end;
  7164. Length := SizeOf(TMPHUndoRec) + 4 + Length;
  7165. Write(Length, 4);
  7166. end;
  7167. var
  7168. LPtrBytes: PByteArray;
  7169. LSStDesc: shortstring;
  7170. begin
  7171. if FUpdateCount < 1 then
  7172. begin
  7173. ResetRedo;
  7174. if sDescription <> '' then
  7175. FDescription := sDescription
  7176. else
  7177. FDescription := STRS_UNDODESC[aKind];
  7178. while (FEditor.FMaxUndo > 0) and (FCount > 0) and (Size > FEditor.FMaxUndo)
  7179. do
  7180. DeleteOldestUndoRec;
  7181. Position := Size;
  7182. Inc(FCount);
  7183. case aKind of
  7184. ufKindBytesChanged:
  7185. begin
  7186. NewFillBuffer(aCount - 1);
  7187. LPtrBytes := PByteArray(@PUndoRec.Buffer);
  7188. FEditor.ReadBuffer(LPtrBytes^, aPosition, aCount);
  7189. if FEditor.HasChanged(aPosition) then
  7190. Include(PUndoRec.Flags, ufFlagByte1Changed);
  7191. if (aCount = 2) and FEditor.HasChanged(aPosition + 1) then
  7192. Include(PUndoRec.Flags, ufFlagByte2Changed);
  7193. UpdateUndoRecord(aCount - 1);
  7194. end;
  7195. ufKindByteRemoved:
  7196. begin
  7197. NewFillBuffer(aCount - 1);
  7198. LPtrBytes := PByteArray(@PUndoRec.Buffer);
  7199. FEditor.ReadBuffer(LPtrBytes^, aPosition, aCount);
  7200. FEditor.AdjustBookmarks(aPosition + aCount, -aCount);
  7201. UpdateUndoRecord(aCount - 1);
  7202. end;
  7203. ufKindInsertBuffer:
  7204. begin
  7205. NewFillBuffer(0);
  7206. FEditor.AdjustBookmarks(aPosition, aCount);
  7207. UpdateUndoRecord;
  7208. end;
  7209. ufKindReplace:
  7210. begin
  7211. NewFillBuffer(aReplaceCount - 1);
  7212. LPtrBytes := PByteArray(@PUndoRec.Buffer);
  7213. FEditor.ReadBuffer(LPtrBytes^, aPosition, aReplaceCount);
  7214. FEditor.AdjustBookmarks(aPosition + aCount, aCount - aReplaceCount);
  7215. UpdateUndoRecord(aReplaceCount - 1);
  7216. end;
  7217. ufKindAppendBuffer:
  7218. begin
  7219. NewFillBuffer(0);
  7220. UpdateUndoRecord;
  7221. end;
  7222. ufKindNibbleInsert:
  7223. begin
  7224. NewFillBuffer(0);
  7225. PUndoRec.Buffer := FEditor.Data[aPosition];
  7226. if FEditor.HasChanged(aPosition) then
  7227. Include(PUndoRec.Flags, ufFlagByte1Changed);
  7228. UpdateUndoRecord;
  7229. end;
  7230. ufKindNibbleDelete:
  7231. begin
  7232. NewFillBuffer(0);
  7233. PUndoRec.Buffer := FEditor.Data[aPosition];
  7234. if FEditor.HasChanged(aPosition) then
  7235. Include(PUndoRec.Flags, ufFlagByte1Changed);
  7236. UpdateUndoRecord;
  7237. end;
  7238. ufKindConvert:
  7239. begin
  7240. NewFillBuffer(aCount - 1);
  7241. LPtrBytes := PByteArray(@PUndoRec.Buffer);
  7242. FEditor.ReadBuffer(LPtrBytes^, aPosition, aCount);
  7243. UpdateUndoRecord(aCount - 1);
  7244. end;
  7245. ufKindSelection:
  7246. begin
  7247. NewFillBuffer(0);
  7248. PUndoRec^.CurPos := APosition;
  7249. UpdateUndoRecord;
  7250. AddSelection(APosition, ACount);
  7251. end;
  7252. ufKindAllData:
  7253. begin
  7254. aCount := FEditor.DataSize;
  7255. if aCount = 0 then
  7256. NewFillBuffer(0)
  7257. else
  7258. NewFillBuffer(aCount - 1);
  7259. LPtrBytes := PByteArray(@PUndoRec.Buffer);
  7260. if aCount > 0 then
  7261. FEditor.ReadBuffer(LPtrBytes^, 0, aCount);
  7262. if aCount = 0 then
  7263. UpdateUndoRecord
  7264. else
  7265. UpdateUndoRecord(aCount - 1);
  7266. end;
  7267. ufKindCombined:
  7268. begin
  7269. LSStDesc := sDescription;
  7270. NewFillBuffer(Length(LSStDesc));
  7271. PUndoRec.Buffer := aCount;
  7272. if FEditor.HasChanged(aPosition) then
  7273. Include(PUndoRec.Flags, ufFlagByte1Changed);
  7274. Move(LSStDesc[0], PUndoRec^.Buffer, Length(LSStDesc) + 1);
  7275. UpdateUndoRecord(Length(LSStDesc));
  7276. end;
  7277. end;
  7278. end;
  7279. end;
  7280. function TMPHUndoStorage.EndUpdate: integer;
  7281. begin
  7282. Dec(FUpdateCount);
  7283. if FUpdateCount < 0 then
  7284. FUpdateCount := 0;
  7285. Result := FUpdateCount;
  7286. end;
  7287. function TMPHUndoStorage.Undo: boolean;
  7288. procedure PopulateUndo(const aBuffer: TMPHUndoRec);
  7289. var
  7290. LRecSel: TUndoSelRec;
  7291. begin
  7292. with FEditor.GetCursorAtPos(aBuffer.CurPos, ufFlagInCharField in
  7293. aBuffer.Flags) do
  7294. begin
  7295. if not (ufFlagInCharField in aBuffer.Flags) then
  7296. if FEditor.DataSize > 0 then
  7297. if ufFlag2ndByteCol in aBuffer.Flags then
  7298. x := x + 1;
  7299. FEditor.MoveColRow(x, y, True, True);
  7300. end;
  7301. FEditor.FModified := ufFlagModified in aBuffer.Flags;
  7302. FEditor.InsertMode := (ufFlagInsertMode in aBuffer.Flags);
  7303. if ufFlagHasSelection in aBuffer.Flags then
  7304. begin
  7305. Position := Size - 4 - sizeof(LRecSel);
  7306. Read(LRecSel, sizeof(LRecSel));
  7307. with LRecSel do
  7308. begin
  7309. if SelEnd = -1 then
  7310. FEditor.Seek(SelStart, FILE_BEGIN)
  7311. else
  7312. FEditor.SetSelection(SelPos, SelStart, SelEnd);
  7313. end;
  7314. end;
  7315. FEditor.UnicodeChars := (ufFlagIsUnicode in aBuffer.Flags);
  7316. FEditor.UnicodeBigEndian := (ufFlagIsUnicodeBigEndian in aBuffer.Flags);
  7317. if not FEditor.UnicodeChars then
  7318. FEditor.Translation := aBuffer.CurTranslation
  7319. else
  7320. FEditor.FTranslation := aBuffer.CurTranslation;
  7321. FEditor.BytesPerUnit := aBuffer.CurBPU;
  7322. FEditor.Invalidate;
  7323. FEditor.Changed;
  7324. end;
  7325. var
  7326. LEnumUndo: TMPHUndoFlag;
  7327. LRecUndo: TMPHUndoRec;
  7328. LIntLoop: integer;
  7329. s: string;
  7330. begin
  7331. Result := False;
  7332. if not CanUndo then
  7333. begin
  7334. Reset(False);
  7335. Exit;
  7336. end;
  7337. if Size >= sizeof(TMPHUndoRec) then
  7338. begin
  7339. // letzten eintrag lesen
  7340. LEnumUndo := ReadUndoRecord(LRecUndo, s);
  7341. // redo erstellen
  7342. CreateRedo(LRecUndo);
  7343. case LEnumUndo of
  7344. ufKindBytesChanged:
  7345. begin
  7346. FEditor.WriteBuffer(PChar(Memory)[Position - 1], LRecUndo.Pos,
  7347. LRecUndo.Count);
  7348. FEditor.SetChanged(LRecUndo.Pos, ufFlagByte1Changed in
  7349. LRecUndo.Flags);
  7350. if LRecUndo.Count = 2 then
  7351. FEditor.SetChanged(LRecUndo.Pos + 1, ufFlagByte2Changed in
  7352. LRecUndo.Flags);
  7353. PopulateUndo(LRecUndo);
  7354. FEditor.RedrawPos(LRecUndo.Pos, LRecUndo.Pos + LRecUndo.Count - 1);
  7355. RemoveLastUndo;
  7356. end;
  7357. ufKindByteRemoved:
  7358. begin
  7359. FEditor.InternalInsertBuffer(Pointer(integer(Memory) + Position - 1),
  7360. LRecUndo.Count, LRecUndo.Pos);
  7361. PopulateUndo(LRecUndo);
  7362. FEditor.AdjustBookmarks(LRecUndo.Pos - LRecUndo.Count,
  7363. LRecUndo.Count);
  7364. if DWORD(FEditor.FModifiedBytes.Size) >= (LRecUndo.Pos) then
  7365. FEditor.FModifiedBytes.Size := LRecUndo.Pos;
  7366. FEditor.Invalidate;
  7367. RemoveLastUndo;
  7368. end;
  7369. ufKindInsertBuffer:
  7370. begin
  7371. FEditor.InternalDelete(LRecUndo.Pos, LRecUndo.Pos + LRecUndo.Count,
  7372. -1, 0);
  7373. PopulateUndo(LRecUndo);
  7374. FEditor.AdjustBookmarks(LRecUndo.Pos, -LRecUndo.Count);
  7375. if DWORD(FEditor.FModifiedBytes.Size) >= (LRecUndo.Pos) then
  7376. FEditor.FModifiedBytes.Size := LRecUndo.Pos;
  7377. FEditor.Invalidate;
  7378. RemoveLastUndo;
  7379. end;
  7380. ufKindSelection:
  7381. begin
  7382. PopulateUndo(LRecUndo);
  7383. RemoveLastUndo;
  7384. end;
  7385. ufKindAllData:
  7386. begin
  7387. FEditor.FDataStorage.Size := LRecUndo.Count;
  7388. FEditor.FDataStorage.WriteBufferAt(Pointer(integer(Memory) + Position
  7389. - 1)^, 0,
  7390. LRecUndo.Count);
  7391. FEditor.CalcSizes;
  7392. PopulateUndo(LRecUndo);
  7393. RemoveLastUndo;
  7394. end;
  7395. ufKindReplace:
  7396. begin
  7397. FEditor.InternalDelete(LRecUndo.Pos, LRecUndo.Pos + LRecUndo.Count,
  7398. -1, 0);
  7399. FEditor.InternalInsertBuffer(Pointer(integer(Memory) + Position - 1),
  7400. LRecUndo.ReplCount, LRecUndo.Pos);
  7401. PopulateUndo(LRecUndo);
  7402. FEditor.AdjustBookmarks(LRecUndo.Pos + LRecUndo.ReplCount,
  7403. LRecUndo.ReplCount - LRecUndo.Count);
  7404. if DWORD(FEditor.FModifiedBytes.Size) >= (LRecUndo.Pos) then
  7405. // was:
  7406. // FEditor.FModifiedBytes.Size := Max(0, LRecUndo.Pos - 1);
  7407. // line above might lead to an integer overflow
  7408. begin
  7409. if LRecUndo.Pos > 0 then
  7410. FEditor.FModifiedBytes.Size := LRecUndo.Pos - 1
  7411. else
  7412. FEditor.FModifiedBytes.Size := 0;
  7413. end;
  7414. FEditor.Invalidate;
  7415. RemoveLastUndo;
  7416. end;
  7417. ufKindAppendBuffer:
  7418. begin
  7419. FEditor.Col := GRID_FIXED;
  7420. FEditor.FDataStorage.Size := LRecUndo.Pos;
  7421. FEditor.CalcSizes;
  7422. if DWORD(FEditor.FModifiedBytes.Size) >= (LRecUndo.Pos) then
  7423. FEditor.FModifiedBytes.Size := LRecUndo.Pos;
  7424. PopulateUndo(LRecUndo);
  7425. FEditor.Invalidate;
  7426. RemoveLastUndo;
  7427. end;
  7428. ufKindNibbleInsert:
  7429. begin
  7430. FEditor.InternalDeleteNibble(LRecUndo.Pos, False);
  7431. FEditor.Data[LRecUndo.Pos] := LRecUndo.Buffer;
  7432. FEditor.SetChanged(LRecUndo.Pos, ufFlagByte1Changed in
  7433. LRecUndo.Flags);
  7434. PopulateUndo(LRecUndo);
  7435. if DWORD(FEditor.FModifiedBytes.Size) >= (LRecUndo.Pos) then
  7436. FEditor.FModifiedBytes.Size := LRecUndo.Pos;
  7437. FEditor.FDataStorage.Size := FEditor.FDataStorage.Size - 1;
  7438. FEditor.CalcSizes;
  7439. FEditor.Invalidate;
  7440. RemoveLastUndo;
  7441. end;
  7442. ufKindNibbleDelete:
  7443. begin
  7444. FEditor.InternalInsertNibble(LRecUndo.Pos, False);
  7445. FEditor.Data[LRecUndo.Pos] := LRecUndo.Buffer;
  7446. FEditor.SetChanged(LRecUndo.Pos, ufFlagByte1Changed in
  7447. LRecUndo.Flags);
  7448. PopulateUndo(LRecUndo);
  7449. if DWORD(FEditor.FModifiedBytes.Size) >= (LRecUndo.Pos) then
  7450. FEditor.FModifiedBytes.Size := LRecUndo.Pos;
  7451. FEditor.FDataStorage.Size := FEditor.FDataStorage.Size - 1;
  7452. FEditor.CalcSizes;
  7453. FEditor.Invalidate;
  7454. RemoveLastUndo;
  7455. end;
  7456. ufKindConvert:
  7457. begin
  7458. FEditor.WriteBuffer(PChar(Memory)[Position - 1], LRecUndo.Pos,
  7459. LRecUndo.Count);
  7460. PopulateUndo(LRecUndo);
  7461. if DWORD(FEditor.FModifiedBytes.Size) >= (LRecUndo.Pos) then
  7462. FEditor.FModifiedBytes.Size := LRecUndo.Pos;
  7463. FEditor.Invalidate;
  7464. RemoveLastUndo;
  7465. end;
  7466. ufKindCombined:
  7467. begin
  7468. LIntLoop := LRecUndo.Count;
  7469. RemoveLastUndo;
  7470. for LIntLoop := 1 to LIntLoop do
  7471. Undo;
  7472. ResetRedo;
  7473. end;
  7474. end;
  7475. end
  7476. else
  7477. Reset;
  7478. end;
  7479. procedure TMPHUndoStorage.RemoveLastUndo;
  7480. var
  7481. LRecUndo: TMPHUndoRec;
  7482. LSStDesc: shortstring;
  7483. LIntRecOffs: integer;
  7484. begin
  7485. if Size < sizeof(TMPHUndoRec) then
  7486. Reset(False)
  7487. else
  7488. begin
  7489. Position := Size - 4;
  7490. Read(LIntRecOffs, 4);
  7491. // restore record in case of a redo
  7492. Seek(-LIntRecOffs, soFromCurrent);
  7493. ReAllocMem(FLastUndo, LIntRecOffs);
  7494. Read(FLastUndo^, LIntRecOffs);
  7495. FLastUndoSize := LIntRecOffs;
  7496. FLastUndoDesc := FDescription;
  7497. // delete last undo record
  7498. SetSize(Max(0, Size - LIntRecOffs));
  7499. Dec(FCount);
  7500. if Size < sizeof(TMPHUndoRec) then
  7501. begin
  7502. Reset(False);
  7503. end
  7504. else
  7505. begin
  7506. if ReadUndoRecord(LRecUndo, FDescription) <> ufKindCombined then
  7507. begin
  7508. if FDescription = '' then
  7509. FDescription := STRS_UNDODESC[GetUndoKind(LRecUndo.Flags)]
  7510. end
  7511. else
  7512. begin
  7513. if LRecUndo.Buffer = 0 then
  7514. LSStDesc := ''
  7515. else
  7516. begin
  7517. Read(LSStDesc[1], LRecUndo.Buffer);
  7518. LSStDesc[0] := char(LRecUndo.Buffer);
  7519. end;
  7520. if LSStDesc = '' then
  7521. FDescription := STRS_UNDODESC[GetUndoKind(LRecUndo.Flags)]
  7522. else
  7523. FDescription := LSStDesc;
  7524. end;
  7525. end;
  7526. end;
  7527. end;
  7528. procedure TMPHUndoStorage.SetSize(NewSize: integer);
  7529. begin
  7530. inherited;
  7531. if NewSize < sizeof(TMPHUndoRec) then
  7532. FCount := 0;
  7533. end;
  7534. procedure TMPHUndoStorage.Reset(AResetRedo: boolean = True);
  7535. begin
  7536. Size := 0;
  7537. FCount := 0;
  7538. FUpdateCount := 0;
  7539. FDescription := '';
  7540. if AResetRedo then
  7541. ResetRedo;
  7542. end;
  7543. procedure TMPHUndoStorage.SetCount(const Value: integer);
  7544. begin
  7545. FCount := Value;
  7546. if FCount < 1 then
  7547. Reset(False);
  7548. end;
  7549. function TMPHUndoStorage.CanRedo: boolean;
  7550. begin
  7551. Result := Assigned(FRedoPointer);
  7552. end;
  7553. function TMPHUndoStorage.Redo: boolean;
  7554. procedure SetEditorStateFromRedoRec(const _2Bytes: Boolean = False);
  7555. begin
  7556. with FRedoPointer^ do
  7557. begin
  7558. Move(PChar(FRedoPointer)[FRedoPointer^.DataLen], FEditor.FBookmarks,
  7559. sizeof(TMPHBookmarks));
  7560. with FEditor.GetCursorAtPos(CurPos, ufFlagInCharField in Flags) do
  7561. begin
  7562. if not (ufFlagInCharField in Flags) then
  7563. if FEditor.DataSize > 0 then
  7564. if ufFlag2ndByteCol in Flags then
  7565. x := x + 1;
  7566. FEditor.MoveColRow(x, y, True, True);
  7567. end;
  7568. FEditor.FModified := ufFlagModified in Flags;
  7569. FEditor.InsertMode := (ufFlagInsertMode in Flags);
  7570. with PUndoSelRec(@(PChar(FRedoPointer)[FRedoPointer^.DataLen +
  7571. sizeof(TMPHBookmarks)]))^ do
  7572. FEditor.SetSelection(SelPos, SelStart, SelEnd);
  7573. FEditor.Translation := CurTranslation;
  7574. FEditor.FTranslation := CurTranslation;
  7575. FEditor.UnicodeChars := (ufFlagIsUnicode in Flags);
  7576. FEditor.UnicodeBigEndian := (ufFlagIsUnicodeBigEndian in Flags);
  7577. FEditor.BytesPerUnit := CurBPU;
  7578. FEditor.InCharField := ufFlagInCharField in Flags;
  7579. FEditor.SetChanged(Pos, ufFlagByte1Changed in Flags);
  7580. if _2Bytes then
  7581. FEditor.SetChanged(Pos + 1, ufFlagByte2Changed in Flags);
  7582. // restore last undo record
  7583. if Assigned(FLastUndo) then
  7584. begin
  7585. Seek(0, soFromEnd);
  7586. Write(FLastUndo^, FLastUndoSize);
  7587. Inc(FCount);
  7588. FreeMem(FLastUndo);
  7589. FLastUndo := nil;
  7590. FLastUndoSize := 0;
  7591. end;
  7592. FDescription := FLastUndoDesc;
  7593. FEditor.Invalidate;
  7594. FEditor.BookmarkChanged;
  7595. end;
  7596. end;
  7597. begin
  7598. Result := CanRedo;
  7599. if Result then
  7600. begin
  7601. case GetUndoKind(FRedoPointer^.Flags) of
  7602. ufKindBytesChanged:
  7603. begin
  7604. FEditor.WriteBuffer(FRedoPointer^.Buffer,
  7605. FRedoPointer^.Pos, FRedoPointer^.Count);
  7606. SetEditorStateFromRedoRec(FRedoPointer^.Count = 2);
  7607. end;
  7608. ufKindByteRemoved:
  7609. begin
  7610. FEditor.InternalDelete(FRedoPointer^.Pos,
  7611. FRedoPointer^.Pos + FRedoPointer^.Count, -1, 0);
  7612. SetEditorStateFromRedoRec;
  7613. end;
  7614. ufKindInsertBuffer:
  7615. begin
  7616. FEditor.InternalInsertBuffer(PChar(@(FRedoPointer^.Buffer)),
  7617. FRedoPointer^.Count, FRedoPointer^.Pos);
  7618. SetEditorStateFromRedoRec;
  7619. end;
  7620. ufKindSelection:
  7621. begin
  7622. SetEditorStateFromRedoRec;
  7623. end;
  7624. ufKindAllData:
  7625. begin
  7626. FEditor.FDataStorage.Size := FRedoPointer^.Count;
  7627. FEditor.FDataStorage.WriteBufferAt(FRedoPointer^.Buffer, 0,
  7628. FRedoPointer^.Count);
  7629. FEditor.CalcSizes;
  7630. SetEditorStateFromRedoRec;
  7631. end;
  7632. ufKindReplace:
  7633. begin
  7634. FEditor.InternalDelete(FRedoPointer^.Pos,
  7635. FRedoPointer^.Pos + FRedoPointer^.ReplCount, -1, 0);
  7636. FEditor.InternalInsertBuffer(PChar(@(FRedoPointer^.Buffer)),
  7637. FRedoPointer^.Count, FRedoPointer^.Pos);
  7638. SetEditorStateFromRedoRec;
  7639. end;
  7640. ufKindConvert:
  7641. begin
  7642. FEditor.InternalDelete(FRedoPointer^.Pos,
  7643. FRedoPointer^.Pos + FRedoPointer^.Count, -1, 0);
  7644. FEditor.InternalInsertBuffer(PChar(@(FRedoPointer^.Buffer)),
  7645. FRedoPointer^.Count, FRedoPointer^.Pos);
  7646. SetEditorStateFromRedoRec;
  7647. end;
  7648. ufKindAppendBuffer:
  7649. begin
  7650. FEditor.InternalAppendBuffer(PChar(@(FRedoPointer^.Buffer)),
  7651. FRedoPointer^.Count);
  7652. SetEditorStateFromRedoRec;
  7653. end;
  7654. ufKindNibbleInsert,
  7655. ufKindNibbleDelete:
  7656. begin
  7657. FEditor.FDataStorage.Size := FRedoPointer^.Count;
  7658. FEditor.FDataStorage.WriteBufferAt(FRedoPointer^.Buffer, 0,
  7659. FRedoPointer^.Count);
  7660. FEditor.CalcSizes;
  7661. SetEditorStateFromRedoRec;
  7662. end;
  7663. end;
  7664. ResetRedo;
  7665. FEditor.Changed;
  7666. end;
  7667. end;
  7668. procedure TMPHUndoStorage.ResetRedo;
  7669. begin
  7670. if Assigned(FRedoPointer) then
  7671. FreeMem(FRedoPointer);
  7672. FRedoPointer := nil;
  7673. if Assigned(FLastUndo) then
  7674. FreeMem(FLastUndo);
  7675. FLastUndo := nil;
  7676. FLastUndoSize := 0;
  7677. FLastUndoDesc := '';
  7678. end;
  7679. procedure TMPHUndoStorage.CreateRedo(const Rec: TMPHUndoRec);
  7680. var
  7681. LIntDataSize: integer;
  7682. procedure AllocRedoPointer;
  7683. begin
  7684. GetMem(FRedoPointer, sizeof(TMPHUndoRec) + sizeof(TMPHBookMarks) +
  7685. sizeof(TUndoSelRec) + LIntDataSize);
  7686. FRedoPointer^.Flags := [GetUndoKind(Rec.Flags)];
  7687. FRedoPointer^.DataLen := sizeof(TMPHUndoRec) + LIntDataSize;
  7688. end;
  7689. procedure FinishRedoPointer;
  7690. begin
  7691. with FRedoPointer^ do
  7692. begin
  7693. CurPos := FEditor.GetPosAtCursor(FEditor.Col, FEditor.Row);
  7694. if not FEditor.FPosInCharField then
  7695. with FEditor.GetCursorAtPos(CurPos, FEditor.FPosInCharField) do
  7696. if (FEditor.Col - x) <> 0 then
  7697. Include(Flags, ufFlag2ndByteCol);
  7698. if FEditor.FPosInCharField then
  7699. Include(Flags, ufFlagInCharField);
  7700. if FEditor.FInsertModeOn then
  7701. Include(Flags, ufFlagInsertMode);
  7702. Pos := Rec.pos;
  7703. Count := Rec.Count;
  7704. ReplCount := Rec.ReplCount;
  7705. CurTranslation := FEditor.FTranslation;
  7706. if FEditor.UnicodeChars then
  7707. Include(Flags, ufFlagIsUnicode);
  7708. if FEditor.UnicodeBigEndian then
  7709. Include(Flags, ufFlagIsUnicodeBigEndian);
  7710. CurBPU := FEditor.BytesPerUnit;
  7711. if FEditor.FModified then
  7712. Include(Flags, ufFlagModified);
  7713. end;
  7714. Move(FEditor.FBookmarks, PChar(FRedoPointer)[FRedoPointer^.DataLen],
  7715. sizeof(TMPHBookmarks));
  7716. with PUndoSelRec(@(PChar(FRedoPointer)[FRedoPointer^.DataLen +
  7717. sizeof(TMPHBookmarks)]))^ do
  7718. begin
  7719. SelStart := FEditor.FSelStart;
  7720. SelPos := FEditor.FSelPosition;
  7721. SelEnd := FEditor.FSelEnd;
  7722. end;
  7723. end;
  7724. begin
  7725. ResetRedo;
  7726. // simple redo, store bookmarks, selection, insertmode, col, row, charfield...
  7727. // and bytes to save
  7728. case GetUndoKind(Rec.Flags) of
  7729. ufKindBytesChanged:
  7730. begin
  7731. LIntDataSize := Rec.Count - 1;
  7732. AllocRedoPointer;
  7733. if FEditor.HasChanged(Rec.Pos) then
  7734. Include(FRedoPointer^.Flags, ufFlagByte1Changed);
  7735. if Rec.Count = 2 then
  7736. if FEditor.HasChanged(Rec.Pos + 1) then
  7737. Include(FRedoPointer^.Flags, ufFlagByte2Changed);
  7738. FEditor.ReadBuffer(FRedoPointer^.Buffer, Rec.Pos, Rec.Count);
  7739. FinishRedoPointer;
  7740. end;
  7741. ufKindByteRemoved:
  7742. begin
  7743. LIntDataSize := 0;
  7744. AllocRedoPointer;
  7745. FinishRedoPointer;
  7746. end;
  7747. ufKindInsertBuffer,
  7748. ufKindReplace,
  7749. ufKindConvert:
  7750. begin
  7751. LIntDataSize := Rec.Count;
  7752. AllocRedoPointer;
  7753. FEditor.ReadBuffer(FRedoPointer^.Buffer, Rec.Pos, Rec.Count);
  7754. FinishRedoPointer;
  7755. end;
  7756. ufKindSelection:
  7757. begin
  7758. LIntDataSize := 0;
  7759. AllocRedoPointer;
  7760. FinishRedoPointer;
  7761. end;
  7762. ufKindAllData:
  7763. begin
  7764. LIntDataSize := FEditor.DataSize;
  7765. AllocRedoPointer;
  7766. FEditor.ReadBuffer(FRedoPointer^.Buffer, 0, FEditor.DataSize);
  7767. FinishRedoPointer;
  7768. FRedoPointer^.Count := FEditor.DataSize;
  7769. end;
  7770. ufKindAppendBuffer:
  7771. begin
  7772. LIntDataSize := FEditor.DataSize - integer(Rec.Pos);
  7773. AllocRedoPointer;
  7774. FEditor.ReadBuffer(FRedoPointer^.Buffer, Rec.Pos, FEditor.DataSize -
  7775. integer(Rec.Pos));
  7776. FinishRedoPointer;
  7777. end;
  7778. ufKindNibbleInsert,
  7779. ufKindNibbleDelete:
  7780. begin
  7781. LIntDataSize := FEditor.DataSize;
  7782. AllocRedoPointer;
  7783. FEditor.ReadBuffer(FRedoPointer^.Buffer, 0, FEditor.DataSize);
  7784. FinishRedoPointer;
  7785. FRedoPointer^.Count := LIntDataSize;
  7786. end;
  7787. end;
  7788. //FEditor.Changed;
  7789. end;
  7790. function TMPHUndoStorage.GetUndoKind(const Flags: TMPHUndoFlags): TMPHUndoFlag;
  7791. begin
  7792. for Result := ufKindBytesChanged to ufKindAllData do
  7793. if Result in Flags then
  7794. Break;
  7795. end;
  7796. procedure TMPHUndoStorage.AddSelection(const APos, ACount: integer);
  7797. var
  7798. P: PMPHUndoRec;
  7799. PSel: PUndoSelRec;
  7800. LIntRecOffset: integer;
  7801. begin
  7802. if CanUndo then
  7803. begin
  7804. Position := Size - 4;
  7805. Read(LIntRecOffset, 4);
  7806. Seek(-LIntRecOffset, soFromCurrent);
  7807. P := Pointer(Integer(Memory) + Position);
  7808. if not (ufFlagHasSelection in P^.Flags) then
  7809. begin
  7810. Size := Size + SizeOf(TUndoSelRec);
  7811. P := Pointer(Integer(Memory) + Position);
  7812. Include(P^.Flags, ufFlagHasSelection);
  7813. Inc(P^.DataLen, sizeof(TUndoSelRec));
  7814. Inc(LIntRecOffset, sizeof(TUndoSelRec));
  7815. Seek(-4, soFromEnd);
  7816. WriteBuffer(LIntRecOffset, 4);
  7817. end;
  7818. P^.CurPos := APos;
  7819. PSel := Pointer(Integer(Memory) + size - 4 - sizeof(TUndoSelRec));
  7820. PSel^.SelStart := APos;
  7821. if aCount = 0 then
  7822. PSel^.SelEnd := -1
  7823. else
  7824. PSel^.SelEnd := APos + Acount - 1;
  7825. PSel^.SelPos := PSel^.SelStart;
  7826. end;
  7827. end;
  7828. function TMPHUndoStorage.ReadUndoRecord(
  7829. var aUR: TMPHUndoRec; var SDescription: string): TMPHUndoFlag;
  7830. var
  7831. LIntRecOffs: integer;
  7832. LIntPos: integer;
  7833. begin
  7834. Position := Size - 4;
  7835. Read(LIntRecOffs, 4);
  7836. Seek(-LIntRecOffs, soFromCurrent);
  7837. Read(aUR, SizeOf(TMPHUndoRec));
  7838. Result := GetUndoKind(aUr.Flags);
  7839. if ufFlagHasDescription in aUr.Flags then
  7840. begin
  7841. LIntPos := Position;
  7842. try
  7843. Position := size - 4 - sizeof(integer);
  7844. if ufFlagHasSelection in aUr.Flags then
  7845. Seek(-sizeof(TUndoSelRec), soFromCurrent);
  7846. Read(LIntRecOffs, sizeof(integer));
  7847. Seek(-(LIntRecOffs + sizeof(integer)), soFromCurrent);
  7848. SetLength(SDescription, LIntRecOffs);
  7849. Read(SDescription[1], LIntRecOffs);
  7850. finally
  7851. Position := LIntPos;
  7852. end;
  7853. end
  7854. else
  7855. SDescription := '';
  7856. end;
  7857. function TMPHUndoStorage.GetLastUndoKind: TMPHUndoFlag;
  7858. var
  7859. recUndo: TMPHUndoRec;
  7860. s: string;
  7861. begin
  7862. Result := ReadUndoRecord(recUndo, s);
  7863. end;
  7864. // initialize tkCustom translation tables
  7865. procedure InitializeCustomTables;
  7866. var
  7867. LBytLoop: byte;
  7868. begin
  7869. for LBytLoop := 0 to 255 do
  7870. begin
  7871. MPHCustomCharConv[cctFromAnsi][LBytLoop] := char(LBytLoop);
  7872. MPHCustomCharConv[cctToAnsi][LBytLoop] := char(LBytLoop);
  7873. end;
  7874. end;
  7875. { TMPHMemoryStream }
  7876. const
  7877. MAX_PER_BLOCK = $F000;
  7878. procedure TMPHMemoryStream.CheckBounds(const AMax: Integer);
  7879. begin
  7880. if (AMax < 0) or (AMax > Size) then
  7881. raise EMPHexEditor.Create(ERR_DATA_BOUNDS);
  7882. end;
  7883. {$IFDEF FASTACCESS}
  7884. function TMPHMemoryStream.GetAddress(const Index, Count: integer): PByte;
  7885. begin
  7886. if (Index < 0) or ((Index+Count) > Size) then
  7887. raise EMPHexEditor.Create(ERR_DATA_BOUNDS);
  7888. Result := Pointer(Integer(Memory)+Index);
  7889. end;
  7890. {$ENDIF}
  7891. function TMPHMemoryStream.GetAsHex(const APosition, ACount: integer;
  7892. const SwapNibbles: Boolean): string;
  7893. begin
  7894. CheckBounds(APosition + ACount);
  7895. SetLength(Result, ACount * 2);
  7896. if ACount > 0 then
  7897. ConvertBinToHex(PointerAt(APosition), @Result[1], ACount, SwapNibbles);
  7898. end;
  7899. procedure TMPHMemoryStream.Move(const AFromPos, AToPos, ACount: Integer);
  7900. begin
  7901. MoveMemory(PointerAt(AToPos), PointerAt(AFromPos), ACount);
  7902. end;
  7903. function TMPHMemoryStream.PointerAt(const APosition: Integer): Pointer;
  7904. begin
  7905. Result := Pointer(LongInt(Memory) + APosition);
  7906. end;
  7907. procedure TMPHMemoryStream.ReadBufferAt(var Buffer; const APosition,
  7908. ACount: Integer);
  7909. var
  7910. LIntPos: Integer;
  7911. begin
  7912. CheckBounds(APosition + ACount);
  7913. LIntPos := Position;
  7914. try
  7915. Position := APosition;
  7916. ReadBuffer(Buffer, ACount);
  7917. finally
  7918. Position := LIntPos;
  7919. end;
  7920. end;
  7921. procedure TMPHMemoryStream.TranslateFromAnsi(const ToTranslation:
  7922. TMPHTranslationKind; const APosition, ACount: integer);
  7923. begin
  7924. if ToTranslation = tkAsIs then
  7925. Exit; // no translation needed
  7926. CheckBounds(APosition + ACount);
  7927. if ACount > 0 then
  7928. TranslateBufferFromAnsi(ToTranslation, PointerAt(APosition),
  7929. PointerAt(APosition), ACount);
  7930. end;
  7931. procedure TMPHMemoryStream.TranslateToAnsi(const FromTranslation:
  7932. TMPHTranslationKind; const APosition, ACount: integer);
  7933. begin
  7934. if FromTranslation = tkAsIs then
  7935. Exit; // no translation needed
  7936. CheckBounds(APosition + ACount);
  7937. if ACount > 0 then
  7938. TranslateBufferToAnsi(FromTranslation, PointerAt(APosition),
  7939. PointerAt(APosition), ACount);
  7940. end;
  7941. procedure TMPHMemoryStream.WriteBufferAt(const Buffer; const APosition,
  7942. ACount: Integer);
  7943. var
  7944. LIntPos: Integer;
  7945. begin
  7946. CheckBounds(APosition + ACount);
  7947. LIntPos := Position;
  7948. try
  7949. Position := APosition;
  7950. WriteBuffer(Buffer, ACount);
  7951. finally
  7952. Position := LIntPos;
  7953. end;
  7954. end;
  7955. initialization
  7956. // initialize custom tables
  7957. InitializeCustomTables;
  7958. end.