Skip to content

Commit 9dbfef8

Browse files
committed
flashmob
1 parent 72cd8f6 commit 9dbfef8

15 files changed

+259
-16
lines changed

package-lock.json

+5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@
2121
"@angular/platform-browser": "~8.2.9",
2222
"@angular/platform-browser-dynamic": "~8.2.9",
2323
"@angular/router": "~8.2.9",
24+
"@types/webaudioapi": "0.0.27",
2425
"d3": "^5.12.0",
2526
"hammerjs": "^2.0.8",
2627
"rxjs": "~6.4.0",

src/app/app-routing.module.ts

+2
Original file line numberDiff line numberDiff line change
@@ -2,11 +2,13 @@ import { NgModule } from '@angular/core';
22
import { Routes, RouterModule } from '@angular/router';
33
import { OrderStatusComponent } from './order-status/order-status.component';
44
import { OrderDeliveryComponent } from './order-delivery/order-delivery.component';
5+
import { FlashMobComponent } from './flash-mob/flash-mob.component';
56

67

78
const routes: Routes = [
89
{ path: 'status', component: OrderStatusComponent },
910
{ path: 'delivery', component: OrderDeliveryComponent },
11+
{ path: 'flashmob', component: FlashMobComponent },
1012
{ path: '', component: OrderStatusComponent }
1113
];
1214

src/app/app.component.html

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,8 @@
1-
<div class="toolbar" role="banner">
1+
<div *ngIf="!chartControlsService.fullScreen" class="toolbar" role="banner">
22
<span>Papa Jon's Pizza Pies</span>
33
</div>
4-
<img class="watermark" src="assets/pizza.png"/>
5-
<button mat-icon-button id="mike" color="accent" (click)="toggleSpeechRecognition()">
4+
<img *ngIf="!chartControlsService.fullScreen" class="watermark" src="assets/pizza.png"/>
5+
<button *ngIf="!chartControlsService.fullScreen" mat-icon-button id="mike" color="accent" (click)="toggleSpeechRecognition()">
66
<mat-icon>mic</mat-icon>
77
</button>
88
<router-outlet></router-outlet>

src/app/app.component.ts

+9-4
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import { ChartControlsService } from './chart-controls.service';
55
//https://stackoverflow.com/questions/38087013/angular2-web-speech-api-voice-recognition
66
export interface IWindow extends Window {
77
webkitSpeechRecognition: any;
8+
webkitAudioContext: any;
89
}
910

