Skip to content

Commit 09176cc

Browse files
committed
More JS, Less jQuery (Drag-and-Drop)
1 parent 52947ad commit 09176cc

File tree

3 files changed

+77
-21
lines changed

3 files changed

+77
-21
lines changed

help_docs/method_summary.md

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -163,7 +163,7 @@ self.hover_and_double_click(
163163

164164
self.drag_and_drop(drag_selector, drop_selector,
165165
drag_by="css selector", drop_by="css selector",
166-
timeout=None)
166+
timeout=None, jquery=False)
167167

168168
self.drag_and_drop_with_offset(
169169
selector, x, y, by="css selector", timeout=None)

seleniumbase/fixtures/base_case.py

Lines changed: 17 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -2275,6 +2275,7 @@ def drag_and_drop(
22752275
drag_by="css selector",
22762276
drop_by="css selector",
22772277
timeout=None,
2278+
jquery=False,
22782279
):
22792280
"""Drag and drop an element from one selector to another."""
22802281
self.__check_scope()
@@ -2292,22 +2293,28 @@ def drag_and_drop(
22922293
drag_selector, by=drag_by, timeout=timeout
22932294
)
22942295
self.__demo_mode_highlight_if_active(drag_selector, drag_by)
2295-
self.wait_for_element_visible(
2296+
drop_element = self.wait_for_element_visible(
22962297
drop_selector, by=drop_by, timeout=timeout
22972298
)
22982299
self.__demo_mode_highlight_if_active(drop_selector, drop_by)
22992300
self.scroll_to(drag_selector, by=drag_by)
23002301
drag_selector = self.convert_to_css_selector(drag_selector, drag_by)
23012302
drop_selector = self.convert_to_css_selector(drop_selector, drop_by)
2302-
drag_and_drop_script = js_utils.get_drag_and_drop_script()
2303-
self.safe_execute_script(
2304-
drag_and_drop_script
2305-
+ (
2306-
"$('%s').simulateDragDrop("
2307-
"{dropTarget: "
2308-
"'%s'});" % (drag_selector, drop_selector)
2303+
if not jquery:
2304+
drag_and_drop_script = js_utils.get_js_drag_and_drop_script()
2305+
self.execute_script(
2306+
drag_and_drop_script, drag_element, drop_element, 0, 0, 1, None
2307+
)
2308+
else:
2309+
drag_and_drop_script = js_utils.get_drag_and_drop_script()
2310+
self.safe_execute_script(
2311+
drag_and_drop_script
2312+
+ (
2313+
"$('%s').simulateDragDrop("
2314+
"{dropTarget: "
2315+
"'%s'});" % (drag_selector, drop_selector)
2316+
)
23092317
)
2310-
)
23112318
if self.demo_mode:
23122319
self.__demo_mode_pause_if_active()
23132320
elif self.slow_mode:
@@ -2332,7 +2339,7 @@ def drag_and_drop_with_offset(
23322339
script = js_utils.get_drag_and_drop_with_offset_script(
23332340
css_selector, x, y
23342341
)
2335-
self.safe_execute_script(script)
2342+
self.execute_script(script)
23362343
if self.demo_mode:
23372344
self.__demo_mode_pause_if_active()
23382345
elif self.slow_mode:

seleniumbase/fixtures/js_utils.py

Lines changed: 59 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -151,8 +151,10 @@ def raise_unable_to_load_jquery_exception(driver):
151151

152152

153153
def activate_jquery(driver):
154-
"""If "jQuery is not defined", use this method to activate it for use.
155-
This happens because jQuery is not always defined on web sites."""
154+
"""
155+
If "jQuery is not defined" on a website, use this method to activate it.
156+
This method is needed because jQuery is not always defined on web sites.
157+
"""
156158
try:
157159
# Let's first find out if jQuery is already defined.
158160
driver.execute_script("jQuery('html');")
@@ -163,20 +165,15 @@ def activate_jquery(driver):
163165
pass
164166
jquery_js = constants.JQuery.MIN_JS
165167
add_js_link(driver, jquery_js)
166-
for x in range(25):
168+
for x in range(27):
167169
# jQuery needs a small amount of time to activate.
168170
try:
169171
driver.execute_script("jQuery('html');")
170172
return
171173
except Exception:
174+
if x == 15:
175+
add_js_link(driver, jquery_js)
172176
time.sleep(0.1)
173-
try:
174-
add_js_link(driver, jquery_js)
175-
time.sleep(0.5)
176-
driver.execute_script("jQuery('head');")
177-
return
178-
except Exception:
179-
pass
180177
# Since jQuery still isn't activating, give up and raise an exception
181178
raise_unable_to_load_jquery_exception(driver)
182179

@@ -1021,6 +1018,8 @@ def slow_scroll_to_element(driver, element, browser):
10211018

10221019

10231020
def get_drag_and_drop_script():
1021+
# This script uses jQuery to perform a Drag-and-Drop action.
1022+
# (Requires the Drag-selector and the Drop-selector to work)
10241023
script = r"""(function( $ ) {
10251024
$.fn.simulateDragDrop = function(options) {
10261025
return this.each(function() {
@@ -1078,7 +1077,57 @@ def get_drag_and_drop_script():
10781077
return script
10791078

10801079

1080+
def get_js_drag_and_drop_script():
1081+
# HTML5 Drag-and-Drop script (Requires extra parameters to work)
1082+
# param1 (WebElement): Source element to drag
1083+
# param2 (WebElement): Target element for the drop (Optional)
1084+
# param3 (int): Optional - Drop offset x relative to the target
1085+
# param4 (int): Optional - Drop offset y relative to the target
1086+
# param4 (int): Optional - Delay in milliseconds (default = 1ms)
1087+
# param5 (string): Optional - Key pressed (ALT or CTRL or SHIFT)
1088+
script = """var t=arguments,e=t[0],n=t[1],i=t[2]||0,o=t[3]||0,r=t[4]||1,
1089+
a=t[5]||'',s='alt'===a||'\ue00a'===a,l='ctrl'===a||'\ue009'===a,
1090+
c='shift'===a||'\ue008'===a,u=e.ownerDocument,
1091+
f=e.getBoundingClientRect(),g=n?n.getBoundingClientRect():f,
1092+
p=f.left+f.width/2,d=f.top+f.height/2,h=g.left+(i||g.width/2),
1093+
m=g.top+(o||g.height/2),v=u.elementFromPoint(p,d),
1094+
y=u.elementFromPoint(h,m);if(!v||!y){
1095+
var E=new Error('source or target element is not interactable');
1096+
throw E.code=15,E}var _={constructor:DataTransfer,effectAllowed:null,
1097+
dropEffect:null,types:[],files:Object.setPrototypeOf([],null),
1098+
_items:Object.setPrototypeOf([],{add:function(t,e){
1099+
this[this.length]={_data:''+t,kind:'string',
1100+
type:e,getAsFile:function(){},getAsString:function(t){t(this._data)}},
1101+
_.types.push(e)},remove:function(t){
1102+
Array.prototype.splice.call(this,65535&t,1),_.types.splice(65535&t,1)},
1103+
clear:function(t,e){this.length=0,_.types.length=0}}),
1104+
setData:function(t,e){this.clearData(t),this._items.add(e,t)},
1105+
getData:function(t){for(var e=this._items.length;
1106+
e--&&this._items[e].type!==t;);return e>=0?this._items[e]._data:null},
1107+
clearData:function(t){for(var e=this._items.length;
1108+
e--&&this._items[e].type!==t;);this._items.remove(e)},
1109+
setDragImage:function(t){}};function w(t,e,n,i){
1110+
for(var o=0;o<e.length;++o){var r=u.createEvent('MouseEvent');
1111+
r.initMouseEvent(e[o],!0,!0,u.defaultView,0,0,0,p,d,l,s,c,!1,0,null),
1112+
t.dispatchEvent(r)}i&&setTimeout(i,n)}function D(t,e,n,i){
1113+
var o=u.createEvent('DragEvent');o.initMouseEvent(
1114+
e,!0,!0,u.defaultView,0,0,0,p,d,l,s,c,!1,0,null),Object.setPrototypeOf(
1115+
o,null),o.dataTransfer=_,Object.setPrototypeOf(o,DragEvent.prototype),
1116+
t.dispatchEvent(o),i&&setTimeout(i,n)}
1117+
'items'in DataTransfer.prototype&&(_.items=_._items),
1118+
w(v,['pointerdown','mousedown'],1,function(){
1119+
for(var t=v;t&&!t.draggable;)t=t.parentElement;if(t&&t.contains(v)){
1120+
var e=y.getBoundingClientRect();D(v,'dragstart',r,function(){
1121+
var t=y.getBoundingClientRect();p=t.left+h-e.left,d=t.top+m-e.top,D(
1122+
y,'dragenter',1,function(){D(y,'dragover',r,
1123+
function(){D(u.elementFromPoint(p,d),'drop',1,function(){D(v,'dragend',
1124+
1,function(){w(u.elementFromPoint(p,d),
1125+
['mouseup','pointerup'])})})})})})}})"""
1126+
return script
1127+
1128+
10811129
def get_drag_and_drop_with_offset_script(selector, x, y):
1130+
# This script uses pure JS (No jQuery)
10821131
script_a = """
10831132
var source = document.querySelector("%s");
10841133
var offsetX = %f;

0 commit comments

Comments
 (0)