Also, if the user switches the theme (e.g. from default to dark), we want to make sure that the styling of our SPFx web part also changes without the user having to reload the tab.
Let's see how we can do this. In this post, depending on the Teams theme, we will dynamically add a CSS class to our top level react component. And when the theme changes, we will change this class as well. That will be the scope of this post without going into further details of Styling/CSS.
Also, since I have been exploring React hooks recently, we will be using them in the demo code.
The very first thing we need to do is to make sure that our SPFx tab loads with the currently selected theme when the tab loads for the first time.. This can be achieved with the teamsJS sdk bundled with SPFx:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
this.context.sdks.microsoftTeams.context.theme |
Now the important part. When the user updates the theme, we want to fire an event which can be used to set the new theme in our tab. Luckily the teamsJS sdk provides us with a function which can be used to register this event handler:
https://docs.microsoft.com/en-us/microsoftteams/platform/tabs/how-to/access-teams-context#theme-change-handling
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
const [themeState, setThemeState] = useState<string>(props.teamsTheme || "default"); | |
//Run effect once when the component loads. We will use this effect to register our Theme Changed handler. | |
useEffect(() => { | |
console.log("registerOnThemeChangeHandler useEffect fired"); | |
props.context.sdks.microsoftTeams.teamsJs.registerOnThemeChangeHandler((theme: string) => { | |
console.log(`theme changed to: ${theme}`); | |
//Update the themeState with the new theme. | |
setThemeState(theme); | |
}); | |
}, []); |
This hook takes in the themeState as a dependency which means that it will run anytime themeState changes. We will use it to update the styleState.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
const [styleState, setStyleState] = useState<string>(styles.containerdefault); | |
//Run anytime the themeState changes. | |
useEffect(() => { | |
console.log("themeState useEffect fired"); | |
switch (themeState) { | |
case "dark": | |
setStyleState(styles.containerdark); | |
break; | |
default: | |
setStyleState(styles.containerdefault); | |
} | |
}, [themeState]); |
Here is the complete code for the React Functional component:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import * as React from 'react'; | |
import styles from './HelloTeams.module.scss'; | |
import { IHelloTeamsProps } from './IHelloTeamsProps'; | |
import { useEffect, useState } from 'react'; | |
const HelloTeams: React.FunctionComponent<IHelloTeamsProps> = (props: IHelloTeamsProps) => { | |
const [themeState, setThemeState] = useState<string>(props.teamsTheme || "default"); | |
const [styleState, setStyleState] = useState<string>(styles.containerdefault); | |
//Run effect once when the component loads. We will use this effect to register our Theme Changed handler. | |
useEffect(() => { | |
console.log("registerOnThemeChangeHandler useEffect fired"); | |
props.context.sdks.microsoftTeams.teamsJs.registerOnThemeChangeHandler((theme: string) => { | |
console.log(`theme changed to: ${theme}`); | |
//Update the themeState with the new theme. | |
setThemeState(theme); | |
}); | |
}, []); | |
//Run anytime the themeState changes. | |
useEffect(() => { | |
console.log("themeState useEffect fired"); | |
switch (themeState) { | |
case "dark": | |
setStyleState(styles.containerdark); | |
break; | |
default: | |
setStyleState(styles.containerdefault); | |
} | |
}, [themeState]); | |
return ( | |
<div className={styles.helloTeams}> | |
<div className={`${styles.container} ${styleState}`}> | |
<div className={styles.row}> | |
<div className={styles.column}> | |
<span className={styles.title}>Welcome to Teams in {themeState} mode!</span> | |
</div> | |
</div> | |
</div> | |
</div> | |
); | |
}; | |
export default HelloTeams; |