slow! / Simple Form Saver

// ==UserScript==
// @name Simple Form Saver
// @version  1.9.1
// @updateURL https://openuserjs.org/meta/slow!/Simple_Form_Saver.meta.js
// @namespace SFS92
// @description Provides a small red button at the top left, click it to save or to fill in forms on the web. Auto-fill, click replay and pasting raw info into forms also featured.
// @license     GPL-3.0
// @copyright   2017, slow! (https://openuserjs.org/users/slow!)
// @icon https://bit.ly/1Qre8Je
// @include *
// @run-at   document-end
// @grant GM_registerMenuCommand
// @grant GM_log
// @grant GM_getResourceText
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_deleteValue
// @grant GM_addStyle
// @grant GM_xmlhttpRequest
// @require https://code.jquery.com/jquery-2.2.1.js
// @require https://code.jquery.com/ui/1.11.4/jquery-ui.js
// @require https://github.com/SloaneFox/code/raw/master/GM_registerMenuCommand_Submenu_JS_Module.js
// @resource jqueryuiCss https://code.jquery.com/ui/1.11.4/themes/vader/jquery-ui.css
// ==/UserScript==
//
// @run-at   document-start //problem, if at start, GM_registerMenuCommand does nothing.
//
// Data other than settings etc. is stored as GM get/setValues named: [current profile]hostFormsList and formsAddressBook.
//

var uwin=unsafeWindow;
uwin.sfs_reddot_running=true;

var old_GM_log=GM_log;
log=function(t) { console.info(t); old_GM_log(t); }; //if (!Chrome) old_GM_log(t);}; 
log2=log;
log2=function(){};
log=function(){};

if (/SFS:KeepFrame/.test(window.name)) {
   window.name=window.name.replace(/SFS:KeepFrame/,"");
   window.onbeforeunload = function(){
      return 'Are you sure you want to leave?  You may want to set Simple Form Saver values from GM menu for frame.';
   };
}
//
// History
//
// updated Dec 2017.   v1.9.1 Added new menu command to manually set active input value
// updated Dec 2016.   v1.9.0 Added user profiles which allows multiple stores of form data on same page.  Removed complex chrome adapter, removed address book functions. Uses github due to traffic at openuserjs.
// updated Sept 2013.  v1.6.3 Can add further clicks to page
// updated Sept 2013.  v1.5.1 Address/Notebook (alt-a), unhide password & Export/Import added.
// updated Sept 2013.  v1.5.0 Updated to coax dynamic forms to work.
// updated for firefox 22.0
// updated March 2013.  Fixed Chrome issue with recording a click.  v.1.3.2
// updated March 2013.  Restricted regexp page matching. v.1.3.1
// updated March 2013.  Dialog window in new version of browser was tiny so text was unviewable, using own versions of dialog boxes, alert, confirm and prompt.  v.1.3.0
// updated February 2011.  See userscripts.org description page for this script regarding change in tick box sizing.  Changes only by request.
// updated January 2011.  Updated for Greasemonkey 0.9, javascript's basic function, eval(),  no longer works fully in GM.
// updated January 2011.  Updated for Firefox 4.
// updated December 2010.  Allows editing of auto-click target element.
// updated 17 October 2010.  Google Chrome platform adaptation.
// updated 7 October 2010.  Added feature allowing data to be pasted directly into a form.  See last part of description below.
// updated 1st of March 2010.  Auto delete an element on a page, select record mouse click and do control-shift-click on the element.  Remove iframes.
// updated 5th of February 2010.  Can fill in pop-ups and other elements that appear during use of a page.
// updated 29th of Janaury 2010.  Facility for editing form storage manually.
// updated 27th of Janaury 2010.  Problem with duplicate input names solved.
// updated 25th of Janaury 2010.  Handling of input name changes and multiline textareas added.
// updated 18th of Janaury 2010.  Added ability to record a mouse click on a page with autoplay.
// updated 16th of Janaury 2010.  Enable selective fill-in of forms on user selected pages.
// updated 16th of Janaury 2010.  Enable iframes with forms to work, and forms that auto update on change.

//
// Globals
//

var inputs_on_page=0, text_inputs=0, no_of_inputs=0, special_ta;
with (window.document)
   var selects=getElementsByTagName("select"), tareas=getElementsByTagName("textarea"), rinputs=getElementsByTagName("input");
var inputs=[], input_names={}, old_elems=[];
var automatic=false;
var reddot=true, suspend_replay=true, full_icon=false, element_to_highlight, logos=[];
var hash_host_list = new Object();
var page_key, site, page_object, regexp_page_match, href, form_data, click, recording;
var img, img2, link;
var iframe;
var pasteWindow;
var auto_fill=false, no_element_for_click=true, prev_msg, msg, unhide, init_fin, await_click;
var ws=window.setTimeout, write_once=true; //noscript can block timeout.
var widener="\n___________________________________________________\n";
var bullet="\u26ab", bullet_tab="\u26ab\t", tab_bullet="\t\u26ab",separator=":\u200e:", mdash_char = "\u2014",
    bullet_regexp=/\s*\u26ab\s*/g, 
    tick = "\u2714", ex="\u2717"; //  bullet_regexp=RegExp("\s*\u26ab\s*", "g"); 
var data_version=1.2;
var awaiting_an_option;
var addedByLine="\nAdded by userscript, Simple Form Saver.", userProfileLine;
var stringify=JSON.stringify, parse=JSON.parse;
chromeInit();

function setValue(n,v) { return GM_setValue(n,stringify(v));}

function getValue(n,v) { if (v===undefined) v=null; return parse(GM_getValue(n,stringify(v))); }

var hostFormsList; // gets set to: profiles.current+"hostFormsList";
var profiles=new profile_bridge();
userProfileLine=profiles.current ? "\nUser profile: "+profiles.current+ "\t(non-default->icon decor)" : "";
var otherProfiles=profiles.checkOtherProfiles();

//console.log("userProfile:"+profiles.current+".","profiles_list:"+profiles+".","hostFormsList:"+hostFormsList+".");

String.prototype.indexOfRegExp = function(regex, nomatch){ var match = this.match(regex); if (match) return this.indexOf(match[0]); else return (nomatch==undefined ? -1 : nomatch);};
String.prototype.trim = function () {    return this.replace(/^\s*|\s*$/g,""); }; //all space chars from wither end.
String.prototype.splitOnce = function (r) {   r=RegExp(r.source||r, "g"); var a=r.exec(this); return [ this.substring(0, a?a.index:""), this.substr(r.lastIndex) ] ; };
String.prototype.tail = function (n) {  var last_pos = this.length - n; if (last_pos < 0) last_pos=0;  return ( n < this.length ? "..." : "" )+this.substr(this.length-n); };
Date.prototype.slashFormat=function(){var dd=this.getDate();if(dd<10)dd='0'+dd;var mm=this.getMonth()+1;if(mm<10)mm='0'+mm;var yyyy=this.getFullYear();return String(mm+"\/"+dd+"\/"+yyyy)};
Date.prototype.ampmFormat=function() { var ap = "am", hour=this.getHours(), mins=this.getMinutes(); if (hour>11) ap = "pm";  if (hour   > 12)  hour = hour - 12; if (hour   == 0)  hour = 12; return hour+":"+(mins<10?"0"+mins:mins)+" "+ap;};
log("Form saver, at "+location+", readyState:"+document.readyState);

if (!chrome) loader();

function loader() { try {
    //   addEventListener("abort", function(){log("ERROR");});
    //   addEventListener("load", function() {
    if (document.readyState=="complete") main();   
    addEventListener("DOMContentLoaded", function() {
	main();
	var da=document.activeElement;
	if (getValue("paste_shortcut", false)) {
	    if (uwin.focusControl)
		uwin.focusControl.blur();
	    else if(da) da.blur();
	}
    },0);
} catch(e) { GM_log(e); throw(e) }
}
//
//Mainline
//
function main() {
    //console.log("Form saver, main(), this is:",this);

    GM_addStyle(jqueryui_dialog_css);
    with (window.document) { // init data not there at document-start
	selects=getElementsByTagName("select"), tareas=getElementsByTagName("textarea"), 
	rinputs=getElementsByTagName("input");
    }
    
    if (window.parent != window) {
	iframe=true;
	//    return;
    }
    readPersistentData();
    //setTimeout(function(){updateValueTitles();},1000);
    //log("suspend_replay: "+suspend_replay);
    //log("page_object.automatic "+(page_object&&page_object.automatic));
    //log("startup, form_data: "+uneval(form_data)+", page key:"+page_key+", regexp match: "+regexp_page_match
    //+ ", inputs "+inputs.length+", jq ips "+$("input").length);
    //if (!chrome) this.submenuModule.register("Simple Form Saver");
    this.submenuModule.register("Simple Form Saver",null,"#cccccc","#002a00");        //"#103010");
    //this.submenuModule.register("Simple Form Saver",null,"#3f005e","#ffffee");
    registerMenus();
    if (hash_host_list.msg || hash_host_list.prev_msg) {
	msg=hash_host_list.msg;
	prev_msg=hash_host_list.prev_msg;
	if ( ! checkIfIntervalLongEnough() && msg) 
	    window.status="Form saver msg:   "+(msg);
	hash_host_list.prev_msg=msg;
	prev_msg=msg;
	if ( ! msg) delete hash_host_list.prev_msg;
	delete hash_host_list.msg;
	persistData();
    }
    if (inputs.length==0 && ! (form_data||click)) {
	log("No inputs");
	return;
    }
   if (getXPathElem())
      no_element_for_click=false;
   if ( page_object.automatic || (click && ! suspend_replay)) {
       addIcon(true); //will register menus
       //console.log("Check interval then fill or replay click");
      if (checkIfIntervalLongEnough()) {
	 auto_fill=true;
	  if (page_object.automatic !== false
	      && (page_object.automatic || ! no_element_for_click)
	     && ! regexp_page_match) {
	    fillForm(false, true);
	 }
	  //console.log("interval passed, Filled form now replay click."+click);
	  replayClick();
	  inOutSet();
	  auto_fill=false;
      }
   }
   if (page_object.click_evidence) 
      delete page_object.click_evidence.sealed;
   
   if ( ! link )
      addIcon(true); //will register menus
   if (/Simple Form Saver replayed/.test(msg)) {
      inOutSet(true);
   }   
   //keepEyeOnIcon();
    if (click && suspend_replay && ! page_object.automatic===true    ) { GM_log("Suppressed replay on newly loaded page, auto-replay is suspended, to replay click icon.") }
    //console.log("onbeforeunload pd:",Object.getOwnPropertyDescriptor(window,"onbeforeunload"));
    //console.log(window.parent != window,"sfs set window.onbeforeunload",onbeforeunload,unsafeWindow.onbeforeunload);
    window.onbeforeunload=function(e) { 
      if (recording) { 
	 var dialog=$("#sfsconfirm3"), openeddialogs;
	 if (dialog.length && dialog.parent().css("display")!="none") 
	    openeddialogs=true;

	 log("onbeforeunload "+openeddialogs+" "+e.target.tagName+" cc "+await_click+" "+uneval(e)+" act:"+document.activeElement.tagName);
	 var roll=""; for (i in e) roll+=i+" "+e[i]+"\n";
	 var pn=e.explicitOriginalTarget.parentNode;
	 log(" "+e.eventPhase+" "
	     +e.originalTarget+" "
	     +(pn?pn.tagName+" ":"")
	     +e.timeStamp);
	 var pev={}
	 pev.target=e.explicitOriginalTarget.parentNode;
	 if (await_click) setTimeout(function() { log("call cl"); recordClick(pev);}, 100);
	 if(await_click || openeddialogs) {
	    var p=prompt_interruption, note="No dialogs open"; //!!! for jdialog???
	    if (prompt_interruption) { sprompt(p.a,p.b,p.c); }
	    log("block exit, call confirm");
	    confirm(dialog.parent().css("display")+".  The page on which you are recording has in fact detected the click and wishes to unload.  "
		    +"However, you must complete interaction with the recording dialogs beforehand."
		    +"\n\nPlease finish with recording dialogue in other window and ONLY then click OK or Cancel here in this window.  A window may flash up momentarily, just ignore it.  "
		   );
	    interrupted=true;
	 }
      }//end if recording
   }; //end beforeunload
    //console.log("done.",window.onbeforeunload, ", pagewin:",unsafeWindow.onbeforeunload);
   window.addEventListener("unload", function(){
       //readPersistentData();
       var click_evidence; try { click_evidence=parse(localStorage.click_evidence||0);} catch(e){console.log("old json click_evidence");}
       localStorage.removeItem("click_evidence");
       page_object.click_evidence=click_evidence;
       if (page_object.click_evidence) {
	 if ( page_object.click_evidence.prejudice) 
	     delete page_object.click_evidence;
	   else
	       page_object.click_evidence.sealed=true;
	   localStorage.click_evidence=stringify(page_object.click_evidence);
	   //log("unload ",page_object.click_evidence,page_object.click_evidence.prejudice);
	   //persistData();
      }
       if (pasteWindow && !pasteWindow.closed) pasteWindow.close();
       }, false); //end unload
   if (click || page_object.automatic) {
      uwin.addEventListener("DOMNodeInserted", handlePopups, false);
   }
   if (getValue("paste_shortcut", false)) {
      addEventListener("keypress", function(e) {
	  if (e.charCode==118 && e.ctrlKey) //118 is v
	      if ( ! /INPUT|TEXTAREA/.test(e.target.tagName)) {
		  special_ta=document.createElement("textarea");
		  special_ta.id="speciality";
		  document.body.appendChild(special_ta);
		  //special_ta.focus();
		  e.preventDefault();   
		  e.stopPropagation();
		  setTimeout(function() {
		      log("tout"); 
		      pasteIntoForm();
		      document.body.removeChild(special_ta);
		      special_ta=null;
		  }, 2);
	      }
	 return true;
	 
      },0);
   }
   init_fin=true;
}

function saveFormData(nosave) {  try {
   var output_string = "";
   var  i, plicate={}, saved_data={}, nothing=true;
   if (regexp_page_match && page_object.form_data && !nosave) saved_data=page_object.form_data;
   log("save data, beg "+inputs.length);
   for(i=0; i < inputs.length; i++) {
      var input=inputs[i];
      var value=input.value;
      var type=input.type;
      if ( value == ""  && /text|textarea/.test(type))
	 continue;
      nothing=false;
      var name=input.name;
      if ( ! name) name="yyNoName"+i;
      var checked=input.checked;
      value = value.replace(/\r\n/g,"\u200d");
      value = value.replace(/\n/g,"\u200d");
      if (type=="checkbox") {
	 name+=mdash_char+value;
	 if (checked) 
	    value=tick;
	 else
	    value=ex;
      }
      if ( ! plicate[name]) 
	 plicate[name]=[];
      plicate[name].push(value) //save name and value in case more than one input with same name.
      
      if ( type == "radio")
	 if ( ! checked ) {
	    plicate[name].pop()
	    continue;
	 }
      if (  type  ==   "select-multiple"  ) { 
	 options_list = []
	 for ( var index=0; index < input.options.length; index++ )
	    if ( input.options.item(index).selected )
	       options_list.push(index);
	 value = options_list;
      }
      if (  type  ==   "select-one"  ) {
	 if (input.selectedIndex==-1) continue;
	 for ( var index=0; index < input.options.length; index++ )
	    if ( input.options.item(index).selected )
	       value=index;
	 if (value.substr)
	    value=input.selectedIndex;
      }	    
      log("Save On  i:"+i+".  Elem_name: "+name+".   type: "+type+".  elem_val:"+value+"."+", typeof value: "+typeof value+", ply: "+plicate[name].length);
      saved_data[name]={};
      saved_data[name].v=value;
      saved_data[name].i=i;
      if (type=="password") {
	 saved_data[name].pw=1;
      }
      if (plicate[name].length > 1)
	 saved_data[name].p=plicate[name];//.slice();
      else
	 delete saved_data[name].p;
      log("Saved Data, val: "+saved_data[name].v+".   i: "+saved_data[name].i);
   } // end for
   var request=true;
   if ( ! nothing || nosave) {
      if (iframe && ! regexp_page_match) 
	 sprompt("***Simple Form Saver***\n\nPage stored: "+page_key+widener+"This form is part of a subpage (an iframe) within the current window.  "
		 +"It is controlled via the icon, or via shortcuts, whereas any forms not in the subpage but in the main window are controlled from the GM menu."
		 +"\n\nYou can get status information via the icon or GM menu and check to which page it is refers (first line of the status information window) "
		 +"\n\nCancel and change the page name below or enter a pattern for which the form data shall be stored.  Any page matching this pattern can then be used for form filling.  Use just the site name for it to be in effect for entire site."
		 +"\n\nHit OK to save as usual for the page: "+page_key
		 , page_key
		 ,function(reply) {
		    
		    if (reply && reply != page_key) {
		       delete hash_host_list[page_key];
		       page_key=reply
		       page_object.form_data=saved_data;
		       log("po "+page_object+"cnt "+ countMembers(page_object));
		       persistData();
		       window.status="Forms saved for this page"
			  +", # input fields stored: "+i+".  At: "+page_key;
		    }
		 }); //end function(reply) 
      page_object.form_data=saved_data;
      log("po "+page_object+"cnt "+ countMembers(page_object));
      persistData();
      window.status="Forms saved for this page"
	 +", # input fields stored: "+i+".  At: "+page_key;
   } //endif ! nothing || nosave
    else {
	if (!nosave)
	    window.status="Nothing on forms to save";
	else return saved_data;
    }
     }  catch(e) {alert("Cannot save form, reload and try again.\n\nError was:"+e+" "+e.lineNumber);throw(e);}
  log("win.status "+window.status+", fd: "+form_data+", po.fd: "+ page_object.form_data+", sd: "+saved_data);
}; //end saveFormData())

