| [459] | 1 | unit uSpell;
 | 
|---|
 | 2 | 
 | 
|---|
 | 3 | {$O-}
 | 
|---|
 | 4 | 
 | 
|---|
 | 5 | interface
 | 
|---|
 | 6 | 
 | 
|---|
 | 7 | uses
 | 
|---|
 | 8 |   Windows, Messages, SysUtils, Classes, Controls, Forms, ComObj, StdCtrls, ComCtrls,
 | 
|---|
 | 9 |   ORSystem, Word2000, ORFn, Variants, rCore, clipbrd;
 | 
|---|
 | 10 | 
 | 
|---|
 | 11 | type
 | 
|---|
 | 12 | 
 | 
|---|
 | 13 |   TSpellCheckAvailable = record
 | 
|---|
 | 14 |     Evaluated: boolean;
 | 
|---|
 | 15 |     Available: boolean;
 | 
|---|
 | 16 |   end;
 | 
|---|
 | 17 | 
 | 
|---|
 | 18 | function  SpellCheckAvailable: Boolean;
 | 
|---|
 | 19 | function  SpellCheckInProgress: Boolean;
 | 
|---|
 | 20 | procedure KillSpellCheck;
 | 
|---|
 | 21 | procedure SpellCheckForControl(AnEditControl: TCustomMemo);
 | 
|---|
 | 22 | procedure GrammarCheckForControl(AnEditControl: TCustomMemo);
 | 
|---|
 | 23 | 
 | 
|---|
 | 24 | implementation
 | 
|---|
 | 25 | 
 | 
|---|
 | 26 | const
 | 
|---|
 | 27 |   TX_WINDOW_TITLE       = 'CPRS-Chart Spell Checking #';
 | 
|---|
 | 28 |   TX_NO_SPELL_CHECK     = 'Spell checking is unavailable.';
 | 
|---|
 | 29 |   TX_NO_GRAMMAR_CHECK   = 'Grammar checking is unavailable.';
 | 
|---|
 | 30 |   TX_SPELL_COMPLETE     = 'The spelling check is complete.';
 | 
|---|
 | 31 |   TX_GRAMMAR_COMPLETE   = 'The grammar check is complete.';
 | 
|---|
 | 32 |   TX_SPELL_ABORT        = 'The spelling check terminated abnormally.';
 | 
|---|
 | 33 |   TX_GRAMMAR_ABORT      = 'The grammar check terminated abnormally.';
 | 
|---|
 | 34 |   TX_SPELL_CANCELLED    = 'Spelling check was cancelled before completion.';
 | 
|---|
 | 35 |   TX_GRAMMAR_CANCELLED  = 'Grammar check was cancelled before completion.';
 | 
|---|
 | 36 |   TX_NO_DETAILS         = 'No further details are available.';
 | 
|---|
 | 37 |   TX_NO_CORRECTIONS     = 'Corrections have NOT been applied.';
 | 
|---|
 | 38 |   CR_LF                 = #13#10;
 | 
|---|
 | 39 |   SPELL_CHECK           = 'S';
 | 
|---|
 | 40 |   GRAMMAR_CHECK         = 'G';
 | 
|---|
 | 41 | 
 | 
|---|
 | 42 | var
 | 
|---|
 | 43 |   WindowList: TList;
 | 
|---|
 | 44 |   OldList, NewList: TList;
 | 
|---|
 | 45 |   MSWord: OleVariant;
 | 
|---|
 | 46 |   uSpellCheckAvailable: TSpellCheckAvailable;
 | 
|---|
 | 47 | 
 | 
|---|
 | 48 | function SpellCheckInProgress: boolean;
 | 
|---|
 | 49 | begin
 | 
|---|
 | 50 |   Result := not VarIsEmpty(MSWord);
 | 
|---|
 | 51 | end;
 | 
|---|
 | 52 | 
 | 
|---|
 | 53 | procedure KillSpellCheck;
 | 
|---|
 | 54 | begin
 | 
|---|
 | 55 |   if SpellCheckInProgress then
 | 
|---|
 | 56 |     begin
 | 
|---|
 | 57 |       MSWord.Quit(wdDoNotSaveChanges);
 | 
|---|
 | 58 |       VarClear(MSWord);
 | 
|---|
 | 59 |     end;
 | 
