Home | History | Annotate | Download | only in cocoa
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #import "chrome/browser/ui/cocoa/dock_icon.h"
      6 
      7 #include "base/logging.h"
      8 #include "base/mac/bundle_locations.h"
      9 #include "base/mac/scoped_nsobject.h"
     10 #include "content/public/browser/browser_thread.h"
     11 #include "ui/gfx/scoped_ns_graphics_context_save_gstate_mac.h"
     12 
     13 using content::BrowserThread;
     14 
     15 namespace {
     16 
     17 // The fraction of the size of the dock icon that the badge is.
     18 const float kBadgeFraction = 0.4f;
     19 
     20 // The indentation of the badge.
     21 const float kBadgeIndent = 5.0f;
     22 
     23 // The maximum update rate for the dock icon. 200ms = 5fps.
     24 const int64 kUpdateFrequencyMs = 200;
     25 
     26 }  // namespace
     27 
     28 // A view that draws our dock tile.
     29 @interface DockTileView : NSView {
     30  @private
     31   int downloads_;
     32   BOOL indeterminate_;
     33   float progress_;
     34 }
     35 
     36 // Indicates how many downloads are in progress.
     37 @property (nonatomic) int downloads;
     38 
     39 // Indicates whether the progress indicator should be in an indeterminate state
     40 // or not.
     41 @property (nonatomic) BOOL indeterminate;
     42 
     43 // Indicates the amount of progress made of the download. Ranges from [0..1].
     44 @property (nonatomic) float progress;
     45 
     46 @end
     47 
     48 @implementation DockTileView
     49 
     50 @synthesize downloads = downloads_;
     51 @synthesize indeterminate = indeterminate_;
     52 @synthesize progress = progress_;
     53 
     54 - (void)drawRect:(NSRect)dirtyRect {
     55   // Not -[NSApplication applicationIconImage]; that fails to return a pasted
     56   // custom icon.
     57   NSString* appPath = [base::mac::MainBundle() bundlePath];
     58   NSImage* appIcon = [[NSWorkspace sharedWorkspace] iconForFile:appPath];
     59   [appIcon drawInRect:[self bounds]
     60              fromRect:NSZeroRect
     61             operation:NSCompositeSourceOver
     62              fraction:1.0];
     63 
     64   if (downloads_ == 0)
     65     return;
     66 
     67   NSRect badgeRect = [self bounds];
     68   badgeRect.size.height = (int)(kBadgeFraction * badgeRect.size.height);
     69   int newWidth = kBadgeFraction * NSWidth(badgeRect);
     70   badgeRect.origin.x = NSWidth(badgeRect) - newWidth;
     71   badgeRect.size.width = newWidth;
     72 
     73   CGFloat badgeRadius = NSMidY(badgeRect);
     74 
     75   badgeRect.origin.x -= kBadgeIndent;
     76   badgeRect.origin.y += kBadgeIndent;
     77 
     78   NSPoint badgeCenter = NSMakePoint(NSMidX(badgeRect), NSMidY(badgeRect));
     79 
     80   // Background
     81   NSColor* backgroundColor = [NSColor colorWithCalibratedRed:0.85
     82                                                        green:0.85
     83                                                         blue:0.85
     84                                                        alpha:1.0];
     85   NSColor* backgroundHighlight =
     86       [backgroundColor blendedColorWithFraction:0.85
     87                                         ofColor:[NSColor whiteColor]];
     88   base::scoped_nsobject<NSGradient> backgroundGradient(
     89       [[NSGradient alloc] initWithStartingColor:backgroundHighlight
     90                                     endingColor:backgroundColor]);
     91   NSBezierPath* badgeEdge = [NSBezierPath bezierPathWithOvalInRect:badgeRect];
     92   {
     93     gfx::ScopedNSGraphicsContextSaveGState scopedGState;
     94     [badgeEdge addClip];
     95     [backgroundGradient drawFromCenter:badgeCenter
     96                                 radius:0.0
     97                               toCenter:badgeCenter
     98                                 radius:badgeRadius
     99                                options:0];
    100   }
    101 
    102   // Slice
    103   if (!indeterminate_) {
    104     NSColor* sliceColor = [NSColor colorWithCalibratedRed:0.45
    105                                                     green:0.8
    106                                                      blue:0.25
    107                                                     alpha:1.0];
    108     NSColor* sliceHighlight =
    109         [sliceColor blendedColorWithFraction:0.4
    110                                      ofColor:[NSColor whiteColor]];
    111     base::scoped_nsobject<NSGradient> sliceGradient(
    112         [[NSGradient alloc] initWithStartingColor:sliceHighlight
    113                                       endingColor:sliceColor]);
    114     NSBezierPath* progressSlice;
    115     if (progress_ >= 1.0) {
    116       progressSlice = [NSBezierPath bezierPathWithOvalInRect:badgeRect];
    117     } else {
    118       CGFloat endAngle = 90.0 - 360.0 * progress_;
    119       if (endAngle < 0.0)
    120         endAngle += 360.0;
    121       progressSlice = [NSBezierPath bezierPath];
    122       [progressSlice moveToPoint:badgeCenter];
    123       [progressSlice appendBezierPathWithArcWithCenter:badgeCenter
    124                                                 radius:badgeRadius
    125                                             startAngle:90.0
    126                                               endAngle:endAngle
    127                                              clockwise:YES];
    128       [progressSlice closePath];
    129     }
    130     gfx::ScopedNSGraphicsContextSaveGState scopedGState;
    131     [progressSlice addClip];
    132     [sliceGradient drawFromCenter:badgeCenter
    133                            radius:0.0
    134                          toCenter:badgeCenter
    135                            radius:badgeRadius
    136                           options:0];
    137   }
    138 
    139   // Edge
    140   {
    141     gfx::ScopedNSGraphicsContextSaveGState scopedGState;
    142     [[NSColor whiteColor] set];
    143     base::scoped_nsobject<NSShadow> shadow([[NSShadow alloc] init]);
    144     [shadow.get() setShadowOffset:NSMakeSize(0, -2)];
    145     [shadow setShadowBlurRadius:2];
    146     [shadow set];
    147     [badgeEdge setLineWidth:2];
    148     [badgeEdge stroke];
    149   }
    150 
    151   // Download count
    152   base::scoped_nsobject<NSNumberFormatter> formatter(
    153       [[NSNumberFormatter alloc] init]);
    154   NSString* countString =
    155       [formatter stringFromNumber:[NSNumber numberWithInt:downloads_]];
    156 
    157   base::scoped_nsobject<NSShadow> countShadow([[NSShadow alloc] init]);
    158   [countShadow setShadowBlurRadius:3.0];
    159   [countShadow.get() setShadowColor:[NSColor whiteColor]];
    160   [countShadow.get() setShadowOffset:NSMakeSize(0.0, 0.0)];
    161   NSMutableDictionary* countAttrsDict =
    162       [NSMutableDictionary dictionaryWithObjectsAndKeys:
    163           [NSColor blackColor], NSForegroundColorAttributeName,
    164           countShadow.get(), NSShadowAttributeName,
    165           nil];
    166   CGFloat countFontSize = badgeRadius;
    167   NSSize countSize = NSZeroSize;
    168   base::scoped_nsobject<NSAttributedString> countAttrString;
    169   while (1) {
    170     NSFont* countFont = [NSFont fontWithName:@"Helvetica-Bold"
    171                                         size:countFontSize];
    172 
    173     // This will generally be plain Helvetica.
    174     if (!countFont)
    175       countFont = [NSFont userFontOfSize:countFontSize];
    176 
    177     // Continued failure would generate an NSException.
    178     if (!countFont)
    179       break;
    180 
    181     [countAttrsDict setObject:countFont forKey:NSFontAttributeName];
    182     countAttrString.reset(
    183         [[NSAttributedString alloc] initWithString:countString
    184                                         attributes:countAttrsDict]);
    185     countSize = [countAttrString size];
    186     if (countSize.width > badgeRadius * 1.5) {
    187       countFontSize -= 1.0;
    188     } else {
    189       break;
    190     }
    191   }
    192 
    193   NSPoint countOrigin = badgeCenter;
    194   countOrigin.x -= countSize.width / 2;
    195   countOrigin.y -= countSize.height / 2.2;  // tweak; otherwise too low
    196 
    197   [countAttrString.get() drawAtPoint:countOrigin];
    198 }
    199 
    200 @end
    201 
    202 
    203 @implementation DockIcon
    204 
    205 + (DockIcon*)sharedDockIcon {
    206   static DockIcon* icon;
    207   if (!icon) {
    208     NSDockTile* dockTile = [[NSApplication sharedApplication] dockTile];
    209 
    210     base::scoped_nsobject<DockTileView> dockTileView(
    211         [[DockTileView alloc] init]);
    212     [dockTile setContentView:dockTileView];
    213 
    214     icon = [[DockIcon alloc] init];
    215   }
    216 
    217   return icon;
    218 }
    219 
    220 - (void)updateIcon {
    221   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    222   static base::TimeDelta updateFrequency =
    223       base::TimeDelta::FromMilliseconds(kUpdateFrequencyMs);
    224 
    225   base::TimeTicks now = base::TimeTicks::Now();
    226   base::TimeDelta timeSinceLastUpdate = now - lastUpdate_;
    227   if (!forceUpdate_ && timeSinceLastUpdate < updateFrequency)
    228     return;
    229 
    230   lastUpdate_ = now;
    231   forceUpdate_ = NO;
    232 
    233   NSDockTile* dockTile = [[NSApplication sharedApplication] dockTile];
    234 
    235   [dockTile display];
    236 }
    237 
    238 - (void)setDownloads:(int)downloads {
    239   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    240   NSDockTile* dockTile = [[NSApplication sharedApplication] dockTile];
    241   DockTileView* dockTileView = (DockTileView*)([dockTile contentView]);
    242 
    243   if (downloads != [dockTileView downloads]) {
    244     [dockTileView setDownloads:downloads];
    245     forceUpdate_ = YES;
    246   }
    247 }
    248 
    249 - (void)setIndeterminate:(BOOL)indeterminate {
    250   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    251   NSDockTile* dockTile = [[NSApplication sharedApplication] dockTile];
    252   DockTileView* dockTileView = (DockTileView*)([dockTile contentView]);
    253 
    254   if (indeterminate != [dockTileView indeterminate]) {
    255     [dockTileView setIndeterminate:indeterminate];
    256     forceUpdate_ = YES;
    257   }
    258 }
    259 
    260 - (void)setProgress:(float)progress {
    261   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    262   NSDockTile* dockTile = [[NSApplication sharedApplication] dockTile];
    263   DockTileView* dockTileView = (DockTileView*)([dockTile contentView]);
    264 
    265   [dockTileView setProgress:progress];
    266 }
    267 
    268 @end
    269