2021-08-04 00:31:12 +00:00
|
|
|
// Copyright (c) 2021 Microsoft, Inc.
|
|
|
|
// Use of this source code is governed by the MIT license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#include "shell/browser/ui/cocoa/window_buttons_proxy.h"
|
|
|
|
|
|
|
|
#include "base/i18n/rtl.h"
|
|
|
|
#include "base/notreached.h"
|
|
|
|
|
|
|
|
@implementation ButtonsAreaHoverView : NSView
|
|
|
|
|
|
|
|
- (id)initWithProxy:(WindowButtonsProxy*)proxy {
|
|
|
|
if ((self = [super init])) {
|
|
|
|
proxy_ = proxy;
|
|
|
|
}
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ignore all mouse events.
|
|
|
|
- (NSView*)hitTest:(NSPoint)aPoint {
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)updateTrackingAreas {
|
|
|
|
[proxy_ updateTrackingAreas];
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation WindowButtonsProxy
|
|
|
|
|
|
|
|
- (id)initWithWindow:(NSWindow*)window {
|
|
|
|
window_ = window;
|
|
|
|
show_on_hover_ = NO;
|
|
|
|
mouse_inside_ = NO;
|
|
|
|
|
|
|
|
// Remember the default margin.
|
|
|
|
margin_ = default_margin_ = [self getCurrentMargin];
|
2022-01-24 22:09:21 +00:00
|
|
|
// Custom height will be used if set larger than default
|
|
|
|
height_ = 0;
|
2021-08-04 00:31:12 +00:00
|
|
|
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)dealloc {
|
|
|
|
if (hover_view_)
|
|
|
|
[hover_view_ removeFromSuperview];
|
|
|
|
[super dealloc];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)setVisible:(BOOL)visible {
|
2021-08-25 23:29:34 +00:00
|
|
|
NSView* titleBarContainer = [self titleBarContainer];
|
|
|
|
if (!titleBarContainer)
|
2021-08-04 00:31:12 +00:00
|
|
|
return;
|
2021-08-25 23:29:34 +00:00
|
|
|
[titleBarContainer setHidden:!visible];
|
2021-08-04 00:31:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isVisible {
|
2021-08-25 23:29:34 +00:00
|
|
|
NSView* titleBarContainer = [self titleBarContainer];
|
|
|
|
if (!titleBarContainer)
|
2021-08-04 00:31:12 +00:00
|
|
|
return YES;
|
2021-08-25 23:29:34 +00:00
|
|
|
return ![titleBarContainer isHidden];
|
2021-08-04 00:31:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)setShowOnHover:(BOOL)yes {
|
2021-08-25 23:29:34 +00:00
|
|
|
NSView* titleBarContainer = [self titleBarContainer];
|
|
|
|
if (!titleBarContainer)
|
2021-08-04 00:31:12 +00:00
|
|
|
return;
|
|
|
|
show_on_hover_ = yes;
|
|
|
|
// Put a transparent view above the window buttons so we can track mouse
|
|
|
|
// events when mouse enter/leave the window buttons.
|
|
|
|
if (show_on_hover_) {
|
2023-07-18 22:26:27 +00:00
|
|
|
hover_view_ = [[ButtonsAreaHoverView alloc] initWithProxy:self];
|
2021-08-04 00:31:12 +00:00
|
|
|
[hover_view_ setFrame:[self getButtonsBounds]];
|
2023-07-18 22:26:27 +00:00
|
|
|
[titleBarContainer addSubview:hover_view_];
|
2021-08-04 00:31:12 +00:00
|
|
|
} else {
|
|
|
|
[hover_view_ removeFromSuperview];
|
2023-07-18 22:26:27 +00:00
|
|
|
hover_view_ = nil;
|
2021-08-04 00:31:12 +00:00
|
|
|
}
|
|
|
|
[self updateButtonsVisibility];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)setMargin:(const absl::optional<gfx::Point>&)margin {
|
|
|
|
if (margin)
|
|
|
|
margin_ = *margin;
|
|
|
|
else
|
|
|
|
margin_ = default_margin_;
|
|
|
|
[self redraw];
|
|
|
|
}
|
|
|
|
|
2022-01-24 22:09:21 +00:00
|
|
|
- (void)setHeight:(const float)height {
|
|
|
|
height_ = height;
|
|
|
|
[self redraw];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)useCustomHeight {
|
|
|
|
NSView* left = [self leftButton];
|
|
|
|
float button_height = NSHeight(left.frame);
|
|
|
|
return height_ > button_height + 2 * default_margin_.y();
|
|
|
|
}
|
|
|
|
|
2021-08-04 00:31:12 +00:00
|
|
|
- (NSRect)getButtonsContainerBounds {
|
|
|
|
return NSInsetRect([self getButtonsBounds], -margin_.x(), -margin_.y());
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)redraw {
|
2021-08-25 23:29:34 +00:00
|
|
|
NSView* titleBarContainer = [self titleBarContainer];
|
|
|
|
if (!titleBarContainer)
|
2021-08-04 00:31:12 +00:00
|
|
|
return;
|
|
|
|
|
2021-08-25 23:29:34 +00:00
|
|
|
NSView* left = [self leftButton];
|
|
|
|
NSView* middle = [self middleButton];
|
|
|
|
NSView* right = [self rightButton];
|
|
|
|
|
|
|
|
float button_width = NSWidth(left.frame);
|
|
|
|
float button_height = NSHeight(left.frame);
|
|
|
|
float padding = NSMinX(middle.frame) - NSMaxX(left.frame);
|
2021-08-04 00:31:12 +00:00
|
|
|
float start;
|
|
|
|
if (base::i18n::IsRTL())
|
|
|
|
start =
|
|
|
|
NSWidth(window_.frame) - 3 * button_width - 2 * padding - margin_.x();
|
|
|
|
else
|
|
|
|
start = margin_.x();
|
|
|
|
|
2021-08-25 23:29:34 +00:00
|
|
|
NSRect cbounds = titleBarContainer.frame;
|
2021-08-04 00:31:12 +00:00
|
|
|
cbounds.size.height = button_height + 2 * margin_.y();
|
2022-01-24 22:09:21 +00:00
|
|
|
// Custom height must be larger than the button height to use
|
|
|
|
if ([self useCustomHeight]) {
|
|
|
|
cbounds.size.height = height_;
|
|
|
|
}
|
2021-08-04 00:31:12 +00:00
|
|
|
cbounds.origin.y = NSHeight(window_.frame) - NSHeight(cbounds);
|
2021-08-25 23:29:34 +00:00
|
|
|
[titleBarContainer setFrame:cbounds];
|
2021-08-04 00:31:12 +00:00
|
|
|
|
2022-01-24 22:09:21 +00:00
|
|
|
[left setFrameOrigin:NSMakePoint(start, [self getCurrentMargin].y())];
|
2021-08-04 00:31:12 +00:00
|
|
|
start += button_width + padding;
|
2022-01-24 22:09:21 +00:00
|
|
|
[middle setFrameOrigin:NSMakePoint(start, [self getCurrentMargin].y())];
|
2021-08-04 00:31:12 +00:00
|
|
|
start += button_width + padding;
|
2022-01-24 22:09:21 +00:00
|
|
|
[right setFrameOrigin:NSMakePoint(start, [self getCurrentMargin].y())];
|
2021-08-04 00:31:12 +00:00
|
|
|
|
|
|
|
if (hover_view_)
|
|
|
|
[hover_view_ setFrame:[self getButtonsBounds]];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)updateTrackingAreas {
|
|
|
|
if (tracking_area_)
|
2023-07-18 22:26:27 +00:00
|
|
|
[hover_view_ removeTrackingArea:tracking_area_];
|
|
|
|
tracking_area_ = [[NSTrackingArea alloc]
|
2021-08-04 00:31:12 +00:00
|
|
|
initWithRect:NSZeroRect
|
|
|
|
options:NSTrackingMouseEnteredAndExited | NSTrackingActiveAlways |
|
|
|
|
NSTrackingInVisibleRect
|
|
|
|
owner:self
|
2023-07-18 22:26:27 +00:00
|
|
|
userInfo:nil];
|
|
|
|
[hover_view_ addTrackingArea:tracking_area_];
|
2021-08-04 00:31:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)mouseEntered:(NSEvent*)event {
|
|
|
|
mouse_inside_ = YES;
|
|
|
|
[self updateButtonsVisibility];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)mouseExited:(NSEvent*)event {
|
|
|
|
mouse_inside_ = NO;
|
|
|
|
[self updateButtonsVisibility];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)updateButtonsVisibility {
|
|
|
|
NSArray* buttons = @[
|
|
|
|
[window_ standardWindowButton:NSWindowCloseButton],
|
|
|
|
[window_ standardWindowButton:NSWindowMiniaturizeButton],
|
|
|
|
[window_ standardWindowButton:NSWindowZoomButton],
|
|
|
|
];
|
|
|
|
// Show buttons when mouse hovers above them.
|
|
|
|
BOOL hidden = show_on_hover_ && !mouse_inside_;
|
|
|
|
// Always show buttons under fullscreen.
|
|
|
|
if ([window_ styleMask] & NSWindowStyleMaskFullScreen)
|
|
|
|
hidden = NO;
|
|
|
|
for (NSView* button in buttons) {
|
|
|
|
[button setHidden:hidden];
|
|
|
|
[button setNeedsDisplay:YES];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the bounds of all 3 buttons.
|
|
|
|
- (NSRect)getButtonsBounds {
|
2021-08-25 23:29:34 +00:00
|
|
|
NSView* left = [self leftButton];
|
|
|
|
NSView* right = [self rightButton];
|
2022-01-24 22:09:21 +00:00
|
|
|
|
2021-08-25 23:29:34 +00:00
|
|
|
return NSMakeRect(NSMinX(left.frame), NSMinY(left.frame),
|
|
|
|
NSMaxX(right.frame) - NSMinX(left.frame),
|
|
|
|
NSHeight(left.frame));
|
2021-08-04 00:31:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Compute margin from position of current buttons.
|
|
|
|
- (gfx::Point)getCurrentMargin {
|
|
|
|
gfx::Point result;
|
2021-08-25 23:29:34 +00:00
|
|
|
NSView* titleBarContainer = [self titleBarContainer];
|
|
|
|
if (!titleBarContainer)
|
2021-08-04 00:31:12 +00:00
|
|
|
return result;
|
|
|
|
|
2021-08-25 23:29:34 +00:00
|
|
|
NSView* left = [self leftButton];
|
|
|
|
NSView* right = [self rightButton];
|
|
|
|
|
2022-01-24 22:09:21 +00:00
|
|
|
if (height_ != 0) {
|
|
|
|
result.set_y((height_ - NSHeight(left.frame)) / 2);
|
|
|
|
|
|
|
|
// Do not center buttons if height and button position specified
|
|
|
|
if (margin_.y() != default_margin_.y())
|
|
|
|
result.set_y(height_ - NSHeight(left.frame) - margin_.y());
|
|
|
|
|
|
|
|
} else {
|
|
|
|
result.set_y((NSHeight(titleBarContainer.frame) - NSHeight(left.frame)) /
|
|
|
|
2);
|
|
|
|
}
|
2021-08-04 00:31:12 +00:00
|
|
|
|
|
|
|
if (base::i18n::IsRTL())
|
2021-08-25 23:29:34 +00:00
|
|
|
result.set_x(NSWidth(window_.frame) - NSMaxX(right.frame));
|
2021-08-04 00:31:12 +00:00
|
|
|
else
|
2021-08-25 23:29:34 +00:00
|
|
|
result.set_x(NSMinX(left.frame));
|
2021-08-04 00:31:12 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-08-25 23:29:34 +00:00
|
|
|
// Receive the titlebar container, which might be nil if the window does not
|
|
|
|
// have the NSWindowStyleMaskTitled style.
|
|
|
|
- (NSView*)titleBarContainer {
|
|
|
|
NSView* left = [self leftButton];
|
|
|
|
if (!left.superview)
|
|
|
|
return nil;
|
|
|
|
return left.superview.superview;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Receive the window buttons, note that the buttons might be removed and
|
|
|
|
// re-added on the fly so we should not cache them.
|
|
|
|
- (NSButton*)leftButton {
|
|
|
|
if (base::i18n::IsRTL())
|
|
|
|
return [window_ standardWindowButton:NSWindowZoomButton];
|
|
|
|
else
|
|
|
|
return [window_ standardWindowButton:NSWindowCloseButton];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSButton*)middleButton {
|
|
|
|
return [window_ standardWindowButton:NSWindowMiniaturizeButton];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSButton*)rightButton {
|
|
|
|
if (base::i18n::IsRTL())
|
|
|
|
return [window_ standardWindowButton:NSWindowCloseButton];
|
|
|
|
else
|
|
|
|
return [window_ standardWindowButton:NSWindowZoomButton];
|
|
|
|
}
|
|
|
|
|
2021-08-04 00:31:12 +00:00
|
|
|
@end
|