|---|
 | 60 | end;
 | 
|---|
 | 61 | 
 | 
|---|
 | 62 | function SpellCheckTitle: string;
 | 
|---|
 | 63 | begin
 | 
|---|
 | 64 |   Result := TX_WINDOW_TITLE + IntToStr(Application.Handle);
 | 
|---|
 | 65 | end;
 | 
|---|
 | 66 | 
 | 
|---|
 | 67 | function GetWindows(Handle: HWND; Info: Pointer): BOOL; stdcall;
 | 
|---|
 | 68 | begin
 | 
|---|
 | 69 |   Result := True;
 | 
|---|
 | 70 |   WindowList.Add(Pointer(Handle));
 | 
|---|
 | 71 | end;
 | 
|---|
 | 72 | 
 | 
|---|
 | 73 | procedure GetWindowList(List: TList);
 | 
|---|
 | 74 | begin
 | 
|---|
 | 75 |   WindowList := List;
 | 
|---|
 | 76 |   EnumWindows(@GetWindows, 0);
 | 
|---|
 | 77 | end;
 | 
|---|
 | 78 | 
 | 
|---|
 | 79 | procedure BringWordToFront(OldList, NewList: TList);
 | 
|---|
 | 80 | var
 | 
|---|
 | 81 |   i, NameLen: integer;
 | 
|---|
 | 82 |   WinName: array[0..160] of char;
 | 
|---|
 | 83 |   NewWinName: PChar;
 | 
|---|
 | 84 |   NewName: string;
 | 
|---|
 | 85 | 
 | 
|---|
 | 86 | begin
 | 
|---|
 | 87 |   NewName := SpellCheckTitle;
 | 
|---|
 | 88 |   NameLen := length(NewName);
 | 
|---|
 | 89 |   for i := 0 to NewList.Count-1 do
 | 
|---|
 | 90 |   begin
 | 
|---|
 | 91 |     if(OldList.IndexOf(NewList[i]) < 0) then
 | 
|---|
 | 92 |     begin
 | 
|---|
 | 93 |       GetWindowText(HWND(NewList[i]), WinName, sizeof(WinName) - 1);
 | 
|---|
 | 94 |       if Pos('CPRS', WinName) > 0 then
 | 
|---|
 | 95 |         NewWinName := PChar(Copy(WinName, Pos('CPRS', WinName), sizeof(WinName) - 1))
 | 
|---|
 | 96 |       else
 | 
|---|
 | 97 |         NewWinName := WinName;
 | 
|---|
 | 98 |       if StrLComp(NewWinName, pchar(NewName), NameLen)=0 then
 | 
|---|
 | 99 |       begin
 | 
|---|
 | 100 |         Application.ProcessMessages;
 | 
|---|
 | 101 |         SetForegroundWindow(HWND(NewList[i]));
 | 
|---|
 | 102 |         break;
 | 
|---|
 | 103 |       end;
 | 
|---|
 | 104 |     end;
 | 
|---|
 | 105 |   end;
 | 
|---|
 | 106 | end;
 | 
|---|
 | 107 | 
 | 
|---|
 | 108 | { Spell Checking using Visual Basic for Applications script }
 | 
|---|
 | 109 | 
 | 
|---|
 | 110 | function SpellCheckAvailable: Boolean;
 | 
|---|
 | 111 | //const
 | 
|---|
 | 112 | //  WORD_VBA_CLSID = 'CLSID\{000209FF-0000-0000-C000-000000000046}';
 | 
|---|
 | 113 | begin
 | 
|---|
 | 114 | // CHANGED FOR PT. SAFETY ISSUE RELEASE 19.16, PATCH OR*3*155 - ADDED NEXT 2 LINES:
 | 
|---|
 | 115 |   //result := false;
 | 
|---|
 | 116 |   //exit;
 | 
|---|
 | 117 | //  Reenabled in version 21.1, via parameter setting  (RV)
 | 
|---|
 | 118 | //  Result := (GetUserParam('ORWOR SPELL CHECK ENABLED?') = '1');
 | 
|---|
 | 119 |   with uSpellCheckAvailable do        // only want to call this once per session!!!  v23.10+
 | 
|---|
 | 120 |     begin
 | 
|---|
 | 121 |       if not Evaluated then
 | 
