Simple Form Saver

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.

// ==UserScript==
// @name Simple Form Saver
// @version  1.8.3
// @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.
// @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://openuserjs.org/src/libs/slow!/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.

//addEventListener("load", function() {GM_platform_wrapper("Simple Form Saver", "1aFdtRM", false);});

// @require https://code.jquery.com/jquery-2.2.1.js
// @require https://code.jquery.com/ui/1.11.4/jquery-ui.js
// @resource jqueryuiCss https://code.jquery.com/ui/1.11.4/themes/vader/jquery-ui.css
// -------------
// @require  https://code.jquery.com/jquery-latest.js
// @require  https://code.jquery.com/ui/1.10.3/jquery-ui.js
// @resource jqueryuiCss https://code.jquery.com/ui/1.10.3/themes/vader/jquery-ui.css


//unsafeWindow.markT("Begin sfs");
//
// Data other than settings etc. is stored as GM get/setValues named: hostFormsList and formsAddressBook.
//
var old_GM_log=GM_log;
log=function(t) { console.info(t); old_GM_log(t); }; //if (!Chrome) old_GM_log(t);}; 
//GM_log=log;
log2=log;
log2=function(){};
log=function(){};

//console.log("SimpleFormSave start. Iframe:", window.parent != window);
GM_platform_wrapper("Simple Form Saver", 6, jsLoadedCB); // Adapter for Google Chrome.

//console.log("SFS startup, window.name "+window.name+" "+location);

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 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 uwin=unsafeWindow;
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;
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,""); };
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(); if (hour>11) ap = "pm";  if (hour   > 12)  hour = hour - 12; if (hour   == 0)  hour = 12; return hour+":"+this.getMinutes()+" "+ap;};

log("Form saver, at "+location+", readyState:"+document.readyState);

function jsLoadedCB() { //Called only on chrome from platform wrapper
    main();
}

if (!Chrome) try {
    //   addEventListener("abort", function(){log("ERROR");});
    //   addEventListener("load", function() {
    if (document.readyState=="complete") main();
    addEventListener("DOMContentLoaded", function() {
	main();
	if (GM_getValue("paste_shortcut", false)) {
	    if (uwin.focusControl)
		uwin.focusControl.blur();
	    else document.activeElement.blur();
	}
    },0);
} catch(e) { GM_log(e); throw(e) }

