Compare commits

...

32 Commits

Author SHA1 Message Date
Federico Hernandez
12c4983936 Release date for 1.8.4 2009-11-17 12:00:36 +01:00
Paul Beckingham
39d9f235de HACK - case-insensitive file system problem again. 2009-11-16 23:42:32 -05:00
Paul Beckingham
7aa0c3698a HACK - case-insensitive file system problem again. 2009-11-16 23:42:00 -05:00
Paul Beckingham
bc40ab63b3 Bug Fix - #312 Changing one task changes another
- Added a warning when modifying recurring tasks, that all instances of
  that task may be modified.  When task confirms a bulk edit the
  recurrence is again indicated (thanks to Cory Donnelly).
2009-11-16 23:24:47 -05:00
Paul Beckingham
6e673d2834 Bug Fix - #313 Edit command fails when data.location includes spaces
- Applied patch from Cory Donnelly.
2009-11-16 22:10:47 -05:00
Federico Hernandez
30c6dd0047 Added Joe to AUTHORS file 2009-11-13 23:32:40 +01:00
Paul Beckingham
64bc2a165a Bug fix - hang on cygwin when task updated.
- Fixed bug that caused a hang on cygwin, when a task with multiple
  annotations was edited (thanks to Joe Pulliam).
2009-11-09 22:35:51 -05:00
Paul Beckingham
5b96dbbce8 Bug Fix - wait date editing
- The "wait" date was not being properly formatted, as are all the other
  dates, in the "edit" command.  The result is that an epoch integer date
  was rendered, instead of something readable and in the preferred format.
2009-11-09 18:19:04 -05:00
Federico Hernandez
77dd930574 Fixed bug in regexp matching of whitespace between month and year. 2009-11-05 21:36:46 +01:00
Federico Hernandez
f6842941f3 Bumped version number to 1.8.4 2009-10-22 22:40:18 +02:00
Federico Hernandez
e2e0851a69 Merge branch '1.8.3' 2009-10-22 22:27:21 +02:00
Federico Hernandez
1299fe468b Added sha1 for 1.8.3 to Changelog 2009-10-22 22:09:53 +02:00
Federico Hernandez
de50b2902c Merge branch '1.8.3' 2009-10-22 22:02:32 +02:00
Federico Hernandez
bcdcbeeea0 Preparing release
- Updating dist documentation
- Release date in man pages
2009-10-21 22:53:26 +02:00
Pietro Cerutti
469cafa053 Make run_all more portable.
- Changed shebang for portability
- Fix date command for FreeBSD

Signed-off-by: Federico Hernandez <ultrafredde@gmail.com>
2009-10-21 22:20:32 +02:00
Pietro Cerutti
fdb359c180 Fix confirm() to handle EOF
Signed-off-by: Federico Hernandez <ultrafredde@gmail.com>
2009-10-21 22:19:37 +02:00
Paul Beckingham
b3d40b2554 Portability - Haiku R1/alpha1
- Added necessary include files and edits in order to build task on
  Haiku R1/alpha1.
2009-09-27 17:02:20 -04:00
Federico Hernandez
fba076a0d0 Bumped version number to 1.8.3 2009-09-07 16:13:37 +02:00
Federico Hernandez
09d7940dd3 Revert "Bumping version number to 1.8.3"
This reverts commit 00031dc1ab.
2009-09-07 16:04:24 +02:00
Federico Hernandez
00031dc1ab Bumping version number to 1.8.3 2009-09-07 15:52:51 +02:00
Paul Beckingham
3ef844de5f Packaging
- Adding updated package files for OSX, from the 1.8.2 release.
2009-09-07 09:51:20 -04:00
Federico Hernandez
bb45d91ddb Added sha1 of 1.8.2 to ChangLog 2009-09-07 15:41:51 +02:00
Federico Hernandez
f243f0ed44 Release date in man pages 2009-09-07 10:18:44 +02:00
Federico Hernandez
b305cc0a60 Release date in ChangeLog 2009-09-07 09:57:09 +02:00
Paul Beckingham
53609b2837 Bug Fix - #288
- Fixed bug which failed to propagate rc overrides from the command
  line to the default command (thanks to Zach Frazier).
2009-09-06 16:27:28 -04:00
Federico Hernandez
bdaa0f89d9 Platform update 2009-09-02 12:50:43 +02:00
Federico Hernandez
2fd8d8aa83 Bug Fix #291
- Fixed bug #291 which generated a false warning about an unrecognized
  variable when enabling default.projects in .taskrc (thanks to Thomas@BIC)
2009-09-02 10:43:04 +02:00
Federico Hernandez
e69fb81b2b Documentation update in Changelog for #289 2009-09-01 21:12:49 +02:00
Federico Hernandez
d2aa0f31b0 Bug Fix - #289
- Fixed bug #289 which imported task from todo.sh
      without valid uuids (thanks to Ben Jackson).
2009-09-01 21:08:13 +02:00
Paul Beckingham
05f67db429 Bug Fix - #287
- Fixed bug #287 which caused color control codes to be written to
  the shadow file (thanks to Richard Querin).
2009-08-29 13:55:59 -04:00
Paul Beckingham
62be3f8acb Feature #282
- Applied large patch to make task return meaningful exit codes.
- Added unit tests to prove this.
- Thanks to Pietro Cerutti.
2009-08-29 09:14:26 -04:00
Federico Hernandez
cc5c99c0a1 Pushing version number to 1.8.2 2009-08-21 01:09:41 +02:00
33 changed files with 446 additions and 333 deletions

View File

@@ -18,6 +18,7 @@ The following submitted code, packages or analysis, and deserve special thanks:
Johan Friis Johan Friis
Steven de Brouwer Steven de Brouwer
Pietro Cerutti Pietro Cerutti
Cory Donnelly
Thanks to the following, who submitted detailed bug reports and excellent suggestions: Thanks to the following, who submitted detailed bug reports and excellent suggestions:
Eugene Kramer Eugene Kramer
@@ -38,4 +39,5 @@ Thanks to the following, who submitted detailed bug reports and excellent sugges
Mike Adonay Mike Adonay
Thomas@BIC Thomas@BIC
Ian Mortimer Ian Mortimer
Zach Frazier
Joe Pulliam

View File