function fillForm(once_off, auto, tmp_data) {  try { //also Click replay if necessary
   log("fillForm(once_off: "+once_off+", auto: "+ auto+", tmp: "+uneval(tmp_data));
   var saved_data= tmp_data ? tmp_data : form_data;
   var name, saved_obj, elem_name = "";
   var changed=null, kick, result=true, fill_msg;
   var type, value,  indeterminate, plicate={};
   var input, option;
   //function setMember(obj, member, value) {
   function setMember(member, value) {
      if ( member != value) changed=input.name||true;
      return;
      /* if (eval(obj+"."+member) != value) { */
      /*   changed=true; */
      /*   var js=obj+"."+member+"='"+value+"'"; */
      /*   if (typeof value=="boolean") */
      /*     js=obj+"."+member+"="+value; */
      /*   eval(js); */
      //}
   }
   var plicate={};
   var finputs=inputs;
   if (once_off) finputs=once_off;

   for(var i=0; i<finputs.length; i++) {
      input=finputs[i]; this.input=input;
      name = elem_name = input.name;
      type=input.type;
      elem_value=input.value;
      value=undefined;
      //fakeClick(input);
      log2("Fill, On  i:"+i+".  Elem_name: "+elem_name+".   type: "+type+".  current elem_val: "+elem_value)
      if ( ! elem_name) name="yyNoName"+i;
      if (type=="checkbox")
	 name=elem_name+mdash_char+elem_value;
      saved_obj=saved_data && saved_data[name];
      log2("saved_data["+name+"]="+(saved_obj ? ".v="+saved_obj.v+". (datatype:"+(typeof saved_obj.v)+"), .i= "+saved_obj.i + ".  .p= "+saved_obj.p : " null"))

      if (saved_obj && (saved_obj.i != undefined) && saved_obj.i  != i  && ! once_off) { 
	 if (finputs[saved_obj.i] && finputs[saved_obj.i].name == elem_name) {
	    indeterminate=true;
	 }
	 else {
	    saved_obj.i=i; 
	 }
      }
      if (saved_obj && saved_obj.p ) {
	 log("plicate "+saved_obj.p);
	 if ( ! plicate[name] && saved_data[name].p) {
	    if (!saved_data[name].p.slice)
		saved_data[name].p = convert_obj_to_array(saved_data[name].p);
	    plicate[name]=saved_data[name].p.slice(); // copy 
	    plicate[name].pop();  // top of stack is also the one in form_data.
	 }
	 value=plicate[name].shift();
	 if (value && indeterminate)
	    indeterminate=false;
      }
      if ( ! value && saved_obj && saved_obj.v) value=saved_obj.v;
      log("value: "+value+", indeterminate "+indeterminate);
      if (indeterminate) { indeterminate=false; value=undefined;} // i and saved_obj.i don't match
      if ( value === undefined && ! once_off && type !="radio") {
	 var res=findRenamedInput(name, i);  //checks for value in saved .i index
	 value=res[0];
	 if ( value) { 
	     fill_msg="The "+ordinal(i+1)+' input field, "'+name+'", may have been renamed, try re-saving form information if it is not an automatic renaming by the server.  ';
	     // console.log(fill_msg+"It was filled in with the previously saved value, " 
	     // 		+value+", for the "+ordinal(i+1)+" field,"+ordinal(res[2])+","+res[1]+".  Page: "+page_key);
	 }
      }
      if (value===undefined) {
	 if (type == "radio" ) {
	    // 	input.value="";
	    value=saved_obj?saved_obj.v:"";
	 }
	 else { 
	    if (type=="checkbox" ) {
	       if (input.checked)
		  changed=true;
	       input.checked=false;
	    }
	    log("Skip "+i);
	    continue;
	 }
      } //end if undefined
      //log("Check type, value: "+value+", type: "+type);
      if (value && typeof value == "string") value = value.replace(/\u200d/g, "\r\n");
      if (type=="checkbox") {
	 if (value== tick) 
	 { setMember(input.checked, true); input.checked=true;}
	 else
	 {setMember(input.checked, false); input.checked=false;}
      } 
      else if ((type == "radio") && elem_value == value)
      {setMember(input.checked, true); input.checked=true}
      else if (type  !=  "radio") {
	 if (finputs.tagName=="TEXTAREA") {
	    if (value != elem_value)
	       changed=true;
	 }
	 else 
	    if (type  ==  "select-multiple" )   { 
	       options_list=value;
	       for (var index=0; index < input.options.length; index++) {
		  if (input.options.length <= index ) { result=false; continue }
		  if ( RegExp ("\\b"+index+"\\b").test(options_list)   )  
		  { setMember(input.options.item(index).selected, true); input.options.item(index).selected=true;}
		  else
		  { setMember(input.options.item(index).selected, false); input.options.item(index).selected=false;}
	       }
	    }
	 else if (  type  ==   "select-one"  ) { 
	    log2(" Select, saved index "+value+".  Current selected Index: "+input.selectedIndex+" of "+input.options.length+".  .value "+input.value);
	    if (input.options.length <= value) {  result=false; continue }
	    var index, set_select;
	    try { if (input.selectedIndex!=value) {changed=true;input.selectedIndex=value; } } catch(e){}
	    for ( index=0; index < input.options.length; index++ )
	       if ( index == value) {
		  setMember(input.options.item(index).selected, true); 
		  input.options.item(index).selected=true;
		  option=input.options.item(index);
		  //fakeClick(input.options.item(index));
	       }
	 }
	 else { 
	     setMember(input.value, value); 
	     if (value.slice) input.value=value.slice(0,value.length/2);
	     else input.value=value;
	     //getAnonymousNodes The getAnonymousNodes method retrieves the anonymous children of the specified element.
	     //only visible in dom inspector (red in color, beneath INPUTs at first as DIV with a BR then when filled as DIV with direct text.
	     setTimeout(function(sinput, svalue) {      //pass parameters to it from stack
		 //log("setTimeout Filler set:"+sinput.name+"="+svalue+"."); 
		 sinput.value=svalue;
		 coaxInputs(sinput, svalue);
	     }, 50, input, value); // input, value parameters passed to setTimeout function stack, 50 is timeout.  To add to eventlistener extra param needed.
	     //input.value=value;
	     //log("last setMember "+value);
	 }
      } //end if type != radio
      log("Set value of input["+i+"].type: "+type+", saved value is:"+value + ".  Current value: "+input.value+ ". seloption: "+input.selectedIndex +".  Changed: "+changed);
      if (changed) {  // trick to simulate event if (changed && false) !!
	 kick=changed;
	 changed=false;
	 var pseudo_event = window.document.createEvent("MouseEvents");// create event
	 pseudo_event.initMouseEvent("change", true, true, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);
	 pseudo_event = window.document.createEvent("MouseEvents");// create event
	 //fakeClick(input);
	 //if (option) { fakeClick(option);option=null;}
      }
   } // end for inputs
    var write_msg=(msg?msg+" ":"")+"Filled in forms"
      +(once_off ? " for dynamic element on "+page_key : " at "+page_key )+"."+userProfileLine
      +( ! result? "Bounds error.  " : "") +( kick ? " Changes were made.  " : "No changes were necessary.  ")
      +(fill_msg ? fill_msg : "");
   window.status=write_msg;
    console.info(write_msg);
   old_elems=inputs.slice(0);
   if ( result && ! once_off && ! auto ) {
      log("Filled form, replay from filler as manual ");
      replayClick(true);
   }
   return result;
} catch(e) { alert("Cannot fill-out form, reload and try again.\n\nError was: "+e+" "+e.lineNumber); throw(e);}
					    }
function profileChange(){
    var prof_ar=profiles.toArray(), curr_prof=profiles.current;
    var msg="Give a new name to create a profile, use an existing profile name, or blank to use default base profile"+ (curr_prof?" (current profile is:"+curr_prof+")" : "" ) +".";
    if (prof_ar.length) msg+="  Available profiles (clickable): "
	+("<a class=profs_c>base</a>, "+prof_ar.reduce((prev,current)=>prev+"<a class=profs_c>"+current+"</a>, ","")).replace(/,\s*$/, "")+"."
	+(otherProfiles.length ? "\n\nProfiles, other than base and current one, that have prior stored values: " + otherProfiles +".": "");
    var dialog=sprompt(msg,"",function(reply) {
	if (reply!=null) {
	    reply=reply.trim();
	    if (reply[0] != "-") {
		if(reply.includes(" ")) { alert("Spaces are not allowed within profile names."); return; }
		profiles.current=reply;

	    } // endif reply[0] != -
	    else { // delete given profile from list.
		reply=reply.substr(1).trim();
		if (profiles.rm(reply)==-1) alert("No such profile to delete.");
	    }
	} // endif reply != null
	
    }); // end sprompt()
    dialog.find(".ui-dialog-content")
	.attr("title","Set the current user profile, create a new profile name (it becomes the current profile automatically), or delete an existing one.\nThe special 'base profile' is the default and has no name."
	      +"\nPut a minus sign (-) before a profile name to delete it.\nEnter a name in the box below or click one of the exising profile names."
	      +"\n\nProfiles allow multiple sets of form data to be stored for the same page, for example, if a form is often filled in on behalf of another "
	      +"person and you also wish to keep differing form info for yourself or others.  Once a profile is current, Simple Form Saver will persistently use that profile session until it is explicity changed or is reverted to the base (default) profile.  Old tabs may need to be reloaded to see the changes made here.");
    dialog.find("a.profs_c").click(e=>{ 
	dialog.trigger(jQuery.Event("keydown",{keyCode:27,key:"Escape"})); //close prompt.
	profiles.current=e.target.textContent.replace(/^base$/,""); // Set profile selected.  BTW, this is el to which handler was attached.
	return false;
    });
    //dialog.find("input").focus();
    //setTimeout(function(){dialog.find("input").focus();},200);
}

function profile_bridge() { // abstract storage (accessors) for back compatibility.  Used at top in: var profiles=new profile_bridge();
    var currentProfile=getValue("currentProfile", "");   //
    var full_list=getValue("profiles_list",[]); // Parallel arrays, full list and list (of strings).      GM_deleteValue("profiles_list");GM_deleteValue("currentProfile");
    var list=full_list.map(el=>el.name);        
    var update=function() { list=full_list.map(el=>el.name); setValue("profiles_list",full_list); setValue("currentProfile", currentProfile); }; 
    var accessors={
	get:function(){ return currentProfile.title||"";},    // "" if not using profiles.
	set:function(newprof_name) {                           // Setting to a new name creates a new profile.  Change profile.
	    if (!newprof_name) currentProfile="";
    	    else {
		if ( ! list.includes(newprof_name)) 
		    full_list.push({name:newprof_name});
		currentProfile={title:newprof_name};
		$("#SFSimg").css({border:"dotted 5px"}); //similar in setLive if not base prof.
	    }
	    console.info("Simple Form Saver, changing profile to "+(this.current||"base profile")+".");
	    submenuModule.changeName("Profile Change.*",
				     "Profile Change ["+(profiles.current?profiles.current:"base profile")+"].");
	    update(); //Does storage of values
	    hostFormsList=this.current+"hostFormsList";
	    readPersistentData("profile_bridge");
	    userProfileLine=this.current ? "\nUser profile: "+this.current+" (nondefault -> icon decor)" : "";
	    setLive();
	}
    };
    Object.defineProperty(this,"current", accessors); // get & set for attrib "current".
    hostFormsList=this.current+"hostFormsList";    
    this.toString=function(){ return list.toString(); };
    this.toArray=function(){ return list; };
    this.rm=function(profile_name) {
	if (!profile_name || !list.includes(profile_name)) return -1;
	full_list.splice(list.indexOf(profile_name),1);
	if (this.current==profile_name) {
	    page_object={}; persistData();
	    this.current="";
	}
	else { let old_prof=this.current; this.current=profile_name; page_object={}; persistData(); this.current=old_prof; }
	update();
    };
    this.checkOtherProfiles=function() {
	// for each profile check if forms_data or click set.
	var curr_prof=this.current, original_hfl=hostFormsList;
	var profiles_with_values=list.filter(x=>{
	    if (x==curr_prof) return;
	    hostFormsList=x+"hostFormsList";
	    readPersistentData();
	    return (form_data || click);
	});
	hostFormsList=original_hfl;
	readPersistentData();
	//console.log("checking other profs curr:",curr_prof," result",profiles_with_values);
	return profiles_with_values;
    };
}

function coaxInputs(input, value) {
    //console.log("coaxing "+input.name+" "+value);
    //Live dynamic inputs need coaxing, test in 1.5.0:
    var act_el=window.document.activeElement;
    input.focus();         try{
	moveCaretToEnd(input);    }catch(e){}
    var evt = document.createEvent("KeyboardEvent"), evt2=document.createEvent("KeyboardEvent"), evt3=document.createEvent("KeyboardEvent"), evt4=document.createEvent("KeyboardEvent");
    //evt.initKeyEvent("keypress", true, true, null, "U+0008", 0, "");
    var keyval=98; // 98 'a', 8 is backspace
    if (!chrome) {
	jQuery.event.trigger({ type : "keypress", which : keyval});
	evt.initKeyEvent ("keydown", true, true, window,   0, true,    false,  false,  keyval,            0); 
	evt2.initKeyEvent ("keypress", true, true, window,   0, true,    false,  false,  keyval,            0); 
	evt3.initKeyEvent ("keyup", true, true,  window,   0, true,    false,  false,  keyval,            0);
	evt4.initKeyEvent ("textinput", true, true,  window,   0, true,    false,  false,  keyval,            0); 
    }
    else {
	evt.initKeyboardEvent ("keydown", true, true, window,   0, true,    false,  false,  keyval,            0); 
	evt2.initKeyboardEvent ("keyup", true, true,  window,   0, true,    false,  false,  keyval,            0); 
	evt3.initKeyboardEvent ("keypress", true, true, window,   0, true,    false,  false,  keyval,            0);
	evt4.initKeyboardEvent ("textinput", true, true,  window,   0, true,    false,  false,  keyval,            0); 
    }
    input.dispatchEvent(evt);
    input.dispatchEvent(evt2);
    input.dispatchEvent(evt3);
    input.dispatchEvent(evt4);

    act_el.focus();
}

function moveCaretToEnd(el) {
   el.focus();
   if (typeof el.selectionStart == "number") {
      el.selectionStart = el.selectionEnd = el.value.length;
   } else if (typeof el.createTextRange != "undefined") {
      var range = el.createTextRange();
      range.collapse(false);
      range.select();
   }
}


function registerMenus(full_menus) {
   //
   // GM menus
   //
   switch(this.phaseMenus) {
   case undefined:
      if (this.done==2) return; 

      GM_registerMenuCommand( "========Simple Form Saver======", function(){});
      
      GM_registerMenuCommand( "Save Form",
			      function() {
				 readPersistentData("Save Form");
				 if ( inputs.length==0)
				    return;
				 useCorrect_page_object();
				 saveFormData();
				 if ( ! link )
				    addIcon(true);
				 setLive();
				 inOutSet();
			      }, "","","F" );

       window.openFormStatus=function() { 
	   readPersistentData();
	   var prompt=salert(getStatusInfo());
	   prompt.find("a.submenu_c").click(e=>{   
	       prompt.trigger(jQuery.Event("keydown",{keyCode:27,key:"Escape"})); // jQ will call close on prompt.
	       submenuModule.open(); // To open another script's menu: $("[script-name^='Simple Form Saver']").parent().addBack().css("display","");
	       return false;
	   });
	   if (element_to_highlight) element_to_highlight.style.borderStyle="none"; 
       };
       GM_registerMenuCommand( "Get Form Status & Settings",window.openFormStatus
			       , "","","G");

       GM_registerMenuCommand("Profile Change ["+(profiles.current?profiles.current:"base profile")+"].", profileChange,"","","P");
       
       GM_registerMenuCommand( "Record next mouse click on document...",
			      function() { 
				 log("ask confirm "+!chrome);
				 readPersistentData();
				 if (click) { 
				     sconfirm("A click has been previously recorded, hit OK to delete it and re-record, or Cancel to make addition clicks or abort", function(reply) { 
				       if (!reply) {
					  log("Reply:"+reply+"."+typeof reply+" "+(!reply));
					  sprompt("Please enter the path (XPath) for the additional item to be clicked, eg, /html/body/div[1]/div, for XPath see console while element highlighting is selected by you when recording a first click.","",
						  function(reply){ 
						     if (reply==null) { GM_log("No extra click recorded");return; }
						     if (!click.further_clicks) click.further_clicks=[];
						     click.further_clicks.push(reply.trim());
						     var xpath=click.further_clicks[click.further_clicks.length-1];
						     var celem=uwin.document.evaluate(xpath,uwin.document,null,6,null).snapshotItem(0);
						     GM_log("Extra click is on:"+celem.tagName+", "+celem.textContent);

						     persistData();
						  });
					  return;
				       }// !reply
				       confirmClickRecording()
				      }); // end confirm3 CB

				 }//end if click
				 else 
				    confirmClickRecording();
			      }, "", "", "M");    //end GM_registerMenu
      
       GM_registerMenuCommand( "Toggle Auto-Click-Replay function on all pages "
			      +(suspend_replay ? "[disabled]": "[enabled]"),
			      function() { 
				 readPersistentData();
				 if (suspend_replay) suspend_replay=false;
				 else suspend_replay=true;
				 persistData();
				 window.status="Auto click replay is now "+(suspend_replay ? "suspended.  ": "on.  ");
				 salert(window.status);
			      }, "","","u");  


      GM_registerMenuCommand( "Icon for Form Saver pages on/off, toggle "+(reddot?"[on]":"[off]"),
			      function() { 
				  readPersistentData();
				  toggleIcon();
				  submenuModule.changeName("Icon for Form.*", "Icon for Form Saver pages on/off, toggle "+(reddot?"[on]":"[off]"));
				  //submenuModule.positionAt("Icon for Form.*", 1);
				  persistData();
				  window.status="Form Saver Icon "+(reddot ? "on": "off");
			      });    
      /* if( ! FireFox)  PROBlem on Chrome, only localstorage is available so imported data cannot be seen when visiting other websites*/
      // export also.  Storing it as uneval or member in window.name fails when "back" is selected.
      /*   GM_registerMenuCommand("Import Form Data", function(){ */
      /* 	 var users_input=prompt(widener+"\nTo import raw forms' data paste here and click OK.\n\nTo get raw data on Firefox, look in about:config for the name 'hostFormsList', copy its value here." , ""); */
      /* 	 if (users_input != null) setValue(hostFormsList, users_input); */
      /* 	 alert(getValue(hostFormsList)); */
      /* 	 readPersistentData(); */
      /* 	 persistData(); */
      /*     }) */
      GM_registerMenuCommand("Paste Data Into Form", pasteIntoForm, "", "", "P");
      GM_registerMenuCommand("Clear the Forms on the Page", function() {
	 readPersistentData();
	 for(var i=0; i < inputs.length; i++) {
	    var input=inputs[i];
	    if (input.selectedIndex != undefined) { input.selectedIndex=-1; input.value=-1; }
	    if (input.checked !== undefined ) 
	       input.checked=false
	    input.value="";
	 }
      }, "", "", "C");
       GM_registerMenuCommand( "List of pages with saved form information ",
			      function() { 
				  readPersistentData();
				  var roll="", len=0;
				  var reply=RegExp( prompt("Give regexp to filter or leave blank for all:"), "i");
				  for (var i in hash_host_list) {
				    var fd=true;
				    if (hash_host_list[i].form_data) {
				       for (var j in hash_host_list[i].form_data)
					  len++;
				       var unevaled=uneval(hash_host_list[i].form_data);
				       if ( ! reply.test(i) && ! reply.test(unevaled)) continue;
					 var sizeof=unevaled.length;
				       roll+="\n"+i+ "; "+len+" fields, "+sizeof+" bytes.\t\t"+prettyPrintUneval(unevaled, true);
				    } else fd=false;
				    if (hash_host_list[i].click) {
				      if ( ! reply.test(i)) continue;
				      roll+=( fd ?  " (" : "\n" + i  + ", " )  + " click path: "+hash_host_list[i].click.xpath.tail(30)+( fd ? ")" : "." ) ;
				    }
				 } //end for i in hash_host_list
				 var sizeof=uneval(hash_host_list).length;
				 //bigAlert(widener+(roll? roll: "none")+"\n \n"+"Size of store: "+Math.round(sizeof/1000)+"k.");
				 var ar=roll.split("\n")
				 ar.sort(function(a,b) { return a.toLowerCase().localeCompare(b.toLowerCase()) });
				 roll=ar.join("\n");
				 salert(reply+widener+(roll? roll: " gave no results.")+"\n \n"+"Size of store: "+Math.round(sizeof/1000)+"k.");
			      });    
      GM_registerMenuCommand( "Export/Import all form data", exportImport);
       //retired      GM_registerMenuCommand("Open Forms' Address/Notebook, alt-a", formsAddressBook, "", "", "A");
       GM_registerMenuCommand("Unhide passwords [once-off]", unhidePasswords, "", "", "A");
       GM_registerMenuCommand("Get/set Script Paste Config Values", getSetConfigs, "", "", "C");
       // addEventListener("keydown", function(e) {
       // 	   //GM_log("Keydown: charcode "+e.charCode +", keycode"+e.keyCode+", alt "+e.altKey+", ctrl "+e.ctrlKey);
       // 	   if (e.keyCode!=65 || ! e.altKey) return true;// alt-a. alt a
       // 	   if (!e.shiftKey)
       // 	       formsAddressBook();
       // 	   else
       // 	       fillFromAddressBook();
       // }, 0);
       
       this.phaseMenus=1;
      break;
   case 1:
       if (full_menus) {
	   registerPageMenus();// ------------>
	   this.phaseMenus=2;
       }
       break;
   }
    function registerPageMenus() {
	
	GM_registerMenuCommand( "                       --------------------------------------------------", function(){});
	
	GM_registerMenuCommand( "Fill-in Form",
			       function() { 
				   readPersistentData("Fill-in Form");
				 if (form_data) {
				     fillForm();
				     inOutSet();
				 }
				   else
				       window.status="No form data to fill";
				   //    persistData(); !!
			       }, "",0,"O");
       submenuModule.positionAt("Fill-in Form",0);
       
       GM_registerMenuCommand( "Delete stored Form info for this page",
			       function() { 
				   readPersistentData();
				   var profile=profiles.current?".  Profile: "+profiles.current : "";
				   var request, page_str="\nPage: "+ page_key + profile + (regexp_page_match ? " (regexp match)." : ".") ;
				   if (click && form_data)
				       sconfirm(page_str+"\n\nTo delete click-replay & form data for this page, choose 'OK'.  "
						+"\n\nTo delete only the click-replay or to abort, choose 'Cancel'",
						function(request){
						    if (request)  { page_object={}; wrapup(); }
						    else 
							sconfirm(page_str+"\n\nOk to delete click recording? ", 
								 function(request) {
								     if (request) { delete page_object.click; wrapup();}
								     else return;
								 });
						});
				   else 
				       if (click || form_data)
					   sconfirm(page_str+"\n\nOk to delete form date for this page?",
						    function(request) { 
							if (request) {
							    page_object={}
							    wrapup();
							}
							else
							    return;
						    }); // end if clk||frm
				   function wrapup(){
				       inOutSet();
				       window.status="Deleted "+(request?"all stored ":"only click-replay")+" info for: "+page_key;
				       persistData();
				       setLive();
				   }
			       });//end GM_registerMenuCommand()    
       
       GM_registerMenuCommand( "Toggle Automatic form fill & Click-replay on this page ["+(page_object.automatic?"on "+(page_object.automatic=="on" ? "[implicit]":""):"off")+"]", function () { 
	   readPersistentData();
	   if (page_object.automatic) { //toggle
	       page_object.automatic=false;
	       persistData();
	       window.status="Ending automatic fill-in of forms at page: "+page_key;
	       salert("Ending automatic fill-in of forms at page: "+page_key);
	   }
	   else {
	       page_object.automatic=true;
	       sconfirm("Page: "+page_key+widener+"\n\nForm data saved by the user shall "
			+"be filled in for this page automatically from now explicitly on.  Auto replay of Clicks on this page will override general setting."
			+"\n\nCancel to explicitly disable auto-fill/click-replay on this page"
		     , function(request) {
			if ( ! request)
			   page_object.automatic=false;
			else page_object.automatic=true;
			window.status="Automatic filling of forms on page "+page_key;
			salert("Automatic filling of forms is explicitly on at page "+page_key);
			inOutSet();
			persistData();
		     } );
	 }
      }//end cb function
			      , "","","" );
      
      GM_registerMenuCommand("Edit stored Form data and site", function() {
	 editFormData();
      }, "", "", "E");

	GM_registerMenuCommand("Edit Click Element data", function() {
	    readPersistentData();
	    editClickData();
	    persistData();
	}, "", "", "E");
	
	GM_registerMenuCommand("Manually set active input value",function(p1,p2){
	    var ae=submenuModule.activeElement;
	    console.log("got ae as",ae);
	    if ( ! /input|textarea/i.test(ae.tagName)) {
		salert("Active element is neither an input not a textarea, please  "); return;
	    }
	    var reply=prompt("Please input to what  input/textarea should be set:");
	    ae.value=reply;
	},"","","I");
	GM_registerMenuCommand( "_____________________________________", function(){});
   }
}

