Subversion Repositories fastphp

Rev

Rev 61 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
8 daniel-mar 1
unit BrowserMain;
2
 
25 daniel-mar 3
{$Include 'FastPHP.inc'}
4
 
8 daniel-mar 5
interface
6
 
7
uses
27 daniel-mar 8
  // TODO: "{$IFDEF USE_SHDOCVW_TLB}_TLB{$ENDIF}" does not work with Delphi 10.2
60 daniel-mar 9
  //       so you have to change the reference SHDocVw / SHDocVw_TLB yourself
25 daniel-mar 10
  Windows, Messages, SysUtils, Variants, Classes, Graphics,
60 daniel-mar 11
  Controls, Forms, Dialogs, OleCtrls, SHDocVw, ExtCtrls, StrUtils,
25 daniel-mar 12
  StdCtrls, activex, UrlMon;
8 daniel-mar 13
 
14
type
15
  TForm2 = class(TForm)
16
    WebBrowser1: TWebBrowser;
17
    Timer1: TTimer;
91 daniel-mar 18
    OpenDialog3: TOpenDialog;
8 daniel-mar 19
    procedure Timer1Timer(Sender: TObject);
20
    procedure WebBrowser1BeforeNavigate2(ASender: TObject;
60 daniel-mar 21
      const pDisp: IDispatch; const URL, Flags, TargetFrameName, PostData,
8 daniel-mar 22
      Headers: OleVariant; var Cancel: WordBool);
60 daniel-mar 23
    procedure WebBrowser1WindowClosing(ASender: TObject;
24
      IsChildWindow: WordBool; var Cancel: WordBool);
12 daniel-mar 25
  strict private
26
    function EmbeddedWBQueryService(const rsid, iid: TGUID; out Obj{: IInterface}): HRESULT;
8 daniel-mar 27
  end;
28
 
29
var
30
  Form2: TForm2;
31
 
32
implementation
33
 
34
{$R *.dfm}
35
 
36
uses
9 daniel-mar 37
  WebBrowserUtils, FastPHPUtils, Functions, ShellAPI;
8 daniel-mar 38
 
39
// TODO: Add a lot of nice stuff to let the PHP script communicate with this host application
40
//       For example, allow window resizing etc.  (See Microsoft HTA for inspiration)
41
// TODO: Ajax gives Access Denied error... Create own security manager?
42
// TODO: History doesn't work?
43
// (All these ToDos: Also fix in the Editor)
9 daniel-mar 44
// TODO: kann man eventuell auch php dateien aus einer DLL rausziehen? das wäre TOLL!!!!
45
// TODO: headers... cookies...
11 daniel-mar 46
// TODO: WebBrowser1BeforeNavigate2 mit einem DLL-callback, sodass entwickler ihre eigenen fastphp:// links machen können, z.B. um DLL-Funktionen aufzurufen! (auch in JavaScript ansteuerbar?)
47
// TODO: let the website decide if the window is maximized etc, as well as it's caption, size and icon
19 daniel-mar 48
// TODO: Pass parameters (argv) to PHP
8 daniel-mar 49
 
12 daniel-mar 50
type
51
  TEmbeddedSecurityManager = class(TInterfacedObject, IInternetSecurityManager)
52
  public
53
    function GetSecuritySite(out ppSite: IInternetSecurityMgrSite): HResult; stdcall;
54
    function MapUrlToZone(pwszUrl: LPCWSTR; out dwZone: DWORD; dwFlags: DWORD): HResult; stdcall;
55
    function GetSecurityId(pwszUrl: LPCWSTR; pbSecurityId: Pointer; var cbSecurityId: DWORD; dwReserved: DWORD): HResult; stdcall;
56
    function ProcessUrlAction(pwszUrl: LPCWSTR; dwAction: DWORD; pPolicy: Pointer; cbPolicy: DWORD; pContext: Pointer; cbContext: DWORD; dwFlags, dwReserved: DWORD): HResult; stdcall;
57
    function QueryCustomPolicy(pwszUrl: LPCWSTR; const guidKey: TGUID; out pPolicy: Pointer; out cbPolicy: DWORD; pContext: Pointer; cbContext: DWORD; dwReserved: DWORD): HResult; stdcall;
