Skip to content

Commit 5c5903e

Browse files
authoredFeb 15, 2021
feat: add button to copy link to example (#922)
Adds the ability to copy an anchor link to a specific example. Has a bit of extra logic to ensure that the browser actually scrolls the element into view. Fixes #21884.
1 parent 4e67c59 commit 5c5903e

File tree

2 files changed

+40
-7
lines changed

2 files changed

+40
-7
lines changed
 

‎material.angular.io/src/app/shared/example-viewer/example-viewer.html

+14-5
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,16 @@
22
<div class="docs-example-viewer-title" *ngIf="view !== 'snippet'">
33
<div class="docs-example-viewer-title-spacer">{{exampleData?.title}}</div>
44

5-
<button mat-icon-button type="button" (click)="toggleCompactView()" [matTooltip]="'View snippet only'"
5+
<button
6+
mat-icon-button
7+
type="button"
8+
[attr.aria-label]="'Copy link to ' + exampleData?.title + ' example to the clipboard'"
9+
matTooltip="Copy link to example"
10+
(click)="_copyLink()">
11+
<mat-icon>link</mat-icon>
12+
</button>
13+
14+
<button mat-icon-button type="button" (click)="toggleCompactView()" matTooltip="View snippet only"
615
aria-label="View less" *ngIf="showCompactToggle">
716
<mat-icon>
817
<svg xmlns="http://www.w3.org/2000/svg" width="100%" height="100%" focusable="false">
@@ -13,7 +22,7 @@
1322
</button>
1423

1524
<button mat-icon-button type="button" (click)="toggleSourceView()"
16-
[matTooltip]="view==='demo' ? 'View code' : 'Hide code'" aria-label="View source">
25+
[matTooltip]="view === 'demo' ? 'View code' : 'Hide code'" aria-label="View source">
1726
<mat-icon>code</mat-icon>
1827
</button>
1928

@@ -25,7 +34,7 @@
2534
<mat-tab *ngFor="let tabName of _getExampleTabNames()" [label]="tabName">
2635
<div class="button-bar">
2736
<button mat-icon-button type="button" (click)="copySource(snippet.toArray()[selectedTab].viewer.textContent)"
28-
class="docs-example-source-copy docs-example-button" [matTooltip]="'Copy example source'"
37+
class="docs-example-source-copy docs-example-button" matTooltip="Copy example source"
2938
title="Copy example source" aria-label="Copy example source to clipboard">
3039
<mat-icon>content_copy</mat-icon>
3140
</button>
@@ -38,12 +47,12 @@
3847
<div class="docs-example-viewer-source-compact" *ngIf="view === 'snippet'">
3948
<div class="button-bar">
4049
<button mat-icon-button type="button" (click)="copySource(snippet.first.viewer.textContent)"
41-
class="docs-example-source-copy docs-example-button" [matTooltip]="'Copy snippet'"
50+
class="docs-example-source-copy docs-example-button" matTooltip="Copy snippet"
4251
title="Copy example source" aria-label="Copy example source to clipboard">
4352
<mat-icon>content_copy</mat-icon>
4453
</button>
4554
<button mat-icon-button type="button" (click)="toggleCompactView()"
46-
class="docs-example-compact-toggle docs-example-button" [matTooltip]="'View full example'"
55+
class="docs-example-compact-toggle docs-example-button" matTooltip="View full example"
4756
aria-label="View less">
4857
<mat-icon>
4958
<svg xmlns="http://www.w3.org/2000/svg" width="100%" height="100%" focusable="false">

‎material.angular.io/src/app/shared/example-viewer/example-viewer.ts

+26-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,11 @@
11
import {
22
Component,
3+
ElementRef,
4+
HostBinding,
35
Input,
4-
NgModuleFactory, OnInit, QueryList,
6+
NgModuleFactory,
7+
OnInit,
8+
QueryList,
59
Type,
610
ViewChildren,
711
ɵNgModuleFactory
@@ -53,6 +57,7 @@ export class ExampleViewer implements OnInit {
5357
@Input() showCompactToggle = false;
5458

5559
/** String key of the currently displayed example. */
60+
@HostBinding('attr.id')
5661
@Input()
5762
get example() { return this._example; }
5863
set example(exampleName: string) {
@@ -75,7 +80,8 @@ export class ExampleViewer implements OnInit {
7580

7681
constructor(
7782
private readonly snackbar: MatSnackBar,
78-
private readonly clipboard: Clipboard) {}
83+
private readonly clipboard: Clipboard,
84+
private readonly elementRef: ElementRef<HTMLElement>) {}
7985

8086
ngOnInit() {
8187
if (this.file) {
@@ -152,6 +158,17 @@ export class ExampleViewer implements OnInit {
152158
});
153159
}
154160

161+
_copyLink() {
162+
// Reconstruct the URL using `origin + pathname` so we drop any pre-existing hash.
163+
const fullUrl = location.origin + location.pathname + '#' + this._example;
164+
165+
if (this.clipboard.copy(fullUrl)) {
166+
this.snackbar.open('Link copied', '', {duration: 2500});
167+
} else {
168+
this.snackbar.open('Link copy failed. Please try again!', '', {duration: 2500});
169+
}
170+
}
171+
155172
/** Loads the component and module factory for the currently selected example. */
156173
private async _loadExampleComponent() {
157174
const {componentName, module} = EXAMPLE_COMPONENTS[this._example];
@@ -170,6 +187,13 @@ export class ExampleViewer implements OnInit {
170187
// class symbol to Ivy's module factory constructor. There is no equivalent for View Engine,
171188
// where factories are stored in separate files. Hence the API is currently Ivy-only.
172189
this._exampleModuleFactory = new ɵNgModuleFactory(moduleExports[module.name]);
190+
191+
// Since the data is loaded asynchronously, we can't count on the native behavior
192+
// that scrolls the element into view automatically. We do it ourselves while giving
193+
// the page some time to render.
194+
if (typeof location !== 'undefined' && location.hash.slice(1) === this._example) {
195+
setTimeout(() => this.elementRef.nativeElement.scrollIntoView(), 300);
196+
}
173197
}
174198

175199
private _generateExampleTabs() {

0 commit comments

Comments
 (0)
Please sign in to comment.