function convert_obj_to_array(obj) {
   var ar=[];
   for (var  i in obj) if ( ! isNaN(i) ) ar[i]=obj[i];
   return ar;
}

// function bigAlert(str) {
//    lines=str.split("\n");
//    var entries=lines.length-5;
//    var roll=entries+" entries.  "+(entries>30?"Pages: "+( (entries/30^0) + 1 ) : "" ) + "\n", i=0;
//    while (lines.length) {
//       while ( i < 30) { roll+=(lines[0]?lines[0]+"\n":""); lines.shift(); i++ }
//       alert2(roll, 0.7);
//       roll=widener, i=0;
//    }
// }

function fixMargins() { // see trickle function
   var results = document.evaluate("//*[contains(@style,'margin-left')]", document, null, 6, null);
   var item, len=results.snapshotLength;
   var margin, parent, pos;
   if (len > 0) 
      for(var i=0; item=results.snapshotItem(i), i < len;  i++ ) {
	 margin=parseInt(item.style.marginLeft);
	 //log("item margin "+margin);
	 parent=item.offsetParent;
	 pos=item.offsetLeft;
	 //log(" item pos "+pos);
	 if (pos < 200 && margin > 5) {
	    item.style.setProperty("margin-left",(Math.max(0, margin - 35) )+"px","important");
	    //log("Fixed margin for "+item+" "+item.id+", "+item.className +", from "+margin+".  At: "+page_key);
	 }
      }
   return;
}

function toggleIcon() {
    if (reddot) {
	reddot=false;
	removeIcon();
    }
    else {
	reddot=true;
	addIcon(true);
    }
}

function removeIcon() {
    if (link.parentNode == window.document.body) {
	window.document.body.removeChild(link);
	delete page_object.position;
    }
}

function addIcon(make, override) { // append, prepend reddot to body in DOM hierarchy.
    if (link && (reddot || override)) {
	//if (window.document.body) window.document.body.insertBefore(link, window.document.body.firstElementChild);
	if (window.document.body) window.document.body.appendChild(link);
	img.style.display="";
	setTimeout(fixMargins, 1000); //!!!!fixMargins
	//   link.style.cssFloat="left";
	//   img.style.cssFloat="left";
    }
    else if ( ! link && make)
	makeReddot();
}

function makeReddot(parent) { // makes a "link" object as: <B><IMG></IMG></B> and registers mouse events on it.
    var lk;
   if (! parent)
      parent=window.document;
   if ( ! link)
      link=parent.createElement("b");
   var lk=link;
   img = parent.createElement("img");
   //img2 = parent.createElement("img");
   //  img2.setAttribute("style", "float: left; z-index:999 ! important; position: relative; margin-right: 20px;margin-left: 20px; ");
   //  img2.setAttribute("style", "left: -2px; top: 5px; z-index:999 ! important; position: fixed; margin-right: 20px;margin-left: 20px; ");
   //img2.setAttribute("style", "left: 3px; top: 5px; z-index:999 ! important; float: left; position: relative; visibility: hidden; margin-right: 12px;");
   img.id="SFSimg"; //img2.id="SFSimg2";
   //img2.height=10; img2.width=19;
   lk.appendChild(img);
   //lk.appendChild(img2);
   addIcon(); // insertsBefore 1st body el., ie, prepends to body.
    //checkPageLayout();  //!!!checkPageLayout
   setLive(); // sets style.
   // var pos=getXY(img); 
   //  try {
   //   var el = document.elementFromPoint(pos.x, pos.y);
   //   trickleUp(el);
   //   trickleUp ( document.elementFromPoint(pos.x, pos.y + 50)) } catch(e){};
    lk.id="hostFormsListButton";
    lk.className="SFSButton";
    var dragged=0;
    setTimeout(function(){
	$(img).draggable({ stop: function( event, ui ) { event.dragEv=true;
							 event.ui=ui;
							 dragged=0;
							 mouseUpCB(event);},
			   drag: function( event, ui ) { dragged++;}
			 });//.draggable()
    },100);
    // $(lk).draggable();
   //$(reddot).draggable();
    
   lk.addEventListener("contextmenu", function(e) {  // right click
       readPersistentData("contextmenu click");
       inOutSet();
       e.preventDefault();   
       e.stopPropagation();
       window.document.body.setAttribute("onClick", "return false;"); //!!!
       useCorrect_page_object();
       //if (form_data) {
       sconfirm("Simple Form Saver, saving data at: \n\n\t"+page_key+(profiles.current?"  User profile: "+profiles.current+"":"")+".  Ok/Cancel?",function(reply){
	   if (!reply) return;
	   saveFormData();
	   setLive();
       });
       //return false;
   }, true);
    // lk.addEventListener("mouseup", function(e) { 	// To prevent clipboard accidentally with href and middlemouse.contentloadurl in about: would need to prevent at win/doc level.
    //    e.preventDefault();   	    
    //    e.stopPropagation(); 
    //    return false;
    // }, true); //mousedown->context->mouseup;
    lk.addEventListener("dblclick", function(e) {
	console.log("dbl Click on sfs reddot.");
    });
    lk.addEventListener("mouseup", function(e) { // click // was "mousedown"
	mouseUpCB(e)}, true);
    function mouseUpCB(e) {
	if (dragged) return;
	if(e.dragEv) {
	    readPersistentData("drag");
	    page_object.position=e.ui.position;
	    persistData(); 
	    log2("Mouse was dragged");
	    // e.preventDefault();   
	    // e.stopPropagation();
	    return;
	}
	//e.preventDefault();   
	//e.stopPropagation();
	if (e.button==0 ) {        //left button 
	    readPersistentData("click");
	    inOutSet();
	    // e.preventDefault();   
	    // e.stopPropagation();
	    if (form_data)
		fillForm();
	    else {
		useCorrect_page_object();
		if (click) replayClick(true);
		saveFormData();
		setLive();
		persistData(); 
	    }
	    //persistData(); //!!
	    log("done but 0");
	}
	if (e.button==1) { // middle click
	    readPersistentData("mid click");
	    if ( ! iframe) {
		window.openFormStatus();
		// var prompt_dialog=salert(getStatusInfo());
		// console.log("salert called",prompt_dialog);
		// prompt_dialog.find("a.submenu_c").click(e=>{ //from <a>Simple Form Saver submenu
		//     console.log("submenu open, closing prompt...");
		//     prompt_dialog.trigger(jQuery.Event("keydown",{keyCode:27,key:"Escape"})); // jQ calls close on prompt.
		//     let submenu=$("[script-name^='Simple Form Saver']");
		//     submenu.parent().addBack().css("display","");
		//     //submenuModule.open();
		//     console.log(e,submenu,"displayed, click END.");
		//     return false;
		// });
	    }
	    else {
		//window.open(location.href);
		top.location.href = document.location.href ;
		top.window.name+="SFS:KeepFrame";
		
		// confirm2(getStatusInfo()+"\n\n"+"Click 'Cancel' to edit this subpage's stored data", function(reply) {
		// if ( ! reply ) {
		//   editFormData();
		//   persistData();
		//} } );
	    } //end else
	    // e.preventDefault();   
	    // e.stopPropagation();
	    if (element_to_highlight) element_to_highlight.style.borderStyle="none"; 
	    //return false;
	}
	// e.preventDefault();   
	// e.stopPropagation();
	// return false;
    } //end mouseUpCB()
}

function setLive() {
    if ( ! link)
	return;
    var redSquareRing="";
    redSquareRing="";
    var redDot_img="";
    redDot_img="";
    redDot_img="";
    redDot_img="";
    var redDot_off="";
    redDot_off="";
    link.addEventListener("mouseover",function(){ 
	//log("mo "+form_data); this.style.opacity=0.7;
	if ( ! (form_data||click)) img.src=redDot_img;  
	//else img.style.height=img.style.width=20;
    }, false);
    link.addEventListener("mouseout",function() {
       this.style.opacity=1; 
       if ( ! (form_data||click) ) img.src=redDot_off;
       //else img.style.height=img.style.width=12;
   }, false);
    //    link.style.opacity=".6";
    link.style.opacity="1";
    link.style.rightMargin="10px";
    with (img.style) {
	cursor="pointer";
	position="fixed";
	top="5px";
	left="18px";
	if (page_object && page_object.position) {
	    top=page_object.position.top+"px";
	    left=page_object.position.left+"px";
	    var msg="Simple Form Saver, user set image position to left:"+left+", top:"+top;
	    GM_log(msg);
	    window.status=msg;
	}
	//	cssFloat="left";
	setProperty("display", "inline", "important");
	setProperty("z-index", "2147483644", "important");
	setProperty("min-width", "6px", "important");
	setProperty("font-size", "small", "important");
	setProperty("box-sizing", "content-box", "important");       
	//zIndex=9999;
    }
    /* img2.src="" */
    /* img2.width=12; */
    /* img2.height=12; */
    //log("setlive");
    if (form_data || click) {
	registerMenus(true);
	img.src=redSquareRing;
	//log("setLive has Form data");
	with (img.style) {
	 setProperty("border","", "important");
  	 setProperty("border-style","outset", "important");
	 borderWidth=".23em";
	 opacity=1.0;
	 borderColor="rgba(200, 200, 200, .9)";
	 // img.height=12;
	 // img.width=12;
	 height="14.4px"; //".6em"
	 width="12px"; //".5em"
	 height = ! form_data ? "10px" : "12px"; //"1.0em" : "1.5em";
	 width=height;  //!form_data ? "1.0em" : "1.5em";
	 link.title="Click to Fill in Form (&/or replay a recorded click).  Right click to Save form.  Drag n Drop to move icon."
	    +"  Middle click "+(iframe?"to open iframe in own window/tab in order to allow access to GM settings for this part of the page."
				:"to get forms' status info.")
		+ (iframe ? "  (Within iframe: "+location.href+")." : "" ) + userProfileLine
		+ (profiles.checkOtherProfiles().length ? "\nOther profiles have data stored for this page (->icon pulse).":"") +addedByLine;
	 link.setAttribute("noautohide","true");
      }
      //img2.style.display="inline";
      //img2.style.setProperty("border-style", "none", "important");
      full_icon=true;
   } 
    else { // no form data, nor click
	if (click)       registerMenus(true);
	img.src= redDot_off; 
	img.height=img.width="10px";
	with (img.style) {
	    height="";
	    width="";
	    height=".5em"
	    width=".5em"
	    if (!click) {
		borderStyle="none";
		borderWidth="";
		//if(!click) 
		setProperty("border","none", "important");
		$(link).prop("title","Click once to save form data "+(click?"or to replay a click":"")+"; middle-click to get forms status info."
			     + (iframe ? "  (Within iframe)." : "" ) +"  Can drag and drop to move icon.  See options to remove.  At: "
			     +page_key
			     +userProfileLine
			     + (profiles.checkOtherProfiles().length ? "\nOther profiles have data stored for this page (->icon pulse).":"")
			     +addedByLine);
	    }
	}
	//img2.style.display="none";
	full_icon=false;
    }
    if (reddot==false)
	img.style.display="none";
    else
	img.style.display="";
    
    var iel=$(img), bwidth=form_data||click ? 9 : 5;
    if (profiles.current=="") bwidth=0;
    iel.css({border:"dotted "+bwidth+"px rgba(0,0,0,0.7)"});
    if (otherProfiles.length) {
	pulse(iel,3);
    }
    // $(img).draggable();
    // $(reddot).draggable();
}// end setLive()

function pulse(el,times,recur) {
    if (!recur) pulse.orig={ w:el.width(), h:el.height() };
    var  factor=1.5, w=pulse.orig.w, h=pulse.orig.h, wf=w*factor, hf=h*factor;
    el.animate({width: wf, height: hf, opacity: 1}, 700, function() {
        el.animate({ width: w, height: h, opacity: (times>1?.5:1) }, 700, function() {
	    if (times==1) return; 
            pulse(el,--times,true);
        });
    }); 
};

function inOutSet(set) {
   if ( full_icon 
	|| ( ! form_data && click ) 
	|| set
      ) {
      if ( ! img.style.borderWidth)
	 img.style.borderWidth=".1px";
      if (!this.prev) {
	 img.style.setProperty("border-style", "inset", "important");
	 this.prev=true;
      }
      else {
	 img.style.setProperty("border-style", "outset", "important");
	 this.prev=false;
      }
   }
}

function checkPageLayout(){
   var logo, p, gp; logos=getByIdOrClass("logo-img-2", "logo", "gh-log", "p-logo", "mw-panel", "logocont");
   for( var i =0; logo=logos[i],  i < logos.length; i++ ) {
      logo.style.setProperty("margin-left", "0px", "important");
      // if (p && p.parentNode && p.parentNode.style) p.parentNode.style.setProperty("margin-left", "0px", "important");
      // if (p && p.style) p.style.setProperty("margin-left", "0px", "important");
      logo.style.setProperty("z-index", "0", "important");
      if ( ! checkPageLayout.count) { checkPageLayout.count=10;}
       logo.addEventListener("DOMAttrModified", checkMargins, 0);
   }
   return;
}

function checkMargins(e){
   log("checkMargins");
   if (e.target != this) return;
   if ( this.style.marginLeft[0] != 0) this.style.setProperty("margin-left", "0px", "important");
   if ( ! -- checkPageLayout.count ) 
      for( var j =0; logoj=logos[j], j < logos.length; j++ ) 
	 logoj.removeEventListener("DOMAttrModified", checkMargins, 0);
}

function findRenamedInput(name, i, saved_value) 
{
   var n, result, saved_data=page_object.form_data;
   for (n in saved_data) {
      if ( i == saved_data[n].i ) {
	 result=saved_data[n].v
	 break;
      }
   }
   return [result, n, i ];
}

function getElementsByTagNames(list,obj, full) {
   if (!obj || ! obj.getElementsByTagName) var obj = window.document;
   var tagNames = list.split(',');
   var resultArray = new Array();
   if (obj.tagName && list.match(obj.tagName.toLowerCase())){
      resultArray.push(obj);
   }
   for (var i=0;i<tagNames.length;i++) {
      var tags = obj.getElementsByTagName(tagNames[i]);
      for (var j=0;j<tags.length;j++) {
	 if (  tags[j].type   
	       &&   !   /^(submit|hidden|image|reset|button)$/.test( tags[j].type.toLowerCase() ) 
	       && ! (tags[j].disabled == true) || full  ) {
	    resultArray.push(tags[j]);
	 }
      }
   }
   var testNode = resultArray[0];
   if (!testNode) return [];
   if (testNode.sourceIndex) {
      resultArray.sort(function (a,b) {
	 return a.sourceIndex - b.sourceIndex;
      });
   }
   else if (testNode.compareDocumentPosition) {
      resultArray.sort(function (a,b) {
	 return 3 - (a.compareDocumentPosition(b) & 6);
      });
   }
   return resultArray;
}

