Fix date pickers in metering modal

The metering forms were using the date picker functionality from the
d3 line chart library, this used to work when it was loaded as a whole
page as the d3 libary would run intending to setup any d3 graphs however
now it is being loaded as a modal the elements that would be affected by
the d3 code do not exist at page load time.

In order to get the date picker events to be added to the elements on
the form when the modal loads, the JS in the forms template has been
extended in this patch to support adding those events.

This patch also changes the data attributes on the form elements so that
they do not conflict with any d3 line chart elements.

Alongside the javascript this patch fixes bug to do with parsing the
dates in metering utils and adds UTs to prevent regression.

Change-Id: I4e239daa03b2f54e434254bac48ba0cceb037b5d
Closes-Bug: 1427756
This commit is contained in:
Sam Betts 2015-03-30 16:49:24 +01:00
parent 11fa3ed6bd
commit 0d010f601c
10 changed files with 88 additions and 61 deletions

View File

@ -757,13 +757,8 @@ horizon.d3_line_chart = {
* connected to charts. * connected to charts.
*/ */
var bind_datepicker_change = function(settings) { var bind_datepicker_change = function(settings) {
var now = new Date();
$(datepicker_selector).each(function() { horizon.datepickers.add(datepicker_selector);
var el = $(this);
el.datepicker({format: 'yyyy-mm-dd',
setDate: new Date(),
showButtonPanel: true});
});
delegate_event_and_refresh_charts(datepicker_selector, 'changeDate', settings); delegate_event_and_refresh_charts(datepicker_selector, 'changeDate', settings);
}; };

View File

@ -0,0 +1,13 @@
horizon.datepickers = {
add: function(selector) {
$(selector).each(function () {
var el = $(this);
el.datepicker({
format: 'yyyy-mm-dd',
setDate: new Date(),
showButtonPanel: true,
language: horizon.datepickerLocale
});
});
}
};

View File

@ -0,0 +1,32 @@
horizon.metering = {
init_create_usage_report_form: function() {
horizon.datepickers.add('input[data-date-picker="True"]');
horizon.metering.add_change_event_to_period_dropdown();
horizon.metering.show_or_hide_date_fields();
},
init_stats_page: function() {
if (typeof horizon.d3_line_chart !== 'undefined') {
horizon.d3_line_chart.init("div[data-chart-type='line_chart']",
{'auto_resize': true});
}
horizon.metering.add_change_event_to_period_dropdown();
horizon.metering.show_or_hide_date_fields();
},
show_or_hide_date_fields: function() {
$("#date_from .controls input, #date_to .controls input").val('');
if ($("#id_period").find("option:selected").val() === "other"){
$("#id_date_from, #id_date_to").parent().parent().show();
return true;
} else {
$("#id_date_from, #id_date_to").parent().parent().hide();
return false;
}
},
add_change_event_to_period_dropdown: function() {
$("#id_period").change(function(evt) {
if (horizon.metering.show_or_hide_date_fields()) {
evt.stopPropagation();
}
});
}
};

View File

@ -76,6 +76,7 @@
<script src='{{ STATIC_URL }}horizon/js/horizon.accordion_nav.js'></script> <script src='{{ STATIC_URL }}horizon/js/horizon.accordion_nav.js'></script>
<script src='{{ STATIC_URL }}horizon/js/horizon.communication.js'></script> <script src='{{ STATIC_URL }}horizon/js/horizon.communication.js'></script>
<script src='{{ STATIC_URL }}horizon/js/horizon.datepickers.js'></script>
<script src='{{ STATIC_URL }}horizon/js/horizon.forms.js'></script> <script src='{{ STATIC_URL }}horizon/js/horizon.forms.js'></script>
<script src='{{ STATIC_URL }}horizon/js/horizon.formset_table.js'></script> <script src='{{ STATIC_URL }}horizon/js/horizon.formset_table.js'></script>
<script src='{{ STATIC_URL }}horizon/js/horizon.instances.js'></script> <script src='{{ STATIC_URL }}horizon/js/horizon.instances.js'></script>
@ -92,6 +93,7 @@
<script src='{{ STATIC_URL }}horizon/js/horizon.templates.js'></script> <script src='{{ STATIC_URL }}horizon/js/horizon.templates.js'></script>
<script src='{{ STATIC_URL }}horizon/js/horizon.users.js'></script> <script src='{{ STATIC_URL }}horizon/js/horizon.users.js'></script>
<script src='{{ STATIC_URL }}horizon/js/horizon.membership.js'></script> <script src='{{ STATIC_URL }}horizon/js/horizon.membership.js'></script>
<script src='{{ STATIC_URL }}horizon/js/horizon.metering.js'></script>
<script src='{{ STATIC_URL }}horizon/js/horizon.networktopology.js'></script> <script src='{{ STATIC_URL }}horizon/js/horizon.networktopology.js'></script>
<script src='{{ STATIC_URL }}horizon/js/horizon.d3piechart.js'></script> <script src='{{ STATIC_URL }}horizon/js/horizon.d3piechart.js'></script>
<script src='{{ STATIC_URL }}horizon/js/horizon.heattop.js'></script> <script src='{{ STATIC_URL }}horizon/js/horizon.heattop.js'></script>

View File

@ -34,12 +34,10 @@ class UsageReportForm(forms.SelfHandlingForm):
choices=PERIOD_CHOICES) choices=PERIOD_CHOICES)
date_from = forms.DateField(label=_("From"), required=False, date_from = forms.DateField(label=_("From"), required=False,
widget=forms.TextInput( widget=forms.TextInput(
attrs={'data-line-chart-command': attrs={'data-date-picker': True}))
'date_picker_change'}))
date_to = forms.DateField(label=_("To"), required=False, date_to = forms.DateField(label=_("To"), required=False,
widget=forms.TextInput( widget=forms.TextInput(
attrs={'data-line-chart-command': attrs={'data-date-picker': True}))
'date_picker_change'}))
def clean_date_from(self): def clean_date_from(self):
period = self.cleaned_data['period'] period = self.cleaned_data['period']