@@ -1,7 +1,36 @@
------ current release --------------------------- ------ current release ---------------------------
1.8.1 (8/20/2009) 1.8.4 (11/17/2009)
+ Fixed bug that caused wait: dates to not be properly rendered in a
readable and preferred format with the "edit" command.
+ Fixed bug that caused a hang on cygwin, when a task with multiple
annotations was edited (thanks to Joe Pulliam).
+ Fixed bug #314 where the edit command fails when data.location includes
directories containing spaces (thanks to Cory Donnelly).
+ Added a warning (issue #312) when modifying recurring tasks, that all
instances of that task may be modified. When task confirms a bulk edit
the recurrence is again indicated (thanks to Cory Donnelly).
------ old releases ------------------------------
1.8.3 (10/21/2009) bcdcbeeea0d92f21c3565aebfaf6332b959f4025
+ Added support for Haiku R1/alpha1
1.8.2 (9/7/2009) f243f0ed443ecd7dde779de8a6525222591024db
+ Added feature #282 that returns useful exit codes to the shell. Now a
script can detect whether no tasks were returned by a report (thanks to
Pietro Cerutti).
+ Fixed bug #287 that causes color control codes to be written to shadow
files (thanks to Richard Querin).
+ Fixed bug #289 which imported task from todo.sh without valid uuids
(thanks to Ben Jackson).
+ Fixed bug #291 which generated a false warning about an unrecognized
variable when enabling default.projects in .taskrc (thanks to Thomas@BIC).
+ Fixed bug #288 which failed to propagate rc file overrides on the command
line to the default command (thanks to Zach Frazier).
1.8.1 (8/20/2009) 35792e7874d2bb664abb1a0a67960b7fe7e0fccf
+ Fixed bug #231 that broke the build on OpenBSD 32-bit due to a time_t + Fixed bug #231 that broke the build on OpenBSD 32-bit due to a time_t
and int collision (thanks to Pietro Cerutti). and int collision (thanks to Pietro Cerutti).
+ Fixed bug #241 that prevented bash's tab-completion of projects in Fedora + Fixed bug #241 that prevented bash's tab-completion of projects in Fedora
@@ -34,8 +63,6 @@
+ Fixed bug that failed to suppress color control code in the header and footnote + Fixed bug that failed to suppress color control code in the header and footnote
when redirecting output to a file (thanks to John Florian). when redirecting output to a file (thanks to John Florian).
------ old releases ------------------------------
1.8.0 (7/21/2009) 14977ef317bd004dae2f2c313e806af9f2a2140c 1.8.0 (7/21/2009) 14977ef317bd004dae2f2c313e806af9f2a2140c
+ Added zsh tab completion script (thanks to P.C. Shyamshankar). + Added zsh tab completion script (thanks to P.C. Shyamshankar).
+ Fixed bug that cause the _forcecolor configuration variable to be + Fixed bug that cause the _forcecolor configuration variable to be

18
NEWS
View File

@@ -9,26 +9,24 @@ New Features in task 1.8
- In addition to being a standard part of Fedora 10 and 11 (yum install task), - In addition to being a standard part of Fedora 10 and 11 (yum install task),
task is now also a standard part of Cygwin 1.5 task is now also a standard part of Cygwin 1.5
- There are new demo movies on taskwarrior.org - There are new demo movies on taskwarrior.org
- Shell-friendly exit codes
Please refer to the ChangeLog file for full details. There are too many to Please refer to the ChangeLog file for full details. There are too many to
list here. list here.
Task has been built and tested on the following configurations: Task has been built and tested on the following configurations:
- OS X 10.5 Leopard - OS X 10.6 Snow Leopard and 10.5 Leopard
- OS X 10.4 Tiger - Fedora 11 Leonidas and 10 Cambridge
- Fedora Core 11 Leonidas - Ubuntu 9.04 Jaunty Jackalope and 8.10 Intrepid Ibex
- Fedora Core 10 Cambridge
- Ubuntu 9.04 Jaunty Jackalope
- Ubuntu 8.10 Intrepid Ibex
- Ubuntu 8.04 Hardy Heron
- Slackware 12.2 - Slackware 12.2
- Arch Linux - Arch Linux
- Solaris 10 - Gentoo Linux
- Solaris 8 - Solaris 10 and 8
- OpenBSD 4.5 - OpenBSD 4.5
- FreeBSD - FreeBSD
- Cygwin 1.5.25-14 - Cygwin 1.5
- Haiku R1/alpha1
While Task has undergone testing, bugs are sure to remain. If you encounter a While Task has undergone testing, bugs are sure to remain. If you encounter a
bug, please enter a new issue at: bug, please enter a new issue at:

View File

@@ -2,7 +2,7 @@
# Process this file with autoconf to produce a configure script. # Process this file with autoconf to produce a configure script.
AC_PREREQ(2.61) AC_PREREQ(2.61)
AC_INIT(task, 1.8.1, support@taskwarrior.org) AC_INIT(task, 1.8.4, support@taskwarrior.org)
CFLAGS="${CFLAGS=}" CFLAGS="${CFLAGS=}"
CXXFLAGS="${CXXFLAGS=}" CXXFLAGS="${CXXFLAGS=}"

View File

@@ -1,4 +1,4 @@
.TH task-tutorial 5 2009-08-14 "task 1.8.1" "User Manuals" .TH task-tutorial 5 2009-10-21 "task 1.8.4" "User Manuals"
.SH NAME .SH NAME
task-tutorial \- A tutorial for the task(1) command line todo manager. task-tutorial \- A tutorial for the task(1) command line todo manager.

View File

@@ -1,4 +1,4 @@
.TH task 1 2009-08-14 "task 1.8.1" "User Manuals" .TH task 1 2009-10-21 "task 1.8.4" "User Manuals"
.SH NAME .SH NAME
task \- A command line todo manager. task \- A command line todo manager.

View File

@@ -1,4 +1,4 @@
.TH taskrc 5 2009-08-14 "task 1.8.1" "User Manuals" .TH taskrc 5 2009-10-21 "task 1.8.4" "User Manuals"
.SH NAME .SH NAME
taskrc \- Configuration file for the task(1) command taskrc \- Configuration file for the task(1) command

View File

@@ -1 +1 @@
<pkg-contents spec="1.12"><f n="task" o="paul" g="staff" p="33261" pt="/Users/paul/task.git/package-config/osx/binary/task" m="false" t="file"/></pkg-contents> <pkg-contents spec="1.12"><f n="task" o="root" g="wheel" p="33261" pt="/Users/paul/task.git/package-config/osx/binary/task" m="false" t="file"><mod>group</mod><mod>owner</mod></f></pkg-contents>

View File

@@ -1 +1 @@
<pkgref spec="1.12" uuid="C71026FD-E252-42CD-89C3-2F6F087AAF17"><config><identifier>com.beckingham.task180.task.pkg</identifier><version>1.8.0</version><description></description><post-install type="none"/><requireAuthorization/><installFrom mod="true">/Users/paul/task.git/package-config/osx/binary/task</installFrom><installTo mod="true" relocatable="true">/usr/local/bin</installTo><flags><followSymbolicLinks/></flags><packageStore type="internal"></packageStore><mod>parent</mod><mod>locationType</mod><mod>relocatable</mod><mod>version</mod><mod>installTo.path</mod><mod>installTo</mod></config><contents><file-list>01task-contents.xml</file-list><filter>/CVS$</filter><filter>/\.svn$</filter><filter>/\.cvsignore$</filter><filter>/\.cvspass$</filter><filter>/\.DS_Store$</filter></contents></pkgref> <pkgref spec="1.12" uuid="C71026FD-E252-42CD-89C3-2F6F087AAF17"><config><identifier>com.beckingham.task182.task.pkg</identifier><version>1.8.2</version><description></description><post-install type="none"/><requireAuthorization/><installFrom mod="true">/Users/paul/task.git/package-config/osx/binary/task</installFrom><installTo mod="true" relocatable="true">/usr/local/bin</installTo><flags><followSymbolicLinks/></flags><packageStore type="internal"></packageStore><mod>installTo</mod><mod>locationType</mod><mod>relocatable</mod><mod>identifier</mod><mod>parent</mod><mod>version</mod><mod>installTo.path</mod></config><contents><file-list>01task-contents.xml</file-list><filter>/CVS$</filter><filter>/\.svn$</filter><filter>/\.cvsignore$</filter><filter>/\.cvspass$</filter><filter>/\.DS_Store$</filter></contents></pkgref>

View File

@@ -1 +1 @@
<pkmkdoc spec="1.12"><properties><title>Task 1.8.0</title><build>/Users/paul/Desktop/task-1.8.0.pkg</build><organization>com.beckingham</organization><userSees ui="easy"/><min-target os="3"/><domain anywhere="true" system="true"/></properties><distribution><versions min-spec="1.000000"/><scripts></scripts></distribution><contents><choice title="task" id="choice0" starts_selected="true" starts_enabled="true" starts_hidden="false"><pkgref id="com.beckingham.task180.task.pkg"/><choice-reqs><requirement id="tosv" operator="ge" value="'10.5.0'" selected="no" enabled="no" hidden="unchanged" startSelected="unchanged" startEnabled="unchanged" startHidden="unchanged"/></choice-reqs></choice></contents><resources bg-scale="proportional" bg-align="center"><locale lang="en"><resource mod="true" type="license">/Users/paul/task.git/package-config/osx/binary/COPYING.txt</resource><resource mod="true" type="readme">/Users/paul/task.git/package-config/osx/binary/README.txt</resource></locale></resources><flags/><item type="file">01task.xml</item><mod>properties.title</mod><mod>properties.anywhereDomain</mod><mod>properties.systemDomain</mod></pkmkdoc> <pkmkdoc spec="1.12"><properties><title>Task 1.8.2</title><build>/Users/paul/Desktop/task-1.8.2.pkg</build><organization>com.beckingham</organization><userSees ui="both"/><min-target os="3"/><domain anywhere="true" system="true"/></properties><distribution><versions min-spec="1.000000"/><scripts></scripts></distribution><contents><choice title="task" id="choice0" starts_selected="true" starts_enabled="true" starts_hidden="false"><pkgref id="com.beckingham.task182.task.pkg"/><choice-reqs><requirement id="tosv" operator="ge" value="'10.6.0'" selected="no" enabled="no" hidden="unchanged" startSelected="unchanged" startEnabled="unchanged" startHidden="unchanged"/></choice-reqs></choice></contents><resources bg-scale="proportional" bg-align="center"><locale lang="en"><resource mod="true" type="license">/Users/paul/task.git/package-config/osx/binary/COPYING.txt</resource><resource mod="true" type="readme">/Users/paul/task.git/package-config/osx/binary/README.txt</resource></locale></resources><flags/><item type="file">01task.xml</item><mod>properties.title</mod><mod>properties.customizeOption</mod><mod>properties.anywhereDomain</mod><mod>properties.systemDomain</mod></pkmkdoc>

View File

@@ -30,6 +30,7 @@
#include <pwd.h> #include <pwd.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
#include <unistd.h>
#include "Context.h" #include "Context.h"
#include "Timer.h" #include "Timer.h"
#include "text.h" #include "text.h"
@@ -49,6 +50,7 @@ Context::Context ()
, tdb () , tdb ()
, stringtable () , stringtable ()
, program ("") , program ("")
, overrides ("")
, cmd () , cmd ()
, inShadow (false) , inShadow (false)
{ {
@@ -126,23 +128,26 @@ void Context::initialize ()
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
int Context::run () int Context::run ()
{ {
int rc;
Timer t ("Context::run"); Timer t ("Context::run");
std::string output; std::string output;
try try
{ {
parse (); // Parse command line. parse (); // Parse command line.
output = dispatch (); // Dispatch to command handlers. rc = dispatch (output); // Dispatch to command handlers.
} }
catch (const std::string& error) catch (const std::string& error)
{ {
footnote (error); footnote (error);
rc = 2;
} }
catch (...) catch (...)
{ {
footnote (stringtable.get (100, "Unknown error.")); footnote (stringtable.get (100, "Unknown error."));
rc = 3;
} }
// Dump all debug messages. // Dump all debug messages.
@@ -170,63 +175,63 @@ int Context::run ()
else else
std::cout << *f << std::endl; std::cout << *f << std::endl;
return 0; return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string Context::dispatch () int Context::dispatch (std::string &out)
{ {
int rc = 0;
Timer t ("Context::dispatch"); Timer t ("Context::dispatch");
// TODO Just look at this thing. It cries out for a dispatch table. // TODO Just look at this thing. It cries out for a dispatch table.
std::string out; if (cmd.command == "projects") { rc = handleProjects (out); }
if (cmd.command == "projects") { out = handleProjects (); } else if (cmd.command == "tags") { rc = handleTags (out); }
else if (cmd.command == "tags") { out = handleTags (); } else if (cmd.command == "colors") { rc = handleColor (out); }
else if (cmd.command == "colors") { out = handleColor (); } else if (cmd.command == "version") { rc = handleVersion (out); }
else if (cmd.command == "version") { out = handleVersion (); } else if (cmd.command == "help") { rc = longUsage (out); }
else if (cmd.command == "help") { out = longUsage (); } else if (cmd.command == "stats") { rc = handleReportStats (out); }
else if (cmd.command == "stats") { out = handleReportStats (); } else if (cmd.command == "info") { rc = handleInfo (out); }
else if (cmd.command == "info") { out = handleInfo (); } else if (cmd.command == "history") { rc = handleReportHistory (out); }
else if (cmd.command == "history") { out = handleReportHistory (); } else if (cmd.command == "ghistory") { rc = handleReportGHistory (out); }
else if (cmd.command == "ghistory") { out = handleReportGHistory (); } else if (cmd.command == "summary") { rc = handleReportSummary (out); }
else if (cmd.command == "summary") { out = handleReportSummary (); } else if (cmd.command == "calendar") { rc = handleReportCalendar (out); }
else if (cmd.command == "calendar") { out = handleReportCalendar (); } else if (cmd.command == "timesheet") { rc = handleReportTimesheet (out); }
else if (cmd.command == "timesheet") { out = handleReportTimesheet (); } else if (cmd.command == "add") { rc = handleAdd (out); }
else if (cmd.command == "add") { out = handleAdd (); } else if (cmd.command == "append") { rc = handleAppend (out); }
else if (cmd.command == "append") { out = handleAppend (); } else if (cmd.command == "annotate") { rc = handleAnnotate (out); }
else if (cmd.command == "annotate") { out = handleAnnotate (); } else if (cmd.command == "done") { rc = handleDone (out); }
else if (cmd.command == "done") { out = handleDone (); } else if (cmd.command == "delete") { rc = handleDelete (out); }
else if (cmd.command == "delete") { out = handleDelete (); } else if (cmd.command == "start") { rc = handleStart (out); }
else if (cmd.command == "start") { out = handleStart (); } else if (cmd.command == "stop") { rc = handleStop (out); }
else if (cmd.command == "stop") { out = handleStop (); } else if (cmd.command == "export") { rc = handleExport (out); }
else if (cmd.command == "export") { out = handleExport (); } else if (cmd.command == "import") { rc = handleImport (out); }
else if (cmd.command == "import") { out = handleImport (); } else if (cmd.command == "duplicate") { rc = handleDuplicate (out); }
else if (cmd.command == "duplicate") { out = handleDuplicate (); } else if (cmd.command == "edit") { rc = handleEdit (out); }
else if (cmd.command == "edit") { out = handleEdit (); }
#ifdef FEATURE_SHELL #ifdef FEATURE_SHELL
else if (cmd.command == "shell") { handleShell (); } else if (cmd.command == "shell") { handleShell ( ); }
#endif #endif
else if (cmd.command == "undo") { handleUndo (); } else if (cmd.command == "undo") { handleUndo ( ); }
else if (cmd.command == "_projects") { out = handleCompletionProjects (); } else if (cmd.command == "_projects") { rc = handleCompletionProjects (out); }
else if (cmd.command == "_tags") { out = handleCompletionTags (); } else if (cmd.command == "_tags") { rc = handleCompletionTags (out); }
else if (cmd.command == "_commands") { out = handleCompletionCommands (); } else if (cmd.command == "_commands") { rc = handleCompletionCommands (out); }
else if (cmd.command == "_ids") { out = handleCompletionIDs (); } else if (cmd.command == "_ids") { rc = handleCompletionIDs (out); }
else if (cmd.command == "_config") { out = handleCompletionConfig (); } else if (cmd.command == "_config") { rc = handleCompletionConfig (out); }
else if (cmd.command == "" && else if (cmd.command == "" &&
sequence.size ()) { out = handleModify (); } sequence.size ()) { rc = handleModify (out); }
// Command that display IDs and therefore need TDB::gc first. // Command that display IDs and therefore need TDB::gc first.
else if (cmd.command == "next") { if (!inShadow) tdb.gc (); out = handleReportNext (); } else if (cmd.command == "next") { if (!inShadow) tdb.gc (); rc = handleReportNext (out); }
else if (cmd.validCustom (cmd.command)) { if (!inShadow) tdb.gc (); out = handleCustomReport (cmd.command); } else if (cmd.validCustom (cmd.command)) { if (!inShadow) tdb.gc (); rc = handleCustomReport (cmd.command, out); }
// If the command is not recognized, display usage. // If the command is not recognized, display usage.
else { out = shortUsage (); } else { rc = shortUsage (out); }
// Only update the shadow file if such an update was not suppressed (shadow), // Only update the shadow file if such an update was not suppressed (shadow),
if (cmd.isWriteCommand () && !inShadow) if (cmd.isWriteCommand () && !inShadow)
shadow (); shadow ();
return out; return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
@@ -252,8 +257,6 @@ void Context::shadow ()
std::string oldCurses = config.get ("curses"); std::string oldCurses = config.get ("curses");
std::string oldColor = config.get ("color"); std::string oldColor = config.get ("color");
config.set ("curses", "off");
config.set ("color", "off");
clear (); clear ();
@@ -264,8 +267,12 @@ void Context::shadow ()
split (args, command, ' '); split (args, command, ' ');
initialize (); initialize ();
config.set ("curses", "off");
config.set ("color", "off");
parse (); parse ();
std::string result = dispatch (); std::string result;
(void)dispatch (result);
std::ofstream out (shadowFile.c_str ()); std::ofstream out (shadowFile.c_str ());
if (out.good ()) if (out.good ())
{ {
@@ -433,6 +440,7 @@ void Context::loadCorrectConfigFile ()
n.getUntilEOS (value)) n.getUntilEOS (value))
{ {
config.set (name, value); config.set (name, value);
overrides += " " + *arg;
footnote (std::string ("Configuration override ") + // TODO i18n footnote (std::string ("Configuration override ") + // TODO i18n
arg->substr (3, std::string::npos)); arg->substr (3, std::string::npos));
} }
@@ -649,12 +657,13 @@ void Context::parse (
// then invoke the default command. // then invoke the default command.
if (parseCmd.command == "" && parseArgs.size () == 0) if (parseCmd.command == "" && parseArgs.size () == 0)
{ {
std::string defaultCommand = config.get ("default.command"); // Apply overrides, if any.
std::string defaultCommand = config.get ("default.command") + overrides;
if (defaultCommand != "") if (defaultCommand != "")
{ {
// Stuff the command line. // Stuff the command line.
parseArgs.clear (); args.clear ();
split (parseArgs, defaultCommand, ' '); split (args, defaultCommand, ' ');
header ("[task " + defaultCommand + "]"); header ("[task " + defaultCommand + "]");
// Reinitialize the context and recurse. // Reinitialize the context and recurse.

View File

@@ -50,7 +50,7 @@ public:
void initialize (); // for reinitializing void initialize (); // for reinitializing
int run (); // task classic int run (); // task classic
int interactive (); // task interactive (not implemented) int interactive (); // task interactive (not implemented)
std::string dispatch (); // command handler dispatch int dispatch (std::string&); // command handler dispatch
void shadow (); // shadow file update void shadow (); // shadow file update
int getWidth (); // determine terminal width int getWidth (); // determine terminal width
@@ -83,6 +83,7 @@ public:
StringTable stringtable; StringTable stringtable;
std::string program; std::string program;
std::vector <std::string> args; std::vector <std::string> args;
std::string overrides;
Cmd cmd; Cmd cmd;
std::map <std::string, std::string> aliases; std::map <std::string, std::string> aliases;
std::vector <std::string> tagAdditions; std::vector <std::string> tagAdditions;

View File

@@ -29,6 +29,7 @@
#include <time.h> #include <time.h>
#include <assert.h> #include <assert.h>
#include <stdlib.h> #include <stdlib.h>
#include <ctype.h>
#include "Date.h" #include "Date.h"
#include "text.h" #include "text.h"
#include "util.h" #include "util.h"
@@ -82,14 +83,14 @@ Date::Date (const std::string& mdy, const std::string& format /* = "m/d/Y" */)
// Single or double digit. // Single or double digit.
case 'm': case 'm':
if (i >= mdy.length () || if (i >= mdy.length () ||
! ::isdigit (mdy[i])) ! isdigit (mdy[i]))
{ {
throw std::string ("\"") + mdy + "\" is not a valid date."; throw std::string ("\"") + mdy + "\" is not a valid date.";
} }
if (i + 1 < mdy.length () && if (i + 1 < mdy.length () &&
(mdy[i + 0] == '0' || mdy[i + 0] == '1') && (mdy[i + 0] == '0' || mdy[i + 0] == '1') &&
::isdigit (mdy[i + 1])) isdigit (mdy[i + 1]))
{ {
month = ::atoi (mdy.substr (i, 2).c_str ()); month = ::atoi (mdy.substr (i, 2).c_str ());
i += 2; i += 2;
@@ -103,14 +104,14 @@ Date::Date (const std::string& mdy, const std::string& format /* = "m/d/Y" */)
case 'd': case 'd':
if (i >= mdy.length () || if (i >= mdy.length () ||
! ::isdigit (mdy[i])) ! isdigit (mdy[i]))
{ {
throw std::string ("\"") + mdy + "\" is not a valid date."; throw std::string ("\"") + mdy + "\" is not a valid date.";
} }
if (i + 1 < mdy.length () && if (i + 1 < mdy.length () &&
(mdy[i + 0] == '0' || mdy[i + 0] == '1' || mdy[i + 0] == '2' || mdy[i + 0] == '3') && (mdy[i + 0] == '0' || mdy[i + 0] == '1' || mdy[i + 0] == '2' || mdy[i + 0] == '3') &&
::isdigit (mdy[i + 1])) isdigit (mdy[i + 1]))
{ {
day = ::atoi (mdy.substr (i, 2).c_str ()); day = ::atoi (mdy.substr (i, 2).c_str ());
i += 2; i += 2;
@@ -125,8 +126,8 @@ Date::Date (const std::string& mdy, const std::string& format /* = "m/d/Y" */)
// Double digit. // Double digit.
case 'y': case 'y':
if (i + 1 >= mdy.length () || if (i + 1 >= mdy.length () ||
! ::isdigit (mdy[i + 0]) || ! isdigit (mdy[i + 0]) ||
! ::isdigit (mdy[i + 1])) ! isdigit (mdy[i + 1]))
{ {
throw std::string ("\"") + mdy + "\" is not a valid date."; throw std::string ("\"") + mdy + "\" is not a valid date.";
} }
@@ -137,8 +138,8 @@ Date::Date (const std::string& mdy, const std::string& format /* = "m/d/Y" */)
case 'M': case 'M':
if (i + 1 >= mdy.length () || if (i + 1 >= mdy.length () ||
! ::isdigit (mdy[i + 0]) || ! isdigit (mdy[i + 0]) ||
! ::isdigit (mdy[i + 1])) ! isdigit (mdy[i + 1]))
{ {
throw std::string ("\"") + mdy + "\" is not a valid date."; throw std::string ("\"") + mdy + "\" is not a valid date.";
} }
@@ -149,8 +150,8 @@ Date::Date (const std::string& mdy, const std::string& format /* = "m/d/Y" */)
case 'D': case 'D':
if (i + 1 >= mdy.length () || if (i + 1 >= mdy.length () ||
! ::isdigit (mdy[i + 0]) || ! isdigit (mdy[i + 0]) ||
! ::isdigit (mdy[i + 1])) ! isdigit (mdy[i + 1]))
{ {
throw std::string ("\"") + mdy + "\" is not a valid date."; throw std::string ("\"") + mdy + "\" is not a valid date.";
} }
@@ -162,10 +163,10 @@ Date::Date (const std::string& mdy, const std::string& format /* = "m/d/Y" */)
// Quadruple digit. // Quadruple digit.
case 'Y': case 'Y':
if (i + 3 >= mdy.length () || if (i + 3 >= mdy.length () ||
! ::isdigit (mdy[i + 0]) || ! isdigit (mdy[i + 0]) ||
! ::isdigit (mdy[i + 1]) || ! isdigit (mdy[i + 1]) ||
! ::isdigit (mdy[i + 2]) || ! isdigit (mdy[i + 2]) ||
! ::isdigit (mdy[i + 3])) ! isdigit (mdy[i + 3]))
{ {
throw std::string ("\"") + mdy + "\" is not a valid date."; throw std::string ("\"") + mdy + "\" is not a valid date.";
} }

View File

@@ -243,7 +243,7 @@ bool Nibbler::getInt (int& result)
++i; ++i;
} }
while (i < mInput.length () && ::isdigit (mInput[i])) while (i < mInput.length () && isdigit (mInput[i]))
++i; ++i;
if (i > mCursor) if (i > mCursor)
@@ -260,7 +260,7 @@ bool Nibbler::getInt (int& result)
bool Nibbler::getUnsignedInt (int& result) bool Nibbler::getUnsignedInt (int& result)
{ {
std::string::size_type i = mCursor; std::string::size_type i = mCursor;
while (i < mInput.length () && ::isdigit (mInput[i])) while (i < mInput.length () && isdigit (mInput[i]))
++i; ++i;
if (i > mCursor) if (i > mCursor)

View File

@@ -57,8 +57,15 @@ bool Permission::confirmed (const Task& task, const std::string& question)
<< task.id << task.id
<< " \"" << " \""
<< task.get ("description") << task.get ("description")
<< "\"" << "\"";
<< std::endl;
if (task.getStatus () == Task::recurring ||
task.has ("parent"))
{
std::cout << " (Recurring)";
}
std::cout << std::endl;
int answer = confirm3 (question); int answer = confirm3 (question);
if (answer == 2) if (answer == 2)

View File

@@ -153,7 +153,7 @@ bool Sequence::validId (const std::string& input) const
return false; return false;
for (size_t i = 0; i < input.length (); ++i) for (size_t i = 0; i < input.length (); ++i)
if (!::isdigit (input[i])) if (!isdigit (input[i]))
return false; return false;
return true; return true;

View File

@@ -876,7 +876,7 @@ void Table::sort (std::vector <int>& order)
if (gap > 1) if (gap > 1)
{ {
gap = (int) ((float)gap / 1.3); gap = (int) ((float)gap / 1.3);
if (gap == 10 or gap == 9) if (gap == 10 || gap == 9)
gap = 11; gap = 11;
} }

View File

@@ -134,7 +134,7 @@ void Task::setEntry ()
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
Task::status Task::getStatus () Task::status Task::getStatus () const
{ {
return textToStatus (get ("status")); // No i18n return textToStatus (get ("status")); // No i18n
} }
@@ -441,10 +441,14 @@ void Task::addAnnotation (const std::string& description)
void Task::removeAnnotations () void Task::removeAnnotations ()
{ {
// Erase old annotations. // Erase old annotations.
Record::iterator i; Record::iterator i = this->begin ();
for (i = this->begin (); i != this->end (); ++i) while (i != this->end ())
{
if (i->first.substr (0, 11) == "annotation_") // No i18n if (i->first.substr (0, 11) == "annotation_") // No i18n
this->erase (i); this->erase (i++);
else
i++;
}
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////

View File

@@ -57,7 +57,7 @@ public:
void setEntry (); void setEntry ();
status getStatus (); status getStatus () const;
void setStatus (status); void setStatus (status);
int getTagCount (); int getTagCount ();

View File

@@ -52,7 +52,7 @@ Timer::~Timer ()
<< mDescription << mDescription
<< " " << " "
<< std::setprecision (6) << std::setprecision (6)
<< std::fixed // << std::fixed
<< ((end.tv_sec - mStart.tv_sec) + ((end.tv_usec - mStart.tv_usec ) << ((end.tv_sec - mStart.tv_sec) + ((end.tv_usec - mStart.tv_usec )
/ 1000000.0)) / 1000000.0))
<< " sec"; << " sec";

View File

@@ -49,7 +49,7 @@
extern Context context; extern Context context;
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleAdd () int handleAdd (std::string &outs)
{ {
std::stringstream out; std::stringstream out;
@@ -110,12 +110,14 @@ std::string handleAdd ()
context.tdb.commit (); context.tdb.commit ();
context.tdb.unlock (); context.tdb.unlock ();
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleProjects () int handleProjects (std::string &outs)
{ {
int rc = 0;
std::stringstream out; std::stringstream out;
context.filter.push_back (Att ("status", "pending")); context.filter.push_back (Att ("status", "pending"));
@@ -195,15 +197,18 @@ std::string handleProjects ()
<< " (" << quantity << (quantity == 1 ? " task" : " tasks") << ")" << " (" << quantity << (quantity == 1 ? " task" : " tasks") << ")"
<< std::endl; << std::endl;
} }
else else {
out << "No projects." out << "No projects."
<< std::endl; << std::endl;
rc = 1;
}
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleCompletionProjects () int handleCompletionProjects (std::string &outs)
{ {
std::vector <Task> tasks; std::vector <Task> tasks;
context.tdb.lock (context.config.get ("locking", true)); context.tdb.lock (context.config.get ("locking", true));
@@ -228,12 +233,14 @@ std::string handleCompletionProjects ()
if (project->first.length ()) if (project->first.length ())
out << project->first << std::endl; out << project->first << std::endl;
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleTags () int handleTags (std::string &outs)
{ {
int rc = 0;
std::stringstream out; std::stringstream out;
context.filter.push_back (Att ("status", "pending")); context.filter.push_back (Att ("status", "pending"));
@@ -290,15 +297,19 @@ std::string handleTags ()
<< " (" << quantity << (quantity == 1 ? " task" : " tasks") << ")" << " (" << quantity << (quantity == 1 ? " task" : " tasks") << ")"
<< std::endl; << std::endl;
} }
else else {
out << "No tags." out << "No tags."
<< std::endl; << std::endl;
rc = 1;
}
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleCompletionTags () int handleCompletionTags (std::string &outs)
{ {
std::vector <Task> tasks; std::vector <Task> tasks;
context.tdb.lock (context.config.get ("locking", true)); context.tdb.lock (context.config.get ("locking", true));
@@ -328,11 +339,12 @@ std::string handleCompletionTags ()
foreach (tag, unique) foreach (tag, unique)
out << tag->first << std::endl; out << tag->first << std::endl;
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleCompletionCommands () int handleCompletionCommands (std::string &outs)
{ {
std::vector <std::string> commands; std::vector <std::string> commands;
context.cmd.allCommands (commands); context.cmd.allCommands (commands);
@@ -342,11 +354,12 @@ std::string handleCompletionCommands ()
foreach (command, commands) foreach (command, commands)
out << *command << std::endl; out << *command << std::endl;
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleCompletionConfig () int handleCompletionConfig (std::string &outs)
{ {
std::vector <std::string> configs; std::vector <std::string> configs;
context.config.all (configs); context.config.all (configs);
@@ -356,11 +369,12 @@ std::string handleCompletionConfig ()
foreach (config, configs) foreach (config, configs)
out << *config << std::endl; out << *config << std::endl;
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleCompletionIDs () int handleCompletionIDs (std::string &outs)
{ {
std::vector <Task> tasks; std::vector <Task> tasks;
context.tdb.lock (context.config.get ("locking", true)); context.tdb.lock (context.config.get ("locking", true));
@@ -381,7 +395,8 @@ std::string handleCompletionIDs ()
foreach (id, ids) foreach (id, ids)
out << *id << std::endl; out << *id << std::endl;
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
@@ -395,8 +410,9 @@ void handleUndo ()
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleVersion () int handleVersion (std::string &outs)
{ {
int rc = 0;
std::stringstream out; std::stringstream out;
// Create a table for the disclaimer. // Create a table for the disclaimer.
@@ -483,8 +499,8 @@ std::string handleVersion ()
" blanklines bulk color color.active color.due color.overdue color.pri.H " " blanklines bulk color color.active color.due color.overdue color.pri.H "
"color.pri.L color.pri.M color.pri.none color.recurring color.tagged " "color.pri.L color.pri.M color.pri.none color.recurring color.tagged "
"color.footnote color.header color.debug confirmation curses data.location " "color.footnote color.header color.debug confirmation curses data.location "
"dateformat debug default.command default.priority defaultwidth due locale " "dateformat debug default.command default.priority default.project defaultwidth "
"displayweeknumber echo.command locking monthsperline nag next project " "due locale displayweeknumber echo.command locking monthsperline nag next project "
"shadow.command shadow.file shadow.notify weekstart editor import.synonym.id " "shadow.command shadow.file shadow.notify weekstart editor import.synonym.id "
"import.synonym.uuid longversion complete.all.projects complete.all.tags " "import.synonym.uuid longversion complete.all.projects complete.all.tags "
#ifdef FEATURE_SHELL #ifdef FEATURE_SHELL
@@ -524,7 +540,7 @@ std::string handleVersion ()
if (unrecognized.size ()) if (unrecognized.size ())
{ {
out << "Your .taskrc file contains these unrecognized variables:" out << "Your .taskrc file contains these unrecognized variables:"
<< std::endl; << std::endl;
foreach (i, unrecognized) foreach (i, unrecognized)
out << " " << *i << std::endl; out << " " << *i << std::endl;
@@ -535,9 +551,11 @@ std::string handleVersion ()
// Verify installation. This is mentioned in the documentation as the way to // Verify installation. This is mentioned in the documentation as the way to
// ensure everything is properly installed. // ensure everything is properly installed.
if (all.size () == 0) if (all.size () == 0) {
out << "Configuration error: .taskrc contains no entries" out << "Configuration error: .taskrc contains no entries"
<< std::endl; << std::endl;
rc = 1;
}
else else
{ {
if (context.config.get ("data.location") == "") if (context.config.get ("data.location") == "")
@@ -551,12 +569,14 @@ std::string handleVersion ()
<< std::endl; << std::endl;
} }
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleDelete () int handleDelete (std::string &outs)
{ {
int rc = 0;
std::stringstream out; std::stringstream out;
context.disallowModification (); context.disallowModification ();
@@ -645,19 +665,23 @@ std::string handleDelete ()
<< std::endl; << std::endl;
} }
} }
else else {
out << "Task not deleted." << std::endl; out << "Task not deleted." << std::endl;
rc = 1;
}
} }
context.tdb.commit (); context.tdb.commit ();
context.tdb.unlock (); context.tdb.unlock ();
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleStart () int handleStart (std::string &outs)
{ {
int rc = 0;
std::stringstream out; std::stringstream out;
context.disallowModification (); context.disallowModification ();
@@ -699,18 +723,21 @@ std::string handleStart ()
<< task->get ("description") << task->get ("description")
<< "' already started." << "' already started."
<< std::endl; << std::endl;
rc = 1;
} }
} }
context.tdb.commit (); context.tdb.commit ();
context.tdb.unlock (); context.tdb.unlock ();
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleStop () int handleStop (std::string &outs)
{ {
int rc = 0;
std::stringstream out; std::stringstream out;
context.disallowModification (); context.disallowModification ();
@@ -746,18 +773,21 @@ std::string handleStop ()
<< task->get ("description") << task->get ("description")
<< "' not started." << "' not started."
<< std::endl; << std::endl;
rc = 1;
} }
} }
context.tdb.commit (); context.tdb.commit ();
context.tdb.unlock (); context.tdb.unlock ();
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleDone () int handleDone (std::string &outs)
{ {
int rc = 0;
int count = 0; int count = 0;
std::stringstream out; std::stringstream out;
@@ -826,6 +856,7 @@ std::string handleDone ()
<< task->get ("description") << task->get ("description")
<< "' is not pending" << "' is not pending"
<< std::endl; << std::endl;
rc = 1;
} }
context.tdb.commit (); context.tdb.commit ();
@@ -839,11 +870,12 @@ std::string handleDone ()
<< " as done" << " as done"
<< std::endl; << std::endl;
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleExport () int handleExport (std::string &outs)
{ {
std::stringstream out; std::stringstream out;
@@ -882,11 +914,12 @@ std::string handleExport ()
} }
} }
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleModify () int handleModify (std::string &outs)
{ {
int count = 0; int count = 0;
std::stringstream out; std::stringstream out;
@@ -925,6 +958,13 @@ std::string handleModify ()
task->get ("parent") == other->get ("parent")) || // Sibling task->get ("parent") == other->get ("parent")) || // Sibling
other->get ("uuid") == task->get ("parent")) // Parent other->get ("uuid") == task->get ("parent")) // Parent
{ {
if (task->has ("parent"))
std::cout << "Task "
<< task->id
<< " is a recurring task, and all other instances of this"
<< " task may be modified."
<< std::endl;
Task before (*other); Task before (*other);
// A non-zero value forces a file write. // A non-zero value forces a file write.
@@ -959,11 +999,12 @@ std::string handleModify ()
if (context.config.get ("echo.command", true)) if (context.config.get ("echo.command", true))
out << "Modified " << count << " task" << (count == 1 ? "" : "s") << std::endl; out << "Modified " << count << " task" << (count == 1 ? "" : "s") << std::endl;
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleAppend () int handleAppend (std::string &outs)
{ {
int count = 0; int count = 0;
std::stringstream out; std::stringstream out;
@@ -1026,11 +1067,12 @@ std::string handleAppend ()
if (context.config.get ("echo.command", true)) if (context.config.get ("echo.command", true))
out << "Appended " << count << " task" << (count == 1 ? "" : "s") << std::endl; out << "Appended " << count << " task" << (count == 1 ? "" : "s") << std::endl;
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleDuplicate () int handleDuplicate (std::string &outs)
{ {
std::stringstream out; std::stringstream out;
int count = 0; int count = 0;
@@ -1095,7 +1137,8 @@ std::string handleDuplicate ()
if (context.config.get ("echo.command", true)) if (context.config.get ("echo.command", true))
out << "Duplicated " << count << " task" << (count == 1 ? "" : "s") << std::endl; out << "Duplicated " << count << " task" << (count == 1 ? "" : "s") << std::endl;
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
@@ -1174,8 +1217,9 @@ void handleShell ()
#endif #endif
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleColor () int handleColor (std::string &outs)
{ {
int rc = 0;
std::stringstream out; std::stringstream out;
if (context.config.get ("color", true) || context.config.get (std::string ("_forcecolor"), false)) if (context.config.get ("color", true) || context.config.get (std::string ("_forcecolor"), false))
{ {
@@ -1257,13 +1301,15 @@ std::string handleColor ()
out << "Color is currently turned off in your .taskrc file. " out << "Color is currently turned off in your .taskrc file. "
"To enable color, create the entry 'color=on'." "To enable color, create the entry 'color=on'."
<< std::endl; << std::endl;
rc = 1;
} }
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleAnnotate () int handleAnnotate (std::string &outs)
{ {
if (!context.task.has ("description")) if (!context.task.has ("description"))
throw std::string ("Cannot apply a blank annotation."); throw std::string ("Cannot apply a blank annotation.");
@@ -1307,7 +1353,8 @@ std::string handleAnnotate ()
context.tdb.commit (); context.tdb.commit ();
context.tdb.unlock (); context.tdb.unlock ();
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////

View File

@@ -52,7 +52,7 @@ static std::vector <std::string> customReports;
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// This report will eventually become the one report that many others morph into // This report will eventually become the one report that many others morph into
// via the .taskrc file. // via the .taskrc file.
std::string handleCustomReport (const std::string& report) int handleCustomReport (const std::string& report, std::string &outs)
{ {
// Load report configuration. // Load report configuration.
std::string columnList = context.config.get ("report." + report + ".columns"); std::string columnList = context.config.get ("report." + report + ".columns");
@@ -94,21 +94,24 @@ std::string handleCustomReport (const std::string& report)
labelList, labelList,
sortList, sortList,
filterList, filterList,
tasks); tasks,
outs);
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// This report will eventually become the one report that many others morph into // This report will eventually become the one report that many others morph into
// via the .taskrc file. // via the .taskrc file.
std::string runCustomReport ( int runCustomReport (
const std::string& report, const std::string& report,
const std::string& columnList, const std::string& columnList,
const std::string& labelList, const std::string& labelList,
const std::string& sortList, const std::string& sortList,
const std::string& filterList, const std::string& filterList,
std::vector <Task>& tasks) std::vector <Task>& tasks,
std::string &outs)
{ {
int rc = 0;
// Load report configuration. // Load report configuration.
std::vector <std::string> columns; std::vector <std::string> columns;
split (columns, columnList, ','); split (columns, columnList, ',');
@@ -547,11 +550,14 @@ std::string runCustomReport (
<< table.rowCount () << table.rowCount ()
<< (table.rowCount () == 1 ? " task" : " tasks") << (table.rowCount () == 1 ? " task" : " tasks")
<< std::endl; << std::endl;
else else {
out << "No matches." out << "No matches."
<< std::endl; << std::endl;
rc = 1;
}
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////

View File

@@ -150,7 +150,7 @@ static std::string formatTask (Task task)
<< " Due: " << formatDate (task, "due") << std::endl << " Due: " << formatDate (task, "due") << std::endl
<< " Until: " << formatDate (task, "until") << std::endl << " Until: " << formatDate (task, "until") << std::endl
<< " Recur: " << task.get ("recur") << std::endl << " Recur: " << task.get ("recur") << std::endl
<< " Wait until: " << task.get ("wait") << std::endl << " Wait until: " << formatDate (task, "wait") << std::endl
<< " Parent: " << task.get ("parent") << std::endl << " Parent: " << task.get ("parent") << std::endl
<< " Foreground color: " << task.get ("fg") << std::endl << " Foreground color: " << task.get ("fg") << std::endl
<< " Background color: " << task.get ("bg") << std::endl << " Background color: " << task.get ("bg") << std::endl
@@ -543,7 +543,7 @@ void editFile (Task& task)
// Complete the command line. // Complete the command line.
editor += " "; editor += " ";
editor += file.str (); editor += "\"" + file.str () + "\"";
ARE_THESE_REALLY_HARMFUL: ARE_THESE_REALLY_HARMFUL:
// Launch the editor. // Launch the editor.
@@ -599,7 +599,7 @@ ARE_THESE_REALLY_HARMFUL:
// Introducing the Silver Bullet. This feature is the catch-all fixative for // Introducing the Silver Bullet. This feature is the catch-all fixative for
// various other ills. This is like opening up the hood and going in with a // various other ills. This is like opening up the hood and going in with a
// wrench. To be used sparingly. // wrench. To be used sparingly.
std::string handleEdit () int handleEdit (std::string &outs)
{ {
std::stringstream out; std::stringstream out;
@@ -643,7 +643,8 @@ std::string handleEdit ()
context.tdb.commit (); context.tdb.commit ();
context.tdb.unlock (); context.tdb.unlock ();
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////

View File

@@ -106,16 +106,16 @@ static fileType determineFileType (const std::vector <std::string>& lines)
{ {
if ( lines[i][0] == 'x' && if ( lines[i][0] == 'x' &&
lines[i][1] == ' ' && lines[i][1] == ' ' &&
::isdigit (lines[i][2]) && isdigit (lines[i][2]) &&
::isdigit (lines[i][3]) && isdigit (lines[i][3]) &&
::isdigit (lines[i][4]) && isdigit (lines[i][4]) &&
::isdigit (lines[i][5]) && isdigit (lines[i][5]) &&
lines[i][6] == '-' && lines[i][6] == '-' &&
::isdigit (lines[i][7]) && isdigit (lines[i][7]) &&
::isdigit (lines[i][8]) && isdigit (lines[i][8]) &&
lines[i][9] == '-' && lines[i][9] == '-' &&
::isdigit (lines[i][10]) && isdigit (lines[i][10]) &&
::isdigit (lines[i][11])) isdigit (lines[i][11]))
return todo_sh_2_0; return todo_sh_2_0;
} }
@@ -126,13 +126,13 @@ static fileType determineFileType (const std::vector <std::string>& lines)
// +project // +project
if (words[w].length () > 1 && if (words[w].length () > 1 &&
words[w][0] == '+' && words[w][0] == '+' &&
::isalnum (words[w][1])) isalnum (words[w][1]))
return todo_sh_2_0; return todo_sh_2_0;
// @context // @context
if (words[w].length () > 1 && if (words[w].length () > 1 &&
words[w][0] == '@' && words[w][0] == '@' &&
::isalnum (words[w][1])) isalnum (words[w][1]))
return todo_sh_2_0; return todo_sh_2_0;
} }
} }
@@ -666,6 +666,7 @@ static std::string importTask_1_6_0 (const std::vector <std::string>& lines)
static std::string importTaskCmdLine (const std::vector <std::string>& lines) static std::string importTaskCmdLine (const std::vector <std::string>& lines)
{ {
std::vector <std::string> failed; std::vector <std::string> failed;
std::string unused;
std::vector <std::string>::const_iterator it; std::vector <std::string>::const_iterator it;
for (it = lines.begin (); it != lines.end (); ++it) for (it = lines.begin (); it != lines.end (); ++it)
@@ -680,7 +681,7 @@ static std::string importTaskCmdLine (const std::vector <std::string>& lines)
context.task.clear (); context.task.clear ();
context.cmd.command = ""; context.cmd.command = "";
context.parse (); context.parse ();
handleAdd (); (void)handleAdd (unused);
context.clearMessages (); context.clearMessages ();
} }
@@ -789,6 +790,8 @@ static std::string importTodoSh_2_0 (const std::vector <std::string>& lines)
context.parse (); context.parse ();
decorateTask (context.task); decorateTask (context.task);
context.task.set ("uuid", uuid ());
if (isPending) if (isPending)
{ {
context.task.setStatus (Task::pending); context.task.setStatus (Task::pending);
@@ -1144,7 +1147,7 @@ static std::string importCSV (const std::vector <std::string>& lines)
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleImport () int handleImport (std::string &outs)
{ {
std::stringstream out; std::stringstream out;
@@ -1212,7 +1215,8 @@ std::string handleImport ()
else else
throw std::string ("You must specify a file to import."); throw std::string ("You must specify a file to import.");
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////

View File

@@ -55,25 +55,25 @@ int getDueState (const std::string&);
bool nag (Task&); bool nag (Task&);
// command.cpp // command.cpp
std::string handleAdd (); int handleAdd (std::string &);
std::string handleAppend (); int handleAppend (std::string &);
std::string handleExport (); int handleExport (std::string &);
std::string handleDone (); int handleDone (std::string &);
std::string handleModify (); int handleModify (std::string &);
std::string handleProjects (); int handleProjects (std::string &);
std::string handleCompletionProjects (); int handleCompletionProjects (std::string &);
std::string handleTags (); int handleTags (std::string &);
std::string handleCompletionTags (); int handleCompletionTags (std::string &);
std::string handleCompletionCommands (); int handleCompletionCommands (std::string &);
std::string handleCompletionIDs (); int handleCompletionIDs (std::string &);
std::string handleCompletionConfig (); int handleCompletionConfig (std::string &);
std::string handleVersion (); int handleVersion (std::string &);
std::string handleDelete (); int handleDelete (std::string &);
std::string handleStart (); int handleStart (std::string &);
std::string handleStop (); int handleStop (std::string &);
std::string handleColor (); int handleColor (std::string &);
std::string handleAnnotate (); int handleAnnotate (std::string &);
std::string handleDuplicate (); int handleDuplicate (std::string &);
void handleUndo (); void handleUndo ();
#ifdef FEATURE_SHELL #ifdef FEATURE_SHELL
void handleShell (); void handleShell ();
@@ -85,27 +85,28 @@ int deltaAttributes (Task&);
int deltaSubstitutions (Task&); int deltaSubstitutions (Task&);
// edit.cpp // edit.cpp
std::string handleEdit (); int handleEdit (std::string &);
// report.cpp // report.cpp
std::string shortUsage (); int shortUsage (std::string &);
std::string longUsage (); int longUsage (std::string &);
std::string handleInfo (); int handleInfo (std::string &);
std::string handleReportSummary (); int handleReportSummary (std::string &);
std::string handleReportNext (); int handleReportNext (std::string &);
std::string handleReportHistory (); int handleReportHistory (std::string &);
std::string handleReportGHistory (); int handleReportGHistory (std::string &);
std::string handleReportCalendar (); int handleReportCalendar (std::string &);
std::string handleReportStats (); int handleReportStats (std::string &);
std::string handleReportTimesheet (); int handleReportTimesheet (std::string &);
std::string getFullDescription (Task&); std::string getFullDescription (Task&);
std::string getDueDate (Task&); std::string getDueDate (Task&);
// custom.cpp // custom.cpp
std::string handleCustomReport (const std::string&); int handleCustomReport (const std::string&, std::string &);
std::string runCustomReport (const std::string&, const std::string&, int runCustomReport (const std::string&, const std::string&,
const std::string&, const std::string&, const std::string&, const std::string&,
const std::string&, std::vector <Task>&); const std::string&, std::vector <Task>&,
std::string&);
void validReportColumns (const std::vector <std::string>&); void validReportColumns (const std::vector <std::string>&);
void validSortColumns (const std::vector <std::string>&, const std::vector <std::string>&); void validSortColumns (const std::vector <std::string>&, const std::vector <std::string>&);
@@ -118,7 +119,7 @@ std::string colorizeFootnote (const std::string&);
std::string colorizeDebug (const std::string&); std::string colorizeDebug (const std::string&);
// import.cpp // import.cpp
std::string handleImport (); int handleImport (std::string&);
// list template // list template
/////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////

View File

@@ -50,7 +50,7 @@
extern Context context; extern Context context;
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string shortUsage () int shortUsage (std::string &outs)
{ {
Table table; Table table;
@@ -209,14 +209,19 @@ std::string shortUsage ()
<< std::endl << std::endl
<< std::endl; << std::endl;
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string longUsage () int longUsage (std::string &outs)
{ {
std::string shortUsageString;
std::stringstream out; std::stringstream out;
out << shortUsage ()
(void)shortUsage(shortUsageString);
out << shortUsageString
<< "ID is the numeric identifier displayed by the 'task list' command. " << "ID is the numeric identifier displayed by the 'task list' command. "
<< "You can specify multiple IDs for task commands, and multiple tasks " << "You can specify multiple IDs for task commands, and multiple tasks "
<< "will be affected. To specify multiple IDs make sure you use one " << "will be affected. To specify multiple IDs make sure you use one "
@@ -276,13 +281,15 @@ std::string longUsage ()
<< " $ ! ' \" ( ) ; \\ ` * ? { } [ ] < > | & % # ~" << "\n" << " $ ! ' \" ( ) ; \\ ` * ? { } [ ] < > | & % # ~" << "\n"
<< std::endl; << std::endl;
return out.str (); outs = out.str();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// Display all information for the given task. // Display all information for the given task.
std::string handleInfo () int handleInfo (std::string &outs)
{ {
int rc = 0;
// Get all the tasks. // Get all the tasks.
std::vector <Task> tasks; std::vector <Task> tasks;
context.tdb.lock (context.config.get ("locking", true)); context.tdb.lock (context.config.get ("locking", true));
@@ -497,18 +504,22 @@ std::string handleInfo ()
<< std::endl; << std::endl;
} }
if (! tasks.size ()) if (! tasks.size ()) {
out << "No matches." << std::endl; out << "No matches." << std::endl;
rc = 1;
}
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// Project Remaining Avg Age Complete 0% 100% // Project Remaining Avg Age Complete 0% 100%
// A 12 13d 55% XXXXXXXXXXXXX----------- // A 12 13d 55% XXXXXXXXXXXXX-----------
// B 109 3d 12h 10% XXX--------------------- // B 109 3d 12h 10% XXX---------------------
std::string handleReportSummary () int handleReportSummary (std::string &outs)
{ {
int rc = 0;
// Scan the pending tasks. // Scan the pending tasks.
std::vector <Task> tasks; std::vector <Task> tasks;
context.tdb.lock (context.config.get ("locking", true)); context.tdb.lock (context.config.get ("locking", true));
@@ -648,10 +659,13 @@ std::string handleReportSummary ()
<< table.rowCount () << table.rowCount ()
<< (table.rowCount () == 1 ? " project" : " projects") << (table.rowCount () == 1 ? " project" : " projects")
<< std::endl; << std::endl;
else else {
out << "No projects." << std::endl; out << "No projects." << std::endl;
rc = 1;
}
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
@@ -673,7 +687,7 @@ std::string handleReportSummary ()
// //
// Make the "three" tasks a configurable number // Make the "three" tasks a configurable number
// //
std::string handleReportNext () int handleReportNext (std::string &outs)
{ {
// Load report configuration. // Load report configuration.
std::string columnList = context.config.get ("report.next.columns"); std::string columnList = context.config.get ("report.next.columns");
@@ -718,7 +732,8 @@ std::string handleReportNext ()
labelList, labelList,
sortList, sortList,
filterList, filterList,
tasks); tasks,
outs);
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
@@ -744,8 +759,9 @@ time_t monthlyEpoch (const std::string& date)
return 0; return 0;
} }
std::string handleReportHistory () int handleReportHistory (std::string &outs)
{ {
int rc = 0;
std::map <time_t, int> groups; // Represents any month with data std::map <time_t, int> groups; // Represents any month with data
std::map <time_t, int> addedGroup; // Additions by month std::map <time_t, int> addedGroup; // Additions by month
std::map <time_t, int> completedGroup; // Completions by month std::map <time_t, int> completedGroup; // Completions by month
@@ -891,15 +907,19 @@ std::string handleReportHistory ()
out << optionalBlankLine () out << optionalBlankLine ()
<< table.render () << table.render ()
<< std::endl; << std::endl;
else else {
out << "No tasks." << std::endl; out << "No tasks." << std::endl;
rc = 1;
}
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleReportGHistory () int handleReportGHistory (std::string &outs)
{ {
int rc = 0;
std::map <time_t, int> groups; // Represents any month with data std::map <time_t, int> groups; // Represents any month with data
std::map <time_t, int> addedGroup; // Additions by month std::map <time_t, int> addedGroup; // Additions by month
std::map <time_t, int> completedGroup; // Completions by month std::map <time_t, int> completedGroup; // Completions by month
@@ -1087,14 +1107,17 @@ std::string handleReportGHistory ()
else else
out << "Legend: + added, X completed, - deleted" << std::endl; out << "Legend: + added, X completed, - deleted" << std::endl;
} }
else else {
out << "No tasks." << std::endl; out << "No tasks." << std::endl;
rc = 1;
}
return out.str (); outs = out.str ();
return rc;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleReportTimesheet () int handleReportTimesheet (std::string &outs)
{ {
// Scan the pending tasks. // Scan the pending tasks.
std::vector <Task> tasks; std::vector <Task> tasks;
@@ -1264,7 +1287,8 @@ std::string handleReportTimesheet ()
end -= 7 * 86400; end -= 7 * 86400;
} }
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
@@ -1376,9 +1400,9 @@ std::string renderMonths (
row = 0; row = 0;
// Loop through days in month and add to table. // Loop through days in month and add to table.
for (int d = 1; d <= daysInMonth.at (mpl); ++d) for (int d = 1; d <= daysInMonth[mpl]; ++d)
{ {
Date temp (months.at (mpl), d, years.at (mpl)); Date temp (months[mpl], d, years[mpl]);
int dow = temp.dayOfWeek (); int dow = temp.dayOfWeek ();
int woy = temp.weekOfYear (weekStart); int woy = temp.weekOfYear (weekStart);
@@ -1396,9 +1420,9 @@ std::string renderMonths (
table.addCell (row, thisCol, d); table.addCell (row, thisCol, d);
if ((context.config.get ("color", true) || context.config.get (std::string ("_forcecolor"), false)) && if ((context.config.get ("color", true) || context.config.get (std::string ("_forcecolor"), false)) &&
today.day () == d && today.day () == d &&
today.month () == months.at (mpl) && today.month () == months[mpl] &&
today.year () == years.at (mpl)) today.year () == years[mpl])
table.setCellFg (row, thisCol, Text::cyan); table.setCellFg (row, thisCol, Text::cyan);
foreach (task, all) foreach (task, all)
@@ -1410,8 +1434,8 @@ std::string renderMonths (
if ((context.config.get ("color", true) || context.config.get (std::string ("_forcecolor"), false)) && if ((context.config.get ("color", true) || context.config.get (std::string ("_forcecolor"), false)) &&
due.day () == d && due.day () == d &&
due.month () == months.at (mpl) && due.month () == months[mpl] &&
due.year () == years.at (mpl)) due.year () == years[mpl])
{ {
table.setCellFg (row, thisCol, Text::black); table.setCellFg (row, thisCol, Text::black);
table.setCellBg (row, thisCol, due < today ? Text::on_red : Text::on_yellow); table.setCellBg (row, thisCol, due < today ? Text::on_red : Text::on_yellow);
@@ -1423,7 +1447,7 @@ std::string renderMonths (
int eow = 6; int eow = 6;
if (weekStart == 1) if (weekStart == 1)
eow = 0; eow = 0;
if (dow == eow && d < daysInMonth.at (mpl)) if (dow == eow && d < daysInMonth[mpl])
row++; row++;
} }
} }
@@ -1432,7 +1456,7 @@ std::string renderMonths (
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleReportCalendar () int handleReportCalendar (std::string &outs)
{ {
// Each month requires 28 text columns width. See how many will actually // Each month requires 28 text columns width. See how many will actually
// fit. But if a preference is specified, and it fits, use it. // fit. But if a preference is specified, and it fits, use it.
@@ -1604,11 +1628,12 @@ std::string handleReportCalendar ()
<< optionalBlankLine () << optionalBlankLine ()
<< std::endl; << std::endl;
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
std::string handleReportStats () int handleReportStats (std::string &outs)
{ {
std::stringstream out; std::stringstream out;
@@ -1836,7 +1861,8 @@ std::string handleReportStats ()
<< table.render () << table.render ()
<< optionalBlankLine (); << optionalBlankLine ();
return out.str (); outs = out.str ();
return 0;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////

View File

@@ -1,83 +0,0 @@
////////////////////////////////////////////////////////////////////////////////
// task - a command line task list manager.
//
// Copyright 2006 - 2009, Paul Beckingham.
// All rights reserved.
//
// This program is free software; you can redistribute it and/or modify it under
// the terms of the GNU General Public License as published by the Free Software
// Foundation; either version 2 of the License, or (at your option) any later
// version.
//
// This program is distributed in the hope that it will be useful, but WITHOUT
// ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
// FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
// details.
//
// You should have received a copy of the GNU General Public License along with
// this program; if not, write to the
//
// Free Software Foundation, Inc.,
// 51 Franklin Street, Fifth Floor,
// Boston, MA
// 02110-1301
// USA
//
////////////////////////////////////////////////////////////////////////////////
#ifndef INCLUDED_TASK
#define INCLUDED_TASK
#include <string>
#include "Record.h"
#include "Subst.h"
#include "Sequence.h"
class Task : public Record
{
public:
Task (); // Default constructor
Task (const Task&); // Copy constructor
Task& operator= (const Task&); // Assignment operator
bool operator== (const Task&); // Comparison operator
Task (const std::string&); // Parse
~Task (); // Destructor
void parse (const std::string&);
std::string composeCSV () const;
// Status values.
enum status {pending, completed, deleted, recurring, waiting};
// Public data.
int id;
// Series of helper functions.
static status textToStatus (const std::string&);
static std::string statusToText (status);
void setEntry ();
status getStatus ();
void setStatus (status);
int getTagCount ();
bool hasTag (const std::string&);
void addTag (const std::string&);
void addTags (const std::vector <std::string>&);
void getTags (std::vector<std::string>&) const;
void removeTag (const std::string&);
void getAnnotations (std::vector <Att>&) const;
void setAnnotations (const std::vector <Att>&);
void addAnnotation (const std::string&);
void removeAnnotations ();
void validate () const;
private:
int determineVersion (const std::string&);
void legacyParse (const std::string&);
};
#endif
////////////////////////////////////////////////////////////////////////////////

View File

@@ -40,7 +40,7 @@ if (open my $fh, '>', 'basic.rc')
# Test the usage command. # Test the usage command.
my $output = qx{../task rc:basic.rc}; my $output = qx{../task rc:basic.rc};
like ($output, qr/You must specify a command, or a task ID to modify/, 'missing command and ID'); like ($output, qr/You must specify a command, or a task ID to modify/m, 'missing command and ID');
# Test the version command. # Test the version command.
$output = qx{../task rc:basic.rc version}; $output = qx{../task rc:basic.rc version};

View File

@@ -58,7 +58,7 @@ if ( $day <= 9)
# task cal and task cal y # task cal and task cal y
my $output = qx{../task rc:cal.rc rc._forcecolor:on cal}; my $output = qx{../task rc:cal.rc rc._forcecolor:on cal};
like ($output, qr/\[36m$day/, 'Current day is highlighted'); like ($output, qr/\[36m$day/, 'Current day is highlighted');
like ($output, qr/$month.* $year/, 'Current month and year are displayed'); like ($output, qr/$month\w+?\s+?$year/, 'Current month and year are displayed');
qx{../task rc:cal.rc add zero}; qx{../task rc:cal.rc add zero};
unlike ($output, qr/\[41m\d+/, 'No overdue tasks are present'); unlike ($output, qr/\[41m\d+/, 'No overdue tasks are present');
unlike ($output, qr/\[43m\d+/, 'No due tasks are present'); unlike ($output, qr/\[43m\d+/, 'No due tasks are present');
@@ -67,9 +67,9 @@ like ($output, qr/Su Mo Tu/, 'Week starts on Sunday');
$output = qx{../task rc:cal.rc rc.weekstart:Monday cal}; $output = qx{../task rc:cal.rc rc.weekstart:Monday cal};
like ($output, qr/Fr Sa Su/, 'Week starts on Monday'); like ($output, qr/Fr Sa Su/, 'Week starts on Monday');
$output = qx{../task rc:cal.rc cal y}; $output = qx{../task rc:cal.rc cal y};
like ($output, qr/$month.* $year/, 'Current month and year are displayed'); like ($output, qr/$month\w+?\s+?$year/, 'Current month and year are displayed');
like ($output, qr/$prevmonth.* $nextyear/, 'Month and year one year ahead are displayed'); like ($output, qr/$prevmonth\w+?\s+?$nextyear/, 'Month and year one year ahead are displayed');
unlike ($output, qr/$month.* $nextyear/, 'Current month and year ahead are not displayed'); unlike ($output, qr/$month\w+?\s+?$nextyear/, 'Current month and year ahead are not displayed');
# task cal due and task cal due y # task cal due and task cal due y
qx{../task rc:cal.rc add due:20190515 one}; qx{../task rc:cal.rc add due:20190515 one};

62
src/tests/feature.exit.t Executable file
View File

@@ -0,0 +1,62 @@
#! /usr/bin/perl
################################################################################
## task - a command line task list manager.
##
## Copyright 2006 - 2009, Paul Beckingham.
## All rights reserved.
##
## This program is free software; you can redistribute it and/or modify it under
## the terms of the GNU General Public License as published by the Free Software
## Foundation; either version 2 of the License, or (at your option) any later
## version.
##
## This program is distributed in the hope that it will be useful, but WITHOUT
## ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
## FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
## details.
##
## You should have received a copy of the GNU General Public License along with
## this program; if not, write to the
##
## Free Software Foundation, Inc.,
## 51 Franklin Street, Fifth Floor,
## Boston, MA
## 02110-1301
## USA
##
################################################################################
use strict;
use warnings;
use Test::More tests => 7;
# Create the rc file.
if (open my $fh, '>', 'exit.rc')
{
print $fh "data.location=.\n",
"confirmation=no\n";
close $fh;
ok (-r 'exit.rc', 'Created exit.rc');
}
qx{../task rc:exit.rc add foo};
my $exit_good = system ('../task rc:exit.rc ls foo 2>&1 >>/dev/null');
is ($exit_good, 0, 'task returns 0 on success');
my $exit_bad = system ('../task rc:exit.rc ls bar 2>&1 >>/dev/null');
isnt ($exit_bad, 0, 'task returns non-zero on failure');
# Cleanup.
unlink 'pending.data';
ok (!-r 'pending.data', 'Removed pending.data');
unlink 'completed.data';
ok (!-r 'completed.data', 'Removed completed.data');
unlink 'undo.data';
ok (!-r 'undo.data', 'Removed undo.data');
unlink 'exit.rc';
ok (!-r 'exit.rc', 'Removed exit.rc');
exit 0;

View File

@@ -1,4 +1,4 @@
#! /bin/bash #! /bin/sh
date > all.log date > all.log
@@ -14,7 +14,7 @@ END=`tail -1 all.log`
OS=`uname` OS=`uname`
case $OS in case $OS in
Darwin) Darwin | FreeBSD)
STARTEPOCH=`date -j -f "%a %b %d %T %Z %Y" "${START}" "+%s"` STARTEPOCH=`date -j -f "%a %b %d %T %Z %Y" "${START}" "+%s"`
ENDEPOCH=`date -j -f "%a %b %d %T %Z %Y" "${END}" "+%s"` ENDEPOCH=`date -j -f "%a %b %d %T %Z %Y" "${END}" "+%s"`
;; ;;

View File

@@ -209,7 +209,7 @@ std::string commify (const std::string& data)
int i; int i;
for (int i = 0; i < (int) data.length (); ++i) for (int i = 0; i < (int) data.length (); ++i)
{ {
if (::isdigit (data[i])) if (isdigit (data[i]))
end = i; end = i;
if (data[i] == '.') if (data[i] == '.')
@@ -227,11 +227,11 @@ std::string commify (const std::string& data)
int consecutiveDigits = 0; int consecutiveDigits = 0;
for (; i >= 0; --i) for (; i >= 0; --i)
{ {
if (::isdigit (data[i])) if (isdigit (data[i]))
{ {
result += data[i]; result += data[i];
if (++consecutiveDigits == 3 && i && ::isdigit (data[i - 1])) if (++consecutiveDigits == 3 && i && isdigit (data[i - 1]))
{ {
result += ','; result += ',';
consecutiveDigits = 0; consecutiveDigits = 0;
@@ -251,11 +251,11 @@ std::string commify (const std::string& data)
int consecutiveDigits = 0; int consecutiveDigits = 0;
for (; i >= 0; --i) for (; i >= 0; --i)
{ {
if (::isdigit (data[i])) if (isdigit (data[i]))
{ {
result += data[i]; result += data[i];
if (++consecutiveDigits == 3 && i && ::isdigit (data[i - 1])) if (++consecutiveDigits == 3 && i && isdigit (data[i - 1]))
{ {
result += ','; result += ',';
consecutiveDigits = 0; consecutiveDigits = 0;
@@ -279,8 +279,8 @@ std::string lowerCase (const std::string& input)
{ {
std::string output = input; std::string output = input;
for (int i = 0; i < (int) input.length (); ++i) for (int i = 0; i < (int) input.length (); ++i)
if (::isupper (input[i])) if (isupper (input[i]))
output[i] = ::tolower (input[i]); output[i] = tolower (input[i]);
return output; return output;
} }
@@ -290,8 +290,8 @@ std::string upperCase (const std::string& input)
{ {
std::string output = input; std::string output = input;
for (int i = 0; i < (int) input.length (); ++i) for (int i = 0; i < (int) input.length (); ++i)
if (::islower (input[i])) if (islower (input[i]))
output[i] = ::toupper (input[i]); output[i] = toupper (input[i]);
return output; return output;
} }
@@ -302,7 +302,7 @@ std::string ucFirst (const std::string& input)
std::string output = input; std::string output = input;
if (output.length () > 0) if (output.length () > 0)
output[0] = ::toupper (output[0]); output[0] = toupper (output[0]);
return output; return output;
} }
@@ -352,7 +352,7 @@ void guess (
bool digitsOnly (const std::string& input) bool digitsOnly (const std::string& input)
{ {
for (size_t i = 0; i < input.length (); ++i) for (size_t i = 0; i < input.length (); ++i)
if (!::isdigit (input[i])) if (!isdigit (input[i]))
return false; return false;
return true; return true;
@@ -362,7 +362,7 @@ bool digitsOnly (const std::string& input)
bool noSpaces (const std::string& input) bool noSpaces (const std::string& input)
{ {
for (size_t i = 0; i < input.length (); ++i) for (size_t i = 0; i < input.length (); ++i)
if (::isspace (input[i])) if (isspace (input[i]))
return false; return false;
return true; return true;

View File

@@ -64,7 +64,7 @@ bool confirm (const std::string& question)
<< " "; << " ";
std::getline (std::cin, answer); std::getline (std::cin, answer);
answer = lowerCase (trim (answer)); answer = std::cin.eof() ? "no" : lowerCase (trim (answer));
} }
while (answer != "y" && // TODO i18n while (answer != "y" && // TODO i18n
answer != "ye" && // TODO i18n answer != "ye" && // TODO i18n