Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit f09c300

Browse files
manucorporatmhartington
authored andcommittedJan 6, 2017
fix(tabs): current tab still active if selected tab does not have a root
The current selected tab should NOT be deselected (i.e. detached from change detection) if the selected tab does not have a root ie. a tab that acts as a button to open a modal, logout etc. Lifecycle events should not be dispatched either. Right now we are dispatching willLeave/willEnter always (this is a bug). Closes #9392. Closes #9811. Closes #9392.
1 parent f7b9beb commit f09c300

File tree

4 files changed

+96
-38
lines changed

4 files changed

+96
-38
lines changed
 

‎src/components/tabs/tabs.ts

+50-35
Original file line numberDiff line numberDiff line change
@@ -374,54 +374,69 @@ export class Tabs extends Ion implements AfterViewInit {
374374
return;
375375
}
376376

377-
const deselectedTab = this.getSelected();
378-
if (selectedTab === deselectedTab) {
379-
// no change
377+
// If the selected tab is the current selected tab, we do not switch
378+
const currentTab = this.getSelected();
379+
if (selectedTab === currentTab) {
380380
return this._touchActive(selectedTab);
381381
}
382382

383-
let deselectedPage: ViewController;
384-
if (deselectedTab) {
385-
deselectedPage = deselectedTab.getActive();
386-
deselectedPage && deselectedPage._willLeave(false);
383+
// If the selected tab does not have a root, we do not switch (#9392)
384+
// it's possible the tab is only for opening modal's or signing out
385+
// and doesn't actually have content. In the case there's no content
386+
// for a tab then do nothing and leave the current view as is
387+
if (!selectedTab.root) {
388+
selectedTab.ionSelect.emit(selectedTab);
389+
this.ionChange.emit(selectedTab);
390+
return;
387391
}
388392

389-
opts.animate = false;
393+
// At this point we are going to perform a page switch
394+
// Let's fire willLeave in the current tab page
395+
let currentPage: ViewController;
396+
if (currentTab) {
397+
currentPage = currentTab.getActive();
398+
currentPage && currentPage._willLeave(false);
399+
}
390400

401+
// Fire willEnter in the new selected tab
391402
const selectedPage = selectedTab.getActive();
392403
selectedPage && selectedPage._willEnter();
393404

394-
selectedTab.load(opts, (alreadyLoaded: boolean) => {
395-
selectedTab.ionSelect.emit(selectedTab);
396-
this.ionChange.emit(selectedTab);
405+
// Let's start the transition
406+
opts.animate = false;
407+
selectedTab.load(opts, () => {
408+
if (opts.updateUrl !== false) {
409+
this._linker.navChange(DIRECTION_SWITCH);
410+
}
411+
this._tabSwitchEnd(selectedTab, selectedPage, currentPage);
412+
});
413+
}
397414

398-
if (selectedTab.root) {
399-
// only show the selectedTab if it has a root
400-
// it's possible the tab is only for opening modal's or signing out
401-
// and doesn't actually have content. In the case there's no content
402-
// for a tab then do nothing and leave the current view as is
403-
this._tabs.forEach(tab => {
404-
tab.setSelected(tab === selectedTab);
405-
});
406-
407-
if (this.tabsHighlight) {
408-
this._highlight.select(selectedTab);
409-
}
415+
_tabSwitchEnd(selectedTab: Tab, selectedPage: ViewController, currentPage: ViewController) {
416+
selectedTab.ionSelect.emit(selectedTab);
417+
this.ionChange.emit(selectedTab);
410418

411-
if (opts.updateUrl !== false) {
412-
this._linker.navChange(DIRECTION_SWITCH);
413-
}
414-
}
419+
// Update tabs selection state
420+
const tabs = this._tabs;
421+
let tab: Tab;
422+
for (var i = 0; i < tabs.length; i++) {
423+
tab = tabs[i];
424+
tab.setSelected(tab === selectedTab);
425+
}
426+
427+
if (this.tabsHighlight) {
428+
this._highlight.select(selectedTab);
429+
}
415430

416-
selectedPage && selectedPage._didEnter();
417-
deselectedPage && deselectedPage._didLeave();
431+
// Fire didEnter/didLeave lifecycle events
432+
selectedPage && selectedPage._didEnter();
433+
currentPage && currentPage._didLeave();
418434

419-
// track the order of which tabs have been selected, by their index
420-
// do not track if the tab index is the same as the previous
421-
if (this._selectHistory[this._selectHistory.length - 1] !== selectedTab.id) {
422-
this._selectHistory.push(selectedTab.id);
423-
}
424-
});
435+
// track the order of which tabs have been selected, by their index
436+
// do not track if the tab index is the same as the previous
437+
if (this._selectHistory[this._selectHistory.length - 1] !== selectedTab.id) {
438+
this._selectHistory.push(selectedTab.id);
439+
}
425440
}
426441

427442
/**

‎src/components/tabs/test/advanced/app-module.ts

+43-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import { Component, NgModule, ViewChild } from '@angular/core';
2-
import { /*DeepLink,*/ DeepLinkConfig, IonicApp, IonicModule, App, NavController, NavParams, ModalController, ViewController, Tabs, Tab } from '../../../..';
2+
import { /*DeepLink,*/AlertController, DeepLinkConfig, IonicApp, IonicModule, App, NavController, NavParams, ModalController, ViewController, Tabs, Tab } from '../../../..';
33

