My Blog List

Friday, August 13, 2010

Autosuggest Example Using ColdFusion

So I got a chance to try this one out. I’ve written auto-suggest boxes in GXT and plain old JS and while its a little harder to get the same level of control and customization ,CF8’s autosuggest is pretty neat for most simple use-cases and for those wanting to whip up an app quickly. I know CF9 has much better handles to the autosuggest object (looking forward to that here's a preview), but till then ..here goes!

Here’s an example of a user looking up an employee by last name.

In this example I use 2 files: one (loopkup.cfm) to display the autosuggest box and the other (a CFC ,MCRequestServer.cfc) to fetch data for it.

First the auto-suggest box code:

  1: <!doctype html public "-//w3c//dtd html 4.01 transitional//en" "http://www.w3.org/tr/html4/loose.dtd">
  2: <html>
  3: 	<head>
  4: 		<meta http-equiv="content-type" content="text/html; charset=iso-8859-1">
  5: 		<title>Search Trainees</title>
  6: 		<link href="AppStyles.css" rel="stylesheet" type="text/css" />
  7: 		<script>
  8: 			/**
  9: 			 * This method handles all bind errors. I'm just alerting.
 10: 			 * args:
 11: 			 * 	i. HTTPStatusCode- Error Code.number
 12: 			 *	ii.message- any message you want to display
 13: 			 */
 14: 			function handleBindError(HTTPStatusCode,message)
 15: 			{
 16: 				alert("Error Code: "+HTTPStatusCode+":\n"+message);
 17: 			}
 18: 			/**
 19: 			 * This method attaches an itemSelect event handler to handle
 20: 			 *  selections (using both mouse and keys).Called by ajaxOnLoad below.
 21: 			 */
 22: 			var init = function()
 23: 			{
 24: 				/** get the autosuggest object */
 25: 				autosuggestobj = ColdFusion.objectCache['TraineeSearchBox'];
 26: 				/** attache an itemselect event handler*/
 27: 				autosuggestobj.itemSelectEvent.subscribe(handleSelection);
 28: 			}
 29: 			/**
 30: 			 * This method handles all selections from the autosuggest list
 31: 			 * arg-
 32: 			 *	i. event-the event name. in this case "itemSelect"
 33: 			 *	ii.args- an array with the follwing data
 34: 			 *			 args[0]- Name/ID of the autocomplete component
 35: 			 *			 args[1]- The actual <LI> element that the user selected
 36: 			 *			 args[2]- The value selected.			 			 
 37: 			 */
 38: 			var handleSelection = function(event,args)
 39: 			{
 40: 				/** Display selection, you could do ur custom code here */
 41: 				document.getElementById("SearchTraineeSpan").innerHTML=args[2];
 42: 			}
 43: 		</script>
 44: 	</head>
 45: 	<body bgcolor="beige">		
 46: 		<cfform>
 47: 			<cfinput type="text" 
 48: 				name="TraineeSearchBox" 
 49: 				autosuggest="cfc:MCRequestServer.getEmployeesByLastName({cfautosuggestvalue})"
 50: 				onbinderror="handleBindError"
 51: 				showautosuggestloadingicon="true"
 52: 				autoSuggestMinLength="1"
 53: 				tooltip="Type employee last name"					
 54: 				onclick="this.value=''"	
 55: 				 />
 56: 				 <!--- Causes the specified JavaScript function to run when the page loads. --->
 57: 				 <cfset ajaxOnLoad("init")>
 58: 		<br>
 59: 		<span id="SearchTraineeSpan"></span>
 60: 		</cfform>
 61: 		
 62: 	</body>
 63: </html>
 64: 

Couple of points to note:


1.  The below line binds the textbox to a CFC called MCRequestServer and more specifically, a function called getEmployeesByLastName(). The argument cfautosuggestvalue is a CF keyword which represents the text the user entered in the textbox.

  1: autosuggest="cfc:MCRequestServer.getEmployeesByLastName({cfautosuggestvalue})"



2. The below line causes the specified JavaScript function init() to run when the page loads. This is to avoid timing issues.

  1: <cfset ajaxOnLoad("init")>

And here is the second file MCRequestServer.cfc. It just returns an array fo employee names.

  1: <cfcomponent displayname="MCRequestServer" hint="Handles all bind requests" output="false">
  2: 
  3: <cffunction name="getEmployeesByLastName" displayname="getEmployeesByLastName" access="remote" output="false" returntype="array">
  4: 		<cfargument name="suggestvalue" required="true" />
  5: 		<!--- The function must return suggestions as an array. --->
  6: 		<cfset var EmplNamesArray = ArrayNew(1) />
  7: 		<cftry>
  8: 			<!--- Get all unique last names that match the typed characters. --->
  9: 			<cfstoredproc datasource="DB" procedure="getMatchingEmployees" >			
 10: 			<cfprocparam type="in" dbvarname="@LastName" cfsqltype="CF_SQL_VARCHAR" value="#arguments.suggestvalue#"  >
 11: 				<!--- Out variable --->
 12: 				<cfprocresult name="rsEmplnames">
 13: 			</cfstoredproc>
 14: 			<!--- Convert the query to an array. --->
 15: 			<cfloop query="rsEmplnames">
 16: 				<cfset arrayAppend(EmplNamesArray, Lastname & ", " & FirstName ) />
 17: 			</cfloop>
 18: 			<cfreturn EmplNamesArray />
 19: 			<cfcatch type="any">
 20: 				<cfset arrayAppend(EmplNamesArray, "Error occurred") />
 21: 				<cfreturn EmplNamesArray />
 22: 			</cfcatch>
 23: 		</cftry>
 24: 	</cffunction>
 25: 
 26: 
 27: </cfcomponent>
 28: 

Friday, July 9, 2010

Simple JavaScript Programming Techniques

 

So I have been working on JavaScript for a little while now and gathered up a few points I thought I would share .This is my first blog post ever, so please forgive (but do let me know) any defects. I have tried to not get too detailed as I am afraid it will overload readers. Needless to say, I will be more than happy to discuss and revise the content to further improve it.

1. Declare commonly used objects globally to improve performance and to reduce your file size.

eg.: Declare the document object and window objects globally and use the references in your code.(I generally like to start my global variable names with a '$' sign,but that's just my naming convention. So at the top of my JS file I would say:

        var $doc = document;var $win=window;)

2. Wrap functions in logical namespaces:  For example if you have a bunch of utility functions , you could wrap them in a "Utility" namespace (as shown below) and call the functions using Namespace.function() syntax. 

[So in this example to call replaceAll(), you would say: Utilities.replaceAll("arg1","arg2","arg3") ] 

  1: var Utilities = {
  2:          stopEventPropagation : function(e) 
  3:                                  {    
  4:                                      if (!e)
  5:                                          e = window.event;                        
  6:                                      if (e.cancelBubble)
  7:                                          e.cancelBubble = true;
  8:                                      else
  9:                                          e.stopPropagation();
 10: 
 11:                                  },
 12:                  formatText      : function(text,color,isBold)
 13:                                  {
 14:                                      // implementation..            
 15:                                  },
 16:                  replaceAll        :function (Source,stringToFind,stringToReplace)
 17:                                  {
 18:                                     // implementation..
 19:                     
 20:                                  }
 21:                      }
This has the following advantages:

  • Avoid variable/function name clashes with (potentially) similarly-named variables in other JS libraries/files.
  •   Makes the code much more readable for other people.(Imagine if your HTML file references 5 JS files and you encounter a method call like validate(), you would have to go look in all to find the where validate() is implemented.I personally try to place one namespace per file,so Utilities namespace would be in Utilities.js, so anyone would know exactly where to look.)
  •   Provides an elegant way to create grouped constants,which improves readability.
  1:    var XHROReadystates {
  2:         UNINITIALIZED :0,    /** The initial value. */
  3:         OPEN :1,             /** The open() method has been successfully called, */
  4:         SENT :2,             /** The UA successfully completed the request, but no data hes yet been received. */
  5:         RECEIVING :3,        /** Immediately before receiving the message body (if any). All HTTP headers have been received. */
  6:         LOADED :4            /** The data transfer has been completed. */
  7:     }

3. Using User-Defined Attributes: HTML allows elements to have user-defined attributes. They can be accessed/modified in the same way as standard attributes.This is a powerful mechanism and can be used for both UI and functionality. A note on accessing attributes, IE allows access to the attributes using the (.) dot operator but this is not the W3C Standard and does not work on DOM compliant browsers like FireFox and Safari.Always use the W3C standards-compliant get/setAttribute methods to use attributes.


Here is an example of a user defined attribute and how it may be used.

eg:

  1: <input type="text" required="true" id="myTB".... 
Here required is not a standard HTML attribute (for simplicity's sake lets not consider HTML5's required attribute) 
Here is a way to use it to perform simple validation:
 
  1: isRequiredFieldsFilled	  : function ()/* Function to test if "required" input elements have been filled */
  2: 			   {
  3: 				   /* initialize result flag */
  4: 				   var isValid = true;
  5: 				   /* We are going to test the following element types */
  6: 				   var types = new Array("SELECT","INPUT","TEXTAREA");
  7: 				   /* For each type */
  8: 				   for(var i=0;i<types.length;i++)
  9: 				   {
 10: 					   /* Get all elements of that type */
 11: 					   var elems = document.getElementsByTagName(types[i]);
 12: 					   /* For each of these elements */
 13: 					   for(var j=0;j<elems.length;j++)
 14: 					   {
 15: 						   /* Ignore hidden fields */
 16: 						   if(elems[j].type=='hidden')
 17: 							   continue;
 18: 						   /* Reset color */
 19: 						   Utilities.setInvalidColor(elems[j], false);
 20: 						   /* Test if the required attribute exists,if so check if it is empty */
 21: 						   if(elems[j].getAttribute("required")!=null 
 22: 		    					&& (elems[j].getAttribute("required")==true 
 23: 			  	 				&& elems[j].value=="")
 24: 						   {
 25: 							   /* Mark as unfilled */
 26: 							   Utilities.setInvalidColor(elems[j], true);
 27: 							   /* Set result flag */
 28: 							   isValid=false;
 29: 						   }
 30: 					   }
 31: 				   }
 32: 				   /* After checking all elements,now check if anything has set the result flag */
 33: 				   if (!isValid) {
 34: 					   /* Alert the user */
 35: 					   alert('Please fill required fields!');
 36: 				   }
 37: 				   return isValid;
 38: 
 39: 			   }
4. Do Feature-detects, NOT browser detects

Doing this allows you to be be detached from the changes made by the browser vendor.Consider the example below, which tests if the browser is IE and if so executes someIEMethod else it executes a  DOM compliant someDOMMethod. Consider the future possibility that IE may decide to be DOM compliant and provide implementation for the DOM method someDOMMethod() and in time,deprecate someIEMethod (). This code would break then. 

  1: if (navigator.userAgent.indexO(”MSIE”) > —1)
  2:     someIEMethod();/* say, a non-DON coin1iant function */
  3: else 
  4:     someDOMMethod();/* say, a DOM compliant function */
  5: 

Here is a more real-world example which prevents event propagation depending on the browser.Consider evt to be the event object. This is not a good way to do it:

  1: if (navigator.userAgent.indexOf('MSIE') > —1)
  2:     evt.cancelBubble = true;/* lE specific */
  3: else 
  4:     evt.stopPropagation();/* DOM compliant */
  5: 
  6: 

A better way is:

  1: if (evt.cancelBubble)
  2:   evt.cancelBubble = true;
  3: else
  4:   evt. stopPropagation();
  5: 



[A note: For some situations a browser detect may still be required, most typically to apply browser specific CSS like opacity etc.]