function chopLongString(form_string) {
   if (form_string.length > 1700)   form_string=form_string.replace(/[\t\n]/g," ").replace(/\u26ab/g,tab_bullet).replace(/\t/g,"")
   return form_string;
}

function getStatusInfo() {
   var data= form_data, n, list="";
   for (n in data)
       if (data[n].p)
	  list+=n+",\t";
    if (list) list=list.substring(0, list.length-2)
    var page_list=null, page_list2="";
    //   if ( ! form_data) { //////////////////////////////////////
    page_list=bullet_tab;
    //	var tinputs = getElementsByTagNames('input,textarea,select', null, true);
    var tinputs = getElementsByTagNames('input,textarea,select');//was showing submit buttons with no text.
    for(var i=0; i<tinputs.length; i++) {
	var inp=tinputs[i];
	var type=0, option=0;
	if (inp.tagName =="INPUT")
	    type=inp.type
	else {
	    type=inp.tagName
	    option="";
	    if (inp.tagName=="SELECT" && Number(inp.value)) {
	       var options=inp.options;
	       option="\t\t(~"+options.item(inp.selectedIndex).textContent.replace(/[\n\t(  )]/g,"")+")\t";
	    }
	 }
	 var val=inp.value;
	 var style=window.document.defaultView.getComputedStyle(inp, null)||{};
	 var visibility=( (style.display=="none" || style.visibility=="hidden") ? " (invisible)" : "");

	 page_list+=(inp.name?inp.name:"null")+"::"
	    +(val? ( inp.type=="password" && ! unhide ? val[0] + "..." + val[val.length-1]+"["+val.length+"]" 
		     : val  +(unhide?" (unhidden":"")      )
	      :"null")
	    +(option?option:"\t\t\t\t")+"\t"+type+visibility+"\n"+bullet_tab;
	if (inp.type=="password" && ! unhide)
	    page_list2+=val[0]+"..."+val[val.length-1]+"["+val.length+"]";
	else
	    page_list2 += val +"\n";
      } // end for inputs[].
      page_list=page_list.substring(0,page_list.length-2);
    // } // end if ! form_data ///////////////////////////////////////
    if ( form_data) { page_list=null }
   var ip_names="";
   for (var i in input_names) ip_names+=i+", "; ip_names=ip_names.substr(0, ip_names.length-2);
    var form_string=page_list || formDataToFromString(true);
    form_string=chopLongString(form_string);
    log("get xpath ");
    var click_elem=getXPathElem();
    var user_profile_info=(profiles.current?"\nUser profile: "+profiles.current+" (non default -> icon decor).":"")
	+(otherProfiles.length? "\nOther page relevant user profiles: "+otherProfiles:"");
   var forms_info =
	   ( form_data ? "Previously saved values and names are listed below."+user_profile_info
	: "Page Values and form names have not been saved for this page")
      +"\nFormat: name ["+mdash_char+"default]::value"+( form_data ? "":"\t\tType of (name::value)")+"."
      + (form_data  ? "\n\n"+bullet_tab+form_string : "\n\n"+form_string+"" )
      +  (list ? "\n\nThere are duplicates whose values may differ from the above for the input(s) named:\t "+list+"."  : "")
   var settings_info=""
      +"Names of all inputs: "+ip_names
      +"\n\nAutomatic form filling on this page "
      + (page_object.automatic ? "is on.  " :  (page_object.automatic !== false ? "has not been turned on, and is off.  " : "and click-replay, have been explicitly disabled.  "))
      +(click ? "\nClick auto-"+(click.ctrlShift?"delete":"replay")+", is set for"
	+(click.own_href == "any" ? " any page with this path.  " :  " only this specific page.  ")
	+"Page's element (XPath) to be auto-"+(click.ctrlShift ? "deleted":"clicked")+" is:\n\n\t\t"+ ( ! (click_elem && click_elem.xx) ? page_object.click.xpath  : page_object.click.xxpath + " as xpath" ) +""
	: "Nothing to replay on this page.  " )
      + ( click && ! click_elem
	  ? " (the element is not currently on this page" + (click.href ? " but has href, " + click.href : ""  )  +  ")." 
	  : ( click && (href == click.own_href || click.own_href == "any" )
	      ? (click?"\n\nThe element is present on the page.\nIts value and text is: "+click_elem.value+" "+click_elem.textContent+", it's href is:"+click.own_href : "none")
	      : (click?", the element is on page but it is for another webpage: \n\t\t"+click.own_href : "none" )))
      +(click && click.onclick_js? "\n\t\tJava to replay: "+ click.onclick_js    : "")
      +(click && click.further_clicks? "\n\nFurther clicks are on elem(s) (XPath(s)): "+click.further_clicks.join(", "):"")
      + "\n\n"+ (suspend_replay ? "Click auto-replay is suspended in general." : "Click auto-replay on all pages is not suspended.")
      + (inputs.length ? "\n\nNumber of useable input areas on page (some may currently be hidden): "+(inputs_on_page) : "")
	+"\nIcon status is: "+(reddot ? "show.": "don't show.")
	+(iframe ? "\nHTML Frame." :"")
	+"\nCurrent values:\n"+page_list2;
    var status="Page "
	    + ( (form_data || click) ?  "stored: " : "not stored: ")
	    + page_key + (regexp_page_match ? "  (matches as regexp within, "+regexp_page_match+", specifically)." : ".")
	    + user_profile_info
    	    + "\n<a class=submenu_c href='#' style='float:right;position:relative;top:-30px;'>Open 'Simple Form Saver' submenu</a>"
	    + widener;
   if (form_data)
      status +="\n\n"+forms_info+"\n\n"+settings_info;
   else { 
      if (page_list)
	 status+="\n"+settings_info+"\n\n"+forms_info;
      else
	 status+="\n"+settings_info;
   }
   if (click && click_elem) {
      element_to_highlight=click_elem;
      with (element_to_highlight.style) { borderColor= "red"; borderWidth= "10px" ; borderStyle= "solid" ; }
   }
   
   var flen=window.frames.length;
   if (flen) status+="\n\n"+"Window contains "+flen+" iframe(s), please use icon to fill or save forms within iframes or invoke GM menu shortcut (alt-m) whilst focus is in iframe.";

   return status;
}


function getXPath(elt, counting) {
   function getElementIdx(elt) {
      var count = 0; // zero meaning only tag of that type here.
      for (var sib = elt.previousSibling; sib ; sib = sib.previousSibling)	{
	 if(sib.nodeType == 1 && sib.tagName == elt.tagName)	{
	    if (count==0) count=1;
	    count++;
	 }
      }
      if (count==0)
	 for (var sib = elt.nextSibling; sib ; sib = sib.nextSibling)	{
	    if(sib.nodeType == 1 && sib.tagName == elt.tagName)	{
	       count=1;break; //1 signalling 1 of many
	    }
	 }
      return count;
   }
   var path = "";
   for (; elt && elt.nodeType == 1; elt = elt.parentNode)    {
      idx = getElementIdx(elt);
      xname = elt.tagName.toLowerCase();
      if (idx > 0) {
	 if (elt.id && ! counting && ! /[0-9]{3,}/.test(elt.id) )
	    xname+="[@id='"+elt.id+"']";
	 else 
	    xname += "[" + idx + "]";
      }
      path = "/" + xname + path;
   }
   return path;	
}

function getXPathElem() {
   if ( ! click) return;
   var snap=uwin.document.evaluate(click.xpath,uwin.document,null,6,null), xx, xpresult;
   log("Called evaluate on "+click.xpath+", results: "+snap.snapshotLength+" pagekey: "+page_key);
   var elem=snap.snapshotItem(0);
   if (snap.snapshotLength > 1 || snap.snapshotLength == 0) {
      snap=uwin.document.evaluate(click.xxpath,uwin.document,null,6,null);
      xx=true;
   }
   if (snap.snapshotLength > 0) 
      xpresult=snap.snapshotItem(0);
   if (!xpresult) log("No xpath result, try id and class if singular, id: "+click.id+", class: "+click.className+".");
   if (!xpresult && click.id) { xpresult=document.getElementById(click.id); if (xpresult) xpresult.classed=click.id;}
   if (!xpresult && click.className) {
      xpresult=document.getElementsByClassName(click.className);
      if (xpresult.length==1) { xpresult=xpresult[0]; xpresult.classed=click.className;} else xpresult=undefined;
   }
   if (xpresult && xx) xpresult.xx=true;
   log(" got xp: "+xpresult);
   return xpresult;
}

function useCorrect_page_object() {
   if ( ! regexp_page_match ) return;
   log("cfm");
   var reply=confirm( "This page's form data is already stored under "
		      + page_key + " (as regexp).  It matches the current page, "+regexp_page_match+"." + widener
		      +"\n\nActual page matched a more general page on which form data was previously stored."
    		      +"\n\nClick 'OK' to save under general page that is already stored."
    		      +"\n\nClick 'Cancel' to save only for this specific page."  );
    if ( reply ) return;
    reply=site+window.document.location.pathname;
    var po=hash_host_list[reply];
    if ( ! po) po=new Object(); 
    hash_host_list[reply]=po;
    //po.click=click; // copy old data?
    persistData();
}

function persistData() {
    hash_host_list["data_version"]=data_version;
    hash_host_list["suspend_replay"]=suspend_replay;
    hash_host_list["reddot"]=reddot;
    //console.log("Saving data --- persistData, page_key",page_key,"profile:",profiles.current,"hostFormsList",hostFormsList,"page_object",page_object,countMembers(page_object));
    if (page_object && countMembers(page_object))
	hash_host_list[page_key]=page_object;
    else 
	delete hash_host_list[page_key];
    updateVars();
    //console.log("uneval:",hash_host_list,typeof hash_host_list,"unevaled:",uneval(hash_host_list));
    var stringed=uneval(hash_host_list);
    //console.log("persistData, write to hostFormsList:",hostFormsList,"to val:", {val:stringed,hash_host_list:hash_host_list});
    setValue(hostFormsList, stringed);
    if (page_object) log("poauto "+page_object.automatic);
    write_once=true;    
}

function readPersistentData(origin) {
    //    if ( write_once == false) return;
    var data;
    data=getValue(hostFormsList); 
    //console.log("readPersistentData, !!data ? "+(!!data)+", origin: "+origin+", hostFormsListname:",hostFormsList);
    if (data) {
	try { hash_host_list = eval(data); } catch(e) { 
	    var etext=e+"";
	    if (etext.indexOf("CSP") == -1) {
		alert("Simple Form Saver ERROR, "
		      +"dumping bad data, all form data cleared,  data corrupted: "+e
		      +".  Simple Form Saver DUMP "+data); 
		hash_host_list={};
	    }
	    else GM_log("CSP error");//security at some sites?
	}
	//console.log("hash_host_list type:",typeof hash_host_list,{list:hash_host_list});
	if (hash_host_list["data_version"] >= 1) { // have new data default to false or empty then no rev change needed.
	    suspend_replay=hash_host_list["suspend_replay"];
	}
	if (hash_host_list["data_version"] < data_version) {
	    window.status="Simple Form Saver database format increment "+data_version+".  Consider saving some forms again or of reverting versions if problems are encountered";
	    console.info("Simple Form Saver database changed.  Consider saving forms again if problems encountered");
	}
    } else { hash_host_list=new Object(); } 
    if (countMembers(hash_host_list) == 0)   {
	hash_host_list["suspend_replay"]=suspend_replay;
	hash_host_list["reddot"]=reddot;
    }
    updateVars();
    write_once=false;
   //log("Read Persist  key: "+  page_key + "\n\nPage data:\n "+uneval(page_object.form_data))
}

function updateVars() { //called after reading and before writing underlying data.
    site=getSite();
    page_key=site;
    if (page_key) {
	try { // pathname gives file after site name but before any additional stuff eg, after a '?' in long href.
	    page_key+=window.document.location.pathname; } catch(e) { page_key+=window.document.title.substring(0,10);}
    }
    else
	page_key=window.document.title.substring(0,40);
    
    checkForInputs();
    getMatchingPageObject();
    form_data=page_object.form_data;
    click=page_object.click;
    suspend_replay=hash_host_list["suspend_replay"];
    reddot=hash_host_list["reddot"];
    //console.log("Updated vars, form_data is: ",form_data,", click:"+click+", key: "+page_key+", reddot:"+reddot+", prof:"+profiles.current);
}

function getMatchingPageObject(key_in) { //dual function updates global vars or returns value;
   var key=key_in||page_key;
   var po=hash_host_list[key];
   var regexp_match=false;
   if ( ! po) {
      for(var i in hash_host_list) { // i as regexp in longer strings site,page_key
	 if (i.match("/")) continue;// only bare site name matches all.
	 if (site.match(i) || key.match(i) ) {
	    log("Match found in hash for:"+i+", in "+site+", &/or in page key: "+key)
	    po=hash_host_list[i];
	    regexp_match=key;
	    key = i;
	    break;
	 }
      }
      if ( ! po)
	 po=new Object();
   } // end if ! po
   if (!key_in) { //update globals
      page_key=key;
      page_object=po;
      regexp_page_match=regexp_match;
   }
   else return po;
}//end matchPageObject()

function getSite() {
   var  host;
   var domain_regexp=/((\.\w+\.|^)\w+.\w*$)/;
   try{ 
      host=window.document.location.host;
      href=window.document.location.href;
      if ( ! host)
	 host="localfile";
   }
   catch(e){ 
      host="";
      if (window.document.title)
	 GM_log("can't get site for doc: "+uwin.document.title)
   }
   try { if ( ! href ) href = window.document.title } catch(e)  { GM_log("Cant get doc href or title")}
   return host;
}

function insertCode(code, win) {
    if (code.length > 1) {
	if (!win) win=window;
	var script = win.document.createElement("script");
	script.type = "application/javascript";
	script.textContent = "(function() {" + code + "})();"; // for to exec anonymously, ie "(funcX)()" a function.
	win.document.body.appendChild(script);
	return true;
    }
}

function replayClick(manual) {
   setTimeout(function() { replayClickSwapped(manual); }, 100);
}

function replayFurtherClicks() {
   if (!click.further_clicks) return;
   for(var i=0, delay=0; i< click.further_clicks.length; i++, delay+=200) {
      setTimeout(function(j) { 
	 var elem, snap=uwin.document.evaluate(click.further_clicks[j],uwin.document,null,6,null);
	 GM_log("Replaying further clicks after 2 sec pause, on xpath, "+click.further_clicks[j]+", # matching elements: "+snap.snapshotLength);
	 elem=snap.snapshotItem(0);
	 if (!elem) return;
	 elem.relatedTarget=elem;
	 fakeClick(elem);
	 var tstamp=(new Date().getTime());
	 log(" tstamp "+tstamp);
      }, 2000+delay, i);
   }
}

//function replayClick(manual) {
function replayClickSwapped(manual) {
    //console.log("replayClickSwapped "+click+", "+suspend_replay);
    if ( ! click || (  ( suspend_replay  || page_object.automatic === false )
		       &&  ! ( manual || page_object.automatic === true)   )) {
	if (suspend_replay && click)
	    GM_log("Click auto-replay suspended from playing.  ");
	return;
    }
    //console.log("ReplayClick(), click's href: "+click.own_href+".  href: "+href+", manual "+manual);
    try{
	var prejudice={};
	function execInPageContext(nohref) {
	 var code="";
	 if ( ! nohref && click.href) 
	    code+=click.href.substring(11);
	 code+=";"+(click.onclick_js ? click.onclick_js : "");
	 return insertCode(code);
      }
      if ( checkIfIntervalLongEnough(prejudice) || manual) {
	 if (click.xpath && ( href == click.own_href || click.own_href == "any"  ) ) {
	     elem=getXPathElem();
	     log("replay-- matches href or href=any, elem.innerHTML: "+(elem?elem.innerHTML:""));
	     var java_run;
	     if ( ! elem ) { 
	       if ( regexp_page_match) return;
	       no_element_for_click=true;
	       log("replay-- doc, "+page_key+" changed.  Recorded Xpath stale: "+click.xpath    +(click.href ? ".  Try href: " + click.href  : "" )  +  (click.onclick_js ? ".  Try javascript: "+click.onclick_js : "" ) ) 
	       if (click.href && click.href.substring(0,11) != "javascript:" ) {
		  java_run=execInPageContext(true);
		  window.document.location = click.href; 
	       }
	       else
		  java_run=execInPageContext();
	       if (java_run || click.href)
		  persistClickEvidence();
	    }
	    else { // else ! !elem
	       if ( click.ctrlShift) try { 
		  elem.parentNode.removeChild(elem); var msg="Auto-Deleted element "+click.xpath; window.status=msg;  } 
	       catch(e) { var msg="Can't delete "+click.xpath; window.status=msg; GM_log(msg); }
	       else {
		  no_element_for_click=false;
		  log("replay-- Have element,  path: "+click.xpath +  (click.href ? ".  Try href: " + click.href  : "" )   +(elem.href ? ".  Try href: " + elem.href  : "" )  +(click.onclick_js ? ".  Try javascript: "+click.onclick_js : "" ) 
		      + ", elem.click:"+ (elem.click?" true.":" none.")); 
		  if (elem.tagName=="A" && elem.href != "" && elem.href.substring(0,11) != "javascript:" ) {
		      if (elem.target == "_blank")  	unsafeWindow.open(elem.href, elem.target); 
		     else window.document.location = elem.href; 
		  }
		  else {
		     if (click.href && click.href.substring(0,11) == "javascript:") 
			java_run=execInPageContext()
		     else if (click.onclick_js) 
			java_run=execInPageContext()
		  }
		  if (elem.click ) {
		     // var scripts=window.document.getElementsByTagName("script");
		     // for (var i in scripts)    if ( scripts[i] && ! /(^\s*$)/.test(scripts[i].textContent || " ") )  try { 
		     //     eval.call(uwin, scripts[i].textContent);  } catch(e){ GM_log("\nParse Error: "+e); } 
		     ///above needs remove scripts too.
		     //eval ("elem.click()"); // an input
		     //dblcheck triggering of event: 
		     //elem.addEventListener("click", function() {alert("clicked"); }, 0);
		     elem.click();
		  } else {
		     var pseudo_event = window.document.createEvent("MouseEvents");
		     // type, canBubble, cancelable, view, detail,           screenX, screenY, clientX, clientY, ctrlKey, altKey, shiftKey, metaKey, button, relatedTarget)
		     pseudo_event.initMouseEvent("click", true, true, window, 1, 0,   0,              0,          0,      false,   false, false, false, 0, null);
		     elem.dispatchEvent(pseudo_event);
		  }
	       } // else ! click.ctrlShift
	       persistClickEvidence();
	    } // end if/else have Xpath elem.
	    var msg="Info: Finished replaying click/delete, at: "+site
	       + ( elem ?  ".  Xpath ok "
		   + (!elem.xx ? click.xpath : click.xxpath+" as xpath")+".  " 
		   + (elem.classed ? elem.classed+" class used, Xpath ignored. ":"")
		   + ( click.ctrlShift ?  "Success, removed element.  " 
		     :  ( elem.tagName=="A" ? "Visited link "  +  (elem.target == "_blank" ? " in new window.  ": ".  " ) : "")
		     + (elem.click ? "Invoked input tagged: "+elem.tagName+"; text: "+elem.textContent+"; class, "+elem.className+"; value: "+elem.value+", with click() function.  " :  "Created a pseudo click event.  ")   )
		   :     ".  Failed, Xpath not on page.  " 
		   + (click.href ? "Visiting link "+click.href + ".  " :  "") )
	       + (java_run ? "Ran javascript code: "+click.onclick_js : "" );
	    hash_host_list.msg=msg;
	    GM_log(msg);
	    //persistData();
	 } // end if click.xpath
      } // end if checkIfIntervalLongEnough
      else { // last click replay was < 10 secs ago.
	 if ( ! prejudice.obj) { // prevent reload and auto replay on reload looping here.
	    persistClickEvidence(10000);// to prevent looping here, from reload(false) below.
	    GM_log("Simple Form Saver--reload page "+page_key);
	    uwin.document.location.reload(false); 
	 }
	 return true;
      }//end else
      replayFurtherClicks();
   } catch(e) {GM_log("Replay error, "+e.lineNumber+" "+e); throw(e); }
}

