2017年11月26日 星期日

Ionic3+ firebase

FireBase推出最新的雲端資料庫的版本(cloud Firestore), 有別之前的RealTime的資料庫,更引進rxjsoberserablesubscription的模式,讓程式更具有高彈性,我們就簡單的實作範例。

一、建立Colud Firestore
選擇下拉式選單è Cloud Firestore



二、引用firestore

安裝angularfire2  
npm install angularfire2 –save 

import { AngularFireModule } from 'angularfire2';
import {AngularFirestoreModule} from 'angularfire2/firestore';

新版的colud firestore的設定檔需要加上projectId
// AF2 Settings
export const firebaseConfig = {
  apiKey: "xxxxxxxxxxx",
  authDomain: "fiyy4.firebaseapp.com",
  databaseURL: "https://firstapp-xxxx.firebaseio.com",
  projectId: "fyyyy4",
  storageBucket: "firstyy.appspot.com",
  messagingSenderId: "1233456789"
};

注入AngularFireModuleAngularFirestoreModule
imports: [
    BrowserModule,
    IonicModule.forRoot(MyApp),
    AngularFireModule.initializeApp(firebaseConfig),
    AngularFirestoreModule
  ],

三、ProvideràMoney_service.ts建立CRUD的函式

AngularFirestoreCollection 用來取得資料列表(查詢)
AngularFirestoreDocument 用來允許編輯/刪除資料
Observable 用來取得資料陣列有更新的值
import { Observable } from 'rxjs/Observable';
import { AngularFirestore, AngularFirestoreCollection, AngularFirestoreDocument } from 'angularfire2/firestore';

宣告
valueChange: 傳回Json的物件,不具有metadata的屬性
export class MoneyServiceProvider {
  itemCollections: AngularFirestoreCollection<any>; //資料庫傳回值型態
  items: Observable<any[]>;
  constructor(public afs: AngularFirestore) {
    this.itemCollections = this.afs.collection('moneyCost', ref => ref.orderBy('costDate')); //取回資料庫moneyCost,並依日期排序
    this.items = this.itemCollections.valueChanges();//取回值
  }

新增
snapshotChanges:可以用來異動資料庫,本身具有metadata的屬性
  public add(item) {
    this.itemCollections.add(item);
    const listObservable = this.itemCollections.snapshotChanges()
  }

刪除
AngularFirestoreDocument:取回Firestore Collection 的物件
public remove(itemID) {
    const itemDoc: AngularFirestoreDocument<any> = this.afs.doc<any>('moneyCost/' + itemID);
    itemDoc.delete();
  }

List:取得id
  public list() {
    return this.itemCollections.snapshotChanges().map(actions => {
      return actions.map(a => {
        const data = a.payload.doc.data() as any;
        data.id = a.payload.doc.id;
        return data;
      });
    });
  }

整理陣列的資料 data[key]=[{},{}]
  public groupList(items:any[]){
    let total=0;
    const groupItem=[];
    items.forEach(eachObj => {
      if(groupItem[eachObj.costDate] == undefined){
        groupItem[eachObj.costDate]=[];
      }
total=+eachObj.cost;
groupItem[eachObj.costDate].push({key:eachObj.costDate,
value:eachObj,sum:total});
    });
    return groupItem;
  }

Reduce:累加器,會自動依序陣陣值取出作運算後,傳回最終結果
public getCostSum(items):Observable<number> {
    // return 1000;
    return items.map((items: any[]) => {
      return items.reduce((prev, curr: any) => {            
              return prev + Number(curr.cost);               
      }, 0);
  })}

四、建立Componentàmoney-list.ts
import { MoneyServiceProvider } from '../../providers/money-service/money-service';
import { AngularFirestore, AngularFirestoreCollection } from 'angularfire2/firestore';

宣告物件
interface money {
  productId: string;
  costDate: string;
  cost: number;
}

export class MoneyListComponent implements OnDestroy {
  private subscription: ISubscription;
  itemCollections: AngularFirestoreCollection<money>;
  items:any[];
  obserItem:Observable<any[]>;
  groupList: any[];
  sumCost: Number;
  constructor(public afs: AngularFirestore, public moneyService: MoneyServiceProvider) {
    this.obserItem= moneyService.list(); //取回資料
    this.obserItem.subscribe(x=>{
//將取回來的值放回到變數中,透過item 在畫面上呈現
      this.items=x;
      this.groupList =moneyService.groupList(this.items);
      this.doSum (this.items); //計算總合
    });
  }
}

//累加器,會自動傳回最後值的運算結果
doSum(items) {
    this.items.reduce((prev, curr: any) =>{
         return this.sumCost = prev + Number(curr.cost);
    } ,0);
  }

  doDelete(itemId: any) {
    this.moneyService.remove(itemId); //刪除document
  }

  ngOnDestroy() {
//解除訂閱
    this.obserItem.subscribe().unsubscribe();
  }
}

Component: money-list.html

建立group pipe 呈現複雜資料結構 data[key]=[{},{}],轉換資料結構 [{key:1,value=data},{key:2,value=data}]
@Pipe({
  name: 'group',
})
export class GroupByPipe implements PipeTransform {
  transform(value: any, args: any[] = null): any {
    let keys = [];
    for (let key in value) {
      keys.push({ key: key, value: value[key] });
    }
    return keys;
  }
}

加上關鍵字group 就會將資料置換{key,value}
<ion-list>
        <ion-item-group *ngFor="let item of groupList|group">
                <ion-item-divider color="light">
                        <h3>
                                <font color=black>{{item.key|date}}</font>
                        </h3>
                </ion-item-divider>
                <ion-item *ngFor="let subItem of item.value">
                        <ion-row>
                                <ion-col col-lg-4>
                                   <font color=blue>{{subItem.value.productId}}</font>
                                </ion-col>
                                <ion-col col-lg-4>
                                <font>{{subItem.value.cost|currency}} </font>
                                </ion-col>
                                <ion-col col-lg-4>
                                        <button ion-button color="danger" small  round (click)="doDelete(subItem.value.id)">刪除</button>
                                </ion-col>
                        </ion-row>
                </ion-item>
        </ion-item-group>
       <h1><font color=red>總額<input type="hidden" [(ngModel)]="sumCost"> {{sumCost}}</font></h1>
     </ion-list>

範例如下
新增

 

列表

以上總結 Colud Firestore 可以讓資料傳回時,可先轉換資料型態/結構/運算值等等,更具有彈性。