web_contents_zoom_controller.cc 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283
  1. // Copyright (c) 2017 GitHub, Inc.
  2. // Use of this source code is governed by the MIT license that can be
  3. // found in the LICENSE file.
  4. #include "shell/browser/web_contents_zoom_controller.h"
  5. #include "content/public/browser/navigation_details.h"
  6. #include "content/public/browser/navigation_entry.h"
  7. #include "content/public/browser/navigation_handle.h"
  8. #include "content/public/browser/render_frame_host.h"
  9. #include "content/public/browser/render_process_host.h"
  10. #include "content/public/browser/render_view_host.h"
  11. #include "content/public/browser/web_contents.h"
  12. #include "content/public/common/page_type.h"
  13. #include "net/base/url_util.h"
  14. #include "third_party/blink/public/common/page/page_zoom.h"
  15. namespace electron {
  16. WebContentsZoomController::WebContentsZoomController(
  17. content::WebContents* web_contents)
  18. : content::WebContentsObserver(web_contents) {
  19. default_zoom_factor_ = kPageZoomEpsilon;
  20. host_zoom_map_ = content::HostZoomMap::GetForWebContents(web_contents);
  21. }
  22. WebContentsZoomController::~WebContentsZoomController() = default;
  23. void WebContentsZoomController::AddObserver(
  24. WebContentsZoomController::Observer* observer) {
  25. observers_.AddObserver(observer);
  26. }
  27. void WebContentsZoomController::RemoveObserver(
  28. WebContentsZoomController::Observer* observer) {
  29. observers_.RemoveObserver(observer);
  30. }
  31. void WebContentsZoomController::SetEmbedderZoomController(
  32. WebContentsZoomController* controller) {
  33. embedder_zoom_controller_ = controller;
  34. }
  35. void WebContentsZoomController::SetZoomLevel(double level) {
  36. if (!web_contents()->GetRenderViewHost()->IsRenderViewLive() ||
  37. blink::PageZoomValuesEqual(GetZoomLevel(), level) ||
  38. zoom_mode_ == ZoomMode::kDisabled)
  39. return;
  40. int render_process_id =
  41. web_contents()->GetRenderViewHost()->GetProcess()->GetID();
  42. int render_view_id = web_contents()->GetRenderViewHost()->GetRoutingID();
  43. if (zoom_mode_ == ZoomMode::kManual) {
  44. zoom_level_ = level;
  45. for (Observer& observer : observers_)
  46. observer.OnZoomLevelChanged(web_contents(), level, true);
  47. return;
  48. }
  49. content::HostZoomMap* zoom_map =
  50. content::HostZoomMap::GetForWebContents(web_contents());
  51. if (zoom_mode_ == ZoomMode::kIsolated ||
  52. zoom_map->UsesTemporaryZoomLevel(render_process_id, render_view_id)) {
  53. zoom_map->SetTemporaryZoomLevel(render_process_id, render_view_id, level);
  54. // Notify observers of zoom level changes.
  55. for (Observer& observer : observers_)
  56. observer.OnZoomLevelChanged(web_contents(), level, true);
  57. } else {
  58. content::HostZoomMap::SetZoomLevel(web_contents(), level);
  59. // Notify observers of zoom level changes.
  60. for (Observer& observer : observers_)
  61. observer.OnZoomLevelChanged(web_contents(), level, false);
  62. }
  63. }
  64. double WebContentsZoomController::GetZoomLevel() {
  65. return zoom_mode_ == ZoomMode::kManual
  66. ? zoom_level_
  67. : content::HostZoomMap::GetZoomLevel(web_contents());
  68. }
  69. void WebContentsZoomController::SetDefaultZoomFactor(double factor) {
  70. default_zoom_factor_ = factor;
  71. }
  72. double WebContentsZoomController::GetDefaultZoomFactor() {
  73. return default_zoom_factor_;
  74. }
  75. void WebContentsZoomController::SetTemporaryZoomLevel(double level) {
  76. old_process_id_ = web_contents()->GetRenderViewHost()->GetProcess()->GetID();
  77. old_view_id_ = web_contents()->GetRenderViewHost()->GetRoutingID();
  78. host_zoom_map_->SetTemporaryZoomLevel(old_process_id_, old_view_id_, level);
  79. // Notify observers of zoom level changes.
  80. for (Observer& observer : observers_)
  81. observer.OnZoomLevelChanged(web_contents(), level, true);
  82. }
  83. bool WebContentsZoomController::UsesTemporaryZoomLevel() {
  84. int render_process_id =
  85. web_contents()->GetRenderViewHost()->GetProcess()->GetID();
  86. int render_view_id = web_contents()->GetRenderViewHost()->GetRoutingID();
  87. return host_zoom_map_->UsesTemporaryZoomLevel(render_process_id,
  88. render_view_id);
  89. }
  90. void WebContentsZoomController::SetZoomMode(ZoomMode new_mode) {
  91. if (new_mode == zoom_mode_)
  92. return;
  93. content::HostZoomMap* zoom_map =
  94. content::HostZoomMap::GetForWebContents(web_contents());
  95. int render_process_id =
  96. web_contents()->GetRenderViewHost()->GetProcess()->GetID();
  97. int render_view_id = web_contents()->GetRenderViewHost()->GetRoutingID();
  98. double original_zoom_level = GetZoomLevel();
  99. switch (new_mode) {
  100. case ZoomMode::kDefault: {
  101. content::NavigationEntry* entry =
  102. web_contents()->GetController().GetLastCommittedEntry();
  103. if (entry) {
  104. GURL url = content::HostZoomMap::GetURLFromEntry(entry);
  105. std::string host = net::GetHostOrSpecFromURL(url);
  106. if (zoom_map->HasZoomLevel(url.scheme(), host)) {
  107. // If there are other tabs with the same origin, then set this tab's
  108. // zoom level to match theirs. The temporary zoom level will be
  109. // cleared below, but this call will make sure this tab re-draws at
  110. // the correct zoom level.
  111. double origin_zoom_level =
  112. zoom_map->GetZoomLevelForHostAndScheme(url.scheme(), host);
  113. zoom_map->SetTemporaryZoomLevel(render_process_id, render_view_id,
  114. origin_zoom_level);
  115. } else {
  116. // The host will need a level prior to removing the temporary level.
  117. // We don't want the zoom level to change just because we entered
  118. // default mode.
  119. zoom_map->SetZoomLevelForHost(host, original_zoom_level);
  120. }
  121. }
  122. // Remove per-tab zoom data for this tab. No event callback expected.
  123. zoom_map->ClearTemporaryZoomLevel(render_process_id, render_view_id);
  124. break;
  125. }
  126. case ZoomMode::kIsolated: {
  127. // Unless the zoom mode was |ZoomMode::kDisabled| before this call, the
  128. // page needs an initial isolated zoom back to the same level it was at
  129. // in the other mode.
  130. if (zoom_mode_ != ZoomMode::kDisabled) {
  131. zoom_map->SetTemporaryZoomLevel(render_process_id, render_view_id,
  132. original_zoom_level);
  133. } else {
  134. // When we don't call any HostZoomMap set functions, we send the event
  135. // manually.
  136. for (Observer& observer : observers_)
  137. observer.OnZoomLevelChanged(web_contents(), original_zoom_level,
  138. false);
  139. }
  140. break;
  141. }
  142. case ZoomMode::kManual: {
  143. // Unless the zoom mode was |ZoomMode::kDisabled| before this call, the
  144. // page needs to be resized to the default zoom. While in manual mode,
  145. // the zoom level is handled independently.
  146. if (zoom_mode_ != ZoomMode::kDisabled) {
  147. zoom_map->SetTemporaryZoomLevel(render_process_id, render_view_id,
  148. GetDefaultZoomLevel());
  149. zoom_level_ = original_zoom_level;
  150. } else {
  151. // When we don't call any HostZoomMap set functions, we send the event
  152. // manually.
  153. for (Observer& observer : observers_)
  154. observer.OnZoomLevelChanged(web_contents(), original_zoom_level,
  155. false);
  156. }
  157. break;
  158. }
  159. case ZoomMode::kDisabled: {
  160. // The page needs to be zoomed back to default before disabling the zoom
  161. zoom_map->SetTemporaryZoomLevel(render_process_id, render_view_id,
  162. GetDefaultZoomLevel());
  163. break;
  164. }
  165. }
  166. zoom_mode_ = new_mode;
  167. }
  168. void WebContentsZoomController::ResetZoomModeOnNavigationIfNeeded(
  169. const GURL& url) {
  170. if (zoom_mode_ != ZoomMode::kIsolated && zoom_mode_ != ZoomMode::kManual)
  171. return;
  172. int render_process_id =
  173. web_contents()->GetRenderViewHost()->GetProcess()->GetID();
  174. int render_view_id = web_contents()->GetRenderViewHost()->GetRoutingID();
  175. content::HostZoomMap* zoom_map =
  176. content::HostZoomMap::GetForWebContents(web_contents());
  177. zoom_level_ = zoom_map->GetDefaultZoomLevel();
  178. double new_zoom_level = zoom_map->GetZoomLevelForHostAndScheme(
  179. url.scheme(), net::GetHostOrSpecFromURL(url));
  180. for (Observer& observer : observers_)
  181. observer.OnZoomLevelChanged(web_contents(), new_zoom_level, false);
  182. zoom_map->ClearTemporaryZoomLevel(render_process_id, render_view_id);
  183. zoom_mode_ = ZoomMode::kDefault;
  184. }
  185. void WebContentsZoomController::DidFinishNavigation(
  186. content::NavigationHandle* navigation_handle) {
  187. if (!navigation_handle->IsInMainFrame() || !navigation_handle->HasCommitted())
  188. return;
  189. if (navigation_handle->IsErrorPage()) {
  190. content::HostZoomMap::SendErrorPageZoomLevelRefresh(web_contents());
  191. return;
  192. }
  193. ResetZoomModeOnNavigationIfNeeded(navigation_handle->GetURL());
  194. SetZoomFactorOnNavigationIfNeeded(navigation_handle->GetURL());
  195. }
  196. void WebContentsZoomController::WebContentsDestroyed() {
  197. for (Observer& observer : observers_)
  198. observer.OnZoomControllerWebContentsDestroyed();
  199. observers_.Clear();
  200. embedder_zoom_controller_ = nullptr;
  201. }
  202. void WebContentsZoomController::RenderFrameHostChanged(
  203. content::RenderFrameHost* old_host,
  204. content::RenderFrameHost* new_host) {
  205. // If our associated HostZoomMap changes, update our event subscription.
  206. content::HostZoomMap* new_host_zoom_map =
  207. content::HostZoomMap::GetForWebContents(web_contents());
  208. if (new_host_zoom_map == host_zoom_map_)
  209. return;
  210. host_zoom_map_ = new_host_zoom_map;
  211. }
  212. void WebContentsZoomController::SetZoomFactorOnNavigationIfNeeded(
  213. const GURL& url) {
  214. if (blink::PageZoomValuesEqual(GetDefaultZoomFactor(), kPageZoomEpsilon))
  215. return;
  216. if (host_zoom_map_->UsesTemporaryZoomLevel(old_process_id_, old_view_id_)) {
  217. host_zoom_map_->ClearTemporaryZoomLevel(old_process_id_, old_view_id_);
  218. }
  219. if (embedder_zoom_controller_ &&
  220. embedder_zoom_controller_->UsesTemporaryZoomLevel()) {
  221. double level = embedder_zoom_controller_->GetZoomLevel();
  222. SetTemporaryZoomLevel(level);
  223. return;
  224. }
  225. // When kZoomFactor is available, it takes precedence over
  226. // pref store values but if the host has zoom factor set explicitly
  227. // then it takes precedence.
  228. // pref store < kZoomFactor < setZoomLevel
  229. std::string host = net::GetHostOrSpecFromURL(url);
  230. std::string scheme = url.scheme();
  231. double zoom_factor = GetDefaultZoomFactor();
  232. double zoom_level = blink::PageZoomFactorToZoomLevel(zoom_factor);
  233. if (host_zoom_map_->HasZoomLevel(scheme, host)) {
  234. zoom_level = host_zoom_map_->GetZoomLevelForHostAndScheme(scheme, host);
  235. }
  236. if (blink::PageZoomValuesEqual(zoom_level, GetZoomLevel()))
  237. return;
  238. SetZoomLevel(zoom_level);
  239. }
  240. WEB_CONTENTS_USER_DATA_KEY_IMPL(WebContentsZoomController)
  241. } // namespace electron