mirror of https://github.com/tasks/tasks
Merge branch 'dev'
commit
a8ca8f346d
@ -0,0 +1,96 @@
|
||||
/**
|
||||
* See the file "LICENSE" for the full license governing this code.
|
||||
*/
|
||||
package com.todoroo.astrid.api;
|
||||
|
||||
import android.os.Parcel;
|
||||
import android.os.Parcelable;
|
||||
import android.widget.RemoteViews;
|
||||
import android.widget.RemoteViews.RemoteView;
|
||||
|
||||
/**
|
||||
* Represents a line of text displayed in the Task List
|
||||
*
|
||||
* @author Tim Su <tim@todoroo.com>
|
||||
*
|
||||
*/
|
||||
public final class TaskDecoration implements Parcelable {
|
||||
|
||||
/**
|
||||
* Place decoration between completion box and task title
|
||||
*/
|
||||
public static final int POSITION_LEFT = 0;
|
||||
|
||||
/**
|
||||
* Place decoration between task title and importance bar
|
||||
*/
|
||||
public static final int POSITION_RIGHT = 1;
|
||||
|
||||
/**
|
||||
* {@link RemoteView} decoration
|
||||
*/
|
||||
public RemoteViews decoration = null;
|
||||
|
||||
/**
|
||||
* Decoration position
|
||||
*/
|
||||
public int position = POSITION_LEFT;
|
||||
|
||||
/**
|
||||
* Decorated task background color. 0 is default
|
||||
*/
|
||||
public int color = 0;
|
||||
|
||||
/**
|
||||
* Creates a TaskDetail object
|
||||
* @param text
|
||||
* text to display
|
||||
* @param color
|
||||
* color to use for text. Use <code>0</code> for default color
|
||||
*/
|
||||
public TaskDecoration(RemoteViews decoration, int position, int color) {
|
||||
this.decoration = decoration;
|
||||
this.position = position;
|
||||
this.color = color;
|
||||
}
|
||||
|
||||
// --- parcelable helpers
|
||||
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
public int describeContents() {
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
public void writeToParcel(Parcel dest, int flags) {
|
||||
dest.writeParcelable(decoration, 0);
|
||||
dest.writeInt(position);
|
||||
dest.writeInt(color);
|
||||
}
|
||||
|
||||
/**
|
||||
* Parcelable creator
|
||||
*/
|
||||
public static final Parcelable.Creator<TaskDecoration> CREATOR = new Parcelable.Creator<TaskDecoration>() {
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
public TaskDecoration createFromParcel(Parcel source) {
|
||||
return new TaskDecoration((RemoteViews)source.readParcelable(
|
||||
RemoteViews.class.getClassLoader()),
|
||||
source.readInt(), source.readInt());
|
||||
}
|
||||
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
public TaskDecoration[] newArray(int size) {
|
||||
return new TaskDecoration[size];
|
||||
};
|
||||
};
|
||||
|
||||
}
|
@ -1,84 +0,0 @@
|
||||
/**
|
||||
* See the file "LICENSE" for the full license governing this code.
|
||||
*/
|
||||
package com.todoroo.astrid.api;
|
||||
|
||||
import android.os.Parcel;
|
||||
import android.os.Parcelable;
|
||||
|
||||
/**
|
||||
* Represents a line of text displayed in the Task List
|
||||
*
|
||||
* @author Tim Su <tim@todoroo.com>
|
||||
*
|
||||
*/
|
||||
public final class TaskDetail implements Parcelable {
|
||||
|
||||
/**
|
||||
* Text of detail
|
||||
*/
|
||||
public String text = null;
|
||||
|
||||
/**
|
||||
* Color to use for text. 0 is default
|
||||
*/
|
||||
public int color = 0;
|
||||
|
||||
/**
|
||||
* Creates a TaskDetail object
|
||||
* @param text
|
||||
* text to display
|
||||
* @param color
|
||||
* color to use for text. Use <code>0</code> for default color
|
||||
*/
|
||||
public TaskDetail(String text, int color) {
|
||||
this.text = text;
|
||||
this.color = color;
|
||||
}
|
||||
|
||||
/**
|
||||
* Convenience constructor to make a TaskDetail with default color
|
||||
* @param text
|
||||
* text to display
|
||||
*/
|
||||
public TaskDetail(String text) {
|
||||
this(text, 0);
|
||||
}
|
||||
|
||||
// --- parcelable helpers
|
||||
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
public int describeContents() {
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
public void writeToParcel(Parcel dest, int flags) {
|
||||
dest.writeString(text);
|
||||
dest.writeInt(color);
|
||||
}
|
||||
|
||||
/**
|
||||
* Parcelable creator
|
||||
*/
|
||||
public static final Parcelable.Creator<TaskDetail> CREATOR = new Parcelable.Creator<TaskDetail>() {
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
public TaskDetail createFromParcel(Parcel source) {
|
||||
return new TaskDetail(source.readString(), source.readInt());
|
||||
}
|
||||
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
public TaskDetail[] newArray(int size) {
|
||||
return new TaskDetail[size];
|
||||
};
|
||||
};
|
||||
|
||||
}
|
@ -0,0 +1,28 @@
|
||||
package com.todoroo.astrid.core;
|
||||
|
||||
import com.todoroo.andlib.service.Autowired;
|
||||
import com.todoroo.andlib.service.DependencyInjectionService;
|
||||
import com.todoroo.astrid.service.TaskService;
|
||||
|
||||
public final class PluginServices {
|
||||
|
||||
@Autowired
|
||||
TaskService taskService;
|
||||
|
||||
private static PluginServices instance;
|
||||
|
||||
private PluginServices() {
|
||||
DependencyInjectionService.getInstance().inject(this);
|
||||
}
|
||||
|
||||
private synchronized static PluginServices getInstance() {
|
||||
if(instance == null)
|
||||
instance = new PluginServices();
|
||||
return instance;
|
||||
}
|
||||
|
||||
public static TaskService getTaskService() {
|
||||
return getInstance().taskService;
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,63 @@
|
||||
/**
|
||||
* See the file "LICENSE" for the full license governing this code.
|
||||
*/
|
||||
package com.todoroo.astrid.timers;
|
||||
|
||||
import android.app.PendingIntent;
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
|
||||
import com.timsu.astrid.R;
|
||||
import com.todoroo.astrid.api.AstridApiConstants;
|
||||
import com.todoroo.astrid.api.TaskAction;
|
||||
import com.todoroo.astrid.api.TaskDecoration;
|
||||
import com.todoroo.astrid.core.PluginServices;
|
||||
import com.todoroo.astrid.model.Task;
|
||||
|
||||
/**
|
||||
* Exposes {@link TaskDecoration} for timers
|
||||
*
|
||||
* @author Tim Su <tim@todoroo.com>
|
||||
*
|
||||
*/
|
||||
public class TimerActionExposer extends BroadcastReceiver {
|
||||
|
||||
private static final String TIMER_ACTION = "com.todoroo.astrid.TIMER_BUTTON"; //$NON-NLS-1$
|
||||
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
long taskId = intent.getLongExtra(AstridApiConstants.EXTRAS_TASK_ID, -1);
|
||||
if(taskId == -1)
|
||||
return;
|
||||
|
||||
Task task = PluginServices.getTaskService().fetchById(taskId, Task.ID, Task.TIMER_START,
|
||||
Task.ELAPSED_SECONDS);
|
||||
|
||||
// was part of a broadcast for actions
|
||||
if(AstridApiConstants.BROADCAST_REQUEST_ACTIONS.equals(intent.getAction())) {
|
||||
String label;
|
||||
if(task.getValue(Task.TIMER_START) == 0)
|
||||
label = context.getString(R.string.TAE_startTimer);
|
||||
else
|
||||
label = context.getString(R.string.TAE_stopTimer);
|
||||
Intent newIntent = new Intent(TIMER_ACTION);
|
||||
newIntent.putExtra(AstridApiConstants.EXTRAS_TASK_ID, taskId);
|
||||
TaskAction action = new TaskAction(label,
|
||||
PendingIntent.getBroadcast(context, (int)taskId, newIntent, 0));
|
||||
|
||||
// transmit
|
||||
Intent broadcastIntent = new Intent(AstridApiConstants.BROADCAST_SEND_ACTIONS);
|
||||
broadcastIntent.putExtra(AstridApiConstants.EXTRAS_ADDON, TimerPlugin.IDENTIFIER);
|
||||
broadcastIntent.putExtra(AstridApiConstants.EXTRAS_RESPONSE, action);
|
||||
broadcastIntent.putExtra(AstridApiConstants.EXTRAS_TASK_ID, taskId);
|
||||
context.sendBroadcast(broadcastIntent, AstridApiConstants.PERMISSION_READ);
|
||||
} else if(TIMER_ACTION.equals(intent.getAction())) {
|
||||
if(task.getValue(Task.TIMER_START) == 0)
|
||||
TimerPlugin.updateTimer(context, task, true);
|
||||
else
|
||||
TimerPlugin.updateTimer(context, task, false);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,79 @@
|
||||
/**
|
||||
* See the file "LICENSE" for the full license governing this code.
|
||||
*/
|
||||
package com.todoroo.astrid.timers;
|
||||
|
||||
import java.util.HashMap;
|
||||
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.graphics.Color;
|
||||
import android.os.SystemClock;
|
||||
import android.widget.RemoteViews;
|
||||
|
||||
import com.timsu.astrid.R;
|
||||
import com.todoroo.andlib.utility.DateUtilities;
|
||||
import com.todoroo.astrid.api.AstridApiConstants;
|
||||
import com.todoroo.astrid.api.TaskDecoration;
|
||||
import com.todoroo.astrid.core.PluginServices;
|
||||
import com.todoroo.astrid.model.Task;
|
||||
|
||||
/**
|
||||
* Exposes {@link TaskDecoration} for timers
|
||||
*
|
||||
* @author Tim Su <tim@todoroo.com>
|
||||
*
|
||||
*/
|
||||
public class TimerDecorationExposer extends BroadcastReceiver {
|
||||
|
||||
private static final int TIMING_BG_COLOR = Color.argb(200, 220, 50, 0);
|
||||
|
||||
private static HashMap<Long, TaskDecoration> decorations =
|
||||
new HashMap<Long, TaskDecoration>();
|
||||
|
||||
public static void removeFromCache(long taskId) {
|
||||
decorations.remove(taskId);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
long taskId = intent.getLongExtra(AstridApiConstants.EXTRAS_TASK_ID, -1);
|
||||
if(taskId == -1)
|
||||
return;
|
||||
|
||||
Task task = PluginServices.getTaskService().fetchById(taskId, Task.ELAPSED_SECONDS, Task.TIMER_START);
|
||||
if(task == null || (task.getValue(Task.ELAPSED_SECONDS) == 0 &&
|
||||
task.getValue(Task.TIMER_START) == 0))
|
||||
return;
|
||||
|
||||
TaskDecoration decoration;
|
||||
if(!decorations.containsKey(taskId)) {
|
||||
RemoteViews remoteViews = new RemoteViews(context.getPackageName(),
|
||||
R.layout.timer_decoration);
|
||||
decoration = new TaskDecoration(remoteViews,
|
||||
TaskDecoration.POSITION_LEFT, 0);
|
||||
decorations.put(taskId, decoration);
|
||||
} else {
|
||||
decoration = decorations.get(taskId);
|
||||
}
|
||||
|
||||
long elapsed = task.getValue(Task.ELAPSED_SECONDS) * 1000L;
|
||||
if(task.getValue(Task.TIMER_START) != 0) {
|
||||
decoration.color = TIMING_BG_COLOR;
|
||||
elapsed += DateUtilities.now() - task.getValue(Task.TIMER_START);
|
||||
}
|
||||
|
||||
// update timer
|
||||
decoration.decoration.setChronometer(R.id.timer, SystemClock.elapsedRealtime() -
|
||||
elapsed, null, decoration.color != 0);
|
||||
|
||||
// transmit
|
||||
Intent broadcastIntent = new Intent(AstridApiConstants.BROADCAST_SEND_DECORATIONS);
|
||||
broadcastIntent.putExtra(AstridApiConstants.EXTRAS_ADDON, TimerPlugin.IDENTIFIER);
|
||||
broadcastIntent.putExtra(AstridApiConstants.EXTRAS_RESPONSE, decoration);
|
||||
broadcastIntent.putExtra(AstridApiConstants.EXTRAS_TASK_ID, taskId);
|
||||
context.sendBroadcast(broadcastIntent, AstridApiConstants.PERMISSION_READ);
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,60 @@
|
||||
/**
|
||||
* See the file "LICENSE" for the full license governing this code.
|
||||
*/
|
||||
package com.todoroo.astrid.timers;
|
||||
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.ContentValues;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.res.Resources;
|
||||
import android.graphics.drawable.BitmapDrawable;
|
||||
|
||||
import com.timsu.astrid.R;
|
||||
import com.todoroo.andlib.sql.Query;
|
||||
import com.todoroo.andlib.sql.QueryTemplate;
|
||||
import com.todoroo.astrid.activity.FilterListActivity;
|
||||
import com.todoroo.astrid.api.AstridApiConstants;
|
||||
import com.todoroo.astrid.api.Filter;
|
||||
import com.todoroo.astrid.api.FilterListItem;
|
||||
import com.todoroo.astrid.core.PluginServices;
|
||||
import com.todoroo.astrid.model.Task;
|
||||
|
||||
/**
|
||||
* Exposes "working on" filter to the {@link FilterListActivity}
|
||||
*
|
||||
* @author Tim Su <tim@todoroo.com>
|
||||
*
|
||||
*/
|
||||
public final class TimerFilterExposer extends BroadcastReceiver {
|
||||
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
|
||||
if(PluginServices.getTaskService().count(Query.select(Task.ID).
|
||||
where(Task.TIMER_START.gt(0))) == 0)
|
||||
return;
|
||||
|
||||
Filter workingOn = createFilter(context);
|
||||
|
||||
// transmit filter list
|
||||
FilterListItem[] list = new FilterListItem[1];
|
||||
list[0] = workingOn;
|
||||
Intent broadcastIntent = new Intent(AstridApiConstants.BROADCAST_SEND_FILTERS);
|
||||
broadcastIntent.putExtra(AstridApiConstants.EXTRAS_RESPONSE, list);
|
||||
context.sendBroadcast(broadcastIntent, AstridApiConstants.PERMISSION_READ);
|
||||
}
|
||||
|
||||
public static Filter createFilter(Context context) {
|
||||
Resources r = context.getResources();
|
||||
ContentValues values = new ContentValues();
|
||||
values.put(Task.TIMER_START.name, Filter.VALUE_NOW);
|
||||
Filter workingOn = new Filter(r.getString(R.string.TFE_workingOn),
|
||||
r.getString(R.string.TFE_workingOn),
|
||||
new QueryTemplate().where(Task.TIMER_START.gt(0)),
|
||||
values);
|
||||
workingOn.listingIcon = ((BitmapDrawable)r.getDrawable(R.drawable.tango_clock)).getBitmap();
|
||||
return workingOn;
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,97 @@
|
||||
package com.todoroo.astrid.timers;
|
||||
|
||||
import android.app.Notification;
|
||||
import android.app.PendingIntent;
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.res.Resources;
|
||||
|
||||
import com.timsu.astrid.R;
|
||||
import com.todoroo.andlib.service.NotificationManager;
|
||||
import com.todoroo.andlib.service.NotificationManager.AndroidNotificationManager;
|
||||
import com.todoroo.andlib.sql.Query;
|
||||
import com.todoroo.andlib.utility.DateUtilities;
|
||||
import com.todoroo.astrid.activity.ShortcutActivity;
|
||||
import com.todoroo.astrid.api.Addon;
|
||||
import com.todoroo.astrid.api.AstridApiConstants;
|
||||
import com.todoroo.astrid.api.Filter;
|
||||
import com.todoroo.astrid.core.PluginServices;
|
||||
import com.todoroo.astrid.model.Task;
|
||||
import com.todoroo.astrid.utility.Constants;
|
||||
|
||||
public class TimerPlugin extends BroadcastReceiver {
|
||||
|
||||
static final String IDENTIFIER = "timer"; //$NON-NLS-1$
|
||||
|
||||
@Override
|
||||
@SuppressWarnings("nls")
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
Addon plugin = new Addon(IDENTIFIER, "Timer", "Todoroo",
|
||||
"Lets you time how long it takes to complete tasks.");
|
||||
|
||||
Intent broadcastIntent = new Intent(AstridApiConstants.BROADCAST_SEND_ADDONS);
|
||||
broadcastIntent.putExtra(AstridApiConstants.EXTRAS_ADDON, plugin);
|
||||
context.sendBroadcast(broadcastIntent, AstridApiConstants.PERMISSION_READ);
|
||||
}
|
||||
|
||||
/**
|
||||
* stops timer and sets elapsed time. you still need to save the task.
|
||||
* @param task
|
||||
* @param start if true, start timer. else, stop it
|
||||
*/
|
||||
public static void updateTimer(Context context, Task task, boolean start) {
|
||||
if(start) {
|
||||
if(task.getValue(Task.TIMER_START) == 0)
|
||||
task.setValue(Task.TIMER_START, DateUtilities.now());
|
||||
} else {
|
||||
if(task.getValue(Task.TIMER_START) > 0) {
|
||||
int newElapsed = (int)((DateUtilities.now() - task.getValue(Task.TIMER_START)) / 1000L);
|
||||
task.setValue(Task.TIMER_START, 0L);
|
||||
task.setValue(Task.ELAPSED_SECONDS,
|
||||
task.getValue(Task.ELAPSED_SECONDS) + newElapsed);
|
||||
}
|
||||
}
|
||||
PluginServices.getTaskService().save(task, true);
|
||||
TimerDecorationExposer.removeFromCache(task.getId());
|
||||
|
||||
// transmit new intents
|
||||
Intent intent = new Intent(AstridApiConstants.BROADCAST_REQUEST_ACTIONS);
|
||||
intent.putExtra(AstridApiConstants.EXTRAS_TASK_ID, task.getId());
|
||||
new TimerDecorationExposer().onReceive(context, intent);
|
||||
new TimerActionExposer().onReceive(context, intent);
|
||||
|
||||
// update notification
|
||||
TimerPlugin.updateNotifications(context);
|
||||
}
|
||||
|
||||
private static void updateNotifications(Context context) {
|
||||
NotificationManager nm = new AndroidNotificationManager(context);
|
||||
|
||||
int count = PluginServices.getTaskService().count(Query.select(Task.ID).
|
||||
where(Task.TIMER_START.gt(0)));
|
||||
if(count == 0) {
|
||||
nm.cancel(Constants.NOTIFICATION_TIMER);
|
||||
} else {
|
||||
Filter filter = TimerFilterExposer.createFilter(context);
|
||||
Intent notifyIntent = ShortcutActivity.createIntent(filter);
|
||||
notifyIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
|
||||
PendingIntent pendingIntent = PendingIntent.getActivity(context,
|
||||
Constants.NOTIFICATION_TIMER, notifyIntent, 0);
|
||||
|
||||
Resources r = context.getResources();
|
||||
String appName = r.getString(R.string.app_name);
|
||||
String text = r.getString(R.string.TPl_notification,
|
||||
r.getQuantityString(R.plurals.Ntasks, count, count));
|
||||
Notification notification = new Notification(
|
||||
R.drawable.timers_notification, text, System.currentTimeMillis());
|
||||
notification.setLatestEventInfo(context, appName,
|
||||
text, pendingIntent);
|
||||
notification.flags |= Notification.FLAG_ONGOING_EVENT | Notification.FLAG_NO_CLEAR;
|
||||
notification.flags &= ~Notification.FLAG_AUTO_CANCEL;
|
||||
|
||||
nm.notify(Constants.NOTIFICATION_TIMER, notification);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,27 @@
|
||||
package com.todoroo.astrid.timers;
|
||||
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
|
||||
import com.todoroo.astrid.api.AstridApiConstants;
|
||||
import com.todoroo.astrid.core.PluginServices;
|
||||
import com.todoroo.astrid.model.Task;
|
||||
|
||||
public class TimerTaskCompleteListener extends BroadcastReceiver {
|
||||
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
long taskId = intent.getLongExtra(AstridApiConstants.EXTRAS_TASK_ID, -1);
|
||||
if(taskId == -1)
|
||||
return;
|
||||
|
||||
Task task = PluginServices.getTaskService().fetchById(taskId, Task.ID, Task.ELAPSED_SECONDS,
|
||||
Task.TIMER_START);
|
||||
if(task == null || task.getValue(Task.TIMER_START) == 0)
|
||||
return;
|
||||
|
||||
TimerPlugin.updateTimer(context, task, false);
|
||||
}
|
||||
|
||||
}
|
Binary file not shown.
After Width: | Height: | Size: 1.4 KiB |
Binary file not shown.
After Width: | Height: | Size: 897 B |
Binary file not shown.
After Width: | Height: | Size: 1.5 KiB |
@ -0,0 +1,25 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<!-- See the file "LICENSE" for the full license governing this code. -->
|
||||
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="fill_parent"
|
||||
android:orientation="vertical"
|
||||
android:paddingLeft="3px">
|
||||
|
||||
<ImageView
|
||||
android:layout_width="fill_parent"
|
||||
android:layout_height="wrap_content"
|
||||
android:layout_weight="5"
|
||||
android:gravity="bottom|center_horizontal"
|
||||
android:scaleType="center"
|
||||
android:src="@drawable/timers_decoration" />
|
||||
|
||||
<Chronometer android:id="@+id/timer"
|
||||
android:layout_width="fill_parent"
|
||||
android:layout_height="wrap_content"
|
||||
android:layout_weight="2"
|
||||
android:gravity="top"
|
||||
android:textSize="10sp" />
|
||||
|
||||
</LinearLayout>
|
||||
|
@ -0,0 +1,22 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<!-- See the file "LICENSE" for the full license governing this code. -->
|
||||
<resources xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
|
||||
<!-- Resources for built-in timers plug-in -->
|
||||
|
||||
<!-- Start Timer button -->
|
||||
<string name="TAE_startTimer">Start Timer</string>
|
||||
|
||||
<!-- Stop Timer button -->
|
||||
<string name="TAE_stopTimer">Stop Timer</string>
|
||||
|
||||
<!-- Notification Title (%s => # tasks) -->
|
||||
<string name="TPl_notification">Timers Active for %s!</string>
|
||||
|
||||
<!-- Filter Header -->
|
||||
<string name="TFE_category">Timer Filters</string>
|
||||
|
||||
<!-- Filter -->
|
||||
<string name="TFE_workingOn">Tasks Being Timed</string>
|
||||
|
||||
</resources>
|
Loading…
Reference in New Issue