Merge pull request #988 from dod-ccpo/task-order-cancel
Save draft TO on cancel
This commit is contained in:
@@ -30,9 +30,9 @@ class BaseForm(FlaskForm):
|
|||||||
_data[field] = None
|
_data[field] = None
|
||||||
return _data
|
return _data
|
||||||
|
|
||||||
def validate(self, *args, **kwargs):
|
def validate(self, *args, flash_invalid=True, **kwargs):
|
||||||
valid = super().validate(*args, **kwargs)
|
valid = super().validate(*args, **kwargs)
|
||||||
if not valid:
|
if not valid and flash_invalid:
|
||||||
flash("form_errors")
|
flash("form_errors")
|
||||||
return valid
|
return valid
|
||||||
|
|
||||||
|
@@ -53,7 +53,11 @@ class CLINForm(FlaskForm):
|
|||||||
|
|
||||||
def validate(self, *args, **kwargs):
|
def validate(self, *args, **kwargs):
|
||||||
valid = super().validate(*args, **kwargs)
|
valid = super().validate(*args, **kwargs)
|
||||||
if self.start_date.data > self.end_date.data:
|
if (
|
||||||
|
self.start_date.data
|
||||||
|
and self.end_date.data
|
||||||
|
and self.start_date.data > self.end_date.data
|
||||||
|
):
|
||||||
self.start_date.errors.append(
|
self.start_date.errors.append(
|
||||||
translate("forms.task_order.start_date_error")
|
translate("forms.task_order.start_date_error")
|
||||||
)
|
)
|
||||||
|
@@ -21,8 +21,16 @@ def render_task_orders_edit(template, portfolio_id=None, task_order_id=None, for
|
|||||||
else:
|
else:
|
||||||
render_args["form"] = form or TaskOrderForm()
|
render_args["form"] = form or TaskOrderForm()
|
||||||
|
|
||||||
render_args["cancel_url"] = url_for(
|
render_args["cancel_save_url"] = url_for(
|
||||||
"task_orders.portfolio_funding", portfolio_id=portfolio_id
|
"task_orders.cancel_edit",
|
||||||
|
task_order_id=task_order_id,
|
||||||
|
portfolio_id=portfolio_id,
|
||||||
|
save=True,
|
||||||
|
)
|
||||||
|
render_args["cancel_discard_url"] = url_for(
|
||||||
|
"task_orders.cancel_edit",
|
||||||
|
task_order_id=task_order_id,
|
||||||
|
portfolio_id=portfolio_id,
|
||||||
)
|
)
|
||||||
|
|
||||||
return render_template(template, **render_args)
|
return render_template(template, **render_args)
|
||||||
@@ -115,6 +123,36 @@ def submit_form_step_one_add_pdf(portfolio_id=None, task_order_id=None):
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@task_orders_bp.route(
|
||||||
|
"/portfolios/<portfolio_id>/task_orders/form/cancel", methods=["POST"]
|
||||||
|
)
|
||||||
|
@task_orders_bp.route("/task_orders/<task_order_id>/form/cancel", methods=["POST"])
|
||||||
|
@user_can(Permissions.CREATE_TASK_ORDER, message="cancel task order form")
|
||||||
|
def cancel_edit(task_order_id=None, portfolio_id=None):
|
||||||
|
save = http_request.args.get("save", False)
|
||||||
|
if save:
|
||||||
|
form_data = {**http_request.form}
|
||||||
|
form = None
|
||||||
|
if task_order_id:
|
||||||
|
task_order = TaskOrders.get(task_order_id)
|
||||||
|
form = TaskOrderForm(form_data, obj=task_order)
|
||||||
|
else:
|
||||||
|
form = TaskOrderForm(form_data)
|
||||||
|
|
||||||
|
if form.validate(flash_invalid=False):
|
||||||
|
task_order = None
|
||||||
|
if task_order_id:
|
||||||
|
task_order = TaskOrders.update(task_order_id, **form.data)
|
||||||
|
else:
|
||||||
|
task_order = TaskOrders.create(
|
||||||
|
g.current_user, portfolio_id, **form.data
|
||||||
|
)
|
||||||
|
|
||||||
|
return redirect(
|
||||||
|
url_for("task_orders.portfolio_funding", portfolio_id=g.portfolio.id)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@task_orders_bp.route("/task_orders/<task_order_id>/form/step_2")
|
@task_orders_bp.route("/task_orders/<task_order_id>/form/step_2")
|
||||||
@user_can(Permissions.CREATE_TASK_ORDER, message="view task order form")
|
@user_can(Permissions.CREATE_TASK_ORDER, message="view task order form")
|
||||||
def form_step_two_add_number(task_order_id):
|
def form_step_two_add_number(task_order_id):
|
||||||
|
@@ -249,3 +249,12 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.task-order__modal-cancel {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.task-order__modal-cancel_buttons {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
{% extends "portfolios/base.html" %}
|
{% extends "portfolios/base.html" %}
|
||||||
|
|
||||||
{% from "components/sticky_cta.html" import StickyCTA %}
|
{% from "components/sticky_cta.html" import StickyCTA %}
|
||||||
|
{% from "components/modal.html" import Modal %}
|
||||||
|
|
||||||
{% block portfolio_content %}
|
{% block portfolio_content %}
|
||||||
<base-form inline-template>
|
<base-form inline-template>
|
||||||
@@ -26,13 +27,23 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
<a
|
<a
|
||||||
href="{{ cancel_url }}"
|
v-on:click="openModal('cancel')"
|
||||||
class="action-group__action icon-link">
|
class="action-group__action icon-link">
|
||||||
{{ "common.cancel" | translate }}
|
{{ "common.cancel" | translate }}
|
||||||
</a>
|
</a>
|
||||||
</span>
|
</span>
|
||||||
{% endcall %}
|
{% endcall %}
|
||||||
|
|
||||||
|
{% call Modal(name='cancel', dismissable=True) %}
|
||||||
|
<div class="task-order__modal-cancel">
|
||||||
|
<h1>Do you want to save this draft?</h1>
|
||||||
|
<div class="task-order__modal-cancel_buttons">
|
||||||
|
<button formaction="{{ cancel_discard_url }}" class="usa-button usa-button-primary" type="submit">No, delete it</button>
|
||||||
|
<button formaction="{{ cancel_save_url }}" class="usa-button usa-button-primary" type="submit">Yes, save for later</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endcall %}
|
||||||
|
|
||||||
{% include "fragments/flash.html" %}
|
{% include "fragments/flash.html" %}
|
||||||
|
|
||||||
<div class="task-order">
|
<div class="task-order">
|
||||||
|
@@ -12,7 +12,6 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{% set next_button_text = "Next: Add TO Number" %}
|
{% set next_button_text = "Next: Add TO Number" %}
|
||||||
{% set previous_button_link = cancel_url %}
|
|
||||||
{% set step = "1" %}
|
{% set step = "1" %}
|
||||||
|
|
||||||
|
|
||||||
|
@@ -1,20 +1,12 @@
|
|||||||
import pytest
|
import pytest
|
||||||
from flask import url_for
|
from flask import url_for, get_flashed_messages
|
||||||
from datetime import timedelta, date
|
from datetime import timedelta, date
|
||||||
|
|
||||||
from atst.domain.permission_sets import PermissionSets
|
|
||||||
from atst.domain.task_orders import TaskOrders
|
from atst.domain.task_orders import TaskOrders
|
||||||
from atst.models.task_order import Status as TaskOrderStatus
|
from atst.models.task_order import Status as TaskOrderStatus
|
||||||
from atst.models import Attachment, TaskOrder
|
from atst.models import TaskOrder
|
||||||
from atst.utils.localization import translate
|
|
||||||
|
|
||||||
from tests.factories import (
|
from tests.factories import CLINFactory, PortfolioFactory, TaskOrderFactory, UserFactory
|
||||||
CLINFactory,
|
|
||||||
PortfolioFactory,
|
|
||||||
PortfolioRoleFactory,
|
|
||||||
TaskOrderFactory,
|
|
||||||
UserFactory,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
@@ -326,6 +318,72 @@ def test_task_orders_edit_redirects_to_latest_incomplete_step(
|
|||||||
assert expected_step in response.location
|
assert expected_step in response.location
|
||||||
|
|
||||||
|
|
||||||
|
def test_can_cancel_edit_and_save_task_order(client, user_session, task_order, session):
|
||||||
|
user_session(task_order.portfolio.owner)
|
||||||
|
response = client.post(
|
||||||
|
url_for("task_orders.cancel_edit", task_order_id=task_order.id, save=True),
|
||||||
|
data={"number": "7896564324567"},
|
||||||
|
)
|
||||||
|
assert response.status_code == 302
|
||||||
|
|
||||||
|
updated_task_order = session.query(TaskOrder).get(task_order.id)
|
||||||
|
assert updated_task_order.number == "7896564324567"
|
||||||
|
|
||||||
|
|
||||||
|
def test_cancel_can_create_new_to(client, user_session, portfolio):
|
||||||
|
user_session(portfolio.owner)
|
||||||
|
response = client.post(
|
||||||
|
url_for("task_orders.cancel_edit", portfolio_id=portfolio.id),
|
||||||
|
data={"number": "7643906432984"},
|
||||||
|
)
|
||||||
|
assert response.status_code == 302
|
||||||
|
|
||||||
|
|
||||||
|
def test_cancel_edit_does_not_save_invalid_form_input(client, user_session, session):
|
||||||
|
task_order = TaskOrderFactory.create()
|
||||||
|
user_session(task_order.portfolio.owner)
|
||||||
|
bad_data = {"clins-0-jedi_clin_type": "foo"}
|
||||||
|
response = client.post(
|
||||||
|
url_for("task_orders.cancel_edit", task_order_id=task_order.id, save=True),
|
||||||
|
data=bad_data,
|
||||||
|
)
|
||||||
|
assert response.status_code == 302
|
||||||
|
|
||||||
|
# CLINs should be unchanged
|
||||||
|
updated_task_order = session.query(TaskOrder).get(task_order.id)
|
||||||
|
assert updated_task_order.clins == task_order.clins
|
||||||
|
|
||||||
|
|
||||||
|
def test_cancel_edit_on_invalid_input_does_not_flash(
|
||||||
|
app, client, user_session, session
|
||||||
|
):
|
||||||
|
task_order = TaskOrderFactory.create()
|
||||||
|
user_session(task_order.portfolio.owner)
|
||||||
|
|
||||||
|
bad_data = {"clins-0-jedi_clin_type": "foo"}
|
||||||
|
|
||||||
|
response = client.post(
|
||||||
|
url_for("task_orders.cancel_edit", task_order_id=task_order.id, save=True),
|
||||||
|
data=bad_data,
|
||||||
|
)
|
||||||
|
|
||||||
|
assert len(get_flashed_messages()) == 0
|
||||||
|
|
||||||
|
|
||||||
|
def test_cancel_edit_without_saving(client, user_session, session):
|
||||||
|
task_order = TaskOrderFactory.create(number=None)
|
||||||
|
user_session(task_order.portfolio.owner)
|
||||||
|
response = client.post(
|
||||||
|
url_for("task_orders.cancel_edit", task_order_id=task_order.id),
|
||||||
|
data={"number": "7643906432984"},
|
||||||
|
)
|
||||||
|
assert response.status_code == 302
|
||||||
|
|
||||||
|
# TO number should be unchanged
|
||||||
|
updated_task_order = session.query(TaskOrder).get(task_order.id)
|
||||||
|
assert updated_task_order.number is None
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.skip(reason="Reevaluate how form handles invalid data")
|
@pytest.mark.skip(reason="Reevaluate how form handles invalid data")
|
||||||
def test_task_orders_update_invalid_data(client, user_session, portfolio):
|
def test_task_orders_update_invalid_data(client, user_session, portfolio):
|
||||||
user_session(portfolio.owner)
|
user_session(portfolio.owner)
|
||||||
|
Reference in New Issue
Block a user