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