44

55
// @DeepLink({ name: 'sign-in' })
@@ -35,7 +35,12 @@ export class TabsPage {
3535

3636
@ViewChild(Tabs) tabs: Tabs;
3737

38-
constructor(public modalCtrl: ModalController, public params: NavParams) {}
38+
constructor(
39+
public navCtrl: NavController,
40+
public modalCtrl: ModalController,
41+
public params: NavParams,
42+
public alertCtrl: AlertController
43+
) { }
3944

4045
ngAfterViewInit() {
4146
this.tabs.ionChange.subscribe((tab: Tab) => {
@@ -51,6 +56,39 @@ export class TabsPage {
5156
console.log('onTabChange');
5257
}
5358

59+
logout() {
60+
this.navCtrl.pop().catch(() => {
61+
console.log('Cannot go back.');
62+
});
63+
}
64+
65+
ionViewCanLeave() {
66+
return new Promise((resolve, reject) => {
67+
let alert = this.alertCtrl.create({
68+
title: 'Log out',
69+
subTitle: 'Are you sure you want to log out?',
70+
buttons: [
71+
{
72+
text: 'No',
73+
role: 'cancel',
74+
handler: () => {
75+
reject();
76+
}
77+
},
78+
{
79+
text: 'Yes',
80+
handler: () => {
81+
alert.dismiss().then(() => {
82+
resolve();
83+
});
84+
}
85+
}
86+
]
87+
});
88+
alert.present();
89+
});
90+
}
91+
5492
chat() {
5593
console.log('Chat clicked!');
5694
this.modalCtrl.create(ChatPage).present();
@@ -104,7 +142,9 @@ export class Tab1Page1 {
104142
}
105143

106144
logout() {
107-
this.app.getRootNav().setRoot(SignIn, null, { animate: true, direction: 'back' });
145+
this.app.getRootNav().setRoot(SignIn, null, { animate: true, direction: 'back' }).catch(() => {
146+
console.debug('logout cancelled');
147+
});
108148
}
109149

110150
ionViewWillEnter() {

‎src/components/tabs/test/advanced/tab1page1.html

+2
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,8 @@
1313
<p><button ion-button (click)="logout()">Logout</button></p>
1414
<p><button ion-button (click)="favoritesTab()">Favorites Tab</button></p>
1515
<p><button ion-button (click)="goBack()">Go Back</button></p>
16+
<p><button ion-button (click)="color = !color" [color]="color ? 'primary':'secondary'">Change color</button></p>
17+
<p>"Change color" should continue working after clicking Logout in the tabbar and cancelling (No in the alert)</p>
1618
<p>UserId: {{userId}}</p>
1719
<div f></div><div f></div><div f></div><div f></div><div f></div><div f></div><div f></div><div f></div><div f></div><div f></div>
1820
<div f></div><div f></div><div f></div><div f></div><div f></div><div f></div><div f></div><div f></div><div f></div><div f></div>

‎src/components/tabs/test/advanced/tabs.html

+1
Original file line numberDiff line numberDiff line change
@@ -4,4 +4,5 @@
44
<ion-tab tabTitle="Favorites" tabIcon="star" root="Tab2Page1" tabsHideOnSubPages="false"></ion-tab>
55
<ion-tab tabTitle="Settings" tabIcon="settings" root="Tab3Page1"></ion-tab>
66
<ion-tab tabTitle="Chat" tabIcon="chatbubbles" (ionSelect)="chat()" [show]="showTab"></ion-tab>
7+
<ion-tab tabTitle="Logout" tabIcon="exit" (ionSelect)="logout()"></ion-tab>
78
</ion-tabs>

0 commit comments

Comments
 (0)
Please sign in to comment.