ChromeMessageService.pas 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. unit ChromeMessageService;
  2. interface
  3. uses
  4. ceflib, Forms, Dialogs, Classes, SysUtils, BaseScope; //其它
  5. type
  6. TBrowserContext = class;
  7. TCustomRenderProcessHandler = class(TCefRenderProcessHandlerOwn)
  8. private
  9. FBrowserContexts: TStringList;
  10. function _CallBack(params, callback: string;
  11. ABrowserContext: TBrowserContext): Boolean;
  12. protected
  13. procedure OnWebKitInitialized; override;
  14. function OnProcessMessageReceived(const browser: ICefBrowser; sourceProcess: TCefProcessId; const message: ICefProcessMessage): Boolean; override;
  15. procedure OnBrowserCreated(const browser: ICefBrowser); override;
  16. procedure OnBrowserDestroyed(const browser: ICefBrowser); override;
  17. public
  18. constructor Create; override;
  19. destructor Destroy; override;
  20. function GetBrowerContext(ABrowser: ICefBrowser): TBrowserContext;
  21. procedure DeleteBrowerContext(ABrowser: ICefBrowser);
  22. end;
  23. TCefExtension = class(TCefv8HandlerOwn)
  24. private
  25. public
  26. function Execute(const name: ustring; const obj: ICefv8Value; const arguments: TCefv8ValueArray; var retval: ICefv8Value; var exception: ustring): Boolean; override;
  27. end;
  28. TBrowserContext = class(TInterfacedObject)
  29. private
  30. FScopes,
  31. FCallBacks: TStringList;
  32. FBrowser: ICefBrowser;
  33. public
  34. constructor Create(const browser: ICefBrowser);
  35. destructor Destroy; override;
  36. function GetScope(AGuid: string): TBaseScope;
  37. function GetCallBack(AGuid: string): TBaseCallBack;
  38. function AddCallBack(ACallBack: ICefv8Value): string;
  39. end;
  40. var
  41. ACefExtension: TCefExtension;
  42. implementation
  43. { TCustomRenderProcessHandler }
  44. var
  45. FCustomRenderProcessHandler: TCustomRenderProcessHandler;
  46. constructor TCustomRenderProcessHandler.Create;
  47. begin
  48. FBrowserContexts := TStringList.Create;
  49. inherited Create;
  50. end;
  51. procedure TCustomRenderProcessHandler.DeleteBrowerContext(
  52. ABrowser: ICefBrowser);
  53. var
  54. iIndex: Integer;
  55. begin
  56. if ABrowser = nil then
  57. Exit;
  58. iIndex := FBrowserContexts.IndexOf(IntToStr(ABrowser.Identifier));
  59. if iIndex > -1 then
  60. FBrowserContexts.Delete(iIndex);
  61. end;
  62. destructor TCustomRenderProcessHandler.Destroy;
  63. begin
  64. FBrowserContexts.Clear;
  65. FreeAndNil(FBrowserContexts);
  66. inherited;
  67. end;
  68. function TCustomRenderProcessHandler.GetBrowerContext(
  69. ABrowser: ICefBrowser): TBrowserContext;
  70. var
  71. iIndex: Integer;
  72. begin
  73. Result := nil;
  74. if ABrowser = nil then
  75. Exit;
  76. iIndex := FBrowserContexts.IndexOf(IntToStr(ABrowser.Identifier));
  77. if iIndex > -1 then
  78. Result := FBrowserContexts.Objects[iIndex] as TBrowserContext
  79. else
  80. begin
  81. Result := TBrowserContext.Create(ABrowser);
  82. FBrowserContexts.AddObject(IntToStr(ABrowser.Identifier), Result);
  83. end;
  84. end;
  85. procedure TCustomRenderProcessHandler.OnBrowserCreated(
  86. const browser: ICefBrowser);
  87. begin
  88. inherited;
  89. GetBrowerContext(browser);
  90. end;
  91. procedure TCustomRenderProcessHandler.OnBrowserDestroyed(
  92. const browser: ICefBrowser);
  93. begin
  94. inherited;
  95. DeleteBrowerContext(browser);
  96. end;
  97. function TCustomRenderProcessHandler._CallBack(params, callback: string; ABrowserContext: TBrowserContext): Boolean;
  98. var
  99. AFun: ICefv8Value;
  100. ACallBack: TBaseCallBack;
  101. Argm: TCefv8ValueArray;
  102. begin
  103. SetLength(Argm, 1);
  104. ACallBack := ABrowserContext.GetCallBack(callback);
  105. if (ACallBack = nil) and (ACallBack.CallBack = nil) then
  106. Exit;
  107. AFun := ACallBack.CallBack;
  108. if (AFun <> nil) and (AFun.IsFunction) then
  109. Exit;
  110. Argm[0] := TCefv8ValueRef.NewString(params);
  111. AFun.ExecuteFunction(nil, Argm);
  112. end;
  113. function TCustomRenderProcessHandler.OnProcessMessageReceived(const browser: ICefBrowser; sourceProcess: TCefProcessId; const message: ICefProcessMessage): Boolean;
  114. var
  115. AScope: TBaseScope;
  116. Argm: TCefv8ValueArray;
  117. ACefv8Context: ICefv8Context;
  118. ABrowserContext: TBrowserContext;
  119. AFun: ICefv8Value;
  120. iCount: Integer;
  121. begin
  122. iCount := message.ArgumentList.GetSize;
  123. if (iCount < 2) or (iCount > 3) then
  124. Exit;
  125. ABrowserContext := GetBrowerContext(browser);
  126. if ABrowserContext = nil then
  127. Exit;
  128. ACefv8Context := browser.MainFrame.GetV8Context;
  129. if (ACefv8Context = nil) or (not ACefv8Context.Enter) then
  130. Exit;
  131. try
  132. if message.ArgumentList.GetString(0) = '_CallBack' then
  133. begin
  134. _CallBack(message.ArgumentList.GetString(1), message.ArgumentList.GetString(2), ABrowserContext);
  135. end
  136. else
  137. begin
  138. AScope := ABrowserContext.GetScope(message.Name);
  139. // ShowMessage(message.Name);
  140. if AScope = nil then
  141. Exit;
  142. AScope.Handle(message);
  143. end;
  144. finally
  145. ACefv8Context.Exit;
  146. end;
  147. end;
  148. procedure TCustomRenderProcessHandler.OnWebKitInitialized;
  149. var
  150. AName, ACode: TCefString;
  151. begin
  152. AName := CefString('v8/lxtalkClient');
  153. ACode := CefString('var lxtalkClient = {}; (function() {lxtalkClient.Invoke = function(guid, method, params, obj) {native function Invoke(guid, method, params, obj);return Invoke(guid, method, params, obj);};})();');
  154. cef_register_extension(@AName, @ACode, CefGetData(ACefExtension));
  155. end;
  156. { TCefExtension }
  157. function TCefExtension.Execute(const name: ustring; const obj: ICefv8Value; const arguments: TCefv8ValueArray; var retval: ICefv8Value; var exception: ustring): Boolean;
  158. var
  159. msg: ICefProcessMessage;
  160. guid, method: string;
  161. g:TGUID;
  162. AScope: TBaseScope;
  163. ABrowser: ICefBrowser;
  164. ABrowserContext: TBrowserContext;
  165. begin
  166. if (Length(arguments) < 2) and (not arguments[0].IsString) and (not arguments[1].IsString) then
  167. begin
  168. exception := '参数不合法';
  169. Result := false;
  170. Exit;
  171. end;
  172. if SameText(arguments[1].GetStringValue, '_Register') and (Length(arguments) <> 4) and (not arguments[3].IsObject) then
  173. begin
  174. exception := '_Register 参数不合法';
  175. Result := false;
  176. Exit;
  177. end;
  178. ABrowser := TCefv8ContextRef.Current.Browser;
  179. ABrowserContext := FCustomRenderProcessHandler.GetBrowerContext(ABrowser);
  180. if ABrowserContext = nil then
  181. Exit;
  182. msg := TCefProcessMessageRef.New(arguments[0].GetStringValue);
  183. if SameText(arguments[1].GetStringValue, '_Register') then
  184. begin
  185. AScope := ABrowserContext.GetScope(arguments[0].GetStringValue);
  186. if AScope = nil then
  187. begin
  188. exception := '没有找到对应组件';
  189. Result := false;
  190. Exit;
  191. end;
  192. AScope.Scope := arguments[3];
  193. msg.ArgumentList.SetString(0, arguments[1].GetStringValue);
  194. msg.ArgumentList.SetString(1, arguments[2].GetStringValue);
  195. end
  196. else
  197. begin
  198. case Length(arguments) of
  199. 2:
  200. begin
  201. msg.ArgumentList.SetString(0, arguments[1].GetStringValue);
  202. end;
  203. 3:
  204. begin
  205. msg.ArgumentList.SetString(0, arguments[1].GetStringValue);
  206. if arguments[2].IsObject then
  207. begin
  208. msg.ArgumentList.SetString(1, '');
  209. msg.ArgumentList.SetString(2, ABrowserContext.AddCallBack(arguments[2]));
  210. end
  211. else
  212. if arguments[2].IsString then
  213. msg.ArgumentList.SetString(1, arguments[2].GetStringValue);
  214. end;
  215. 4:
  216. begin
  217. msg.ArgumentList.SetString(0, arguments[1].GetStringValue);
  218. msg.ArgumentList.SetString(1, arguments[2].GetStringValue);
  219. msg.ArgumentList.SetString(2, ABrowserContext.AddCallBack(arguments[3]));
  220. end;
  221. end;
  222. end;
  223. TCefv8ContextRef.Current.Browser.SendProcessMessage(PID_BROWSER, msg);
  224. Result := true;
  225. end;
  226. { TBrowserContext }
  227. function TBrowserContext.AddCallBack(ACallBack: ICefv8Value): string;
  228. var
  229. ABaseCallBack: TBaseCallBack;
  230. begin
  231. ABaseCallBack := TBaseCallBack.Create(ACallBack);
  232. FCallBacks.AddObject(ABaseCallBack.ID, ABaseCallBack);
  233. Result := ABaseCallBack.ID;
  234. end;
  235. constructor TBrowserContext.Create(const browser: ICefBrowser);
  236. begin
  237. FBrowser := browser;
  238. FScopes := TStringList.Create;
  239. FCallBacks := TStringList.Create;
  240. inherited Create();
  241. end;
  242. destructor TBrowserContext.Destroy;
  243. begin
  244. FScopes.Clear;
  245. FCallBacks.Clear;
  246. FreeAndNil(FScopes);
  247. FreeAndNil(FCallBacks);
  248. inherited;
  249. end;
  250. function TBrowserContext.GetCallBack(AGuid: string): TBaseCallBack;
  251. var
  252. iIndex: Integer;
  253. begin
  254. Result := nil;
  255. if AGuid = '' then
  256. Exit;
  257. iIndex := FCallBacks.IndexOf(AGuid);
  258. if iIndex > -1 then
  259. begin
  260. Result := FCallBacks.Objects[iIndex] as TBaseCallBack;
  261. FCallBacks.Delete(iIndex);
  262. end;
  263. end;
  264. function TBrowserContext.GetScope(AGuid: string): TBaseScope;
  265. var
  266. i: Integer;
  267. AClass: TBaseScopeClass;
  268. begin
  269. Result := nil;
  270. i := FScopes.Indexof(AGuid);
  271. if i > -1 then
  272. begin
  273. Result := FScopes.Objects[i] as TBaseScope;
  274. Exit;
  275. end;
  276. AClass := TScopeFactory.Current.GetScopeClass(AGuid);
  277. if AClass = nil then
  278. Exit;
  279. Result := AClass.Create;
  280. FScopes.AddObject(AGuid, Result);
  281. end;
  282. initialization
  283. CefRemoteDebuggingPort := 9000;
  284. FCustomRenderProcessHandler := TCustomRenderProcessHandler.Create;
  285. CefRenderProcessHandler := FCustomRenderProcessHandler;
  286. CefBrowserProcessHandler := TCefBrowserProcessHandlerOwn.Create;
  287. ACefExtension := TCefExtension.Create;
  288. finalization
  289. end.