//
//Mainline
//
function main() {
    log("Form saver, main()");
    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");
    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
      if (checkIfIntervalLongEnough()) {
	 auto_fill=true;
	 if (page_object.automatic !== false && (page_object.automatic || ! no_element_for_click)) {
	    fillForm(false, true);
	 }
	 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.") }
   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) { prompt3(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
   window.addEventListener("unload", function(){
       //readPersistentData();
       log("unload");
       //alert("unload");
       var click_evidence=local_getValue("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;
	   local_setValue("click_evidence", 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 (GM_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) 
	 prompt3("***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 )+".  "
      +( ! result? "Bounds error.  " : "") +( kick ? "changed ( "+kick+"...)" : "No changes were necessary.  ")
      +(fill_msg ? fill_msg : "");
   window.status=write_msg;
   GM_log(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 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 (FireFox) {
	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); 
    }
    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();
				 if ( inputs.length==0)
				    return;
				 useCorrect_page_object();
				 saveFormData();
				 if ( ! link )
				    addIcon(true);
				 setLive();
				 inOutSet();
			      }, "","","F" );
      
      GM_registerMenuCommand( "Record next mouse click on document...",
			      function() { 
				 log("ask confirm "+FireFox);
				 readPersistentData();
				 if (click) { 
				    confirm3("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));
					  prompt3("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( "Get Form Status & Settings",
			      function() { 
				 readPersistentData();
				 alert3(getStatusInfo());
				 if (element_to_highlight) element_to_highlight.style.borderStyle="none"; 
			      }, "","","G");  

      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.  ");
				 alert3(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) GM_setValue('hostFormsList', users_input); */
      /* 	 alert(GM_getValue('hostFormsList')); */
      /* 	 readPersistentData(); */
      /* 	 persistData(); */
      /*     }) */
      GM_registerMenuCommand("Paste Data Into Form, Ctrl-v", 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");
				 alert3(widener+(roll? roll: "none")+"\n \n"+"Size of store: "+Math.round(sizeof/1000)+"k.");
			      });    
      GM_registerMenuCommand( "Export/Import all form data", exportImport);
      GM_registerMenuCommand("Open Forms' Address/Notebook, alt-a", formsAddressBook, "", "", "A");
       GM_registerMenuCommand("Unhide passwords [once-off]", unhidePasswords, "", "", "A");
       GM_registerMenuCommand("Get/set Script 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();
				 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 request, page_str="\nPage: "+ page_key + (regexp_page_match ? " (regexp match)." : ".") ;
				 if (click && form_data)
				    confirm3(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 
						   confirm3(page_str+"\n\nOk to delete click recording? ", 
							    function(request) {
							       if (request) { delete page_object.click; wrapup();}
							       else return;
							    });
					     });
				 else 
				    if (click || form_data)
				       confirm3(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;
	    alert3("Ending automatic fill-in of forms at page: "+page_key);
	 }
	 else {
	    page_object.automatic=true;
	    confirm3("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;
			alert3("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( "_____________________________________", 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) {
    if (link && (reddot || override)) {
	if (window.document.body) window.document.body.insertBefore(link, window.document.body.firstElementChild);
	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();
    //checkPageLayout();  //!!!checkPageLayout
   setLive();
   // 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();
       inOutSet();
       e.preventDefault();   
       e.stopPropagation();
       window.document.body.setAttribute("onClick", "return false;"); //!!!
       useCorrect_page_object();
       //if (form_data) {
       confirm3("Simple Form Saver, saving data at: \n\n\t"+page_key+" Ok/Cancel?",function(reply){
	   if (!reply) return;
	   saveFormData();
	   setLive();
       });
       //return false;
   }, true);
    // lk.addEventListener("mouseup", function(e) { 	    
    //    e.preventDefault();   	    
    //    e.stopPropagation(); 
    //    return false;
    // }, true); //mousedown->context->mouseup;
    lk.addEventListener("mouseup", function(e) { // click // was "mousedown"
	mouseUpCB(e)}, true);
    function mouseUpCB(e) {
	log2("mouse btn event, on lk "+e.button+", dragged?:"+dragged+", from drag:"+e.dragEv+" new pos: "+(e.ui?uneval(e.ui.position):""));
	if (dragged) return;
	if(e.dragEv) {
	    readPersistentData();
	    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();
	    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();
	    if ( ! iframe) 
		alert3(getStatusInfo());
	    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="";
   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");
      //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+")." : "" ) + "  Simple Form Saver, GM script.";
	 link.setAttribute("noautohide","true");
      }
      //img2.style.display="inline";
      //img2.style.setProperty("border-style", "none", "important");
      full_icon=true;
   } 
   else { // no form data, nor click
      log("setLive, no form data");
      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.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+".  Added by GM script, Simple Form Saver.";
	 }
      }
      //img2.style.display="none";
      full_icon=false;
   }
   if (reddot==false)
     img.style.display="none";
   else
     img.style.display="";
   // $(img).draggable();
   // $(reddot).draggable();
}// end setLive()

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 ");
   click_elem=getXPathElem();
   var forms_info =
      ( form_data ? "Previously saved values and names are listed below" 
	: "Page Values and form names have not been saved for this page")
      +" (format: 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)." : ".") + 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;

   if (page_object && countMembers(page_object))
      hash_host_list[page_key]=page_object;
   else 
      delete hash_host_list[page_key]
   updateVars();
   var stringed=uneval(hash_host_list);
   GM_setValue('hostFormsList', stringed);
   log("PerSIST end ");
   if (page_object) log("poauto "+page_object.automatic);
   write_once=true;    
}

function readPersistentData(origin) {
   //    if ( write_once == false) return;
   var data;
   data=GM_getValue('hostFormsList'); 
   //log("readPersistentData "+(!data)+" "+origin);
   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?
      }
      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";
	 GM_log("Simple Form Saver database changed.  Consider saving forms again if problems encountered");
      }
   }
   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"];
   //log("updated vars, form_data is: "+form_data+", click:"+click+", key: "+page_key);
}

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) {
    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;
    }
    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.
   click_evidence={};
   click_evidence.tstamp=(new Date().getTime() + ( prejudice ? prejudice : 0));
   if (prejudice)
      click_evidence.prejudice=true;
   page_object.click_evidence=click_evidence;
   local_setValue("click_evidence", click_evidence);
   //persistData();
   //log("persistClickEvidence "+click_evidence.tstamp+", prej: "+prejudice);
}

function checkIfIntervalLongEnough(prejudice) {
   var result=true, stamp=new Date().getTime();
   var dc=local_getValue("click_evidence","");
   page_object.click_evidence=dc;
   //log("Check click_evidence: "+dc.tstamp+", prej: "+dc.prejudice);
   //var dc=page_object.click_evidence;
   if (dc && dc.sealed) {
      var diff=stamp - dc.tstamp;
      //log2("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();
      local_setValue("click_evidence", page_object.click_evidence);
   }
   //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=GM_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;
   var pwin=prompt3(widener+'\nThe following is the form filling information previously saved, '
		    + '\nedit the values in the box at the bottom '
		    +'(you may need to scroll down).'
		    +'\nthen Click OK, Click Cancel/Next to edit sites to\nwhich data applies or to simply cancel.\n\nKeep to the punctuation format.  '
		    +'Do not edit the special\ncharacters, "'+bullet+'<tab>", leave them in place,\nunless '
		    +'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,\nits value is what gets filled-in by the user.  '
		    +'Pass the mouse over a form\nto see the internal\nnames of the fields.'
		    +'They are given in the form name:: value & must have a\nspecial 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):
			  pwin.dontclose=true;
			  log("Next, prompt3 for window2");
			  prompt3(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 prompt3 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
			  log("pwin "+pwin);
		       }//else
		       
		    } ); //result_handler end function
}

