Weather is done
This commit is contained in:
parent
a158b4cb15
commit
83a61d50c7
33
.circleci/config.yml
Normal file
33
.circleci/config.yml
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
version: 2
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
docker:
|
||||||
|
- image: circleci/node:10.4-browsers
|
||||||
|
|
||||||
|
working_directory: ~/repo
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
|
||||||
|
- restore_cache:
|
||||||
|
keys:
|
||||||
|
- v1-dependencies-{{ checksum "package.json" }}
|
||||||
|
- v1-dependencies-
|
||||||
|
|
||||||
|
- run:
|
||||||
|
name: Install Dependancies
|
||||||
|
command: yarn
|
||||||
|
|
||||||
|
- save_cache:
|
||||||
|
key: v1-dependencies-{{ checksum "package.json" }}
|
||||||
|
paths:
|
||||||
|
- node_modules
|
||||||
|
- functions/node_modules
|
||||||
|
|
||||||
|
- run:
|
||||||
|
name: Build
|
||||||
|
command: yarn build
|
||||||
|
|
||||||
|
- run:
|
||||||
|
name: Deploy
|
||||||
|
command: yarn deploy
|
@ -4,7 +4,7 @@ root = true
|
|||||||
[*]
|
[*]
|
||||||
charset = utf-8
|
charset = utf-8
|
||||||
indent_style = space
|
indent_style = space
|
||||||
indent_size = 2
|
indent_size = 4
|
||||||
insert_final_newline = true
|
insert_final_newline = true
|
||||||
trim_trailing_whitespace = true
|
trim_trailing_whitespace = true
|
||||||
|
|
||||||
|
5
.firebaserc
Normal file
5
.firebaserc
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
{
|
||||||
|
"projects": {
|
||||||
|
"default": "homefront-2ccb4"
|
||||||
|
}
|
||||||
|
}
|
@ -23,7 +23,6 @@
|
|||||||
"polyfills": "src/polyfills.ts",
|
"polyfills": "src/polyfills.ts",
|
||||||
"tsConfig": "src/tsconfig.app.json",
|
"tsConfig": "src/tsconfig.app.json",
|
||||||
"assets": [
|
"assets": [
|
||||||
"src/favicon.ico",
|
|
||||||
"src/assets"
|
"src/assets"
|
||||||
],
|
],
|
||||||
"styles": [
|
"styles": [
|
||||||
@ -87,7 +86,6 @@
|
|||||||
],
|
],
|
||||||
"scripts": [],
|
"scripts": [],
|
||||||
"assets": [
|
"assets": [
|
||||||
"src/favicon.ico",
|
|
||||||
"src/assets"
|
"src/assets"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
16
firebase.json
Normal file
16
firebase.json
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
{
|
||||||
|
"hosting": {
|
||||||
|
"public": "dist/HomeFront",
|
||||||
|
"ignore": [
|
||||||
|
"firebase.json",
|
||||||
|
"**/.*",
|
||||||
|
"**/node_modules/**"
|
||||||
|
],
|
||||||
|
"rewrites": [
|
||||||
|
{
|
||||||
|
"source": "**",
|
||||||
|
"destination": "/index.html"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
14
package.json
14
package.json
@ -1,26 +1,31 @@
|
|||||||
{
|
{
|
||||||
"name": "home-front",
|
"name": "home-front",
|
||||||
"version": "0.0.0",
|
"version": "0.2.0",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"ng": "ng",
|
"ng": "ng",
|
||||||
"start": "ng serve",
|
"start": "ng serve",
|
||||||
"build": "ng build",
|
"build": "ng build --configuration=production",
|
||||||
"test": "ng test",
|
"test": "ng test",
|
||||||
"lint": "ng lint",
|
"lint": "ng lint",
|
||||||
"e2e": "ng e2e"
|
"e2e": "ng e2e",
|
||||||
|
"deploy": "firebase deploy --token \"${FIREBASE_TOKEN}\""
|
||||||
},
|
},
|
||||||
"private": true,
|
"private": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@angular/animations": "~7.0.0",
|
"@angular/animations": "~7.0.0",
|
||||||
|
"@angular/cdk": "^7.0.4",
|
||||||
"@angular/common": "~7.0.0",
|
"@angular/common": "~7.0.0",
|
||||||
"@angular/compiler": "~7.0.0",
|
"@angular/compiler": "~7.0.0",
|
||||||
"@angular/core": "~7.0.0",
|
"@angular/core": "~7.0.0",
|
||||||
"@angular/forms": "~7.0.0",
|
"@angular/forms": "~7.0.0",
|
||||||
"@angular/http": "~7.0.0",
|
"@angular/http": "~7.0.0",
|
||||||
|
"@angular/material": "^7.0.4",
|
||||||
"@angular/platform-browser": "~7.0.0",
|
"@angular/platform-browser": "~7.0.0",
|
||||||
"@angular/platform-browser-dynamic": "~7.0.0",
|
"@angular/platform-browser-dynamic": "~7.0.0",
|
||||||
"@angular/router": "~7.0.0",
|
"@angular/router": "~7.0.0",
|
||||||
"core-js": "^2.5.4",
|
"core-js": "^2.5.4",
|
||||||
|
"firebase": "^5.5.8",
|
||||||
|
"hammerjs": "^2.0.8",
|
||||||
"rxjs": "~6.3.3",
|
"rxjs": "~6.3.3",
|
||||||
"zone.js": "~0.8.26"
|
"zone.js": "~0.8.26"
|
||||||
},
|
},
|
||||||
@ -29,10 +34,11 @@
|
|||||||
"@angular/cli": "~7.0.5",
|
"@angular/cli": "~7.0.5",
|
||||||
"@angular/compiler-cli": "~7.0.0",
|
"@angular/compiler-cli": "~7.0.0",
|
||||||
"@angular/language-service": "~7.0.0",
|
"@angular/language-service": "~7.0.0",
|
||||||
"@types/node": "~8.9.4",
|
|
||||||
"@types/jasmine": "~2.8.8",
|
"@types/jasmine": "~2.8.8",
|
||||||
"@types/jasminewd2": "~2.0.3",
|
"@types/jasminewd2": "~2.0.3",
|
||||||
|
"@types/node": "^10.12.7",
|
||||||
"codelyzer": "~4.5.0",
|
"codelyzer": "~4.5.0",
|
||||||
|
"firebase-tools": "^6.1.0",
|
||||||
"jasmine-core": "~2.99.1",
|
"jasmine-core": "~2.99.1",
|
||||||
"jasmine-spec-reporter": "~4.2.1",
|
"jasmine-spec-reporter": "~4.2.1",
|
||||||
"karma": "~3.0.0",
|
"karma": "~3.0.0",
|
||||||
|
25
src/app/animations.ts
Normal file
25
src/app/animations.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import {
|
||||||
|
trigger,
|
||||||
|
animate,
|
||||||
|
transition,
|
||||||
|
style,
|
||||||
|
query, group
|
||||||
|
} from '@angular/animations';
|
||||||
|
|
||||||
|
export const routerTransition = trigger('routerTransition', [
|
||||||
|
transition('* <=> *', [
|
||||||
|
/* order */
|
||||||
|
/* 1 */ query(':enter, :leave', style({ position: 'fixed', width:'100%' })
|
||||||
|
, { optional: true }),
|
||||||
|
/* 2 */ group([ // block executes in parallel
|
||||||
|
query(':enter', [
|
||||||
|
style({ transform: 'translateX(100%)' }),
|
||||||
|
animate('0.5s ease-in-out', style({ transform: 'translateX(0%)' }))
|
||||||
|
], { optional: true }),
|
||||||
|
query(':leave', [
|
||||||
|
style({ transform: 'translateX(0%)' }),
|
||||||
|
animate('0.5s ease-in-out', style({ transform: 'translateX(-100%)' }))
|
||||||
|
], { optional: true }),
|
||||||
|
])
|
||||||
|
])
|
||||||
|
])
|
@ -1,7 +1,19 @@
|
|||||||
import {NgModule} from '@angular/core';
|
import {NgModule} from '@angular/core';
|
||||||
import {Routes, RouterModule} from '@angular/router';
|
import {Routes, RouterModule} from '@angular/router';
|
||||||
|
import {DashboardComponent} from './dashboard/dashboard.component';
|
||||||
|
import {WeatherComponent} from './weather/weather.component';
|
||||||
|
import {SecurityComponent} from './security/security.component';
|
||||||
|
import {SettingsComponent} from './settings/settings.component';
|
||||||
|
import {BatteryComponent} from './battery/battery.component';
|
||||||
|
|
||||||
const routes: Routes = [];
|
const routes: Routes = [
|
||||||
|
{path: 'dashboard', component: DashboardComponent},
|
||||||
|
{path: 'battery', component: BatteryComponent},
|
||||||
|
{path: 'weather', component: WeatherComponent},
|
||||||
|
{path: 'security', component: SecurityComponent},
|
||||||
|
{path: 'settings', component: SettingsComponent},
|
||||||
|
{path: '**', redirectTo: '/dashboard'}
|
||||||
|
];
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
imports: [RouterModule.forRoot(routes)],
|
imports: [RouterModule.forRoot(routes)],
|
||||||
|
@ -1,21 +1,28 @@
|
|||||||
<!--The content below is only a placeholder and can be replaced.-->
|
<mat-toolbar class="bg-primary">
|
||||||
<div style="text-align:center">
|
<mat-icon *ngIf="mobile" class="mr-2" (click)="open = !open">menu</mat-icon>
|
||||||
<h1>
|
<img src="assets/icon.png" class="mr-2" height="24px" width="auto">
|
||||||
Welcome to {{ title }}!
|
<span style="font-weight: 500;">Home Front</span>
|
||||||
</h1>
|
<small class="text-muted ml-2">v{{environment.version}}</small>
|
||||||
<img width="300" alt="Angular Logo" src="">
|
</mat-toolbar>
|
||||||
</div>
|
<mat-drawer-container class="h-100 w-100 example-container" [hasBackdrop]="false">
|
||||||
<h2>Here are some links to help you start: </h2>
|
<mat-drawer class="bg-primary" [mode]="mobile ? 'push' : 'side'" [opened]="open" [disableClose]="!mobile" [autoFocus]="false">
|
||||||
<ul>
|
<mat-nav-list class="p-0">
|
||||||
<li>
|
<mat-divider></mat-divider>
|
||||||
<h2><a target="_blank" rel="noopener" href="https://angular.io/tutorial">Tour of Heroes</a></h2>
|
<a mat-list-item routerLink="dashboard" routerLinkActive="active"><span class="p-3 pl-0 mr-5"><mat-icon class="mr-2">insert_chart</mat-icon> Dashboard</span></a>
|
||||||
</li>
|
<mat-divider></mat-divider>
|
||||||
<li>
|
<a mat-list-item routerLink="battery" routerLinkActive="active"><span class="p-3 pl-0 mr-5"><mat-icon class="mr-2">{{batteryService.icon}}</mat-icon> Power Wall</span></a>
|
||||||
<h2><a target="_blank" rel="noopener" href="https://github.com/angular/angular-cli/wiki">CLI Documentation</a></h2>
|
<mat-divider></mat-divider>
|
||||||
</li>
|
<a mat-list-item routerLink="weather" routerLinkActive="active"><span class="p-3 pl-0 mr-5"><mat-icon class="mr-2">cloud</mat-icon> Weather</span></a>
|
||||||
<li>
|
<mat-divider></mat-divider>
|
||||||
<h2><a target="_blank" rel="noopener" href="https://blog.angular.io/">Angular blog</a></h2>
|
<a mat-list-item routerLink="security" routerLinkActive="active"><span class="p-3 pl-0 mr-5"><mat-icon class="mr-2">security</mat-icon> Security</span></a>
|
||||||
</li>
|
<mat-divider></mat-divider>
|
||||||
</ul>
|
<a mat-list-item routerLink="settings" routerLinkActive="active"><span class="p-3 pl-0 mr-5"><mat-icon class="mr-2">settings</mat-icon> Settings</span></a>
|
||||||
|
<mat-divider></mat-divider>
|
||||||
<router-outlet></router-outlet>
|
</mat-nav-list>
|
||||||
|
</mat-drawer>
|
||||||
|
<mat-drawer-content class="bg-secondary text-white p-4" (click)="open = (mobile && open) ? false : open">
|
||||||
|
<main [@routerTransition]="o.isActivated ? o.activatedRoute : ''">
|
||||||
|
<router-outlet #o="outlet"></router-outlet>
|
||||||
|
</main>
|
||||||
|
</mat-drawer-content>
|
||||||
|
</mat-drawer-container>
|
||||||
|
@ -1,35 +0,0 @@
|
|||||||
import { TestBed, async } from '@angular/core/testing';
|
|
||||||
import { RouterTestingModule } from '@angular/router/testing';
|
|
||||||
import { AppComponent } from './app.component';
|
|
||||||
|
|
||||||
describe('AppComponent', () => {
|
|
||||||
beforeEach(async(() => {
|
|
||||||
TestBed.configureTestingModule({
|
|
||||||
imports: [
|
|
||||||
RouterTestingModule
|
|
||||||
],
|
|
||||||
declarations: [
|
|
||||||
AppComponent
|
|
||||||
],
|
|
||||||
}).compileComponents();
|
|
||||||
}));
|
|
||||||
|
|
||||||
it('should create the app', () => {
|
|
||||||
const fixture = TestBed.createComponent(AppComponent);
|
|
||||||
const app = fixture.debugElement.componentInstance;
|
|
||||||
expect(app).toBeTruthy();
|
|
||||||
});
|
|
||||||
|
|
||||||
it(`should have as title 'HomeFront'`, () => {
|
|
||||||
const fixture = TestBed.createComponent(AppComponent);
|
|
||||||
const app = fixture.debugElement.componentInstance;
|
|
||||||
expect(app.title).toEqual('HomeFront');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should render title in a h1 tag', () => {
|
|
||||||
const fixture = TestBed.createComponent(AppComponent);
|
|
||||||
fixture.detectChanges();
|
|
||||||
const compiled = fixture.debugElement.nativeElement;
|
|
||||||
expect(compiled.querySelector('h1').textContent).toContain('Welcome to HomeFront!');
|
|
||||||
});
|
|
||||||
});
|
|
@ -1,10 +1,23 @@
|
|||||||
import {Component} from '@angular/core';
|
import {Component} from '@angular/core';
|
||||||
|
import {BatteryService} from './battery/battery.service';
|
||||||
|
import {BreakpointObserver, Breakpoints} from '@angular/cdk/layout';
|
||||||
|
import {environment} from '../environments/environment';
|
||||||
|
import {routerTransition} from './animations';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-root',
|
selector: 'app-root',
|
||||||
templateUrl: './app.component.html',
|
templateUrl: './app.component.html',
|
||||||
styleUrls: ['./app.component.scss']
|
animations: [routerTransition]
|
||||||
})
|
})
|
||||||
export class AppComponent {
|
export class AppComponent {
|
||||||
title = 'HomeFront';
|
mobile = true;
|
||||||
|
open = false;
|
||||||
|
environment = environment;
|
||||||
|
|
||||||
|
constructor(public batteryService: BatteryService, breakpointObserver: BreakpointObserver) {
|
||||||
|
breakpointObserver.observe([Breakpoints.Handset]).subscribe(result => {
|
||||||
|
this.mobile = result.matches;
|
||||||
|
this.open = !this.mobile;
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,16 +3,47 @@ import { NgModule } from '@angular/core';
|
|||||||
|
|
||||||
import {AppRoutingModule} from './app-routing.module';
|
import {AppRoutingModule} from './app-routing.module';
|
||||||
import {AppComponent} from './app.component';
|
import {AppComponent} from './app.component';
|
||||||
|
import {BrowserAnimationsModule} from '@angular/platform-browser/animations';
|
||||||
|
import {FormsModule} from '@angular/forms';
|
||||||
|
import {
|
||||||
|
MatCardModule,
|
||||||
|
MatDividerModule,
|
||||||
|
MatIconModule,
|
||||||
|
MatListModule,
|
||||||
|
MatSidenavModule,
|
||||||
|
MatToolbarModule
|
||||||
|
} from '@angular/material';
|
||||||
|
import { DashboardComponent } from './dashboard/dashboard.component';
|
||||||
|
import { BatteryComponent } from './battery/battery.component';
|
||||||
|
import { WeatherComponent } from './weather/weather.component';
|
||||||
|
import { SecurityComponent } from './security/security.component';
|
||||||
|
import { SettingsComponent } from './settings/settings.component';
|
||||||
|
import {HttpClientModule} from '@angular/common/http';
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
declarations: [
|
declarations: [
|
||||||
AppComponent
|
AppComponent,
|
||||||
|
DashboardComponent,
|
||||||
|
BatteryComponent,
|
||||||
|
WeatherComponent,
|
||||||
|
SecurityComponent,
|
||||||
|
SettingsComponent
|
||||||
],
|
],
|
||||||
imports: [
|
imports: [
|
||||||
|
AppRoutingModule,
|
||||||
BrowserModule,
|
BrowserModule,
|
||||||
AppRoutingModule
|
BrowserAnimationsModule,
|
||||||
|
FormsModule,
|
||||||
|
HttpClientModule,
|
||||||
|
MatCardModule,
|
||||||
|
MatDividerModule,
|
||||||
|
MatIconModule,
|
||||||
|
MatListModule,
|
||||||
|
MatSidenavModule,
|
||||||
|
MatToolbarModule,
|
||||||
],
|
],
|
||||||
providers: [],
|
providers: [],
|
||||||
bootstrap: [AppComponent]
|
bootstrap: [AppComponent]
|
||||||
})
|
})
|
||||||
export class AppModule { }
|
export class AppModule {
|
||||||
|
}
|
||||||
|
3
src/app/battery/battery.component.html
Normal file
3
src/app/battery/battery.component.html
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<p>
|
||||||
|
batterys works!
|
||||||
|
</p>
|
14
src/app/battery/battery.component.ts
Normal file
14
src/app/battery/battery.component.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import { Component, OnInit } from '@angular/core';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'app-batterys',
|
||||||
|
templateUrl: './battery.component.html'
|
||||||
|
})
|
||||||
|
export class BatteryComponent implements OnInit {
|
||||||
|
|
||||||
|
constructor() { }
|
||||||
|
|
||||||
|
ngOnInit() {
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
42
src/app/battery/battery.service.ts
Normal file
42
src/app/battery/battery.service.ts
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
import {Injectable} from '@angular/core';
|
||||||
|
|
||||||
|
@Injectable({
|
||||||
|
providedIn: 'root'
|
||||||
|
})
|
||||||
|
export class BatteryService {
|
||||||
|
percentage: number[] = [0];
|
||||||
|
charging: boolean;
|
||||||
|
|
||||||
|
get average() {
|
||||||
|
return this.percentage.reduce((acc, battery) => acc + battery, 0) / this.percentage.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
get icon() {
|
||||||
|
if (this.charging == null) return 'battery_unknown';
|
||||||
|
|
||||||
|
let temp = 'battery';
|
||||||
|
if (this.charging) temp += '_charging';
|
||||||
|
|
||||||
|
let average = this.average;
|
||||||
|
if (average <= 20) {
|
||||||
|
temp += '_20';
|
||||||
|
} else if (average <= 30) {
|
||||||
|
temp += '_30';
|
||||||
|
} else if (average <= 50) {
|
||||||
|
temp += '_50';
|
||||||
|
} else if (average <= 60) {
|
||||||
|
temp += '_60';
|
||||||
|
} else if (average <= 80) {
|
||||||
|
temp += '_80';
|
||||||
|
} else if (average <= 90) {
|
||||||
|
temp += '_90';
|
||||||
|
} else if (average > 90) {
|
||||||
|
temp += 'full'
|
||||||
|
}
|
||||||
|
|
||||||
|
return temp;
|
||||||
|
}
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
}
|
||||||
|
}
|
3
src/app/dashboard/dashboard.component.html
Normal file
3
src/app/dashboard/dashboard.component.html
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<p>
|
||||||
|
dashboard works!
|
||||||
|
</p>
|
14
src/app/dashboard/dashboard.component.ts
Normal file
14
src/app/dashboard/dashboard.component.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import { Component, OnInit } from '@angular/core';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'app-dashboard',
|
||||||
|
templateUrl: './dashboard.component.html'
|
||||||
|
})
|
||||||
|
export class DashboardComponent implements OnInit {
|
||||||
|
|
||||||
|
constructor() { }
|
||||||
|
|
||||||
|
ngOnInit() {
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
3
src/app/security/security.component.html
Normal file
3
src/app/security/security.component.html
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<p>
|
||||||
|
security works!
|
||||||
|
</p>
|
14
src/app/security/security.component.ts
Normal file
14
src/app/security/security.component.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import { Component, OnInit } from '@angular/core';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'app-security',
|
||||||
|
templateUrl: './security.component.html'
|
||||||
|
})
|
||||||
|
export class SecurityComponent implements OnInit {
|
||||||
|
|
||||||
|
constructor() { }
|
||||||
|
|
||||||
|
ngOnInit() {
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
3
src/app/settings/settings.component.html
Normal file
3
src/app/settings/settings.component.html
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<p>
|
||||||
|
settings works!
|
||||||
|
</p>
|
14
src/app/settings/settings.component.ts
Normal file
14
src/app/settings/settings.component.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import { Component, OnInit } from '@angular/core';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'app-settings',
|
||||||
|
templateUrl: './settings.component.html'
|
||||||
|
})
|
||||||
|
export class SettingsComponent implements OnInit {
|
||||||
|
|
||||||
|
constructor() { }
|
||||||
|
|
||||||
|
ngOnInit() {
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
27
src/app/weather/weather.component.html
Normal file
27
src/app/weather/weather.component.html
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
<div class="d-flex flex-column flex-md-row justify-content-center align-items-center" style="flex-grow: 1;">
|
||||||
|
<i [class]="'wi wi-fw wi-' + weatherService.icon" style="font-size: 6rem"></i>
|
||||||
|
<div class="ml-0 ml-md-3">
|
||||||
|
<h1 class="my-4 my-md-0 font-weight-bold text-center text-md-left">{{weatherService.temp}} °C</h1>
|
||||||
|
<h3 class="m-0">{{weatherService.weather}}</h3>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="my-4 d-flex mx-auto" style="max-width: 450px">
|
||||||
|
<div class="text-center" style="flex-grow: 1">
|
||||||
|
<i class="wi wi-fw wi-humidity"></i> {{weatherService.humidity}} %
|
||||||
|
</div>
|
||||||
|
<div class="text-center" style="flex-grow: 1">
|
||||||
|
<i class="wi wi-fw wi-cloud"></i> {{weatherService.cloudCover}} %
|
||||||
|
</div>
|
||||||
|
<div class="text-center" style="flex-grow: 1">
|
||||||
|
<i class="wi wi-fw wi-strong-wind"></i> {{weatherService.wind[1]}} KM/H
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<mat-divider></mat-divider>
|
||||||
|
<div class="my-4 d-flex justify-content-center">
|
||||||
|
<div *ngFor="let w of weatherService.forecast" class="d-flex flex-column align-items-center" style="max-width: 75px; flex-grow: 1">
|
||||||
|
{{w.day}}
|
||||||
|
<i [class]="'my-2 wi wi-fw wi-' + w.icon" style="font-size: 2rem"></i>
|
||||||
|
{{w.temp}} °C
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<mat-divider></mat-divider>
|
15
src/app/weather/weather.component.ts
Normal file
15
src/app/weather/weather.component.ts
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
import { Component, OnInit } from '@angular/core';
|
||||||
|
import {WeatherService} from './weather.service';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'app-weather',
|
||||||
|
templateUrl: './weather.component.html'
|
||||||
|
})
|
||||||
|
export class WeatherComponent implements OnInit {
|
||||||
|
|
||||||
|
constructor(public weatherService: WeatherService) { }
|
||||||
|
|
||||||
|
ngOnInit() {
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
56
src/app/weather/weather.service.ts
Normal file
56
src/app/weather/weather.service.ts
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
import {Injectable} from '@angular/core';
|
||||||
|
import {HttpClient} from '@angular/common/http';
|
||||||
|
import {timer} from 'rxjs';
|
||||||
|
|
||||||
|
@Injectable({
|
||||||
|
providedIn: 'root'
|
||||||
|
})
|
||||||
|
export class WeatherService {
|
||||||
|
readonly apiKey = 'e8391af54b6fc09dc82b019fc68b8409';
|
||||||
|
readonly city = 'London';
|
||||||
|
readonly countryCode = 'CA';
|
||||||
|
readonly days = ['SUN', 'MON', 'TUE', 'WED', 'THU', 'FRI', 'SAT'];
|
||||||
|
readonly weatherCodes = require('./weatherCodes.json');
|
||||||
|
|
||||||
|
// Weather information
|
||||||
|
cloudCover = 0;
|
||||||
|
forecast = [];
|
||||||
|
humidity = 0;
|
||||||
|
icon: string;
|
||||||
|
pressure = 0;
|
||||||
|
sunrise: Date;
|
||||||
|
sunset: Date;
|
||||||
|
temp = 0;
|
||||||
|
tempMin = 0;
|
||||||
|
tempMax = 0;
|
||||||
|
weather: string = '';
|
||||||
|
wind: [number, number] = [0, 0];
|
||||||
|
|
||||||
|
constructor(httpClient: HttpClient) {
|
||||||
|
timer(0, 5 * 60000).subscribe(async () => {
|
||||||
|
httpClient.get(`https://api.openweathermap.org/data/2.5/weather?q=${this.city},${this.countryCode}&APPID=${this.apiKey}&units=metric`).toPromise().then((weather: any) => {
|
||||||
|
this.cloudCover = weather.clouds.all;
|
||||||
|
this.humidity = weather.main.humidity;
|
||||||
|
this.icon = this.weatherCodes[weather.weather[0].id].icon;
|
||||||
|
this.pressure = weather.main.pressure;
|
||||||
|
this.sunrise = new Date(weather.sys.sunrise);
|
||||||
|
this.sunset = new Date(weather.sys.sunset);
|
||||||
|
this.temp = Math.round(weather.main.temp);
|
||||||
|
this.tempMin = Math.round(weather.main.temp_min);
|
||||||
|
this.tempMax = Math.round(weather.main.temp_max);
|
||||||
|
this.weather = weather.weather[0].description;
|
||||||
|
this.wind = [weather.wind.deg, Math.round(weather.wind.speed)];
|
||||||
|
});
|
||||||
|
httpClient.get(`https://api.openweathermap.org/data/2.5/forecast?q=${this.city},${this.countryCode}&APPID=${this.apiKey}&units=metric`).toPromise().then((weather: any) => {
|
||||||
|
let temp = weather.list.filter(weather => weather.dt_txt.indexOf('12:00:00') != -1);
|
||||||
|
temp.splice(0, temp.length - 5);
|
||||||
|
this.forecast = temp.map(weather => ({
|
||||||
|
day: this.days[new Date(weather.dt_txt).getDay()],
|
||||||
|
icon: this.weatherCodes[weather.weather[0].id].icon,
|
||||||
|
temp: Math.round(weather.main.temp)
|
||||||
|
}));
|
||||||
|
console.log(this.forecast[3])
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
367
src/app/weather/weatherCodes.json
Normal file
367
src/app/weather/weatherCodes.json
Normal file
@ -0,0 +1,367 @@
|
|||||||
|
{
|
||||||
|
"200": {
|
||||||
|
"label": "thunderstorm with light rain",
|
||||||
|
"icon": "storm-showers"
|
||||||
|
},
|
||||||
|
|
||||||
|
"201": {
|
||||||
|
"label": "thunderstorm with rain",
|
||||||
|
"icon": "storm-showers"
|
||||||
|
},
|
||||||
|
|
||||||
|
"202": {
|
||||||
|
"label": "thunderstorm with heavy rain",
|
||||||
|
"icon": "storm-showers"
|
||||||
|
},
|
||||||
|
|
||||||
|
"210": {
|
||||||
|
"label": "light thunderstorm",
|
||||||
|
"icon": "storm-showers"
|
||||||
|
},
|
||||||
|
|
||||||
|
"211": {
|
||||||
|
"label": "thunderstorm",
|
||||||
|
"icon": "thunderstorm"
|
||||||
|
},
|
||||||
|
|
||||||
|
"212": {
|
||||||
|
"label": "heavy thunderstorm",
|
||||||
|
"icon": "thunderstorm"
|
||||||
|
},
|
||||||
|
|
||||||
|
"221": {
|
||||||
|
"label": "ragged thunderstorm",
|
||||||
|
"icon": "thunderstorm"
|
||||||
|
},
|
||||||
|
|
||||||
|
"230": {
|
||||||
|
"label": "thunderstorm with light drizzle",
|
||||||
|
"icon": "storm-showers"
|
||||||
|
},
|
||||||
|
|
||||||
|
"231": {
|
||||||
|
"label": "thunderstorm with drizzle",
|
||||||
|
"icon": "storm-showers"
|
||||||
|
},
|
||||||
|
|
||||||
|
"232": {
|
||||||
|
"label": "thunderstorm with heavy drizzle",
|
||||||
|
"icon": "storm-showers"
|
||||||
|
},
|
||||||
|
|
||||||
|
"300": {
|
||||||
|
"label": "light intensity drizzle",
|
||||||
|
"icon": "sprinkle"
|
||||||
|
},
|
||||||
|
|
||||||
|
"301": {
|
||||||
|
"label": "drizzle",
|
||||||
|
"icon": "sprinkle"
|
||||||
|
},
|
||||||
|
|
||||||
|
"302": {
|
||||||
|
"label": "heavy intensity drizzle",
|
||||||
|
"icon": "sprinkle"
|
||||||
|
},
|
||||||
|
|
||||||
|
"310": {
|
||||||
|
"label": "light intensity drizzle rain",
|
||||||
|
"icon": "sprinkle"
|
||||||
|
},
|
||||||
|
|
||||||
|
"311": {
|
||||||
|
"label": "drizzle rain",
|
||||||
|
"icon": "sprinkle"
|
||||||
|
},
|
||||||
|
|
||||||
|
"312": {
|
||||||
|
"label": "heavy intensity drizzle rain",
|
||||||
|
"icon": "sprinkle"
|
||||||
|
},
|
||||||
|
|
||||||
|
"313": {
|
||||||
|
"label": "shower rain and drizzle",
|
||||||
|
"icon": "sprinkle"
|
||||||
|
},
|
||||||
|
|
||||||
|
"314": {
|
||||||
|
"label": "heavy shower rain and drizzle",
|
||||||
|
"icon": "sprinkle"
|
||||||
|
},
|
||||||
|
|
||||||
|
"321": {
|
||||||
|
"label": "shower drizzle",
|
||||||
|
"icon": "sprinkle"
|
||||||
|
},
|
||||||
|
|
||||||
|
"500": {
|
||||||
|
"label": "light rain",
|
||||||
|
"icon": "rain"
|
||||||
|
},
|
||||||
|
|
||||||
|
"501": {
|
||||||
|
"label": "moderate rain",
|
||||||
|
"icon": "rain"
|
||||||
|
},
|
||||||
|
|
||||||
|
"502": {
|
||||||
|
"label": "heavy intensity rain",
|
||||||
|
"icon": "rain"
|
||||||
|
},
|
||||||
|
|
||||||
|
"503": {
|
||||||
|
"label": "very heavy rain",
|
||||||
|
"icon": "rain"
|
||||||
|
},
|
||||||
|
|
||||||
|
"504": {
|
||||||
|
"label": "extreme rain",
|
||||||
|
"icon": "rain"
|
||||||
|
},
|
||||||
|
|
||||||
|
"511": {
|
||||||
|
"label": "freezing rain",
|
||||||
|
"icon": "rain-mix"
|
||||||
|
},
|
||||||
|
|
||||||
|
"520": {
|
||||||
|
"label": "light intensity shower rain",
|
||||||
|
"icon": "showers"
|
||||||
|
},
|
||||||
|
|
||||||
|
"521": {
|
||||||
|
"label": "shower rain",
|
||||||
|
"icon": "showers"
|
||||||
|
},
|
||||||
|
|
||||||
|
"522": {
|
||||||
|
"label": "heavy intensity shower rain",
|
||||||
|
"icon": "showers"
|
||||||
|
},
|
||||||
|
|
||||||
|
"531": {
|
||||||
|
"label": "ragged shower rain",
|
||||||
|
"icon": "showers"
|
||||||
|
},
|
||||||
|
|
||||||
|
"600": {
|
||||||
|
"label": "light snow",
|
||||||
|
"icon": "snow"
|
||||||
|
},
|
||||||
|
|
||||||
|
"601": {
|
||||||
|
"label": "snow",
|
||||||
|
"icon": "snow"
|
||||||
|
},
|
||||||
|
|
||||||
|
"602": {
|
||||||
|
"label": "heavy snow",
|
||||||
|
"icon": "snow"
|
||||||
|
},
|
||||||
|
|
||||||
|
"611": {
|
||||||
|
"label": "sleet",
|
||||||
|
"icon": "sleet"
|
||||||
|
},
|
||||||
|
|
||||||
|
"612": {
|
||||||
|
"label": "shower sleet",
|
||||||
|
"icon": "sleet"
|
||||||
|
},
|
||||||
|
|
||||||
|
"615": {
|
||||||
|
"label": "light rain and snow",
|
||||||
|
"icon": "rain-mix"
|
||||||
|
},
|
||||||
|
|
||||||
|
"616": {
|
||||||
|
"label": "rain and snow",
|
||||||
|
"icon": "rain-mix"
|
||||||
|
},
|
||||||
|
|
||||||
|
"620": {
|
||||||
|
"label": "light shower snow",
|
||||||
|
"icon": "rain-mix"
|
||||||
|
},
|
||||||
|
|
||||||
|
"621": {
|
||||||
|
"label": "shower snow",
|
||||||
|
"icon": "rain-mix"
|
||||||
|
},
|
||||||
|
|
||||||
|
"622": {
|
||||||
|
"label": "heavy shower snow",
|
||||||
|
"icon": "rain-mix"
|
||||||
|
},
|
||||||
|
|
||||||
|
"701": {
|
||||||
|
"label": "mist",
|
||||||
|
"icon": "sprinkle"
|
||||||
|
},
|
||||||
|
|
||||||
|
"711": {
|
||||||
|
"label": "smoke",
|
||||||
|
"icon": "smoke"
|
||||||
|
},
|
||||||
|
|
||||||
|
"721": {
|
||||||
|
"label": "haze",
|
||||||
|
"icon": "day-haze"
|
||||||
|
},
|
||||||
|
|
||||||
|
"731": {
|
||||||
|
"label": "sand, dust whirls",
|
||||||
|
"icon": "cloudy-gusts"
|
||||||
|
},
|
||||||
|
|
||||||
|
"741": {
|
||||||
|
"label": "fog",
|
||||||
|
"icon": "fog"
|
||||||
|
},
|
||||||
|
|
||||||
|
"751": {
|
||||||
|
"label": "sand",
|
||||||
|
"icon": "cloudy-gusts"
|
||||||
|
},
|
||||||
|
|
||||||
|
"761": {
|
||||||
|
"label": "dust",
|
||||||
|
"icon": "dust"
|
||||||
|
},
|
||||||
|
|
||||||
|
"762": {
|
||||||
|
"label": "volcanic ash",
|
||||||
|
"icon": "smog"
|
||||||
|
},
|
||||||
|
|
||||||
|
"771": {
|
||||||
|
"label": "squalls",
|
||||||
|
"icon": "day-windy"
|
||||||
|
},
|
||||||
|
|
||||||
|
"781": {
|
||||||
|
"label": "tornado",
|
||||||
|
"icon": "tornado"
|
||||||
|
},
|
||||||
|
|
||||||
|
"800": {
|
||||||
|
"label": "clear sky",
|
||||||
|
"icon": "day-sunny"
|
||||||
|
},
|
||||||
|
|
||||||
|
"801": {
|
||||||
|
"label": "few clouds",
|
||||||
|
"icon": "cloudy"
|
||||||
|
},
|
||||||
|
|
||||||
|
"802": {
|
||||||
|
"label": "scattered clouds",
|
||||||
|
"icon": "cloudy"
|
||||||
|
},
|
||||||
|
|
||||||
|
"803": {
|
||||||
|
"label": "broken clouds",
|
||||||
|
"icon": "cloudy"
|
||||||
|
},
|
||||||
|
|
||||||
|
"804": {
|
||||||
|
"label": "overcast clouds",
|
||||||
|
"icon": "cloudy"
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
"900": {
|
||||||
|
"label": "tornado",
|
||||||
|
"icon": "tornado"
|
||||||
|
},
|
||||||
|
|
||||||
|
"901": {
|
||||||
|
"label": "tropical storm",
|
||||||
|
"icon": "hurricane"
|
||||||
|
},
|
||||||
|
|
||||||
|
"902": {
|
||||||
|
"label": "hurricane",
|
||||||
|
"icon": "hurricane"
|
||||||
|
},
|
||||||
|
|
||||||
|
"903": {
|
||||||
|
"label": "cold",
|
||||||
|
"icon": "snowflake-cold"
|
||||||
|
},
|
||||||
|
|
||||||
|
"904": {
|
||||||
|
"label": "hot",
|
||||||
|
"icon": "hot"
|
||||||
|
},
|
||||||
|
|
||||||
|
"905": {
|
||||||
|
"label": "windy",
|
||||||
|
"icon": "windy"
|
||||||
|
},
|
||||||
|
|
||||||
|
"906": {
|
||||||
|
"label": "hail",
|
||||||
|
"icon": "hail"
|
||||||
|
},
|
||||||
|
|
||||||
|
"951": {
|
||||||
|
"label": "calm",
|
||||||
|
"icon": "day-sunny"
|
||||||
|
},
|
||||||
|
|
||||||
|
"952": {
|
||||||
|
"label": "light breeze",
|
||||||
|
"icon": "cloudy-gusts"
|
||||||
|
},
|
||||||
|
|
||||||
|
"953": {
|
||||||
|
"label": "gentle breeze",
|
||||||
|
"icon": "cloudy-gusts"
|
||||||
|
},
|
||||||
|
|
||||||
|
"954": {
|
||||||
|
"label": "moderate breeze",
|
||||||
|
"icon": "cloudy-gusts"
|
||||||
|
},
|
||||||
|
|
||||||
|
"955": {
|
||||||
|
"label": "fresh breeze",
|
||||||
|
"icon": "cloudy-gusts"
|
||||||
|
},
|
||||||
|
|
||||||
|
"956": {
|
||||||
|
"label": "strong breeze",
|
||||||
|
"icon": "cloudy-gusts"
|
||||||
|
},
|
||||||
|
|
||||||
|
"957": {
|
||||||
|
"label": "high wind, near gale",
|
||||||
|
"icon": "cloudy-gusts"
|
||||||
|
},
|
||||||
|
|
||||||
|
"958": {
|
||||||
|
"label": "gale",
|
||||||
|
"icon": "cloudy-gusts"
|
||||||
|
},
|
||||||
|
|
||||||
|
"959": {
|
||||||
|
"label": "severe gale",
|
||||||
|
"icon": "cloudy-gusts"
|
||||||
|
},
|
||||||
|
|
||||||
|
"960": {
|
||||||
|
"label": "storm",
|
||||||
|
"icon": "thunderstorm"
|
||||||
|
},
|
||||||
|
|
||||||
|
"961": {
|
||||||
|
"label": "violent storm",
|
||||||
|
"icon": "thunderstorm"
|
||||||
|
},
|
||||||
|
|
||||||
|
"962": {
|
||||||
|
"label": "hurricane",
|
||||||
|
"icon": "cloudy-gusts"
|
||||||
|
}
|
||||||
|
}
|
7
src/assets/bootstrap.min.css
vendored
Normal file
7
src/assets/bootstrap.min.css
vendored
Normal file
File diff suppressed because one or more lines are too long
BIN
src/assets/icon.png
Normal file
BIN
src/assets/icon.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 35 KiB |
@ -1,3 +1,12 @@
|
|||||||
export const environment = {
|
export const environment = {
|
||||||
production: true
|
firebase: {
|
||||||
|
apiKey: "AIzaSyAs3FvBCADM66wR1-leBz6aIjK1wZfUxRo",
|
||||||
|
authDomain: "homefront-2ccb4.firebaseapp.com",
|
||||||
|
databaseURL: "https://homefront-2ccb4.firebaseio.com",
|
||||||
|
projectId: "homefront-2ccb4",
|
||||||
|
storageBucket: "homefront-2ccb4.appspot.com",
|
||||||
|
messagingSenderId: "482384317544"
|
||||||
|
},
|
||||||
|
production: true,
|
||||||
|
version: require('../../package.json').version
|
||||||
};
|
};
|
||||||
|
@ -1,16 +1,12 @@
|
|||||||
// This file can be replaced during build by using the `fileReplacements` array.
|
|
||||||
// `ng build --prod` replaces `environment.ts` with `environment.prod.ts`.
|
|
||||||
// The list of file replacements can be found in `angular.json`.
|
|
||||||
|
|
||||||
export const environment = {
|
export const environment = {
|
||||||
production: false
|
firebase: {
|
||||||
|
apiKey: "AIzaSyAs3FvBCADM66wR1-leBz6aIjK1wZfUxRo",
|
||||||
|
authDomain: "homefront-2ccb4.firebaseapp.com",
|
||||||
|
databaseURL: "https://homefront-2ccb4.firebaseio.com",
|
||||||
|
projectId: "homefront-2ccb4",
|
||||||
|
storageBucket: "homefront-2ccb4.appspot.com",
|
||||||
|
messagingSenderId: "482384317544"
|
||||||
|
},
|
||||||
|
production: false,
|
||||||
|
version: require('../../package.json').version
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
|
||||||
* For easier debugging in development mode, you can import the following file
|
|
||||||
* to ignore zone related error stack frames such as `zone.run`, `zoneDelegate.invokeTask`.
|
|
||||||
*
|
|
||||||
* This import should be commented out in production mode because it will have a negative impact
|
|
||||||
* on performance if an error is thrown.
|
|
||||||
*/
|
|
||||||
// import 'zone.js/dist/zone-error'; // Included with Angular CLI.
|
|
||||||
|
BIN
src/favicon.ico
BIN
src/favicon.ico
Binary file not shown.
Before Width: | Height: | Size: 5.3 KiB |
@ -6,7 +6,8 @@
|
|||||||
<base href="/">
|
<base href="/">
|
||||||
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
<link rel="icon" type="image/x-icon" href="favicon.ico">
|
<link rel="icon" type="image/x-icon" href="assets/icon.png">
|
||||||
|
<link rel="stylesheet" type="text/css" href="assets/bootstrap.min.css">
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<app-root></app-root>
|
<app-root></app-root>
|
||||||
|
@ -4,6 +4,8 @@ import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';
|
|||||||
import { AppModule } from './app/app.module';
|
import { AppModule } from './app/app.module';
|
||||||
import { environment } from './environments/environment';
|
import { environment } from './environments/environment';
|
||||||
|
|
||||||
|
import 'hammerjs';
|
||||||
|
|
||||||
if (environment.production) {
|
if (environment.production) {
|
||||||
enableProdMode();
|
enableProdMode();
|
||||||
}
|
}
|
||||||
|
@ -1 +1,58 @@
|
|||||||
/* You can add global styles to this file, and also import other style files */
|
/* You can add global styles to this file, and also import other style files */
|
||||||
|
@import url('https://fonts.googleapis.com/css?family=Archivo|Material+Icons');
|
||||||
|
@import "~@angular/material/prebuilt-themes/indigo-pink.css";
|
||||||
|
@import url('https://cdnjs.cloudflare.com/ajax/libs/weather-icons/2.0.9/css/weather-icons.min.css');
|
||||||
|
|
||||||
|
html, body {
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
font-family: 'Archivo', sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mat-list-item {
|
||||||
|
height: auto !important;
|
||||||
|
color: #9CA4B6 !important;
|
||||||
|
|
||||||
|
&.active {
|
||||||
|
background-color: #262930 !important;
|
||||||
|
color: #1CA8DD !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.mat-divider {
|
||||||
|
color: #9CA4B6 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bg-primary {
|
||||||
|
background-color: #f8f8f8 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bg-secondary {
|
||||||
|
background-color: #2F323A !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.scale-150 {
|
||||||
|
transform: scale(1.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
.scale-200 {
|
||||||
|
transform: scale(2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.scale-300 {
|
||||||
|
transform: scale(3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.scale-500 {
|
||||||
|
transform: scale(5);
|
||||||
|
}
|
||||||
|
|
||||||
|
.material-icons {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
"extends": "../tsconfig.json",
|
"extends": "../tsconfig.json",
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"outDir": "../out-tsc/app",
|
"outDir": "../out-tsc/app",
|
||||||
"types": []
|
"types": ["node"]
|
||||||
},
|
},
|
||||||
"exclude": [
|
"exclude": [
|
||||||
"test.ts",
|
"test.ts",
|
||||||
|
Loading…
Reference in New Issue
Block a user