How to set a theme to whole application in code, but not in the Manifest?

前端 未结 6 1501
萌比男神i
萌比男神i 2020-12-16 17:11

I know how to set theme to whole application in manifest,but how to set theme to whole application programmatically ? I am trying this: getApplicationContext.se

相关标签:
6条回答
  • 2020-12-16 17:38

    You can create BaseActivity. Change your style in this Activity. All your activity inherit from this Activity.

    0 讨论(0)
  • 2020-12-16 17:50

    I use a custom activity for all activities in my application.

    Then I check a preference value in the onCreate of my inherited activity e.g.

    public class MyCustomActivity extends Activity {
    
    protected void onCreate(Bundle savedInstanceState) {
            SharedPreferences prefs = PreferenceManager
                    .getDefaultSharedPreferences(this);
            if(prefs.getBoolean("use_light_theme",false)==true)
            {
                setTheme(R.style.AppThemeLight);
            }
            super.onCreate(savedInstanceState);
    
    }
    

    //styles.xml

        <!-- Base application theme. -->
        <style name="AppTheme" parent="android:Theme.Holo">
            <!-- Customize your theme here. -->
        </style>
        <style name="AppThemeLight" parent="android:Theme.Light">
            <!-- Customize your theme here. -->
        </style>
    
    0 讨论(0)
  • 2020-12-16 17:53

    Call setTheme before super.onCreate() similar to below code

      public void onCreate(Bundle icicle) {
               if(Utility.isThemed)
                 setTheme(R.style.Mytheme);
               super.onCreate(icicle);
        .....
        .....
    }
    
    0 讨论(0)
  • 2020-12-16 17:54

    In setTheme, the documentation says:

    Note that this should be called before any views are instantiated in the Context (for example before calling setContentView(View) or inflate(int, ViewGroup)).

    Have you taken care of that?

    0 讨论(0)
  • 2020-12-16 17:59

    You can't apply it to a whole application from Java, unless you use Java code to modify the manifest file. The other activities you want to apply the theme to might not even be running so how would Android apply a theme to them? And after exiting and returning to your program all the changes would be lost and you'd need to apply the theme again.

    getApplicationContext does return the application context - but just because a method takes a Context doesn't mean that passing it an ApplicationContext will suddenly make it affect the whole application. In fact in general it won't, and will just work as if you used a normal Context.

    Rather the importance of the different contexts is that they are around for different amounts of time - an activity context is created and destroyed with the activity, but an Application context is created when the first application component runs and destroyed when the last component is destroyed.

    0 讨论(0)
  • 2020-12-16 18:00

    I spent a lot of time getting this to work and now my app has selectable Light and Dark themes which can even be selected dynamically coming immediately into play including Prefs. Below is the resource I use with some other ideas commented out which I played with at some starge.

    <?xml version="1.0" encoding="utf-8"?>
    <resources>
        <!--The base theme ensures nothing is shown until the first activity. All activities and fragmenst must-->
        <!--set a them in onCreate or in AndroidManifext or they wil crash because of no title.-->
        <!--Using Theme.Holo.NoActionBar suppresses the ActionBar initially so Icon doesn't show until new theme is set.-->
        <style name="MyAppThemeInitial" parent="@android:style/Theme.Holo">
            <!--<item name="android:windowBackground">@color/initial_background_grey</item>-->
            <!--<item name="android:actionBarStyle">@style/MyActionBar</item>-->
            <item name="android:windowDisablePreview">true</item>
    
        </style>
        <style name="MyAppThemeDark" parent="@android:style/Theme.Holo">
            <item name="android:windowBackground">@color/black</item>
        </style>
        <style name="MyAppThemeLight" parent="android:Theme.Holo.Light">
        <item name="android:windowBackground">@color/white</item>
        </style>
    
    
    <!--&lt;!&ndash;This is so that only the icon is showing in the intial theme&ndash;&gt;-->
        <!--&lt;!&ndash; ActionBar styles &ndash;&gt;-->
        <!--<style name="MyActionBar" parent="@android:style/Widget.Holo.ActionBar">-->
            <!--<item name="android:background">@android:color/transparent</item>-->
            <!--<item name="android:titleTextStyle">@style/MyActionBarTextAppearance</item>-->
        <!--</style>-->
    
        <!--<style name="MyActionBarTextAppearance">-->
            <!--<item name="android:textColor">@android:color/transparent</item>-->
        <!--</style>-->
    
    </resources> 
    

    The key item here is

    <item name="android:windowDisablePreview">true</item>
    

    and that took me a while to realize. My app does some heavy lifting while starting so it was important to not have the mandatory Manifest theme showing before the one I set in onCreate.

    To be able to restart the app dynamically I check if this was how the app was started/restarted like this.

    SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
    boolean useThemeLight = sp.getBoolean("useThemeLight", false);
    //Since this Activity can also be started by the Theme Toggle in the Action bar we need to see if
    //there is a TOGGLE_THEME extra which only it uses
    Intent curIntent = this.getIntent();
    if (curIntent.getExtras() != null && curIntent.getExtras().containsKey(TOGGLE_THEME)) {
        if(curIntent.getStringExtra(TOGGLE_THEME).equals("Dark")){
            this.setTheme(R.style.MyAppThemeDark);
            CurrentTheme = "Dark";
        }else{
            this.setTheme(R.style.MyAppThemeLight);
            CurrentTheme = "Light";
        }
        activityThemeToggleActive = true;
    } else {
        if (useThemeLight) {
            this.setTheme(R.style.MyAppThemeLight);
            CurrentTheme = "Light";
        } else {
            this.setTheme(R.style.MyAppThemeDark);
            CurrentTheme = "Dark";
        }
    }
    

    In Preferences I do this.

    SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
    boolean useThemeLight = sp.getBoolean("useThemeLight", false);
    if (useThemeLight) {
        this.setTheme(R.style.MyAppThemeLight);
    } else {
        this.setTheme(R.style.MyAppThemeDark);
    }
    

    Hope this gets you started. Regards, John.

    0 讨论(0)
提交回复
热议问题