AngularFire Observables / Subscribe - 未在同一范围内设置 Class 变量

AngularFire Observables / Subscribe - Not Setting Class Variable in Same Scope

我正在使用 Firebase 和 AngularFire2 构建一个 Ionic 应用程序。在 Angularfire2 中,您可以进行身份​​验证,但它是可观察的,必须订阅才能获取用户对象。问题是我似乎无法设置 this.user = user 因为 .subscribe 似乎不在同一范围内。

举个例子,我有一个auth.ts和一个app.component.ts。在我的服务中,我进行身份验证,

auth.ts

export class AuthService {

// user: Observable<firebase.User>;
user: object = {
    displayName: null,
    email: null,
    emailVerified: null,
    photoUrl: null,
    uid: null
};

constructor(private facebook: Facebook,
            private googlePlus: GooglePlus,
            private platform: Platform,
            private afAuth: AngularFireAuth) {

    // this.user = afAuth.authState;
    afAuth.authState.subscribe((user: firebase.User) => {

        if (!user) {
            this.user = null;
            return;
        }
        this.user =  {
            displayName: user.displayName,
            email: user.email,
            emailVerified: user.emailVerified,
            photoUrl: user.photoURL,
            uid: user.uid
        }
    });

}

在我的 app.component.ts 中:

app.component.ts

export class MytApp {

rootPage: any = TabsPage;
signinPage = SigninPage;
@ViewChild('nav') nav: NavController;

userProfile: any = null;

constructor(private platform: Platform, 
            private menuCtrl: MenuController,
            private statusBar: StatusBar, 
            private splashScreen: SplashScreen,
            private authService: AuthService) {

    // this.authService.user.subscribe((user: firebase.User) => {
        // var user = authService.user;
        // if (user) {
        //  this.userProfile = user;
        //  this.onLoad(TabsPage);
        // } else {
        //  this.userProfile = null;
        //  this.onLoad(SigninPage);
        // }
    // });

    platform.ready().then(() => {
        // Okay, so the platform is ready and our plugins are available.
        // Here you can do any higher level native things you might need.
        statusBar.styleDefault();
        splashScreen.hide();

        var user = authService.user; /* HERE */
        console.log(user);
        if (user) {
            this.userProfile = user;
            this.onLoad(TabsPage);
        } else {
            this.userProfile = null;
            this.onLoad(SigninPage);
        }
    });
}

}

如您所见,在 var user = authService.user; 我 运行 console.log(user); 之后,我得到的是:

console.log(用户);

   displayName: null,
   email: null,
   emailVerified: null,
   photoUrl: null,
   uid: null

我在 Firebase 中使用用户对象来定位用户数据,所以这是必要的。问题是用户对象没有更新,就像它在 afAuth.authState.subscribe((user: firebase.User) => {} 的 "scope" 之外。

知道我怎么不能订阅,只是简单地获取用户对象,或者从订阅中提取数据以便我可以轻松地从其他组件检索它(因为它是一项服务)吗?请帮忙!

您不能在您的服务中执行 subscribe()。您想要做的是 return 和 Observable,让组件自己处理 Observables,而不是在服务中。这是因为您的 API 调用是异步的,您的组件无法等待 http 调用的解析。

你可以做的是 return 通过将你的 AngularFireAuth 映射到你想要的对象来实现 Observable。

export class AuthService {
    public user: Observable<any>; //this can be any type you wish
    constructor(private facebook: Facebook,
                private googlePlus: GooglePlus,
                private platform: Platform,
                private afAuth: AngularFireAuth) {

        this.user =
            afAuth.authState.map((user: firebase.User) => {
                if (!user) {
                    return null;
                }
                else return {
                    displayName: user.displayName,
                    email: user.email,
                    emailVerified: user.emailVerified,
                    photoUrl: user.photoURL,
                    uid: user.uid
                }
            });
    }
}

现在,您可以在任何其他组件中订阅您的 AuthService.user

export class MytApp {

    rootPage: any = TabsPage;
    signinPage = SigninPage;
    @ViewChild('nav') nav: NavController;

    userProfile: any = null;

    constructor(private platform: Platform,
                private menuCtrl: MenuController,
                private statusBar: StatusBar,
                private splashScreen: SplashScreen,
                private authService: AuthService) {


        platform.ready().then(() => {
            // Okay, so the platform is ready and our plugins are available.
            // Here you can do any higher level native things you might need.
            statusBar.styleDefault();
            splashScreen.hide();
            //subscribe to AuthService
            authService.subscribe(user => {
                console.log(user);
                if (user) {
                    this.userProfile = user;
                    this.onLoad(TabsPage);
                } else {
                    this.userProfile = null;
                    this.onLoad(SigninPage);
                }
            })
        });
    }
}