]> git.ktnx.net Git - mobile-ledger.git/blob - app/src/main/java/net/ktnx/mobileledger/utils/Colors.java
themed color for disabled text
[mobile-ledger.git] / app / src / main / java / net / ktnx / mobileledger / utils / Colors.java
1 /*
2  * Copyright © 2019 Damyan Ivanov.
3  * This file is part of MoLe.
4  * MoLe is free software: you can distribute it and/or modify it
5  * under the term of the GNU General Public License as published by
6  * the Free Software Foundation, either version 3 of the License, or
7  * (at your opinion), any later version.
8  *
9  * MoLe is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12  * GNU General Public License terms for details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with MoLe. If not, see <https://www.gnu.org/licenses/>.
16  */
17
18 package net.ktnx.mobileledger.utils;
19
20 import android.app.Activity;
21 import android.content.res.ColorStateList;
22 import android.content.res.Resources;
23 import android.util.TypedValue;
24
25 import androidx.annotation.ColorInt;
26 import androidx.annotation.ColorLong;
27 import androidx.annotation.NonNull;
28 import androidx.lifecycle.MutableLiveData;
29
30 import net.ktnx.mobileledger.BuildConfig;
31 import net.ktnx.mobileledger.R;
32 import net.ktnx.mobileledger.model.Data;
33 import net.ktnx.mobileledger.model.MobileLedgerProfile;
34 import net.ktnx.mobileledger.ui.HueRing;
35
36 import java.util.ArrayList;
37 import java.util.Collections;
38 import java.util.Locale;
39
40 import static java.lang.Math.abs;
41 import static net.ktnx.mobileledger.utils.Logger.debug;
42
43 public class Colors {
44     public static final int DEFAULT_HUE_DEG = 261;
45     public static final int THEME_HUE_STEP_DEG = 5;
46     public static final int baseHueStep = 60;
47     private static final float blueLightness = 0.665f;
48     private static final float yellowLightness = 0.350f;
49     private static final int[][] EMPTY_STATES = new int[][]{new int[0]};
50     public static @ColorInt
51     int accent;
52     @ColorInt
53     public static int tableRowLightBG;
54     @ColorInt
55     public static int tableRowDarkBG;
56     @ColorInt
57     public static int primary, defaultTextColor, defaultTextColorDisabled;
58     public static int profileThemeId = -1;
59     public static MutableLiveData<Integer> themeWatch = new MutableLiveData<>(0);
60     private static int[] themeIDs =
61             {R.style.AppTheme_NoActionBar_000, R.style.AppTheme_NoActionBar_005,
62              R.style.AppTheme_NoActionBar_010, R.style.AppTheme_NoActionBar_015,
63              R.style.AppTheme_NoActionBar_020, R.style.AppTheme_NoActionBar_025,
64              R.style.AppTheme_NoActionBar_030, R.style.AppTheme_NoActionBar_035,
65              R.style.AppTheme_NoActionBar_040, R.style.AppTheme_NoActionBar_045,
66              R.style.AppTheme_NoActionBar_050, R.style.AppTheme_NoActionBar_055,
67              R.style.AppTheme_NoActionBar_060, R.style.AppTheme_NoActionBar_065,
68              R.style.AppTheme_NoActionBar_070, R.style.AppTheme_NoActionBar_075,
69              R.style.AppTheme_NoActionBar_080, R.style.AppTheme_NoActionBar_085,
70              R.style.AppTheme_NoActionBar_090, R.style.AppTheme_NoActionBar_095,
71              R.style.AppTheme_NoActionBar_100, R.style.AppTheme_NoActionBar_105,
72              R.style.AppTheme_NoActionBar_110, R.style.AppTheme_NoActionBar_115,
73              R.style.AppTheme_NoActionBar_120, R.style.AppTheme_NoActionBar_125,
74              R.style.AppTheme_NoActionBar_130, R.style.AppTheme_NoActionBar_135,
75              R.style.AppTheme_NoActionBar_140, R.style.AppTheme_NoActionBar_145,
76              R.style.AppTheme_NoActionBar_150, R.style.AppTheme_NoActionBar_155,
77              R.style.AppTheme_NoActionBar_160, R.style.AppTheme_NoActionBar_165,
78              R.style.AppTheme_NoActionBar_170, R.style.AppTheme_NoActionBar_175,
79              R.style.AppTheme_NoActionBar_180, R.style.AppTheme_NoActionBar_185,
80              R.style.AppTheme_NoActionBar_190, R.style.AppTheme_NoActionBar_195,
81              R.style.AppTheme_NoActionBar_200, R.style.AppTheme_NoActionBar_205,
82              R.style.AppTheme_NoActionBar_210, R.style.AppTheme_NoActionBar_215,
83              R.style.AppTheme_NoActionBar_220, R.style.AppTheme_NoActionBar_225,
84              R.style.AppTheme_NoActionBar_230, R.style.AppTheme_NoActionBar_235,
85              R.style.AppTheme_NoActionBar_240, R.style.AppTheme_NoActionBar_245,
86              R.style.AppTheme_NoActionBar_250, R.style.AppTheme_NoActionBar_255,
87              R.style.AppTheme_NoActionBar_260, R.style.AppTheme_NoActionBar_265,
88              R.style.AppTheme_NoActionBar_270, R.style.AppTheme_NoActionBar_275,
89              R.style.AppTheme_NoActionBar_280, R.style.AppTheme_NoActionBar_285,
90              R.style.AppTheme_NoActionBar_290, R.style.AppTheme_NoActionBar_295,
91              R.style.AppTheme_NoActionBar_300, R.style.AppTheme_NoActionBar_305,
92              R.style.AppTheme_NoActionBar_310, R.style.AppTheme_NoActionBar_315,
93              R.style.AppTheme_NoActionBar_320, R.style.AppTheme_NoActionBar_325,
94              R.style.AppTheme_NoActionBar_330, R.style.AppTheme_NoActionBar_335,
95              R.style.AppTheme_NoActionBar_340, R.style.AppTheme_NoActionBar_345,
96              R.style.AppTheme_NoActionBar_350, R.style.AppTheme_NoActionBar_355,
97              };
98     public static void refreshColors(Resources.Theme theme) {
99         TypedValue tv = new TypedValue();
100         theme.resolveAttribute(R.attr.table_row_dark_bg, tv, true);
101         tableRowDarkBG = tv.data;
102         theme.resolveAttribute(R.attr.table_row_light_bg, tv, true);
103         tableRowLightBG = tv.data;
104         theme.resolveAttribute(R.attr.colorPrimary, tv, true);
105         primary = tv.data;
106         theme.resolveAttribute(R.attr.textColor, tv, true);
107         defaultTextColor = tv.data;
108         defaultTextColorDisabled = 0x7f000000 | 0x00ffffff & defaultTextColor;
109         theme.resolveAttribute(R.attr.colorAccent, tv, true);
110         accent = tv.data;
111
112         // trigger theme observers
113         themeWatch.postValue(themeWatch.getValue() + 1);
114     }
115     public static @ColorInt
116     int hslColor(float hueRatio, float saturation, float lightness) {
117         return 0xff000000 | hslTriplet(hueRatio, saturation, lightness);
118     }
119     public static @ColorInt
120     int hslTriplet(float hueRatio, float saturation, float lightness) {
121         @ColorLong long result;
122         float h = hueRatio * 6;
123         float c = (1 - abs(2f * lightness - 1)) * saturation;
124         float h_mod_2 = h % 2;
125         float x = c * (1 - Math.abs(h_mod_2 - 1));
126         int r, g, b;
127         float m = lightness - c / 2f;
128
129         if (h < 1 || h == 6)
130             return tupleToColor(c + m, x + m, 0 + m);
131         if (h < 2)
132             return tupleToColor(x + m, c + m, 0 + m);
133         if (h < 3)
134             return tupleToColor(0 + m, c + m, x + m);
135         if (h < 4)
136             return tupleToColor(0 + m, x + m, c + m);
137         if (h < 5)
138             return tupleToColor(x + m, 0 + m, c + m);
139         if (h < 6)
140             return tupleToColor(c + m, 0 + m, x + m);
141
142         throw new IllegalArgumentException(String.format(
143                 "Unexpected value for h (%1.3f) while converting hsl(%1.3f, %1.3f, %1.3f) to rgb",
144                 h, hueRatio, saturation, lightness));
145     }
146     public static @ColorInt
147     int tupleToColor(float r, float g, float b) {
148         int r_int = Math.round(255 * r);
149         int g_int = Math.round(255 * g);
150         int b_int = Math.round(255 * b);
151         return (r_int << 16) | (g_int << 8) | b_int;
152     }
153     public static float baseHueLightness(int baseHueDegrees) {
154         switch (baseHueDegrees % 360) {
155             case 0:
156                 return 0.450f;   // red
157             case 60:
158                 return 0.400f;  // yellow
159             case 120:
160                 return 0.400f;  // green
161             case 180:
162                 return 0.400f;  // cyan
163             case 240:
164                 return 0.750f;  // blue
165             case 300:
166                 return 0.500f;   // magenta
167             default:
168                 throw new IllegalStateException(
169                         String.format(Locale.US, "baseHueLightness called with invalid value %d",
170                                 baseHueDegrees));
171         }
172     }
173     public static float hueLightness(int hueDegrees) {
174         int mod = hueDegrees % baseHueStep;
175         int x0 = hueDegrees - mod;
176         int x1 = x0 + baseHueStep;
177
178         float y0 = baseHueLightness(x0);
179         float y1 = baseHueLightness(x1);
180
181         return y0 + (hueDegrees - x0) * (y1 - y0) / (x1 - x0);
182     }
183     public static @ColorInt
184     int getPrimaryColorForHue(int hueDegrees) {
185         int result = hslColor(hueDegrees / 360f, 0.845f, hueLightness(hueDegrees));
186         debug("colors", String.format(Locale.ENGLISH, "getPrimaryColorForHue(%d) = %x", hueDegrees,
187                 result));
188         return result;
189     }
190     public static void setupTheme(Activity activity) {
191         MobileLedgerProfile profile = Data.profile.getValue();
192         setupTheme(activity, profile);
193     }
194     public static void setupTheme(Activity activity, MobileLedgerProfile profile) {
195         final int themeHue = (profile == null) ? -1 : profile.getThemeHue();
196         setupTheme(activity, themeHue);
197     }
198     public static int getThemeIdForHue(int themeHue) {
199         int themeId = -1;
200         if (themeHue == 360)
201             themeHue = 0;
202         if ((themeHue >= 0) && (themeHue < 360)) {
203             int index;
204             if ((themeHue % HueRing.hueStepDegrees) != 0) {
205                 Logger.warn("profiles",
206                         String.format(Locale.US, "Adjusting unexpected hue %d", themeHue));
207                 index = Math.round(1f * themeHue / HueRing.hueStepDegrees);
208             }
209             else
210                 index = themeHue / HueRing.hueStepDegrees;
211
212             themeId = themeIDs[index];
213         }
214
215         if (themeId < 0) {
216             themeId = R.style.AppTheme_NoActionBar;
217             debug("profiles",
218                     String.format(Locale.ENGLISH, "Theme hue %d not supported, using the default",
219                             themeHue));
220         }
221
222         return themeId;
223     }
224     public static void setupTheme(Activity activity, int themeHue) {
225         int themeId = getThemeIdForHue(themeHue);
226         activity.setTheme(themeId);
227
228         refreshColors(activity.getTheme());
229     }
230
231     public static @NonNull
232     ColorStateList getColorStateList() {
233         return getColorStateList(profileThemeId);
234     }
235     public static @NonNull
236     ColorStateList getColorStateList(int hue) {
237         return new ColorStateList(EMPTY_STATES, getColors(hue));
238     }
239     public static int[] getColors() {
240         return getColors(profileThemeId);
241     }
242     public static int[] getColors(int hue) {
243         int[] colors = new int[]{0, 0, 0, 0, 0, 0};
244         for (int i = 0; i < 6; i++, hue = (hue + 60) % 360) {
245             colors[i] = getPrimaryColorForHue(hue);
246         }
247         return colors;
248     }
249     public static int getNewProfileThemeHue(ArrayList<MobileLedgerProfile> profiles) {
250         if ((profiles == null) || (profiles.size() == 0))
251             return DEFAULT_HUE_DEG;
252
253         int chosenHue;
254
255         if (profiles.size() == 1) {
256             int opposite = profiles.get(0)
257                                    .getThemeHue() + 180;
258             opposite %= 360;
259             chosenHue = opposite;
260         }
261         else {
262             ArrayList<Integer> hues = new ArrayList<>();
263             for (MobileLedgerProfile p : profiles) {
264                 int hue = p.getThemeHue();
265                 if (hue == -1)
266                     hue = DEFAULT_HUE_DEG;
267                 hues.add(hue);
268             }
269             Collections.sort(hues);
270             if (BuildConfig.DEBUG) {
271                 StringBuilder huesSB = new StringBuilder();
272                 for (int h : hues) {
273                     if (huesSB.length() > 0)
274                         huesSB.append(", ");
275                     huesSB.append(String.valueOf(h));
276                 }
277                 debug("profiles", String.format("used hues: %s", huesSB.toString()));
278             }
279             hues.add(hues.get(0));
280
281             int lastHue = -1;
282             int largestInterval = 0;
283             ArrayList<Integer> largestIntervalStarts = new ArrayList<>();
284
285             for (int h : hues) {
286                 if (lastHue == -1) {
287                     lastHue = h;
288                     continue;
289                 }
290
291                 int interval;
292                 if (h > lastHue)
293                     interval = h - lastHue;     // 10 -> 20 is a step of 10
294                 else
295                     interval = h + (360 - lastHue);    // 350 -> 20 is a step of 30
296
297                 if (interval > largestInterval) {
298                     largestInterval = interval;
299                     largestIntervalStarts.clear();
300                     largestIntervalStarts.add(lastHue);
301                 }
302                 else if (interval == largestInterval) {
303                     largestIntervalStarts.add(lastHue);
304                 }
305
306                 lastHue = h;
307             }
308
309             final int chosenIndex = (int) (Math.random() * largestIntervalStarts.size());
310             int chosenIntervalStart = largestIntervalStarts.get(chosenIndex);
311
312             debug("profiles",
313                     String.format(Locale.US, "Choosing the middle colour between %d and %d",
314                             chosenIntervalStart, chosenIntervalStart + largestInterval));
315
316             if (largestInterval % 2 != 0)
317                 largestInterval++;    // round up the middle point
318
319             chosenHue = (chosenIntervalStart + (largestInterval / 2)) % 360;
320         }
321
322         final int mod = chosenHue % THEME_HUE_STEP_DEG;
323         if (mod != 0) {
324             if (mod > THEME_HUE_STEP_DEG / 2)
325                 chosenHue += (THEME_HUE_STEP_DEG - mod); // 13 += (5-3) = 15
326             else
327                 chosenHue -= mod;       // 12 -= 2 = 10
328         }
329
330         debug("profiles", String.format(Locale.US, "New profile hue: %d", chosenHue));
331
332         return chosenHue;
333     }
334 }