|---|
 | 122 |         begin
 | 
|---|
 | 123 |           Available := (GetUserParam('ORWOR SPELL CHECK ENABLED?') = '1');
 | 
|---|
 | 124 |           Evaluated := True;
 | 
|---|
 | 125 |         end;
 | 
|---|
 | 126 |       Result := Available;
 | 
|---|
 | 127 |     end;
 | 
|---|
 | 128 | end;
 | 
|---|
 | 129 | 
 | 
|---|
 | 130 | procedure SpellAndGrammarCheckForControl(var AnotherEditControl: TCustomMemo; ACheck: Char);
 | 
|---|
 | 131 | var
 | 
|---|
 | 132 |   NoLFText, LFText: string;
 | 
|---|
 | 133 |   OneChar: char;
 | 
|---|
 | 134 |   ErrMsg: string;
 | 
|---|
 | 135 |   FinishedChecking: boolean;
 | 
|---|
 | 136 |   OldSaveInterval, i: integer;
 | 
|---|
 | 137 |   MsgText: string;
 | 
|---|
 | 138 |   FirstLineBlank: boolean;
 | 
|---|
 | 139 |   OldLine0: string;
 | 
|---|
 | 140 | begin
 | 
|---|
 | 141 |   if AnotherEditControl = nil then Exit;
 | 
|---|
 | 142 |   OldList := TList.Create;
 | 
|---|
 | 143 |   NewList := TList.Create;
 | 
|---|
 | 144 |   FinishedChecking := False;
 | 
|---|
 | 145 |   FirstLineBlank := False;
 | 
|---|
 | 146 |   NoLFText := '';
 | 
|---|
 | 147 |   OldLine0 := '';
 | 
|---|
 | 148 |   ClipBoard.Clear;
 | 
|---|
 | 149 |   try
 | 
|---|
 | 150 |     try
 | 
|---|
 | 151 |       GetWindowList(OldList);
 | 
|---|
 | 152 |       try
 | 
|---|
 | 153 |         Screen.Cursor := crHourGlass;
 | 
|---|
 | 154 |         MSWord := CreateOLEObject('Word.Application');
 | 
|---|
 | 155 |       except   // MSWord not available, so exit now
 | 
|---|
 | 156 |         Screen.Cursor := crDefault;
 | 
|---|
 | 157 |         case ACheck of
 | 
|---|
 | 158 |           SPELL_CHECK  :  MsgText := TX_NO_SPELL_CHECK;
 | 
|---|
 | 159 |           GRAMMAR_CHECK:  MsgText := TX_NO_GRAMMAR_CHECK;
 | 
|---|
 | 160 |           else            MsgText := ''
 | 
|---|
 | 161 |         end;
 | 
|---|
 | 162 |         Application.MessageBox(PChar(MsgText), PChar(Application.Title), MB_ICONWARNING);
 | 
|---|
 | 163 |         Exit;
 | 
|---|
 | 164 |       end;
 | 
|---|
 | 165 | 
 | 
|---|
 | 166 |       GetWindowList(NewList);
 | 
|---|
 | 167 |       try
 | 
|---|
 | 168 |         MSWord.Application.Caption := SpellCheckTitle;
 | 
|---|
 | 169 |         // Position Word off screen to avoid having document visible...
 | 
|---|
 | 170 |         MSWord.WindowState := 0;
 | 
|---|
 | 171 |         MSWord.Top := -3000;
 | 
|---|
 | 172 |         OldSaveInterval := MSWord.Application.Options.SaveInterval;
 | 
|---|
 | 173 |         MSWord.Application.Options.SaveInterval := 0;
 | 
|---|
 | 174 |         MSWord.Application.Options.AutoFormatReplaceQuotes := False;
 | 
|---|
| [460] | 175 |         MSWord.Application.Options.AutoFormatAsYouTypeReplaceQuotes := False;
 | 
|---|
| [459] | 176 |         MSWord.ResetIgnoreAll;
 | 
|---|
 | 177 | 
 | 
|---|
 | 178 |         MSWord.Documents.Add;                                              // FileNew
 | 
|---|
 | 179 |         MSWord.ActiveDocument.TrackRevisions := False;
 | 
|---|
 | 180 |         with AnotherEditControl do
 | 