58
    function SetZoneMapping(dwZone: DWORD; lpszPattern: PWideChar; dwFlags: DWORD): HResult; stdcall;
59
    function GetZoneMappings(dwZone: DWORD;out ppenumString: IEnumString; dwFlags: DWORD): HResult; stdcall;
60
    function SetSecuritySite(pSite: IInternetSecurityMgrSite): HResult; stdcall;
61
  end;
62
 
63
function TEmbeddedSecurityManager.SetSecuritySite(pSite: IInternetSecurityMgrSite): HResult; stdcall;
64
begin
65
  Result := INET_E_DEFAULT_ACTION;
66
end;
67
function TEmbeddedSecurityManager.GetSecuritySite(
68
  out ppSite: IInternetSecurityMgrSite): HResult; stdcall;
69
begin
70
  Result := INET_E_DEFAULT_ACTION;
71
end;
72
function TEmbeddedSecurityManager.GetSecurityId(pwszUrl: LPCWSTR; pbSecurityId: Pointer;
73
  var cbSecurityId: DWORD; dwReserved: DWORD): HResult; stdcall;
74
begin
75
  Result := INET_E_DEFAULT_ACTION;
76
end;
77
function TEmbeddedSecurityManager.ProcessUrlAction(pwszUrl: LPCWSTR; dwAction: DWORD;
78
  pPolicy: Pointer; cbPolicy: DWORD; pContext: Pointer; cbContext: DWORD;
79
  dwFlags, dwReserved: DWORD): HResult; stdcall;
80
begin
81
  // Result := INET_E_DEFAULT_ACTION;
82
 
83
  // TODO: Doesn't work... Cross-Domain is still not allowed.
84
  PDWORD(pPolicy)^ := URLPOLICY_ALLOW;
85
  Result := S_OK;
86
end;
87
function TEmbeddedSecurityManager.QueryCustomPolicy(pwszUrl: LPCWSTR; const guidKey: TGUID;
88
  out pPolicy: Pointer; out cbPolicy: DWORD; pContext: Pointer; cbContext: DWORD;
89
  dwReserved: DWORD): HResult; stdcall;
90
begin
91
  // Result := INET_E_DEFAULT_ACTION;
92
 
93
  // TODO: Doesn't work... Cross-Domain is still not allowed.
94
  PDWORD(pPolicy)^ := URLPOLICY_ALLOW;
95
  Result := S_OK;
96
end;
97
function TEmbeddedSecurityManager.SetZoneMapping(dwZone: DWORD; lpszPattern: PWideChar;
98
  dwFlags: DWORD): HResult; stdcall;
99
begin
100
  Result := INET_E_DEFAULT_ACTION;
101
end;
102
function TEmbeddedSecurityManager.GetZoneMappings(dwZone: DWORD;out ppenumString: IEnumString;
103
  dwFlags: DWORD): HResult; stdcall;
104
begin
105
  Result := INET_E_DEFAULT_ACTION;
106
end;
107
function TEmbeddedSecurityManager.MapUrlToZone(pwszUrl: LPCWSTR; out dwZone: DWORD; dwFlags: DWORD): HResult;
108
begin
16 daniel-mar 109
  dwZone := URLZONE_TRUSTED;
110
  Result := S_OK;
111
end;
12 daniel-mar 112
 
113
function TForm2.EmbeddedWBQueryService(const rsid, iid: TGUID; out Obj{: IInterface}): HRESULT;
114
var
91 daniel-mar 115
  sam: IInternetSecurityManager;
12 daniel-mar 116
begin
91 daniel-mar 117
  Result := E_NOINTERFACE;
12 daniel-mar 118
 
91 daniel-mar 119
  //rsid ==> Service Identifier
120
  //iid ==> Interface identifier
121
  if IsEqualGUID(rsid, IInternetSecurityManager) and IsEqualGUID(iid, IInternetSecurityManager) then
122
  begin
123
      sam := TEmbeddedSecurityManager.Create;
124
      IInterface(Obj) := sam;
125
      Result := S_OK;
126
  end;
12 daniel-mar 127
end;
128
 