View File

@ -27,31 +27,12 @@
{% block modal-js %} {% block modal-js %}
<script type="text/javascript"> <script type="text/javascript">
if (typeof $ !== 'undefined') { if (typeof horizon.metering !== 'undefined') {
show_hide_datepickers(); horizon.metering.init_create_usage_report_form();
} else { } else {
addHorizonLoadEvent(function() { addHorizonLoadEvent(function() {
show_hide_datepickers(); horizon.metering.init_create_usage_report_form();
}); });
} }
function show_hide_datepickers() {
$("#id_period").change(function(evt) {
// Enhancing behaviour of selectbox, on 'other' value selected, I don't
// want to refresh, but show hide the date fields
if ($(this).find("option:selected").val() === "other"){
evt.stopPropagation();
$("#date_from .controls input, #date_to .controls input").val('');
$("#id_date_from, #id_date_to").parent().parent().show();
} else {
$("#id_date_from, #id_date_to").parent().parent().hide();
}
});
if ($("#id_period").find("option:selected").val() === "other"){
$("#id_date_from, #id_date_to").parent().parent().show();
} else {
$("#id_date_from, #id_date_to").parent().parent().hide();
}
}
</script> </script>
{% endblock %} {% endblock %}

View File

@ -166,35 +166,11 @@
</div> </div>
<script type="text/javascript"> <script type="text/javascript">
if (typeof horizon.d3_line_chart !== 'undefined') {
horizon.d3_line_chart.init("div[data-chart-type='line_chart']",
{'auto_resize': true});
}
if (typeof $ !== 'undefined') { if (typeof $ !== 'undefined') {
show_hide_datepickers(); horizon.metering.init_stats_page();
} else { } else {
addHorizonLoadEvent(function() { addHorizonLoadEvent(function() {
show_hide_datepickers(); horizon.metering.init_stats_page();
}); });
} }
function show_hide_datepickers() {
$("#date_options").change(function(evt) {
// Enhancing behaviour of selectbox, on 'other' value selected, I don't
// want to refresh, but show hide the date fields
if ($(this).find("option:selected").val() == "other"){
evt.stopPropagation();
$("#date_from input, #date_to input").val('');
$("#date_from, #date_to").show();
} else {
$("#date_from, #date_to").hide();
}
});
if ($("#date_options").find("option:selected").val() == "other"){
$("#date_from, #date_to").show();
} else {
$("#date_from, #date_to").hide();
}
}
</script> </script>

