web_contents_zoom_controller.cc 10 KB

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