1 | unit rHTMLTools;
|
---|
2 | (*This entire unit was created by K. Toppenberg, starting on 5/27/05
|
---|
3 | It will hold additional functions to support HTML display of notes
|
---|
4 | and printing of HTML notes. *)
|
---|
5 |
|
---|
6 | interface
|
---|
7 |
|
---|
8 | uses Windows, SysUtils, Classes, Printers, ComCtrls,
|
---|
9 | ShDocVw, {//kt added ShDocVw 5-2-05 for TWebBrowser access}
|
---|
10 | Dialogs,
|
---|
11 | Forms,
|
---|
12 | Registry, {elh 6/19/09}
|
---|
13 | ORFn; {//kt for RedrawActivate}
|
---|
14 |
|
---|
15 | var
|
---|
16 | DesiredHTMLFontSize : byte;
|
---|
17 |
|
---|
18 | procedure PrintHTMLReport(Lines: TStringList; var ErrMsg: string;
|
---|
19 | PtName, DOB, VisitDate, Location:string; Application : TApplication); //kt added 5-2-05
|
---|
20 | function IsHTML(Lines : TStrings): boolean; overload;
|
---|
21 | function IsHTML(Line : String): boolean; overload;
|
---|
22 | function HasHTMLTags(Text: string) : boolean;
|
---|
23 | procedure FixHTML(Lines : TStrings);
|
---|
24 | function FixHTMLCRLF(Text : String) : string;
|
---|
25 | procedure SplitToArray (HTMLText: string; Lines : TStrings);
|
---|
26 | procedure StripBeforeAfterHTML(Lines,OutBefore,OutAfter : TStrings);
|
---|
27 | function UnwrapHTML(HTMLText : string) : string;
|
---|
28 | function WrapHTML(HTMLText : string) : string;
|
---|
29 | // function WaitForBrowserOK(MaxSecDelay: integer; Application : TApplication) : boolean;
|
---|
30 | function CheckForImageLink(Lines: TStrings; ImageList : TStringList) : boolean;
|
---|
31 | function ProtectHTMLSpaces(Text : String) : string;
|
---|
32 | function Text2HTML(Lines : TStrings) : String; overload;
|
---|
33 | function Text2HTML(text : string) : String; overload;
|
---|
34 | procedure SetRegHTMLFontSize(Size: byte);
|
---|
35 | procedure RestoreRegHTMLFontSize;
|
---|
36 | procedure SetupHTMLPrinting(Name,DOB,VisitDate,Location,Institution : string);
|
---|
37 | procedure RestoreIEPrinting;
|
---|
38 | function ExtractDateOfNote(Lines : TStringList) : string;
|
---|
39 |
|
---|
40 | implementation
|
---|
41 |
|
---|
42 | uses fNotes,
|
---|
43 | fImages,
|
---|
44 | Messages,
|
---|
45 | Graphics, //For color constants
|
---|
46 | fTMGPrintingAnimation,
|
---|
47 | ExtCtrls,
|
---|
48 | StrUtils;
|
---|
49 |
|
---|
50 | type
|
---|
51 | TPrinterEvents = class
|
---|
52 | public
|
---|
53 | SavedDefaultPrinter : string;
|
---|
54 | LastChosenPrinterName : string;
|
---|
55 | RestorePrinterTimer : TTimer;
|
---|
56 | PrintingNow : boolean;
|
---|
57 | procedure HandleRestorePrinting (Sender: TObject);
|
---|
58 | Constructor Create;
|
---|
59 | Destructor Destroy; override;
|
---|
60 | end;
|
---|
61 |
|
---|
62 | var
|
---|
63 | PrinterEvents : TPrinterEvents;
|
---|
64 |
|
---|
65 | const CRLF = #$0D#$0A;
|
---|
66 |
|
---|
67 |
|
---|
68 | function GetCurrentPrinterName : string;
|
---|
69 | //var ResStr: array[0..255] of Char;
|
---|
70 | begin
|
---|
71 | //GetProfileString('Windows', 'device', '', ResStr, 255);
|
---|
72 | //Result := StrPas(ResStr);
|
---|
73 | if (Printer.PrinterIndex > 0)then begin
|
---|
74 | Result := Printer.Printers[Printer.PrinterIndex];
|
---|
75 | end else begin
|
---|
76 | Result := '';
|
---|
77 | end;
|
---|
78 | end;
|
---|
79 |
|
---|
80 | procedure SetDefaultPrinter(PrinterName: String) ;
|
---|
81 | var
|
---|
82 | j : Integer;
|
---|
83 | Device, Driver, Port : PChar;
|
---|
84 | HdeviceMode : THandle;
|
---|
85 | aPrinter : TPrinter;
|
---|
86 | begin
|
---|
87 | Printer.PrinterIndex := -1;
|
---|
88 | getmem(Device, 255) ;
|
---|
89 | getmem(Driver, 255) ;
|
---|
90 | getmem(Port, 255) ;
|
---|
91 | aPrinter := TPrinter.create;
|
---|
92 | j := Printer.Printers.IndexOf(PrinterName);
|
---|
93 | if j >= 0 then begin
|
---|
94 | aprinter.printerindex := j;
|
---|
95 | aPrinter.getprinter(device, driver, port, HdeviceMode) ;
|
---|
96 | StrCat(Device, ',') ;
|
---|
97 | StrCat(Device, Driver ) ;
|
---|
98 | StrCat(Device, Port ) ;
|
---|
99 | WriteProfileString('windows', 'device', Device) ;
|
---|
100 | StrCopy( Device, 'windows' ) ;
|
---|
101 | //SendMessage(HWND_BROADCAST, WM_WININICHANGE, 0, Longint(@Device)) ;
|
---|
102 | end;
|
---|
103 | Freemem(Device, 255) ;
|
---|
104 | Freemem(Driver, 255) ;
|
---|
105 | Freemem(Port, 255) ;
|
---|
106 | aPrinter.Free;
|
---|
107 | end;
|
---|
108 |
|
---|
109 |
|
---|
110 | procedure Wait(Sec : byte; Application : TApplication);
|
---|
111 | var StartTime : TDateTime;
|
---|
112 | const OneSec = 0.000012;
|
---|
113 | begin
|
---|
114 | StartTime := GetTime;
|
---|
115 | repeat
|
---|
116 | Application.ProcessMessages;
|
---|
117 | until (GetTime-StartTime) > (OneSec*Sec);
|
---|
118 | end;
|
---|
119 |
|
---|
120 |
|
---|
121 | procedure PrintHTMLReport(Lines: TStringList; var ErrMsg: string;
|
---|
122 | PtName, DOB, VisitDate, Location:string;
|
---|
123 | Application : TApplication);
|
---|
124 | // Web browser printing options:
|
---|
125 | // OLECMDEXECOPT_DODEFAULT Use the default behavior, whether prompting the user for input or not.
|
---|
126 | // OLECMDEXECOPT_PROMPTUSER Execute the command after obtaining user input.
|
---|
127 | // OLECMDEXECOPT_DONTPROMPTUSER Execute the command without prompting the user.
|
---|
128 |
|
---|
129 | {Notice: When IE is asked to print, it immediately returns from the function,
|
---|
130 | but the printing has not yet occured. If UI is requested, then the
|
---|
131 | printing will not start until after the user selects a printer and
|
---|
132 | presses [OK]. I could not find any reliable way to determine when the
|
---|
133 | print job had been created. I had to know this event because I need to
|
---|
134 | restore some IE settings AFTER the printing has finished. I even tried to
|
---|
135 | get the active windows and see if it was a print dialog. But IE print dlg
|
---|
136 | apparently is owned by another thread than CPRS, because GetActiveWindow would
|
---|
137 | not bring back a handle to the printer dialog window. I therefore told IE
|
---|
138 | to print WITHOUT asking which printer via UI. In that case it prints to the
|
---|
139 | system wide default printer. So I have to set the default printer to the
|
---|
140 | user's choice, and then change it back again. This is bit of a kludge,
|
---|
141 | but I couldn't figure out any other way after hours of trial and error.
|
---|
142 | NOTE: I tried to query IE to see if it was able to print, thinking that it
|
---|
143 | would return NO if in the process of currently printing. It didn't work,
|
---|
144 | and would return OK immediately.
|
---|
145 |
|
---|
146 | ADDENDUM: I was getting errors and inconsistent behavior with this, so I
|
---|
147 | have decided to try to let the user click a button when the printer has
|
---|
148 | been selected. }
|
---|
149 |
|
---|
150 | var
|
---|
151 | UseUI : OleVariant;
|
---|
152 | //NewPrinterName : string;
|
---|
153 | //dlgWinPrinter : TPrintDialog;
|
---|
154 | begin
|
---|
155 | //if PrinterEvents.RestorePrinterTimer.Enabled = false then begin
|
---|
156 | // PrinterEvents.SavedDefaultPrinter := GetCurrentPrinterName;
|
---|
157 | //end;
|
---|
158 | if PrinterEvents.PrintingNow then exit; // prevent double printing (it has happened)
|
---|
159 |
|
---|
160 | try
|
---|
161 | frmNotes.SetDisplayToHTMLvsText([vmView,vmHTML],Lines); //ActivateHtmlViewer(Lines);
|
---|
162 | if frmNotes.HtmlViewer.WaitForDocComplete = false then begin
|
---|
163 | ErrMsg := 'The web browser timed out trying to set up document.';
|
---|
164 | exit;
|
---|
165 | end;
|
---|
166 | PrinterEvents.PrintingNow := true;
|
---|
167 | SetupHTMLPrinting(PtName,DOB,VisitDate,Location,' '); {elh 6/19/09} //kt
|
---|
168 | frmNotes.HtmlViewer.PrintFinished := false;
|
---|
169 | UseUI := true;
|
---|
170 | frmNotes.HtmlViewer.PrintDocument(UseUI); //Returns immediately, not after printing done.
|
---|
171 | frmTMGPrinting.ShowModal; // Let user show when print job has been launched.
|
---|
172 | PrinterEvents.RestorePrinterTimer.Enabled := true; //launch a restore event in 30 seconds
|
---|
173 | //RestoreIEPrinting; //elh - This was omitted from below. Not sure why. 11/10/09
|
---|
174 | finally
|
---|
175 | PrinterEvents.PrintingNow := false;
|
---|
176 | end;
|
---|
177 | end;
|
---|
178 |
|
---|
179 | (*
|
---|
180 | Safe copy of above. Delete later...
|
---|
181 |
|
---|
182 | procedure PrintHTMLReport(Lines: TStringList; var ErrMsg: string;
|
---|
183 | PtName, DOB, Location:string;
|
---|
184 | Application : TApplication);
|
---|
185 | // Web browser printing options:
|
---|
186 | // OLECMDEXECOPT_DODEFAULT Use the default behavior, whether prompting the user for input or not.
|
---|
187 | // OLECMDEXECOPT_PROMPTUSER Execute the command after obtaining user input.
|
---|
188 | // OLECMDEXECOPT_DONTPROMPTUSER Execute the command without prompting the user.
|
---|
189 |
|
---|
190 | {Notice: When IE is asked to print, it immediately returns from the function,
|
---|
191 | but the printing has not yet occured. If UI is requested, then the
|
---|
192 | printing will not start until after the user selects a printer and
|
---|
193 | presses [OK]. I could not find any reliable way to determine when the
|
---|
194 | print job had been created. I had to know this event because I need to
|
---|
195 | restore some IE settings AFTER the printing has finished. I even tried to
|
---|
196 | get the active windows and see if it was a print dialog. But IE print dlg
|
---|
197 | apparently is owned by another thread than CPRS, because GetActiveWindow would
|
---|
198 | not bring back a handle to the printer dialog window. I therefore told IE
|
---|
199 | to print WITHOUT asking which printer via UI. In that case it prints to the
|
---|
200 | system wide default printer. So I have to set the default printer to the
|
---|
201 | user's choice, and then change it back again. This is bit of a kludge,
|
---|
202 | but I couldn't figure out any other way after hours of trial and error.
|
---|
203 | NOTE: I tried to query IE to see if it was able to print, thinking that it
|
---|
204 | would return NO if in the process of currently printing. It didn't work,
|
---|
205 | and would return OK immediately. }
|
---|
206 |
|
---|
207 |
|
---|
208 |
|
---|
209 | var
|
---|
210 | UseUI : OleVariant;
|
---|
211 | NewPrinterName : string;
|
---|
212 | dlgWinPrinter : TPrintDialog;
|
---|
213 | begin
|
---|
214 | if PrinterEvents.RestorePrinterTimer.Enabled = false then begin
|
---|
215 | PrinterEvents.SavedDefaultPrinter := GetCurrentPrinterName;
|
---|
216 | end;
|
---|
217 | dlgWinPrinter := TPrintDialog.Create(nil);
|
---|
218 | frmTMGPrinting.Show;
|
---|
219 | //FIX: get printer name for the one used last time somehow...
|
---|
220 | if dlgWinPrinter.Execute then begin //only sets a local printer
|
---|
221 | NewPrinterName := GetCurrentPrinterName;
|
---|
222 | SetDefaultPrinter(NewPrinterName); //Set global setting that IE will use.
|
---|
223 | PrinterEvents.LastChosenPrinterName := NewPrinterName;
|
---|
224 | try
|
---|
225 | frmNotes.SetDisplayToHTMLvsText([vmView,vmHTML],Lines); //ActivateHtmlViewer(Lines);
|
---|
226 | if frmNotes.HtmlViewer.WaitForDocComplete = false then begin
|
---|
227 | ErrMsg := 'The web browser timed out trying to set up document.';
|
---|
228 | exit;
|
---|
229 | end;
|
---|
230 | SetupHTMLPrinting(PtName,DOB,Location,' '); {elh 6/19/09} //kt
|
---|
231 | frmNotes.HtmlViewer.PrintFinished := false;
|
---|
232 | UseUI := false; //UseUI := true;
|
---|
233 | frmNotes.HtmlViewer.PrintDocument(UseUI); //Returns immediately, not after printing done.
|
---|
234 | PrinterEvents.RestorePrinterTimer.Enabled := true; //launch a restore event in 30 seconds
|
---|
235 | Wait(4,Application);
|
---|
236 | //WaitForBrowserOK(10, Application); //wait up to 10 seconds //Note: this doesn't do what I want. Status is immediately OK.
|
---|
237 | //RestoreIEPrinting; {elh 6/19/09} //kt
|
---|
238 | finally //any needed final code goes here.
|
---|
239 | //SetDefaultPrinter(DefaultPrinter);
|
---|
240 | //beep;
|
---|
241 | end;
|
---|
242 | end;
|
---|
243 | dlgWinPrinter.Free;
|
---|
244 | frmTMGPrinting.Hide;
|
---|
245 | end;
|
---|
246 | *)
|
---|
247 | (*
|
---|
248 | function WaitForBrowserOK(MaxSecDelay: integer; Application : TApplication) : boolean;
|
---|
249 | //Returns TRUE if can print
|
---|
250 | var
|
---|
251 | StartTime : TDateTime;
|
---|
252 | Status: OLECMDF;
|
---|
253 | MaxDelay,ElapsedTime : Double;
|
---|
254 | CanPrint : boolean;
|
---|
255 | const
|
---|
256 | OneMin = 0.0007; //note: 0.0007 is about 1 minute
|
---|
257 | begin
|
---|
258 | StartTime := GetTime;
|
---|
259 | MaxDelay := OneMin * MaxSecDelay;
|
---|
260 | repeat
|
---|
261 | Status := frmNotes.HtmlViewer.QueryStatusWB(OLECMDID_PRINT); //"can you print?" -- get print command status
|
---|
262 | CanPrint := (Status and OLECMDF_ENABLED) > 0;
|
---|
263 | ElapsedTime := GetTime-StartTime;
|
---|
264 | Application.ProcessMessages;
|
---|
265 | until (ElapsedTime > MaxDelay) or CanPrint or frmNotes.HtmlViewer.PrintFinished;
|
---|
266 | Result := CanPrint;
|
---|
267 | end;
|
---|
268 | *)
|
---|
269 |
|
---|
270 | Procedure ScanForSubs(Lines : TStrings);
|
---|
271 | //Purpose: To scan note for constant $CPRS$ and replace with CPRS's actual directory
|
---|
272 | var i : integer;
|
---|
273 | CPRSDir : string;
|
---|
274 | begin
|
---|
275 | for i := 0 to Lines.Count-1 do begin
|
---|
276 | if Pos('$CPRSDIR$',Lines.Strings[i])>0 then begin
|
---|
277 | CPRSDir := ExcludeTrailingPathDelimiter(ExtractFilePath(ParamStr(0)));
|
---|
278 | Lines.Strings[i] := AnsiReplaceStr(Lines.Strings[i],'$CPRSDIR$',CPRSDir);
|
---|
279 | //Ensure images are downloaded before passing page to web browser
|
---|
280 | frmImages.timLoadImagesTimer(nil);
|
---|
281 | end;
|
---|
282 | end;
|
---|
283 | end;
|
---|
284 |
|
---|
285 |
|
---|
286 | function IsHTML(Line : String): boolean;
|
---|
287 | {Purpose: To look at the Text and determine if it is an HTML document.
|
---|
288 | Test used: if document contains <!DOCTYPE HTML" or <HTML> or </BODY>
|
---|
289 | This is not a fool-proof test...
|
---|
290 | NOTE: **This does NOT call ScanForSubs as the other IsHTML(TStrings) function does. }
|
---|
291 |
|
---|
292 | begin
|
---|
293 | Result := false; //default of false
|
---|
294 | Line := UpperCase(Line);
|
---|
295 | if (Pos('<!DOCTYPE HTML',Line) > 0)
|
---|
296 | or (Pos('<HTML>',Line) > 0)
|
---|
297 | or (Pos('</BODY>',Line) > 0)then begin
|
---|
298 | Result := true;
|
---|
299 | end;
|
---|
300 | end;
|
---|
301 |
|
---|
302 |
|
---|
303 | function IsHTML(Lines : TStrings): boolean;
|
---|
304 | //Purpose: To look at the note loaded into Lines and determine if it is
|
---|
305 | // an HTML document. See other IsHTML(String) function for test used.
|
---|
306 | begin
|
---|
307 | Result := false;
|
---|
308 | if Lines = nil then exit;
|
---|
309 | Result := IsHTML(Lines.Text);
|
---|
310 | if Result = true then ScanForSubs(Lines);
|
---|
311 | end;
|
---|
312 |
|
---|
313 |
|
---|
314 | function HasHTMLTags(Text: string) : boolean;
|
---|
315 | function GetTag(p1,p2 : integer; var Text : string) : string;
|
---|
316 | var i : integer;
|
---|
317 | begin
|
---|
318 | Result := MidStr(Text,p1, p2-p1);
|
---|
319 | if Result[1] = '/' then Result := MidStr(Result,2,999);
|
---|
320 | i := Pos(' ',Result);
|
---|
321 | if i >0 then Result := MidStr(Result,1,i-1);
|
---|
322 | end;
|
---|
323 |
|
---|
324 | var p1,p2: integer;
|
---|
325 | Tag : string;
|
---|
326 | begin
|
---|
327 | Result := False; //default to ignore
|
---|
328 | Text := UpperCase(Text);
|
---|
329 | if (Pos('&NBSP;',Text)>0) then Result := true
|
---|
330 | else if (Pos('<P>',Text)>0) then Result := true
|
---|
331 | else if (Pos('<BR>',Text)>0) then Result := true
|
---|
332 | else if (Pos('<HTML>',Text)>0) then Result := true
|
---|
333 | else begin
|
---|
334 | p1 := Pos('<',Text); if p1 = 0 then exit;
|
---|
335 | p2 := Pos('>',Text); if p2 = 0 then exit;
|
---|
336 | Tag := GetTag(p1,p2,Text);
|
---|
337 | if Tag='' then exit;
|
---|
338 | if Pos('/'+Tag+'>',Text)>0 then result := true;
|
---|
339 | end;
|
---|
340 | {
|
---|
341 | if (Pos('<BR>',Text)>0) or
|
---|
342 | (Pos('</P>',Text)>0) or
|
---|
343 | (Pos('<UL>',Text)>0) or
|
---|
344 | (Pos('</UL>',Text)>0) or
|
---|
345 | (Pos('<LI>',Text)>0) or
|
---|
346 | (Pos('</LI>',Text)>0) or
|
---|
347 | (Pos('<OL>',Text)>0) or
|
---|
348 | (Pos('</OL>',Text)>0) or
|
---|
349 | (Pos('&NBSP;',Text)>0) or
|
---|
350 | (Pos('<P>',Text)>0) then begin
|
---|
351 | Result := true;
|
---|
352 | end;
|
---|
353 | }
|
---|
354 | end;
|
---|
355 |
|
---|
356 |
|
---|
357 | function LineAfterTag(Lines : TStrings; Tag : string) : integer;
|
---|
358 | //returns index of line directly after tag (-1 if not found)
|
---|
359 | //Note: only 1st tag is found (stops looking after that)
|
---|
360 | var p,i : integer;
|
---|
361 | s,s1,s2 : string;
|
---|
362 | begin
|
---|
363 | result := -1;
|
---|
364 | Tag := UpperCase(Tag);
|
---|
365 | for i := 0 to Lines.Count-1 do begin
|
---|
366 | s := UpperCase(Lines.Strings[i]);
|
---|
367 | p := Pos(Tag,s);
|
---|
368 | if p=0 then continue;
|
---|
369 | if (p+length(Tag)-1) < length(s) then begin //extra stuff after tag on line --> split line
|
---|
370 | s1 := MidStr(Lines.Strings[i],1,p+length(Tag)-1);
|
---|
371 | s2 := MidStr(Lines.Strings[i],p+length(Tag),9999);
|
---|
372 | Lines.Strings[i] := s1;
|
---|
373 | Lines.Insert(i+1,s2);
|
---|
374 | end;
|
---|
375 | //Lines.Insert(i+1,'');
|
---|
376 | result := i+1;
|
---|
377 | break;
|
---|
378 | end;
|
---|
379 | end;
|
---|
380 |
|
---|
381 | function LineBeforeTag(Lines : TStrings; Tag : string) : integer;
|
---|
382 | //returns index of newly added blank line, directly before tag (-1 if not found)
|
---|
383 | //Note: only 1st tag is found (stops looking after that)
|
---|
384 | var p,i,idx : integer;
|
---|
385 | s,s1,s2 : string;
|
---|
386 | begin
|
---|
387 | result := -1;
|
---|
388 | idx := -1;
|
---|
389 | Tag := UpperCase(Tag);
|
---|
390 | for i := 0 to Lines.Count-1 do begin
|
---|
391 | s := UpperCase(Lines.Strings[i]);
|
---|
392 | p := Pos(Tag,s);
|
---|
393 | if p>0 then begin
|
---|
394 | idx := i;
|
---|
395 | break;
|
---|
396 | end;
|
---|
397 | end;
|
---|
398 | if idx <> -1 then begin
|
---|
399 | p := Pos(Tag,UpperCase(Lines.Strings[idx]));
|
---|
400 | if p>1 then begin //extra stuff after tag on line --> split line
|
---|
401 | s1 := MidStr(Lines.Strings[idx],1,p-1);
|
---|
402 | s2 := MidStr(Lines.Strings[idx],p,9999);
|
---|
403 | Lines.Strings[idx] := s1;
|
---|
404 | Lines.Insert(idx+1,s2);
|
---|
405 | inc(idx);
|
---|
406 | end;
|
---|
407 | //Lines.Insert(idx-1,'');
|
---|
408 | result := idx;
|
---|
409 | end;
|
---|
410 | end;
|
---|
411 |
|
---|
412 | procedure SplitLineAfterTag(Lines : TStrings; Tag : string);
|
---|
413 | //Purpose: To ensure that text that occurs after Tag is split and wrapped
|
---|
414 | // to the next line.
|
---|
415 | //Note: It is assumed that tag is in form of <TAGName> or <SomeReallyLongText...
|
---|
416 | // if a closing '>' is not provided in the tag name, then the part provided
|
---|
417 | // is used for matching, and then a search for the closing '>' is made, and
|
---|
418 | // the split will occur after that.
|
---|
419 | //Note: Only the first instance of Tag will be found, stops searching after that.
|
---|
420 | //Note: Tag beginning and ending MUST occur on same line (wrapping of a long tag NOT supported)
|
---|
421 | var i,p1,p2 : integer;
|
---|
422 | s,s1,s2 : string;
|
---|
423 | begin
|
---|
424 | Tag := UpperCase(Tag);
|
---|
425 | for i := 0 to Lines.Count-1 do begin
|
---|
426 | s := UpperCase(Lines.Strings[i]);
|
---|
427 | p1 := Pos(Tag,s);
|
---|
428 | if p1=0 then continue;
|
---|
429 | p2 := PosEx('>',s,p1);
|
---|
430 | if p2=0 then continue; //this is a problem, no closing '>' found... ignore for now.
|
---|
431 | if p2 = length(s) then break;
|
---|
432 | s1 := MidStr(Lines.Strings[i],1,p2);
|
---|
433 | S2 := MidStr(Lines.Strings[i],p2+1,999);
|
---|
434 | Lines.Strings[i] := s1;
|
---|
435 | Lines.Insert(i+1,s2);
|
---|
436 | break;
|
---|
437 | end;
|
---|
438 | end;
|
---|
439 |
|
---|
440 | procedure SplitLineBeforeTag(Lines : TStrings; Tag : string);
|
---|
441 | //Purpose: To ensure that text that occurs before Tag is split and Tag
|
---|
442 | // is wrapped to the next line.
|
---|
443 | //Note: It is assumed that tag is in form of <TAGName> or <SomeReallyLongText...
|
---|
444 | //Note: Only the first instance of Tag will be found, stops searching after that.
|
---|
445 | var i,p1 : integer;
|
---|
446 | s1,s2 : string;
|
---|
447 | begin
|
---|
448 | Tag := UpperCase(Tag);
|
---|
449 | for i := 0 to Lines.Count-1 do begin
|
---|
450 | p1 := Pos(Tag,UpperCase(Lines.Strings[i]));
|
---|
451 | if p1=0 then continue;
|
---|
452 | s1 := MidStr(Lines.Strings[i],1,p1-1);
|
---|
453 | S2 := MidStr(Lines.Strings[i],p1,999);
|
---|
454 | Lines.Strings[i] := s1;
|
---|
455 | Lines.Insert(i+1,s2);
|
---|
456 | break;
|
---|
457 | end;
|
---|
458 | end;
|
---|
459 |
|
---|
460 | function IndexOfHoldingLine(Lines : TStrings; Tag : string) : integer;
|
---|
461 | var i : integer;
|
---|
462 | s : string;
|
---|
463 | begin
|
---|
464 | result := -1;
|
---|
465 | Tag := UpperCase(Tag);
|
---|
466 | for i := 0 to Lines.Count-1 do begin
|
---|
467 | s := UpperCase(Lines.Strings[i]);
|
---|
468 | if Pos (Tag,s)=0 then continue;
|
---|
469 | result := i;
|
---|
470 | break;
|
---|
471 | end;
|
---|
472 | end;
|
---|
473 |
|
---|
474 | procedure EnsureTrailingBR(Lines : TStrings);
|
---|
475 | var p,i : integer;
|
---|
476 | begin
|
---|
477 | for i := 0 to Lines.Count-1 do begin //Ensure each line ends with <BR>
|
---|
478 | p := Pos('<BR>',Lines.Strings[i]);
|
---|
479 | if p+3=length(Lines.Strings[i]) then continue;
|
---|
480 | Lines.Strings[i] := Lines.Strings[i] + '<BR>';
|
---|
481 | end;
|
---|
482 | end;
|
---|
483 |
|
---|
484 | procedure MergeLines(Lines,BeforeLines,AfterLines : TStrings);
|
---|
485 | var i : integer;
|
---|
486 | Result : TStringList;
|
---|
487 | begin
|
---|
488 | Result := TStringList.Create;
|
---|
489 | SplitLineAfterTag(Lines,'<!DOCTYPE HTML');
|
---|
490 | SplitLineBeforeTag(Lines,'</BODY>');
|
---|
491 | Result.Add(Lines.Strings[0]); //Should be line with <!DOCTYPE HTML...
|
---|
492 | for i := 0 to BeforeLines.Count-1 do begin //Add Before-Lines text
|
---|
493 | Result.Add(BeforeLines.Strings[i]);
|
---|
494 | end;
|
---|
495 | for i := 1 to Lines.Count-2 do begin //Add back regular lines text
|
---|
496 | Result.Add(Lines.Strings[i]);
|
---|
497 | end;
|
---|
498 | for i := 1 to AfterLines.Count-1 do begin //Add After-Lines text
|
---|
499 | Result.Add(AfterLines.Strings[i]);
|
---|
500 | end;
|
---|
501 | Result.Add(Lines.Strings[Lines.count-1]); //Should be '</BODY></HTML>' line
|
---|
502 |
|
---|
503 | Lines.Assign(Result);
|
---|
504 | end;
|
---|
505 |
|
---|
506 | procedure StripBeforeAfterHTML(Lines,OutBefore,OutAfter : TStrings);
|
---|
507 | //Purpose: Strip all text that comes before <!DOCTYPE ... line and store in OutBefore;
|
---|
508 | // Strip all text that comes after </HTML> ... line and store in OutAfter;
|
---|
509 | var i : integer;
|
---|
510 | DocTypeLine,EndHTMLLine: integer;
|
---|
511 | begin
|
---|
512 | OutBefore.Clear;
|
---|
513 | OutAfter.Clear;
|
---|
514 | DocTypeLine := IndexOfHoldingLine(Lines,'<!DOCTYPE HTML');
|
---|
515 | if DocTypeLine>1 then begin
|
---|
516 | for i := 0 to DocTypeLine-1 do OutBefore.Add(Lines.Strings[i]);
|
---|
517 | for i := 0 to DocTypeLine-1 do Lines.Delete(0);
|
---|
518 | end;
|
---|
519 | EndHTMLLine := IndexOfHoldingLine(Lines,'</HTML>');
|
---|
520 | if (EndHTMLLine>0) and (EndHTMLLine < (Lines.Count-1)) then begin
|
---|
521 | for i := EndHTMLLine+1 to Lines.Count-1 do OutAfter.Add(Lines.Strings[i]);
|
---|
522 | for i := EndHTMLLine+1 to Lines.Count-1 do Lines.Delete(EndHTMLLine+1);
|
---|
523 | end;
|
---|
524 | end;
|
---|
525 |
|
---|
526 | Function FixHTMLCRLF(Text : String) : string;
|
---|
527 | begin
|
---|
528 | Result := AnsiReplaceText(Text,'<NO DATA>',#$1E); //protect sequences we want
|
---|
529 | Result := AnsiReplaceText(Result,'>'+CRLF,'>'#$1F); //protect sequences we want
|
---|
530 | Result := AnsiReplaceText(Result,CRLF,'<BR>'+CRLF); //Add <BR>'s to CrLf's
|
---|
531 | Result := AnsiReplaceText(Result,'>'#$1F,'>'+CRLF); //Restore sequences we wanted
|
---|
532 | Result := AnsiReplaceText(Result,#$1E,'<NO DATA>'); //Restore sequences we wanted
|
---|
533 | end;
|
---|
534 |
|
---|
535 |
|
---|
536 | procedure FixHTML(Lines : TStrings); //kt 6/20/09
|
---|
537 | //Purpose: to put header info that VistA adds to note into proper formatting.
|
---|
538 | var BeforeLines,AfterLines : TStringList;
|
---|
539 | begin
|
---|
540 | BeforeLines := TStringList.Create;
|
---|
541 | AfterLines := TStringList.Create;
|
---|
542 | StripBeforeAfterHTML(Lines,BeforeLines,AfterLines);
|
---|
543 | EnsureTrailingBR(BeforeLines);
|
---|
544 | if BeforeLines.Count>0 then begin //Wrap Before-Lines with formatting
|
---|
545 | BeforeLines.Insert(0,'<CODE>');
|
---|
546 | //<---Existing text remains in between --->
|
---|
547 | BeforeLines.Add('</CODE>');
|
---|
548 | BeforeLines.Add('<HR><P>'); //horizontal line
|
---|
549 | end;
|
---|
550 | EnsureTrailingBR(AfterLines);
|
---|
551 | if AfterLines.Count > 0 then begin //Wrap After-Lines with formatting
|
---|
552 | AfterLines.Insert(0,'<P><CODE>');
|
---|
553 | //<---Existing text remains in between --->
|
---|
554 | AfterLines.Add('</CODE>');
|
---|
555 | end;
|
---|
556 | MergeLines(Lines,BeforeLines,AfterLines);
|
---|
557 | BeforeLines.Free;
|
---|
558 | AfterLines.Free;
|
---|
559 | end;
|
---|
560 |
|
---|
561 | procedure SplitToArray (HTMLText: string; Lines : TStrings);
|
---|
562 | var tempS : string;
|
---|
563 | InEscapeCode, InTagCode : boolean;
|
---|
564 | i, LastGoodBreakI : integer;
|
---|
565 | TagStart,TagEnd : integer;
|
---|
566 | TagText : string;
|
---|
567 | TextLen : integer;
|
---|
568 | MaxLineLen : integer;
|
---|
569 | begin
|
---|
570 | Lines.Clear;
|
---|
571 | MaxLineLen := 80;
|
---|
572 | Repeat
|
---|
573 | InEscapeCode := False;
|
---|
574 | InTagCode := False;
|
---|
575 | LastGoodBreakI := 0;
|
---|
576 | TextLen := length(HTMLText);
|
---|
577 | TagText := '';
|
---|
578 | TagStart := 0; TagEnd := 0;
|
---|
579 | if TextLen > 80 then TextLen := MaxLineLen;
|
---|
580 | for i := 1 to TextLen do begin
|
---|
581 | if (HTMLText[i] = '<') then begin
|
---|
582 | InTagCode := True;
|
---|
583 | TagStart := i;
|
---|
584 | TagText := '';
|
---|
585 | LastGoodBreakI := i-1;
|
---|
586 | end;
|
---|
587 | if (HTMLText[i] = '&') then begin
|
---|
588 | InEscapeCode := True;
|
---|
589 | LastGoodBreakI := i-1;
|
---|
590 | end;
|
---|
591 | if InEscapeCode and (HTMLText[i] = ';') then begin
|
---|
592 | InEscapeCode := False;
|
---|
593 | LastGoodBreakI := i;
|
---|
594 | end;
|
---|
595 | if InTagCode and (HTMLText[i] = '>') then begin
|
---|
596 | InTagCode := False;
|
---|
597 | TagEnd := i;
|
---|
598 | TagText := UpperCase(MidStr(HTMLText,TagStart+1,(TagEnd-TagStart-1)));
|
---|
599 | LastGoodBreakI := i;
|
---|
600 | end;
|
---|
601 | if (HTMLText[i] = ',') and (MaxLineLen > 80) then begin
|
---|
602 | LastGoodBreakI := i;
|
---|
603 | break;
|
---|
604 | end;
|
---|
605 | if (TagText='BR') or (TagText='/P') then begin
|
---|
606 | LastGoodBreakI := TagEnd;
|
---|
607 | break;
|
---|
608 | end else TagText := '';;
|
---|
609 | if (not InTagCode) and (not InEscapeCode)
|
---|
610 | and (HTMLText[i] = ' ') then LastGoodBreakI := i;
|
---|
611 | end;
|
---|
612 | if LastGoodBreakI > 0 then begin
|
---|
613 | tempS := MidStr(HTMLText,1,LastGoodBreakI); //get next 80 chars, or less if at the end.
|
---|
614 | HTMLText := Rightstr(HTMLText, length(HTMLText)- LastGoodBreakI); //characters 81 ... the end
|
---|
615 | Lines.Add(tempS);
|
---|
616 | end else begin
|
---|
617 | if MaxLineLen < 250 then begin
|
---|
618 | MaxLineLen := 250; //emergency mode
|
---|
619 | end else begin //i.e. couldn't find any break within 250 chars. So just chop off.
|
---|
620 | tempS := MidStr(HTMLText,1,80);
|
---|
621 | HTMLText := Rightstr(HTMLText, length(HTMLText)- 80); //characters 81 ... the end
|
---|
622 | Lines.Add(tempS);
|
---|
623 | end;
|
---|
624 | end;
|
---|
625 | until length(HTMLText)=0;
|
---|
626 | end;
|
---|
627 |
|
---|
628 |
|
---|
629 | function WrapHTML(HTMLText : string) : string; //kt 6/3/09
|
---|
630 | //Purpose: take HTML formatted text and sure it has proper headers and footers etc.
|
---|
631 | // i.e. 'wrap' partial HTML into formal format.
|
---|
632 | begin
|
---|
633 | if Pos('<BODY>',HTMLText)=0 then HTMLText := '<BODY>' + HTMLText;
|
---|
634 | if Pos('</BODY>',HTMLText)=0 then HTMLText := HTMLText + '</BODY>';
|
---|
635 | if Pos('<HTML>',HTMLText)=0 then HTMLText := '<HTML>' + HTMLText;
|
---|
636 | if Pos('</HTML>',HTMLText)=0 then HTMLText := HTMLText + '</HTML>';
|
---|
637 | if Pos('<!DOCTYPE HTML',HTMLText)=0 then begin
|
---|
638 | HTMLText := '<!DOCTYPE HTML PUBLIC "-//WC3//DTD HTML 3.2//EN">'+ #10#13 + HTMLText;
|
---|
639 | end;
|
---|
640 | result := HTMLText;
|
---|
641 | end;
|
---|
642 |
|
---|
643 | function UnwrapHTML(HTMLText : string) : string;
|
---|
644 | //Purpose: take HTML formatted text and remove proper headers and footers etc.
|
---|
645 | // i.e. 'unwrap' formal HTML into partial HTML format.
|
---|
646 | begin
|
---|
647 | HTMLText := piece(HTMLText,'<HTML>',2);
|
---|
648 | HTMLText := piece(HTMLText,'</HTML>',1);
|
---|
649 | HTMLText := piece(HTMLText,'<BODY>',2);
|
---|
650 | HTMLText := piece(HTMLText,'</BODY>',1);
|
---|
651 | result := HTMLText;
|
---|
652 | end;
|
---|
653 |
|
---|
654 | function CheckForImageLink(Lines: TStrings; ImageList : TStringList) : boolean;
|
---|
655 | {Purpose: To scan memNote memo for a link to an image. If found, return link(s)
|
---|
656 | input: none:
|
---|
657 | output: Will return a string list holding 1 or more links
|
---|
658 | Notes: Here will be the <img .. > format scanned for:
|
---|
659 |
|
---|
660 | Here is some opening text...
|
---|
661 | <img src="http://www.geocities.com/kdtop3/OpenVistA.jpg" alt="Image Title 1">
|
---|
662 | And here is some more text
|
---|
663 | <img src="http://www.geocities.com/kdtop3/OpenVistA_small.jpg" alt="Image Title 2">
|
---|
664 | And the saga continues...
|
---|
665 | <img src="http://www.geocities.com/kdtop3/pics/Image100.gif" alt="Image Title 3">
|
---|
666 | As with html, end-of-lines and white space is not preserved or significant
|
---|
667 | }
|
---|
668 |
|
---|
669 | function GetBetween (var Text : AnsiString; OpenTag,CloseTag : string;
|
---|
670 | KeepTags : Boolean) : string;
|
---|
671 | {Purpose: Gets text between Open and Close tags. Removes any CR's or LF's
|
---|
672 | Input: Text - the text to work on. It IS changed as code is removed
|
---|
673 | KeepTags - true if want tag return in result
|
---|
674 | false if tag not in result (still is removed from Text)
|
---|
675 | Output: Text is changed.
|
---|
676 | Result=the code between the opening and closing tags
|
---|
677 | Note: Both OpenTag and CloseTag MUST be present for anything to happen.
|
---|
678 | }
|
---|
679 |
|
---|
680 | procedure CutInThree(var Text : AnsiString; p1, p2 : Integer; var s1,s2,s3 : AnsiString);
|
---|
681 | {Purpose: Cut input string Text into 3 parts, with cut points given by p1 & p2.
|
---|
682 | p1 points to first character to be in s2
|
---|
683 | p2 points to last character to be in s2 }
|
---|
684 | begin
|
---|
685 | s1 := ''; s2 := ''; s3 := '';
|
---|
686 | if p1 > 1 then s1 := MidStr(Text, 1, p1-1);
|
---|
687 | s2 := MidStr(Text, p1, p2-p1+1);
|
---|
688 | s3 := MidStr(Text, p2+1, Length(Text)-p2);
|
---|
689 | end;
|
---|
690 |
|
---|
691 | var
|
---|
692 | p1,p2 : integer;
|
---|
693 | s1,s2,s3 : AnsiString;
|
---|
694 |
|
---|
695 | begin
|
---|
696 | Result := ''; //default of no result.
|
---|
697 |
|
---|
698 | p1 := Pos(UpperCase(OpenTag), UpperCase(Text));
|
---|
699 | if (p1 > 0) then begin
|
---|
700 | p2 := Pos(UpperCase(CloseTag),UpperCase(Text)) + Length(CloseTag) -1;
|
---|
701 | if ((p2 > 0) and (p2 > p1)) then begin
|
---|
702 | CutInThree (Text, p1,p2, s1,Result,s3);
|
---|
703 | Text := s1+s3;
|
---|
704 | //Now, remove any CR's or LF's
|
---|
705 | repeat
|
---|
706 | p1 := Pos (Chr(13),Result);
|
---|
707 | if p1= 0 then p1 := Pos (Chr(10),Result);
|
---|
708 | if (p1 > 0) then begin
|
---|
709 | CutInThree (Result, p1,p1, s1,s2,s3);
|
---|
710 | Result := s1+s3;
|
---|
711 | // Text := MidStr(Text,1,p1-1) + MidStr(Text,p1+1,Length(Text)-p1);
|
---|
712 | end;
|
---|
713 | until (p1=0);
|
---|
714 | //Now cut off boundry tags if requested.
|
---|
715 | if not KeepTags then begin
|
---|
716 | p1 := Length(OpenTag) + 1;
|
---|
717 | p2 := Length (Result) - Length (CloseTag);
|
---|
718 | CutInThree (Result, p1,p2, s1,s2,s3);
|
---|
719 | Result := s2;
|
---|
720 | end;
|
---|
721 | end;
|
---|
722 | end;
|
---|
723 | end;
|
---|
724 |
|
---|
725 | var
|
---|
726 | Text : AnsiString;
|
---|
727 | LineStr : string;
|
---|
728 |
|
---|
729 | begin
|
---|
730 | Result := false; //set default
|
---|
731 | if (ImageList = nil) or (Lines = nil) then exit;
|
---|
732 | ImageList.Clear; //set default
|
---|
733 | Text := Lines.Text; //Get entire note into one long string
|
---|
734 | repeat
|
---|
735 | LineStr := GetBetween (Text, '<img', '>', true);
|
---|
736 | if LineStr <> '' then begin
|
---|
737 | ImageList.Add(LineStr);
|
---|
738 | Result := true;
|
---|
739 | end;
|
---|
740 | until LineStr = '';
|
---|
741 | //Note: The following works, but need to replace removed links
|
---|
742 | // with "[Title]" Work on later...
|
---|
743 | //memNote.Lines.Text := Text;
|
---|
744 | end;
|
---|
745 |
|
---|
746 | function ProtectHTMLSpaces(Text : String) : string;
|
---|
747 | begin
|
---|
748 | Result := AnsiReplaceStr(Text, #9, ' ');
|
---|
749 | while Pos(' ',Result)>0 do begin //preserve whitespace
|
---|
750 | Result := AnsiReplaceStr(Result, ' ', ' ');
|
---|
751 | end;
|
---|
752 | end;
|
---|
753 |
|
---|
754 | function Text2HTML(Lines : TStrings) : String;
|
---|
755 | //Purpose: Take plain text, and prep for viewing in HTML viewer.
|
---|
756 | //i.e. convert TABs into  's and add <br> at end of line etc.
|
---|
757 | var i : integer;
|
---|
758 | tempS : string;
|
---|
759 | begin
|
---|
760 | for i := 0 to Lines.Count-1 do begin
|
---|
761 | tempS := TrimRight(Lines.Strings[i]);
|
---|
762 | if i = Lines.Count-1 then tempS := tempS + '<P>'
|
---|
763 | else tempS := tempS + '<BR>';
|
---|
764 | Lines.Strings[i] := tempS;
|
---|
765 | end;
|
---|
766 | Result := ProtectHTMLSpaces(Lines.Text)
|
---|
767 | end;
|
---|
768 |
|
---|
769 | function Text2HTML(text : string) : String; overload;
|
---|
770 | var Lines : TStringList;
|
---|
771 | begin
|
---|
772 | Lines := TStringList.create;
|
---|
773 | Lines.Text := text;
|
---|
774 | Result := Text2HTML(Lines);
|
---|
775 | Lines.Free;
|
---|
776 | end;
|
---|
777 |
|
---|
778 | type
|
---|
779 | TFontSizeData = record
|
---|
780 | case byte of 1: (Data : array[0..3] of byte);
|
---|
781 | 2: (Size : byte; Filler : array[1..3] of byte);
|
---|
782 | end;
|
---|
783 |
|
---|
784 | var
|
---|
785 | StoredFontSize : TFontSizeData;
|
---|
786 | FontSizeReg: TRegistry;
|
---|
787 |
|
---|
788 | procedure SetRegHTMLFontSize(Size: byte);
|
---|
789 | //Purpose: To set the internet explorer Font Size value via the registry.
|
---|
790 | //Expected input: HTML_SMALLEST, HTML_SMALL, HTML_MEDIUM,HTML_LARGE, HTML_LARGEST
|
---|
791 | // Value should be 0 (smallest) - 4 (largest)
|
---|
792 | const HTML_BLANK : TFontSizeData =(Data: (0,0,0,0));
|
---|
793 | var Value : TFontSizeData;
|
---|
794 |
|
---|
795 | begin
|
---|
796 | if Size > 4 then Size := 4;
|
---|
797 | Value := HTML_BLANK; Value.Size := Size;
|
---|
798 | FontSizeReg := TRegistry.Create; //To be freed in RestoreRegHTMLFontSize
|
---|
799 | try
|
---|
800 | FontSizeReg.Rootkey := HKEY_CURRENT_USER;
|
---|
801 | if FontSizeReg.OpenKey('\Software\Microsoft\Internet Explorer\International\Scripts\3', False) then begin
|
---|
802 | FontSizeReg.ReadBinaryData('IEFontSize',StoredFontSize,Sizeof(StoredFontSize));
|
---|
803 | FontSizeReg.WriteBinaryData('IEFontSize',Value,SizeOf(Value));
|
---|
804 | end;
|
---|
805 | finally
|
---|
806 | end;
|
---|
807 | end;
|
---|
808 |
|
---|
809 | procedure RestoreRegHTMLFontSize;
|
---|
810 | //Purpose: To restore the Internet Explorer zoom value to a stored value..
|
---|
811 | //elh 6/19/09
|
---|
812 | begin
|
---|
813 | if not assigned(FontSizeReg) then FontSizeReg := TRegistry.Create;
|
---|
814 | try
|
---|
815 | FontSizeReg.WriteBinaryData('IEFontSize',StoredFontSize,SizeOf(StoredFontSize));
|
---|
816 | finally
|
---|
817 | FontSizeReg.Free;
|
---|
818 | end;
|
---|
819 | end;
|
---|
820 |
|
---|
821 | var
|
---|
822 | StoredIEHeader, StoredIEFooters : string;
|
---|
823 | Reg : TRegistry;
|
---|
824 |
|
---|
825 | procedure SetupHTMLPrinting(Name,DOB,VisitDate,Location,Institution : string);
|
---|
826 | //Purpose: To open the IE header and footer registry keys, save the
|
---|
827 | //current value and then replace with passed patient data. elh 6/19/09
|
---|
828 | //NOTE: There does not seem to be any other way to do this programatically.
|
---|
829 | var NewHeader,NewFooter : string;
|
---|
830 | begin
|
---|
831 | if DesiredHTMLFontSize > 0 then begin
|
---|
832 | SetRegHTMLFontSize(DesiredHTMLFontSize-1); //Downsize by 1 step
|
---|
833 | end;
|
---|
834 | NewHeader := Location + ' &b ' + Institution + ' &b Printed: &d &t';
|
---|
835 | NewFooter := Name + ' (' + DOB + ') &b Note: ' + VisitDate + ' &b &p of &P';
|
---|
836 | Reg := TRegistry.Create; //to be freed in RestoreIEPrinting
|
---|
837 | try
|
---|
838 | Reg.Rootkey := HKEY_CURRENT_USER;
|
---|
839 | if Reg.OpenKey('\Software\Microsoft\Internet Explorer\PageSetup', False) then begin
|
---|
840 | StoredIEFooters := Reg.ReadString('footer');
|
---|
841 | StoredIEHeader := Reg.ReadString('header');
|
---|
842 | Reg.WriteString('header',NewHeader);
|
---|
843 | Reg.WriteString('footer',NewFooter);
|
---|
844 | end;
|
---|
845 | finally
|
---|
846 | end;
|
---|
847 | end;
|
---|
848 |
|
---|
849 | procedure RestoreIEPrinting;
|
---|
850 | //Purpose: To restore the IE header and footer registry with the initial value
|
---|
851 | //NOTE: The below function was used to restore the previous value to the registry
|
---|
852 | // but got commented above, so the registry retained the patient's data
|
---|
853 | // to resolve this, we are now setting this to a default value.
|
---|
854 | //elh 6/19/09
|
---|
855 | begin
|
---|
856 | if not assigned(Reg) then Reg := TRegistry.Create;
|
---|
857 | try
|
---|
858 | StoredIEFooters := '&u&b&d'; //Comment this line to restore previous value
|
---|
859 | StoredIEHeader := '&d&b&t&bPage &p of &P'; //Comment this line to restore previous value
|
---|
860 | Reg.WriteString('footer',StoredIEFooters);
|
---|
861 | Reg.WriteString('header',StoredIEHeader);
|
---|
862 | RestoreRegHTMLFontSize;
|
---|
863 | finally
|
---|
864 | Reg.Free;
|
---|
865 | end;
|
---|
866 | end;
|
---|
867 |
|
---|
868 | function ExtractDateOfNote(Lines : TStringList) : string;
|
---|
869 | //Scan note and return date found after 'DATE OF NOTE:', if present.
|
---|
870 | var i,p : integer;
|
---|
871 | s : string;
|
---|
872 | begin
|
---|
873 | Result := '';
|
---|
874 | if Lines = nil then exit;
|
---|
875 | for i := 0 to Lines.Count-1 do begin
|
---|
876 | p := Pos('DATE OF NOTE:',Lines.Strings[i]);
|
---|
877 | if p<1 then continue;
|
---|
878 | s := Piece(Lines.Strings[i],'DATE OF NOTE:',2);
|
---|
879 | s := Piece(s,'@',1);
|
---|
880 | Result := Trim(s);
|
---|
881 | end;
|
---|
882 | end;
|
---|
883 |
|
---|
884 | //===============================================================
|
---|
885 |
|
---|
886 | Constructor TPrinterEvents.Create;
|
---|
887 | begin
|
---|
888 | RestorePrinterTimer := TTimer.Create(frmNotes);
|
---|
889 | RestorePrinterTimer.Enabled := false;
|
---|
890 | RestorePrinterTimer.Interval := 30000; //30 seconds to complete print job.
|
---|
891 | RestorePrinterTimer.OnTimer := HandleRestorePrinting;
|
---|
892 | PrintingNow := false;
|
---|
893 | end;
|
---|
894 |
|
---|
895 | Destructor TPrinterEvents.Destroy;
|
---|
896 | begin
|
---|
897 | RestorePrinterTimer.Free;
|
---|
898 | inherited Destroy;
|
---|
899 | end;
|
---|
900 |
|
---|
901 |
|
---|
902 | procedure TPrinterEvents.HandleRestorePrinting (Sender: TObject);
|
---|
903 | begin
|
---|
904 | if PrinterEvents.PrintingNow then begin
|
---|
905 | RestorePrinterTimer.Enabled := true; // reset timer for later.
|
---|
906 | exit;
|
---|
907 | end;
|
---|
908 | RestorePrinterTimer.Enabled := false;
|
---|
909 | RestoreIEPrinting; {elh 6/19/09} //kt
|
---|
910 | //kt SetDefaultPrinter(SavedDefaultPrinter);
|
---|
911 | //beep;
|
---|
912 | end;
|
---|
913 |
|
---|
914 | //===============================================================
|
---|
915 |
|
---|
916 |
|
---|
917 | initialization
|
---|
918 | DesiredHTMLFontSize := 2; //probably overwritten in fNotes initialization
|
---|
919 | PrinterEvents := TPrinterEvents.Create;
|
---|
920 |
|
---|
921 | finalization
|
---|
922 | PrinterEvents.Free;
|
---|
923 | end.
|
---|