View File

@ -618,6 +618,11 @@ td.loading {
/* Forms */ /* Forms */
.datepicker {
z-index: $zindex-popover !important;
}
.datepicker input{ .datepicker input{
@extend .form-control; @extend .form-control;
} }

View File

@ -13,10 +13,12 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import datetime
import uuid import uuid
from openstack_dashboard.test import helpers as test from openstack_dashboard.test import helpers as test
from openstack_dashboard.utils import filters from openstack_dashboard.utils import filters
from openstack_dashboard.utils import metering
class UtilsFilterTests(test.TestCase): class UtilsFilterTests(test.TestCase):
@ -38,3 +40,26 @@ class UtilsFilterTests(test.TestCase):
def test_reject_random_string(self): def test_reject_random_string(self):
val = '55WbJTpJDf' val = '55WbJTpJDf'
self.assertRaises(ValueError, filters.get_int_or_uuid, val) self.assertRaises(ValueError, filters.get_int_or_uuid, val)
class UtilsMeteringTests(test.TestCase):
def test_calc_date_args_strings(self):
date_from, date_to = metering.calc_date_args(
"2012-04-11", "2012-04-12", "other")
self.assertTrue(type(date_from) is datetime.datetime)
self.assertTrue(type(date_to) is datetime.datetime)
self.assertEqual(str(date_from.tzinfo), "UTC")
self.assertEqual(str(date_to.tzinfo), "UTC")
def test_calc_date_args_datetime_dates(self):
date_from, date_to = metering.calc_date_args(
datetime.date(2012, 4, 11), datetime.date(2012, 4, 12), "other")
self.assertTrue(type(date_from) is datetime.datetime)
self.assertTrue(type(date_to) is datetime.datetime)
self.assertEqual(str(date_from.tzinfo), "UTC")
self.assertEqual(str(date_to.tzinfo), "UTC")
def test_calc_date_args_invalid(self):
self.assertRaises(
ValueError, metering.calc_date_args, object, object, "other")

View File

@ -61,7 +61,7 @@ def calc_date_args(date_from, date_to, date_options):
try: try:
if date_from: if date_from:
date_from = pytz.utc.localize( date_from = pytz.utc.localize(
datetime.datetime.strptime(date_from, "%Y-%m-%d")) datetime.datetime.strptime(str(date_from), "%Y-%m-%d"))
else: else:
# TODO(lsmola) there should be probably the date # TODO(lsmola) there should be probably the date
# of the first sample as default, so it correctly # of the first sample as default, so it correctly
@ -70,7 +70,7 @@ def calc_date_args(date_from, date_to, date_options):
pass pass
if date_to: if date_to:
date_to = pytz.utc.localize( date_to = pytz.utc.localize(
datetime.datetime.strptime(date_to, "%Y-%m-%d")) datetime.datetime.strptime(str(date_to), "%Y-%m-%d"))
# It returns the beginning of the day, I want the end of # It returns the beginning of the day, I want the end of
# the day, so I add one day without a second. # the day, so I add one day without a second.
date_to = (date_to + datetime.timedelta(days=1) - date_to = (date_to + datetime.timedelta(days=1) -