By: Team SE-EDU
Since: Jun 2016
Licence: MIT
-
JDK
9
or later⚠️ JDK 10
on Windows will fail to run tests in headless mode due to a JavaFX bug. Windows developers are highly recommended to use JDK9
. -
IntelliJ IDE
ℹ️IntelliJ by default has Gradle and JavaFx plugins installed.
Do not disable them. If you have disabled them, go toFile
>Settings
>Plugins
to re-enable them.
-
Fork this repo, and clone the fork to your computer
-
Open IntelliJ (if you are not in the welcome screen, click
File
>Close Project
to close the existing project dialog first) -
Set up the correct JDK version for Gradle
-
Click
Configure
>Project Defaults
>Project Structure
-
Click
New…
and find the directory of the JDK
-
-
Click
Import Project
-
Locate the
build.gradle
file and select it. ClickOK
-
Click
Open as Project
-
Click
OK
to accept the default settings -
Open a console and run the command
gradlew processResources
(Mac/Linux:./gradlew processResources
). It should finish with theBUILD SUCCESSFUL
message.
This will generate all resources required by the application and tests. -
Open
XmlAdaptedWish.java
andMainWindow.java
and check for any code errors-
Due to an ongoing issue with some of the newer versions of IntelliJ, code errors may be detected even if the project can be built and run successfully
-
To resolve this, place your cursor over any of the code section highlighted in red. Press ALT+ENTER, and select
Add '--add-modules=…' to module compiler options
for each error
-
-
Repeat this for the test folder as well (e.g. check
XmlUtilTest.java
andHelpWindowTest.java
for code errors, and if so, resolve it the same way)
-
Run the
seedu.url.MainApp
and try a few commands -
Run the tests to ensure they all pass.
This project follows oss-generic coding standards. IntelliJ’s default style is mostly compliant with ours but it uses a different import order from ours. To rectify,
-
Go to
File
>Settings…
(Windows/Linux), orIntelliJ IDEA
>Preferences…
(macOS) -
Select
Editor
>Code Style
>Java
-
Click on the
Imports
tab to set the order-
For
Class count to use import with '*'
andNames count to use static import with '*'
: Set to999
to prevent IntelliJ from contracting the import statements -
For
Import Layout
: The order isimport static all other imports
,import java.*
,import javax.*
,import org.*
,import com.*
,import all other imports
. Add a<blank line>
between eachimport
-
Optionally, you can follow the UsingCheckstyle.adoc document to configure Intellij to check style-compliance as you write code.
After forking the repo, the documentation will still have the SE-EDU branding and refer to the CS2103-AY1819S1-T16-1/main
repo.
If you plan to develop this fork as a separate product (i.e. instead of contributing to CS2103-AY1819S1-T16-1/main
), you should do the following:
-
Configure the site-wide documentation settings in
build.gradle
, such as thesite-name
, to suit your own project. -
Replace the URL in the attribute
repoURL
inDeveloperGuide.adoc
andUserGuide.adoc
with the URL of your fork.
Set up Travis to perform Continuous Integration (CI) for your fork. See UsingTravis.adoc to learn how to set it up.
After setting up Travis, you can optionally set up coverage reporting for your team fork (see UsingCoveralls.adoc).
ℹ️
|
Coverage reporting could be useful for a team repository that hosts the final version but it is not that useful for your personal fork. |
Optionally, you can set up AppVeyor as a second CI (see UsingAppVeyor.adoc).
ℹ️
|
Having both Travis and AppVeyor ensures your App works on both Unix-based platforms and Windows-based platforms (Travis is Unix-based and AppVeyor is Windows-based) |
When you are ready to start coding,
-
Get some sense of the overall design by reading Section 2.1, “Architecture”.
-
Take a look at [GetStartedProgramming].
The Architecture Diagram given above explains the high-level design of the App. Given below is a quick overview of each component.
💡
|
The .pptx files used to create diagrams in this document can be found in the diagrams folder. To update a diagram, modify the diagram in the pptx file, select the objects of the diagram, and choose Save as picture .
|
Main
has only one class called MainApp
. It is responsible for,
-
At app launch: Initializes the components in the correct sequence, and connects them up with each other.
-
At shut down: Shuts down the components and invokes cleanup method where necessary.
Commons
represents a collection of classes used by multiple other components. Two of those classes play important roles at the architecture level.
-
EventsCenter
: This class (written using Google’s Event Bus library) is used by components to communicate with other components using events (i.e. a form of Event Driven design) -
LogsCenter
: Used by many classes to write log messages to the App’s log file.
The rest of the App consists of four components.
Each of the four components
-
Defines its API in an
interface
with the same name as the Component. -
Exposes its functionality using a
{Component Name}Manager
class.
For example, the Logic
component (see the class diagram given below) defines it’s API in the Logic.java
interface and exposes its functionality using the LogicManager.java
class.
The Sequence Diagram below shows how the components interact for the scenario where the user issues the command delete 1
.
ℹ️
|
Note how the Model simply raises a WishBookChangedEvent when the Wish Book data is changed, instead of asking the Storage to save the updates to the hard disk. This event also triggers the save of wish histories to disk.
|
The diagram below shows how the EventsCenter
reacts to that event, which eventually results in the updates being saved to the hard disk and the status bar of the UI being updated to reflect the 'Last Updated' time.
ℹ️
|
Note how the event is propagated through the EventsCenter to the Storage and UI without Model having to be coupled to either of them. This is an example of how this Event Driven approach helps us reduce direct coupling between components.
|
The sections below give more details of each component.
API : Ui.java
The UI consists of a MainWindow
that is made up of parts e.g.CommandBox
, ResultDisplay
, WishListPanel
, StatusBarFooter
, BrowserPanel
etc. All these, including the MainWindow
, inherit from the abstract UiPart
class.
The UI
component uses JavaFx UI framework. The layout of these UI parts are defined in matching .fxml
files that are in the src/main/resources/view
folder. For example, the layout of the MainWindow
is specified in MainWindow.fxml
The UI
component,
-
Executes user commands using the
Logic
component. -
Binds itself to some data in the
Model
so that the UI can auto-update when data in theModel
change. -
Responds to events raised from various parts of the App and updates the UI accordingly.
API :
Logic.java
-
Logic
uses theWishBookParser
class to parse the user command. -
This results in a
Command
object which is executed by theLogicManager
. -
The command execution can affect the
Model
(e.g. adding a wish) and/or raise events. -
The result of the command execution is encapsulated as a
CommandResult
object which is passed back to theUi
.
Given below is the Sequence Diagram for interactions within the Logic
component for the execute("delete 1")
API call.
API : Model.java
The Model
,
-
stores a
UserPref
object that represents the user’s preferences. -
stores the Wish Book data.
-
stores the data of wish histories.
-
exposes an unmodifiable
ObservableList<Wish>
that can be 'observed' e.g. the UI can be bound to this list so that the UI automatically updates when the data in the list change. The elements of theObservableList<Wish>
can also be filtered and sorted to suit the needs of specific commands. -
does not depend on any of the other three components.
API : Storage.java
The Storage
component,
-
can save
UserPref
objects in json format and read it back. -
can save the Wish Book data in xml format and read it back.
This section describes some noteworthy details on how certain features are implemented.
A wish is uniquely identified by its Universal Unique Identifier (UUID) which is generated randomly only once for a
particular wish, upon its creation through the AddCommand
. A wish stores the following primary attributes:
-
Name
-
Price
-
Date
-
Saved Amount
-
Url
-
Remark
-
Tags
-
UUID
ℹ️
|
It is impossible for the user to create a duplicate wish as it is impossible to modify a wish’s UUID. |
A wish needs to be prioritised in a specific order such that the wishes with the highest priority will be visible
on the top of the list. In WishBook, the priority is determined primarily by the due date of the wish which is
stored in every wish’s Date
attribute. Ties are broken by Name
. Further ties are broken by UUID
as it is possible
for the Date
and Name
of two wishes to be identical.
The sorting of the displayed results is done by the filteredSortedWishes
list. The sorting order is specified by
WishComparator
.
-
Alternative 1(current choice): Identify a
Wish
by a randomly generated UUID.-
Pros: Extremely low probability of collision.
-
Pros: No extra maintenance required upon generation as every
Wish
is unique. -
Cons: UUID does not map to any real world entity and it is used strictly for identification.
-
Cons: It is more difficult to system test the
AddCommand
with the current group of methods for system tests as UUID is randomly generated each time.
-
-
Alternative 2: Identify a wish by
Name
,Price
,Date
,Url
,Tags
. Wishes with identical values for these attributes will be represented by a singleWishCard
. TheWishCard
will be augmented with aMultiplicity
to indicate the number of identical wishes.-
Pros: WishBook will be more compact and every attribute stored in a
Wish
maps to a real entity. -
Cons: Additional attribute
Multiplicity
may have to be frequently edited as it is another attribute that is affected by multiple commands.
-
-
Alternative 3: Identify a wish by a new attribute
CreatedTime
, which is derived from the system time when the wish is created.-
Pros: The attribute maps to a real entity. It can be an additional information presented to the user about a wish.
-
Cons: There might be collisions in
CreatedTime
if the the system time is incorrect.
-
The undo/redo mechanism is facilitated by VersionedWishBook
.
It extends WishBook
with an undo/redo history, stored internally as an wishBookStateList
and currentStatePointer
.
Additionally, it implements the following operations:
-
VersionedWishBook#commit()
— Saves the current wish book state in its history. -
VersionedWishBook#undo()
— Restores the previous wish book state from its history. -
VersionedWishBook#redo()
— Restores a previously undone wish book state from its history.
These operations are exposed in the Model
interface as Model#commitWishBook()
, Model#undoWishBook()
and Model#redoWishBook()
respectively.
Given below is an example usage scenario and how the undo/redo mechanism behaves at each step.
Step 1. The user launches the application for the first time. The VersionedWishBook
will be initialized with the initial wish book state, and the currentStatePointer
pointing to that single url book state.
Step 2. The user executes delete 5
command to delete the 5th wish in the wish book. The delete
command calls Model#commitWishBook()
, causing the modified state of the url book after the delete 5
command executes to be saved in the wishBookStateList
, and the currentStatePointer
is shifted to the newly inserted url book state.
Step 3. The user executes add n/David …
to add a new wish. The add
command also calls Model#commitWishBook()
, causing another modified wish book state to be saved into the wishBookStateList
.
ℹ️
|
If a command fails its execution, it will not call Model#commitWishBook() , so the wish book state will not be saved into the wishBookStateList .
|
Step 4. The user now decides that adding the wish was a mistake, and decides to undo that action by executing the undo
command. The undo
command will call Model#undoWishBook()
, which will shift the currentStatePointer
once to the left, pointing it to the previous wish book state, and restores the url book to that state.
ℹ️
|
If the currentStatePointer is at index 0, pointing to the initial wish book state, then there are no previous url book states to restore. The undo command uses Model#canUndoWishBook() to check if this is the case. If so, it will return an error to the user rather than attempting to perform the undo.
|
The following sequence diagram shows how the undo operation works:
The redo
command does the opposite — it calls Model#redoWishBook()
, which shifts the currentStatePointer
once to the right, pointing to the previously undone state, and restores the wish book to that state.
ℹ️
|
If the currentStatePointer is at index wishBookStateList.size() - 1 , pointing to the latest wish book state, then there are no undone url book states to restore. The redo command uses Model#canRedoWishBook() to check if this is the case. If so, it will return an error to the user rather than attempting to perform the redo.
|
Step 5. The user then decides to execute the command list
. Commands that do not modify the wish book, such as list
, will usually not call Model#commitWishBook()
, Model#undoWishBook()
or Model#redoWishBook()
. Thus, the wishBookStateList
remains unchanged.
Step 6. The user executes clear
, which calls Model#commitWishBook()
. Since the currentStatePointer
is not pointing at the end of the wishBookStateList
, all wish book states after the currentStatePointer
will be purged. We designed it this way because it no longer makes sense to redo the add n/David …
command. This is the behavior that most modern desktop applications follow.
The following activity diagram summarizes what happens when a user executes a new command:
-
Alternative 1 (current choice): Saves the entire wish book.
-
Pros: Easy to implement.
-
Cons: May have performance issues in terms of memory usage.
-
-
Alternative 2: Individual command knows how to undo/redo by itself.
-
Pros: Will use less memory (e.g. for
delete
, just save the wish being deleted). -
Cons: We must ensure that the implementation of each individual command are correct.
-
-
Alternative 1 (current choice): Use a list to store the history of wish book states.
-
Pros: Easy for new Computer Science student undergraduates to understand, who are likely to be the new incoming developers of our project.
-
Cons: Logic is duplicated twice. For example, when a new command is executed, we must remember to update both
HistoryManager
andVersionedWishBook
.
-
-
Alternative 2: Use
HistoryManager
for undo/redo-
Pros: We do not need to maintain a separate list, and just reuse what is already in the codebase.
-
Cons: Requires dealing with commands that have already been undone: We must remember to skip these commands. Violates Single Responsibility Principle and Separation of Concerns as
HistoryManager
now needs to do two different things.
-
The current state of the savings history of the WishBook
is captured by VersionedWishTransaction
.
VersionedWishTransaction
extends WishTransaction
and has an undo/redo history, similar to the implementation of the Undo/Redo feature, and is stored internally as a wishStateList
and currentStatePointer
. Additionally, it implements VersionedModel
and so contains the implementation of the following operations:
-
VersionedWishTransaction#commit()
— Saves the current wish transaction state in its history. -
VersionedWishTransaction#undo()
— Restores the previous wish transaction state from its history. -
VersionedWishTransaction#redo()
— Restores a previously undone wish transaction state from its history.
These operations are exposed in the Model
interface as Model#commitWishBook()
, Model#undoWishBook()
and Model#redoWishBook()
respectively.
WishTransaction
keeps track of the state of all wishes in WishBook
via a wishMap
which maps the unique ID of a Wish
to a list of Wish
states. WishTransaction
implements ActionCommandListener
such that any state changing command performed to a Wish
or the WishBook
such as AddCommand()
, EditCommand()
, SaveCommand()
, etc will result in the WishMap
being updated accordingly in WishTransaction
.
VersionedWishTransaction
, WishTransaction
can be easily converted to and from xml using XmlWishTransactions
. XmlWishTransactions
is saved as an xml file when the user explicitly closes the window, thereby invoking MainApp#stop()
which saves the current state of VersionedWishTransaction
in the wishStateList
to hard disk.
If the user’s command triggers a change in the state of the WishBook
, a WishBookChangedEvent
will be raised, causing the subscribed StorageManager
to respond by saving both the current state of the WishBook
and WishTransaction
to disk.
Given below is an example usage scenario and how the savings history mechanism behaves at each step.
Step 1. The user launches the application. The default file path storing the previous state of the WishTransaction
will be retrieved, unless otherwise specified by the user, and the contents from the xml file will be parsed and converted into a WishTransaction
object via the XmlWishTransactions
object. If the file at the specified location is behind the current state of the WishBook
, content of the WishTransaction
will be overwritten by the WishBook
.
ℹ️
|
The wishStateList starts off with the initial state of the WishTransaction as the first item in the list.
|
Step 2. The user executes add n/iPhone …
to add a new wish. The add
command calls Model#commitWishBook()
, causing the current state of the modified wish transaction state to be saved into wishStateList
. As this is a command that changes the state of the WishBook
, Model#addWish()
will call VersionedWishTransaction#addWish()
to add a new wish to the WishMap
.
-
If a command fails its execution, it will not call
Model#commitWishBook()
, so the wish transaction state will not be saved into thewishStateList
. -
If the
WishMap
contains an identical wish (such is identified byWish#isSameWish()
), then the call to add this wish will fail. As such, the wish will not be added to theWishMap
or theWishBook
.
Step 3. The user now decides that adding the wish was a mistake, and decides to undo that action by executing the undo
command. The undo
command will call Model#undoWishBook()
, which will shift the currentStatePointer
once to the left, pointing it to the previous wish transaction state, and restores the wish transaction to that state.
ℹ️
|
If the currentStatePointer is at index 0, pointing to the initial wish transaction state, then there are no previous wish transaction states to restore. The undo command uses Model#canUndoWishBook() to check if this is the case. If so, it will return an error to the user rather than attempting to perform the undo.
|
The redo
command does the opposite — it calls Model#redoWishBook()
, which shifts the currentStatePointer
once to the right, pointing to the previously undone state, and restores the wish transaction to that state.
ℹ️
|
If the currentStatePointer is at index wishStateList.size() - 1 , pointing to the latest wish transaction state, then there are no undone wish transaction states to restore. The redo command uses Model#canRedoWishBook() to check if this is the case. If so, it will return an error to the user rather than attempting to perform the redo.
|
Step 4. The user then decides to execute the command list
. Commands that do not modify the state of the WishBook
, such as list
, will usually not call Model#commitWishBook()
, Model#undoWishBook()
or Model#redoWishBook()
. Thus, the wishBookStateList
remains unchanged.
Step 5. The user finally exits the app by clicking on the close button. The most recent state of the WishTransaction
will be converted into xml format via the the XmlWishTransactions
object and be saved into the same file path it was first retrieved from.
ℹ️
|
If there was some error saving the current state of the WishTransaction to the specified file path in hard disk, an exception will be thrown and a warning will be shown to the user. The current state of the WishTransaction object will not be saved to hard disk.
|
The Add Wish feature is executed through an AddCommand
by the user, which after parsing,
is facilitated mainly by the ModelManager
which implements Model
.
It also affects versionedWishBook
and versionedWishTransaction
by adding the
resultant wish to both of their respective data structures.
After adding a Wish
, the filteredSortedWishes
is also updated to reflect the
latest version of WishBook.
The UI is also prompted to refresh through a WishBookChangedEvent
.
AddCommandParser
parses the user’s input for parameters using prefixes,
and checks them against their respective regular expressions (regex), specified
in their respective classes.
The following prefix/parameter pairs are compulsory, where a user’s input will be rejected if they are not provided:
-
n/
: Name -
p/
: Price -
One of the following Date parameters:
-
d/
: Exact expiry date -
a/
: duration (or lifetime) from time when command is entered
-
The following prefix/parameter pairs are optional, where a user’s input will be successful even if they are not provided:
-
t/
: tags (more than one allowed) -
u/
: product’s URL (product page)
ℹ️
|
Regarding Duration (
|
Given below is an example usage scenario and how an AddCommand is carried out.
Step 1. The user types in a valid AddCommand
, add n/1 TB Toshiba SSD p/158 a/200d
, and the current date is 2nd October 2017 (2/10/2017).
The AddCommandParser
will employ ParserUtil
to parse the attributes specified after each prefix. The parsing of the
Duration
attribute which follows a/
in the command will be discussed below.
Since Duration
prefix is used, the computation of a wish’s expiry date is handled
internally in the ParserUtil
class, which ParserUtil#parseDate()
parses and converts
the input string into a Period
object (if input is valid),
and adds the resultant Period
to the current Date
to get the desired
Date
of the Wish
.
The resultant Wish
will have the following properties:
-
id:
a randomly-generated UUID
-
Name: 1TB Toshiba SSD
-
SavedAmount: 0.00
-
Price: 158.00
-
Date: 20/4/2018 (20th April 2018)
-
URL:
empty string
-
Remark:
empty string
-
Tags:
none
-
Fulfilled:
false
-
Expired:
false
The resultant wish is pass into VersionedWishBook#addWish
and VersionedWishTransaction#addWish
,
which tracks the history of the WishBook
and Wish
respectively. The list of wishes shown on the UI is also updated to show all wishes again,
as filteredSortedWishes
is updated to have all wishes in WishBook
and a WishBookChangedEvent
is fired.
The following sequence diagram shows how an AddCommand
is processed in WB:
Step 2. Some time later, the user decides that she wants the exact same wish,
but duplicated, and enters the exact same command, but with an exact Date
instead of Duration
, so the command entered is
add n/1 TB Toshiba SSD p/158 d/20/4/2018
.
Since Date
prefix is used, the ParserUtil
parses the string into a Date
object,
and the resultant object is used directly for the resultant Wish
.
Similar to in Step 1, the command will be parsed successfully and a second Wish
will be added, albeit with a different (hidden) id generated.
The resultant Wish
will have the following properties:
-
id:
another randomly-generated UUID
-
Name: 1TB Toshiba SSD
-
SavedAmount: 0.00
-
Price: 158.00
-
Date: 20/4/2018 (20th April 2018)
-
URL:
empty string
-
Remark:
empty string
-
Tags:
none
-
Fulfilled:
false
-
Expired:
false
-
Alternative 1 (current choice): Different prefixes for
Duration
andDate
.-
Pros: More focused user experience. User get more specific feedback depending on their preferred way of inputting date if a wrong input was made. If user uses
a/
and enters an incorrectDuration
, the user will not receive an error message about the correct format for an exactDate
, and will only be notified of the correct format of aDuration
. -
Pros: Easier to implement and handle isolate errors related to respective input parameters.
-
Cons: More prefixes for user to remember.
-
-
Alternative 2: Have
Duration
andDate
use the same prefix.-
Pros: More natural usage of one prefix to determine
Wish
's desired expiry date. -
Cons: Conflating implementation of
Duration
andDate
, hence harder to debug. -
Cons: Tricky to implement, as we are parsing one input for two different desired formats.
-
The Save Amount feature is executed through a SaveCommand
by the user, which after parsing,
is facilitated mainly by the ModelManager
which implements Model
.
Wish stores the price
and savedAmount
of Wish
, helping to track the progress of the savings towards the price
.
Meanwhile, WishBook stores an unusedFunds
, which is an unallocated pool of funds that can be used in the future.
After adding a saving, the filteredSortedWishes
in ModelManager
is updated to reflect the latest observable WishBook.
Given below is an example usage scenario and how the SaveCommand behaves at each step:
Step 1. The user executes save 1 10
, to save $10 into an existing wish with Index
1 and Price
$15. The $10 is
wrapped in an Amount
and a SaveCommand
instance is created with the Amount
. Amount
is then used to make an updated
instance of the Wish
at index 1 whose SavedAmount
will be updated. Model#updateWish
is then called to update this
wish with the old one in WishBook
.
ℹ️
|
The Index of each Wish is labelled at the side of the app.
|
The resultant wish will have the following properties:
-
Name: 1TB Toshiba SSD
-
SavedAmount: 10.00
-
Price: 15.00
-
Date: 20/4/2018 (20th April 2018)
-
URL:
empty string
-
Remark:
empty string
-
Tags:
none
-
Fulfilled:
false
-
Expired:
false
ℹ️
|
Amount to be saved can be a negative value where it would mean a withdrawal of money from a particular wish.
|
ℹ️
|
SavedAmount of a wish cannot be negative. This means that an Amount cannot be negative enough to cause SavedAmount
to be negative.
|
Step 2. The user decides to execute save 1 10
again. However, SaveCommand checks that savedAmount
> price
.
SaveCommand#execute creates a new updated Wish
with savedAmount = wish.getPrice()
.
The resultant wish will have the following properties:
-
Name: 1TB Toshiba SSD
-
SavedAmount: 15.00
-
Price: 15.00
-
Date: 20/4/2018 (20th April 2018)
-
URL:
empty string
-
Remark:
empty string
-
Tags:
none
-
Fulfilled:
true
-
Expired:
false
Step 3. The excess amount of $5 is stored in a new Amount
variable excess
.
SaveCommand#execute then calls Model#updateUnusedFunds(excess) to update the unusedFunds
in WishBook.
In WishBook, the result would be:
-
unusedFunds: 5.00
Step 4. The user tries to execute save 1 10
again. However, since the value for Wish#isFulfilled is true, the amount
will not be saved. SaveCommand#execute will throw a CommandException, with the message "Wish has already been fulfilled!".
The following sequence diagram shows how the save operation works:
-
Alternative 1 (current choice): Store it in a
SavedAmount
variable inWishBook
.-
Pros: Easy to implement.
-
Cons: More methods needed when needing to move funds from
unusedFunds
to other wishes.
-
-
Alternative 2: Store it as a pseudo wish with index 0.
-
Pros: It can be treated as another
wish
, hence existing methods can be used without needing to create much more new ones. -
Cons: Requires dealing with an extra wish that has to be hidden on the
WishListPanel
and displayed separately on the UI. We must remember to skip this wish in methods that involve displaying the WishList.
-
The find mechanism is supported by FindCommandParser
. It implements Parser
that implements the following operation:
-
FindCommandParser#parse()
— Checks the arguments for empty strings and throws aParseException
if empty string is found. It then splits the arguments usingArgumentTokenizer#tokenize()
and returns anArgumentMultimap
. Keywords of the same prefix are then grouped usingArgumentMultimap#getAllValues()
.
The find mechanism is also facilitated by FindCommand
. It extends Command
and implements the
following operation:
-
FindCommand#execute()
— Executes the command by updating the currentFilteredSortedWishList
with theWishContainsKeywordPredicate
.
The predicate WishContainsKeywordsPredicate
, takes in three lists of the keywords for the following attributes:
-
Name
-
Tags
-
Remark
and also the isExactMatch
argument. The result of the predicate is determined by checking
whether a Wish
contains the given keywords at their corresponding attributes. The match threshold
is dictated by the value of isExactMatch
.
Given below is an example usage scenario and how the Find mechanism behaves at each step.
Step 1. The user launches the application for the first time.
Step 2. The user executes find n/wat n/apple t/impor
command to get all wishes whose name contains the
keywords 'iphone' or 'tablet'.
Step 3. The FindCommandParser#parse()
is called and the WishContainsKeywordPredicate
is constructed with
the arguments of the find command.
Step 4. FindCommand#execute()
is then called.
Step 5. The entire list of wishes is filtered by the predicate WishContainsKeywordsPredicate
.
Step 6. The filtered list of wishes is returned to the GUI.
-
Alternative 1 (Current choice): Require the user to prepend every keyword argument with the appropriate Wish attribute prefix.
-
Pros: Easier to implement as it easier to match keyword against a Wish if the attribute to match against is known.
-
Pros: User has more control over the results returned.
-
Cons: User is required to type slightly more.
-
-
Alternative 2: No prefixes are required in the arguments. Keywords can match with any one of the following chosen wish attributes:
Name
,Tags
orRemark
.-
Pros: Less typing required from user.
-
Cons: Command might be slightly slower as every keyword has to be checked against all chosen attributes of the wish.
-
Cons: User has less control over the results returned.
-
-
Alternative 1 (Current choice): Keywords appended to different prefixes are grouped with a logical AND and keywords appended to the same prefixes are grouped with a logical OR when being matched against a
Wish
.-
Pros: A more intuitive way to find wishes.
-
Cons: Can be restrictive in some situations.
-
-
Alternative 2: Keywords appended to different prefixes are grouped with a logical OR and keywords appended to the same prefixes are grouped with a logical OR when being matched against a
Wish
.-
Pros: Search results will be more inclusive.
-
Cons: Very slim chance for such a use case.
-
The list -c and list -u
command allows the user to view the list of all wishes, completed and ongoing, respectively.
A wish is completed if the savedAmount is greater or equal to the price of the wish.
Given below is an example usage scenario and how the list overdue mechanism behaves at each step:
-
The user executes the command
list -c
. -
model.updateFilteredWishList()
will update the wish list withWishCompletedPredicate
as the parameter (boolean).wish.isFulfilled()
is called to check whether the wish is completed or not. -
The updated wish list would be reflected on the UI to be displayed to the user.
The following sequence diagram shows how the Wish Detail Panel displays its updated content:
The UI has been redesigned to implement the following UI components required for WishBook:
-
Command Box
-
Wish List Panel
-
Wish Detail Panel
The Wish List Panel consists of a list of Wish Card which contains 4 UI elements:
-
WishCard#nameLabel
- AText
element that displays the wish’s name. -
WishCard#progressLabel
- AText
element that displays the wish’s saving progress in percentage format. -
WishCard#tags
- AFlowPane
element that contains aText
element which displays the wish’s assigned tags. -
WishCard#progressBar
- AprogressBar
element that visually presents the percentage of the wish’s current saving progress.
Whenever the user adds a new wish or edits an existing wish, a new WishCard containing the wish will be added to the Wish List Panel or the content in the existing WishCard will be updated respectively.
The user will be able to view the wish’s current saving progress both in terms of text on the progressLabel (e.g. ’80%’) and the progressBar. Also, the user will be able to see all the tags he/she assigned to categorize the wish.
The UI (MainWindow) constructs the WishListPanel
by obtaining an ObservableList
of wish cards from Model
, this list is assigned when UI starts, and will never be re-assigned.
The UI "observes" the list and updates when it is modified.
This approach works well for the WishListPanel
because WishBook contains only 1 list of wish cards.
However, the saving history list in the WishDetailPanel
can not be updated in the same manner because Model component will change its card list’s reference when a user adds a new wish or updates the content of the wish.
In this case, the WishDetailPanel
in UI will not be updated because the card list of which UI has reference to is actually not changed.
-
Alternative 1 (current choice): Have a wishList in
Model
and keep it updated with the current list of cards-
Explanation: The UI needs only 1 reference to this
wishList
, each time user executes any changes,wishList
is cleared and the new list of cards is copy to thewishList
. -
Pros: The structure of
Model
and UI component needs not be changed -
Cons: Need to keep a copy of the current card list, copying the whole list of cards for each command operation has bad effect on performance .
-
-
Alternative 2: Model component raises an event when its current card list’s reference is changed
-
Explanation: When user adds a new wish or executes save,
Model
will raise an event (WishPanelUpdatedEvent
), which is subscribed by UI, then UI can re-assign its list of cards and update the cards panel accordingly. -
Pros: Better performance
-
Cons: Need to re-design
Model
and UI components
-
The Wish Detail Panel consists of 3 UI sub-components:
-
WishDetailSavingAmount
that containsText
elements to display price and the saving progress of the wish -
WishDetailSavingHistory
that contains aList
of history of saving inputs of the wish -
WishBrowserPanel
that displaysWebView
of the URL of the wish.
Whenever the user adds a new wish or edits an existing wish, the content of the wish in Wish Detail Panel will be updated. The user will be able to view the wish’s current saving progress and the history of his/her saving inputs of the wish in the list format. Also, the user will be able to browse through the wish’s product page via its assigned URL.
Every time a new Wish is added or an existing wish is updated by the commands such as save, it raises a WishDataUpdatedEvent
.
The UI will then handle that event and update the WishDetailPanel
with the new version of wish.
Given below is an example usage scenario and how the WishBook behaves and WishDetailPanel
is updated at each step:
-
The user executes the command
save 1 1000
.
ℹ️
|
If a command fails its execution, WishDataUpdatedEvent will not be posted. |
-
The save command updates the model with the new wish and raises a new
WishDataUpdatedEvent
. -
WishDetailSavingAmount
andWishDetailSavingHistory
responds to theWishDataUpdatedEvent
withWishListPanel#handleWishUpdatedEvent()
. -
The
WishDetailSavingAmount
updates the wish’s current saving progress whenWishDetailSavingAmount#loadWishDetails
is called. -
The progress is calculated from when
Wish#getProgress
is called. The value is saveAmount / price. Then the progress label for the wish is set to that fraction. -
The
WishDetailSavingHistory
updates the wish’s saving history list whenWishDetailSavingHistory#loadWishDetails
is called. -
The saving history list is cleared.
-
The new set of history entry is retrieved from
wishTransaction#getWishMap
and the saved amount is calculated from subtracting previous saving amount from the next one. -
The saving history list is now filled with the new list of updated saving history.
The following sequence diagram shows how the Wish Detail Panel displays its updated content:
Aspect: How to update the progress and saving history on UI
-
Alternative 1 (current choice): Clear all the sub components and add new sub components accordingly
-
Pros: No matter which progress or history is changed, or what type of change (ie. delete, add, or edit), this change can be handled by the same method each time.
-
Cons: It is redundant to clear everything and replace them with new sub components.
-
-
Alternative 2: Handle different kinds of changes to the progress or history lists.
-
Pros: It is a lot faster to only change the sub component that is affected.
-
Cons: There are too many cases for how the lists can be changed. (ie. a different change is needed for each of these cases: wish is deleted/edited/created/cleared, or a
wishTransaction
is deleted/added)
-
Some users may have many wishes, all of which have a different targeted date of completion and different price. It may thus be difficult for users to keep track of how much they need to consistently save to fulfil their various wishes on time. This Savings Notification feature will allow users to opt for daily/weekly/monthly notifications for each specific wish, reminding them of the amount that they need to save at the beginning of the chosen time period. This will help users to consistently save towards their wishes.
We are using java.util.logging
package for logging. The LogsCenter
class is used to manage the logging levels and logging destinations.
-
The logging level can be controlled using the
logLevel
setting in the configuration file (See Section 3.11, “Configuration”) -
The
Logger
for a class can be obtained usingLogsCenter.getLogger(Class)
which will log messages according to the specified logging level -
Currently log messages are output through:
Console
and to a.log
file.
Logging Levels
-
SEVERE
: Critical problem detected which may possibly cause the termination of the application -
WARNING
: Can continue, but with caution -
INFO
: Information showing the noteworthy actions by the App -
FINE
: Details that is not usually noteworthy but may be useful in debugging e.g. print the actual list instead of just its size
We use asciidoc for writing documentation.
ℹ️
|
We chose asciidoc over Markdown because asciidoc, although a bit more complex than Markdown, provides more flexibility in formatting. |
See UsingGradle.adoc to learn how to render .adoc
files locally to preview the end result of your edits.
Alternatively, you can download the AsciiDoc plugin for IntelliJ, which allows you to preview the changes you have made to your .adoc
files in real-time.
See UsingTravis.adoc to learn how to deploy GitHub Pages using Travis.
We use Google Chrome for converting documentation to PDF format, as Chrome’s PDF engine preserves hyperlinks used in webpages.
Here are the steps to convert the project documentation files to PDF format.
-
Follow the instructions in UsingGradle.adoc to convert the AsciiDoc files in the
docs/
directory to HTML format. -
Go to your generated HTML files in the
build/docs
folder, right click on them and selectOpen with
→Google Chrome
. -
Within Chrome, click on the
Print
option in Chrome’s menu. -
Set the destination to
Save as PDF
, then clickSave
to save a copy of the file in PDF format. For best results, use the settings indicated in the screenshot below.
The build.gradle
file specifies some project-specific asciidoc attributes which affects how all documentation files within this project are rendered.
💡
|
Attributes left unset in the build.gradle file will use their default value, if any.
|
Attribute name | Description | Default value |
---|---|---|
|
The name of the website. If set, the name will be displayed near the top of the page. |
not set |
|
URL to the site’s repository on GitHub. Setting this will add a "View on GitHub" link in the navigation bar. |
not set |
|
Define this attribute if the project is an official SE-EDU project. This will render the SE-EDU navigation bar at the top of the page, and add some SE-EDU-specific navigation items. |
not set |
Each .adoc
file may also specify some file-specific asciidoc attributes which affects how the file is rendered.
Asciidoctor’s built-in attributes may be specified and used as well.
💡
|
Attributes left unset in .adoc files will use their default value, if any.
|
Attribute name | Description | Default value |
---|---|---|
|
Site section that the document belongs to.
This will cause the associated item in the navigation bar to be highlighted.
One of: * Official SE-EDU projects only |
not set |
|
Set this attribute to remove the site navigation bar. |
not set |
The files in docs/stylesheets
are the CSS stylesheets of the site.
You can modify them to change some properties of the site’s design.
The files in docs/templates
controls the rendering of .adoc
files into HTML5.
These template files are written in a mixture of Ruby and Slim.
|
Modifying the template files in |
There are three ways to run tests.
💡
|
The most reliable way to run tests is the 3rd one. The first two methods might fail some GUI tests due to platform/resolution-specific idiosyncrasies. |
Method 1: Using IntelliJ JUnit test runner
-
To run all tests, right-click on the
src/test/java
folder and chooseRun 'All Tests'
-
To run a subset of tests, you can right-click on a test package, test class, or a test and choose
Run 'ABC'
Method 2: Using Gradle
-
Open a console and run the command
gradlew clean allTests
(Mac/Linux:./gradlew clean allTests
)
ℹ️
|
See UsingGradle.adoc for more info on how to run tests using Gradle. |
Method 3: Using Gradle (headless)
Thanks to the TestFX library we use, our GUI tests can be run in the headless mode. In the headless mode, GUI tests do not show up on the screen. That means the developer can do other things on the Computer while the tests are running.
To run tests in headless mode, open a console and run the command gradlew clean headless allTests
(Mac/Linux: ./gradlew clean headless allTests
)
We have two types of tests:
-
GUI Tests - These are tests involving the GUI. They include,
-
System Tests that test the entire App by simulating user actions on the GUI. These are in the
systemtests
package. -
Unit tests that test the individual components. These are in
seedu.url.ui
package.
-
-
Non-GUI Tests - These are tests not involving the GUI. They include,
-
Unit tests targeting the lowest level methods/classes.
e.g.seedu.url.commons.StringUtilTest
-
Integration tests that are checking the integration of multiple code units (those code units are assumed to be working).
e.g.seedu.url.storage.StorageManagerTest
-
Hybrids of unit and integration tests. These test are checking multiple code units as well as how the are connected together.
e.g.seedu.url.logic.LogicManagerTest
-
See UsingGradle.adoc to learn how to use Gradle for build automation.
We use Travis CI and AppVeyor to perform Continuous Integration on our projects. See UsingTravis.adoc and UsingAppVeyor.adoc for more details.
We use Coveralls to track the code coverage of our projects. See UsingCoveralls.adoc for more details.
When a pull request has changes to asciidoc files, you can use Netlify to see a preview of how the HTML version of those asciidoc files will look like when the pull request is merged. See UsingNetlify.adoc for more details.
Here are the steps to create a new release.
-
Update the version number in
MainApp.java
. -
Generate a JAR file using Gradle.
-
Tag the repo with the version number. e.g.
v0.1
-
Create a new release using GitHub and upload the JAR file you created.
A project often depends on third-party libraries. For example, Wish Book depends on the Jackson library for XML parsing. Managing these dependencies can be automated using Gradle. For example, Gradle can download the dependencies automatically, which is better than these alternatives.
a. Include those libraries in the repo (this bloats the repo size)
b. Require developers to download those libraries manually (this creates extra work for developers)
Target user profile:
-
has a need to manage savings for a significant number of items to buy
-
prefer desktop apps over other types
-
can type fast
-
prefers typing over mouse input
-
is reasonably comfortable using CLI apps
Value proposition: manage savings faster than a typical mouse/GUI driven app
Priorities: High (must have) - * * *
, Medium (nice to have) - * *
, Low (unlikely to have) - *
Priority | As a … | I want to … | So that I can… |
---|---|---|---|
|
new user |
see usage instructions |
refer to instructions when I forget how to use the App |
|
user |
add a new wish |
keep track of the things I want to purchase |
|
user |
add savings to selected wishes |
make faster progress towards certain wishes |
|
user |
delete a wish |
remove items that I no longer need |
|
user |
find a wish by name |
locate details of a wish without having to go through the entire list |
|
user |
view all fulfilled wishes |
so I can keep track of items I have bought |
|
user |
view all past savings for my wishes |
have a better idea of my saving habits in general |
|
user |
view all my wishes |
monitor the progress I have made in all my wishes |
|
user |
undo past commands |
reverse wrong commands |
|
user |
distribute a saving to a few wishes |
make equal progress to a few of my wishes |
|
user |
rank my wishes |
prioritise certain wishes over others so that money can be allocated accordingly |
|
user |
transfer money from one wish to another |
progress towards other wishes faster |
|
user |
withdraw from savings |
spend the money if need be |
|
user |
reorder the priority of a wish |
fulfil the specified wish faster |
|
user |
save money without a wish |
allocate my savings to a wish later |
|
user |
receive email reminders about wishes that are due |
be more mindful of my savings to fulfil wishes |
|
user |
view all past savings for a particular wish |
have a better idea of my saving habits for a wish |
{More to be added}
(For all use cases below, the System is the WishBook
and the Actor is the user
, unless specified otherwise)
MSS
-
Actor enters a wish with Name, Date, Price.
-
System adds wish to the wish list.
Use case ends.
Extensions
-
2a. Actor fails to specify any of the compulsory fields (Name, Price, and Date/Duration).
-
2a1. System shows a correct Add command usage with example.
Use case ends.
-
-
2b. Actor enters incorrectly formatted arguments.
-
2b1. System shows Add command usage.
-
2b2. Actor is prompted to enter a valid argument in a specific format shown.
Use case ends.
-
MSS
-
Actor requests to list wishes.
-
System shows a list of wishes.
-
Actor requests to delete a specific wish in the list.
-
System deletes the wish.
Use case ends.
Extensions
-
2a. The list is empty.
Use case ends.
-
3a. The given index is invalid.
-
3a1. System shows an error message.
Use case resumes at step 2.
-
3b1. Wish requested to be deleted has a non-zero savings amount.
-
3b2. System displays warning to user that wish to be deleted has a non-zero savings amount.
Use case resumes at step 2.
-
MSS
-
Actor requests to edit wish.
-
System updates wish and shows updated wish to Actor.
Use case ends.
Extensions
-
1a. System has no recorded wishes.
-
1a1. Actor is prompted to add a wish.
Use case ends.
-
-
1b. Actor enters invalid arguments
-
1b1. System shows Edit command usage.
-
1b2. Actor is prompted to enter a valid argument.
Use case ends.
-
MSS
-
Actor specifies the search predicate.
-
System shows all wishes matching the given search predicate.
Use case ends.
Extensions
-
1a. System has no recorded wishes.
-
1a1. System shows dialog notifying Actor that no relevant results can be found.
Use case ends.
-
-
1b. Actor enters invalid arguments
-
1b1. System shows Find command usage.
-
1b2. Actor is prompted to enter a valid argument.
Use case ends.
-
-
1c. System unable to find any matching wishes.
-
1c1. System shows dialog notifying Actor that no relevant results can be found.
Use case ends.
-
-
1d. Actor enters empty prefixes for arguments.
-
1c1. System shows all wishes in the WishBook.
Use case ends.
-
MSS
-
Actor specifies the origin wish index, the destination wish index, and amount to move.
-
System moves specified amount from origin wish to destination wish.
Use case ends.
Extensions
-
1a. System has no recorded wishes.
-
1a1. Actor is prompted to add a wish.
Use case ends.
-
-
1b. Actor enters invalid arguments.
-
1b1. System shows Move command usage.
-
1b2. Actor is prompted to enter a valid argument.
Use case ends.
-
-
1c. Actor enters the same index for origin wish and destination wish.
-
1c1. System shows Move command usage.
-
1c2. Actor is prompted to enter a valid argument.
Use case ends.
-
-
1d. The amount to move is greater than the existing saved amount in the origin wish.
-
1d1. System reports that saved amount of wish cannot become negative.
Use case ends.
-
-
1e. The amount moved to destination wish exceeds the amount needed to fulfil the wish.
-
1e1. System moves to destination wish the required amount needed to fulfil it.
-
1e2. System moves the excess amount to unused funds.
Use case ends.
-
-
1f. Actor specifies unused funds as the origin wish.
-
1f1. System moves amount from unused funds to the destination wish.
Use case ends.
-
-
1g. Actor specifies unused funds as the destination wish.
-
1g1. System moves amount from origin wish to unused funds.
Use case ends.
-
MSS
-
Actor enters X amount of money to be saved.
-
System transfers X to the wish with the specified index.
Use case ends.
Extensions
-
1a. Actor specifies to allocate the money to unused funds.
-
1a1. System adds X to unused funds.
Use case ends.
-
-
1b. System has no recorded wishes.
-
1b1. Actor is prompted to add a wish.
Use case ends.
-
-
1c. Actor enters an invalid value of money to be saved.
-
1c1. Actor is prompted to enter a valid value.
Use case ends.
-
-
1d. Actor enters an amount of money that causes the wish’s saved amount to exceed the amount needed to fulfil the wish.
-
1d1. System adds the required amount to fulfil the saved amount of the wish at the specified index.
-
1d2. System adds the excess amount to unused funds.
Use case ends.
-
-
1e. Actor enters an amount that causes the wish’s resulting saved amount to become negative.
-
1e1. Actor is prompted to enter a valid value.
Use case ends.
-
MSS
-
Actor requests to view all wishes.
-
System shows all wishes.
Use case ends.
Extensions
-
1a. System has no recorded wishes.
-
1a1. User is prompted to add a wish.
Use case ends.
-
-
1b. System has no recorded wishes.
-
1b1. System shows dialog notifying Actor that there are no such wishes.
Use case ends.
-
MSS
-
Actor requests to view uncompleted wishes.
-
System shows all uncompleted wishes.
Use case ends.
Extensions
-
1a. System has no recorded uncompleted wishes.
-
1a1. User is prompted to add a wish.
Use case ends.
-
-
1b. System has no recorded uncompleted wishes.
-
1b1. System shows dialog notifying Actor that there are no such wishes.
Use case ends.
-
MSS
-
Actor requests to list completed wishes.
-
System shows a list of completed wishes.
Use case ends.
Extensions
-
2a. List is empty.
Use case ends.
MSS
-
Actor requests to view history of commands entered.
-
System shows all commands entered.
Use case ends.
Extensions
-
1a. System has no recorded commands.
-
1a1. User is prompted to enter a command.
Use case ends.
-
MSS
-
Actor requests to view history of savings entered.
-
System shows all savings entered, from newest to oldest.
Use case ends.
Extensions
-
1a. System has no recorded savings.
-
1a1. User is prompted to enter a saving.
Use case ends.
-
{More to be added}
-
Should work on any mainstream OS as long as it has Java
9
or higher installed. -
Should be able to hold up to 1000 wishes without user experiencing a drop in application performance.
-
A user with above average typing speed for regular English text (i.e. not code, not system admin commands) should be able to accomplish most of the tasks faster using commands than using the mouse.
-
User data can be transferred across different machines (of different platforms).
-
The software should not use a DBMS (Database Management System) to store data.
-
User data is stored locally.
-
User data is human readable and can be edited.
-
Friendly towards color-blind users.
-
Command Line Interface (CLI) is the primary mode of input. GUI is used mainly for visual feedback rather than to collect input. Usage of mouse should be minimized.
-
The software should follow the Object-Oriented Paradigm.
-
The software should work without requiring an installer.
{More to be added}
Given below are instructions to test the app manually.
ℹ️
|
These instructions only provide a starting point for testers to work on; testers are expected to do more exploratory testing. |
-
Initial launch
-
Download the jar file and copy into an empty folder
-
Double-click the jar file
Expected: Shows the GUI with a set of sample contacts. The window size may not be optimum.
-
-
Saving window preferences
-
Resize the window to an optimum size. Move the window to a different location. Close the window.
-
Re-launch the app by double-clicking the jar file.
Expected: The most recent window size and location is retained.
-
{ more test cases … }
-
Deleting a wish while all wishes are listed
-
Prerequisites: List all wishes using the
list
command. Multiple wishes in the list. -
Test case:
delete 1
Expected: First contact is deleted from the list. Details of the deleted contact shown in the status message. Timestamp in the status bar is updated. -
Test case:
delete 0
Expected: No wish is deleted. Error details shown in the status message. Status bar remains the same. -
Other incorrect delete commands to try:
delete
,delete x
(where x is larger than the list size) {give more}
Expected: Similar to previous.
-
{ more test cases … }