|---|
 | 181 |           if (Lines.Count > 0) and (not ContainsVisibleChar(Lines[0])) then
 | 
|---|
 | 182 |             begin
 | 
|---|
 | 183 |               FirstLineBlank := True;  //MS bug when spell-checking document with blank first line  (RV - v22.6)
 | 
|---|
 | 184 |               OldLine0 := Lines[0];
 | 
|---|
 | 185 |               Lines.Delete(0);
 | 
|---|
 | 186 |             end;
 | 
|---|
 | 187 |         MSWord.ActiveDocument.Content.Text := (AnotherEditControl.Text);   // The Text property returns the plain, unformatted text of the selection or range.
 | 
|---|
 | 188 |                                                                            // When you set this property, the text of the range or selection is replaced.
 | 
|---|
 | 189 |         BringWordToFront(OldList, NewList);
 | 
|---|
 | 190 |         MSWord.ActiveDocument.Content.SpellingChecked := False;
 | 
|---|
 | 191 |         MSWord.ActiveDocument.Content.GrammarChecked := False;
 | 
|---|
 | 192 | 
 | 
|---|
 | 193 |         case ACheck of
 | 
|---|
 | 194 |           SPELL_CHECK  :  begin
 | 
|---|
 | 195 |                             MSWord.ActiveDocument.Content.CheckSpelling;                       // ToolsSpelling
 | 
|---|
 | 196 |                             FinishedChecking := MSWord.ActiveDocument.Content.SpellingChecked;
 | 
|---|
 | 197 |                           end;
 | 
|---|
 | 198 |           GRAMMAR_CHECK:  begin
 | 
|---|
 | 199 |                             MSWord.ActiveDocument.Content.CheckGrammar;                       // ToolsGrammar
 | 
|---|
 | 200 |                             FinishedChecking := MSWord.ActiveDocument.Content.GrammarChecked;
 | 
|---|
 | 201 |                           end;
 | 
|---|
 | 202 |         end;
 | 
|---|
 | 203 |         if FinishedChecking then    // not cancelled?
 | 
|---|
 | 204 |           NoLFText := MSWord.ActiveDocument.Content.Text                   // EditSelectAll
 | 
|---|
 | 205 |         else
 | 
|---|
 | 206 |           NoLFText := '';
 | 
|---|
 | 207 |       finally
 | 
|---|
 | 208 |         Screen.Cursor := crDefault;
 | 
|---|
 | 209 |         MSWord.Application.Options.SaveInterval := OldSaveInterval;
 | 
|---|
 | 210 |         case ACheck of
 | 
|---|
 | 211 |           SPELL_CHECK  :  FinishedChecking := MSWord.ActiveDocument.Content.SpellingChecked;
 | 
|---|
 | 212 |           GRAMMAR_CHECK:  FinishedChecking := MSWord.ActiveDocument.Content.GrammarChecked;   
 | 
|---|
 | 213 |         end;
 | 
|---|
 | 214 |         MSWord.Quit(wdDoNotSaveChanges);
 | 
|---|
 | 215 |         VarClear(MSWord);
 | 
|---|
 | 216 |       end;
 | 
|---|
 | 217 |     finally
 | 
|---|
 | 218 |       OldList.Free;
 | 
|---|
 | 219 |       NewList.Free;
 | 
|---|
 | 220 |     end;
 | 
|---|
 | 221 |   except
 | 
|---|
 | 222 |     on E: Exception do
 | 
|---|
 | 223 |       begin
 | 
|---|
 | 224 |         ErrMsg := E.Message;
 | 
|---|
 | 225 |         FinishedChecking := False;
 | 
|---|
 | 226 |       end;
 | 
|---|
 | 227 |   end;
 | 
|---|
 | 228 | 
 | 
|---|
 | 229 |   Screen.Cursor := crDefault;
 | 
|---|
 | 230 |   Application.BringToFront;
 | 
|---|
 | 231 |   if FinishedChecking then
 | 
|---|
 | 232 |     begin
 | 
|---|
 | 233 |       if (Length(NoLFText) > 0) then
 | 
|---|
 | 234 |         begin
 | 
|---|
 | 235 |           LFText := '';
 | 
|---|
 | 236 |           for i := 1 to Length(NoLFText) do
 | 
