diff options
author | rsesek@chromium.org <rsesek@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-10-25 16:55:16 +0000 |
---|---|---|
committer | rsesek@chromium.org <rsesek@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-10-25 16:55:16 +0000 |
commit | 44c8ca1a251f5b0dd3108f5088ec281d68358f7e (patch) | |
tree | 7df215e81b55c50d1cf9951d248bf184c69811f9 /chrome/browser/cocoa | |
parent | 17e0b44e09d940f8d6ee0f3011ded0f31996fd64 (diff) | |
download | chromium_src-44c8ca1a251f5b0dd3108f5088ec281d68358f7e.zip chromium_src-44c8ca1a251f5b0dd3108f5088ec281d68358f7e.tar.gz chromium_src-44c8ca1a251f5b0dd3108f5088ec281d68358f7e.tar.bz2 |
Schedule layout of the Page Info Bubble using RunnableMethod rather than |-performSelector:...|.
This will hopefully prevent a zombie crash because RunnableMethod is scoped to
a factory and will not run if the bridge gets deleted.
BUG=59619
TEST=Hard. Look at crash data.
Review URL: http://codereview.chromium.org/3979007
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@63740 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'chrome/browser/cocoa')
-rw-r--r-- | chrome/browser/cocoa/page_info_bubble_controller.mm | 32 |
1 files changed, 27 insertions, 5 deletions
diff --git a/chrome/browser/cocoa/page_info_bubble_controller.mm b/chrome/browser/cocoa/page_info_bubble_controller.mm index b10c1cf..c2c2d36 100644 --- a/chrome/browser/cocoa/page_info_bubble_controller.mm +++ b/chrome/browser/cocoa/page_info_bubble_controller.mm @@ -6,7 +6,9 @@ #include "app/l10n_util.h" #include "app/l10n_util_mac.h" +#include "base/message_loop.h" #include "base/sys_string_conversions.h" +#include "base/task.h" #include "chrome/browser/browser_list.h" #include "chrome/browser/cert_store.h" #include "chrome/browser/certificate_viewer.h" @@ -88,20 +90,33 @@ const CGFloat kTextXPosition = kTextXPositionNoImage + kImageSize + const CGFloat kTextWidth = kWindowWidth - (kImageSize + kImageSpacing + kFramePadding * 2); - // Bridge that listens for change notifications from the model. class PageInfoModelBubbleBridge : public PageInfoModel::PageInfoModelObserver { public: - PageInfoModelBubbleBridge() : controller_(nil) {} + PageInfoModelBubbleBridge() + : controller_(nil), + ALLOW_THIS_IN_INITIALIZER_LIST(task_factory_(this)) { + } // PageInfoModelObserver implementation. virtual void ModelChanged() { + // Check to see if a layout has already been scheduled. + if (!task_factory_.empty()) + return; + // Delay performing layout by a second so that all the animations from // InfoBubbleWindow and origin updates from BaseBubbleController finish, so // that we don't all race trying to change the frame's origin. - [controller_ performSelector:@selector(performLayout) - withObject:nil - afterDelay:1.0]; + // + // Using ScopedRunnableMethodFactory is superior here to |-performSelector:| + // because it will not retain its target; if the child outlives its parent, + // zombies get left behind (http://crbug.com/59619). This will also cancel + // the scheduled Tasks if the controller (and thus this bridge) get + // destroyed before the message can be delivered. + MessageLoop::current()->PostDelayedTask(FROM_HERE, + task_factory_.NewRunnableMethod( + &PageInfoModelBubbleBridge::PerformLayout), + 1000 /* milliseconds */); } // Sets the controller. @@ -110,7 +125,14 @@ class PageInfoModelBubbleBridge : public PageInfoModel::PageInfoModelObserver { } private: + void PerformLayout() { + [controller_ performLayout]; + } + PageInfoBubbleController* controller_; // weak + + // Factory that vends RunnableMethod tasks for scheduling layout. + ScopedRunnableMethodFactory<PageInfoModelBubbleBridge> task_factory_; }; } // namespace |