Como dijiste, over
(como su contraparte out
) solo se levanta una vez en la lista desplegable. Por otro lado, el evento drag
del que se puede arrastrar se genera cada vez que se mueve el mouse, y parece apropiado para la tarea. Hay, sin embargo, dos problemas con esta estrategia:
drag
se eleva o no la pueda arrastrar en realidad se encuentra en un lanzables,
- incluso en ese caso, el lanzables no se pasa al controlador de eventos.
Una manera de resolver ambos problemas es asociar la lanzables y de la que pueden arrastrarse en el controlador over
, utilizando las instalaciones de jQuery data(), y disociar en las out
y drop
manipuladores:
$("tr.droppable").droppable({
over: function(event, ui) {
if (/* mouse is in top half of row */) {
$(this).removeClass("droppable-below")
.addClass("droppable-above");
}
else {
$(this).removeClass("droppable-above")
.addClass("droppable-below");
}
ui.draggable.data("current-droppable", $(this)); // Associate.
},
out: function(event, ui) {
ui.draggable.removeData("current-droppable"); // Break association.
$(this).removeClass("droppable-above droppable-below");
},
drop: function(event, ui) {
ui.draggable.removeData("current-droppable"); // Break association.
$(this).removeClass("droppable-above droppable-below");
if (/* mouse is in top half of row */) {
// Add new row above the dropped row.
}
else {
// Add new row below the dropped row.
}
}
});
Ahora que la arrastrable conoce el lanzables que está mintiendo otra vez, podemos actualizar la apariencia del elemento dentro de un manejador de eventos drag
:
$(".draggable").draggable({
drag: function(event, ui) {
var $droppable = $(this).data("current-droppable");
if ($droppable) {
if (/* mouse is in top half of row */) {
$droppable.removeClass("droppable-below")
.addClass("droppable-above");
} else {
$droppable.removeClass("droppable-above")
.addClass("droppable-below");
}
}
}
});
El código que sigue es un caso de prueba simple que demuestra esta solución (básicamente llena los vacíos comentados anteriormente y refactores patrones comunes en funciones auxiliares). La configuración desplegable es un poco más complicada que en el ejemplo anterior, principalmente porque las filas de la tabla recién creadas deben ser desplegables como sus hermanos.
Puede ver los resultados en this fiddle.
HTML:
<div class="draggable">New item 1</div>
<div class="draggable">New item 2</div>
<div class="draggable">New item 3</div>
<div class="draggable">New item 4</div>
<div class="draggable">New item 5</div>
<p>Drag the items above into the table below.</p>
<table>
<tr class="droppable"><td>Item 1</td></tr>
<tr class="droppable"><td>Item 2</td></tr>
<tr class="droppable"><td>Item 3</td></tr>
<tr class="droppable"><td>Item 4</td></tr>
<tr class="droppable"><td>Item 5</td></tr>
</table>
CSS:
p {
line-height: 32px;
}
table {
width: 100%;
}
.draggable {
background-color: #d0ffff;
border: 1px solid black;
cursor: pointer;
padding: 6px;
}
.droppable {
background-color: #ffffd0;
border: 1px solid black;
}
.droppable td {
padding: 10px;
}
.droppable-above {
border-top: 3px solid blue;
}
.droppable-below {
border-bottom: 3px solid blue;
}
Javascript:
$(document).ready(function() {
$(".draggable").draggable({
helper: "clone",
drag: function(event, ui) {
var $droppable = $(this).data("current-droppable");
if ($droppable) {
updateHighlight(ui, $droppable);
}
}
});
initDroppable($(".droppable"));
function initDroppable($elements)
{
$elements.droppable({
over: function(event, ui) {
var $this = $(this);
updateHighlight(ui, $this);
ui.draggable.data("current-droppable", $this);
},
out: function(event, ui) {
cleanupHighlight(ui, $(this));
},
drop: function(event, ui) {
var $this = $(this);
cleanupHighlight(ui, $this);
var $new = $this.clone().children("td:first")
.html(ui.draggable.html()).end();
if (isInUpperHalf(ui, $this)) {
$new.insertBefore(this);
} else {
$new.insertAfter(this);
}
initDroppable($new);
}
});
}
function isInUpperHalf(ui, $droppable)
{
var $draggable = ui.draggable || ui.helper;
return (ui.offset.top + $draggable.outerHeight()/2
<= $droppable.offset().top + $droppable.outerHeight()/2);
}
function updateHighlight(ui, $droppable)
{
if (isInUpperHalf(ui, $droppable)) {
$droppable.removeClass("droppable-below")
.addClass("droppable-above");
} else {
$droppable.removeClass("droppable-above")
.addClass("droppable-below");
}
}
function cleanupHighlight(ui, $droppable)
{
ui.draggable.removeData("current-droppable");
$droppable.removeClass("droppable-above droppable-below");
}
});
¿Se puede elegir una opción en lugar de la información del borde? es decir, cuando lo arrastre, aparecerá espacio para que se inserte la fila –
@Gary: gracias por la sugerencia, pero no creo que los sortables me ayuden aquí. Mi problema es específico para agregar nuevas filas, en lugar de ordenar las filas. Mi UI realmente permite al usuario ordenar las filas de la tabla después de agregarlas, y esto se hace usando el plugin jDuend de tablaDnD. –