Subversion Repositories fastphp

Rev

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