1011
@Component({
@@ -15,6 +16,7 @@ export interface IWindow extends Window {
1516
export class AppComponent implements OnInit {
1617
title = 'Using d3 within Angular 8';
1718
speechRecogitionState = false;
19+
flashMob = false;
1820

1921
constructor(
2022
private router: Router,
@@ -26,7 +28,7 @@ export class AppComponent implements OnInit {
2628

2729
toggleSpeechRecognition() {
2830
if ('webkitSpeechRecognition' in window) {
29-
const { webkitSpeechRecognition }: IWindow = <IWindow>window;
31+
const { webkitSpeechRecognition }: IWindow = <IWindow>(window as unknown);
3032
const recognition = new webkitSpeechRecognition();
3133

3234
recognition.continuous = true;
@@ -45,9 +47,7 @@ export class AppComponent implements OnInit {
4547
const statusMatcher = new RegExp('.*status.*',"i");
4648
const hideMatcher = new RegExp('.*hide.*data.*',"i");
4749
const showMatcher = new RegExp('.*show.*data.*',"i");
48-
console.log(command);
49-
console.log(deliveryMatcher.test(command.trim()));
50-
console.log(statusMatcher.test(command.trim()));
50+
const flashMatcher = new RegExp('.*flash.*',"i");
5151
if (deliveryMatcher.test(command.trim())) {
5252
this.navigate('/delivery');
5353
return;
@@ -64,6 +64,11 @@ export class AppComponent implements OnInit {
6464
this.chartControlsService.showData = false;
6565
return;
6666
}
67+
if (flashMatcher.test(command.trim())) {
68+
this.flashMob = true;
69+
this.navigate('/flashmob');
70+
return;
71+
}
6772
}
6873

6974
navigate(path) {

src/app/app.module.ts

+3-1
Original file line numberDiff line numberDiff line change
@@ -15,14 +15,16 @@ import { OrderStatusComponent } from './order-status/order-status.component';
1515
import { DonutChartComponent } from './donut-chart/donut-chart.component';
1616
import { OrderDeliveryComponent } from './order-delivery/order-delivery.component';
1717
import { AreaChartComponent } from './area-chart/area-chart.component';
18+
import { FlashMobComponent } from './flash-mob/flash-mob.component';
1819

1920
@NgModule({
2021
declarations: [
2122
AppComponent,
2223
OrderStatusComponent,
2324
DonutChartComponent,
2425
OrderDeliveryComponent,
25-
AreaChartComponent
26+
AreaChartComponent,
27+
FlashMobComponent
2628
],
2729
imports: [
2830
BrowserModule,

src/app/area-chart/area-chart.component.ts

+7-5
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,9 @@ export class AreaDatum {
1414
styleUrls: ['./area-chart.component.scss']
1515
})
1616
export class AreaChartComponent implements OnInit, OnChanges {
17-
17+
@Input() transitionTime = 1000;
18+
@Input() xmax = 45;
19+
@Input() ymax = 200;
1820
@Input() data: number[];
1921
hostElement; // Native element hosting the SVG container
2022
svg; // Top level SVG element
@@ -64,7 +66,7 @@ export class AreaChartComponent implements OnInit, OnChanges {
6466

6567
this.histogram = d3.histogram()
6668
.value((datum) => datum)
67-
.domain([0, 45])
69+
.domain([0, this.xmax])
6870
.thresholds(this.x.ticks(60));
6971

7072
// data has to be processed after area and histogram functions are defined
@@ -103,7 +105,7 @@ export class AreaChartComponent implements OnInit, OnChanges {
103105

104106
private createXAxis() {
105107
this.x = d3.scaleLinear()
106-
.domain([0, 45])
108+
.domain([0, this.xmax])
107109
.range([30, 170]);
108110
this.g.append('g')
109111
.attr('transform', 'translate(0,90)')
@@ -126,7 +128,7 @@ export class AreaChartComponent implements OnInit, OnChanges {
126128

127129
private createYAxis() {
128130
this.y = d3.scaleLinear()
129-
.domain([0, 200])
131+
.domain([0, this.ymax])
130132
.range([90, 10]);
131133
this.g.append('g')
132134
.attr('transform', 'translate(30,0)')
@@ -173,7 +175,7 @@ export class AreaChartComponent implements OnInit, OnChanges {
173175
private updateAreaCharts() {
174176
this.paths.forEach((path, index) => {
175177
path.datum(this.bins[index])
176-
.transition().duration(1000)
178+
.transition().duration(this.transitionTime)
177179
.attr('d', d3.area()
178180
.x((datum: any) => this.x(d3.mean([datum.x1, datum.x2])))
179181
.y0(this.y(0))

src/app/chart-controls.service.ts

+1
Original file line numberDiff line numberDiff line change
@@ -6,5 +6,6 @@ import { Injectable } from '@angular/core';
66
export class ChartControlsService {
77

88
showData = false;
9+
fullScreen = false;
910
constructor() { }
1011
}
+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
<div class="container">
2+
<app-area-chart #areaChart [data]="chartData" [transitionTime]="transitionTime"
3+
xmax="50"
4+
ymax="250"
5+
></app-area-chart>
6+
</div>
7+
<audio id="audio" #audio controls crossorigin="anonymous">
8+
<!-- <source src="https://d3ng.s3.us-east-2.amazonaws.com/xlf.m4a" type="audio/mpeg"> -->
9+
<source src="https://d3ng.s3.us-east-2.amazonaws.com/dd.mp3" type="audio/mpeg">
10+
<!-- <source src="assets/xlf.m4a" type="audio/mpeg"> -->
11+
Your browser does not support the audio element.
12+
</audio>
+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
:host {
2+
display: flex;
3+
justify-content: space-evenly;
4+
align-items: center;
5+
}
6+
7+
.container {
8+
display: flex;
9+
// background-color: #EAFAF1;
10+
justify-content: center;
11+
align-items: center;
12+
height: 100vh;
13+
width: 100vw;
14+
app-area-chart {
15+
display: flex;
16+
justify-content: center;
17+
align-items: center;
18+
height: 100%;
19+
width: 100%;
20+
}
21+
}
22+
#audio {
23+
position: absolute;
24+
top: 1rem;
25+
right: 1rem;
26+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
import { async, ComponentFixture, TestBed } from '@angular/core/testing';
2+
3+
import { FlashMobComponent } from './flash-mob.component';
4+
5+
describe('FlashMobComponent', () => {
6+
let component: FlashMobComponent;
7+
let fixture: ComponentFixture<FlashMobComponent>;
8+
9+
beforeEach(async(() => {
10+
TestBed.configureTestingModule({
11+
declarations: [ FlashMobComponent ]
12+
})
13+
.compileComponents();
14+
}));
15+
16+
beforeEach(() => {
17+
fixture = TestBed.createComponent(FlashMobComponent);
18+
component = fixture.componentInstance;
19+
fixture.detectChanges();
20+
});
21+
22+
it('should create', () => {
23+
expect(component).toBeTruthy();
24+
});
25+
});
+157
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,157 @@
1+
import { Component, OnInit, OnDestroy, AfterContentInit, ElementRef, isDevMode } from '@angular/core';
2+
import { Router } from '@angular/router';
3+
import { ViewChild } from '@angular/core';
4+
import * as d3 from 'd3';
5+
6+
import { AreaChartComponent } from './../area-chart/area-chart.component';
7+
import { ChartControlsService } from '../chart-controls.service';
8+
import { DeliveryMetric } from '../order-delivery/order-delivery.component';
9+
import { IWindow } from '../app.component';
10+
11+
@Component({
12+
selector: 'app-flash-mob',
13+
templateUrl: './flash-mob.component.html',
14+
styleUrls: ['./flash-mob.component.scss']
15+
})
16+
export class FlashMobComponent implements OnInit, OnDestroy, AfterContentInit {
17+
@ViewChild('areaChart', { static: true }) chart: AreaChartComponent;
18+
@ViewChild('audio', { static: true }) audio: ElementRef;
19+
20+
chartData = [];
21+
22+
transitionTime = 200;
23+
refreshInterval;
24+
25+
audioContext;
26+
analyzer;
27+
dataArray;
28+
29+
30+
31+
constructor(private chartControlsService: ChartControlsService) {
32+
this.chartControlsService.fullScreen = true;
33+
}
34+
35+
ngOnInit() {
36+
37+
// if(isDevMode()) {
38+
// this.audio.nativeElement.source = 'assets/xlf.m4a';
39+
// }
40+
41+
}
42+
43+
initialize() {
44+
45+
if ('AudioContext' in window) {
46+
const audioContext = new AudioContext();
47+
const audioSrc = audioContext.createMediaElementSource(this.audio.nativeElement);
48+
this.analyzer = audioContext.createAnalyser();
49+
50+
51+
const bufferLength = this.analyzer.frequencyBinCount;
52+
console.log('bufferlength', bufferLength);
53+
this.dataArray = new Uint8Array(bufferLength);
54+
this.analyzer.fftSize = 2048;
55+
audioSrc.connect(this.analyzer);
56+
audioSrc.connect(audioContext.destination);
57+
this.analyzer.getByteFrequencyData(this.dataArray);
58+
59+
}
60+
61+
62+
if (this.refreshInterval) {
63+
clearInterval(this.refreshInterval);
64+
}
65+
this.generateData();
66+
this.chart.data = [...this.chartData];
67+
this.audio.nativeElement.play();
68+
this.refreshInterval = setInterval(() => {
69+
if(document.hasFocus()) {
70+
this.generateData();
71+
this.chart.data = [...this.chartData];
72+
}
73+
}, this.transitionTime);
74+
75+
}
76+
77+
ngOnDestroy() {
78+
if (this.refreshInterval) {
79+
clearInterval(this.refreshInterval);
80+
}
81+
}
82+
83+
ngAfterContentInit() {
84+
this.initialize();
85+
}
86+
87+
generateData() {
88+
this.chartData = [];
89+
let maxDecibelRatio = 1;
90+
let mf = 1.0;
91+
if (this.analyzer) {
92+
93+
this.analyzer.getByteFrequencyData(this.dataArray);
94+
// maxDecibelRatio = this.analyzer.maxDecibels / 100 ;
95+
// console.log(maxDecibelRatio);
96+
console.log('data array mean', 1.1 * d3.mean(this.dataArray));
97+
mf = 200.0 / d3.mean(this.dataArray) ;
98+
console.log(mf);
99+
100+
//this.chartData.push(this.dataArray);
101+
}
102+
const meanPrepTime = randomInt(10, 11) + 0.0 ;
103+
const meanWaitTime = randomInt(8, 9) + 0.0;
104+
const meanTransitTime = randomInt(9, 10) + 0.0;
105+
106+
const meanTotalTime = meanPrepTime + meanWaitTime + meanTransitTime;
107+
108+
// const sigmaPrepTime = mf * randomInt(1, 1) ;
109+
const sigmaPrepTime = mf * randomInt(1, 2) ;
110+
console.log('sprep', mf);
111+
const sigmaWaitTime = randomInt(2, 3);
112+
const sigmaTransitTime = randomInt(1, 2);
113+
114+
const sigmaTotalTime = Math.floor(
115+
Math.sqrt(Math.pow(sigmaPrepTime, 2) +
116+
Math.pow(sigmaWaitTime, 2) +
117+
Math.pow(sigmaTransitTime, 2))
118+
);
119+
120+
121+
let prandomizer = d3.randomNormal(8, sigmaPrepTime);
122+
let wrandomizer = d3.randomNormal(16, sigmaPrepTime);
123+
let trandomizer = d3.randomNormal(24, sigmaPrepTime);
124+
let qrandomizer = d3.randomNormal(32, sigmaPrepTime);
125+
let srandomizer = d3.randomNormal(40, sigmaPrepTime);
126+
127+
const ptimes = [];
128+
const wtimes = [];
129+
const ttimes = [];
130+
const qtimes = [];
131+
const stimes = [];
132+
for (let i = 0; i < 1000; i++) {
133+
const p = Math.floor(prandomizer());
134+
const w = Math.floor(wrandomizer());
135+
const t = Math.floor(trandomizer());
136+
const q = Math.floor(qrandomizer());
137+
const s = srandomizer();
138+
ptimes.push(p);
139+
wtimes.push(w);
140+
ttimes.push(t);
141+
qtimes.push(q);
142+
stimes.push(s);
143+
}
144+
this.chartData.push(ptimes);
145+
this.chartData.push(wtimes);
146+
this.chartData.push(ttimes);
147+
this.chartData.push(qtimes);
148+
this.chartData.push(stimes);
149+
// this.chartData.push(totaltimes);
150+
}
151+
152+
}
153+
154+
export function randomInt(min, max) {
155+
return Math.floor(Math.random() * (max - min + 1)) + min;
156+
}
157+

0 commit comments

Comments
 (0)