|---|
 | 237 |           begin
 | 
|---|
 | 238 |             OneChar := NoLFText[i];
 | 
|---|
 | 239 |             LFText := LFText + OneChar;
 | 
|---|
 | 240 |             if OneChar = #13 then LFText := LFText + #10;
 | 
|---|
 | 241 |           end;
 | 
|---|
 | 242 |           with AnotherEditControl do if Lines.Count > 0 then
 | 
|---|
 | 243 |             begin
 | 
|---|
 | 244 |               Text := LFText;
 | 
|---|
 | 245 |               if FirstLineBlank then Text := OldLine0 + Text;
 | 
|---|
 | 246 |             end;
 | 
|---|
 | 247 |           case ACheck of
 | 
|---|
 | 248 |             SPELL_CHECK  : MsgText := TX_SPELL_COMPLETE;
 | 
|---|
 | 249 |             GRAMMAR_CHECK: MsgText := TX_GRAMMAR_COMPLETE;
 | 
|---|
 | 250 |             else           MsgText := ''
 | 
|---|
 | 251 |           end;
 | 
|---|
 | 252 |           Application.MessageBox(PChar(MsgText), PChar(Application.Title), MB_ICONINFORMATION);
 | 
|---|
 | 253 |         end
 | 
|---|
 | 254 |       else
 | 
|---|
 | 255 |         begin
 | 
|---|
 | 256 |           case ACheck of
 | 
|---|
 | 257 |             SPELL_CHECK  : MsgText := TX_SPELL_CANCELLED;
 | 
|---|
 | 258 |             GRAMMAR_CHECK: MsgText := TX_GRAMMAR_CANCELLED;
 | 
|---|
 | 259 |             else           MsgText := ''
 | 
|---|
 | 260 |           end;
 | 
|---|
 | 261 |           Application.MessageBox(PChar(MsgText + CR_LF + CR_LF + TX_NO_CORRECTIONS), PChar(Application.Title), MB_ICONINFORMATION);
 | 
|---|
 | 262 |         end;
 | 
|---|
 | 263 |     end
 | 
|---|
 | 264 |   else   // error during spell or grammar check
 | 
|---|
 | 265 |     begin
 | 
|---|
 | 266 |       case ACheck of
 | 
|---|
 | 267 |         SPELL_CHECK  :  MsgText := TX_SPELL_ABORT;
 | 
|---|
 | 268 |         GRAMMAR_CHECK:  MsgText := TX_GRAMMAR_ABORT;
 | 
|---|
 | 269 |         else            MsgText := ''
 | 
|---|
 | 270 |       end;
 | 
|---|
 | 271 |       if ErrMsg = '' then ErrMsg := TX_NO_DETAILS;
 | 
|---|
 | 272 |       Application.MessageBox(PChar(MsgText + CR_LF + ErrMsg + CR_LF + CR_LF + TX_NO_CORRECTIONS), PChar(Application.Title), MB_ICONWARNING);
 | 
|---|
 | 273 |     end;
 | 
|---|
 | 274 |   SendMessage(TRichEdit(AnotherEditControl).Handle, WM_VSCROLL, SB_TOP, 0);
 | 
|---|
 | 275 |   AnotherEditControl.SetFocus;
 | 
|---|
 | 276 | end;
 | 
|---|
 | 277 | 
 | 
|---|
 | 278 | procedure SpellCheckForControl(AnEditControl: TCustomMemo);
 | 
|---|
 | 279 | begin
 | 
|---|
 | 280 |   if AnEditControl = nil then Exit;
 | 
|---|
 | 281 |   SpellAndGrammarCheckForControl(AnEditControl, SPELL_CHECK);
 | 
|---|
 | 282 | end;
 | 
|---|
 | 283 | 
 | 
|---|
 | 284 | procedure GrammarCheckForControl(AnEditControl: TCustomMemo);
 | 
|---|
 | 285 | begin
 | 
|---|
 | 286 |   if AnEditControl = nil then Exit;
 | 
|---|
 | 287 |   SpellAndGrammarCheckForControl(AnEditControl, GRAMMAR_CHECK);
 | 
|---|
 | 288 | end;
 | 
|---|
 | 289 | 
 | 
|---|
 | 290 | 
 | 
|---|
 | 291 | end.
 | 
|---|