function editClickData(){
   prompt3(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();
}

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

function parseDateTime(val, i) {
   var datetime=GM_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=GM_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=GM_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=GM_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.  Paste-in as follows in three lines, with the first line, John, second line, Webster, third line, male.  "
	+"To do it in just one line use two colons, type or paste, ::John Webster male.  If there was also a field for the middle name to set it as null put: ::John null Webster male."
	+"If one of the values after the double colon (::) had spaces in it, it would need to be surrounded by double quotes, eg, ::John "+'"Boyle OConnor Fitzmaurice Tisdall OFarell"'+" male.  "
	+"If you know the names of the inputs or can guess or check them by looking at the pages HTML then you could use, "+"  name: Peter   "+", 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>This 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=GM_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; }
	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; 
	log(" 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);
	confirm3(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();
	}); //confirm3()
	
    }; // 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 (){
	processTextarea();                          ///////////////////////
	//log2("processTextarea "+ta.value+".");
    };
    //ta.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;
	    form_inputs[1].do_onclick();
	}
	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() {
   confirm3("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)
      confirm3( "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
      confirm3( "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
	  prompt3("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"+GM_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=GM_getValue("formsAddressBook","");
    var last_elem, old_style={}, flag;
    var pwin=prompt3("", addressbook, 
		     function(edited_text) {
			 log("reply callback txt:"+edited_text+", type: "+typeof edited_text);
			 restoreStyle();
			 if (edited_text != null) GM_setValue("formsAddressBook", edited_text );
		     });
    pwin.moveTo(screen.width,screen.height/2-300);
    var textarea=pwin.document.getElementById("p2reply");
    $(textarea).attr("ondblclick", "window.fsfdblclick=this; ");
    textarea.style.height=Math.round(pwin.innerHeight*0.66)+"px";
    var hfl=pwin.document.getElementById("hostFormsListButton");
    var heading=pwin.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) {
		GM_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) {
   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=GM_getValue('hostFormsList'); 
   log("export "+typeof data);
   if (data && data.substr) log("10 "+data.substr(0,10));
   else data=JSON.stringify(data);
   if ( data && data[0] != "(" ) data = "(" + data + ")";
   var pwin=prompt3("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"
		    +"GM script 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=JSON.parse(reply);
			     log(" JSON res:"+reply);
			  }
			  else reply=eval(reply);
		       }catch(e) {GM_log("Caught error: "+e);}
		       log("Ex/In Got object "+reply);
		       //}
		       GM_setValue('hostFormsList', reply);
		       readPersistentData("export");
		    });
   var textarea=pwin.document.getElementById("p2reply");
   textarea.style.setProperty("height", "80%", "");
   setTimeout(function(){textarea.scrollTop=9999999;}, 400);
}

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);
	}
    }
}
//prompt3("abc","", function(v){alert("value:"+v+". type:"+typeof v)});