function persistClickEvidence(prejudice) { //if click causes reload ensures no looping.
    var click_evidence={};
    click_evidence.tstamp=(new Date().getTime() + ( prejudice ? prejudice : 0));
    if (prejudice)
       click_evidence.prejudice=true;
    page_object.click_evidence=click_evidence;
    localStorage.click_evidence=stringify(click_evidence);
    //persistData();
    //console.log("persistClickEvidence ",click_evidence.tstamp,", prej: ",prejudice, "ls.ce",localStorage.click_evidence);
}

function checkIfIntervalLongEnough(prejudice) {
    var result=true, stamp=new Date().getTime(), dc;
    try { dc=parse(localStorage.click_evidence); }catch(e) { localStorage.removeItem("click_evidence"); }; // prev version may have left bad string.
    localStorage.removeItem("click_evidence");
    page_object.click_evidence=dc;
    var dc=page_object.click_evidence;
    if (dc && dc.sealed) {
	var diff=stamp - dc.tstamp;
	//console.log("Interval since last auto-click: ",diff/1000,".  If < 5 no auto re-click.");
	if ( diff < 5000) 
	    result=false;
	if (diff < 0 && prejudice)
	    prejudice.obj=true;
	if (prejudice) delete page_object.click_evidence;
	//persistData();
	localStorage.click_evidence=stringify(page_object.click_evidence);
    }
    //console.log("checkIfIntervalLongEnough: "+result+" "+(dc? "s: "+dc.sealed+" t: "+dc.tstamp+" p:"+dc.prejudice:""));
    return result;
}

function fakeClick(target) {
   var e = window.document.createEvent("MouseEvents");// create event
   var pseudo_event_md = window.document.createEvent("MouseEvents");// create event
   var pseudo_event_mu = window.document.createEvent("MouseEvents");// create event
   var pseudo_event_click = window.document.createEvent("MouseEvents");// create event
   //type, canBubble, cancelable, view, detail, screenX, screenY, clientX, clientY, ctrlKey, altKey, shiftKey, metaKey, button, relatedTarget
   pseudo_event_md.initMouseEvent("mousedown", true, e.cancelable, e.view, e.detail, 
				  e.screenX, e.screenY, e.clientX, e.clientY, 
				  e.ctrlKey, e.altKey, e.shiftKey, e.metaKey, 
				  0, target.relatedTarget);
   pseudo_event_mu.initMouseEvent("mouseup", true, e.cancelable, e.view, e.detail, 
				  e.screenX, e.screenY, e.clientX, e.clientY, 
				  e.ctrlKey, e.altKey, e.shiftKey, e.metaKey, 
				  0, target.relatedTarget);
   
   pseudo_event_click.initMouseEvent("click", true, e.cancelable, e.view, 318153143, 
				     e.screenX, e.screenY, e.clientX, e.clientY, 
				     e.ctrlKey, e.altKey, e.shiftKey, e.metaKey, 
				     0, target.relatedTarget);
   
   pseudo_event_click.fakeClick=true;
   target.dispatchEvent(pseudo_event_md); 
   target.dispatchEvent(pseudo_event_mu); 
   target.dispatchEvent(pseudo_event_click);
   //log2("Dispatched fake click on: "+target.tagName+", "+target.id+", "+target.className+", "+target.textContent)
}

function ordinal(n) {
   var sfx = ["th","st","nd","rd"];
   var val = n%100;
   return n + (sfx[(val-20)%10] || sfx[val] || sfx[0]);
}

function updateValueTitles() {
    var just_inputs = getElementsByTagNames('input');
    for(var i=0; i<just_inputs.length; i++) {
	var title=just_inputs[i].title;
	title=title.replace(/Value:.*(?=Type:)/,"Value: "+just_inputs[i].value+".  "); //regexp "lookaround", not matching for replace /1
	just_inputs[i].title=title
    }
}
function checkForInputs() {
    if (no_of_inputs != selects.length + rinputs.length + tareas.length) {
	no_of_inputs = selects.length + rinputs.length + tareas.length;
	var new_inputs = getElementsByTagNames('input,textarea,select');
	//log("checkForInputs "+new_inputs.length);
	var new_names={}, new_ones=[];
	text_inputs = inputs_on_page = 0;
	for(var i=0; i<new_inputs.length; i++) {
	    var input=new_inputs[i];
	    var name=input.name;
	    var type=input.type;
	    if (type=="checkbox") 
		name+=mdash_char+input.value;
	    if ( ! name) {	name="yyNoName"+i; } //new_inputs[i].name=name; } ///RENAMEs
	    new_names[name]=input;
	    if ( input_names[input.name] )
		continue;
	    new_ones.push(input);
	    var style=window.document.defaultView.getComputedStyle(input, null)||{};
	    if (style.display=="none" || style.visibility=="hidden")	continue;
	    if (reddot) {
		//if (input.title.match(/Name:.*Value:.*Type/)) continue;
		// if ( !  ( /q|Gw/.test(name) && type=="text")  && ! /textarea/i.test(input.nodeName)) {
		//     if (type=="checkbox")  		  input.title+= "Name"+mdash_char+"value: "+name;
		//     else if (name)	    input.title+= "Name: "+name;
		//     else if (input.id)      	input.title+="Id: "+input.id;
		    var val=input.value;
		    // input.title+= ( /checkbox/.test(type) ? ""
		    // 		    : ".  Value: " + ( val ?  
		    // 				       ( input.type=="password" ? val[0] + "..." + val[val.length-1] : val)
		    // 				  : null)
		    // 		  );
	       //input.title+=".  Type: "+type+".";
	       if (/checkbox|radio/.test(type)) {
		  // var w=parseInt(style.width), h = parseInt(style.height), min_size=18, hi_size=20;
		  // //log("Size of tick "+h+"x"+w);
		  // var fixed_sizes=getValue("fixed_sizes", false);
		  // if ( (w < min_size || h < min_size) && ! fixed_sizes) {
		  //   //log("Too small, set to "+min_size);
		  //   input.style.setProperty("height", min_size+"px", "important"); 
		  //   input.style.setProperty("width", min_size+"px", "important");
		  //   input.style.setProperty("-moz-appearance", "none", "important");
		  //   // input.addEventListener("mouseover", function(){ this.style.setProperty("height", hi_size+"px", "important");		  this.style.setProperty("width", hi_size+"px", "important");		}, false);
		  //   // input.addEventListener("mouseout", function(){ this.style.setProperty("height", min_size+"px", "important");		  this.style.setProperty("width", min_size+"px", "important");		}, false);
		  // }
		  //var roll="";	    for(var j in style) roll+=j+":"+style[j]+"::"+style[style[j]]+", "
		  //log("Computed style "+roll);
		  // input.style.MozAppearance="none";
		  // input.style.height="1em";
		  // input.style.width="1em";
	       }
	    //} // if ! q|Gw
	    //if (iframe) input.title+=" (Within iframe).";
	    if (/text/.test(type))           	text_inputs++;
	    if ( ! /hidden|submit|image|reset/.test(type) )           		inputs_on_page++;
	 }
      }
      inputs=new_inputs;
      input_names=new_names;
      return new_ones;
   } //end if change in number of inputs
   else {
      var new_inputs = getElementsByTagNames('input,textarea,select');
      for(var i=0; i<new_inputs.length; i++) {
	 var input=new_inputs[i];
	 var name=input.name;
	 var type=input.type;
	 var style=window.document.defaultView.getComputedStyle(input, null)||{};
	 if (style.display=="none" || style.visibility=="hidden")	continue;
	 if (reddot) {
	    // if (input.title.match(/Name:.*Value:.*Type/)) //update title value if necessary
	     //input.title="";
	    // if (type=="checkbox")  		  input.title+= "Name: "+mdash_char+"value: "+name;
	    // else if (name)	    input.title+= "Name: "+name;
	    // else if (input.id)      	input.title+="Id: "+input.id;
	    // var val=input.value;
	    // input.title+= ( /checkbox/.test(type) ? ""
	    // 		    : ".  Value: " + ( val ? ( input.type=="password" ? val[0] + "..." + val[val.length-1] : val ) : null));
	    // input.title+=".  Type: "+type+".";
	    //if (iframe) input.title+=" (Within iframe).";
	 }
      }
   }
} //end checkForInputs()

function prettyPrintUneval(str, oneline) {
    return str.substring(2, str.length-2);
}

function editFormData() {
   readPersistentData();
   if ( ! form_data && ! click) return;
   sprompt(widener+'\nThe following is the form filling information previously saved, '
		    + 'edit the values in the box at the bottom '
		    +'(you may need to scroll down).'
		    +'then Click OK, Click Cancel/Next to edit sites to which data applies or to simply cancel.\n\nKeep to the punctuation format.  '
		    +'Do not edit the special characters, "'+bullet+'<tab>", leave them in place, unless '
		    +'you want to remove or add an entire name/value pair.'
		    +'\n\nNote.  Each input box or tick box on a webpage has an internal name, its value is what gets filled-in by the user.  '
		    +'Pass the mouse over a form to see the internal names of the fields.'
		    +'They are given in the form name:: value & must have a special character between pairs.  Checkbox names also have their default\nvalue appearing with their name after a dash ('+mdash_char+').\n\n' 
		    +'To just fill just the values without names, clear the input and type in\nthe values leaving a space between them.\nValues with spaces must be surrounded by double quotes.\n\n'
		    + chopLongString(bullet_tab+formDataToFromString(true)), formDataToFromString(false, undefined, "", true)
		    , function(users_input) {    
		       log("Got users_input "+users_input);
		       if (users_input != null) {
			  log("ip  "+users_input +" match: "+users_input.match(separator) );
			  if (  users_input && ! users_input.match(separator)  )
			     res=formDataToFromString(false, users_input, "name_value_paste");
			  else
			     res=formDataToFromString(false, users_input);
			  if (res) {
			     log("Store "+uneval(res)+"\n\nAt page_key: "+page_key);
			     page_object.form_data=res;
			  } else  {
			     delete page_object.form_data;
			  }
			  persistData();
		       }
		       else { //is null (ie, a cancel):
			  log("Next, sprompt for window2");
			   sprompt(widener+"\nThe following is the page(s) for which the form data and click information is valid."
				       +"\nEdit the page name to indicate page name(s) where stored form data may be applied and click OK.\nClick CANCEL to leave as is."
				       +"\n\nFor example, remove all but the site name to make it valid for the entire site or for a number of sites (can use regexp or a part of the name but no slashes)"
				       , page_key, function(users_input) {
					  log("From sprompt got:  "+users_input);
					  if (users_input != null && page_key != users_input) {
					     log(1);
					     hash_host_list[users_input]=page_object;
					     log(1);
					     page_object=null;
					  }
					  persistData();
				       } ); //result_handler end of function
		       }//else
		       
		    } ); //result_handler end function
}

function editClickData(){
   sprompt(widener+"\nThe following is the XPath address of the auto-click element on this page."
	   +"\nEdit it and click OK.\nClick CANCEL to leave as is.", (click?click.xpath:""),
	   function(users_input) { 
	      log("Got "+users_input+", click: "+click);
	      if ( users_input != null ) {
		 if ( ! click) click={};
		 click.xpath=users_input;
		 click.xxpath=users_input;
		 persistData();
		 GM_log("Set click xpath to:"+click.xxpath+".");
	      }
	      else GM_log("No input");
	   });
}

function formDataToFromString(breaks, str_in, value_paste, isedit, tmp_data) {
    var data= form_data||{};
    if (tmp_data) data=tmp_data;
    var j=0, v, n, form_str="", extra_break="";
    tab=" "+bullet+" ";
    if (breaks) tab ="\n"+bullet_tab
    if ( str_in == undefined ) {  //toString
      for (n in data) {
	 j++;
	 v=data[n].v;
	 if ( ! breaks ) {
	    if ( ! v && v != 0)
	       v="null";
	    if (v==tick || v==ex) {
		data[n].checkbox=true;
		(v==tick ? v = "on":  v= "off");
	    }
	 }
	  else { // breaks.
 	      if (typeof v == "object") {
		  v = v + "\t\t[ "+getOptionsNames(n, v)+" ]"
		  data[n].select_multi=true;
	      }
	      if (input_names[n] && input_names[n].type=="select-one")
		  v = v  + "\t\t[ " + getOptionsNames(n , v) + " ]"
	      if ( j % 3 == 0 ) extra_break="\n"
	      else extra_break="";
	  } // end ! breaks
	  if ( ! (data[n].pw) || unhide || isedit) {
	      form_str+=n+separator;
	      form_str+=(breaks ? "\t" : " ") +v+(unhide?"(unhidden)":"")+extra_break+tab;
	  }
	  else    { 
	     form_str+=n+separator
	     this.pw=v;
	     var i=0
	     form_str += (breaks ? "\t" : " ") + (v[0]?v[0]:"");
	     while (i++ < v.length-2)
		form_str+="*";
	     form_str+=(v[i]?v[i]:"");
	     form_str+=tab;
	 }
      } //end for n in data
      return form_str.substring(0,form_str.length-tab.length);
   }
    //
    //endif str_in==undefined, hence it is fromString:
   if (value_paste) return doValuePaste(str_in, data);
   name_value=str_in.split(bullet);
   var val;
   for(var i=0; i < name_value.length; i++) {
      log("for loop, str_in, nm pair:"+name_value[i]);
      if ( ! name_value[i])
	 continue;
      words=name_value[i].split(separator);
      if (words.length != 2)
	 continue;
      var name=words[0].replace(/^\s+|\s+$/g,"");
      val = words[1].replace(bullet_regexp, "").replace(/^\s+|\s+$/g,"");
      if (val =="null")
	 val=null;
      if ( ! data[name] ) data[name]=new Object();
      //if (data[name].pw)   continue;
      data[name].v=val;                       ////
      log("not pw, data[name] val: "+data[name].v);
      data[name].user_set=true;
      if (data[name].checkbox) {
	 val=val.replace(/\s*/g,"");
	 if (/^on$/i.test(val))
	    data[name].v=tick;
	 else
	    data[name].v=ex;
      }
      
      if (input_names[name] && input_names[name].type=="select-one") 
	 if (isNaN(val) ) 
	    data[name].v = getOptionsIndex(input_names[name], val, data[name]);
      else
	 data[name].v = val;
      log("Set val: "+data[name].v+", type "+(typeof data[name].v));

      if (data[name].select_multi) {
	 data[name].v=val.split(/\s*,\s*/);
      }
   }
   for(var n in data) {
      if ( ! data[n].user_set) {
	 delete data[n]
      }
      else {
	 delete data[n].user_set
	 delete data[n].checkbox;
	 delete data[n].select_multi;
      }
   }
   if (countMembers(data) == 0)
      return 0;
   return data;
}

function doValuePaste(str_in, data) {
   str_in=stuffQuotedSpaces(str_in);
   log("Do paste str_in: "+str_in);
   var values=str_in.split(/\s+/);
   var val, name;
   for(var i=0; i < values.length; i++) {
      log("str in: "+values[i]);
      if ( ! values[i] )
	 continue;
      val = values[i].replace(bullet_regexp, " ").replace(/^\s+|\s+$/g,"").replace(/[""]/g,"");//2 doubles for indent
      if (val =="null")
	 val=null;
      name=nameAtI(data, i);
       log2(i+"th, val "+val+" name:" +name+ " "+(input_names[name]?input_names[name].type:"no ip"));
      if (name==false) continue;
      if ( ! data[name] ) data[name]=new Object();
      if (data[name].pw)   continue;
      data[name].v=val;                       ////
      data[name].user_set=true;
       if (input_names[name] && input_names[name].type=="select-one") {
	   if (isNaN(val)) 
	       data[name].v = getOptionsIndex(input_names[name], val, data[name]);
	   else
	       data[name].v = Number(val);
       }
      log2( "paste val: "+data[name].v+", typeof val "+(typeof data[name].v) );
      
      if (data[name].checkbox) {
	 val=val.replace(/\s*/g,"");
	 if (/^on$/i.test(val))
	    data[name].v=tick;
	 else
	    data[name].v=ex;
      }
      if (data[name].select_multi)
	 data[name].v=val.split(/\s*,\s*/);
   }	//matches end of for loop over values
   for(var n in data) {
      if ( ! data[n].user_set) {
	 delete data[n]
      }
      else {
	 delete data[n].user_set
	 delete data[n].checkbox;
	 delete data[n].select_multi;
      }
   }
   if (countMembers(data) == 0)
      return 0;
   return data;
} //actually matches end of doValuePaste()

function doNameValuePaste(str_in, data){
   name_value=str_in.split("\n");
   var val;
   for(var i=0; i < name_value.length; i++) {
      if ( ! name_value[i])
	 continue;
      var words=name_value[i].split(":");
      if (words.length != 2)
	 continue;
      var name=words[0].trim();
      val = words[1];
      if (val =="null")
	 val=null;
      name=matchInputName(name, inputs);
      //log("Got name match of "+name+" data[name] :" +data[name]);
      if ( ! name ) continue;
      if ( ! data[name] ) data[name]=new Object();
      if (data[name].pw)   continue;
      data[name].v=val;                       ////
      data[name].user_set=true;
      if (data[name].checkbox) {
	 val=val.replace(/\s*/g,"");
	 if (/^on$/i.test(val))
	    data[name].v=tick;
	 else
	    data[name].v=ex;
      }

      if (input_names[name].input_names[name].type=="select-one")
	 if (isNaN(val) ) 
	    data[name].v = getOptionsIndex(input_names[name], val, data[name]);
      else
	 data[name].v = val;
      log("Paste set val: "+data[name].v+", type "+(typeof data[name].v));


      if (data[name].select_multi) {
	 data[name].v=val.split(/\s*,\s*/);
      }
   }
   for(var n in data) {
      if ( ! data[n].user_set) {
	 delete data[n]
      }
      else {
	 delete data[n].user_set
	 delete data[n].checkbox;
	 delete data[n].select_multi;
      }
   }
   if (countMembers(data) == 0)
      return 0;
   return data;
}

function countMembers(obj) { var cnt=0;     for(var i in obj) if ( ! obj.hasOwnProperty || obj.hasOwnProperty(i)) cnt++; 	return cnt;    }

function matchInputName(name, data) {
   var res, ratio=0, high_ratio=0;
   for(var i in data) {
      ratio=ratioOfMatchedWords(name, data[i].name);
      //log("got ratio of "+ratio+" for "+name+" in data[i].name: "+data[i].name);
      if ( ratio > .5) {
	 if (ratio > high_ratio) {
	    high_ratio=ratio;
	    res=data[i].name;
	 }
      }
      if (ratio == 1)
	 return data[i].name;
   }
   return res;
}

