diff --git a/src/demo-app/button/button-demo.html b/src/demo-app/button/button-demo.html index 2e260122533f..fcc074bfa143 100644 --- a/src/demo-app/button/button-demo.html +++ b/src/demo-app/button/button-demo.html @@ -1,7 +1,7 @@

Buttons

- + diff --git a/src/demo-app/toolbar/toolbar-demo.html b/src/demo-app/toolbar/toolbar-demo.html index fa7f900c945d..3e2ec07f5cd8 100644 --- a/src/demo-app/toolbar/toolbar-demo.html +++ b/src/demo-app/toolbar/toolbar-demo.html @@ -1,40 +1,61 @@
-

- menu - Default Toolbar + + Default Toolbar - code + + + +

- menu - Primary Toolbar + + Primary Toolbar - code + + +

- menu - Accent Toolbar + + Accent Toolbar - code + + + +

- + First Row Second Row diff --git a/src/demo-app/toolbar/toolbar-demo.scss b/src/demo-app/toolbar/toolbar-demo.scss index a433f45b5cb9..fa679ae89f1f 100644 --- a/src/demo-app/toolbar/toolbar-demo.scss +++ b/src/demo-app/toolbar/toolbar-demo.scss @@ -9,4 +9,7 @@ flex: 1 1 auto; } + button { + margin: 0 4px; + } } diff --git a/src/lib/button/_button-theme.scss b/src/lib/button/_button-theme.scss index c3d961da123a..250249b5e873 100644 --- a/src/lib/button/_button-theme.scss +++ b/src/lib/button/_button-theme.scss @@ -2,7 +2,7 @@ @import '../core/typography/typography-utils'; // Applies a focus style to an mat-button element for each of the supported palettes. -@mixin _mat-button-focus-color($theme) { +@mixin _mat-button-focus-overlay-color($theme) { $primary: map-get($theme, primary); $accent: map-get($theme, accent); $warn: map-get($theme, warn); @@ -24,7 +24,7 @@ } } -@mixin _mat-button-ripple-color($theme, $hue, $opacity: 0.2) { +@mixin _mat-button-ripple-color($theme, $hue, $opacity: 0.1) { $primary: map-get($theme, primary); $accent: map-get($theme, accent); $warn: map-get($theme, warn); @@ -43,7 +43,7 @@ } // Applies a property to an mat-button element for each of the supported palettes. -@mixin _mat-button-theme-color($theme, $property, $color: 'default') { +@mixin _mat-button-theme-property($theme, $property, $hue) { $primary: map-get($theme, primary); $accent: map-get($theme, accent); $warn: map-get($theme, warn); @@ -51,13 +51,13 @@ $foreground: map-get($theme, foreground); &.mat-primary { - #{$property}: mat-color($primary, $color); + #{$property}: mat-color($primary, $hue); } &.mat-accent { - #{$property}: mat-color($accent, $color); + #{$property}: mat-color($accent, $hue); } &.mat-warn { - #{$property}: mat-color($warn, $color); + #{$property}: mat-color($warn, $hue); } &.mat-primary, &.mat-accent, &.mat-warn, &[disabled] { @@ -76,51 +76,37 @@ $foreground: map-get($theme, foreground); .mat-button, .mat-icon-button, .mat-stroked-button { - background: transparent; - - @include _mat-button-focus-color($theme); - @include _mat-button-theme-color($theme, 'color'); - } - - .mat-raised-button, .mat-fab, .mat-mini-fab { - // Default properties when not using any [color] value. color: mat-color($foreground, text); - background-color: mat-color($background, raised-button); - - @include _mat-button-theme-color($theme, 'color', default-contrast); - @include _mat-button-theme-color($theme, 'background-color'); + background: transparent; - // Add ripple effect with contrast color to buttons that don't have a focus overlay. - @include _mat-button-ripple-color($theme, default-contrast); - } + @include _mat-button-theme-property($theme, 'color', default); + @include _mat-button-focus-overlay-color($theme); - // Add ripple effect with default color to flat buttons, which also have a focus overlay. - .mat-button { - @include _mat-button-ripple-color($theme, default, 0.1); + // Setup the ripple color to be based on the color palette. The opacity can be a bit weaker + // than for icon-buttons, because normal and stroked buttons have a focus overlay. + @include _mat-button-ripple-color($theme, default); } - .mat-flat-button { - // Default properties when not using any [color] value. + .mat-flat-button, .mat-raised-button, .mat-fab, .mat-mini-fab { + // Default font and background color when not using any color palette. color: mat-color($foreground, text); - background-color: mat-color($background, raised-button); - @include _mat-button-theme-color($theme, 'color', default-contrast); - @include _mat-button-theme-color($theme, 'background-color'); - // Add ripple effect with contrast color to buttons that don't have a focus overlay. + @include _mat-button-theme-property($theme, 'color', default-contrast); + @include _mat-button-theme-property($theme, 'background-color', default); @include _mat-button-ripple-color($theme, default-contrast); } - // Add ripple effect with default color to the icon button. Ripple color needs to be stronger - // since the icon button doesn't have a focus overlay. + // Since icon buttons don't have a focus overlay, the ripple opacity should be the higher + // than the default value. .mat-icon-button { - @include _mat-button-ripple-color($theme, default); + @include _mat-button-ripple-color($theme, default, 0.2); } } @mixin mat-button-typography($config) { - .mat-button, .mat-raised-button, .mat-icon-button, .mat-stroked-button, .mat-flat-button, - .mat-fab, .mat-mini-fab { + .mat-button, .mat-raised-button, .mat-icon-button, .mat-stroked-button, + .mat-flat-button, .mat-fab, .mat-mini-fab { font: { family: mat-font-family($config, button); size: mat-font-size($config, button); diff --git a/src/lib/button/button.scss b/src/lib/button/button.scss index 0f39d8791bab..6f0bd6c9babe 100644 --- a/src/lib/button/button.scss +++ b/src/lib/button/button.scss @@ -67,14 +67,6 @@ } } -// The text and icon should be vertical aligned inside a button -.mat-button, .mat-raised-button, .mat-icon-button, .mat-fab, .mat-mini-fab { - color: currentColor; - .mat-button-wrapper > * { - vertical-align: middle; - } -} - // The ripple container should match the bounds of the entire button. .mat-button-ripple, .mat-button-focus-overlay { @include mat-fill; @@ -111,6 +103,14 @@ z-index: 1; } +// Elements inside of all type of buttons should be vertical aligned in the middle. +.mat-button, .mat-flat-button, .mat-stroked-button, .mat-raised-button, .mat-icon-button, +.mat-fab, .mat-mini-fab { + .mat-button-wrapper > * { + vertical-align: middle; + } +} + // Add an outline to make it more visible in high contrast mode. @include cdk-high-contrast { .mat-button, .mat-raised-button, .mat-icon-button, .mat-fab, .mat-mini-fab { diff --git a/src/lib/button/button.spec.ts b/src/lib/button/button.spec.ts index 846f89563dbd..c48a3c0a79d0 100644 --- a/src/lib/button/button.spec.ts +++ b/src/lib/button/button.spec.ts @@ -2,7 +2,7 @@ import {async, ComponentFixture, TestBed} from '@angular/core/testing'; import {Component, DebugElement} from '@angular/core'; import {By} from '@angular/platform-browser'; import {MatButtonModule, MatButton} from './index'; -import {MatRipple} from '@angular/material/core'; +import {MatRipple, ThemePalette} from '@angular/material/core'; describe('MatButton', () => { @@ -41,6 +41,30 @@ describe('MatButton', () => { expect(aDebugElement.nativeElement.classList).not.toContain('mat-accent'); }); + it('should mark buttons without a background color and theme as plain buttons', () => { + const fixture = TestBed.createComponent(TestApp); + const buttonDebugEl = fixture.debugElement.query(By.css('button')); + const anchorDebugEl = fixture.debugElement.query(By.css('a')); + const fabDebugEl = fixture.debugElement.query(By.css('[mat-fab]')); + + fixture.detectChanges(); + + // Buttons that have no background color and theme palette are considered as plain buttons. + expect(buttonDebugEl.nativeElement.classList).toContain('mat-plain-button'); + expect(anchorDebugEl.nativeElement.classList).toContain('mat-plain-button'); + expect(fabDebugEl.nativeElement.classList).not.toContain('mat-plain-button'); + + fixture.componentInstance.buttonColor = 'primary'; + fixture.detectChanges(); + + // Buttons that have no background color, but use an explicit theme palette, are not + // considered as plain buttons. + expect(buttonDebugEl.nativeElement.classList).not.toContain('mat-plain-button'); + expect(anchorDebugEl.nativeElement.classList).not.toContain('mat-plain-button'); + expect(fabDebugEl.nativeElement.classList).not.toContain('mat-plain-button'); + }); + + it('should expose the ripple instance', () => { const fixture = TestBed.createComponent(TestApp); const button = fixture.debugElement.query(By.css('button')).componentInstance as MatButton; @@ -259,6 +283,7 @@ class TestApp { clickCount: number = 0; isDisabled: boolean = false; rippleDisabled: boolean = false; + buttonColor: ThemePalette; increment() { this.clickCount++; diff --git a/src/lib/button/button.ts b/src/lib/button/button.ts index feb8840dbba0..21195e969283 100644 --- a/src/lib/button/button.ts +++ b/src/lib/button/button.ts @@ -65,6 +65,7 @@ export const _MatButtonMixinBase = mixinColor(mixinDisabled(mixinDisableRipple(M exportAs: 'matButton', host: { '[disabled]': 'disabled || null', + '[class.mat-button-plain]': '_isPlainButton()', }, templateUrl: 'button.html', styleUrls: ['button.css'], @@ -76,12 +77,18 @@ export const _MatButtonMixinBase = mixinColor(mixinDisabled(mixinDisableRipple(M export class MatButton extends _MatButtonMixinBase implements OnDestroy, CanDisable, CanColor, CanDisableRipple { + /** Whether the button is a normal button. */ + _isNormalButton: boolean = this._hasHostAttributes('mat-button'); + /** Whether the button is round. */ _isRoundButton: boolean = this._hasHostAttributes('mat-fab', 'mat-mini-fab'); /** Whether the button is icon button. */ _isIconButton: boolean = this._hasHostAttributes('mat-icon-button'); + /** Whether the button is a stroked button. */ + _isStrokedButton: boolean = this._hasHostAttributes('mat-stroked-button'); + /** Reference to the MatRipple instance of the button. */ @ViewChild(MatRipple) ripple: MatRipple; @@ -124,15 +131,16 @@ export class MatButton extends _MatButtonMixinBase return this.disableRipple || this.disabled; } + /** + * Whether the button is a plain button. Plain buttons are buttons without a background + * color and theme color set. + */ + _isPlainButton() { + return !this.color && (this._isIconButton || this._isNormalButton || this._isStrokedButton); + } + /** Gets whether the button has one of the given attributes. */ _hasHostAttributes(...attributes: string[]) { - // If not on the browser, say that there are none of the attributes present. - // Since these only affect how the ripple displays (and ripples only happen on the client), - // detecting these attributes isn't necessary when not on the browser. - if (!this._platform.isBrowser) { - return false; - } - return attributes.some(attribute => this._getHostElement().hasAttribute(attribute)); } } @@ -149,6 +157,7 @@ export class MatButton extends _MatButtonMixinBase '[attr.tabindex]': 'disabled ? -1 : 0', '[attr.disabled]': 'disabled || null', '[attr.aria-disabled]': 'disabled.toString()', + '[class.mat-button-plain]': '_isPlainButton()', '(click)': '_haltDisabledEvents($event)', }, inputs: ['disabled', 'disableRipple', 'color'], @@ -159,10 +168,8 @@ export class MatButton extends _MatButtonMixinBase changeDetection: ChangeDetectionStrategy.OnPush, }) export class MatAnchor extends MatButton { - constructor( - platform: Platform, - focusMonitor: FocusMonitor, - elementRef: ElementRef) { + + constructor(platform: Platform, focusMonitor: FocusMonitor, elementRef: ElementRef) { super(elementRef, platform, focusMonitor); } diff --git a/src/lib/toolbar/toolbar.scss b/src/lib/toolbar/toolbar.scss index e6b4cb21650b..398d2df20aca 100644 --- a/src/lib/toolbar/toolbar.scss +++ b/src/lib/toolbar/toolbar.scss @@ -30,6 +30,12 @@ $mat-toolbar-row-padding: 16px !default; // Per Material specs a toolbar cannot have multiple lines inside of a single row. // Disable text wrapping inside of the toolbar. Developers are still able to overwrite it. white-space: nowrap; + + // Plain buttons (buttons without a background color and color palette) should inherit the color + // from the toolbar row, because otherwise the text will be unreadable on themed toolbars. + .mat-button-plain { + color: inherit; + } } .mat-toolbar-multiple-rows {