function getSetConfigs() {
    prompt3("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=GM_getValue(configName);
	    prompt3("Current value is given here, enter new value to change it:", val, function(configValue){
		if (configValue==null) return;
		GM_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);
    }
    
}
/////////////////
/////////////////// ////////////WRAPPER for Google Chrome etc.///////////////////////////////////////////
///////////////////Wrapper version 4.0.3
// Notes: the this pointer on chrome may differ from ff.
//              keypress does not pass on altKey setting (charCode is not set for keypress but for keydown for both).
//
// Functions: 1. Provides platform independence, certain functions on Firefox but missing on Chrome are provided.
//            2. Alert and prompt windows of reasonable size compared to restricted native ones.
//            3. GMsetValue etc. overridden to write/read json values.
//            4. On Chrome GM menu is provided as an GM icon top right, it overrides GM_registerMenuCommand().  Now using GM_registerMenuCommand JS module.
//            5. On Chrome enables dynamic loading of jquery and jqueryui.
//            6. On Chrome provides access to own version of status bar.

function GM_platform_wrapper(title, use_jquery, loadedCB) {  //use_jquery =[1,2,4,6], bit 2 for UI, eg for prompt3 function, bit 4 for new GM_menu registration.
    var csname=title.replace(/\W*/g,""), uwin=unsafeWindow, bg_color="rgb(173,216,239, 0.8)",depends=[],tmp_store={}; //"#add8e6"
    String.prototype.parse = function (r, limit_str) { var i=this.lastIndexOf(r);var end=this.lastIndexOf(limit_str);if (end==-1) end=this.length; if(i!=-1) return this.substring(i+r.length, end); };  //return string after "r" and before "limit_str" or end of string. 
    String.prototype.trim = function (charset) { if (!charset) return this.replace(/^\s*|\s*$/g,""); else return this.replace( RegExp("^["+charset+"]*|["+charset+"]*$", "g" ) , "");}; //trim spaces or any set of characters.
    window.outerHTML = function (obj) { return new XMLSerializer().serializeToString(obj); };
    window.FireFox=false;     window.Chrome=false; window.uscript_name=title;
    window.confirm3=confirm3;  window.prompt3=prompt3;  window.alert3=alert3; 
    window.stringify=JSON.stringify;
    window.local_getValue=local_getValue; window.local_setValue=local_setValue; 
    
    //problem with localStorage is that webpage has full access to it and may delete it all, as bitlee dotcom does at very end, after beforeunload & unload events.
    function local_setValue(name, value) { name="GMxs_"+name; if ( ! value && value !=0 ) {   try{ localStorage.removeItem(name); } catch(e){};      return;    }
					   var str=JSON.stringify(value);  try {localStorage.setItem(name,  str );}catch(e){}
					 }
    function local_getValue(name, defaultValue) { name="GMxs_"+name;  var value = null; try{value = localStorage.getItem(name);}catch(e){};    if (value==null) return defaultValue;    
						 value=JSON.parse(value);    return value;  
					       }   //on FF it's in webappsstore.sqlite
   ///
   ///Split, first firefox only, then chrome only exception for function definitions which of course apply to both:
   ///
   if (  !  /^Goo/.test (navigator.vendor) )  { /////////Firefox:
       window.FireFox=true;
       window.brversion=parseInt(navigator.userAgent.parse("Firefox/"));
       var old_set=GM_setValue, old_get=GM_getValue;
       GM_setValue=function(name, value) { return old_set( name, JSON.stringify(value));	};
       GM_getValue=function(name, defaulT) { var res=old_get ( name, JSON.stringify (defaulT) );  if (res!="") try { return JSON.parse(res); } catch(e) { console.log(uscript_name+" during parse, JSON Error: "+name+", will return result anyway, type:"+(typeof res)+", value:"+res+", error:"+e);   }	 return res; };	//rmed eval less backcompat
       if (use_jquery&2 && ! document.getElementById("jqueryuiCss") ) {
	   var src=GM_getResourceText("jqueryuiCss");
	   $('head').append("<style id=jqueryuiCss>"+src+"</style>");
	   // $("head").append ("<link id=jqueryuiCss href="+url+"rel=stylesheet type=text/css>" ); //would also load ui images but extra load.
       }
       return;
   } //end ua==Firefox
   /////////////
   ///////////////////// Only Google Chrome from here, except for functions used above defined below :
   ///////////
    window.Chrome=true;
    window.brversion=parseInt(navigator.userAgent.parse("Chrome/"));
    GM_setValue = function(name, value) { name=title+":"+name; local_setValue(name, value);};
    GM_getValue = function(name, defval) { name=title+":"+name; return local_getValue(name, defval); };
    GM_deleteValue = function(name) { localStorage.removeItem(title+":"+name);  };
    
    // Only Chromium not able for @require...
    if (use_jquery) { //Can be 1/true (default jquery), 2: jquery-ui, 4:GM_registerMenuCommand, or 6 both.
	var cb_countdown=1;
	switch(use_jquery) { case 2: cb_countdown+=2;break; case 4: cb_countdown+=1;break; case 6: cb_countdown+=3; }
	if(!window.jQuery)
	    loadScript("https://code.jquery.com/jquery-latest.js");
	else if (use_jquery==1) sourceIn("");
	if (use_jquery & 2) { //010 // eg, parseInt("010", 2) ==> 2
	    loadScript("https://code.jquery.com/ui/1.10.3/jquery-ui.js");
	    loadScript("https://code.jquery.com/ui/1.10.3/themes/vader/jquery-ui.css");
	}
	if (use_jquery & 4) { //100, eg parseInt("100", 2) => 4, 110 => 6
	    loadScript("https://openuserjs.org/src/libs/slow!/GM_registerMenuCommand_Submenu_JS_Module.js");
	}
    }//if use_jquery
    function sourceIn(filename, filetext) {
	if (/\.css$/.test(filename)) GM_addStyle(filetext);
	else tmp_store[filename]=filetext;
	cb_countdown--;
	if (cb_countdown==0) {
	    while(depends.length)
		try { window.unsafeWindow=unsafeWindow;eval.call(window,tmp_store[depends.shift()]); } catch(e){ console.log("Platform Wrapper userscript, js eval err"+e); }
	    setTimeout(loadedCB,0);
	}
    }
    function loadScript(url) {
	var filen=url.split("/").splice(-1);
	var file=GM_getValue(filen,"");
	depends.push(filen);
	if (file) {
	    //console.info("Using cached "+filen+".  To delete, close chromium and find sqlite file in chromium main dir under dir 'Local Storage' as a file with site name suffixed with .locastorage");
	    sourceIn(filen,file); return;
	}
	GM_xmlhttpRequest(  { method: "GET", url: url, onload:function(r) { //asynch
	    GM_setValue(filen, r.responseText);
	    sourceIn(filen, r.responseText);
	} });
    }
    uneval=function(x) {
      return "("+JSON.stringify(x)+")";
    };
    GM_addStyle = function(css, doc) {
	if (!doc) doc=window.document;
	var style = doc.createElement('style');
	style.textContent = css;
	doc.getElementsByTagName('head')[0].appendChild(style);
    };
    function setStatus(s) {
	//if (s)  s = s.toLowerCase ? s.toLowerCase() : s;
	setStatus.value=s;
	var div=document.getElementById("GMstatus");
	if (div) {	
   	    if (s) { div.textContent=s;	    div.style.display="block";	    setDivStyle(); }
   	    else   { setDivStyle();	    div.style.display="none"; }
	} 
	else  if (s) { 
   	    div=document.createElement('div');
   	    div.textContent=s;
   	    div.setAttribute('id','GMstatus');
   	    if (document.body) document.body.appendChild(div);
   	    setDivStyle();
   	    div.addEventListener('mouseout', function(e){ setStatus(); },false);
	}
	if (s) setTimeout( function() {  if (s==setStatus.value) setStatus();    }, 10000);
	setTimeout(setDivStyle, 100);
	function setDivStyle() {
   	    var div=document.getElementById("GMstatus");
   	    if ( ! div ) return;
   	    var display=div.style.display; 
   	    div.style.cssText="border-top-right-radius: 6px; "
		+"background: linear-gradient(#fff,#ddd);"
		+"color: rgba(0,0,0,0.8) ! important; font-weight:500;"  //text-shadow: 0 1px 0 rgba(0,0,0,.4);"
   		+"font-family: Helvetica; font-size: 15px;line-height:20px; z-index: 9909099; "
		+"padding: 2px; padding-top:0px; border: 1px solid #82a2ad; "//Lucida Sans Unicode;
   		+"position: fixed ! important; bottom: 0px; " + (FireFox && brversion >= 4 ? "left: -1px" : "" );
   	    div.style.display=display;
	}
    }//setStatus()
    initStatus();
    function initStatus() {
   	window.__defineSetter__("status", function(val){  setStatus(val); });
   	window.__defineGetter__("status", function(){    return setStatus.value; });
    }
    function prompt3(text, init_value, handler, following_text, title){
	if (!init_value) init_value="";
	var box=confirm3(text, handler, "", "", title, init_value, following_text);
	return box;
    } //prompt3()
    function alert3(text){
	confirm3(text, "", null);
    }
    
    function confirm3(text, handler, btn1, btn2, title, init_answer, following_text ) {
	if(btn1 !== null) btn1=btn1||"Cancel";
	if (!confirm3.cnt) confirm3.cnt=1; else confirm3.cnt++;
	if (!handler) handler=function(){};
	btn2=btn2||"OK";
	title=title||"";
	var topleft_btn="\u2573"; //"x";
	text=text.replace(/</g,"&lt;").replace(/\n/g,"<br>").replace(/\t/g,"&nbsp;&nbsp;&nbsp;&nbsp;");
	following_text=following_text||"";
	following_text=following_text.replace(/</g,"&lt;").replace(/\n/g,"<br>").replace(/\t/g,"&nbsp;&nbsp;&nbsp;&nbsp;");
	var res, buttons={}, box, dinput="";
	if (init_answer!==undefined) {
	    dinput="<input id=fsfdinput style='width:100%; background:white;color:black'></input>";
	    if (!text) dinput="<textarea id=fsfdinput style='width:100%; height:98%; white-space:pre-wrap;'></textarea>";
	}
	box=$("<div id=sfsconfirm3 title=x><div id=sfsmsgtxt></div>"
	      +dinput+"<div id=sfsfollowing></div></div>");
	var ip=box.find("input, textarea");ip.val(init_answer);
	ip.css({bordertyle:"indent", borderWidth:"4px",padding:0,margin:0});
	var msgtxt=box.find("#sfsmsgtxt"); msgtxt.html(text); 
	var followtxt=box.find("#sfsfollowing"); followtxt.html(following_text);
	//GM_addStyle("pre { font-family: Sans-Serif;}"); // if wrap text in <pre>
	$("body").append(box);
	
	buttons[btn2]=function(e) { //OK 
	    var ip=$(this).find("input, textarea");//can't use id since may be duplicated, queueing of dialoges?
	    handler(ip.length?ip[0].value:1);
	    $(this).dialog("close");
	    e.stopPropagation();	    e.preventDefault();
	};
	if(btn1 !== null) buttons[btn1]=function(e) { //cancel
		handler(null); $(this).dialog("close");
		e.stopPropagation();	    e.preventDefault();	 
	};
	box.dialog({ position: "center", modal: true, buttons: buttons, resizable: true,height:"auto", width:"auto", overflow:"auto" }); //wraps box in dialog elems.
	if (!ip[0] || ip[0].tagName=="INPUT")  box.keydown(function (event) {
	    if (event.keyCode == 13)  {   $(this).parent().find("button:eq(1)").trigger("click");    }  });
	//
	// Dialog element has three component elements: titlebar, box and buttonpane.
	var dialog=box.closest(".ui-dialog"), titlebar=$(".ui-dialog-titlebar",dialog), buttonpane=$(".ui-dialog-buttonpane",dialog), limit=window.innerHeight*0.66*0.85, longD;
	if (box.height()>limit) {  longD=true;log("too long "+limit);    box.height(limit);       dialog.css("top", window.innerHeight*(1/7)+"px");   }
	limit=window.innerWidth*.66;
	if (box.width()>limit) {      box.width(limit);    dialog.css("left", window.innerWidth*(1/6)+"px");   }
	dialog.css({ zIndex:2147483642, textAlign:"left", position: "fixed",
		     left:"15%", width:"50%",height:"70%",top:"10%", fontSize:"medium" });
	$(dialog).add(titlebar).css({background:"#002400", boxSizing: "content-box"});
	buttonpane.css({background: "#001800",marginTop:0, boxSizing: "content-box"});
	GM_addStyle(".sfsdiax {white-space:nowrap}");
	$(".ui-button").css("font-size", "small");

	buttonpane.css({width:"-moz-available",position:"absolute",bottom:0});
	box.css({width:"-moz-available",position: "absolute"});
	box.css({background:"#002000", color:"#eeeeee", overflow:"auto", fontSize:"medium" });
	box.children().css({background:"#002000", color:"#eeeeee"});
	dialog.find("*").addClass("sfsdiax");

	with($(".ui-dialog-title"))	{
	    children(":eq(0)").title="close";	width("5%");	css({cursor:"pointer",marginTop:"-4px",marginLeft:"-4px"});
	    click(function() { box.dialog("close"); } );
	    hover(function() { log("ft "); $(this).toggleClass("whitebtn");});
	}
	with(titlebar.find("button")) { text("x"); css({top:7,right:0}); }
	jQuery.fn.reverse = [].reverse;
	if(btn1===null) { var btn=$(".ui-dialog-buttonset .ui-button");btn.css({left:"40%"}); ; 
			  $(".ui-button")[0].focus();$(".ui-dialog-buttonset").css({float:"none"});}
	else $(".ui-dialog-buttonset button").reverse().each(function() {this.focus(); return false;});  //return false, break from each(), return true, equiv to continue
	if (longD) box.focus();
	if (ip[0]) { ip.focus();} //else dialog.focus();
	//console.log("outerHeights:",buttonpane,buttonpane.outerHeight(), titlebar,titlebar.outerHeight());
	titlebar.height("7%");box.height("65%");buttonpane.height("10%");
	box.css({bottom: buttonpane.outerHeight(), top: titlebar.outerHeight(), boxSizing: "content-box" });
	box.scrollTop(0);
	// var wrapper="<div id=iwrapper style='position:fixed !important; width:60%; height:80%;'></div>";
	// dialog.wrap(wrapper);
	// wrapper.draggable();
	return box;
    }//end fun confirm3();
    
} //end platform_wrapper()