mirror of
https://github.com/iSoron/uhabits.git
synced 2025-12-06 09:08:52 -06:00
Merge branch 'hotfix/1.7.5'
This commit is contained in:
@@ -67,6 +67,8 @@ public class BaseAndroidTest
|
|||||||
|
|
||||||
protected ModelFactory modelFactory;
|
protected ModelFactory modelFactory;
|
||||||
|
|
||||||
|
private boolean isDone = false;
|
||||||
|
|
||||||
@Before
|
@Before
|
||||||
public void setUp()
|
public void setUp()
|
||||||
{
|
{
|
||||||
@@ -117,6 +119,25 @@ public class BaseAndroidTest
|
|||||||
assertTrue(latch.await(60, TimeUnit.SECONDS));
|
assertTrue(latch.await(60, TimeUnit.SECONDS));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected void runConcurrently(Runnable... runnableList) throws Exception
|
||||||
|
{
|
||||||
|
isDone = false;
|
||||||
|
ExecutorService executor = Executors.newFixedThreadPool(100);
|
||||||
|
List<Future> futures = new LinkedList<>();
|
||||||
|
for (Runnable r : runnableList)
|
||||||
|
futures.add(executor.submit(() ->
|
||||||
|
{
|
||||||
|
while (!isDone) r.run();
|
||||||
|
return null;
|
||||||
|
}));
|
||||||
|
|
||||||
|
Thread.sleep(3000);
|
||||||
|
isDone = true;
|
||||||
|
executor.shutdown();
|
||||||
|
for(Future f : futures) f.get();
|
||||||
|
while (!executor.isTerminated()) Thread.sleep(50);
|
||||||
|
}
|
||||||
|
|
||||||
protected void setTheme(@StyleRes int themeId)
|
protected void setTheme(@StyleRes int themeId)
|
||||||
{
|
{
|
||||||
targetContext.setTheme(themeId);
|
targetContext.setTheme(themeId);
|
||||||
|
|||||||
@@ -61,29 +61,6 @@ public class SQLiteScoreListTest extends BaseAndroidTest
|
|||||||
day = DateUtils.millisecondsInOneDay;
|
day = DateUtils.millisecondsInOneDay;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
|
||||||
public void testGetAll()
|
|
||||||
{
|
|
||||||
List<Score> list = scores.toList();
|
|
||||||
assertThat(list.size(), equalTo(121));
|
|
||||||
assertThat(list.get(0).getTimestamp(), equalTo(today));
|
|
||||||
assertThat(list.get(10).getTimestamp(), equalTo(today - 10 * day));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void testInvalidateNewerThan()
|
|
||||||
{
|
|
||||||
scores.getTodayValue(); // force recompute
|
|
||||||
List<ScoreRecord> records = getAllRecords();
|
|
||||||
assertThat(records.size(), equalTo(121));
|
|
||||||
|
|
||||||
scores.invalidateNewerThan(today - 10 * day);
|
|
||||||
|
|
||||||
records = getAllRecords();
|
|
||||||
assertThat(records.size(), equalTo(110));
|
|
||||||
assertThat(records.get(0).timestamp, equalTo(today - 11 * day));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testAdd()
|
public void testAdd()
|
||||||
{
|
{
|
||||||
@@ -101,6 +78,15 @@ public class SQLiteScoreListTest extends BaseAndroidTest
|
|||||||
assertThat(records.get(0).timestamp, equalTo(today));
|
assertThat(records.get(0).timestamp, equalTo(today));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetAll()
|
||||||
|
{
|
||||||
|
List<Score> list = scores.toList();
|
||||||
|
assertThat(list.size(), equalTo(121));
|
||||||
|
assertThat(list.get(0).getTimestamp(), equalTo(today));
|
||||||
|
assertThat(list.get(10).getTimestamp(), equalTo(today - 10 * day));
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testGetByInterval()
|
public void testGetByInterval()
|
||||||
{
|
{
|
||||||
@@ -115,6 +101,16 @@ public class SQLiteScoreListTest extends BaseAndroidTest
|
|||||||
assertThat(list.get(7).getTimestamp(), equalTo(today - 10 * day));
|
assertThat(list.get(7).getTimestamp(), equalTo(today - 10 * day));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetByInterval_concurrent() throws Exception
|
||||||
|
{
|
||||||
|
Runnable block1 = () -> scores.invalidateNewerThan(0);
|
||||||
|
Runnable block2 =
|
||||||
|
() -> assertThat(scores.getByInterval(today, today).size(),
|
||||||
|
equalTo(1));
|
||||||
|
runConcurrently(block1, block2);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testGetByInterval_withLongInterval()
|
public void testGetByInterval_withLongInterval()
|
||||||
{
|
{
|
||||||
@@ -125,6 +121,30 @@ public class SQLiteScoreListTest extends BaseAndroidTest
|
|||||||
assertThat(list.size(), equalTo(201));
|
assertThat(list.size(), equalTo(201));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetTodayValue_concurrent() throws Exception
|
||||||
|
{
|
||||||
|
Runnable block1 = () -> scores.invalidateNewerThan(0);
|
||||||
|
Runnable block2 =
|
||||||
|
() -> assertThat(scores.getTodayValue(), equalTo(18407827));
|
||||||
|
|
||||||
|
runConcurrently(block1, block2);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInvalidateNewerThan()
|
||||||
|
{
|
||||||
|
scores.getTodayValue(); // force recompute
|
||||||
|
List<ScoreRecord> records = getAllRecords();
|
||||||
|
assertThat(records.size(), equalTo(121));
|
||||||
|
|
||||||
|
scores.invalidateNewerThan(today - 10 * day);
|
||||||
|
|
||||||
|
records = getAllRecords();
|
||||||
|
assertThat(records.size(), equalTo(110));
|
||||||
|
assertThat(records.get(0).timestamp, equalTo(today - 11 * day));
|
||||||
|
}
|
||||||
|
|
||||||
private List<ScoreRecord> getAllRecords()
|
private List<ScoreRecord> getAllRecords()
|
||||||
{
|
{
|
||||||
return new Select()
|
return new Select()
|
||||||
|
|||||||
@@ -21,8 +21,8 @@
|
|||||||
<manifest
|
<manifest
|
||||||
package="org.isoron.uhabits"
|
package="org.isoron.uhabits"
|
||||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
android:versionCode="31"
|
android:versionCode="32"
|
||||||
android:versionName="1.7.4">
|
android:versionName="1.7.5">
|
||||||
|
|
||||||
<uses-permission android:name="android.permission.VIBRATE"/>
|
<uses-permission android:name="android.permission.VIBRATE"/>
|
||||||
|
|
||||||
|
|||||||
@@ -256,7 +256,7 @@ public class FrequencyChart extends ScrollableChart
|
|||||||
float scale = 1.0f/maxFreq * value;
|
float scale = 1.0f/maxFreq * value;
|
||||||
float radius = maxRadius * scale;
|
float radius = maxRadius * scale;
|
||||||
|
|
||||||
int colorIndex = Math.round((colors.length-1) * scale);
|
int colorIndex = Math.min(colors.length - 1, Math.round((colors.length - 1) * scale));
|
||||||
pGraph.setColor(colors[colorIndex]);
|
pGraph.setColor(colors[colorIndex]);
|
||||||
canvas.drawCircle(rect.centerX(), rect.centerY(), radius, pGraph);
|
canvas.drawCircle(rect.centerX(), rect.centerY(), radius, pGraph);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -81,7 +81,7 @@ public abstract class ScoreList implements Iterable<Score>
|
|||||||
* @param timestamp the timestamp of a day
|
* @param timestamp the timestamp of a day
|
||||||
* @return score value for that day
|
* @return score value for that day
|
||||||
*/
|
*/
|
||||||
public final synchronized int getValue(long timestamp)
|
public synchronized final int getValue(long timestamp)
|
||||||
{
|
{
|
||||||
compute(timestamp, timestamp);
|
compute(timestamp, timestamp);
|
||||||
Score s = getComputedByTimestamp(timestamp);
|
Score s = getComputedByTimestamp(timestamp);
|
||||||
|
|||||||
@@ -100,7 +100,8 @@ public class SQLiteScoreList extends ScoreList
|
|||||||
|
|
||||||
@NonNull
|
@NonNull
|
||||||
@Override
|
@Override
|
||||||
public List<Score> getByInterval(long fromTimestamp, long toTimestamp)
|
public synchronized List<Score> getByInterval(long fromTimestamp,
|
||||||
|
long toTimestamp)
|
||||||
{
|
{
|
||||||
check(habit.getId());
|
check(habit.getId());
|
||||||
compute(fromTimestamp, toTimestamp);
|
compute(fromTimestamp, toTimestamp);
|
||||||
@@ -137,7 +138,7 @@ public class SQLiteScoreList extends ScoreList
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int getTodayValue()
|
public synchronized int getTodayValue()
|
||||||
{
|
{
|
||||||
if (cache == null || cache.expired())
|
if (cache == null || cache.expired())
|
||||||
cache = new CachedData(super.getTodayValue());
|
cache = new CachedData(super.getTodayValue());
|
||||||
@@ -146,7 +147,7 @@ public class SQLiteScoreList extends ScoreList
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void invalidateNewerThan(long timestamp)
|
public synchronized void invalidateNewerThan(long timestamp)
|
||||||
{
|
{
|
||||||
cache = null;
|
cache = null;
|
||||||
invalidateStatement.bindLong(1, habit.getId());
|
invalidateStatement.bindLong(1, habit.getId());
|
||||||
|
|||||||
@@ -28,6 +28,8 @@ import org.isoron.uhabits.utils.*;
|
|||||||
|
|
||||||
import javax.inject.*;
|
import javax.inject.*;
|
||||||
|
|
||||||
|
import static org.isoron.uhabits.utils.DateUtils.*;
|
||||||
|
|
||||||
@ReceiverScope
|
@ReceiverScope
|
||||||
public class ReminderController
|
public class ReminderController
|
||||||
{
|
{
|
||||||
@@ -66,7 +68,7 @@ public class ReminderController
|
|||||||
{
|
{
|
||||||
long snoozeInterval = preferences.getSnoozeInterval();
|
long snoozeInterval = preferences.getSnoozeInterval();
|
||||||
|
|
||||||
long now = DateUtils.getLocalTime();
|
long now = applyTimezone(getLocalTime());
|
||||||
long reminderTime = now + snoozeInterval * 60 * 1000;
|
long reminderTime = now + snoozeInterval * 60 * 1000;
|
||||||
|
|
||||||
reminderScheduler.schedule(habit, reminderTime);
|
reminderScheduler.schedule(habit, reminderTime);
|
||||||
|
|||||||
Reference in New Issue
Block a user