function nameAtI(obj, i) {
   var index=i;
   for(var j in obj) {
      if ( i == 0 ) return j || "yyNoName"+index;
      i--;
   }
   return false;
}

function  stuffQuotedSpaces(str) {
   var phrase_pos=0;
   var i=str.indexOf('"', phrase_pos);
   while (i !== -1) {
      var j=str.indexOf('"', i+1);
      var phrase=str.substring(i, j);
      str=stringCutAndPaste(str, phrase, phrase.replace(/\s/g, bullet), i);
      i=str.indexOf('"', j+1);
   }
   return str;
}

function stringCutAndPaste(str, tocutout, topaste, start) {
   var n=str.indexOf(tocutout, start||0);
   if (n != -1) 
      return str.substring(0,n) + ( topaste || "") + str.substring(n+tocutout.length); 
   return str;
}

function getOptionsNames(name, value_s) {
   var result="";
   var option_names=[], options;
   var ip=input_names[name];
   if (ip) options=ip.options;
   if (typeof value_s == "object") {
      if (ip && ip.type == "select-multiple" )
	 for(var i in value_s)
	    option_names[i]=getOptionsNames(name, value_s[i]);
      result =String(option_names).replace(/,\b/g,", ");
   }
   else {
      for (var i =0; i < options.length; i++) {
	 //log("opt.inx: "+options.item(i).index +", look for: "+value_s +".  Opt.value: "+options.item(i).value );
	 var option=options.item(i);
	 if (option.index== value_s ) {
	    if (option.textContent)
	       result = option.textContent;
	    else
	       result = option.value;
	 }
      }// end for.
   }
   return result.replace(/[\r\n]/g, "").replace(/^\s+|\s+$/g,"");
}

function getOptionsIndex(ip , in_val, name) {
   var index = 0, best_match = 0;
   for (var i =0; i < ip.options.length; i++) {
      var option=ip.options.item(i);
      var opts_val=option.value;
      if ( option.textContent && option.textContent.trim() )
	 opts_val=option.textContent;
      opts_val=opts_val.toLowerCase().trim();
      in_val=in_val.toLowerCase().trim();
      var match_ratio=ratioOfMatchedWords(in_val, opts_val);
      if (match_ratio > best_match ) {
	 best_match=match_ratio;
	 name.str=opts_val;
	  //index=String(option.index);
	  index=option.index;
	 if (match_ratio > 0.9) 
	     //	    return String(option.index);
	     return option.index;
      }
   }
   return index;
}

function ratioOfMatchedWords(source_words, target_words) {
   source_words=source_words.replace(/[^a-zA-Z0-9_ \t]+/g," ").split(/\s+/);
   var matches=0, ratio=0, targ_ratio=0;
   for(var i in source_words)
      if (target_words.match( RegExp(source_words[i],"i") ))
	 matches++;
   ratio = matches/source_words.length;
   targ_ratio = matches/target_words.replace(/[^a-zA-Z0-9_ \t]+/g," ").split(/\s+/).length;
   ratio = (ratio + targ_ratio) / 2;
   //log("opt match "+source_words+", in targ "+target_words+".  # Matches "+matches+", in "+source_words.length+" , targ ratio:, "+targ_ratio+" ratio "+ratio)
   return ratio;
} 

function parseTextarea(ta) {
   var result="";
   var lines=ta.value.trim();
   //lines=lines.replace(/[\r\n]\s*[\n\r]/g, "\n"); //removes empty lines
   log("lines:\n "+uneval(lines));
   lines=lines.split(/\n/); // check MSDOz!!
   lines=reorderLines(lines);
   for(var i=0; i < lines.length; i++) {
       var val, res;
       var line=lines[i].trim();
       if (line && ! /:/.test(line))
	  line=": "+line;
       if (/::/.test(line) || multiMarked(i) ) {
	   val=line.splitOnce(/\s*:+\s*/)[1];
	   if (val) result += " " + parseOptionalValues(val, i);
       }
       else {
	   val=line.splitOnce(/:\s*/)[1];
	   if (res=parseDateTime(val, i)) result += " " + res;
	   //	  else if (val) result += ' "'+val.replace(/[\x27\x22]/g,"")+'"';
	   else if (val) result += ' "'+val.replace(/[\x22]/g,"").replace(/[\x27]/g,"\\\x27")+'"';
	   else result += " null ";
       }
   }
    return result.trim();
}

// setValue("line_order", "3*, 1, 2DsT, 4");
// setValue("optionals", "line1word2of3");

function parseDateTime(val, i) {
   var datetime=getValue('line_order', ""), date="", time=""; //eg, "3*, 1, 2DsT, 4"
   if ( ! /[DT]/i.test(datetime))
      return null;
   datetime=datetime.split(/,/)[i];
   if ( ! /[DT]/i.test(datetime))
      return null;
   if (/Ds/i.test(datetime)) 
      date=new Date(val).slashFormat();
   if (/T/i.test(datetime)) {
      time=new Date(val).ampmFormat();
      time += " " + getTimeZone();
   }
   result = date+" "+time;
   if (/TD/.test(datetime))  result = time+" "+date;
   //log("ret date/time "+result);
   return result;
}

function parseOptionalValues(val, i) {
    var opts=getValue("optionals", ""), opt_word, max_words;
    var pos=opts.indexOf( "line" + (i+1) ); //line number not zero counted.
    if (pos != -1) {
	pos= pos + ("line"+i).length + 4;//eg, "line1word2of3"
	opt_word=parseInt ( opts.substr (pos) );
	pos += (opt_word+"").length + 2;
	max_words=parseInt ( opts.substr(pos) );
	//log(pos+ "<pos.  opt_word "+opt_word+", max "+max_words);
	var words=val.split(/\s+/);
	if (words.length < max_words) {
	    words.splice(opt_word-1, 0, "null"); //insert 'null' if optional is not given.
	}
	val=words.join(" ");
    }
    return val;
}

function multiMarked(i) {
   var marked=getValue('line_order', "1, 2, 3, 4, 5");
   marked=marked.split(/,/);
   return /\*/.test(marked[i]);
}

//greasemonkey.scriptvals.userscripts.org/Simple Form Saver.line_order
function reorderLines(lines){
   order=getValue('line_order', "1, 2, 3, 4, 5").replace(/[^\d,]/g,"");
   order=eval("["+order+"]");
   var newlines=[];
   for( var i in lines)
      if (i  <  order.length && order[i] != 0) {
	 var old_pos=order[i]-1;
	 if (old_pos < lines.length)
	    newlines[i]=lines[old_pos];
	 else
	    newlines[i]="null";
      }
   else
      newlines[i]=lines[i];
   return newlines;
}

