Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fixed datepicker positioning calculation #2337

Open
wants to merge 2 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
79 changes: 79 additions & 0 deletions tests/unit/datepicker/core.js
Original file line number Diff line number Diff line change
Expand Up @@ -540,4 +540,83 @@ QUnit.test( "mouse", function( assert ) {
"Mouse click inline - next" );
} );

QUnit.test( "position", function( assert ) {
assert.expect( 4 );

var $container = $( "#qunit-fixture" ),
$modal = $container.children('div'),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Make sure lint passes. Because it fails at the lint stage, it doesn't even run tests in half of the cases.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That's a bit fragile to depend on it always being a single div. If you need a reference to this element, can you maybe start from the inp input and go to the parent?

$inputOne = testHelper.init( "#inp" ),
$inputTwo = testHelper.init( "#alt" );

// Undo the weird positioning on the container
$container.css({
position: "static",
top: 0,
left: 0,
height: "5000px",
});
Comment on lines +552 to +557
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let's not change these; this is so that the fixture is not obscuring the test report view if not necessary. Also, I'm not sure if these styles are reset between runs.

If for some reason this has to be visible on the screen, please append another container to #qunit-fixture and style it.


// Position the modal in the middle of the screen
$modal.css({
display: "none", // Initially must be hidden
position: "fixed",
"overflow": "hidden auto",
top: "50%",
left: "50%",
right: "auto",
width: "500px",
height: "500px",
marginTop: "-100px",
marginLeft: "-100px"
});

// Add an internal wrapper around all the children nodes
$modal.wrapInner(`<div id="relative-wrapper" style="position: relative"></div>`)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ES6+ is not currently allowed; ESLint also catches that.

var $wrapper = $modal.children('#relative-wrapper');
$wrapper.prepend(`<div style="height: 300px;"></div>`);

// Wrap inputs with a container
$inputOne.wrap(`<div id="input-one-container" style="height: 200px;"></div>`);
$inputTwo.wrap(`<div id="input-two-container" style="height: 200px;"></div>`);
Comment on lines +578 to +580
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is doing the wrapping after the widget is already initialized. I think we'd rather want to prepare the intended HTML structure before initializing the widget. If you modify the structure of an already running widget, various things may not work correctly and I don't think this issue requires any of that, does it?

Also, are all those modifications really needed? If it gets too heavy, it might be easier to write a separate iframe test where you can put all the style info in a style tag. But let's first verify why all those modifications are needed. Can you explain them all?

Plus, you're wrapping two different inputs with divs with the same ID; that's not compliant.


assert.ok( $modal.is(":hidden"), "Modal is hidden" );

// Set the page scroll position way down the page
var done = assert.async();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If the test is async, please put this declaration at the top of the test, as in other tests.

$(document).ready(function() {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It shouldn't be needed to wrap any code in a test in document ready.

$(window).scrollTop(2000);

// Disable the scrollbar
$('body').css('overflow', 'hidden');
Comment on lines +589 to +590
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this necessary? Why? All such code modifying global elements needs to be reset manually at the end of the test or it may affect other ones.


// Now show the wrapper with the input
$modal.show();

// Initialize datepickers
$modal.find('input').datepicker();

setTimeout(() => {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why is the setTimeout needed, especially with such a long delay?

$("#input-one-container").hide();
$("#input-two-container").css("padding-top", "100px");

var $inputTwo = $("#input-two-container").find("input");

$inputTwo.focus();
assert.ok( $("#ui-datepicker-div").is(":visible"), "Datepicker is visible" );
done();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let's not call done before all the code is run, and we have some below.


// Get the top position of the input and compare it to the datepicker to ensure they're
// both positioned correctly
var inputTop = $inputTwo.offset().top;
var dpTop = $("#ui-datepicker-div").offset().top;
assert.ok( Math.abs(inputTop - dpTop) < 25, "Datepicker is positioned correctly" );

// It should also be left aligned with the input
var inputLeft = $inputTwo.offset().left;
var dpLeft = $("#ui-datepicker-div").offset().left;
assert.ok( Math.abs(inputLeft - dpLeft) < 5, "Datepicker is left aligned with the input" );
}, 200);
});
} );

} );
2 changes: 1 addition & 1 deletion ui/widgets/datepicker.js
Original file line number Diff line number Diff line change
Expand Up @@ -816,7 +816,7 @@ $.extend( Datepicker.prototype, {
}
if ( !$.datepicker._pos ) { // position below input
$.datepicker._pos = $.datepicker._findPos( input );
$.datepicker._pos[ 1 ] += input.offsetHeight; // add the height
$.datepicker._pos[ 1 ] += inst.input.outerHeight(); // add the height
}

isFixed = false;
Expand Down
Loading