8 daniel-mar 129
procedure TForm2.Timer1Timer(Sender: TObject);
130
var
131
  phpScript: string;
61 daniel-mar 132
  sl: TStringList;
91 daniel-mar 133
resourcestring
134
  SRunningScriptPleaseWait = 'Running script... please wait...';
135
  SPleaseEnterPHPScript = 'Please enter a PHP file to execute.';
136
  SFileDoesNotExist = 'File %s does not exist.';
137
  SFastPHP = 'ViaThinkSoft FastPHP';
8 daniel-mar 138
begin
139
  Timer1.Enabled := false;
140
  phpScript := ParamStr(1);
141
 
12 daniel-mar 142
  // Remove Security
143
  WebBrowser1.ServiceQuery := EmbeddedWBQueryService;
144
 
91 daniel-mar 145
  WebBrowser1.LoadHTML('<h1>'+SFastPHP+'</h1>'+SRunningScriptPleaseWait);
8 daniel-mar 146
 
147
  // TODO: nice HTML error/intro pages (as resource?)
148
  if phpScript = '' then
149
  begin
91 daniel-mar 150
    WebBrowser1.LoadHTML('<h1>'+SFastPHP+'</h1>'+SPleaseEnterPHPScript);
151
    if not OpenDialog3.Execute then
152
    begin
153
      Abort;
154
    end;
155
    phpScript := OpenDialog3.FileName;
8 daniel-mar 156
  end;
157
 
158
  if not FileExists(phpScript) then
159
  begin
91 daniel-mar 160
    WebBrowser1.LoadHTML(Format('<h1>'+SFastPHP+'</h1>'+SFileDoesNotExist, [phpScript]));
8 daniel-mar 161
    Abort;
162
  end;
163
 
164
  WebBrowser1.LoadHTML(RunPHPScript(phpScript), phpScript);
61 daniel-mar 165
 
166
  Application.ProcessMessages; // This is important, otherwise the metatags can't be read...
167
 
168
  sl := TStringList.Create;
169
  try
170
    WebBrowser1.ReadMetaTags(sl);
171
    // TODO: case insensitive
172
    if sl.Values['fastphp_title'] <> '' then Caption := sl.Values['fastphp_title'];
173
    if sl.Values['fastphp_width'] <> '' then ClientWidth := StrToInt(sl.Values['fastphp_width']);
174
    if sl.Values['fastphp_height'] <> '' then ClientHeight := StrToInt(sl.Values['fastphp_height']);
175
    // TODO: Add more attributes, like HTA applications had
176
    // TODO: Additionally implement "HTA:APPLICATION" element, see https://docs.microsoft.com/en-us/previous-versions//ms536495%28v%3dvs.85%29
177
  finally
178
    FreeAndNil(sl);
179
  end;
8 daniel-mar 180
end;
181
 
182
procedure TForm2.WebBrowser1BeforeNavigate2(ASender: TObject;
60 daniel-mar 183
  const pDisp: IDispatch; const URL, Flags, TargetFrameName, PostData,
8 daniel-mar 184
  Headers: OleVariant; var Cancel: WordBool);
185
var
16 daniel-mar 186
  myURL, myUrl2, getData: string;
8 daniel-mar 187
  p: integer;
9 daniel-mar 188
  background: boolean;
189
  ArgGet, ArgPost, ArgHeader: string;
91 daniel-mar 190
resourcestring
191
  SOnlyWorksInEditor = 'This action only works in FastPHP editor.';
8 daniel-mar 192
begin
91 daniel-mar 193
  background := Pos('background|', URL) >= 1; // do not translate
9 daniel-mar 194
 
8 daniel-mar 195
  {$REGION 'Line number references (PHP errors and warnings)'}
196
  if Copy(URL, 1, length(FASTPHP_GOTO_URI_PREFIX)) = FASTPHP_GOTO_URI_PREFIX then
197
  begin
9 daniel-mar 198
    // TODO: maybe we could even open that file in the editor!
91 daniel-mar 199
    ShowMessage(SOnlyWorksInEditor);
8 daniel-mar 200
    Cancel := true;
201
    Exit;
202
  end;
203
  {$ENDREGION}
