tl;dr Revising the concept of changing app colors at runtime in an Ionic app
Backstory
A while back, I wrote a blog post about how to customize the colors of Ionic components using a custom directive. The method used in that blog posts works but it’s highly fragile since it’s completely dependent on the HTML elements of the components and not the CSS classes. With the help of Simon’s blog post about dynamic theming at Devdactic and David Walsh’s blog post about dynamic CSS I was able to achieve the same results with much less fuss.
Problem
In our app, the users can choose what color they want to use for the menu. We use that primary color in a bunch of different locations. It helps give the user a level of ownership if they can choose how things look. In the mobile app for the product, built with Ionic, I wanted to recreate that feature but wasn’t sure how to change CSS colors at runtime. I think I’ve figured out how to do so by injecting new rules into our stylesheets based on the results of the network request to our servers to get the users’ custom color.
Solution
Start with a blank Ionic app: ionic start DynamicColorsApp blank
.
Open up app.component.ts
found in DynamicColorApp/src/app/
.
Once there, let’s add a property to the class to hold the custom background color we’re going to give the app:
@Component({
templateUrl: 'app.html'
})
export class MyApp {
rootPage:any = HomePage;
public customBgColor:any;
Now at this point, we should be ready to make an AJAX request back to the server to fetch the users actual background color, but that’s outside the scope of this post. For now, let’s just assume we have the color string. Let’s store that in a property we just created and then call a function to add the necessary CSS rules to our stylesheet
constructor() {
// Some async request that will fetch data that will include the custom color goes here
this.customBgColor = "#F0F";
this.createCSSRules();
Our createCSSRules()
method looks like this:
createCSSRules(){
this.addCSSRule(document.styleSheets[0], ".header .toolbar-background, .button", `background-color: ${this.customBgColor} !important;`, 1);
}
This method does 2 things:
- targets the
.header.toolbar-background
and.button
elements and sets their background-color property to our custom color. - Sends all of that information to another method,
addCSSRule()
to inject the new rule into the stylesheet. Which looks like this:
addCSSRule(sheet, selector, rules, index) {
if("insertRule" in sheet) {
sheet.insertRule(selector + "{" + rules + "}", index);
}else if("addRule" in sheet) {
sheet.addRule(selector, rules, index);
}
}
Now I shamefully stole this code from David Walsh’s blog. Check out the link at the top of the post for more information.
What this does is create a new CSS rule, targeting our selector (in our case, the toolbar and button backgrounds) and using the color we’ve loaded from a remote server. It then inserts that rule into the stylesheet of our app. The addCssRule()
checks to see if the insertRule
method is available, and if not, uses addRule
instead, so we get some cross-browser compatibility to boot!
End result should look something like this:
I also did some additional work on making sure the font color would be visible on any color of background. I used the npm package contrast (npm install contrast --save
) and used it (import * as contrast from 'contrast';
) to test whether the this.customBgColor
was light or dark and then change the font color to suit. You can see my full test file here.
This all works in the browser, I have not deployed this code to devices yet so your milage may vary.