function pasteIntoForm() {
    // 1. Open a window containing help info with a list of form field names and into which user can paste form values, then 
    // 2. Label all form elements in page with a visible sequence number, then
    // 3. Set up a click handler for user's 'OK', which
    // 4. Calls parseTextarea() and doValuePaste() to set up values given by user, then
    // 5. Ask user to confirm the values to be set, finally call fillForm() with ordered array tmp_data.

    log2("pasteIntoForm() "+typeof pasteWindow);
    //readPersistentData();
    if (special_ta) log("special_ta tc "+special_ta.value+" id "+special_ta.id);
    if (getById("SimpleFormSave")) return;
    var ta_val="";
    //var pasteWindow=uwin.open("ex_form.html#bizarreoKeyer","_blank", "foreground");
    try { if (pasteWindow && !pasteWindow.closed) {  // in order to focus on pasteWindow
	var texta = pasteWindow.div.getElementsByTagName("textarea")[0];
	ta_val=texta.value;
	log2("ta_val "+ta_val);
	pasteWindow.close();
    }}  catch(e) { pasteWindow=null; } //catch occurs if window was closed, cant acess dead object.

    if (chrome) pasteWindow=window;
    else pasteWindow=unsafeWindow.open("","pasteIntoForm","menubar=no,scrollbars,location=no,toolbar=no, status=no,left="+screen.width+"px,top="+(screen.height/2-300)+"px,width=600em");
    log("pasteWindow "+pasteWindow);
    //pasteWindow.document.addEventListener("keydown", function(e) { if (e.keyCode == 27)  pasteWindow.sfspasteCANCEL=true;}, 0);
    var div=pasteWindow.document.createElement("div"); div.id="SimpleFormSave";
    div.setAttribute("onkeydown","if (event.keyCode == 27)  window.sfspasteCANCEL=true;");
    pasteWindow.div=div;
    var body=pasteWindow.document.body;
    function listFields() { var skip, roll="field(s) named, "; 
			    for (var i=0; i < inputs.length; i++) {
				var ip=inputs[i];
				if(skip==ip.name) continue;
				if (ip.type=="radio") skip=ip.name;
				var style=window.document.defaultView.getComputedStyle(ip, null)||{};
				var visibility=( (style.display=="none" || style.visibility=="hidden") ? " (invisible)" : "");
				ip.visibility=visibility;
				roll+= ip.name+visibility+", ";
			    }
			    return roll; }
    var tatitle="For an example form with three fields: name, last-name, sex,\npaste-in as follows in three lines,\nJohn\nWebster\nmale"
	+"\nTo do it in just one line use two colons, type or paste,\n::John Webster male\nIf there was also a field for the middle name to set it as null put:\n::John null Webster male"
	+"\nIf one of the values after the double colon (::) had spaces in it, it would need to be surrounded by double quotes, eg,\n::John "+'"Boyle OConnor Fitzmaurice Tisdall OFarell"'+" male.  "
	+"\nIf you know the names of the inputs or can guess or check them by looking at the HTML of the page then you could use, eg, "+"\nlastname: Smith"+", on any line and it would fill it in the right field";
    div.innerHTML="<p>Enter or Paste form field values into the box directly below this text; do it for each field of the form.  Click OK when complete, or leave blank and click OK to see current values.  You can put a colon "
	+"before the value if desired.  You may need to maintain the same order as the page's form so fields are now numbered with '#' on the webpage, watch out for invisible (red) fields.  "
	+"<br><br>Use a new line for each field, or use double colons (::) to put more than one field's values on a single line.  "
	+"Put the mouse over this and below for instructions and examples.  You can put the mouse over the fields of the form to see details of form.<br><br>The page has "+listFields()+"</p>"
	+"<textarea style='width:100%;height:20em;  border: double;background-color:#fffff4;color:midnightblue;' title='"+tatitle+"';>"
	+(special_ta?special_ta.value:ta_val)+"</textarea>";
    var ta=div.getElementsByTagName("textarea")[0]; 
    div.innerHTML+="<form>"
	+"<input type=button value='Cancel/Next' style='width:100; height:50;font-size: 12' "
	+"onclick='window.sfspasteCANCEL=true' " //workaround for window perms problem.
	+">"
	+"<input type=button value='OK' style='width:100; height:50;font-size: 24;' "
	+"onclick='window.sfspasteOK=true' "
	+"></form>";
    div.firstElementChild.title='Give one value on each line eg, "Pepsi" or put them in the form name \u2039colon\u203a value, eg, "Company: Pepsi" or indeed just \u2039colon\u203a value, eg, ":Pepsi". For fields not to be filled-in at all just put a single colon on that line.  '
	+"If names are given the name need not match the real name; each value is copied into the approriate field based not on the name but on the field ordering"
	+" unless the config value for pasting names is set in "
	+"about:config.  The first line is copied to the first form field, etc..  One field value per line or,"
	+" to allow more values on one line, use double colons for contiguous fields, eg, 'time:: 12:23 Eastern' will fill-in two fields, the time and the"
	+" time zone.  To skip a field use the value 'null' or leave the part after the colon empty.  Hover over below to see help text.";
    //ta.value="\n  ";
	div.innerHTML+=""
	+"<input title='When above text box is empty click Prev./Save to paste values that were previously saved here in this dialog.  Or, when not empty, "
	+"click it to save the values currently in above text box of this dialog.  To get current values, click OK when empty, to see current values that are set in the form and choice values/names middle click on red dot.' type=button value='Prev./Save' style='width:100; height:50;font-size: 12' "
	+"onmouseup='window.sfspastePrev=event.which' "
	+">";
    log2("prevPasteVals:"+page_object.prevPasteVals+".");
    if (page_object.prevPasteVals) 
	div.innerHTML+="<div id=prevsave>Previously saved at this dialog:<br><pre>"+page_object.prevPasteVals+"</pre></div>"
    else
	div.innerHTML+="<div id=prevsave></div>"
    
    ta.value=ta_val; //set from innerHTML not here!
    log2("set ta_val "+ta.value);
    var title=div.firstElementChild;
    body.appendChild(div);
    if (chrome) {
	div.style.cssText=" z-index: 99; top: 40px; height: 50%; opacity: 0.85; background-color: white; font-family: Helvetica; font-weight: bold; font-size: small; padding: 10px";
	ta.style.cssText="width: 50%; margin: 0;  border: none;"
	    +" -moz-appearance: none; border-left: double; min-height: 300px; ";
	title.style.cssText=	"background-color: -moz-field; margin:0; border: double;";
	log("set cssTExt "+ta.style.cssText);
    }
    else {
	div.style.cssText=" top: 40px; height: 50%; font-family: Helvetica; font-weight: bold; font-size: small; padding: 10px";
	// ta.style.cssText="width: 50%; height: 150px; margin: 0;  border: none;"
	// 	+" -moz-appearance: none; border-left: double;"
	title.style.cssText=	"background-color: -moz-field; color: -moz-FieldText; margin:0; border: double;";
    }
    //title.scrollIntoView();
    //pasteWindow.resizeTo(body.offsetWidth+50,body.offsetHeight+100);
    var labels=[], radio_names={}, detract=0; 
    for(i=0; i < inputs.length; i++) {
	var input=inputs[i];
	var l=document.createElement("label");
	var prev=input.previousSibling;
	if ( prev && prev.tagName=="LABEL" && /#\d$/.test (prev.textContent) ) continue;
	input.parentNode.insertBefore(l, input);
	if (input.visibility) { l.title="Invisible input, provide a value or leave line blank, name="+input.name; l.style.color="red";l.style.backgroundColor="black";}
	if (input.type=="radio") {
	    if (radio_names[input.name]==undefined)
		radio_names[input.name]=(i+1-detract);
	    else detract++;
	    l.textContent=" #"+(radio_names[input.name]);
	}
	else l.textContent=" #"+(i+1-detract);
	labels.push(l);
    }
    var processTextarea=function() {
	var ta = div.getElementsByTagName("textarea")[0];
	log("ta css "+ta.style.cssText+" tag "+ta.tagName);
	log("processing val:"+ta.value);
	log("processing ");
	var str_in, parse_method=getValue("parse_method", "values"); //can be set to "names", default is "values" which uses the order to know which value to paste.
	if (parse_method[0] != "n") 	  parse_method=true; else parse_method=false;
	if (ta.value=="") { fillWithCurrentValues(ta); return -1; }
	if (parse_method)
	    str_in=parseTextarea(ta);
	else str_in=ta.value.trim();
	log("parse_method "+parse_method+", after parse, str_in: "+str_in);
	var data={};
	for(i=0; i < inputs.length; i++) {
	    var input=inputs[i], name=input.name;
	    data[name]={};
	    data[name].i=i;
	    if (input.type=="checkbox") {
		data[name].checkbox=true;
		var cbname=name+mdash_char+input.value;
		data[cbname]=data[name];
		delete data[name];
	    }
	}
	var res;
	if (parse_method)
	    res=doValuePaste(str_in, data), list=widener, j=0; 
	else
	    res=doNameValuePaste(str_in, data), list=widener, j=0; 
	//console.log("Paste, res: ",res," data.len: "+data.length);
	for (var i in data) {
	    if (typeof data[i] === 'function') continue;
	    //log2("for i in data.  i:"+i+"="+data[i]+", type: "+typeof data[i]);
	    if (parse_method)
		list +=  "" //"field #"  + ( ( j++ ) + 1) 
		+"To be filled-in: "+data[i].v+" \t\t ("+i+")\n";
	    //		+"To be filled-in: "+(data[i].str ? data[i].str : (data[i].v||"") )+" \t\t ("+(inputs[j-1].name||"")+")\n";
	    else 
		list +=   "input name: "+i
		+", to be filled-in with: "+(data[i].str ? data[i].str : (data[i].v||"") )+"\n";
	}
	list=list.replace(/\\\x27/g,"\x27");
	//      var go_ahead=
	var w=pasteWindow.outerWidth, h=pasteWindow.outerHeight;
	//pasteWindow.resizeTo(1,1); 
	//window.focus();
	//log2("w h "+pasteWindow.outerWidth+" "+pasteWindow.outerHeight);
	sconfirm(list+"\n\nConfirm or Cancel", function(go_ahead) {
	    if (go_ahead) {
		if (res) fillForm(false, false, res);
		inOutSet();
	    }
	    if (chrome) {
		div.style.display="none";
		div.parentNode.removeChild(div);
		delete div;
	    }
	    for (var i in labels) {
		var parent=labels[i].parentNode;
		if (parent) parent.removeChild(labels[i]);
	    }
	    //pasteWindow.resizeTo(w, h);
	    pasteWindow.focus();
	}); //sconfirm()
	
    }; // end processTextarea() ///////////////////////////////
    
    var form_inputs=div.getElementsByTagName("input");
    form_inputs[0].do_onclick=function() {  // was "onclick" workaround window perms problem.
	pasteWindow.close(); log("cancel");	    
	for (var i in labels)
	    if (labels[i].parentNode) labels[i].parentNode.removeChild(labels[i]);
	if (chrome) {
	    div.style.display="none";
	    div.parentNode.removeChild(div);
	    delete div;
	}
    };
    form_inputs[0].style.cssFloat="left";
    form_inputs[1].do_onclick=function (){
	return processTextarea();                          ///////////////////////
	//log2("processTextarea "+ta.value+".");
    };
    ta.focus();
    pasteWindow.focus();
    if (form_inputs[2])
	form_inputs[2].do_onclick=function (which_button) {
	    log2("Which_button "+which_button)
	    var ta = div.getElementsByTagName("textarea")[0];
	    var taval=ta.value.trim();
	    if (taval=="") { ta.value=page_object.prevPasteVals; }
	    else {
		readPersistentData();
		page_object.prevPasteVals=taval;
		var paradiv=$(div).find("#prevsave");
		paradiv.html("Previous saved at this dialog:<br><pre>"+page_object.prevPasteVals+"</pre>");
		persistData();
		//alert("Saved "+page_object.prevPasteVals)
	    }
	};
    log2("was set ta_val "+ta.value);
    pasteWindow.ta=ta;
    pasteWindow.setTimeout(function() { ta.focus(); }, 3000);

    var siid=setInterval(function() { //Added in cos of new window's lack of perms to run code in this file.
	try { if (pasteWindow.closed) clearInterval(siid); var tmp=pasteWindow.sfspasteOK; } catch(e) { clearInterval(siid);}
	if (pasteWindow.sfspasteOK) {
	    pasteWindow.sfspasteOK=false;
	    if (form_inputs[1].do_onclick() != -1) {
		pasteWindow.blur();
		window.focus();
	    }
	}
	if (pasteWindow.sfspasteCANCEL) {
	    clearInterval(siid);
	    pasteWindow.sfspasteCANCEL=false;
	    form_inputs[0].do_onclick();
	    pasteWindow.close();   //cancel
	}
	if (pasteWindow.sfspastePrev) {
	    // tbd, paste prev values or save existing ones.
	    form_inputs[2].do_onclick(pasteWindow.sfspastePrev); //see page_object.prevPasteVals
	    pasteWindow.sfspastePrev=false;
	}
    }, 200);
} // end pasteIntoForm()


function fillWithCurrentValues(ta) { //clicking ok when pasting values and empty fills with current values.
    var tmp_data=saveFormData(true);
    var str_in=formDataToFromString(false, undefined, false, false, tmp_data);
    var vals=str_in.split(separator);  //separator is "::" w/ ctrl char betwixt.
    var roll="";
    for (var i=1; i< vals.length;i++) {
	roll+=vals[i].split(bullet)[0].trim()+"\n"; //bullet is ⚫ 
    }
    ta.value=roll; //+"\n\n";
}

function confirmClickRecording() {
   sconfirm("The next part of the webpage on which you click shall be recorded and\ncan be clicked again on subsequent visits to this page, automatically or manually."
	    +"\n\nTo ensure you click on the correct element all elements can be\nhighlighted with a red outline & show XPath (it also appears in the console,\nsee status bar (if allowed by about:config, see userscript webpage) for element tag."
	    +"\n\nClick OK to proceed\n\nHit Cancel/Next to hightlight elements.\nNote, highlighted elements may move position, hit 'Esc' key to clear highlighting."
	    +"\n\nThen click on chosen element to have a click replayed on this page when visited in future.\nShift-click searches for nearest input, form, link, or onclick element.  "
	    //+"\n\nInstead, to have an element permanently deleted from the page, do, control-shift-click, on the element.\nRecording a click will not involve a real click on the page."
	    , function(reply) { //result handler function
	       recording=reply;
	       if ( recording == false) {
		  GM_addStyle("* { border-color: red ! important ; border-width: 1px ! important ; border-style: solid ! important ; } ");
		  window.document.body.addEventListener("mouseover", mouseOver, false);
		  addEventListener("keyup", function(e) {
		     if (e.keyCode!=27) return; //esc
		     var headNode=window.document.getElementsByTagName("head")[0];
		     headNode.removeChild(headNode.lastElementChild);
		     window.document.body.removeEventListener("mouseover", mouseOver, false);
		     window.document.body.removeEventListener("keyup", arguments.callee, false);
		  }, 0);
	       }
	       //window.document.body.addEventListener("click", recordClick, false);
	       window.document.body.addEventListener("click", recordClick, true); //capture mode
	       window.status="Shall save next click on this page: "+page_key;
	       await_click=true;
	    })
}

function recordClick(e) {
   log("recordClick "+e.target.className+" "+e.target.tagName+" "+recording);
   if ( $(e.target).hasClass("sfsdiax") ) return true;
   if ( recording === undefined) return true;
   // if ( ! FireFox && ! window.counted_confirm_click) { 
   //   window.counted_confirm_click=true;
   //   return true;
   // }
   await_click=false;
   if (e.preventDefault) {
      e.preventDefault();   
      e.stopPropagation();
      e.stopImmediatePropagation();
   }
   if ( recording == false) {
      var headNode=window.document.getElementsByTagName("head")[0];
      headNode.removeChild(headNode.lastElementChild);
      window.document.body.removeEventListener("mouseover", mouseOver, false);
   }
   this.removeEventListener("click", arguments.callee, true);
   //img.inOutSet(); might be on different page, so cant access this img.style w/o security  error
   //readPersistentData();
   var click={};
   var elem=e.target;
   var cond=false, onclick;
   var orig_onclick=elem.getAttribute("onclick") // or onmousedown, up
   if (e.ctrlKey && e.shiftKey)
      click.ctrlShift=true;
   else {
      if (e.shiftKey) {
	 while ( elem && elem.tagName ) {
	    onclick=elem.getAttribute("onclick");
	    cond= /^(input|form|a)$/.test ( elem.tagName.toLowerCase() );
	    cond = cond || onclick;
	    if (cond) break;
	    elem=elem.parentNode;
	 }
	 if ( ! elem || ! elem.tagName)
	    elem=e.target;
      }
      click.onclick_js=elem.getAttribute("onclick") 
      click.href=elem.href;
   }
   log("rem click listnr");
   log("store click info");
   click.own_href=href;
   click.xpath=getXPath(elem, true);
   click.id=elem.id;
   click.className=elem.className;
   click.xxpath=getXPath(elem);
   log("Click was on element, innerHTML: "+elem.innerHTML);
   log(" onc "+orig_onclick+" "+elem.onclick);
   elem.id=null;
   //elem.parentNode.removeChild(elem);
   page_object.click=click;
   var click_msg="Mouse click to save was on element: "+elem.tagName+", text, "+elem.textContent+", class, "+elem.className+" path: "+click.xpath;
   GM_log(click_msg+" for page "+page_key);
   var subpage=href.substring ( href.lastIndexOf ( page_key ) + page_key.length)
   log("confirm again on subpage: "+subpage);
   if (click.ctrlShift) // asych calls, events can intervene (eg, unload and clobber data)
      sconfirm( "Page name: "+page_key+"\n" +"Element xpath: " + click.xpath+(click.xpath!=click.xxpath ? " ("+click.xxpath+")":"")+widener
		+"\n\nThe element on which the mouse click was just made shall be automatically deleted from this page from now on, "
		+"unless automatic form-fill & replay are explicitly disabled for the page." 
		+"\n\n"
		+ "Click 'OK' to auto-delete this element in future on this specific page name, even when the page name is  suffixed."
		+"\n\nClick 'Cancel ' to abort or to specify on which pages the element is to be deleted"
		+(subpage  ?  ", ie, on entire site or only on pages ending with the suffix:\n\t\t" + subpage : "" )  + "."
		+"\n\n"
		+(suspend_replay ? "\n\nNOTE: auto/replay-delete is currently suspended, to enable it go to the toggle in the GM menu." : ""),
		reply_handler
	      );
   else
      sconfirm( "Page name: "+page_key+"\n" + "Element xpaths: " + click.xpath+(click.xpath!=click.xxpath ? "\n\n("+click.xxpath+")":"")+"\n"+elem.textContent.replace(/\n/g,"")+widener
		+"\n\nThe mouse click just made shall be replayed automatically on this page from now on " 
		+"unless automatic form-fill & replay are explicitly disabled for the page or in general.  " 
		+"\nAny form data for the page will also be filled in prior to replaying the click unless auto-fill has been explicitly disabled for this page."
		+"\n\nTo access this page in future it may be necessary to first select GM icon 'User Script Commands' menu option 'Suspend the Auto-Replay' or explicitly disable auto-replay for the page so click will only replay when the form is manually filled in via the red icon at top left or the GM icon"
		+"\n\n"
		+ "Click 'OK' to auto-replay this click in future on this specific page name, even when the page name is suffixed."
		+"\n\nClick 'Cancel/Next ' to specify on which page(s) to replay it, "
		+(subpage  ?  ", ie, on entire site or only on this page ending with the suffix:\n\t\t" + subpage : "" )  + ", or to abort."
		+""
		+(suspend_replay ? "\n\nNOTE: auto-replay is currently suspended, to enable it go to the toggle in the GM menu." : ""),
		reply_handler
	      );
   function reply_handler(reply) {
      log("reply_handler for record "+reply);
      if (reply) {
	 page_object.click=click;
	 if(page_object.automatic!=false) page_object.automatic="on";
	 click.own_href="any";
	 recording=undefined;
	 hash_host_list.msg=click_msg
	 persistData();
	 window.status=": "+hash_host_list.msg;
      }
      else { //cancel
	  sprompt("Will replay click (or element deletion) on the following pages.  "
		  +"\nThe site is, "+site+", edit the details as required or delete all but "
		  +"the site name for it to operate on entire site."
		  +"\nForward slashes[/] after the site name specifies"
		  +"\nthat it only applies to that page not whole site. "
		  +"\n\nClick 'Cancel' to abort entire recording.", decodeURIComponent(page_key+subpage), 
		  function(subpage_reply)	{ 
		    if (subpage_reply) {
		       log("page_key "+page_key+", == reply?: "+subpage_reply);
		       if (page_key != subpage_reply) { //move all to new page object, ie, & del this one. 
			  click.own_href="any"; 
			  var po=hash_host_list[subpage_reply];
			  GM_log("New page object for, "+subpage_reply+", previous, "+page_key);
			  if ( ! po) { 
			     po=new Object(); 
			     function copy(destination, source) {
				for (var property in source) {
				   if (typeof source[property] === "object" &&
				       source[property] !== null ) {
				      destination[property] = destination[property] || {};
				      arguments.callee(destination[property], source[property]);
				   } else {
				      destination[property] = source[property];
				   }
				}
				return destination;
			     };
			     copy(po, page_object);
			     log("no page for it, copied page_object "+page_object+" to po");
			  }// end if !po
			  
			  hash_host_list[subpage_reply]=po;
			  po.click=click; 
			  if(po.automatic!=false) po.automatic=true;
			  //page_object={};
			  log("cleared page obj and set hash_host_list of given to "+hash_host_list[subpage_reply]);
			  GM_log("Operates on different pages: "+subpage_reply);
		       } //end if != subpage_reply
		       else { 
			  page_object.click=click;
			  if(page_object.automatic!=false) page_object.automatic=true;
			  click.own_href="any";
			  log(" == to page_key, own_href: "+click.own_href)
		       }
		    }
		    else { window.status="Click recording aborted"; GM_log("Recoding aborted"); return;}
		    persistData();
		    recording=undefined;
		 });
      }// end else //cancel
      
   } // end reply handler
} // end recordClick()

function mouseOver(e) {
   var msg="Mouse is over: "+e.target.nodeName+".   Xpath: "+getXPath(e.target);
   window.status=msg;
   GM_log( msg+"\n"+e.target.textContent.substr(0,40).replace(/\n+/g," ") );
}

function handlePopups(e) { try {     //node was inserted
   if (!e.target.tagName) return;
   if (recording !== undefined) return;
   setTimeout(function() {   
      //log("handlePopups elem: "+e.target.tagName+", tc: "+e.target.textContent.substr(0,10).replace(/\s+/g,"")+", class: "+e.target.className);
      readPersistentData("popups");   },  0);
   var target=e.target;
   if (page_object.automatic) {
      var opt_flag;
      var new_elems=getElementsByTagNames("input,textarea,select"
					  +(awaiting_an_option ? ",option" : "") , target);
      var news=new_elems.slice(0);
      while ( target=new_elems.pop() ) {
	 if ( old_elems.some(function(e) {return target==e; }) ) continue;
	 if (target.tagName.toLowerCase()=="option") { opt_flag=target; target=target.parentNode;}
	 if (/^(input|textarea|select)$/.test(target.tagName.toLowerCase()))
	    if (form_data && form_data[target.name] ) {
	       var once_off=[];
	       once_off[0]=target;
	       if (target.options) {
		  var index=form_data[target.name].v;
		  if (target.options.length <= index) { awaiting_an_option = target; continue }
		  if (opt_flag && opt_flag.index!=index) continue;
		  if (awaiting_an_option == target) awaiting_an_option=false;
	       }
	       setTimeout(function() {      fillForm(once_off) }, 0);
	       //target.title  +=  (   target.name  ?   "Name: "   +   target.name    :      (target.id ?   "Id: "   +  target.id    :  ""   )    )  +     ".  Value: "  +  target.value  +   ".  Type: "   +    target.type
	    }
      } // end while
      old_elems=news.length? news : old_elems;
   }
   if (click && ! awaiting_an_option && no_element_for_click && ! suspend_replay) {
      if (getXPathElem()) { 
	 ws(function() {  	if (no_element_for_click) {    if (fillForm(false, true)) no_element_for_click=false;    }  	    }, 0);
      }
   }
}catch(e){GM_log(e); throw(e);}
			 }

function log(str) {
   //GM_log(str); return;
   if ( typeof dcount == "undefined" ) { dcount=2; }
   var d=new Date();  d=d.getMinutes()+"m:"+d.getSeconds()+"s:"+d.getMilliseconds()+"ms ";
   if ( ! log.win) {
      log.win=unsafeWindow.open(""); // need to allow site in noscript & be online for this.
      log.doc=log.win.document; 
      str=d+" i-0: Doc: "+window.document.location.href+".<br>"+d+" i-1: "+str;
   }
   var style="style='margin-left : 100px; border-bottom: solid 1px; font-size: 14pt;line-height: 2em ' ondblclick='document.body.innerHTML=null'";
   try{  log.doc.writeln("<div "+style+">"+d+"i-"+dcount+":   "+str+" "+(iframe?"(iframe msg)":"")+"</div>"); log.doc.title=dcount; dcount++; }
   catch(e){ window.setTimeout(function() {log(str)}, 0);	}
}

function getXY(obj) {
   var curleft = 0;  var curtop = obj.offsetHeight + 5;  var border;
   function getStyle(obj, prop) {	    return document.defaultView.getComputedStyle(obj,null).getPropertyValue(prop);    }
   if (obj.offsetParent)    {
      do	{
	 //  If the element is position: relative we have to add borderWidth
	 if (  /^rel/.test ( getStyle (obj, 'position') )  ) {
	    if (border = getStyle(obj, 'border-top-width')) curtop += parseInt(border);
	    if (border = getStyle(obj, 'border-left-width')) curleft += parseInt(border);
	 }
	 curleft += obj.offsetLeft;
	 curtop += obj.offsetTop;
      }
      while (obj = obj.offsetParent)
   }
   else if (obj.x)    {
      curleft += obj.x;
      curtop += obj.y;
   }
   return {'x': curleft, 'y': curtop};
}

function trickleUp(el) { try {
   var style=window.document.defaultView.getComputedStyle(el, null);
   //if (/^rel/.test(style.position))     el.style.setProperty("position","static", "important");
   if (/^abs/.test(style.position))  {   el.style.setProperty("margin-left","0px", "important"); 
					 el.style.setProperty("margin-right","0px", "important"); 
					 log("trickleUp unset abs, margins, on "+el.tagName+" "+el.id);
				     }
   if (el.parentNode) trickleUp(el.parentNode); } catch(e){}
		       }

function getTimeZone() {
   var right_now = new Date();
   var jan_here = new Date(right_now.getFullYear(), 0, 1, 0, 0, 0, 0);  
   var temp = jan_here.toGMTString();
   var jan_GMT = new Date(temp.substring(0, temp.lastIndexOf(" ")-1));
   var offset =  (jan_GMT - jan_here) / (1000 * 60 * 60);
   offset|=0;
   //log("off set "+offset);
   switch(offset) {
   case 0: return "GMT";
   case 5: return "eastern";
   case 6: return "central";
   case 7: return "mountain";
   case 8: return "pacific";
   default: return "GMT+"+-offset;
   }
}

function getById(id) {
   var el=window.document.getElementById(id);
   return el;
}

function getByIdOrClass() {
   var res=[];
   for(var i=0; i<arguments.length; i++) {
      var els=[];
      els[0]=window.document.getElementById( arguments[i] );
      if ( ! els[0] )  els=window.document.getElementsByClassName( arguments[i] ); 
      for (var j=0; j < els.length; j++) {
	 log(arguments[i]+"<len: "+els.length+" push "+els[j].tagName+", id: "+els[j].id+", class "+els[j].className);
	 res.push(els[j]);
      }
   }
   return res;
}

// function fillFromAddressBook() { //alt shift a
//     var addressbook="\n"+getValue("formsAddressBook","");
//     var act_el=window.document.activeElement;
//     var key=act_el.value;
//     var line=addressbook.match( RegExp("\\n\\s*"+key+".*", "i") );
//     if (!line) line=addressbook.match( RegExp(".*"+key+".*", "i") );
//     if (line) line=line[0];
//     else {
// 	alert("Alt-shift-a was hit.  There is no line in addressbook with:"+key+", open address book with alt-a to check."+line);
// 	return;
//     }
//     var sel=line.match( RegExp("\\n\\s*"+key+"\\S*\\s\(.*\)", "i") );  // Brackets splits to give sel[1] below 
//     if (!sel) {
// 	sel=line.match( RegExp("\(\\S*"+key+"\\S*\\s*.*\)", "i") );
// 	log2("not keyed, sel "+sel+", line:"+line);
//     }
//     if (sel) {
// 	sel=sel[1];
// 	if (sel=="") sel=line;
// 	act_el.title="From:"+line.replace(/^\n/,"")+", line in address book (alt-shift-a)";
// 	$(act_el).trigger("mouseout");//over
// 	$(act_el).trigger("mouseover");//over
//     }
//     else sel=line;

//     act_el.value=sel;
//     var tip=$("<p><strong>"+act_el.title+"</strong></p>");
//     $(act_el).after(tip);
//     tip.fadeOut(20000);
// }

function esc(str) {
  return str.replace(/[\-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g, "\\$&");
}

// function formsAddressBook() { //alt a
//     var addressbook=getValue("formsAddressBook","");
//     var last_elem, old_style={}, flag;
//     var dialog=sprompt("", addressbook, 
// 		     function(edited_text) {
// 			 log("reply callback txt:"+edited_text+", type: "+typeof edited_text);
// 			 restoreStyle();
// 			 if (edited_text != null) setValue("formsAddressBook", edited_text );
// 		     });
//     //.moveTo(screen.width,screen.height/2-300);
//     var textarea=dialog.find("textarea,input")[0];  //document.getElementById("p2reply");
//     $(textarea).attr("ondblclick", "window.fsfdblclick=this; ");
//     textarea.style.height=Math.round(dialog.innerHeight*0.66)+"px";
//     //var hfl=dialog.document.getElementById("hostFormsListButton");
//     var heading=document.createElement("div");
//     heading.id="sfshead";
//     // on https, secure sites, cannot set tags in innerHTML nor access window.document.
//     heading.innerHTML="Simple Form Saver Address/Note Book.  Shortcuts: Alt-a for this window. Alt-Shift-a "
// 	+"to fill value in using name typed in form input. <BR/>Select or Edit text below.  "  
// 	+"Hit OK to save.  Put one item on each line, eg, Steve 056 703 4567."
// 	+"<br/>Double click on a word or number below to select it and the text up to the end of "
// 	+"line (numbers may be automatically parsed-in on certain sites); "
// 	+"this selection is then automatically entered into the field of form on the main webpage.<br/>"
// 	+"<input type='checkbox'>Tick to keep this window open; after each selection the next field on "
// 	+"the form is selected.</input>";
//     heading.title="Use \\n in text in order to copy a newline into a textarea.  "
// 	+"Double click on the words 'on' or 'off' to change tickboxes.  "
// 	+"Double click on a word that is the same as the name from a drop-down selection box.";
//     heading.style.cssText="font-size: small;";
//     textarea.parentNode.insertBefore(heading, textarea);
//     saveAndSetStyle(window.document.activeElement);
//     var intid=setInterval(function() {
// 	try{ var h=prompt_win.innerHeight; } catch(e){ log("clear interv"); 	 clearInterval(intid);       }
// 	if (pwin.fsfdblclick) {
// 	    pwin.fsfdblclick=false;
// 	    var textarea=pwin.document.getElementById("p2reply");
// 	    var s=textarea.selectionStart;
// 	    var text=textarea.value;
// 	    var sel=text.substring(s).split("\n")[0];
// 	    textarea.selectionEnd = s + sel.length;
// 	    setTimeout(function(){textarea.selectionStart=textarea.selectionEnd=0;}, 750);
// 	    sel=sel.trim();
// 	    sel=sel.replace(/\\n/g,"\n");// convert \n to a real newline
// 	    var heading=pwin.document.getElementById("sfshead");
// 	    var keep_open_tick=heading.lastElementChild.checked;
	 
// 	    var act_el=window.document.activeElement;
// 	    log("dblclick:"+sel+". active:"+act_el.tagName+", type: "+act_el.type+", name "
// 		+act_el.name+" "+keep_open_tick);
// 	    if (/SELECT/i.test(act_el.tagName)) {
// 		sel=sel.split(" ")[0];/*first word*/
// 		for (var i=0; act_el[i]; i++)
// 		    if (act_el[i].textContent.match(sel)) {
// 			act_el.value=act_el[i].value;break; }
// 	    }
// 	    else if (act_el.type=="checkbox") {
// 		sel=sel.split(" ")[0];/*first word*/
// 		if (sel=="on") act_el.checked=true;
// 		else if (sel=="off") act_el.checked=false;
// 	    }
// 	    else if (act_el.type=="radio") {
// 		sel=sel.split(" ")[0];/*first word*/  
// 		var radios = window.document.getElementsByName( act_el.name );
// 		log("radio, tc "+act_el.textContent+", val "+act_el.value+" name "+act_el.name
// 		    +" #of name:"+radios.length); 
// 		for( i = 0; i < radios.length; i++ ) {
// 		    var rad=radios[i];   
// 		    log("radio value:"+rad.value+"=="+sel+".");
// 		    if (rad.value.match(sel)) { rad.checked=true;log("checkd it!");}    
// 		    log("radio "+i+" "+rad.checked+" "+rad.textContent);
// 		    rad.focus();  }
// 	    }
// 	    else /* ordinary input/textarea: */
// 		act_el.value=sel.substr(sel.indexOfRegExp(/[0-9]/,0)); // set to 1st number on, or all if none.
	    
// 	    if (!keep_open_tick) {
// 		setValue("formsAddressBook", text.replace(/^\s*|\s*$/g,""));//trim and save
// 		restoreStyle();
// 		pwin.close();
// 	    } //no tick
// 	 else { // document.activeElement focused element;
// 	   act_el=window.document.activeElement;
// 	   var tinputs = getElementsByTagNames('input,textarea,select'), firstOne, found;
// 	   log("inputs "+tinputs.length);
// 	   for(var i=0; i<tinputs.length; i++) {
// 	     var inp=tinputs[i];
// 	     var style=window.document.defaultView.getComputedStyle(inp, null)||{};
// 	     var invisible=style.display=="none" || style.visibility=="hidden" || inp.offsetHeight==0, seek;
// 	     if (seek && !found && !invisible) { log("seek now true, n: "+inp.name);found=inp;}
// 	     if (act_el==inp) { log("This inp is active: "); seek=true;}
// 	     log("tinputs "+inp.id+" "+style.display+" "+style.visibility+", h:"+inp.offsetHeight+" "+seek+" "+inp.tagName+" act: "+act_el.tagName+", n "+inp.name);
// 	     if (!invisible && !firstOne) firstOne=inp;
// 	   }
// 	   if (!found) {  found=firstOne;}
// 	     // log2("found, id:"+found.id+", tag "+found.tagName+", type "+found.type+", name "+found.name+", title "+found.title
// 	     // 	  +" dim "+found.offsetWidth+"x"+found.offsetHeight
// 	     // 	  +"\n style "+found.style.cssText);
// 	     found.focus(); 
// 	     //found.select();
// 	     scrollToMid(found);
// 	   restoreStyle();
// 	   //save
// 	   saveAndSetStyle(found);
// 	 }
//        }//end if fsfdblclick 
//      },500); //end textarea.addEventListener()
   
//    function saveAndSetStyle(el) {
//      var bcolor;
//      if (el.type=="radio"||el.type=="checkbox") { el=el.parentNode; bcolor="orange"; }
//      //save
//      var style=getComputedStyle(el, null);
//      with (style) { //log("get settings "+borderWidth+" "+borderStyle+" "+borderColor); 
//        old_style.bw=borderWidth;  old_style.bs=borderStyle ; old_style.bc=borderColor }
//      //set 
//      with (el.style) { 
//        borderColor= bcolor||"red"; borderWidth= "2px" ; margin="0px"; padding="2px"; borderStyle= "dashed" ; }
//      last_elem=el;
//    }
//    function restoreStyle() {
//      if (last_elem) with (last_elem.style) { borderColor= old_style.bc; borderWidth= old_style.bw; 
// 	 borderStyle= old_style.bs;  }
//    }
// } //end formsAddressBook()

scrollToMid=function(elem) {
    console.log("scrollToMid",elem);
   var pos=getXY(elem);
   var  midY=window.innerHeight/2|0, midX=window.innerWidth/2|0;
    //log2("mid "+midX+" "+midY+", scrollPos:"+window.scrollX +" "+window.scrollY+". Elem pos: "+uneval(pos));
   pos.x -= window.scrollX+midX; pos.y -= window.scrollY+midY
   scrollBy(pos.x, pos.y)
}

function exportImport() { 
   log("Export/Import called at "+location.href);
   var data;
   data=getValue(hostFormsList); 
   log("export "+typeof data);
   if (data && data.substr) log("10 "+data.substr(0,10));
   else data=stringify(data);
   if ( data && data[0] != "(" ) data = "(" + data + ")";
   sprompt("Warning, clicking OK will set the entire form "
		    +"data to the values below.  The existing form data so far stored by this script is "
		    +"given below and can be copied and pasted to the same export/import window on "
		    +"another browser, or backed up to a text file and restored later on in this same browser.  "
		    +"Or, if you risk it, you can even edit the form data and click OK to save it, for example if "
		    +"the host/website name has changed"
		    +"Userscript Simple Form Saver.\n\n",
		    data?data:"", 
		    function(reply) {
		       if (reply==null) { log("rnull"); return;}
		       log("reply "+typeof reply);
		       if (reply) log("10 "+reply.substr(0,10)+" size "+reply.length);
		       //if (reply[2]!='"') {
		       //translate 2xUnevals (an FF write) to 2xJSON.stringifys (a chrome write)
		       //on chrome, read is JSON parse followed by an eval, FF is 2xeval.  One of each is in callees other is in GM_*Value wrappers below.  Redef of uneval on chrome.
		       log("eval "+eval);
		       try {
			  if (chrome) { 
			     reply=parse(reply);
			     log(" JSON res:"+reply);
			  }
			  else reply=eval(reply);
		       }catch(e) {GM_log("Caught error: "+e);}
		       log("Ex/In Got object "+reply);
		       //}
		       setValue(hostFormsList, reply);
		       readPersistentData("export");
		    });
}

function unhidePasswords() {
    //showHiddenElements();
    var pws = document.querySelectorAll('input[type="password"]');
    unhide="onceoff";
    for (var i=0; i < pws.length; i++) {
	var inp=pws[i];
	inp.form.autocomplete="off";
	inp.type="text";
	moveCaretToEnd(inp);
	if (inp.form) {
	    inp.form.addEventListener("mousedown", function(e) {
		inp.type="password";
		setTimeout(function(){ inp.type="text";inp.focus();moveCaretToEnd(inp);},500);
	    }, true);
	}
    }
}
//sprompt("abc","", function(v){alert("value:"+v+". type:"+typeof v)});

function getSetConfigs() {
    sprompt("Enter config value name to get or set its value,\nsee userscripts/openuserjs.org websites for meaning.  "
	    +"\nValue names/value types are:\npaste_shortcut (ctrl-v), use value 'true' or leave blank."
	    +"\nline_order"
	    +"\noptionals"
	    +"\nparse_method"
	    ,"",function(configName){
	if (configName != null){
	    var val=getValue(configName);
	    sprompt("Current value is given here, enter new value to change it:", val, function(configValue){
		if (configValue==null) return;
		setValue(configName, configValue);
		alert("Set "+configName+" for "+scriptName+" to:"+configValue)
	    });
	}
    });
}
function showHiddenAncestors(el) {
    var ancestors=el.parents();
    ancestors=ancestors.add(el);
    ancestors.each(function(){this.type="";});
    ancestors.each(function(){$(this).css("visibility","visible"); $(this).css("opacity","1"); });
    
    ancestors.each(function(){
	if (this.style.display=="none") this.style.display="inline";
	if ($(this).width()==0) $(this).width("100%");
	if ($(this).height()==0) $(this).height("100%");
    });
    ancestors.each(function(){$(this).css({display: "inline", zIndex:"300"}); });
}

function showHiddenElements_full() {
    var hiddeninputs, op0orvishidden, jqhidden, dispnone;
    gethiddens(1); //fills vars hiddeninputs and op0orvishidden, jqhidden and dispnone.
  
    hiddeninputs.each(function(){this.type="";});
    op0orvishidden.each(function(){$(this).css("visibility","visible"); $(this).css("opacity","1"); });
    
    jqhidden.each(function(){
	if (this.style.display=="none") this.style.display="inline";
	if ($(this).width()==0) $(this).width("100%");
	if ($(this).height()==0) $(this).height("100%");
    });
    dispnone.each(function(){$(this).css({display: "inline", zIndex:"-300"}); });
    gethiddens(2);
    if (op0orvishidden.length)
	op0orvishidden.each(function(){
	    GM_log("style.vis "+this.style.visibility+", style.op "+ this.style.opacity 
		   +" it was "+$(this).data("opandviswas"));
	});
    function gethiddens(run) {
	hiddeninputs=$("input[type=hidden]");
	op0orvishidden=$("*").filter(function() {
	    $(this).data("opandviswas", run +"# op: "+$(this).css("opacity") + ", vis: " + $(this).css("visibility"));
	return ( $(this).css("opacity") < 0.25 || $(this).css("visibility") == "hidden" );
	} );
    
	jqhidden=$(":hidden");
	jqhidden=jqhidden.filter(function() { return ! /script|style/i.test(this.tagName); });
	dispnone=$("*").filter(function() {
	    return $(this).css("display")=="none" && ! /script|style/i.test(this.tagName);;
	});
	GM_log("Hidden inputs "+hiddeninputs.length+", invisible or low opacity elements "+op0orvishidden.length+", JQ :hidden elements "+jqhidden.length+", dispnone "+dispnone.length);
    }
}

function sprompt(pretext,initval,cb,cancelbtnText="Cancel",okbtnText="OK") { // "Cancel" has reply of false or null (if a prompt), "OK" gives reply of true or "", Escape key returns undefined reply.  undefined==null is true. but not for ""
    var input_tag, input_style="width:80%;font-size:small;";
    var confirm_prompt=initval===undefined;
    if (!confirm_prompt) input_tag=initval.length<50 ? "input" : (input_style="width:95%;height:100px;","textarea");
    if (!cb) cb=x=>x;
    var content=$("<div class=sfs-content tabindex=2 style='outline:none;white-space:pre-wrap;'>"
		  +"<div>"+pretext+"</div>"
		  +(initval!==undefined ? "<"+input_tag+" spellcheck='false' style='"+input_style+"'  tabindex='1'></"+input_tag+">":"")+"</div>");
    content.find("input:not(:checkbox),textarea").val(initval);
    content.resizable();
    var sp1=$(document).scrollTop();
    var dfunc=content.dialog.bind(content);
    var dialog=content.dialog({
	modal: true, width:"auto", // position: { my: "center", at: "center center-25%", of: window }, // Greater percent further to top.
	buttons: {
	    [cancelbtnText]: function(e) { if (confirm_prompt) cb(false); else cb(null, $(this).find("input,textarea").val()); dfunc("close"); return false;},
	    [okbtnText]: function(e) {  if (confirm_prompt) cb(true); else cb( $(this).find("input,textarea").val() || ""); dfunc("close"); return false;}
	},
	close: function(e) { dialog.off("keydown"); $(document).scrollTop(sp1); if (e.key=="Escape") cb(undefined); dfunc("destroy");}  // Called from jQ by Escape, trigger with Event("keydown")
    }).parent();
    if (cancelbtnText==-1) { dialog.find("button").each(function(){   if (this.textContent=="-1") $(this).remove(); }); }
    dialog.wrap("<div class=sfs-sprompt></div>"); // allows css rules to exclude other jqueryUi css on webpage from own settings, a
    dialog.keydown(function(e){	if (e.key == "Enter" && !/textarea/i.test(e.target.tagName)) $("button:contains("+okbtnText+")",this).click();  }); 
    dialog.css({"z-index":2147483647, position:"fixed", left:200, top: "50px", width:550 });
    dialog.find(".ui-dialog-titlebar").remove(); // No img in css for close 'x' at top right so remove.  Title bar not in normal confirm anyhow.
    dialog.draggable("option","handle", ".ui-dialog-buttonpane"); //
    dialog.resizable();
    content.css({"overflow-x":"hidden","max-height":innerHeight-dialog.position().top-$(".ui-dialog-buttonpane").height()-30});
    setTimeout(function(){ content.scrollTop(0);
			   console.log("scrollTop",content.scrollTop(),content);
			   var ips=dialog.find("input,textarea");if (ips.length) ips.focus(); else content.focus();  },1000);
    return dialog; //.ui-dialog
}
function sconfirm(msg,cb,cancelbtnText,okbtnText) { return sprompt(msg,undefined,cb,cancelbtnText,okbtnText); }
function salert(msg,cb) { return sprompt(msg,undefined,cb,-1,"OK"); }

function chromeInit() {
    window.chrome=false;
    if (!this.GM_getValue || "Barychelidae"!=GM_getValue("arachnoidal","Barychelidae")){ //chromium
	console.info("Simple Form Saver userscript in non GM_ mode for chrome/safari etc.");
	window.chrome=true; window.unsafeWindow=window;
	this.GM_getValue=function(a,b) { return localStorage[a]||b; };
	this.GM_setValue=function(a,b) { localStorage[a]=b; };
	this.GM_getResourceURL=function(url) { return ""; };
	this.GM_registerMenuCommand=x=>null;
	this.uneval=function(x) { return "("+JSON.stringify(x)+")";  }; //Diff is that uneval brackets string and json excludes code only data allowed in json.
	var xhr_queue=[], xhr=new XMLHttpRequest(), next_inqueue;
	xhr.onload=x=> { //arrow function means this remains window not xhr (as a function would).
	    eval.call(window,xhr.response);
	    if (next_inqueue=xhr_queue.shift()) {  xhr.open('GET', next_inqueue); xhr.send(); }
	    else loader.call(window);
	};
	xhr.onerror=e=>console.error("Error",e);
	xhr_queue.push("https://code.jquery.com/jquery-3.1.1.js");
	xhr_queue.push("https://github.com/SloaneFox/code/raw/master/GM_registerMenuCommand_Submenu_JS_Module.js");
	xhr_queue.push("https://code.jquery.com/ui/1.12.0/jquery-ui.js");
	
	xhr.open('GET', xhr_queue.shift()); xhr.send();
	
	return true;
    } else 
	return false;
} // chromeInit()

var jqueryui_dialog_css=(
    ""
	+".ui-dialog-content,.ui-dialog,.ui-dialog textarea { font-size: 12px; font-family: Arial,Helvetica,sans-serif; border: 1px solid #757575; "
    //	+"       background:#fff; color:#333; padding:12px;margin:5px;} "
    	+"    background:#002000; color:#ccc; padding:12px;margin:5px; } "
	+".ui-dialog-buttonpane { font-size: 10px; cursor:move; border: 1px solid #ddd; overflow:hidden; } "
	+".ui-dialog-buttonset { float:right; } "
	+".ui-widget-overlay { background: #aaaaaa none repeat scroll 0 0; opacity: 0.3;height: 100%; left: 0;position: fixed;  top: 0; width: 100%;}"
	+".ui-button,.ui-widget-content { text-align:left; border: solid 1px #757575; padding: 6px 13px;margin: 4px 3px 4px 0;} "
	+".ui-corner-all, .ui-dialog-buttonpane { border-radius: 5px; border-bottom-left-radius:30px; }"
	// +".ui-corner-all,.ui-dialog-buttonpane {border-bottom-left-radius:30px;}"
	// +".ui-corner-all, .ui-corner-bottom, .ui-corner-right, .ui-corner-br { border-bottom-right-radius: 5px; }"
	// +".ui-corner-all, .ui-corner-top, .ui-corner-right, .ui-corner-tr { border-top-right-radius: 5px; }"
	// +".ui-corner-all, .ui-corner-top, .ui-corner-left, .ui-corner-tl { border-top-left-radius: 5px; }"
	+".ui-button:hover { background-color: #ededed;color:#333; } "
	+".ui-button { background-color: #f6f6f6; color:#333; }"
	+".ui-dialog {position:absolute;padding:3px;outline:none;}"
	+".ui-resizable-handle { position:absolute; cursor: url(data:image/svg+xml;base64,"
	+"iVBORw0KGgoAAAANSUhEUgAAABQAAAAUCAAAAACo4kLRAAAACXBIWXMAAAsTAAALEwEAmpwYAAABAUlEQVQY022RsWrCYACEvz9EK1HQCilYu+jWdNKlj9BNN1/Cp/IBFHTMExihdTCIm0sR26W6KNHkvw6lunjLjffdnXn7fG0/PzzeG0A/m+/Ve/REB3CL/laStn7RBTpOG0iTXgWg0ktSoO20DJDv5gBy3TxgWkQFAG+QSdnAAyhErMtuFSiN0nRUAqpuec2x2V/4YOr7fd2AH/ebR+zhbMOaCWJrF4GphfZ8sEhSNm3EVrJxY5pJkhGAkjtzNRxuyAGws2Ap0DKYWYDbQRek3e6K9A8/TNPhBf5mzbEBvDCTpCz0ADN25gJOkzPAeXICNL85spu8/N0B8LDafK0+ouQXfemVYVtdIewAAAAASUVORK5CYII="
	+") 10 10, row-resize; } .ui-resizable-sw {bottom:5px;left:5px;}"
	+".ui-resizable-w, .ui-resizable-e { width:10px;height:100%;top:-5px;} .ui-resizable-n, .ui-resizable-s { width:100%;height:10px;} .ui-resizable-n {top:-5px; } .ui-resizable-w {left:-1px; } .ui-resizable-e {right:-15px; }"
	+".ui-tooltip { font-size: 7px; }"
	+".sfs-ticks * {font-size:11px;padding:0px;margin:2px;}"
	+".ui-dialog .sfs-content { position:static; overflow:auto;}"
    	+ (str=>str+str.replace(/-moz-/g,"-webkit-"))(
	    ".sfs-content :-moz-any(div) { font-size:13px;padding:6px;margin:4px 3px 4px 0;color:#ccc;}"
		+".sfs-content :-moz-any(textarea,input) { font-size:13px;padding:6px;margin:4px 3px 4px 0;color:#eee;background-color:#002a00;}"
		+".sfs-content :-moz-any(span) { font-size:13px;padding:0;margin:0;color:#ccc;}"
		+".sfs-content :-moz-any(a,a:visited) { color:#ccc;text-decoration:underline; padding:0;margin:0;}")
	+".sfs-content a:hover {opacity:0.5;}")
	.replace(/\.ui/g,".sfs-sprompt .ui"); //prefix classes with namespace prevent interference.