Compare commits
No commits in common. "f36826ef2914e20392d2823ebfafdffbfc34b9f9" and "f7105f041cc7c66d8bd6f45f5c9115ad621dcd88" have entirely different histories.
f36826ef29
...
f7105f041c
|
@ -104,16 +104,16 @@ public class CropDetailController {
|
||||||
* open dialog to set area
|
* open dialog to set area
|
||||||
*/
|
*/
|
||||||
@FXML
|
@FXML
|
||||||
void setArea() throws IOException {
|
void setArea() {
|
||||||
openTextFieldDialog("set Text Area", "Text Area", area_label.getText(), false);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* open dialog to set location
|
* open dialog to set location
|
||||||
*/
|
*/
|
||||||
@FXML
|
@FXML
|
||||||
void setLocation() throws IOException {
|
void setLocation() {
|
||||||
openTextFieldDialog("set Location", "Location", location_label.getText(), true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -218,7 +218,6 @@ public class CropDetailController {
|
||||||
setIconToButton(edit, "editIcon.png");
|
setIconToButton(edit, "editIcon.png");
|
||||||
setIconToButton(delete, "deleteIcon.png");
|
setIconToButton(delete, "deleteIcon.png");
|
||||||
edit.setOnAction(getEditTaskEvent(task));
|
edit.setOnAction(getEditTaskEvent(task));
|
||||||
delete.setOnAction(deleteTask(task));
|
|
||||||
|
|
||||||
hBox.getChildren().addAll(taskName, taskDescription, edit, delete);
|
hBox.getChildren().addAll(taskName, taskDescription, edit, delete);
|
||||||
return hBox;
|
return hBox;
|
||||||
|
@ -261,12 +260,6 @@ public class CropDetailController {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
private EventHandler<ActionEvent> deleteTask(Task task) {
|
|
||||||
return (event) -> {
|
|
||||||
showDeleteTask(task);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
private void createTaskDialog(boolean newTask, Task givenTask) throws IOException {
|
private void createTaskDialog(boolean newTask, Task givenTask) throws IOException {
|
||||||
Dialog<Task> dialog = new Dialog<>();
|
Dialog<Task> dialog = new Dialog<>();
|
||||||
dialog.setTitle("Set Task");
|
dialog.setTitle("Set Task");
|
||||||
|
@ -285,7 +278,6 @@ public class CropDetailController {
|
||||||
|
|
||||||
if (appLoader.loadPaneToDialog("TaskFormular.fxml", dialogPane) instanceof TaskFormularController controller) {
|
if (appLoader.loadPaneToDialog("TaskFormular.fxml", dialogPane) instanceof TaskFormularController controller) {
|
||||||
controller.setCorp(this.crop);
|
controller.setCorp(this.crop);
|
||||||
controller.initSaveButton((Button) dialogPane.lookupButton(saveTask));
|
|
||||||
if (!newTask) {
|
if (!newTask) {
|
||||||
controller.setTaskValue(givenTask);
|
controller.setTaskValue(givenTask);
|
||||||
}
|
}
|
||||||
|
@ -308,56 +300,4 @@ public class CropDetailController {
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void openTextFieldDialog(String title, String labelDescription, String value, boolean isLocation) throws IOException {
|
|
||||||
Dialog<String> dialog = new Dialog<>();
|
|
||||||
dialog.setTitle(title);
|
|
||||||
dialog.setHeaderText(title);
|
|
||||||
dialog.setResizable(false);
|
|
||||||
|
|
||||||
DialogPane dialogPane = dialog.getDialogPane();
|
|
||||||
|
|
||||||
ButtonType save = new ButtonType("Save", ButtonBar.ButtonData.OK_DONE);
|
|
||||||
dialogPane.getButtonTypes().addAll(save, ButtonType.CANCEL);
|
|
||||||
|
|
||||||
if (appLoader.loadPaneToDialog("TextFieldFormular.fxml", dialogPane) instanceof TextFieldFormularController controller) {
|
|
||||||
controller.setDescription_label(labelDescription);
|
|
||||||
controller.setValueTextArea(value);
|
|
||||||
controller.initSaveButton((Button) dialogPane.lookupButton(save));
|
|
||||||
|
|
||||||
dialog.setResultConverter(button -> button.equals(save) ? controller.getValue() : null);
|
|
||||||
|
|
||||||
dialog.showAndWait()
|
|
||||||
.ifPresent(string -> {
|
|
||||||
if (isLocation) {
|
|
||||||
System.out.println(string);
|
|
||||||
//ToDo method to set location
|
|
||||||
location_label.setText(string);
|
|
||||||
} else {
|
|
||||||
System.out.println(string);
|
|
||||||
//ToDo method to set area of crop in garden
|
|
||||||
area_label.setText(string);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void showDeleteTask(Task task) {
|
|
||||||
Alert alert = new Alert(Alert.AlertType.CONFIRMATION);
|
|
||||||
alert.setTitle("Delete " + task.getName());
|
|
||||||
alert.setHeaderText("Are you sure want to delete this Task?");
|
|
||||||
|
|
||||||
alert.showAndWait()
|
|
||||||
.ifPresent(buttonType -> {
|
|
||||||
if (buttonType == ButtonType.OK) {
|
|
||||||
try {
|
|
||||||
gardenSchedule.removeTask(task);
|
|
||||||
setTaskListProperty(this.crop);
|
|
||||||
} catch (IOException e) {
|
|
||||||
// TODO: Show error alert
|
|
||||||
LOG.log(Level.SEVERE, "Could not remove crop.", e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,9 +11,6 @@ import javafx.scene.image.Image;
|
||||||
import javafx.scene.image.ImageView;
|
import javafx.scene.image.ImageView;
|
||||||
import javafx.scene.layout.AnchorPane;
|
import javafx.scene.layout.AnchorPane;
|
||||||
import javafx.scene.layout.Pane;
|
import javafx.scene.layout.Pane;
|
||||||
import javafx.stage.Modality;
|
|
||||||
import javafx.stage.Stage;
|
|
||||||
import javafx.stage.WindowEvent;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.logging.Level;
|
import java.util.logging.Level;
|
||||||
|
@ -43,9 +40,6 @@ public class MainFXMLController {
|
||||||
@FXML
|
@FXML
|
||||||
private Button tutorial_button;
|
private Button tutorial_button;
|
||||||
|
|
||||||
private final Stage tutorialModal = new Stage();
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* go to home pane
|
* go to home pane
|
||||||
*/
|
*/
|
||||||
|
@ -101,23 +95,11 @@ public class MainFXMLController {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Show the tutorial window
|
* go to Tutorial pane
|
||||||
*/
|
*/
|
||||||
public void showTutorial() {
|
public void goToTutorial() {
|
||||||
if (!tutorialModal.isShowing()) {
|
showPaneAsMainView("Tutorial.fxml");
|
||||||
if (tutorialModal.getScene() == null) {
|
styleChangeButton(tutorial_button);
|
||||||
try {
|
|
||||||
appLoader.loadSceneToStage("Tutorial.fxml", tutorialModal);
|
|
||||||
tutorialModal.initModality(Modality.NONE);
|
|
||||||
tutorialModal.setResizable(false);
|
|
||||||
tutorialModal.sizeToScene();
|
|
||||||
} catch (IOException e) {
|
|
||||||
LOG.log(Level.SEVERE, "Could not load Tutorial");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
tutorialModal.show();
|
|
||||||
}
|
|
||||||
tutorialModal.requestFocus();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -150,6 +132,7 @@ public class MainFXMLController {
|
||||||
appLoader.loadAndCacheFxml("MyGarden.fxml");
|
appLoader.loadAndCacheFxml("MyGarden.fxml");
|
||||||
appLoader.loadAndCacheFxml("MySchedule.fxml");
|
appLoader.loadAndCacheFxml("MySchedule.fxml");
|
||||||
appLoader.loadAndCacheFxml("Plants.fxml");
|
appLoader.loadAndCacheFxml("Plants.fxml");
|
||||||
|
appLoader.loadAndCacheFxml("Tutorial.fxml");
|
||||||
}
|
}
|
||||||
|
|
||||||
private void styleChangeButton(Button button) {
|
private void styleChangeButton(Button button) {
|
||||||
|
@ -170,14 +153,12 @@ public class MainFXMLController {
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
LOG.log(Level.SEVERE, "Failed to load FXML-Pane!", e);
|
LOG.log(Level.SEVERE, "Failed to load FXML-Pane!", e);
|
||||||
}
|
}
|
||||||
mainPane.getScene().getWindow().setOnCloseRequest(this::closeWindowHandler);
|
|
||||||
setIconToButton(home_button, "homeIcon.png");
|
setIconToButton(home_button, "homeIcon.png");
|
||||||
setIconToButton(settings_button, "settingsIcon.png");
|
setIconToButton(settings_button, "settingsIcon.png");
|
||||||
tutorial_button.visibleProperty().bind(Settings.getInstance().getShowTutorialProperty());
|
Settings.getInstance().getShowTutorialProperty().addListener((observable, oldValue, newValue) -> {
|
||||||
}
|
tutorial_button.setVisible(newValue);
|
||||||
|
});
|
||||||
private void closeWindowHandler(WindowEvent windowEvent) {
|
tutorial_button.setVisible(Settings.getInstance().getShowTutorial());
|
||||||
tutorialModal.close();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -192,4 +173,7 @@ public class MainFXMLController {
|
||||||
imageView.setPreserveRatio(true);
|
imageView.setPreserveRatio(true);
|
||||||
button.setGraphic(imageView);
|
button.setGraphic(imageView);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2,14 +2,12 @@ package ch.zhaw.gartenverwaltung;
|
||||||
|
|
||||||
import ch.zhaw.gartenverwaltung.types.Crop;
|
import ch.zhaw.gartenverwaltung.types.Crop;
|
||||||
import ch.zhaw.gartenverwaltung.types.Task;
|
import ch.zhaw.gartenverwaltung.types.Task;
|
||||||
import javafx.beans.binding.Binding;
|
|
||||||
import javafx.beans.binding.Bindings;
|
|
||||||
import javafx.beans.binding.BooleanBinding;
|
|
||||||
import javafx.beans.value.ChangeListener;
|
|
||||||
import javafx.beans.value.ObservableValue;
|
|
||||||
import javafx.fxml.FXML;
|
import javafx.fxml.FXML;
|
||||||
import javafx.fxml.Initializable;
|
import javafx.fxml.Initializable;
|
||||||
import javafx.scene.control.*;
|
import javafx.scene.control.DateCell;
|
||||||
|
import javafx.scene.control.DatePicker;
|
||||||
|
import javafx.scene.control.TextArea;
|
||||||
|
import javafx.scene.control.TextField;
|
||||||
import javafx.util.Callback;
|
import javafx.util.Callback;
|
||||||
|
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
|
@ -58,7 +56,7 @@ public class TaskFormularController implements Initializable {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private Callback<DatePicker, DateCell> getDayCellFactoryStartDate() {
|
private Callback<DatePicker, DateCell> getDayCellFactory() {
|
||||||
|
|
||||||
return (datePicker) -> new DateCell() {
|
return (datePicker) -> new DateCell() {
|
||||||
private final LocalDate today = LocalDate.now();
|
private final LocalDate today = LocalDate.now();
|
||||||
|
@ -73,61 +71,16 @@ public class TaskFormularController implements Initializable {
|
||||||
setDisable(false);
|
setDisable(false);
|
||||||
setStyle("-fx-background-color: #32CD32;");
|
setStyle("-fx-background-color: #32CD32;");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (end_datePicker.getValue() != null && item.compareTo(end_datePicker.getValue()) > 0) {
|
|
||||||
setDisable(true);
|
|
||||||
setStyle("-fx-background-color: #ffc0cb;");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
private Callback<DatePicker, DateCell> getDayCellFactoryEndDate() {
|
|
||||||
|
|
||||||
return (datePicker) -> new DateCell() {
|
|
||||||
private final LocalDate today = LocalDate.now();
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void updateItem(LocalDate item, boolean empty) {
|
|
||||||
super.updateItem(item, empty);
|
|
||||||
setDisable(true);
|
|
||||||
setStyle("-fx-background-color: #ffc0cb;");
|
|
||||||
|
|
||||||
if (item.compareTo(today) > 0 && item.compareTo(crop.getStartDate()) > 0) {
|
|
||||||
setDisable(false);
|
|
||||||
setStyle("-fx-background-color: #32CD32;");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (start_datePicker.getValue() != null && item.compareTo(start_datePicker.getValue()) < 0) {
|
|
||||||
setDisable(true);
|
|
||||||
setStyle("-fx-background-color: #ffc0cb;");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public void initSaveButton(Button button) {
|
|
||||||
interval_field.textProperty().addListener((observable, oldValue, newValue) -> {
|
|
||||||
if (!newValue.matches("\\d*")) {
|
|
||||||
interval_field.setText(newValue.replaceAll("[^\\d]", ""));
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
button.disableProperty().bind(start_datePicker.valueProperty().isNull()
|
|
||||||
.or(end_datePicker.valueProperty().isNull())
|
|
||||||
.or(taskName_field.textProperty().isEmpty())
|
|
||||||
.or(description_area.textProperty().isEmpty())
|
|
||||||
.or(interval_field.textProperty().isEmpty()));
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void initialize(URL location, ResourceBundle resources) {
|
public void initialize(URL location, ResourceBundle resources) {
|
||||||
start_datePicker.setDayCellFactory(getDayCellFactoryStartDate());
|
start_datePicker.setDayCellFactory(getDayCellFactory());
|
||||||
start_datePicker.setEditable(false);
|
start_datePicker.setEditable(false);
|
||||||
|
|
||||||
end_datePicker.setDayCellFactory(getDayCellFactoryEndDate());
|
end_datePicker.setDayCellFactory(getDayCellFactory());
|
||||||
end_datePicker.setEditable(false);
|
end_datePicker.setEditable(false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,32 +0,0 @@
|
||||||
package ch.zhaw.gartenverwaltung;
|
|
||||||
|
|
||||||
import javafx.fxml.FXML;
|
|
||||||
import javafx.scene.control.Button;
|
|
||||||
import javafx.scene.control.Label;
|
|
||||||
import javafx.scene.control.TextField;
|
|
||||||
|
|
||||||
public class TextFieldFormularController {
|
|
||||||
|
|
||||||
@FXML
|
|
||||||
private Label description_label;
|
|
||||||
|
|
||||||
@FXML
|
|
||||||
private TextField text_area;
|
|
||||||
|
|
||||||
|
|
||||||
public void setDescription_label(String string) {
|
|
||||||
description_label.setText(string);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setValueTextArea(String string) {
|
|
||||||
text_area.setText(string);
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getValue() {
|
|
||||||
return text_area.getText();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void initSaveButton(Button button) {
|
|
||||||
button.disableProperty().bind(text_area.textProperty().isEmpty());
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,61 +1,6 @@
|
||||||
package ch.zhaw.gartenverwaltung;
|
package ch.zhaw.gartenverwaltung;
|
||||||
|
|
||||||
|
|
||||||
import javafx.fxml.FXML;
|
|
||||||
import javafx.scene.control.Button;
|
|
||||||
import javafx.scene.image.Image;
|
|
||||||
import javafx.scene.image.ImageView;
|
|
||||||
import javafx.scene.layout.StackPane;
|
|
||||||
import javafx.stage.Stage;
|
|
||||||
|
|
||||||
public class TutorialController {
|
public class TutorialController {
|
||||||
|
|
||||||
@FXML
|
|
||||||
public Button previousPageButton;
|
|
||||||
@FXML
|
|
||||||
public Button nextPageButton;
|
|
||||||
@FXML
|
|
||||||
public StackPane tourPages;
|
|
||||||
public ImageView imgAddNewPlant;
|
|
||||||
public ImageView imgTaskList;
|
|
||||||
public ImageView imgSelectDate;
|
|
||||||
|
|
||||||
private int page = 0;
|
|
||||||
|
|
||||||
@FXML
|
|
||||||
public void initialize() {
|
|
||||||
switchViews();
|
|
||||||
setButtonAbilities();
|
|
||||||
|
|
||||||
Image placeholder = new Image(String.valueOf(PlantsController.class.getResource("placeholder.png")));
|
|
||||||
imgAddNewPlant.setImage(placeholder);
|
|
||||||
imgSelectDate.setImage(placeholder);
|
|
||||||
imgTaskList.setImage(placeholder);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void viewNextPage() {
|
|
||||||
page++;
|
|
||||||
switchViews();
|
|
||||||
setButtonAbilities();
|
|
||||||
}
|
|
||||||
public void viewPreviousPage() {
|
|
||||||
page--;
|
|
||||||
switchViews();
|
|
||||||
setButtonAbilities();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void setButtonAbilities() {
|
|
||||||
previousPageButton.setDisable(page <= 0);
|
|
||||||
nextPageButton.setDisable(page >= tourPages.getChildren().size() - 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void switchViews() {
|
|
||||||
tourPages.getChildren().forEach(node -> node.setOpacity(0));
|
|
||||||
tourPages.getChildren().get(page).setOpacity(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void closeTutorial() {
|
|
||||||
Stage root = (Stage) tourPages.getScene().getWindow();
|
|
||||||
root.close();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -73,10 +73,7 @@ public class AppLoader {
|
||||||
public Object loadSceneToStage(String fxmlFile, Stage appendee) throws IOException {
|
public Object loadSceneToStage(String fxmlFile, Stage appendee) throws IOException {
|
||||||
FXMLLoader loader = new FXMLLoader(Objects.requireNonNull(HelloApplication.class.getResource(fxmlFile)));
|
FXMLLoader loader = new FXMLLoader(Objects.requireNonNull(HelloApplication.class.getResource(fxmlFile)));
|
||||||
Pane root = loader.load();
|
Pane root = loader.load();
|
||||||
Scene scene = new Scene(root);
|
appendee.setScene(new Scene(root));
|
||||||
String css = Objects.requireNonNull(this.getClass().getResource("styles.css")).toExternalForm();
|
|
||||||
appendee.setScene(scene);
|
|
||||||
scene.getStylesheets().add(css);
|
|
||||||
Object controller = loader.getController();
|
Object controller = loader.getController();
|
||||||
annotationInject(controller);
|
annotationInject(controller);
|
||||||
return controller;
|
return controller;
|
||||||
|
|
|
@ -9,7 +9,6 @@ import java.time.LocalDate;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Comparator;
|
import java.util.Comparator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.function.Consumer;
|
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
public class GardenSchedule {
|
public class GardenSchedule {
|
||||||
|
@ -20,7 +19,6 @@ public class GardenSchedule {
|
||||||
* Comparators to create sorted Task List
|
* Comparators to create sorted Task List
|
||||||
*/
|
*/
|
||||||
static final Comparator<Task> sortByStartDate = Comparator.comparing(Task::getStartDate);
|
static final Comparator<Task> sortByStartDate = Comparator.comparing(Task::getStartDate);
|
||||||
static final Comparator<Task> sortByNextExecution = Comparator.comparing(Task::getNextExecution);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor to create Database Objects.
|
* Constructor to create Database Objects.
|
||||||
|
@ -48,8 +46,7 @@ public class GardenSchedule {
|
||||||
*/
|
*/
|
||||||
public void planTasksForCrop(Crop crop) throws PlantNotFoundException, HardinessZoneNotSetException, IOException {
|
public void planTasksForCrop(Crop crop) throws PlantNotFoundException, HardinessZoneNotSetException, IOException {
|
||||||
Plant plant = plantList.getPlantById(Settings.getInstance().getCurrentHardinessZone(), crop.getPlantId()).orElseThrow(PlantNotFoundException::new);
|
Plant plant = plantList.getPlantById(Settings.getInstance().getCurrentHardinessZone(), crop.getPlantId()).orElseThrow(PlantNotFoundException::new);
|
||||||
int growPhaseGroup = plant.getGrowphaseGroupForDate(crop.getStartDate());
|
for (GrowthPhase growthPhase : plant.lifecycle()) {
|
||||||
for (GrowthPhase growthPhase : plant.lifecycleForGroup(growPhaseGroup)) {
|
|
||||||
for (TaskTemplate taskTemplate : growthPhase.taskTemplates()) {
|
for (TaskTemplate taskTemplate : growthPhase.taskTemplates()) {
|
||||||
addTask(taskTemplate.generateTask(crop.getStartDate(), crop.getCropId().orElse(0L)));
|
addTask(taskTemplate.generateTask(crop.getStartDate(), crop.getCropId().orElse(0L)));
|
||||||
}
|
}
|
||||||
|
@ -138,22 +135,10 @@ public class GardenSchedule {
|
||||||
* @throws IOException If the database cannot be accessed
|
* @throws IOException If the database cannot be accessed
|
||||||
*/
|
*/
|
||||||
public List<List<Task>> getTasksUpcomingWeek() throws IOException {
|
public List<List<Task>> getTasksUpcomingWeek() throws IOException {
|
||||||
final int listLength = 7;
|
|
||||||
List<Task> weekTasks = taskList.getTaskList(LocalDate.now(), LocalDate.now().plusDays(listLength - 1));
|
|
||||||
List<List<Task>> dayTaskList = new ArrayList<>();
|
List<List<Task>> dayTaskList = new ArrayList<>();
|
||||||
for(int i = 0; i < listLength; i++) {
|
for(int i = 0; i < 7; i++) {
|
||||||
LocalDate date = LocalDate.now().plusDays(i);
|
LocalDate date = LocalDate.now().plusDays(i);
|
||||||
dayTaskList.add(new ArrayList<>());
|
dayTaskList.add(taskList.getTaskList(date, date));
|
||||||
final int finalI = i;
|
|
||||||
weekTasks.forEach(task -> {
|
|
||||||
LocalDate checkDate = task.getNextExecution();
|
|
||||||
do {
|
|
||||||
if (date.equals(checkDate) && !date.isAfter(task.getEndDate().orElse(LocalDate.MIN))) {
|
|
||||||
dayTaskList.get(finalI).add(task);
|
|
||||||
}
|
|
||||||
checkDate = checkDate.plusDays(task.getInterval().orElse(0));
|
|
||||||
} while (task.getInterval().isPresent() && checkDate.isBefore(LocalDate.now().plusDays(listLength)));
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
return dayTaskList;
|
return dayTaskList;
|
||||||
}
|
}
|
||||||
|
@ -164,8 +149,11 @@ public class GardenSchedule {
|
||||||
* @throws IOException If the database cannot be accessed
|
* @throws IOException If the database cannot be accessed
|
||||||
*/
|
*/
|
||||||
public List<List<Task>> getTasksUpcomingWeekForCrop(Long cropId) throws IOException {
|
public List<List<Task>> getTasksUpcomingWeekForCrop(Long cropId) throws IOException {
|
||||||
List<List<Task>> dayTaskList = getTasksUpcomingWeek();
|
List<List<Task>> dayTaskList = new ArrayList<>();
|
||||||
dayTaskList.forEach(taskList -> taskList.removeIf(task -> task.getCropId() != cropId));
|
for(int i = 0; i < 7; i++) {
|
||||||
|
LocalDate date = LocalDate.now().plusDays(i);
|
||||||
|
dayTaskList.add(filterListByCrop(taskList.getTaskList(date, date), cropId));
|
||||||
|
}
|
||||||
return dayTaskList;
|
return dayTaskList;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -177,7 +165,7 @@ public class GardenSchedule {
|
||||||
* @throws IOException If the database cannot be accessed
|
* @throws IOException If the database cannot be accessed
|
||||||
*/
|
*/
|
||||||
public List<Task> getFilteredTaskList(LocalDate start, LocalDate end) throws IOException {
|
public List<Task> getFilteredTaskList(LocalDate start, LocalDate end) throws IOException {
|
||||||
return getSortedTaskList(taskList.getTaskList(start, end), sortByNextExecution);
|
return getSortedTaskList(taskList.getTaskList(start, end), sortByStartDate);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -39,16 +39,6 @@ public record Plant(
|
||||||
.collect(Collectors.toList());
|
.collect(Collectors.toList());
|
||||||
}
|
}
|
||||||
|
|
||||||
public int getGrowphaseGroupForDate(LocalDate date) {
|
|
||||||
for(GrowthPhase growthPhase : lifecycle){
|
|
||||||
MonthDay plantingDate = MonthDay.of(date.getMonth().getValue(), date.getDayOfMonth());
|
|
||||||
if(plantingDate.isAfter(growthPhase.startDate()) && plantingDate.isBefore(growthPhase.endDate())){
|
|
||||||
return growthPhase.group();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return 0; // TODO implement
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* get sow date from given harvest day from lifecycle group
|
* get sow date from given harvest day from lifecycle group
|
||||||
* @param harvestDate date of the harvest
|
* @param harvestDate date of the harvest
|
||||||
|
|
|
@ -15,8 +15,6 @@ public class Task {
|
||||||
private final LocalDate startDate;
|
private final LocalDate startDate;
|
||||||
private Integer interval;
|
private Integer interval;
|
||||||
private LocalDate endDate;
|
private LocalDate endDate;
|
||||||
private LocalDate nextExecution;
|
|
||||||
private LocalDate nextNotification;
|
|
||||||
private long cropId;
|
private long cropId;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -27,16 +25,12 @@ public class Task {
|
||||||
name= "";
|
name= "";
|
||||||
description= "";
|
description= "";
|
||||||
startDate = LocalDate.now();
|
startDate = LocalDate.now();
|
||||||
endDate = startDate;
|
|
||||||
nextExecution = startDate;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public Task(String name, String description, LocalDate startDate, long cropId) {
|
public Task(String name, String description, LocalDate startDate, long cropId) {
|
||||||
this.name = name;
|
this.name = name;
|
||||||
this.description = description;
|
this.description = description;
|
||||||
this.startDate = startDate;
|
this.startDate = startDate;
|
||||||
this.endDate = startDate;
|
|
||||||
nextExecution = startDate;
|
|
||||||
this.cropId = cropId;
|
this.cropId = cropId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,7 +38,6 @@ public class Task {
|
||||||
this.name = name;
|
this.name = name;
|
||||||
this.description = description;
|
this.description = description;
|
||||||
this.startDate = startDate;
|
this.startDate = startDate;
|
||||||
nextExecution = startDate;
|
|
||||||
this.endDate = endDate;
|
this.endDate = endDate;
|
||||||
this.interval = interval;
|
this.interval = interval;
|
||||||
this.cropId = cropId;
|
this.cropId = cropId;
|
||||||
|
@ -65,32 +58,10 @@ public class Task {
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean isInTimePeriode(LocalDate searchStartDate, LocalDate searchEndDate){
|
public boolean isInTimePeriode(LocalDate searchStartDate, LocalDate searchEndDate){
|
||||||
return endDate.isAfter(searchStartDate) && startDate.isBefore(searchEndDate);
|
return startDate.isAfter(searchStartDate) && startDate.isBefore(searchEndDate);
|
||||||
}
|
|
||||||
|
|
||||||
public void done(){
|
|
||||||
if(interval != null && !nextExecution.plusDays(interval).isAfter(endDate)){
|
|
||||||
nextExecution = nextExecution.plusDays(interval);
|
|
||||||
} else {
|
|
||||||
nextExecution = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isDone(){
|
|
||||||
return nextExecution == null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setNextExecution(LocalDate nextExecution) {
|
|
||||||
this.nextExecution = nextExecution;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setNextNotification(LocalDate nextNotification) {
|
|
||||||
this.nextNotification = nextNotification;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Getters
|
// Getters
|
||||||
public LocalDate getNextNotification() { return nextNotification; }
|
|
||||||
public LocalDate getNextExecution() { return nextExecution; }
|
|
||||||
public Optional<Long> getId() { return Optional.ofNullable(id); }
|
public Optional<Long> getId() { return Optional.ofNullable(id); }
|
||||||
public String getName() { return name; }
|
public String getName() { return name; }
|
||||||
public String getDescription() { return description; }
|
public String getDescription() { return description; }
|
||||||
|
|
|
@ -16,6 +16,10 @@ public class TaskTemplate {
|
||||||
@JsonProperty
|
@JsonProperty
|
||||||
private Integer interval;
|
private Integer interval;
|
||||||
|
|
||||||
|
// TODO: reconsider if we need this
|
||||||
|
@JsonProperty
|
||||||
|
private boolean isOptional = false;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Default constructor
|
* Default constructor
|
||||||
* (Used by deserializer)
|
* (Used by deserializer)
|
||||||
|
@ -44,10 +48,6 @@ public class TaskTemplate {
|
||||||
public Task generateTask(LocalDate realStartDate, long cropId) {
|
public Task generateTask(LocalDate realStartDate, long cropId) {
|
||||||
LocalDate endDate = relativeEndDate != null ? realStartDate.plusDays(relativeEndDate) : null;
|
LocalDate endDate = relativeEndDate != null ? realStartDate.plusDays(relativeEndDate) : null;
|
||||||
|
|
||||||
if (interval == null) {
|
|
||||||
this.interval = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return new Task(name, description, realStartDate.plusDays(relativeStartDate), cropId)
|
return new Task(name, description, realStartDate.plusDays(relativeStartDate), cropId)
|
||||||
.withInterval(interval)
|
.withInterval(interval)
|
||||||
.withEndDate(endDate);
|
.withEndDate(endDate);
|
||||||
|
|
|
@ -12,7 +12,7 @@
|
||||||
<Button fx:id="home_button" mnemonicParsing="false" onAction="#goToHome" prefHeight="38.0" prefWidth="40.0" HBox.hgrow="NEVER" />
|
<Button fx:id="home_button" mnemonicParsing="false" onAction="#goToHome" prefHeight="38.0" prefWidth="40.0" HBox.hgrow="NEVER" />
|
||||||
<Button fx:id="myGarden_button" layoutX="10.0" layoutY="10.0" mnemonicParsing="false" onAction="#goToMyPlants" prefHeight="38.0" prefWidth="121.0" text="My Garden" />
|
<Button fx:id="myGarden_button" layoutX="10.0" layoutY="10.0" mnemonicParsing="false" onAction="#goToMyPlants" prefHeight="38.0" prefWidth="121.0" text="My Garden" />
|
||||||
<Button fx:id="mySchedule_button" layoutX="10.0" layoutY="10.0" mnemonicParsing="false" onAction="#goToMySchedule" prefHeight="38.0" prefWidth="121.0" text="My Schedule" />
|
<Button fx:id="mySchedule_button" layoutX="10.0" layoutY="10.0" mnemonicParsing="false" onAction="#goToMySchedule" prefHeight="38.0" prefWidth="121.0" text="My Schedule" />
|
||||||
<Button fx:id="tutorial_button" layoutX="10.0" layoutY="10.0" mnemonicParsing="false" onAction="#showTutorial" prefHeight="38.0" prefWidth="121.0" text="Tutorial" />
|
<Button fx:id="tutorial_button" layoutX="10.0" layoutY="10.0" mnemonicParsing="false" onAction="#goToTutorial" prefHeight="38.0" prefWidth="121.0" text="Tutorial" />
|
||||||
<Pane maxWidth="1.7976931348623157E308" prefHeight="200.0" prefWidth="200.0" HBox.hgrow="ALWAYS" />
|
<Pane maxWidth="1.7976931348623157E308" prefHeight="200.0" prefWidth="200.0" HBox.hgrow="ALWAYS" />
|
||||||
<Button fx:id="settings_button" maxHeight="1.7976931348623157E308" mnemonicParsing="false" onAction="#openSettings" prefHeight="38.0" prefWidth="40.0" HBox.hgrow="NEVER" />
|
<Button fx:id="settings_button" maxHeight="1.7976931348623157E308" mnemonicParsing="false" onAction="#openSettings" prefHeight="38.0" prefWidth="40.0" HBox.hgrow="NEVER" />
|
||||||
</children>
|
</children>
|
||||||
|
|
|
@ -1,23 +0,0 @@
|
||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
|
|
||||||
<?import javafx.geometry.Insets?>
|
|
||||||
<?import javafx.scene.control.Label?>
|
|
||||||
<?import javafx.scene.control.TextField?>
|
|
||||||
<?import javafx.scene.layout.AnchorPane?>
|
|
||||||
<?import javafx.scene.layout.HBox?>
|
|
||||||
|
|
||||||
|
|
||||||
<AnchorPane maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="64.0" prefWidth="298.0" xmlns="http://javafx.com/javafx/17"
|
|
||||||
xmlns:fx="http://javafx.com/fxml/1" fx:controller="ch.zhaw.gartenverwaltung.TextFieldFormularController">
|
|
||||||
<children>
|
|
||||||
<HBox alignment="CENTER" layoutX="153.0" layoutY="33.0" prefHeight="100.0" prefWidth="200.0" spacing="10.0" AnchorPane.bottomAnchor="0.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" AnchorPane.topAnchor="0.0">
|
|
||||||
<children>
|
|
||||||
<Label fx:id="description_label" maxWidth="1.7976931348623157E308" text="Label" HBox.hgrow="ALWAYS" />
|
|
||||||
<TextField fx:id="text_area" HBox.hgrow="NEVER" />
|
|
||||||
</children>
|
|
||||||
<padding>
|
|
||||||
<Insets bottom="10.0" left="10.0" right="10.0" top="10.0" />
|
|
||||||
</padding>
|
|
||||||
</HBox>
|
|
||||||
</children>
|
|
||||||
</AnchorPane>
|
|
|
@ -1,141 +1,27 @@
|
||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
|
||||||
<?import javafx.geometry.Insets?>
|
<?import javafx.geometry.Insets?>
|
||||||
<?import javafx.scene.control.Button?>
|
<?import javafx.scene.control.Label?>
|
||||||
<?import javafx.scene.control.ButtonBar?>
|
<?import javafx.scene.layout.AnchorPane?>
|
||||||
<?import javafx.scene.control.CheckBox?>
|
|
||||||
<?import javafx.scene.control.Separator?>
|
|
||||||
<?import javafx.scene.image.ImageView?>
|
|
||||||
<?import javafx.scene.layout.BorderPane?>
|
|
||||||
<?import javafx.scene.layout.HBox?>
|
|
||||||
<?import javafx.scene.layout.StackPane?>
|
|
||||||
<?import javafx.scene.layout.VBox?>
|
<?import javafx.scene.layout.VBox?>
|
||||||
<?import javafx.scene.text.Font?>
|
<?import javafx.scene.text.Font?>
|
||||||
<?import javafx.scene.text.Text?>
|
|
||||||
<?import javafx.scene.text.TextFlow?>
|
|
||||||
|
|
||||||
<BorderPane maxHeight="470.0" maxWidth="724.0" minHeight="470.0" minWidth="724.0" prefHeight="470.0" prefWidth="724.0" xmlns="http://javafx.com/javafx/17" xmlns:fx="http://javafx.com/fxml/1" fx:controller="ch.zhaw.gartenverwaltung.TutorialController">
|
|
||||||
<bottom>
|
<AnchorPane maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="400.0" prefWidth="600.0" xmlns="http://javafx.com/javafx/17"
|
||||||
<HBox alignment="CENTER" prefHeight="0.0" prefWidth="724.0" BorderPane.alignment="CENTER">
|
xmlns:fx="http://javafx.com/fxml/1" fx:controller="ch.zhaw.gartenverwaltung.TutorialController">
|
||||||
|
<children>
|
||||||
|
<VBox layoutX="7.0" layoutY="8.0" prefHeight="200.0" prefWidth="100.0" AnchorPane.bottomAnchor="0.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" AnchorPane.topAnchor="0.0">
|
||||||
<children>
|
<children>
|
||||||
<CheckBox mnemonicParsing="false" text="Don't show again" />
|
<Label text="Tutorial">
|
||||||
<Separator prefWidth="50.0" visible="false" HBox.hgrow="ALWAYS" />
|
<font>
|
||||||
<ButtonBar prefHeight="40.0" prefWidth="200.0">
|
<Font size="18.0" />
|
||||||
<buttons>
|
</font>
|
||||||
<Button cancelButton="true" contentDisplay="CENTER" graphicTextGap="5.0" mnemonicParsing="false" text="Close" onAction="#closeTutorial"/>
|
</Label>
|
||||||
<Button fx:id="previousPageButton" mnemonicParsing="false" text="Previous" onAction="#viewPreviousPage"/>
|
<Label text="To be added" />
|
||||||
<Button fx:id="nextPageButton" defaultButton="true" mnemonicParsing="false" text="Next" onAction="#viewNextPage" />
|
|
||||||
</buttons>
|
|
||||||
</ButtonBar>
|
|
||||||
</children>
|
</children>
|
||||||
<padding>
|
<padding>
|
||||||
<Insets bottom="15.0" left="15.0" right="15.0" top="15.0" />
|
<Insets bottom="10.0" left="10.0" right="10.0" top="10.0" />
|
||||||
</padding>
|
</padding>
|
||||||
</HBox>
|
</VBox>
|
||||||
</bottom>
|
</children>
|
||||||
<center>
|
</AnchorPane>
|
||||||
<StackPane fx:id="tourPages" prefHeight="150.0" prefWidth="200.0" BorderPane.alignment="CENTER">
|
|
||||||
<children>
|
|
||||||
<VBox layoutX="30.0" layoutY="26.0" opacity="0.0" prefHeight="200.0" prefWidth="100.0">
|
|
||||||
<children>
|
|
||||||
<Text strokeType="OUTSIDE" strokeWidth="0.0" text="Managing Your Crops">
|
|
||||||
<font>
|
|
||||||
<Font size="24.0" />
|
|
||||||
</font>
|
|
||||||
</Text>
|
|
||||||
<Separator prefWidth="200.0">
|
|
||||||
<VBox.margin>
|
|
||||||
<Insets bottom="15.0" top="10.0" />
|
|
||||||
</VBox.margin>
|
|
||||||
</Separator>
|
|
||||||
<Text strokeType="OUTSIDE" strokeWidth="0.0" text="In the "My Garden" tab, you can see all of the plants you're planning on growing.">
|
|
||||||
<VBox.margin>
|
|
||||||
<Insets bottom="20.0" />
|
|
||||||
</VBox.margin>
|
|
||||||
</Text>
|
|
||||||
<HBox prefHeight="100.0" prefWidth="200.0">
|
|
||||||
<children>
|
|
||||||
<TextFlow lineSpacing="4.0" prefHeight="90.0" prefWidth="500.0">
|
|
||||||
<children>
|
|
||||||
<Text strokeType="OUTSIDE" strokeWidth="0.0" text="To get started, click on the "Add new Plant" button in the bottom left. This will open a list of all the plants you can grow. Use the search bar at the top, the filters on the left, or simply browse the list until you've found a plant you want to grow in your garden." wrappingWidth="345.80316162109375" />
|
|
||||||
</children>
|
|
||||||
<HBox.margin>
|
|
||||||
<Insets bottom="20.0" />
|
|
||||||
</HBox.margin>
|
|
||||||
</TextFlow>
|
|
||||||
<Separator prefWidth="200.0" visible="false" HBox.hgrow="ALWAYS" />
|
|
||||||
<ImageView fx:id="imgAddNewPlant" fitHeight="98.0" fitWidth="200.0" pickOnBounds="true" preserveRatio="true" />
|
|
||||||
</children>
|
|
||||||
</HBox>
|
|
||||||
<HBox prefHeight="100.0" prefWidth="200.0">
|
|
||||||
<children>
|
|
||||||
<TextFlow lineSpacing="4.0" prefHeight="200.0" prefWidth="500.0">
|
|
||||||
<children>
|
|
||||||
<Text strokeType="OUTSIDE" strokeWidth="0.0" text="Once you've found what you're looking for, click the "Select Sow/Harvest Date" button at the bottom of the window. In the subsequently shown dialog, you can select the date you'd like to sow or harvest the crop. The date selector shows possible dates in green. Confirm your selection with the "Save" button." />
|
|
||||||
</children>
|
|
||||||
</TextFlow>
|
|
||||||
<Separator prefWidth="200.0" visible="false" HBox.hgrow="ALWAYS" />
|
|
||||||
<ImageView fx:id="imgSelectDate" fitHeight="150.0" fitWidth="200.0" pickOnBounds="true" preserveRatio="true" />
|
|
||||||
</children>
|
|
||||||
</HBox>
|
|
||||||
</children>
|
|
||||||
<opaqueInsets>
|
|
||||||
<Insets />
|
|
||||||
</opaqueInsets>
|
|
||||||
</VBox>
|
|
||||||
<VBox prefHeight="200.0" prefWidth="100.0">
|
|
||||||
<children>
|
|
||||||
<Text strokeType="OUTSIDE" strokeWidth="0.0" text="Managing Your Tasks">
|
|
||||||
<font>
|
|
||||||
<Font size="24.0" />
|
|
||||||
</font>
|
|
||||||
</Text>
|
|
||||||
<Separator prefWidth="200.0">
|
|
||||||
<VBox.margin>
|
|
||||||
<Insets bottom="15.0" top="10.0" />
|
|
||||||
</VBox.margin>
|
|
||||||
</Separator>
|
|
||||||
<Text strokeType="OUTSIDE" strokeWidth="0.0" text="In the "My Schedule" tab, you can see all of the tasks you need to complete for a successful harvest.">
|
|
||||||
<VBox.margin>
|
|
||||||
<Insets bottom="20.0" />
|
|
||||||
</VBox.margin>
|
|
||||||
</Text>
|
|
||||||
<HBox prefHeight="100.0" prefWidth="200.0">
|
|
||||||
<children>
|
|
||||||
<TextFlow lineSpacing="4.0" prefHeight="90.0" prefWidth="500.0">
|
|
||||||
<children>
|
|
||||||
<Text strokeType="OUTSIDE" strokeWidth="0.0" text="You can view the tasks for a specific crop by selecting it from the left sidebar. To see all tasks again, click the "Clear Filter" button at the bottom." wrappingWidth="345.80316162109375" />
|
|
||||||
</children>
|
|
||||||
<HBox.margin>
|
|
||||||
<Insets bottom="20.0" />
|
|
||||||
</HBox.margin>
|
|
||||||
</TextFlow>
|
|
||||||
<Separator prefWidth="200.0" visible="false" HBox.hgrow="ALWAYS" />
|
|
||||||
<ImageView fx:id="imgTaskList" fitHeight="98.0" fitWidth="200.0" pickOnBounds="true" preserveRatio="true" />
|
|
||||||
</children>
|
|
||||||
</HBox>
|
|
||||||
<HBox prefHeight="100.0" prefWidth="200.0">
|
|
||||||
<children>
|
|
||||||
<TextFlow lineSpacing="4.0" prefHeight="200.0" prefWidth="500.0">
|
|
||||||
<children>
|
|
||||||
<Text strokeType="OUTSIDE" strokeWidth="0.0" text="You can also add your own custom tasks, by clicking the "Add Task" button. In the subsequently shown dialog, you can enter the corresponding details. Note: If you want to make a task recurring, you need to set both an interval (in days) AND an end date, so the task won't repeat for all eternity." />
|
|
||||||
</children>
|
|
||||||
</TextFlow>
|
|
||||||
<Separator prefWidth="200.0" visible="false" HBox.hgrow="ALWAYS" />
|
|
||||||
<ImageView fitHeight="150.0" fitWidth="200.0" pickOnBounds="true" preserveRatio="true" />
|
|
||||||
</children></HBox>
|
|
||||||
</children>
|
|
||||||
<opaqueInsets>
|
|
||||||
<Insets />
|
|
||||||
</opaqueInsets>
|
|
||||||
</VBox>
|
|
||||||
</children>
|
|
||||||
<BorderPane.margin>
|
|
||||||
<Insets />
|
|
||||||
</BorderPane.margin>
|
|
||||||
<padding>
|
|
||||||
<Insets bottom="20.0" left="20.0" right="20.0" top="20.0" />
|
|
||||||
</padding>
|
|
||||||
</StackPane>
|
|
||||||
</center>
|
|
||||||
</BorderPane>
|
|
||||||
|
|
|
@ -1,6 +0,0 @@
|
||||||
.button{
|
|
||||||
-fx-text-fill: rgb(49, 89, 23);
|
|
||||||
-fx-border-color: rgb(49, 89, 23);
|
|
||||||
-fx-border-radius: 5;
|
|
||||||
-fx-padding: 3 6 6 6;
|
|
||||||
}
|
|
|
@ -32,7 +32,8 @@
|
||||||
"relativeStartDate": -14,
|
"relativeStartDate": -14,
|
||||||
"relativeEndDate": null,
|
"relativeEndDate": null,
|
||||||
"description": "Take an egg carton and fill it with soil. Put the seedling deep enough so its half covered with soil. Keep it in 10-15 * Celsius with lots of light.",
|
"description": "Take an egg carton and fill it with soil. Put the seedling deep enough so its half covered with soil. Keep it in 10-15 * Celsius with lots of light.",
|
||||||
"interval": null
|
"interval": null,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
@ -53,7 +54,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": null,
|
"relativeEndDate": null,
|
||||||
"description": "When the plants are 20 cm tall, begin hilling the potatoes by gently mounding the soil from the center of your rows around the stems of the plant. Mound up the soil around the plant until just the top few leaves show above the soil. Two weeks later, hill up the soil again when the plants grow another 20 cm.",
|
"description": "When the plants are 20 cm tall, begin hilling the potatoes by gently mounding the soil from the center of your rows around the stems of the plant. Mound up the soil around the plant until just the top few leaves show above the soil. Two weeks later, hill up the soil again when the plants grow another 20 cm.",
|
||||||
"interval": 21
|
"interval": 21,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
@ -74,7 +76,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": null,
|
"relativeEndDate": null,
|
||||||
"description": "Once the foliage has wilted and dried completely, harvest on a dry day. Store in a dark and cool location.",
|
"description": "Once the foliage has wilted and dried completely, harvest on a dry day. Store in a dark and cool location.",
|
||||||
"interval": null
|
"interval": null,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
@ -103,7 +106,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": 0,
|
"relativeEndDate": 0,
|
||||||
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
||||||
"interval": null
|
"interval": null,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
@ -126,7 +130,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": null,
|
"relativeEndDate": null,
|
||||||
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
||||||
"interval": 15
|
"interval": 15,
|
||||||
|
"isOptional": true
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
@ -147,7 +152,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": 14,
|
"relativeEndDate": 14,
|
||||||
"description": "When the leaves turn to a yellowish brown. Do not harvest earlier. The plant will show when it's ready.",
|
"description": "When the leaves turn to a yellowish brown. Do not harvest earlier. The plant will show when it's ready.",
|
||||||
"interval": null
|
"interval": null,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
@ -186,7 +192,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": 0,
|
"relativeEndDate": 0,
|
||||||
"description": "Plant the sets about 5cm deep into the soil.",
|
"description": "Plant the sets about 5cm deep into the soil.",
|
||||||
"interval": null
|
"interval": null,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
@ -209,7 +216,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": null,
|
"relativeEndDate": null,
|
||||||
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
||||||
"interval": 15
|
"interval": 15,
|
||||||
|
"isOptional": true
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
@ -231,7 +239,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": 14,
|
"relativeEndDate": 14,
|
||||||
"description": "When ready for harvest, the leaves on your onion plants will start to flop over. This happens at the \"neck\" of the onion and it signals that the plant has stopped growing and is ready for storage. Onions should be harvested soon thereafter",
|
"description": "When ready for harvest, the leaves on your onion plants will start to flop over. This happens at the \"neck\" of the onion and it signals that the plant has stopped growing and is ready for storage. Onions should be harvested soon thereafter",
|
||||||
"interval": null
|
"interval": null,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,10 +4,8 @@
|
||||||
"name": "sow plant",
|
"name": "sow plant",
|
||||||
"description": "Plant the seeds, crops in de bed.",
|
"description": "Plant the seeds, crops in de bed.",
|
||||||
"startDate": "2022-05-01",
|
"startDate": "2022-05-01",
|
||||||
"nextExecution": "2022-05-01",
|
|
||||||
"nextNotification": "2022-05-01",
|
|
||||||
"endDate": "2022-05-01",
|
"endDate": "2022-05-01",
|
||||||
"interval": null,
|
"interval": 0,
|
||||||
"cropId": 0
|
"cropId": 0
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -15,8 +13,6 @@
|
||||||
"name": "water plant",
|
"name": "water plant",
|
||||||
"description": "water the plant, so that the soil is wet around the plant.",
|
"description": "water the plant, so that the soil is wet around the plant.",
|
||||||
"startDate": "2022-05-01",
|
"startDate": "2022-05-01",
|
||||||
"nextExecution": "2022-05-01",
|
|
||||||
"nextNotification": "2022-05-01",
|
|
||||||
"endDate": "2022-09-01",
|
"endDate": "2022-09-01",
|
||||||
"interval": 2,
|
"interval": 2,
|
||||||
"cropId": 0
|
"cropId": 0
|
||||||
|
@ -26,8 +22,6 @@
|
||||||
"name": "fertilize plant",
|
"name": "fertilize plant",
|
||||||
"description": "The fertilizer has to be mixed with water. Then fertilize the plants soil with the mixture",
|
"description": "The fertilizer has to be mixed with water. Then fertilize the plants soil with the mixture",
|
||||||
"startDate": "2022-06-01",
|
"startDate": "2022-06-01",
|
||||||
"nextExecution": "2022-06-01",
|
|
||||||
"nextNotification": "2022-06-01",
|
|
||||||
"endDate": "2022-08-01",
|
"endDate": "2022-08-01",
|
||||||
"interval": 28,
|
"interval": 28,
|
||||||
"cropId": 0
|
"cropId": 0
|
||||||
|
@ -37,10 +31,8 @@
|
||||||
"name": "covering plant",
|
"name": "covering plant",
|
||||||
"description": "Take a big enough coverage for the plants. Cover the whole plant with a bit space between the plant and the coverage",
|
"description": "Take a big enough coverage for the plants. Cover the whole plant with a bit space between the plant and the coverage",
|
||||||
"startDate": "2022-07-01",
|
"startDate": "2022-07-01",
|
||||||
"nextExecution": "2022-07-01",
|
|
||||||
"nextNotification": "2022-07-01",
|
|
||||||
"endDate": "2022-07-01",
|
"endDate": "2022-07-01",
|
||||||
"interval": null,
|
"interval": 0,
|
||||||
"cropId": 0
|
"cropId": 0
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -48,8 +40,6 @@
|
||||||
"name": "look after plant",
|
"name": "look after plant",
|
||||||
"description": "Look for pest or illness at the leaves of the plant. Check the soil around the plant, if the roots are enough covered with soil",
|
"description": "Look for pest or illness at the leaves of the plant. Check the soil around the plant, if the roots are enough covered with soil",
|
||||||
"startDate": "2022-05-01",
|
"startDate": "2022-05-01",
|
||||||
"nextExecution": "2022-05-01",
|
|
||||||
"nextNotification": "2022-05-01",
|
|
||||||
"endDate": "2022-09-01",
|
"endDate": "2022-09-01",
|
||||||
"interval": 5,
|
"interval": 5,
|
||||||
"cropId": 0
|
"cropId": 0
|
||||||
|
@ -59,10 +49,8 @@
|
||||||
"name": "harvest plant",
|
"name": "harvest plant",
|
||||||
"description": "Pull the ripe vegetables out from the soil. Clean them with clear, fresh water. ",
|
"description": "Pull the ripe vegetables out from the soil. Clean them with clear, fresh water. ",
|
||||||
"startDate": "2022-09-01",
|
"startDate": "2022-09-01",
|
||||||
"nextExecution": "2022-09-01",
|
|
||||||
"nextNotification": "2022-09-01",
|
|
||||||
"endDate": "2022-09-01",
|
"endDate": "2022-09-01",
|
||||||
"interval": null,
|
"interval": 0,
|
||||||
"cropId": 0
|
"cropId": 0
|
||||||
}
|
}
|
||||||
]
|
]
|
|
@ -1,49 +1,38 @@
|
||||||
package ch.zhaw.gartenverwaltung.models;
|
package ch.zhaw.gartenverwaltung.taskList;
|
||||||
|
|
||||||
import ch.zhaw.gartenverwaltung.io.*;
|
import ch.zhaw.gartenverwaltung.io.*;
|
||||||
import ch.zhaw.gartenverwaltung.models.GardenSchedule;
|
import ch.zhaw.gartenverwaltung.models.GardenSchedule;
|
||||||
import ch.zhaw.gartenverwaltung.types.*;
|
import ch.zhaw.gartenverwaltung.types.HardinessZone;
|
||||||
|
import ch.zhaw.gartenverwaltung.types.Plant;
|
||||||
|
import ch.zhaw.gartenverwaltung.types.Task;
|
||||||
import org.junit.jupiter.api.BeforeEach;
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.time.LocalDate;
|
import java.time.LocalDate;
|
||||||
import java.time.MonthDay;
|
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
|
|
||||||
import static org.junit.jupiter.api.Assertions.*;
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
import static org.mockito.Mockito.*;
|
import static org.mockito.Mockito.*;
|
||||||
|
|
||||||
class GardenScheduleTest {
|
class GardenScheduleTest {
|
||||||
LocalDate exampleStartDate = LocalDate.of(2020, 2, 2);
|
|
||||||
TaskList taskList;
|
TaskList taskList;
|
||||||
PlantList plantList;
|
PlantList plantList;
|
||||||
List<Plant> examplePlantList;
|
|
||||||
List<Task> exampleTaskList;
|
List<Task> exampleTaskList;
|
||||||
List<TaskTemplate> exampleTaskTemplateList;
|
Map<Long, Plant> examplePlantMap;
|
||||||
GardenSchedule model;
|
GardenSchedule model;
|
||||||
|
|
||||||
@BeforeEach
|
@BeforeEach
|
||||||
void setUp() throws IOException, HardinessZoneNotSetException {
|
void setUp() throws IOException {
|
||||||
createExampleTaskList();
|
createExampleTaskList();
|
||||||
createExampleTaskTemplateList();
|
|
||||||
createExamplePlantList();
|
|
||||||
taskList = mockTaskDatabase(exampleTaskList);
|
taskList = mockTaskDatabase(exampleTaskList);
|
||||||
plantList = mockPlantDatabase(examplePlantList);
|
plantList = mockPlantDatabase(examplePlantMap);
|
||||||
model = new GardenSchedule(taskList, plantList);
|
model = new GardenSchedule(taskList, plantList);
|
||||||
}
|
}
|
||||||
|
|
||||||
PlantList mockPlantDatabase(List<Plant> plantList) throws HardinessZoneNotSetException, IOException {
|
|
||||||
PlantList plantDatabase = mock(JsonPlantList.class);
|
|
||||||
when(plantDatabase.getPlantList(HardinessZone.ZONE_8A)).thenReturn(plantList);
|
|
||||||
when(plantDatabase.getPlantById(HardinessZone.ZONE_8A,20)).thenReturn(Optional.of(examplePlantList.get(0)));
|
|
||||||
return plantDatabase;
|
|
||||||
}
|
|
||||||
|
|
||||||
private TaskList mockTaskDatabase(List<Task> exampleTaskList) throws IOException {
|
private TaskList mockTaskDatabase(List<Task> exampleTaskList) throws IOException {
|
||||||
TaskList taskList = mock(JsonTaskList.class);
|
TaskList taskList = mock(JsonTaskList.class);
|
||||||
when(taskList.getTaskList(LocalDate.MIN, LocalDate.MAX)).thenReturn(exampleTaskList);
|
when(taskList.getTaskList(LocalDate.MIN, LocalDate.MAX)).thenReturn(exampleTaskList);
|
||||||
when(taskList.getTaskList(LocalDate.now(), LocalDate.now().plusDays(7 - 1))).thenReturn(exampleTaskList);
|
|
||||||
|
|
||||||
when(taskList.getTaskList(LocalDate.now(), LocalDate.MAX)).thenReturn((exampleTaskList.subList(1, 4)));
|
when(taskList.getTaskList(LocalDate.now(), LocalDate.MAX)).thenReturn((exampleTaskList.subList(1, 4)));
|
||||||
|
|
||||||
|
@ -65,39 +54,18 @@ class GardenScheduleTest {
|
||||||
return taskList;
|
return taskList;
|
||||||
}
|
}
|
||||||
|
|
||||||
void createExampleTaskTemplateList(){
|
private PlantList mockPlantDatabase(Map<Long, Plant> examplePlantMap) {
|
||||||
exampleTaskTemplateList = new ArrayList<>();
|
return new PlantList() {
|
||||||
exampleTaskTemplateList.add(mock(TaskTemplate.class));
|
@Override
|
||||||
exampleTaskTemplateList.add(mock(TaskTemplate.class));
|
public List<Plant> getPlantList(HardinessZone zone) {
|
||||||
exampleTaskTemplateList.add(mock(TaskTemplate.class));
|
return null;
|
||||||
exampleTaskTemplateList.add(mock(TaskTemplate.class));
|
}
|
||||||
when(exampleTaskTemplateList.get(0).generateTask(exampleStartDate, 30)).thenReturn(exampleTaskList.get(0));
|
|
||||||
when(exampleTaskTemplateList.get(1).generateTask(exampleStartDate, 30)).thenReturn(exampleTaskList.get(1));
|
|
||||||
when(exampleTaskTemplateList.get(2).generateTask(exampleStartDate, 30)).thenReturn(exampleTaskList.get(2));
|
|
||||||
when(exampleTaskTemplateList.get(3).generateTask(exampleStartDate, 30)).thenReturn(exampleTaskList.get(3));
|
|
||||||
}
|
|
||||||
|
|
||||||
void createExamplePlantList(){
|
@Override
|
||||||
examplePlantList = new ArrayList<>();
|
public Optional<Plant> getPlantById(HardinessZone zone, long id) {
|
||||||
examplePlantList.add(new Plant(
|
return Optional.ofNullable(examplePlantMap.get(id));
|
||||||
20,
|
}
|
||||||
"summertime onion",
|
};
|
||||||
"Onion, (Allium cepa), herbaceous biennial plant in the amaryllis family (Amaryllidaceae) grown for its edible bulb. The onion is likely native to southwestern Asia but is now grown throughout the world, chiefly in the temperate zones. Onions are low in nutrients but are valued for their flavour and are used widely in cooking. They add flavour to such dishes as stews, roasts, soups, and salads and are also served as a cooked vegetable.",
|
|
||||||
null,
|
|
||||||
"15,30,2",
|
|
||||||
0,
|
|
||||||
"sandy to loamy, loose soil, free of stones",
|
|
||||||
new ArrayList<>(),
|
|
||||||
List.of(
|
|
||||||
new GrowthPhase(MonthDay.of(6, 4), MonthDay.of(12, 4), 0, new WateringCycle(0, 0, null), GrowthPhaseType.HARVEST, HardinessZone.ZONE_8A, List.of(
|
|
||||||
exampleTaskTemplateList.get(0),
|
|
||||||
exampleTaskTemplateList.get(1)
|
|
||||||
)),
|
|
||||||
new GrowthPhase(MonthDay.of(4, 3), MonthDay.of(12, 4), 0, new WateringCycle(0, 0, null), GrowthPhaseType.PLANT, HardinessZone.ZONE_8A, List.of(
|
|
||||||
exampleTaskTemplateList.get(2),
|
|
||||||
exampleTaskTemplateList.get(3)
|
|
||||||
))
|
|
||||||
)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void createExampleTaskList() {
|
void createExampleTaskList() {
|
||||||
|
@ -106,9 +74,15 @@ class GardenScheduleTest {
|
||||||
exampleTaskList.add(new Task("name", "description", LocalDate.now().plusDays(1), 2L));
|
exampleTaskList.add(new Task("name", "description", LocalDate.now().plusDays(1), 2L));
|
||||||
exampleTaskList.add(new Task("name", "description", LocalDate.now(), 1L));
|
exampleTaskList.add(new Task("name", "description", LocalDate.now(), 1L));
|
||||||
exampleTaskList.add(new Task("name", "description", LocalDate.of(9019, 5, 5), 1L));
|
exampleTaskList.add(new Task("name", "description", LocalDate.of(9019, 5, 5), 1L));
|
||||||
exampleTaskList.add(new Task("name", "description", LocalDate.now().minusDays(10), LocalDate.now().plusDays(4),2, 2L));
|
exampleTaskList.add(new Task("name", "description", LocalDate.of(2019, 5, 5), 2L));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void createExamplePlantMap() {
|
||||||
|
examplePlantMap = new HashMap<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void addTask() throws IOException {
|
void addTask() throws IOException {
|
||||||
Task taskToAdd = new Task("name", "description", LocalDate.now(), 1L);
|
Task taskToAdd = new Task("name", "description", LocalDate.now(), 1L);
|
||||||
|
@ -158,22 +132,19 @@ class GardenScheduleTest {
|
||||||
assertEquals(7, dayList.size());
|
assertEquals(7, dayList.size());
|
||||||
|
|
||||||
//Check day 0
|
//Check day 0
|
||||||
assertEquals(2, dayList.get(0).size());
|
assertEquals(1, dayList.get(0).size());
|
||||||
assertTrue(dayList.get(0).contains(exampleTaskList.get(2)));
|
assertEquals(exampleTaskList.get(2), dayList.get(0).get(0));
|
||||||
assertTrue(dayList.get(0).contains(exampleTaskList.get(4)));
|
|
||||||
|
|
||||||
//Check day 1
|
//Check day 1
|
||||||
assertEquals(1, dayList.get(1).size());
|
assertEquals(1, dayList.get(1).size());
|
||||||
assertTrue(dayList.get(1).contains(exampleTaskList.get(1)));
|
assertEquals(exampleTaskList.get(1), dayList.get(1).get(0));
|
||||||
|
|
||||||
//Check day 2
|
//Check day 2
|
||||||
assertEquals(1, dayList.get(2).size());
|
assertEquals(0, dayList.get(2).size());
|
||||||
assertTrue(dayList.get(2).contains(exampleTaskList.get(4)));
|
|
||||||
//Check day 3
|
//Check day 3
|
||||||
assertEquals(0, dayList.get(3).size());
|
assertEquals(0, dayList.get(3).size());
|
||||||
//Check day 4
|
//Check day 4
|
||||||
assertEquals(1, dayList.get(4).size());
|
assertEquals(0, dayList.get(4).size());
|
||||||
assertTrue(dayList.get(4).contains(exampleTaskList.get(4)));
|
|
||||||
//Check day 5
|
//Check day 5
|
||||||
assertEquals(0, dayList.get(5).size());
|
assertEquals(0, dayList.get(5).size());
|
||||||
//Check day 6
|
//Check day 6
|
||||||
|
@ -185,17 +156,4 @@ class GardenScheduleTest {
|
||||||
model.removeTasksForCrop(1L);
|
model.removeTasksForCrop(1L);
|
||||||
verify(taskList, times(1)).removeTasksForCrop(1L);
|
verify(taskList, times(1)).removeTasksForCrop(1L);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
|
||||||
void planTasksForCrop() throws HardinessZoneNotSetException, PlantNotFoundException, IOException {
|
|
||||||
model.planTasksForCrop(new Crop(20, exampleStartDate).withId(30));
|
|
||||||
verify(exampleTaskTemplateList.get(0), times(1)).generateTask(exampleStartDate, 30);
|
|
||||||
verify(exampleTaskTemplateList.get(1), times(1)).generateTask(exampleStartDate, 30);
|
|
||||||
verify(exampleTaskTemplateList.get(2), times(1)).generateTask(exampleStartDate, 30);
|
|
||||||
verify(exampleTaskTemplateList.get(3), times(1)).generateTask(exampleStartDate, 30);
|
|
||||||
verify(taskList, times(1)).saveTask(exampleTaskList.get(0));
|
|
||||||
verify(taskList, times(1)).saveTask(exampleTaskList.get(1));
|
|
||||||
verify(taskList, times(1)).saveTask(exampleTaskList.get(2));
|
|
||||||
verify(taskList, times(1)).saveTask(exampleTaskList.get(3));
|
|
||||||
}
|
|
||||||
}
|
}
|
|
@ -32,7 +32,8 @@
|
||||||
"relativeStartDate": -14,
|
"relativeStartDate": -14,
|
||||||
"relativeEndDate": null,
|
"relativeEndDate": null,
|
||||||
"description": "Take an egg carton and fill it with soil. Put the seedling deep enough so its half covered with soil. Keep it in 10-15 * Celsius with lots of light.",
|
"description": "Take an egg carton and fill it with soil. Put the seedling deep enough so its half covered with soil. Keep it in 10-15 * Celsius with lots of light.",
|
||||||
"interval": null
|
"interval": null,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
@ -53,7 +54,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": null,
|
"relativeEndDate": null,
|
||||||
"description": "When the plants are 20 cm tall, begin hilling the potatoes by gently mounding the soil from the center of your rows around the stems of the plant. Mound up the soil around the plant until just the top few leaves show above the soil. Two weeks later, hill up the soil again when the plants grow another 20 cm.",
|
"description": "When the plants are 20 cm tall, begin hilling the potatoes by gently mounding the soil from the center of your rows around the stems of the plant. Mound up the soil around the plant until just the top few leaves show above the soil. Two weeks later, hill up the soil again when the plants grow another 20 cm.",
|
||||||
"interval": 21
|
"interval": 21,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
@ -74,7 +76,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": null,
|
"relativeEndDate": null,
|
||||||
"description": "Once the foliage has wilted and dried completely, harvest on a dry day. Store in a dark and cool location.",
|
"description": "Once the foliage has wilted and dried completely, harvest on a dry day. Store in a dark and cool location.",
|
||||||
"interval": null
|
"interval": null,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
@ -103,7 +106,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": 0,
|
"relativeEndDate": 0,
|
||||||
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
||||||
"interval": null
|
"interval": null,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
@ -126,7 +130,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": null,
|
"relativeEndDate": null,
|
||||||
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
||||||
"interval": 15
|
"interval": 15,
|
||||||
|
"isOptional": true
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
@ -147,7 +152,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": 14,
|
"relativeEndDate": 14,
|
||||||
"description": "When the leaves turn to a yellowish brown. Do not harvest earlier. The plant will show when it's ready.",
|
"description": "When the leaves turn to a yellowish brown. Do not harvest earlier. The plant will show when it's ready.",
|
||||||
"interval": null
|
"interval": null,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
@ -187,7 +193,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": 0,
|
"relativeEndDate": 0,
|
||||||
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
||||||
"interval": null
|
"interval": null,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
@ -210,7 +217,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": null,
|
"relativeEndDate": null,
|
||||||
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
||||||
"interval": 15
|
"interval": 15,
|
||||||
|
"isOptional": true
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
@ -233,7 +241,8 @@
|
||||||
"relativeStartDate": 0,
|
"relativeStartDate": 0,
|
||||||
"relativeEndDate": 14,
|
"relativeEndDate": 14,
|
||||||
"description": "When ready for harvest, the leaves on your onion plants will start to flop over. This happens at the \"neck\" of the onion and it signals that the plant has stopped growing and is ready for storage. Onions should be harvested soon thereafter",
|
"description": "When ready for harvest, the leaves on your onion plants will start to flop over. This happens at the \"neck\" of the onion and it signals that the plant has stopped growing and is ready for storage. Onions should be harvested soon thereafter",
|
||||||
"interval": null
|
"interval": null,
|
||||||
|
"isOptional": false
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue