UNPKG

ngx-ice-tweet

Version:

tweeter component to show tweets by a hashtag given, it can be order by text, date, or user name

242 lines (234 loc) 19.6 kB
import { Injectable, NgModule, Component, EventEmitter, Input, Output, ViewEncapsulation, defineInjectable } from '@angular/core'; import { HttpClient, HttpHeaders, HttpParams, HttpClientModule } from '@angular/common/http'; import { Subject } from 'rxjs'; import { takeUntil } from 'rxjs/operators'; import { BrowserModule } from '@angular/platform-browser'; import { BrowserAnimationsModule } from '@angular/platform-browser/animations'; import { FormsModule } from '@angular/forms'; import { MatCardModule, MatSelectModule } from '@angular/material'; /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ class TweetService { constructor() { } } TweetService.decorators = [ { type: Injectable, args: [{ providedIn: 'root' },] } ]; /** @nocollapse */ TweetService.ctorParameters = () => []; /** @nocollapse */ TweetService.ngInjectableDef = defineInjectable({ factory: function TweetService_Factory() { return new TweetService(); }, token: TweetService, providedIn: "root" }); /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ /** @type {?} */ const TweetConfig = { tokenUrl: 'https://api.twitter.com/oauth2/token', tweetSearchUrl: 'https://api.twitter.com/1.1/search/tweets.json' }; class TweetComponent { /** * @param {?} http */ constructor(http) { this.http = http; this.errorMesage = new EventEmitter(); this.filterBy = 'created_at'; this.filterOptions = ['created_at', 'text', 'user.name']; this._unsubscribeAll = new Subject(); } /** * @return {?} */ ngOnInit() { if (this.evalData()) { this.getToken(); } else { this.sendErrorMesage('No Public, Privite Key'); this.ngOnDestroy(); } } /** * @return {?} */ setFilterBy() { return this.tweets.sort((a, b) => a[this.filterBy] > b[this.filterBy] ? 1 : a[this.filterBy] === b[this.filterBy] ? 0 : -1); } /** * @private * @return {?} */ evalData() { return (this.keyPublic && this.keyPrivate && this.keyPublic !== null && this.keyPrivate !== null && this.keyPublic !== '' && this.keyPrivate !== ''); } /** * @private * @return {?} */ evaHash() { return (this.hashTag && this.hashTag !== null && this.hashTag !== ''); } /** * @private * @param {?} token * @return {?} */ evalToken(token) { return (token.token_type && token.access_token && token.token_type !== null && token.access_token !== null && token.token_type !== '' && token.access_token !== ''); } /** * @private * @return {?} */ getToken() { if (this.evaHash()) { /** @type {?} */ const basic = btoa(`${this.keyPublic}:${this.keyPrivate}`); /** @type {?} */ const headers = new HttpHeaders({ 'Content-Type': 'application/x-www-form-urlencoded', 'Authorization': `Basic ${basic}` }); /** @type {?} */ const payload = new FormData(); payload.append('grant_type', 'client_credentials'); this.http.post(TweetConfig.tokenUrl, payload, { headers: headers }) .pipe(takeUntil(this._unsubscribeAll)) .subscribe(token => this.getTweetList(token), error => this.sendErrorMesage(error)); } else { this.sendErrorMesage('No HasgTag set For Searching'); } } /** * @private * @param {?} token * @return {?} */ getTweetList(token) { if (this.evalToken(token)) { /** @type {?} */ const headers = new HttpHeaders({ 'Authorization': `${token.token_type} ${token.access_token}` }); /** @type {?} */ const params = new HttpParams().set('q', `%23${this.hashTag}`); this.http.get(TweetConfig.tweetSearchUrl, { headers: headers, params: params }) .pipe(takeUntil(this._unsubscribeAll)) .subscribe(({ statuses }) => this.tweets = (/** @type {?} */ (statuses)), error => this.sendErrorMesage(error)); } else { this.sendErrorMesage('Error Tweeter Token'); } } /** * @private * @param {?} men * @return {?} */ sendErrorMesage(men) { this.errorMesage.emit(men); } /** * @return {?} */ ngOnDestroy() { this._unsubscribeAll.next(); this._unsubscribeAll.complete(); } } TweetComponent.decorators = [ { type: Component, args: [{ selector: 'tweet', template: `<div class="container" *ngIf="tweets && tweets.length > 0"> <div> <h4>filter by:</h4> <mat-form-field> <mat-select [(value)]="filterBy"> <mat-option *ngFor="let opt of filterOptions" value="opt">{{opt}}</mat-option> </mat-select> </mat-form-field> </div> <mat-card *ngFor="let tt of setFilterBy()" > <mat-card-header> <mat-card-title>{{tt.user.screen_name}}</mat-card-title> <mat-card-subtitle>@{{tt.user.name}}</mat-card-subtitle> </mat-card-header> <img matCardImage [src]="tt.user.profile_image_url"> <mat-card-content> {{tt.text}} </mat-card-content> <mat-card-actions align="start"> <!--<button mat-button (click)="onAction1">Action1</button>--> </mat-card-actions> <mat-card-footer> </mat-card-footer> </mat-card> </div>`, encapsulation: ViewEncapsulation.Native, styles: [` .container { width: 100%; margin: 2rem; display: flex; flex-direction: column; } mat-card { margin-bottom: 2rem; flex: 1 1 100%; }`] }] } ]; /** @nocollapse */ TweetComponent.ctorParameters = () => [ { type: HttpClient } ]; TweetComponent.propDecorators = { keyPublic: [{ type: Input }], keyPrivate: [{ type: Input }], hashTag: [{ type: Input }], errorMesage: [{ type: Output }] }; /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ class TweetModule { } TweetModule.decorators = [ { type: NgModule, args: [{ declarations: [TweetComponent], imports: [ BrowserModule, HttpClientModule, BrowserAnimationsModule, MatCardModule, MatSelectModule, FormsModule ], exports: [TweetComponent] },] } ]; /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ export { TweetService, TweetComponent, TweetModule }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ngx-ice-tweet.js.map","sources":["ng://ngx-ice-tweet/lib/tweet.service.ts","ng://ngx-ice-tweet/lib/tweet.component.ts","ng://ngx-ice-tweet/lib/tweet.module.ts"],"sourcesContent":["import { Injectable } from '@angular/core';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class TweetService {\n\n  constructor() { }\n}\n","import {Component, EventEmitter, Input, OnDestroy, OnInit, Output, ViewEncapsulation} from '@angular/core';\nimport {HttpClient, HttpHeaders, HttpParams} from '@angular/common/http';\nimport {Subject} from 'rxjs';\nimport {takeUntil} from 'rxjs/operators';\n\nconst TweetConfig = {\n  tokenUrl: 'https://api.twitter.com/oauth2/token',\n  tweetSearchUrl: 'https://api.twitter.com/1.1/search/tweets.json'\n};\n\ninterface Token {\n  token_type: string;\n  access_token: string;\n}\n\ninterface Tweets {\n  created_at: Date;\n  id: number;\n  id_str: string;\n  text: string;\n  user: any;\n  entities: any;\n}\n\n@Component({\n  selector: 'tweet',\n  template: `<div class=\"container\" *ngIf=\"tweets && tweets.length > 0\">\n    <div>\n      <h4>filter by:</h4>\n      <mat-form-field>\n        <mat-select [(value)]=\"filterBy\">\n          <mat-option *ngFor=\"let opt of filterOptions\" value=\"opt\">{{opt}}</mat-option>\n        </mat-select>\n      </mat-form-field>\n    </div>\n  <mat-card\n    *ngFor=\"let tt of setFilterBy()\"\n  >\n    <mat-card-header>\n      <mat-card-title>{{tt.user.screen_name}}</mat-card-title>\n      <mat-card-subtitle>@{{tt.user.name}}</mat-card-subtitle>\n    </mat-card-header>\n    <img matCardImage [src]=\"tt.user.profile_image_url\">\n    <mat-card-content>\n      {{tt.text}}\n    </mat-card-content>\n    <mat-card-actions align=\"start\">\n      <!--<button mat-button (click)=\"onAction1\">Action1</button>-->\n    </mat-card-actions>\n    <mat-card-footer>\n\n    </mat-card-footer>\n  </mat-card>\n  </div>`,\n  styles: [`\n    .container {\n      width: 100%;\n      margin: 2rem;\n      display: flex;\n      flex-direction: column;\n    }\n  mat-card {\n    margin-bottom: 2rem;\n    flex: 1 1 100%;\n  }`],\n  encapsulation: ViewEncapsulation.Native\n})\nexport class TweetComponent implements OnInit, OnDestroy {\n  @Input() keyPublic: string;\n  @Input() keyPrivate: string;\n  @Input() hashTag: string;\n  @Output() errorMesage = new EventEmitter<string>();\n  private _unsubscribeAll: Subject<any>;\n  tweets: Tweets[];\n  filterBy = 'created_at';\n  filterOptions = ['created_at', 'text', 'user.name'];\n\n  constructor(private http: HttpClient) {\n    this._unsubscribeAll = new Subject();\n  }\n  ngOnInit(): void {\n    if (this.evalData()) {\n      this.getToken();\n    } else {\n      this.sendErrorMesage('No Public, Privite Key');\n      this.ngOnDestroy();\n    }\n  }\n  setFilterBy() {\n    return this.tweets.sort((a, b) => a[this.filterBy] > b[this.filterBy] ? 1 : a[this.filterBy] === b[this.filterBy] ? 0 : -1);\n  }\n  private evalData(): boolean {\n    return (\n      this.keyPublic && this.keyPrivate\n      && this.keyPublic !== null && this.keyPrivate !== null\n      && this.keyPublic !== '' && this.keyPrivate !== ''\n    );\n  }\n  private evaHash(): boolean {\n    return (\n      this.hashTag\n      && this.hashTag !== null\n      && this.hashTag !== ''\n    );\n  }\n  private evalToken (token: Token) {\n    return (\n      token.token_type && token.access_token\n      && token.token_type !== null && token.access_token !== null\n      && token.token_type !== '' && token.access_token !== ''\n    );\n  }\n  private getToken(): void {\n    if (this.evaHash()) {\n      const basic = btoa(`${this.keyPublic}:${this.keyPrivate}`);\n      const headers = new HttpHeaders({\n        'Content-Type'  : 'application/x-www-form-urlencoded',\n        'Authorization'   : `Basic  ${basic}`\n      });\n      const payload = new FormData();\n      payload.append('grant_type', 'client_credentials');\n      this.http.post<Token>(TweetConfig.tokenUrl, payload, { headers: headers})\n        .pipe(takeUntil(this._unsubscribeAll))\n        .subscribe(\n          token => this.getTweetList(token),\n          error => this.sendErrorMesage(error)\n        );\n    } else {\n      this.sendErrorMesage('No HasgTag set For Searching');\n    }\n  }\n  private getTweetList(token: Token) {\n    if (this.evalToken(token)) {\n      const headers = new HttpHeaders({\n        'Authorization'   : `${token.token_type} ${token.access_token}`\n      });\n      const params = new HttpParams().set('q', `%23${this.hashTag}`);\n      this.http.get<any>(TweetConfig.tweetSearchUrl, {headers: headers, params: params})\n        .pipe(takeUntil(this._unsubscribeAll))\n        .subscribe(\n          ({statuses}) => this.tweets = <Tweets[]>statuses,\n          error => this.sendErrorMesage(error)\n        );\n    } else {\n      this.sendErrorMesage('Error Tweeter Token');\n    }\n  }\n  private sendErrorMesage(men: string): void {\n    this.errorMesage.emit(men);\n  }\n  ngOnDestroy(): void {\n    this._unsubscribeAll.next();\n    this._unsubscribeAll.complete();\n  }\n}\n","import { NgModule } from '@angular/core';\nimport { TweetComponent } from './tweet.component';\nimport {BrowserModule} from '@angular/platform-browser';\nimport {HttpClientModule} from '@angular/common/http';\nimport {BrowserAnimationsModule} from '@angular/platform-browser/animations';\nimport {FormsModule} from '@angular/forms';\nimport {MatCardModule, MatSelectModule} from '@angular/material';\n\n@NgModule({\n  declarations: [TweetComponent],\n  imports: [\n    BrowserModule,\n    HttpClientModule,\n    BrowserAnimationsModule,\n    MatCardModule,\n    MatSelectModule,\n    FormsModule\n  ],\n  exports: [TweetComponent]\n})\nexport class TweetModule { }\n"],"names":[],"mappings":";;;;;;;;;;;;;AAAA,MAKa,YAAY;IAEvB,iBAAiB;;;YALlB,UAAU,SAAC;gBACV,UAAU,EAAE,MAAM;aACnB;;;;;;;;;;ACJD;MAKM,WAAW,GAAG;IAClB,QAAQ,EAAE,sCAAsC;IAChD,cAAc,EAAE,gDAAgD;CACjE;MA2DY,cAAc;;;;IAUzB,YAAoB,IAAgB;QAAhB,SAAI,GAAJ,IAAI,CAAY;QAN1B,gBAAW,GAAG,IAAI,YAAY,EAAU,CAAC;QAGnD,aAAQ,GAAG,YAAY,CAAC;QACxB,kBAAa,GAAG,CAAC,YAAY,EAAE,MAAM,EAAE,WAAW,CAAC,CAAC;QAGlD,IAAI,CAAC,eAAe,GAAG,IAAI,OAAO,EAAE,CAAC;KACtC;;;;IACD,QAAQ;QACN,IAAI,IAAI,CAAC,QAAQ,EAAE,EAAE;YACnB,IAAI,CAAC,QAAQ,EAAE,CAAC;SACjB;aAAM;YACL,IAAI,CAAC,eAAe,CAAC,wBAAwB,CAAC,CAAC;YAC/C,IAAI,CAAC,WAAW,EAAE,CAAC;SACpB;KACF;;;;IACD,WAAW;QACT,OAAO,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;KAC7H;;;;;IACO,QAAQ;QACd,QACE,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,UAAU;eAC9B,IAAI,CAAC,SAAS,KAAK,IAAI,IAAI,IAAI,CAAC,UAAU,KAAK,IAAI;eACnD,IAAI,CAAC,SAAS,KAAK,EAAE,IAAI,IAAI,CAAC,UAAU,KAAK,EAAE,EAClD;KACH;;;;;IACO,OAAO;QACb,QACE,IAAI,CAAC,OAAO;eACT,IAAI,CAAC,OAAO,KAAK,IAAI;eACrB,IAAI,CAAC,OAAO,KAAK,EAAE,EACtB;KACH;;;;;;IACO,SAAS,CAAE,KAAY;QAC7B,QACE,KAAK,CAAC,UAAU,IAAI,KAAK,CAAC,YAAY;eACnC,KAAK,CAAC,UAAU,KAAK,IAAI,IAAI,KAAK,CAAC,YAAY,KAAK,IAAI;eACxD,KAAK,CAAC,UAAU,KAAK,EAAE,IAAI,KAAK,CAAC,YAAY,KAAK,EAAE,EACvD;KACH;;;;;IACO,QAAQ;QACd,IAAI,IAAI,CAAC,OAAO,EAAE,EAAE;;kBACZ,KAAK,GAAG,IAAI,CAAC,GAAG,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;;kBACpD,OAAO,GAAG,IAAI,WAAW,CAAC;gBAC9B,cAAc,EAAI,mCAAmC;gBACrD,eAAe,EAAK,UAAU,KAAK,EAAE;aACtC,CAAC;;kBACI,OAAO,GAAG,IAAI,QAAQ,EAAE;YAC9B,OAAO,CAAC,MAAM,CAAC,YAAY,EAAE,oBAAoB,CAAC,CAAC;YACnD,IAAI,CAAC,IAAI,CAAC,IAAI,CAAQ,WAAW,CAAC,QAAQ,EAAE,OAAO,EAAE,EAAE,OAAO,EAAE,OAAO,EAAC,CAAC;iBACtE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;iBACrC,SAAS,CACR,KAAK,IAAI,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,EACjC,KAAK,IAAI,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CACrC,CAAC;SACL;aAAM;YACL,IAAI,CAAC,eAAe,CAAC,8BAA8B,CAAC,CAAC;SACtD;KACF;;;;;;IACO,YAAY,CAAC,KAAY;QAC/B,IAAI,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;;kBACnB,OAAO,GAAG,IAAI,WAAW,CAAC;gBAC9B,eAAe,EAAK,GAAG,KAAK,CAAC,UAAU,IAAI,KAAK,CAAC,YAAY,EAAE;aAChE,CAAC;;kBACI,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC,GAAG,CAAC,GAAG,EAAE,MAAM,IAAI,CAAC,OAAO,EAAE,CAAC;YAC9D,IAAI,CAAC,IAAI,CAAC,GAAG,CAAM,WAAW,CAAC,cAAc,EAAE,EAAC,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,EAAC,CAAC;iBAC/E,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;iBACrC,SAAS,CACR,CAAC,EAAC,QAAQ,EAAC,KAAK,IAAI,CAAC,MAAM,sBAAa,QAAQ,EAAA,EAChD,KAAK,IAAI,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CACrC,CAAC;SACL;aAAM;YACL,IAAI,CAAC,eAAe,CAAC,qBAAqB,CAAC,CAAC;SAC7C;KACF;;;;;;IACO,eAAe,CAAC,GAAW;QACjC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;KAC5B;;;;IACD,WAAW;QACT,IAAI,CAAC,eAAe,CAAC,IAAI,EAAE,CAAC;QAC5B,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,CAAC;KACjC;;;YAjIF,SAAS,SAAC;gBACT,QAAQ,EAAE,OAAO;gBACjB,QAAQ,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;;SA2BH;gBAYP,aAAa,EAAE,iBAAiB,CAAC,MAAM;yBAX9B;;;;;;;;;;IAUP;aAEH;;;;YAjEO,UAAU;;;wBAmEf,KAAK;yBACL,KAAK;sBACL,KAAK;0BACL,MAAM;;;;;;;ACvET,MAoBa,WAAW;;;YAZvB,QAAQ,SAAC;gBACR,YAAY,EAAE,CAAC,cAAc,CAAC;gBAC9B,OAAO,EAAE;oBACP,aAAa;oBACb,gBAAgB;oBAChB,uBAAuB;oBACvB,aAAa;oBACb,eAAe;oBACf,WAAW;iBACZ;gBACD,OAAO,EAAE,CAAC,cAAc,CAAC;aAC1B;;;;;;;;;;;;;;;"}