/public/javascripts/dojo/release/dojo/dijit/form/Form.js

http://enginey.googlecode.com/ · JavaScript · 508 lines · 219 code · 41 blank · 248 comment · 50 complexity · 0629043bbece4d41495224d77fcab7f1 MD5 · raw file

  1. /*
  2. Copyright (c) 2004-2008, The Dojo Foundation All Rights Reserved.
  3. Available via Academic Free License >= 2.1 OR the modified BSD license.
  4. see: http://dojotoolkit.org/license for details
  5. */
  6. if(!dojo._hasResource["dijit.form.Form"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
  7. dojo._hasResource["dijit.form.Form"] = true;
  8. dojo.provide("dijit.form.Form");
  9. dojo.require("dijit._Widget");
  10. dojo.require("dijit._Templated");
  11. dojo.declare("dijit.form._FormMixin", null,
  12. {
  13. //
  14. // summary:
  15. // Widget corresponding to HTML form tag, for validation and serialization
  16. //
  17. // example:
  18. // | <form dojoType="dijit.form.Form" id="myForm">
  19. // | Name: <input type="text" name="name" />
  20. // | </form>
  21. // | myObj = {name: "John Doe"};
  22. // | dijit.byId('myForm').attr('value', myObj);
  23. // |
  24. // | myObj=dijit.byId('myForm').attr('value');
  25. /*=====
  26. // value: Object
  27. // Name/value hash for each form element.
  28. // If there are multiple elements w/the same name, value is an array,
  29. // unless they are radio buttons in which case value is a scalar since only
  30. // one can be checked at a time.
  31. //
  32. // If the name is a dot separated list (like a.b.c.d), it's a nested structure.
  33. // Only works on widget form elements.
  34. // example:
  35. // | { name: "John Smith", interests: ["sports", "movies"] }
  36. =====*/
  37. // TODO:
  38. // * Repeater
  39. // * better handling for arrays. Often form elements have names with [] like
  40. // * people[3].sex (for a list of people [{name: Bill, sex: M}, ...])
  41. //
  42. //
  43. reset: function(){
  44. dojo.forEach(this.getDescendants(), function(widget){
  45. if(widget.reset){
  46. widget.reset();
  47. }
  48. });
  49. },
  50. validate: function(){
  51. // summary: returns if the form is valid - same as isValid - but
  52. // provides a few additional (ui-specific) features.
  53. // 1 - it will highlight any sub-widgets that are not
  54. // valid
  55. // 2 - it will call focus() on the first invalid
  56. // sub-widget
  57. var didFocus = false;
  58. return dojo.every(dojo.map(this.getDescendants(), function(widget){
  59. // Need to set this so that "required" widgets get their
  60. // state set.
  61. widget._hasBeenBlurred = true;
  62. var valid = widget.disabled || !widget.validate || widget.validate();
  63. if (!valid && !didFocus) {
  64. // Set focus of the first non-valid widget
  65. dijit.scrollIntoView(widget.containerNode||widget.domNode);
  66. widget.focus();
  67. didFocus = true;
  68. }
  69. return valid;
  70. }), function(item) { return item; });
  71. },
  72. setValues: function(val){
  73. dojo.deprecated(this.declaredClass+"::setValues() is deprecated. Use attr('value', val) instead.", "", "2.0");
  74. return this.attr('value', val);
  75. },
  76. _setValueAttr: function(/*object*/obj){
  77. // summary: Fill in form values from according to an Object (in the format returned by attr('value'))
  78. // generate map from name --> [list of widgets with that name]
  79. var map = { };
  80. dojo.forEach(this.getDescendants(), function(widget){
  81. if(!widget.name){ return; }
  82. var entry = map[widget.name] || (map[widget.name] = [] );
  83. entry.push(widget);
  84. });
  85. for(var name in map){
  86. if(!map.hasOwnProperty(name)){
  87. continue;
  88. }
  89. var widgets = map[name], // array of widgets w/this name
  90. values = dojo.getObject(name, false, obj); // list of values for those widgets
  91. if(values===undefined){
  92. continue;
  93. }
  94. if(!dojo.isArray(values)){
  95. values = [ values ];
  96. }
  97. if(typeof widgets[0].checked == 'boolean'){
  98. // for checkbox/radio, values is a list of which widgets should be checked
  99. dojo.forEach(widgets, function(w, i){
  100. w.attr('value', dojo.indexOf(values, w.value) != -1);
  101. });
  102. }else if(widgets[0]._multiValue){
  103. // it takes an array (e.g. multi-select)
  104. widgets[0].attr('value', values);
  105. }else{
  106. // otherwise, values is a list of values to be assigned sequentially to each widget
  107. dojo.forEach(widgets, function(w, i){
  108. w.attr('value', values[i]);
  109. });
  110. }
  111. }
  112. /***
  113. * TODO: code for plain input boxes (this shouldn't run for inputs that are part of widgets)
  114. dojo.forEach(this.containerNode.elements, function(element){
  115. if (element.name == ''){return}; // like "continue"
  116. var namePath = element.name.split(".");
  117. var myObj=obj;
  118. var name=namePath[namePath.length-1];
  119. for(var j=1,len2=namePath.length;j<len2;++j){
  120. var p=namePath[j - 1];
  121. // repeater support block
  122. var nameA=p.split("[");
  123. if (nameA.length > 1){
  124. if(typeof(myObj[nameA[0]]) == "undefined"){
  125. myObj[nameA[0]]=[ ];
  126. } // if
  127. nameIndex=parseInt(nameA[1]);
  128. if(typeof(myObj[nameA[0]][nameIndex]) == "undefined"){
  129. myObj[nameA[0]][nameIndex] = { };
  130. }
  131. myObj=myObj[nameA[0]][nameIndex];
  132. continue;
  133. } // repeater support ends
  134. if(typeof(myObj[p]) == "undefined"){
  135. myObj=undefined;
  136. break;
  137. };
  138. myObj=myObj[p];
  139. }
  140. if (typeof(myObj) == "undefined"){
  141. return; // like "continue"
  142. }
  143. if (typeof(myObj[name]) == "undefined" && this.ignoreNullValues){
  144. return; // like "continue"
  145. }
  146. // TODO: widget values (just call attr('value', ...) on the widget)
  147. switch(element.type){
  148. case "checkbox":
  149. element.checked = (name in myObj) &&
  150. dojo.some(myObj[name], function(val){ return val==element.value; });
  151. break;
  152. case "radio":
  153. element.checked = (name in myObj) && myObj[name]==element.value;
  154. break;
  155. case "select-multiple":
  156. element.selectedIndex=-1;
  157. dojo.forEach(element.options, function(option){
  158. option.selected = dojo.some(myObj[name], function(val){ return option.value == val; });
  159. });
  160. break;
  161. case "select-one":
  162. element.selectedIndex="0";
  163. dojo.forEach(element.options, function(option){
  164. option.selected = option.value == myObj[name];
  165. });
  166. break;
  167. case "hidden":
  168. case "text":
  169. case "textarea":
  170. case "password":
  171. element.value = myObj[name] || "";
  172. break;
  173. }
  174. });
  175. */
  176. },
  177. getValues: function(){
  178. dojo.deprecated(this.declaredClass+"::getValues() is deprecated. Use attr('value') instead.", "", "2.0");
  179. return this.attr('value');
  180. },
  181. _getValueAttr: function(){
  182. // summary:
  183. // Returns Object representing form values.
  184. // description:
  185. // Returns name/value hash for each form element.
  186. // If there are multiple elements w/the same name, value is an array,
  187. // unless they are radio buttons in which case value is a scalar since only
  188. // one can be checked at a time.
  189. //
  190. // If the name is a dot separated list (like a.b.c.d), creates a nested structure.
  191. // Only works on widget form elements.
  192. // example:
  193. // | { name: "John Smith", interests: ["sports", "movies"] }
  194. // get widget values
  195. var obj = { };
  196. dojo.forEach(this.getDescendants(), function(widget){
  197. var name = widget.name;
  198. if(!name||widget.disabled){ return; }
  199. // Single value widget (checkbox, radio, or plain <input> type widget
  200. var value = widget.attr('value');
  201. // Store widget's value(s) as a scalar, except for checkboxes which are automatically arrays
  202. if(typeof widget.checked == 'boolean'){
  203. if(/Radio/.test(widget.declaredClass)){
  204. // radio button
  205. if(value !== false){
  206. dojo.setObject(name, value, obj);
  207. }
  208. }else{
  209. // checkbox/toggle button
  210. var ary=dojo.getObject(name, false, obj);
  211. if(!ary){
  212. ary=[];
  213. dojo.setObject(name, ary, obj);
  214. }
  215. if(value !== false){
  216. ary.push(value);
  217. }
  218. }
  219. }else{
  220. // plain input
  221. dojo.setObject(name, value, obj);
  222. }
  223. });
  224. /***
  225. * code for plain input boxes (see also dojo.formToObject, can we use that instead of this code?
  226. * but it doesn't understand [] notation, presumably)
  227. var obj = { };
  228. dojo.forEach(this.containerNode.elements, function(elm){
  229. if (!elm.name) {
  230. return; // like "continue"
  231. }
  232. var namePath = elm.name.split(".");
  233. var myObj=obj;
  234. var name=namePath[namePath.length-1];
  235. for(var j=1,len2=namePath.length;j<len2;++j){
  236. var nameIndex = null;
  237. var p=namePath[j - 1];
  238. var nameA=p.split("[");
  239. if (nameA.length > 1){
  240. if(typeof(myObj[nameA[0]]) == "undefined"){
  241. myObj[nameA[0]]=[ ];
  242. } // if
  243. nameIndex=parseInt(nameA[1]);
  244. if(typeof(myObj[nameA[0]][nameIndex]) == "undefined"){
  245. myObj[nameA[0]][nameIndex] = { };
  246. }
  247. } else if(typeof(myObj[nameA[0]]) == "undefined"){
  248. myObj[nameA[0]] = { }
  249. } // if
  250. if (nameA.length == 1){
  251. myObj=myObj[nameA[0]];
  252. } else{
  253. myObj=myObj[nameA[0]][nameIndex];
  254. } // if
  255. } // for
  256. if ((elm.type != "select-multiple" && elm.type != "checkbox" && elm.type != "radio") || (elm.type=="radio" && elm.checked)){
  257. if(name == name.split("[")[0]){
  258. myObj[name]=elm.value;
  259. } else{
  260. // can not set value when there is no name
  261. }
  262. } else if (elm.type == "checkbox" && elm.checked){
  263. if(typeof(myObj[name]) == 'undefined'){
  264. myObj[name]=[ ];
  265. }
  266. myObj[name].push(elm.value);
  267. } else if (elm.type == "select-multiple"){
  268. if(typeof(myObj[name]) == 'undefined'){
  269. myObj[name]=[ ];
  270. }
  271. for (var jdx=0,len3=elm.options.length; jdx<len3; ++jdx){
  272. if (elm.options[jdx].selected){
  273. myObj[name].push(elm.options[jdx].value);
  274. }
  275. }
  276. } // if
  277. name=undefined;
  278. }); // forEach
  279. ***/
  280. return obj;
  281. },
  282. // TODO: ComboBox might need time to process a recently input value. This should be async?
  283. isValid: function(){
  284. // summary: make sure that every widget that has a validator function returns true
  285. this._invalidWidgets = [];
  286. return dojo.every(this.getDescendants(), function(widget){
  287. var isValid = widget.disabled || !widget.isValid || widget.isValid();
  288. if(!isValid){
  289. this._invalidWidgets.push(widget);
  290. }
  291. return isValid;
  292. }, this);
  293. },
  294. onValidStateChange: function(isValid){
  295. // summary: stub function to connect to if you want to do something
  296. // (like disable/enable a submit button) when the valid
  297. // state changes on the form as a whole.
  298. },
  299. _widgetChange: function(widget){
  300. // summary: connected to a widgets onChange function - update our
  301. // valid state, if needed.
  302. var isValid = this._lastValidState;
  303. if(!widget || this._lastValidState===undefined){
  304. // We have passed a null widget, or we haven't been validated
  305. // yet - let's re-check all our children
  306. // This happens when we connect (or reconnect) our children
  307. isValid = this.isValid();
  308. if(this._lastValidState===undefined){
  309. // Set this so that we don't fire an onValidStateChange
  310. // the first time
  311. this._lastValidState = isValid;
  312. }
  313. }else if(widget.isValid){
  314. this._invalidWidgets = dojo.filter(this._invalidWidgets||[], function(w){
  315. return (w != widget);
  316. }, this);
  317. if(!widget.isValid() && !widget.attr("disabled")){
  318. this._invalidWidgets.push(widget);
  319. }
  320. isValid = (this._invalidWidgets.length === 0);
  321. }
  322. if (isValid !== this._lastValidState){
  323. this._lastValidState = isValid;
  324. this.onValidStateChange(isValid);
  325. }
  326. },
  327. connectChildren: function(){
  328. // summary: connects to the onChange function of all children to
  329. // track valid state changes. You can call this function
  330. // directly, ie. in the event that you programmatically
  331. // add a widget to the form *after* the form has been
  332. // initialized
  333. dojo.forEach(this._changeConnections, dojo.hitch(this, "disconnect"));
  334. var _this = this;
  335. // we connect to validate - so that it better reflects the states
  336. // of the widgets - also, we only connect if it has a validate
  337. // function (to avoid too many unneeded connections)
  338. var conns = this._changeConnections = [];
  339. dojo.forEach(dojo.filter(this.getDescendants(),
  340. function(item){ return item.validate; }
  341. ),
  342. function(widget){
  343. // We are interested in whenever the widget is validated - or
  344. // whenever the disabled attribute on that widget is changed
  345. conns.push(_this.connect(widget, "validate",
  346. dojo.hitch(_this, "_widgetChange", widget)));
  347. conns.push(_this.connect(widget, "_setDisabledAttr",
  348. dojo.hitch(_this, "_widgetChange", widget)));
  349. });
  350. // Call the widget change function to update the valid state, in
  351. // case something is different now.
  352. this._widgetChange(null);
  353. },
  354. startup: function(){
  355. this.inherited(arguments);
  356. // Initialize our valid state tracking. Needs to be done in startup
  357. // because it's not guaranteed that our children are initialized
  358. // yet.
  359. this._changeConnections = [];
  360. this.connectChildren();
  361. }
  362. });
  363. dojo.declare(
  364. "dijit.form.Form",
  365. [dijit._Widget, dijit._Templated, dijit.form._FormMixin],
  366. {
  367. // summary:
  368. // Adds conveniences to regular HTML form
  369. // HTML <FORM> attributes
  370. name: "",
  371. action: "",
  372. method: "",
  373. encType: "",
  374. "accept-charset": "",
  375. accept: "",
  376. target: "",
  377. templateString: "<form dojoAttachPoint='containerNode' dojoAttachEvent='onreset:_onReset,onsubmit:_onSubmit' name='${name}'></form>",
  378. attributeMap: dojo.mixin(dojo.clone(dijit._Widget.prototype.attributeMap),
  379. {action: "", method: "", encType: "", "accept-charset": "", accept: "", target: ""}),
  380. execute: function(/*Object*/ formContents){
  381. // summary:
  382. // Deprecated: use submit()
  383. },
  384. onExecute: function(){
  385. // summary:
  386. // Deprecated: use onSubmit()
  387. },
  388. _setEncTypeAttr: function(/*String*/ value){
  389. this.encType = value;
  390. dojo.attr(this.domNode, "encType", value);
  391. if(dojo.isIE){ this.domNode.encoding = value; }
  392. },
  393. postCreate: function(){
  394. // IE tries to hide encType
  395. // TODO: this code should be in parser, not here.
  396. if(dojo.isIE && this.srcNodeRef && this.srcNodeRef.attributes){
  397. var item = this.srcNodeRef.attributes.getNamedItem('encType');
  398. if(item && !item.specified && (typeof item.value == "string")){
  399. this.attr('encType', item.value);
  400. }
  401. }
  402. this.inherited(arguments);
  403. },
  404. onReset: function(/*Event?*/e){
  405. // summary:
  406. // Callback when user resets the form. This method is intended
  407. // to be over-ridden. When the `reset` method is called
  408. // programmatically, the return value from `onReset` is used
  409. // to compute whether or not resetting should proceed
  410. return true; // Boolean
  411. },
  412. _onReset: function(e){
  413. // create fake event so we can know if preventDefault() is called
  414. var faux = {
  415. returnValue: true, // the IE way
  416. preventDefault: function(){ // not IE
  417. this.returnValue = false;
  418. },
  419. stopPropagation: function(){}, currentTarget: e.currentTarget, target: e.target
  420. };
  421. // if return value is not exactly false, and haven't called preventDefault(), then reset
  422. if(!(this.onReset(faux) === false) && faux.returnValue){
  423. this.reset();
  424. }
  425. dojo.stopEvent(e);
  426. return false;
  427. },
  428. _onSubmit: function(e){
  429. var fp = dijit.form.Form.prototype;
  430. // TODO: remove ths if statement beginning with 2.0
  431. if(this.execute != fp.execute || this.onExecute != fp.onExecute){
  432. dojo.deprecated("dijit.form.Form:execute()/onExecute() are deprecated. Use onSubmit() instead.", "", "2.0");
  433. this.onExecute();
  434. this.execute(this.getValues());
  435. }
  436. if(this.onSubmit(e) === false){ // only exactly false stops submit
  437. dojo.stopEvent(e);
  438. }
  439. },
  440. onSubmit: function(/*Event?*/e){
  441. // summary:
  442. // Callback when user submits the form. This method is
  443. // intended to be over-ridden, but by default it checks and
  444. // returns the validity of form elements. When the `submit`
  445. // method is called programmatically, the return value from
  446. // `onSubmit` is used to compute whether or not submission
  447. // should proceed
  448. return this.isValid(); // Boolean
  449. },
  450. submit: function(){
  451. // summary:
  452. // programmatically submit form if and only if the `onSubmit` returns true
  453. if(!(this.onSubmit() === false)){
  454. this.containerNode.submit();
  455. }
  456. }
  457. }
  458. );
  459. }