204
 
205
  {$REGION 'Intelligent browser (executes PHP scripts)'}
206
  if URL <> 'about:blank' then
207
  begin
208
    myUrl := URL;
209
 
91 daniel-mar 210
    myurl := StringReplace(myurl, 'background|', '', []); // do not translate
9 daniel-mar 211
 
8 daniel-mar 212
    p := Pos('?', myUrl);
9 daniel-mar 213
    if p >= 1 then
214
    begin
215
      getData := copy(myURL, p+1, Length(myURL)-p);
216
      myURL := copy(myURL, 1, p-1);
217
    end
218
    else
219
    begin
220
      getData := '';
221
    end;
8 daniel-mar 222
 
16 daniel-mar 223
    myURL := StringReplace(myURL, 'http://wa.viathinksoft.de', '', []);
224
 
9 daniel-mar 225
    myURL := StringReplace(myURL, 'file:///', '', []);
226
    myURL := StringReplace(myURL, '/', '\', [rfReplaceAll]);
8 daniel-mar 227
 
9 daniel-mar 228
    // TODO: real myURL urldecode
229
    myURL := StringReplace(myURL, '+', ' ', []);
230
    myURL := StringReplace(myURL, '%20', ' ', []);
231
    myURL := StringReplace(myURL, '%%', '%', []);
232
 
233
    ArgHeader := '';
234
    ArgHeader := MyVarToStr(Headers);
235
    ArgHeader := StringReplace(ArgHeader, #13, '|CR|', [rfReplaceAll]);
236
    ArgHeader := StringReplace(ArgHeader, #10, '|LF|', [rfReplaceAll]);
237
 
11 daniel-mar 238
    // *.xphp is ViaThinkSoft's extension associated to FastPHPBrowser
239
    // This allows the "executable PHP scripts" to be executed via double click.--
240
    if FileExists(myURL) and (EndsText('.xphp', myURL) or EndsText('.php', myURL) or EndsText('.php3', myURL) or EndsText('.php4', myURL) or EndsText('.php5', myURL) or EndsText('.phps', myURL)) then
8 daniel-mar 241
    begin
9 daniel-mar 242
      if background then
243
      begin
244
        // TODO: how to detach the process?
245
        ShellExecute(0, 'open', PChar(GetPHPExe), PChar('"'+myURL+'" "'+ArgGet+'" "'+ArgPost+'" "'+ArgHeader+'"'), PChar(ExtractFileDir(Application.ExeName)), SW_HIDE);
246
      end
247
      else
248
      begin
249
        // TODO: somehow prepend fastphp_server.inc.php (populates the $_GET and $_POST arrays)
250
        // TODO: is there a maximal length for the command line?
251
        ArgGet := MyVarToStr(getData);
252
        ArgPost := MyVarToStr(PostData);
16 daniel-mar 253
 
254
        myUrl2 := myUrl;
255
        myUrl2 := StringReplace(myUrl2, '\', '/', [rfReplaceAll]);
256
        // TODO: real myURL urlencode
257
        myUrl2 := StringReplace(myUrl2, '%', '%%', []);
258
        //myUrl2 := StringReplace(myUrl2, ' ', '%20', []);
259
        myUrl2 := StringReplace(myUrl2, ' ', '+', []);
260
        myUrl2 := 'http://wa.viathinksoft.de/' + myUrl2;
261
 
42 daniel-mar 262
        // showmessage(myUrl2);
263
        WebBrowser1.LoadHTML(GetDosOutput('"'+GetPHPExe+'" -f "'+myURL+'" -- "'+ArgGet+'" "'+ArgPost+'" "'+ArgHeader+'"', ExtractFileDir(Application.ExeName)), myUrl2);
9 daniel-mar 264
      end;
8 daniel-mar 265
      Cancel := true;
266
    end;
267
  end;
268
  {$ENDREGION}
269
end;
270
 
60 daniel-mar 271
procedure TForm2.WebBrowser1WindowClosing(ASender: TObject;
272
  IsChildWindow: WordBool; var Cancel: WordBool);
273
begin
274
  Close;
275
  Cancel := true;
276
end;
277
 
8 daniel-mar 278
end.