Index: openacs-4/packages/q-forms/INSTALL.TXT
===================================================================
RCS file: /usr/local/cvsroot/openacs-4/packages/q-forms/INSTALL.TXT,v
diff -u
--- /dev/null	1 Jan 1970 00:00:00 -0000
+++ openacs-4/packages/q-forms/INSTALL.TXT	14 Nov 2014 18:27:52 -0000	1.1
@@ -0,0 +1,64 @@
+Q-FORMS Installation Instructions
+
+To install in an OpenACS system, follow the standard practices for
+downloading, uncompressing, and installing a local copy of this package.
+
+Detailed instructions follow.
+
+Download Q-Forms
+
+The easiest way is to go to https://github.com/tekbasse/q-forms
+and click on the ZIP button on the left side.
+
+# From shell, that would be something like:
+
+wget https://github.com/tekbasse/q-forms/archive/master.zip
+
+# if wget isn't available locally, try fetch or curl.
+
+Uncompress Q-Forms
+
+# Once downloaded, unzip the file with 'unzip'.
+# That might have to be grabbed from freebsd ports.
+
+unzip master.zip
+
+# Then rename the directory:
+
+mv q-forms-master q-forms
+
+# move the directory to the openacs/packages/ directory
+
+mv q-forms your-openacs-4/packages/.
+
+# change to the openacs-4/packages directory
+
+cd your-openacs-4/packages
+
+# changes permissions to match other packages
+# replace user_name and user_group with your local ones. Sometimes they're openacs:openacs, sometimes nobody:nogroup etc
+
+chown -R user_name:user_group q-forms
+
+# make sure the file permissions are consistent with the rest of openacs
+
+chmod -R 754 q-forms
+
+# Then start openacs if it isn't already.
+
+# Browse to your website's local url:
+
+/acs-admin/
+
+# click "Install Software"
+
+# click "Install from Local"
+
+# click on the box next to Q-Forms
+
+# click the "Install or update checked applications" button
+
+# follow instructions (usually requiring server restart) 
+
+# That's it!
+
Index: openacs-4/packages/q-forms/README.md
===================================================================
RCS file: /usr/local/cvsroot/openacs-4/packages/q-forms/README.md,v
diff -u
--- /dev/null	1 Jan 1970 00:00:00 -0000
+++ openacs-4/packages/q-forms/README.md	14 Nov 2014 18:27:52 -0000	1.1
@@ -0,0 +1,72 @@
+Q-Forms
+=======
+
+The lastest version of the code is available at the site:
+ http://github.com/dcpm/q-forms
+
+introduction
+------------
+
+Q-Forms provides procedures for building forms dynamically in OpenACS.
+It is an OpenACS service package that allows convenient building and
+interpreting of web-based forms via tcl in a web page.
+
+Q-Forms procedures parallel html's form tags with many automatic 
+defaults that remove the tedious nature of building forms 
+via html or an alternate form building context, such as OpenACS form
+builder, ad_form or acs-templating.
+
+license
+-------
+Copyright (c) 2013 Benjamin Brink
+po box 20, Marylhurst, OR 97036-0020 usa
+email: tekbasse@yahoo.com
+
+Q-Forms is open source and published under the GNU General Public License, 
+consistent with the OpenACS system: http://www.gnu.org/licenses/gpl.html
+A local copy is available at q-forms/www/doc/LICENSE.html
+
+    This program is free software: you can redistribute it and/or modify
+    it under the terms of the GNU General Public License as published by
+    the Free Software Foundation, either version 3 of the License, or
+    (at your option) any later version.
+
+    This program is distributed in the hope that it will be useful,
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+    GNU General Public License for more details.
+
+    You should have received a copy of the GNU General Public License
+    along with this program.  If not, see <http://www.gnu.org/licenses/>.
+
+features
+--------
+
+Low learning-curve. Uses tcl context. Procedures match tags. List friendly.
+
+Built-in API defaults. Takes less keystrokes to build a form than typing manually.
+
+Can build multiple forms concurently using Tcl file terminology.
+
+No limitations to building dynamic forms with specialized inputs.
+
+Form values are retrieved as an array named by the programmer.
+
+Form values are automatically quoted, a requirement of secure input handling.
+
+Optional automatic hash generation helps secure form transactions 
+and ignores multiple posts caused from mouse double-clicks and browsing page history.
+
+Multiple values of same key can be combined as a list (instead of producing
+a form post error).
+
+html can be inserted in any form at any point during the build.
+
+No UI javascript is used. Technologies with limited UI or cpu power can use it.
+
+Integrates with acs-templating features.
+
+
+installation
+------------
+See file q-forms/INSTALL.TXT for directions on installing.
Index: openacs-4/packages/q-forms/q-forms.info
===================================================================
RCS file: /usr/local/cvsroot/openacs-4/packages/q-forms/q-forms.info,v
diff -u
--- /dev/null	1 Jan 1970 00:00:00 -0000
+++ openacs-4/packages/q-forms/q-forms.info	14 Nov 2014 18:27:52 -0000	1.1
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<!-- Generated by the OpenACS Package Manager -->
+
+<package key="q-forms" url="http://openacs.org/repository/apm/packages/q-forms" type="apm_service">
+    <package-name>Q-Forms</package-name>
+    <pretty-plural>Q-Forms</pretty-plural>
+    <initial-install-p>f</initial-install-p>
+    <singleton-p>t</singleton-p>
+    <implements-subsite-p>f</implements-subsite-p>
+    <inherit-templates-p>f</inherit-templates-p>
+    
+    <version name="0.035" url="http://openacs.org/repository/download/apm/q-forms-0.035.apm">
+        <owner url="mailto:tekbasse@yahoo.com">Benjamin Brink</owner>
+        <owner url="http://openacs.org">OpenACS Community</owner>
+        <summary>Provides code for building forms dynamically in tcl</summary>
+        <description format="text/plain">Provides qf_* form bulding and interpreting procedures, especially designed for building forms dynamically.</description>
+        <maturity>1</maturity>
+
+        <provides url="q-forms" version="0.035"/>
+
+        <callbacks>
+        </callbacks>
+        <parameters>
+        <!-- No version parameters -->
+        </parameters>
+
+    </version>
+</package>
Index: openacs-4/packages/q-forms/tcl/form-helper-procs.tcl
===================================================================
RCS file: /usr/local/cvsroot/openacs-4/packages/q-forms/tcl/form-helper-procs.tcl,v
diff -u
--- /dev/null	1 Jan 1970 00:00:00 -0000
+++ openacs-4/packages/q-forms/tcl/form-helper-procs.tcl	14 Nov 2014 18:27:52 -0000	1.1
@@ -0,0 +1,888 @@
+ad_library {
+    
+    procedures for helping render form data or presentation for form data
+    @creation-date 15 May 2012
+    @cs-id $Id:
+
+    to vertically align textarea label, add to css: 
+
+    textarea { vertical-align: top; }
+    # replace top with middle or other options to adjust alignment.
+    # textarea's border is used for alignment, so alignment is affected by font-size and line-height ratio 
+    # in addition to borders and margins.
+
+}
+
+
+
+ad_proc -public qss_txt_to_tcl_list_of_lists {
+    textarea
+    linebreak
+    delimiter
+} {
+    Converts a csv/txt style table into a tcl list_of_lists
+} {
+    set lists_list [list]
+    set row_list [split $textarea $linebreak]
+    # clean the rows of any extra linefeeds etc
+
+    foreach row $row_list {
+        set columns_list [split $row $delimiter]
+        # rebuild columns_list to clean it of any remaining invisible characters
+        set column_set [list ]
+        foreach column $columns_list {
+            regsub -all -- {[\n\r\f\v]} $column {} col_version2
+            regsub -expanded -all -- {[[:cntrl:]]} $col_version2 {} col_version3
+            lappend column_set $col_version3
+        }
+        set columns_list $column_set
+
+        set columns [llength $columns_list]
+#        ns_log Notice "qss_txt_to_tcl_list_of_lists: col len $columns, columns_list ${columns_list}"
+        if { $columns > 0 } {
+            lappend lists_list $columns_list
+        }
+    }
+    return $lists_list
+}
+
+ad_proc -public qss_txt_table_stats { 
+    textarea 
+} {
+    determines the best guess linebreak_char delimiter rows_count columns_count of a cvs/txt table
+    and returns these values a list
+} {
+    # scan to guess # of rows and cols
+
+    set linebreaks_list [list \n \r \f \v ]
+    set array table_arr
+
+    # determine row delimiter
+    set lineC 0
+    set max_rows 0
+    foreach linebreak $linebreaks_list {
+        set row_set [split $textarea $linebreak]
+        set linesC [llength $row_set]
+#ns_log Notice "qss_txt_table_stats: rows $linesC for linebreak_idx/lineC $lineC"
+        if { $linesC > $max_rows } {
+            set linebreak_idx $lineC
+            set max_rows $linesC
+            set linebreak_char $linebreak
+#            set rows_set $row_set 
+            # remove any remaining delimiters
+            set rows_set [list ]
+            foreach line $row_set {
+                regsub -all -- {[\n\r\f\v]} $line {} line2
+#                regsub -expanded -all -- {[[:cntrl:]]} $line2 {} line3
+                lappend rows_set $line2
+            }
+            set rows_count [llength $rows_set]
+        }
+        incr lineC
+    }
+#ns_log Notice "qss_txt_table_stats: rows_set: $rows_set"
+
+
+    set rowsC [llength $rows_set]
+    # determine column delimiter
+    set delimiters_list [list "\t" " " "," "|" "!"]
+
+    set delimC 0
+    set columns_arr(0-avg) 0.
+    foreach delimiter $delimiters_list {
+        array unset columns_arr
+        set max_cols 0
+        set many_cols_sum 0.
+        set few_cols_sum 0.
+        set many_cols_rows 0
+        set few_cols_rows 0
+        set colC_list [list]
+        set cols_sum 0.
+        # get average number of rows and avg variance for each delim
+        # When avg cols/row is > 2 ignore rows with (0 or) one column when calculating avg variance
+        #   Do this by counting these rows, averaging them, then averaging to the other set if <= 2.
+
+        # if there is a significant median value, use it instead.
+        foreach row $rows_set {
+            set col_set [split $row $delimiter]
+            set colsC [llength $col_set]
+# ns_log Notice "qss_txt_table_stats: delimiter $delimiter colsC $colsC"
+            if { [info exists columns_arr(${colsC})] } {
+                set columns_arr(${colsC}) [expr { $columns_arr(${colsC}) + 1 } ]
+            } else {
+                set columns_arr(${colsC}) 1
+            }
+            set cols_sum [expr { $cols_sum + $colsC } ]
+            lappend colC_list $colsC
+            if { $colsC > 2 } {
+                set many_cols_sum [expr { $many_cols_sum + $colsC } ]
+                incr many_cols_rows
+            } else {
+                set few_cols_sum [expr { $few_cols_sum + $colsC } ]
+                incr few_cols_rows
+            }
+        }
+        if { $few_cols_rows > 0 } {
+            set few_cols_avg [expr { $few_cols_sum / $few_cols_rows } ]
+        } else {
+            set few_cols_avg 0
+        }
+        if { $many_cols_rows > 0 } {
+            set many_cols_avg [expr { $many_cols_sum / $many_cols_rows } ]
+        } else {
+            set many_cols_avg 0
+        }
+        set cols_avg [expr { $cols_sum / $max_rows } ]
+        if { $cols_avg > 2 } {
+            set cols_avg $many_cols_avg
+#            set rowsC $many_cols_rows
+        } else {
+            set cols_avg $few_cols_avg
+#            set rowsC $few_cols_rows
+        }
+        # determine variance
+        set sum2 0.
+        set rowCt 0
+        foreach colCt $colC_list {
+            if { $colCt > 0 } {
+                set sum2 [expr { $sum2 + pow( $colCt - $cols_avg , 2. ) } ]
+                incr rowCt
+            }
+        }
+        if { $rowCt > 1 } {
+            set variance [expr { $sum2 / ( $rowCt - 1. ) } ]
+        } else {
+            set variance 99999.
+        }
+
+        #what is median of columns?
+        set median 0
+        foreach { column count } [array get columns_arr] {
+            if { $count > $median } {
+                set median_old $median
+                set median $column
+            }
+        }
+
+        # column count expands (not contracts) if delimeter is shared in data
+        if { $median_old == 0 } {
+            set median_old $median
+        }
+        set median_diff [expr { $median - $median_old } ]
+
+        set median_pct_diff [expr { $median_diff / $median_old } ]
+
+            
+        if { $median_pct_diff > 1.1 && $median_pct_diff < 2.0 } {
+            set median_old2 $median_old
+            set median_old $median
+            set median $median_old2
+        }
+# if row and column delimiter are same (such as space), manually step through table collecting info?
+# determine likely matrix size and variations, then step through to sqrt(max_rows) looking for data type patterns.
+# NOT IMPLEMENTED
+
+        # For best guess, the average converges toward the median..
+        set median_diff_abs [expr { abs( $median_diff ) } ]
+        if { $variance < $median_diff_abs && $cols_avg < $median } {
+            set bguess $cols_avg
+            set bguessD $variance
+        } else {
+            set bguess $median
+            set bguessD $median_diff_abs
+        }
+
+        set table_arr(${delimC}-avg) $cols_avg
+        set table_arr(${delimC}-variance) $variance
+        set table_arr(${delimC}-median) $median
+        set table_arr(${delimC}-medianD) $median_old
+        set table_arr(${delimC}-bguess) $bguess
+        set table_arr(${delimC}-bguessD) $bguessD
+        set table_arr(${delimC}-rows) $rowCt
+        set table_arr(${delimC}-delim) $delimiter
+        set table_arr(${delimC}-linebrk) $linebreak_char
+        ns_log Notice "qss_txt_table_stats: delimC '$delimC' delimiter '${delimiter}' cols_avg $cols_avg variance $variance median $median median_old $median_old bguess $bguess bguessD $bguessD rowCt $rowCt"
+        incr delimC
+    }
+
+    # convert the table into a sortable list of lists
+    # First choice:
+    # select bguess >= 2 with smallest variance.
+    set bguess_lists [list ]
+    for { set i 0 } { $i < $delimC } { incr i } {
+        if { $table_arr(${i}-bguess) >= 2. } {
+            set bg_list [list $i $table_arr(${i}-avg) $table_arr(${i}-variance) ]
+            ns_log Notice "qss_txt_table_stats.215: i $i table_arr(${i}-bguess) $table_arr(${i}-bguess) table_arr(${i}-bguessD) $table_arr(${i}-bguessD) bg_list ${bg_list}"
+            lappend bguess_lists $bg_list
+        }
+    }
+    # sort by smallest variance
+    set sorted_bg_lists [lsort -increasing -real -index 2 $bguess_lists]
+    ns_log Notice "qss_txt_table_stats.220: sorted_bg_lists ${sorted_bg_lists}"
+    set i [lindex [lindex $sorted_bg_lists 0] 0]
+    set bguess $table_arr(${i}-bguess)
+    set bguessD $table_arr(${i}-bguessD)
+    set rows_count $table_arr(${i}-rows)
+    set delimiter $table_arr(${i}-delim)
+
+    # If there are no bguesses over 2, then use this process:
+    if { [llength $bguess_lists] == 0 } {
+        # This following techinque is not dynamic enough to handle all conditions.
+        set bguessD $table_arr(0-bguessD)
+        set bguess $table_arr(0-bguess)
+        set rows_count $table_arr(0-rows)
+        set delimiter $table_arr(0-delim)
+        # bguessD is absolute value of bguess from variance
+        for { set i 0 } { $i < $delimC } { incr i } {
+            if { ( $table_arr(${i}-bguessD) <= $bguessD ) && $table_arr(${i}-bguess) > 1 } {
+                if { ( $bguess > 1 && $table_arr(${i}-bguess) < $bguess ) || $bguess < 2 } {
+                    set bguess $table_arr(${i}-bguess)
+                    set bguessD $table_arr(${i}-bguessD)
+                    set rows_count $table_arr(${i}-rows)
+                    set delimiter $table_arr(${i}-delim)
+                }
+            }
+        }
+    }
+    ns_log Notice "qss_txt_table_stats linebreak '${linebreak_char}' delim '${delimiter}' rows '${rows_count}' columns '${bguess}'"
+    set return_list [list $linebreak_char $delimiter $rows_count $bguess]
+#    ns_log Notice "qss_txt_table_stats: return_list $return_list"
+    return $return_list
+}
+
+
+
+ad_proc -public qss_list_of_lists_to_html_table { 
+    table_list_of_lists 
+    {table_attribute_list ""}
+    {td_attribute_lists ""}
+} {
+    Converts a tcl list_of_lists to an html table, returns table as text/html
+    table_attribute_list can be a list of attribute pairs to pass to the TABLE tag: attribute1 value1 attribute2 value2..
+    The td_attribute_lists adds attributes to TD tags at the same position as table_list_of_lists 
+    the list is represented {row1 {cell1} {cell2} {cell3} .. {cell x} } {row2 {cell1}...}
+    Note that attribute - value pairs in td_attribute_lists can be added uniquely to each TD tag.
+} {
+    set table_html "<table"
+    foreach {attribute value} $table_attribute_list {
+        regsub -all -- {\"} $value {\"} value
+        append table_html " $attribute=\"$value\""
+    }
+    append table_html ">\n"
+    set row_i 0
+    set column_i 0
+    #setup repeat pattern for formatting rows, if last formatting row is not blank
+    set repeat_last_row_p 0
+    if { [llength [lindex $td_attribute_lists end] ] > 0 } {
+        # this feature only comes into play if td_attrubte_lists is not as long as table_list_of_lists
+        set repeat_last_row_p 1
+        set repeat_row [expr { [llength $td_attribute_lists] - 1 } ]
+    }
+    foreach row_list $table_list_of_lists {
+        append table_html "<tr>"
+        foreach column $row_list {
+            append table_html "<td"
+            if { $repeat_last_row_p && $row_i > $repeat_row } {
+                set attribute_value_list [lindex [lindex $td_attribute_lists $repeat_row] $column_i]
+
+            } else {
+                set attribute_value_list [lindex [lindex $td_attribute_lists $row_i] $column_i]
+            }
+            foreach {attribute value} $attribute_value_list {
+                regsub -all -- {\"} $value {\"} value
+                append table_html " $attribute=\"$value\""
+            }
+            append table_html ">${column}</td>"
+            incr column_i
+        }
+        append table_html "</tr>\n"
+        incr row_i
+        set column_i 0
+    }
+    append table_html "</table>\n"
+    return $table_html
+}
+
+ad_proc -public qss_lists_to_text { 
+    table_list_of_lists
+    {row_delimiter "\n"}
+    {column_delimiter ","}
+ } {
+    Converts a tcl list_of_lists to content suitable to be used with a textarea tag.
+} {
+    foreach row_list $table_list_of_lists {
+        set col_delim ""
+        foreach column $row_list {
+            append table_html $col_delim
+            append table_html $column
+            set col_delim $column_delimiter
+        }
+        append table_html $row_delimiter
+    }
+    return $table_html
+}
+
+
+ad_proc -public qss_form_table_to_table_lists {
+    table_array_name
+} {
+    returns a table represented as a list of lists from a table represtented as an array.
+} {
+    upvar $table_array_name table_array
+
+    # get array indices as a sorted list
+    set array_idx_list [lsort [array names table_array]]
+}
+
+ad_proc -public qss_table_lists_normalize {
+    table_lists
+    new_min
+    new_max
+} {
+    given a list_of_lists table, returns the table normalized to max/min parameters
+} {
+    set element_1 [lindex [lindex $table_lists 0] 0]
+    set element_max $element_1
+    set element_min $element_1
+    foreach row_list $table_lists {
+        foreach element $row_list {
+            if { $element > $element_max } {
+                set element_max $element
+            }
+            if { $element < $element_min } {
+                set element_min $element
+            }
+        }
+    }
+    set delta_old [expr { $element_max - $element_min } ]
+    set delta_new [expr { $new_max - $new_min } ]
+
+    foreach row_list $table_lists {
+        foreach element $row_list {
+            # transform value to new range
+            set element_new [expr { ( $element - $element_min ) * $delta_new  / $delta_old + $new_min } ]
+        }
+    }
+}
+
+
+
+ad_proc -public qss_progression_x1x2xc {
+    min
+    max
+    count
+} {
+    given:  x1 start, x2 end, and xc (the number of points)
+
+    returns a list of xc elements starting from x1 to x2
+} {
+
+    set dx [expr { $x2 - $x1 } ]
+    set x_list [list ]
+    if { $xc != 0 && $dx != 0 } {
+        if { $x1 > $x2 } { 
+            set step [expr { $dx / $xc } ]
+            for { set x $x1 } { $x >= $x2 } { set x [expr { $x + $step } ] } {
+                lappend x_list $x
+            }
+        } else {
+            set step [expr { $dx / $xc } ]
+            for { set x $x1 } { $x <= $x2 } { set x [expr { $x + $step } ] } {
+                lappend x_list $x
+            }
+        }
+    }
+    return $x_list
+}
+
+ad_proc -public qf_is_natural_number {
+    value
+} {
+    answers question: is value a natural counting number (non-negative integer)?
+    returns 0 or 1
+} {
+    set is_natural [regexp {^(0*)(([1-9][0-9]*|0))$} $value match zeros value]
+    return $is_natural
+}
+
+ad_proc -public qf_remove_from_list {
+    value value_list
+} {
+    removes multiple of a specific value from a list
+    returns list without the passed value
+} {
+
+    set value_indexes [lsearch -all -exact $value_list $value]
+    while { [llength $value_indexes] > 0 } {
+        set next_index [lindex $value_indexes 0]
+        set value_list [lreplace $value_list $next_index $next_index]
+        set value_indexes [lsearch -all -exact $value_list $value]
+    }
+    return $value_list
+}
+
+ad_proc -public qf_get_contents_from_tag {
+    start_tag
+    end_tag
+    page
+    {start_index 0}
+} {
+    Returns content of an html/xml or other bracketing tag that is uniquely identified within a page fragment or string.
+    helps pan out the golden nuggets of data from the waste text when given some garbage with input for example
+} {
+    set tag_contents ""
+    set start_col [string first $start_tag $page $start_index]
+    set end_col [string first $end_tag $page $start_col]
+    if { $end_col > $start_col && $start_col > -1 } {
+        set tag_contents [string trim [string range $page [expr { $start_col + [string length $start_tag] } ] [expr { $end_col -1 } ]]]
+    } else {
+        set starts_with "${start_tag}.*"
+        set ends_with "${end_tag}.*"
+        if { [regexp -- ${starts_with} $page tag_contents ]} {
+            if { [regexp -- ${ends_with} $tag_contents tail_piece] } {
+                set tag_contents [string range $tag_contents 0 [expr { [string length $tag_contents] - [string length $tail_piece] - 1 } ] ]
+            } else {
+                ns_log Notice "Warning no contents for tag $start_tag"
+                set tag_contents ""
+            }
+        }
+    }
+    return $tag_contents
+}
+
+ad_proc -public qf_get_contents_from_tags_list {
+    start_tag
+    end_tag
+    page
+} {
+    Returns content (as a list) of all occurances of an html/xml or other bracketing tag that is somewhat uniquely identified within a page fragment or string.
+    helps pan out the golden nuggets of data from the waste text when given some garbage with input for example
+} {
+    set start_index 0
+    set tag_contents_list [list]
+    set start_tag_len [string length $start_tag]
+    set start_col [string first $start_tag $page 0]
+    set end_col [string first $end_tag $page $start_col]
+    set tag_contents [string range $page [expr { $start_col + $start_tag_len } ] [expr { $end_col - 1 } ]]
+    while { $start_col != -1 && $end_col != -1 } {
+#        lappend tag_contents_list [string trim $tag_contents]
+        lappend tag_contents_list $tag_contents
+
+        set start_index [expr { $end_col + 1 }]
+        set start_col [string first $start_tag $page $start_index]
+        set end_col [string first $end_tag $page $start_col]
+        set tag_contents [string range $page [expr { $start_col + $start_tag_len } ] [expr { $end_col - 1 } ]]
+    }
+    return $tag_contents_list
+}
+
+ad_proc -public qf_remove_tag_contents {
+    start_tag
+    end_tag
+    page
+} {
+    Returns everything but the content between start_tag and end_tag (as a list) 
+    of all occurances on either side of an html/xml or other bracketing tag 
+    that is somewhat uniquely identified within a page fragment or string.
+    This is handy to remove script tags and < ! - - web comments - - > etc
+    helps pan out the golden nuggets of data from the waste text when given some garbage with input for example
+} {
+    # start and end refer to the tags and their contents that are to be removed
+ns_log Notice "qf_remove_tag_contents: start_tag $start_tag end_tag $end_tag page $page"
+    set start_index 0
+    set tag_contents_list [list]
+    set start_tag_len [string length $start_tag]
+    set end_tag_len [string length $end_tag]
+    set start_col [string first $start_tag $page $start_index]
+    set end_col [string first $end_tag $page $start_col]
+    # set tag_contents [string range $page 0 [expr { $start_col - 1 } ] ]
+    while { $start_col != -1 && $end_col != -1 } {
+        set tag_contents [string range $page $start_index [expr { $start_col - 1 } ] ]
+#        lappend tag_contents_list [string trim $tag_contents]
+        lappend tag_contents_list $tag_contents
+ns_log Notice "qf_remove_tag_contents(465): tag_contents '$tag_contents'"
+        # start index is where we begin the next clip        
+        set start_index [expr { $end_col + $end_tag_len } ]
+        set start_col [string first $start_tag $page $start_index]
+        set end_col [string first $end_tag $page $start_col]
+        # and the new clip ends at the start of the next tag
+    }
+    # append any trailing portion
+    lappend tag_contents_list [string range $page $start_index end]
+#    set remaining_contents \[join $tag_contents_list " "\]
+    return $tag_contents_list
+}
+
+
+ad_proc -public qf_convert_html_list_to_tcl_list {
+    html_list
+} {
+    converts a string containing an html list to a tcl list
+    Assumes there are no embedded sublists, and strips remaining html
+} {
+    set draft_list $html_list
+
+    #we standardize the start and end of the list, so we know where to clip
+
+    if { [regsub -nocase -- {<[ou][l][^\>]*>} $draft_list "<ol>" draft_list ] ne 1 } {
+        # no ol/ul tag, lets create the list container anyway
+        set draft_list "<ol> ${draft_list}"
+
+    } else {
+        # ol/ul tag exists, trim garbage before list
+        set draft_list [string range $draft_list [string first "<ol>" $draft_list ] end ]
+    }
+
+    if { [regsub -nocase -- {</li>[ ]*</[ou]l[^\>]*>} $draft_list "</li></ol>" draft_list ] ne 1 } {
+        # end list tag may not exist or is not in standard form
+        if { [regsub -nocase -- {</[ou]l[^\>]*>} $draft_list "</li></ol>" draft_list ] ne 1 } {
+            # assume for now that there was no end li tag before the list (bad html)
+        } else {
+            # no ol/ul list tag, assume no end </li> either?
+            append draft_list "</li></ol>"
+        }
+    }
+
+    # end ol tag exists, trim garbage after list
+    # choosing the last end list tag in case there is a list in one of the lists
+    set draft_list [string range $draft_list 0 [expr { [string last "</ol>" $draft_list ] + 4} ] ]
+
+    # simplify li tags, with a common delimiter
+    regsub -nocase -all -- {<li[^\>]*>} $draft_list {|} draft_list
+    # remove other html tags
+
+    set draft_list [qf_webify $draft_list]
+
+    # remove excess spaces
+    regsub -all -- {[ ]+} $draft_list " " draft_list
+    set draft_list [string trim $draft_list]
+
+    # remove excess commas and format poorly placed ones
+    regsub -all -- {[ ],} $draft_list "," draft_list
+
+    regsub -all -- {[,]+} $draft_list "," draft_list
+
+    # put colons in good form
+    regsub -all -- {[ ]:} $draft_list ":" draft_list
+
+    regsub -all -- {:,} $draft_list ":" draft_list
+    # remove : in cases where first column is blank, ie li should not start with a colon
+
+    regsub -all -- {\|:} $draft_list {|} draft_list
+
+    set tcl_list [split $draft_list {|}]
+    # first lindex will be blank, so remove it
+    set tcl_list [lrange $tcl_list 1 end]
+#ns_log Notice "qf_convert_html_list_to_tcl_list: tcl_list $tcl_list"
+    return $tcl_list
+}
+
+ad_proc -public qf_convert_html_table_to_list {
+    html_string
+    {list_style ul}
+} {
+    converts a string containing an html table to an html list
+    assumes first column is a heading (with no rows as headings), and remaining columns are values
+    defaults to li list style, should return list in good html form even if table is not quite that way
+} {
+
+    if { [regsub -nocase -- {<table[^\>]*>} $html_string "<${list_style}>" draft_list ] ne 1 } {
+        # no table tag, lets create the list container anyway
+        set draft_list "<${list_style}> ${html_string}"
+    } else {
+        # table tag exists, trim garbage before list
+        set draft_list [string range $draft_list [string first "<${list_style}>" $draft_list ] end ]
+    }
+
+    if { [regsub -nocase -- {</tr>[ ]*</table[^\>]*>} $draft_list "</li></${list_style}>" draft_list ] ne 1 } {
+        # end table tag may not exist or is not in standard form
+        if { [regsub -nocase -- {</table[^\>]*>} $draft_list "</li></${list_style}>" draft_list ] ne 1 } {
+            # assume for now that there was no end tr tag before the table (bad html)
+        } else {
+            # no table tag, assume no end </tr> either?
+            append draft_list "</li></${list_style}>"
+        }
+    }
+
+    # end table tag exists, trim garbage after list
+    # choosing the last end list tag in case there is a list in one of the table cells
+    set draft_list [string range $draft_list 0 [expr { [string last "</${list_style}>" $draft_list ] + 4} ] ]
+
+    # simplify tr and td tags, but do not replace yet, because we want to use them for markers when replacing td tags
+    regsub -nocase -all -- {<tr[^\>]+>} $draft_list "<tr>" draft_list
+    regsub -nocase -all -- {</tr[^\>]+>} $draft_list "</tr>" draft_list
+    regsub -nocase -all -- {<td[^\>]+>} $draft_list "<td>" draft_list
+    regsub -nocase -all -- {</td[^\>]+>} $draft_list "</td>" draft_list
+
+    # clean out other content junk tags
+    regsub -nocase -all -- {<[^luot\/\>][^\>]*>} $draft_list "" draft_list
+    regsub -nocase -all -- {</[^luot\>][^\>]*>} $draft_list "" draft_list
+
+    set counterA 0
+    while { [string match -nocase "*<tr>*" $draft_list ] } {
+
+       if { [incr counterA ] > 300 } {
+           ns_log Error "convert_html_table_to_list, ref: counterA detected possible infinite loop."
+           doc_adp_abort
+        }
+        # get row range
+        set start_tr [string first "<tr>" $draft_list ]
+        set end_tr [string first "</tr>" $draft_list ]
+
+        # make sure that the tr end tag matches the current tr tag
+        if { $end_tr == -1 } {
+            set next_start_tr [string first "<tr>" $draft_list [expr { $start_tr + 4 } ] ]
+        } else {
+            set next_start_tr [string first "<tr>" $draft_list $end_tr ]
+        }
+
+        regsub -- {<tr>} $draft_list "<li>" draft_list
+
+        if { $end_tr < $next_start_tr && $end_tr > -1 } {
+            regsub -- {</tr>} $draft_list "     " draft_list
+            # common sense says we replace </tr> with </li>, but then there may be cases missing a </tr>
+            # and if so, we would have to insert a </li> which would mess up the position values for use
+            # later on. Instead, at the end, we convert <li> to </li><li> and take care of the special 1st and last cases
+        } 
+
+        # we are assuming any td/tr tags occur within the table, since table has been trimmed above
+        set start_td [string first "<td>" $draft_list ]
+        set end_td [string first "</td>" $draft_list ]
+        set next_start_td [string first "<td>" $draft_list [expr { $start_td + 3 } ] ]
+
+        if { $next_start_td == -1 || ( $next_start_td > $next_start_tr && $next_start_tr > -1 )} {
+            # no more td tags for this row.. only one column in this table
+
+        } else {
+            # setup first special case of first column
+            # replacing with strings of same length to keep references current throughout loops
+            set draft_list [string replace $draft_list $start_td [expr { $start_td + 3 } ] "    " ]
+
+            if { $end_td < $next_start_tr && $end_td > -1 } {
+                # there is an end td tag for this td cell, replace with :
+                set draft_list [string replace $draft_list $end_td [expr { $end_td + 4 } ] ":    " ]
+
+            } else {
+                # insert special case, just prior to new td tag
+                set draft_list "[string range ${draft_list} 0 [expr { ${next_start_td} - 1 } ] ]: [string range ${draft_list} ${next_start_td} end ]"
+                if { $next_start_tr > 0 } {
+                    incr next_start_tr 2
+                }
+            }
+        }
+
+        # process remaining td cells in row, separating cells by comma
+        set column_separator "    "
+        if { $next_start_tr == -1 } {
+            set end_of_row [string length $draft_list ]
+        } else {
+            set end_of_row [expr { $next_start_tr + 3 } ]
+        }
+
+        set columns_to_convert [string last "<td>" [string range $draft_list 0 $end_of_row ] ]
+        set counterB 0
+        while { $columns_to_convert > -1 } {
+
+            if { [incr counterB ] > 200 } {
+                ns_log Error "convert_html_table_to_list, ref: counterB detected possible infinite loop."
+                doc_adp_abort
+            }
+
+            set start_td [string first "<td>" $draft_list ]
+            set end_td [string first "</td>" $draft_list ]
+            set next_start_td [string first "<td>" $draft_list [expr { $start_td + 3 } ] ]
+
+            if { $next_start_td == -1 } {
+                # no more td tags for all rows.. still need to process this one.
+                set columns_to_convert -1
+                set draft_list [string replace $draft_list $start_td [expr { $start_td + 3 } ] $column_separator ]
+
+            } elseif { ( $next_start_td > $next_start_tr && $next_start_tr > -1 ) } {
+                # no more td tags for this row..
+                set columns_to_convert -1
+
+            } else {
+                # add a comma before the value, if this is not the first value
+                set draft_list [string replace $draft_list $start_td [expr { $start_td + 3 } ] $column_separator ]
+
+            }
+
+            if { $end_td > -1 && ( $end_td < $next_start_td || $next_start_td == -1 ) } {
+                # there is an end td tag for this td cell, remove it
+                regsub -- {</td>} $draft_list "" draft_list
+            }
+
+            set column_separator ",    "
+            # next column
+        }
+
+
+        # next row
+    }
+
+    # clean up list, add </li>
+    regsub -all -- "<li>" $draft_list "</li><li>" draft_list
+    # change back first case
+    regsub -- "</li><li>" $draft_list "<li>" draft_list
+    # a /li tag is already included with the  list container end tag
+
+    # remove excess spaces
+    regsub -all -- {[ ]+} $draft_list " " draft_list
+
+    # remove excess commas and format poorly placed ones
+    regsub -all -- {[ ],} $draft_list "," draft_list
+    regsub -all -- {[,]+} $draft_list "," draft_list
+
+    # put colons in good form
+    regsub -all -- {[ ]:} $draft_list ":" draft_list
+    regsub -all -- {:,} $draft_list ":" draft_list
+    # remove : in cases where first column is blank, ie li should not start with a colon
+    regsub -all -- {<li>:} $draft_list "<li>" draft_list
+
+   return $draft_list
+}
+ad_proc -public qf_remove_html {
+    description
+    {delimiter ":"}
+} {
+
+    remvoves html and converts common delimiters to something that works in html tag attributes, default delimiter is ':'
+
+} {
+    # remove tags
+    regsub -all -- "<\[^\>\]*>" $description " " description
+
+    # convert fancy delimiter to one that complies with meta tag values
+    regsub -all -- "&\#187;" $description $delimiter description
+
+    # convert bracketed items as separate (delimited) items
+    regsub -all -- {\]} $description "" description
+    regsub -all -- {\[} $description $delimiter description
+
+    # convert any dangling lt/gt signs to delimiters
+    regsub -all -- ">" $description $delimiter description
+    regsub -all -- "<" $description $delimiter description
+
+    # remove characters that
+    # can munge some meta tag values or how they are interpreted
+    regsub -all -- {\'} $description {} description
+    regsub -all -- {\"} $description {} description
+
+    # remove html entities, such as &trade; &copy; etc.
+    regsub -all -nocase -- {&[a-z]+;} $description {} description
+
+    # filter extra spaces
+    regsub -all -- {\s+} $description { } description
+    set description "[string trim $description]"
+
+return $description
+}
+
+ad_proc -public qf_remove_attributes_from_html {
+    description
+} {
+
+    remvoves attributes from html
+
+} {
+    # filter extra spaces
+    regsub -all -- {\s+} $description { } description
+    set description "[string trim $description]"
+
+    # remove attributes from tags
+    regsub -all -nocase -- {(<[/]?[a-z]*)[^\>]*(\>)} $description {\1\2} description
+    
+return $description
+}
+
+ad_proc -public qf_abbreviate {
+    phrase
+    {max_length {}}
+} {
+    abbreviates a pretty title or phrase to first word, or to max_length characters if max_length is a number > 0
+} {
+    set suffix ".."
+    set suffix_len [string length $suffix]
+
+    if { [ad_var_type_check_number_p $max_length] && $max_length > 0 } {
+        set phrase_len_limit [expr { $max_length - $suffix_len } ]
+        regsub -all -- { / } $phrase {/} phrase
+        if { [string length $phrase] > $max_length } {
+            set cat_end [expr { [string last " " [string range $phrase 0 $max_length] ] - 1 } ]
+            if { $cat_end < 0 } {
+                set cat_end $phrase_len_limit
+            }
+            set phrase [string range $phrase 0 $cat_end ]
+        append phrase $suffix
+            regsub {[^a-zA-Z0-9]+\.\.} $phrase $suffix phrase
+        }
+        regsub -all -- { } $phrase {\&nbsp;} phrase
+        set abbrev_phrase $phrase
+
+    } else {
+        regsub -all { .*$} $phrase $suffix abbrev1
+        regsub -all {\-.*$} $abbrev1 $suffix abbrev
+        regsub -all {\,.*$} $abbrev $suffix abbrev1
+        set abbrev_phrase $abbrev1
+    }
+    return $abbrev_phrase
+}
+
+ad_proc -public qf_webify {
+ description
+} {
+   standardizes and sanitizes some junky data for use in web content
+} {
+    # need to remove code between script tags and hidden comments
+    set description_list [qf_remove_tag_contents {<script} {</script>} $description ]
+    set description_new ""
+    foreach desc_part $description_list {
+        append description_new $desc_part
+    }
+    set description_list [qf_remove_tag_contents {<!--} {-->} $description_new ]
+    set description_new ""
+    foreach desc_part $description_list {
+        append description_new $desc_part
+    }
+    regsub -all "<\[^\>\]*>" $description_new "" description1
+    regsub -all "<" $description1 ":" description
+    regsub -all ">" $description ":" description1
+    regsub -all -nocase {\"} $description1 {} description
+    regsub -all -nocase {\'} $description {} description1
+    regsub -all -nocase {&[a-z]+;} $description1 {} description
+    return $description
+}
+
+ad_proc -public qf_is_decimal {
+ value
+} {
+   checks if value is a decimal number that can be used in tcl decimal math. Returns 1 if true, otherwise 0.
+} {
+    # following regexp from acs-tcl/tcl/json-procs.tcl which references json.org, ietf.org, Thomas Maeder, Glue Software Engineering AG and Don Baccus
+    
+    # tokens consisting of a single character
+    #variable singleCharTokens { "{" "}" ":" "\\[" "\\]" "," }
+    #variable singleCharTokenRE "\[[join $singleCharTokens {}]\]"
+    
+    # quoted string tokens
+    #variable escapableREs { "[\\\"\\\\/bfnrt]" "u[[:xdigit:]]{4}" }
+    #variable escapedCharRE "\\\\(?:[join $escapableREs |])"
+    #variable unescapedCharRE {[^\\\"]}
+    #variable stringRE "\"(?:$escapedCharRE|$unescapedCharRE)*\""
+    
+    # (unquoted) words
+    #variable wordTokens { "true" "false" "null" }
+    #variable wordTokenRE [join $wordTokens "|"]
+    
+    # number tokens
+    # negative lookahead (?!0)[[:digit:]]+ might be more elegant, but
+    # would slow down tokenizing by a factor of up to 3!
+    set positiveRE {[1-9][[:digit:]]+[.]?|[[:digit:]][.]?}
+    set cardinalRE "-?(?:$positiveRE)?"
+    set fractionRE {[.][[:digit:]]+}
+    set exponentialRE {[eE][+-]?[[:digit:]]+}
+    set numberRE "^${cardinalRE}(?:$fractionRE)?(?:$exponentialRE)?$"
+    set type_decimal_p [regexp -- $numberRE $value]
+    return $type_decimal_p
+}
+
Index: openacs-4/packages/q-forms/tcl/form-procs.tcl
===================================================================
RCS file: /usr/local/cvsroot/openacs-4/packages/q-forms/tcl/form-procs.tcl,v
diff -u
--- /dev/null	1 Jan 1970 00:00:00 -0000
+++ openacs-4/packages/q-forms/tcl/form-procs.tcl	14 Nov 2014 18:27:52 -0000	1.1
@@ -0,0 +1,1564 @@
+ad_library {
+
+    routines for creating, managing input via html forms
+    @creation-date 21 Nov 2010
+    @cs-id $Id:
+}
+
+# use _ to clear a new default
+# use upvar to grab previous defaults and re-use (with qf_input only)
+# main namespace vars:
+# __form_input_arr = array that contains existing form input and defaults, only one form can be posted at a time
+# __form_ids_list  = list that contains existing form ids
+# __form_ids_open_list = list that contains ids of forms that are not closed
+# __form_ids_fieldset_open_list = list that contains form ids where a fieldset tag is open
+# __form_arr contains an array of forms. Each form built as a string by appending tags, indexed by form_id, for example __form_arr($id)
+# __qf_arr contains last attribute values of a tag (for all forms), indexed by {tag}_attribute, __form_last_id is in __qf_arr(form_id)
+# a blank id passed in anything other than qf_form assumes the current (most recent used form_id)
+
+# to fix:  id for nonform tag should not be same as form_id. use an attribute "form_id" for assigning tags to specific forms.
+
+#use following to limit access to page requests via post.. to reduce vulnerability to url hack and insertion attacks from web:
+#if { [ad_conn method] != POST } {
+#  ad_script_abort
+#}
+#also see patch: http://openacs.org/forums/message-view?message_id=182057
+
+# for early example and discussion, see http://openacs.org/forums/message-view?message_id=3602056
+
+ad_proc -private qf_form_key_create {
+    {key_id ""}
+    {action_url "/"}
+    {instance_id ""}
+} {
+    creates the form key for a more secure form transaction. Returns the security hash. See also qf_submit_key_accepted_p
+} {
+    # This proc is inspired from sec_random_token
+    if { $instance_id eq "" } {
+        # set instance_id package_id
+        set instance_id [ad_conn package_id]
+    }
+#    set time_sec \[ns_time\]
+# need more time separation
+    if { $key_id eq "" } {
+        set key_id [expr { int( [clock clicks] * [ns_rand] ) } ]
+    }
+    set time_sec [clock clicks -milliseconds]
+    set start_clicks [ad_conn start_clicks]
+    if { [ad_conn -connected_p] } {
+        set client_ip [ns_conn peeraddr]
+        #        set request \[ad_conn request\]
+        set secure_p [security::secure_conn_p]
+        set session_id [ad_conn session_id]
+        set action_url [ns_conn url]
+ #       set render_timestamp $time_sec
+    } else {
+        set server_ip [ns_config ns/server/[ns_info server]/module/nssock Address]
+        if { $server_ip eq "" } {
+            set server_ip "127.0.0.1"
+        }
+        set client_ip $server_ip
+        # time_sec s/b circa clock seconds
+        #set request \[string range $time_sec \[expr { floor( ( \[ns_rand\] * \[string length $time_sec\] ) ) }\] end\]
+        set secure_p [expr { floor( [ns_rand] + 0.5 ) } ]
+
+        set session_id [expr { floor( $time_sec / 4 ) } ]
+#        set action_url "/"
+#        set render_timestamp $time_sec
+    }
+    append sec_hash_string $start_clicks $session_id $secure_p $client_ip $action_url $time_sec $key_id
+    set sec_hash [ns_sha1 $sec_hash_string]
+    db_dml qf_form_key_create {insert into qf_key_map
+                  (instance_id,rendered_timestamp,sec_hash,key_id,session_id,action_url,secure_conn_p,client_ip)
+        values (:instance_id,:time_sec,:sec_hash,:key_id,:session_id,:action_url,:secure_p,:client_ip) }
+    return $sec_hash
+}
+
+ad_proc -private qf_submit_key_accepted_p {
+    {sec_hash ""}
+    {instance_id ""}
+} {
+    Checks the form key against existing ones. Returns 1 if matches and unexpired, otherwise returns 0.
+} {
+    # This proc is inspired from sec_random_token
+    if { $instance_id eq "" } {
+        # set instance_id package_id
+        set instance_id [ad_conn package_id]
+    }
+    set connected_p [ad_conn -connected_p]
+    if { $connected_p } {
+        set client_ip [ns_conn peeraddr]
+        set secure_p [security::secure_conn_p]
+        set session_id [ad_conn session_id]
+        set action_url [ns_conn url]
+
+    } else {
+        set server_ip [ns_config ns/server/[ns_info server]/module/nssock Address]
+        if { $server_ip eq "" } {
+            set server_ip "127.0.0.1"
+        }
+        set client_ip $server_ip
+        set secure_p ""
+        set session_id ""
+        set action_url [ns_conn url]
+    }
+    # the key_id is used to help generate unpredictable hashes, but isn't used at this level of input validation
+    set accepted_p [db_0or1row qf_form_key_check_hash { 
+        select session_id as session_id_i, action_url as action_url_i, secure_conn_p as secure_conn_p_i, client_ip as client_ip_i from qf_key_map
+        where instance_id =:instance_id and sec_hash =:sec_hash and submit_timestamp is null } ]
+    if { !$accepted_p } {
+        # there is nothing to compare. log current values:
+        ns_log Warning "qf_submit_key_accepted_p: is false. action_url '$action_url' sec_hash '$sec_hash'"
+        if { $connected_p } {
+            ns_log Warning "qf_submit_key_accepted_p: session_id '$session_id' secure_p '$secure_p' client_ip '$client_ip'"
+        }
+    } else {
+        # Mark the key expired
+        set submit_timestamp [ns_time]
+        db_dml qf_form_key_expire { update qf_key_map
+            set submit_timestamp = :submit_timestamp where instance_id =:instance_id and sec_hash =:sec_hash and submit_timestamp is null }
+    }
+    return $accepted_p
+}
+
+
+
+ad_proc -public qf_get_inputs_as_array {
+    {form_array_name "__form_input_arr"}
+    {arg1 ""}
+    {arg2 ""}
+    {arg3 ""}
+    {arg4 ""}
+    {arg5 ""}
+    {arg6 ""}
+} {
+    Get inputs from form submission, quotes all input values. Use ad_unquotehtml to unquote a value.
+    Returns 1 if form inputs exist, otherwise returns 0.
+    If duplicate_key_check is 1, checks if an existing key/value pair already exists, otherwise just overwrites existing value.  
+    Overwriting is programmatically useful to overwrite preset defaults, for example.
+} {
+    # get args
+    upvar 1 $form_array_name __form_input_arr
+    set array __form_buffer_arr
+    set arg_arr(duplicate_key_check) 0
+    set arg_arr(multiple_key_as_list) 0
+    set arg_arr(hash_check) 0
+    set arg_full_list [list duplicate_key_check multiple_key_as_list hash_check]
+    set arg_list [list $arg1 $arg2 $arg3 $arg4 $arg5 $arg6 ]
+    set args_list [list]
+    foreach {name value} $arg_list {
+        set arg_index [lsearch -exact $arg_full_list $name]
+        if { $arg_index > -1 } {
+            set arg_arr($name) $value
+        } elseif { $value eq "" } {
+            # ignore
+        } else {
+            ns_log Error "qf_get_inputs_as_array: $name is not a valid name invoked with name value pairs. Separate each with a space."
+        }
+    }
+
+    # get form variables passed with connection
+    set __form_input_exists 0
+    set __form [ns_getform]
+    if { $__form eq "" } {
+        set __form_size 0
+    } else {
+        set __form_size [ns_set size $__form]
+    }
+    #ns_log Notice "qf_get_inputs_as_array: formsize $__form_size"
+    for { set __form_counter_i 0 } { $__form_counter_i < $__form_size } { incr __form_counter_i } {
+
+        regexp -nocase -- {^[a-z][a-z0-9_\.\:\(\)]*} [ns_set key $__form $__form_counter_i] __form_key
+        # Why doesn't work for  regexp -nocase -- {^[a-z][a-z0-9_\.\:\(\)]*$ }    ?
+        set __form_key_exists [info exists __form_key]
+        # ns_log Notice "qf_get_inputs_as_array: __form_key_exists = ${__form_key_exists}"
+
+        # no inserting tcl commands etc!
+        if { $__form_key_exists == 0 || ( $__form_key_exists == 1 && [string length $__form_key] == 0 ) } {
+            # let's make this an error for now, so we log any attempts
+#            ns_log Notice "qf_get_inputs_as_array: __form_key_exists ${__form_key_exists} length __form_key [string length ${__form_key}]"
+ #           ns_log Notice "qf_get_inputs_as_array(ref156: attempt to insert unallowed characters to user input '{__form_key}' as '[ns_set key $__form $__form_counter_i]' for counter ${__form_counter_i}."
+            if { $__form_counter_i > 0 } {
+                ns_log Notice "qf_get_inputs_as_array: attempt to insert unallowed characters to user input '{__form_key}'."
+            }
+        } else {
+            set __form_key [ad_quotehtml $__form_key]
+            # The name of the argument passed in the form
+            # no legitimate argument should be affected by quoting:
+    
+            # This is the value
+            set __form_input [ad_quotehtml [ns_set value $__form $__form_counter_i]]
+
+            set __form_input_exists 1
+            # check for duplicate key?
+            if { $arg_arr(duplicate_key_check) && [info exists __form_buffer_arr($__form_key) ] } {
+                if { $__form_input ne $__form_buffer_arr($__form_key) } {
+                    # which one is correct? log error
+                    ns_log Error "qf_get_form_input: form input error. duplcate key provided for ${__form_key}"
+                    ad_script_abort
+                    # set __form_input_exists to -1 instead of ad_script_abort?
+                } else {
+                    ns_log Warning "qf_get_form_input: notice, form has a duplicate key with multiple values containing same info.."
+                }
+            } elseif { $arg_arr(multiple_key_as_list) } {
+                ns_log Notice "qf_get_inputs_as_array: A key has been posted with multible values. Values assigned to the key as a list."
+                if { [llength $__form_buffer_arr($__form_key)] > 1 } {
+                    # value is a list, lappend
+                    lappend __form_buffer_arr($__form_key) $__form_input
+                } else {
+                    # convert the key value to a list
+                    set __value_one $__form_buffer_arr($__form_key)
+                    unset __form_buffer_arr($__form_key)
+                    set __form_buffer_arr($__form_key) [list $__value_one $__form_input]
+                }
+            } else {
+                set __form_buffer_arr($__form_key) $__form_input
+#                ns_log Debug "qf_get_inputs_as_array: set ${form_array_name}($__form_key) '${__form_input}'."
+            }
+
+            # next key-value pair
+        }
+    }
+    if { $arg_arr(hash_check) } {
+        if { [info exists __form_buffer_arr(qf_security_hash) ] } {
+            set accepted_p [qf_submit_key_accepted_p $__form_buffer_arr(qf_security_hash) ]
+            if { $accepted_p } {
+                unset __form_buffer_arr(qf_security_hash)
+                array set __form_input_arr [array get __form_buffer_arr]
+                return $__form_input_exists
+            } else {
+                ns_log Notice "qf_get_inputs_as_array: hash_check with form input of '$__form_buffer_arr(qf_security_hash)' did not match."
+                return 0
+            }
+        } else {
+            set accepted_p 0
+            ns_log Notice "qf_get_inputs_as_array: hash_check requires qf_security_hash, but was not included with form input."
+            return 0
+        }
+    } else {
+        array set __form_input_arr [array get __form_buffer_arr]
+        return $__form_input_exists
+    }
+}
+
+ad_proc -public qf_remember_attributes {
+    {arg1 "1"}
+} {
+    Changes qf_* form building procs to use the previous attributes and their values used with the last tag of same type (input,select,button etc) if arg1 is 1. 
+} {
+    upvar __qf_remember_attributes __qf_remember_attributes
+    if { $arg1 eq 0 } {
+        set __qf_remember_attributes 0
+    } else {
+        set __qf_remember_attributes 1
+    }
+}
+
+ad_proc -public qf_form { 
+    {arg1 ""}
+    {arg2 ""}
+    {arg3 ""}
+    {arg4 ""}
+    {arg5 ""}
+    {arg6 ""}
+    {arg7 ""}
+    {arg8 ""}
+    {arg9 ""}
+    {arg10 ""}
+    {arg11 ""}
+    {arg12 ""}
+    {arg13 ""}
+    {arg14 ""}
+    {arg15 ""}
+    {arg16 ""}
+    {arg17 ""}
+    {arg18 ""}
+    {arg19 ""}
+    {arg20 ""}
+    {arg21 ""}
+    {arg22 ""}
+} {
+    Initiates a form with form tag and supplied attributes. Returns an id. A clumsy url based id is provided if not passed (not recommended). 
+    If hash_check passed, creates a hash to be checked on submit for server-client transaction continuity.
+} {
+    # use upvar to set form content, set/change defaults
+    # __qf_arr contains last attribute values of tag, indexed by {tag}_attribute, __form_last_id is in __qf_arr(form_id)
+    upvar 1 __form_ids_list __form_ids_list
+    upvar 1 __form_arr __form_arr
+    upvar 1 __form_ids_open_list __form_ids_open_list
+    upvar 1 __qf_remember_attributes __qf_remember_attributes
+    upvar 1 __qf_arr __qf_arr
+
+    # if proc was passed a list of parameters, parse
+    if { [llength $arg1] > 1 && [llength $arg2] == 0 } {
+        set arg1_list $arg1
+        set lposition 1
+        foreach arg $arg1_list {
+            set arg${lposition} $arg
+            incr lposition
+        }
+        unset arg1_list
+    }
+
+    set attributes_tag_list [list action class id method name style target title]
+    set attributes_full_list $attributes_tag_list
+    lappend attributes_full_list form_id hash_check key_id
+    set arg_list [list $arg1 $arg2 $arg3 $arg4 $arg5 $arg6 $arg7 $arg8 $arg9 $arg10 $arg11 $arg12 $arg13 $arg14 $arg15 $arg16 $arg17 $arg18 $arg19 $arg20 $arg21 $arg22]
+    set attributes_list [list]
+    foreach {attribute value} $arg_list {
+        set attribute_index [lsearch -exact $attributes_full_list $attribute]
+        if { $attribute_index > -1 } {
+            set attributes_arr($attribute) $value
+            if { [lsearch -exact $attributes_tag_list $attribute] > -1 } {
+                lappend attributes_list $attribute
+            }
+        } elseif { $value eq "" } {
+            # ignore
+        } else {
+            ns_log Error "qf_form: $attribute is not a valid attribute. invoke with attribute value pairs. Separate each with a space."
+        }
+    }
+    if { ![info exists attributes_arr(method)] } {
+        set attributes_arr(method) "post"
+        lappend attributes_list "method"
+    }
+# if html5 should we default novalidate to novalidate? No for now.
+
+    if { ![info exists __qf_remember_attributes] } {
+        set __qf_remember_attributes 0
+    }
+    if { ![info exists __form_ids_list] } {
+        set __form_ids_list [list]
+    }
+    if { ![info exists __form_ids_open_list] } {
+        set __form_ids_open_list [list]
+    }
+    # use previous tag attribute values?
+    if { $__qf_remember_attributes } {
+        foreach attribute $attributes_list {
+            if { $attribute ne "id" && ![info exists attributes_arr($attribute)] && [info exists __qf_arr(form_$attribute)] } {
+                set attributes_arr($attribute) $__qf_arr(form_$attribute)
+            }
+        }
+    }
+    # every form gets a form_id
+    set form_id_exists [info exists attributes_arr(form_id) ]
+    if { $form_id_exists == 0 || ( $form_id_exists == 1 && $attributes_arr(form_id) eq "" ) } { 
+        set id_exists [info exists attributes_arr(id) ]
+        if { $id_exists == 0 || ( $id_exists == 1 && $attributes_arr(id) eq "" ) } { 
+            regsub {/} [ad_conn url] {-} form_key
+            append form_key "-[llength $__form_ids_list]"
+        } else {
+            # since a FORM id has to be unique, lets use it
+            set form_key $attributes_arr(id)
+        }
+        set attributes_arr(form_id) $form_key
+        ns_log Notice "qf_form: generating form_id $attributes_arr(form_id)"
+    }
+
+    # prepare attributes to process
+    set tag_attributes_list [list]
+    foreach attribute $attributes_list {
+        set __qf_arr(form_$attribute) $attributes_arr($attribute)
+        # if a form tag requires an attribute, the following test needs to  be forced true
+        if { $attributes_arr($attribute) ne "" } {
+            lappend tag_attributes_list $attribute $attributes_arr($attribute)
+        }
+    }
+    
+    set tag_html "<form[qf_insert_attributes $tag_attributes_list]>"
+    # set results  __form_arr 
+    append __form_arr($attributes_arr(form_id)) "$tag_html\n"
+    if { [lsearch $__form_ids_list $attributes_arr(form_id)] == -1 } {
+        lappend __form_ids_list $attributes_arr(form_id)
+
+    }
+    if { [lsearch $__form_ids_open_list $attributes_arr(form_id)] == -1 } {
+        lappend __form_ids_open_list $attributes_arr(form_id)
+    }
+
+    #  append an input tag for qf_security_hash?
+    if { [info exists attributes_arr(hash_check)] && $attributes_arr(hash_check) eq 1 } {
+        if { ![info exists attributes_arr(key_id) ] } {
+            set attributes_arr(key_id) ""
+        }
+        set tag_html "<input[qf_insert_attributes [list type hidden name qf_security_hash value [qf_form_key_create $attributes_arr(key_id) $attributes_arr(action)]]]>"
+        append __form_arr($attributes_arr(form_id)) "$tag_html\n"
+        ns_log Notice "qf_form: adding $tag_html"
+    }
+    
+    set __qf_arr(form_id) $attributes_arr(form_id)
+    return $attributes_arr(form_id)
+}
+
+
+ad_proc -public qf_fieldset { 
+    {arg1 ""}
+    {arg2 ""}
+    {arg3 ""}
+    {arg4 ""}
+    {arg5 ""}
+    {arg6 ""}
+    {arg7 ""}
+    {arg8 ""}
+    {arg9 ""}
+    {arg10 ""}
+    {arg11 ""}
+    {arg12 ""}
+    {arg13 ""}
+    {arg14 ""}
+} {
+    Starts a form fieldset by appending a fieldset tag.  Fieldset closes when form is closed or another fieldset defined in same form.
+} {
+    # use upvar to set form content, set/change defaults
+    # __qf_arr contains last attribute values of tag, indexed by {tag}_attribute, __form_last_id is in __qf_arr(form_id)
+    upvar 1 __form_ids_list __form_ids_list
+    upvar 1 __form_arr __form_arr
+    upvar 1 __qf_remember_attributes __qf_remember_attributes
+    upvar 1 __qf_arr __qf_arr
+    upvar 1 __form_ids_fieldset_open_list __form_ids_fieldset_open_list
+
+    # if proc was passed a list of parameters, parse
+    if { [llength $arg1] > 1 && [llength $arg2] == 0 } {
+        set arg1_list $arg1
+        set lposition 1
+        foreach arg $arg1_list {
+            set arg${lposition} $arg
+            incr lposition
+        }
+        unset arg1_list
+    }
+
+    set attributes_tag_list  [list align class id style title valign]
+    set attributes_full_list $attributes_tag_list
+    lappend attributes_full_list form_id
+    set arg_list [list $arg1 $arg2 $arg3 $arg4 $arg5 $arg6 $arg7 $arg8 $arg9 $arg10 $arg11 $arg12 $arg13 $arg14]
+    set attributes_list [list]
+    foreach {attribute value} $arg_list {
+        set attribute_index [lsearch -exact $attributes_full_list $attribute]
+        if { $attribute_index > -1 } {
+            set attributes_arr($attribute) $value
+            if { [lsearch -exact $attributes_tag_list $attribute] > -1 } {
+                lappend attributes_list $attribute
+            }
+        } elseif { $value eq "" } {
+            # do nothing                  
+        } else {
+            ns_log Error "qf_fieldset: $attribute is not a valid attribute. invoke with attribute value pairs. Separate each with a space."
+            ad_script_abort
+        }
+    }
+
+    if { ![info exists __qf_remember_attributes] } {
+        ns_log Error "qf_fieldset: invoked before qf_form or used in a different namespace than qf_form.."
+        ad_script_abort
+    }
+    if { ![info exists __form_ids_list] } {
+        ns_log Error "qf_fieldset: invoked before qf_form or used in a different namespace than qf_form.."
+        ad_script_abort
+    }
+    # default to last modified form_id
+    set form_id_exists [info exists attributes_arr(form_id)]
+    if { $form_id_exists == 0 || ( $form_id_exists == 1 && $attributes_arr(form_id) eq "" ) } { 
+        set attributes_arr(form_id) $__qf_arr(form_id) 
+    }
+    if { [lsearch $__form_ids_list $attributes_arr(form_id)] == -1 } {
+        ns_log Error "qf_fieldset: unknown form_id $attributes_arr(form_id)"
+        ad_script_abort
+    }
+
+    # use previous tag attribute values?
+    if { $__qf_remember_attributes } {
+        foreach attribute $attributes_list {
+            if { $attribute ne "id" && ![info exists attributes_arr($attribute)] && [info exists __qf_arr(fieldset_$attribute)] } {
+                set attributes_arr($attribute) $__qf_arr(form_$attribute)
+            }
+        }
+    }
+
+    # prepare attributes to process
+    set tag_attributes_list [list]
+    foreach attribute $attributes_list {
+        set __qf_arr(fieldset_$attribute) $attributes_arr($attribute)
+        lappend tag_attributes_list $attribute $attributes_arr($attribute)
+    }
+    set tag_html ""
+    set previous_fs 0
+    # first close any existing fieldset tag with form_id
+    set __fieldset_open_list_exists [info exists __form_ids_fieldset_open_list]
+    if { $__fieldset_open_list_exists } {
+        if { [lsearch $__form_ids_fieldset_open_list $attributes_arr(form_id)] > -1 } {
+            append tag_html "</fieldset>\n"
+            set previous_fs 1
+        }
+    }
+    append tag_html "<fieldset[qf_insert_attributes $tag_attributes_list]>"
+
+    # set results __form_ids_fieldset_open_list
+    if { $previous_fs } {
+        # no changes needed, "fieldset open" already indicated
+    } else {
+        if { $__fieldset_open_list_exists } {
+            lappend __form_ids_fieldset_open_list $attributes_arr(form_id)
+        } else {
+            set __form_ids_fieldset_open_list [list $attributes_arr(form_id)]
+        }
+    }
+    # set results  __form_arr, we checked form_id above.
+    append __form_arr($attributes_arr(form_id)) "$tag_html\n"
+}
+
+ad_proc -public qf_textarea { 
+    {arg1 ""}
+    {arg2 ""}
+    {arg3 ""}
+    {arg4 ""}
+    {arg5 ""}
+    {arg6 ""}
+    {arg7 ""}
+    {arg8 ""}
+    {arg9 ""}
+    {arg10 ""}
+    {arg11 ""}
+    {arg12 ""}
+    {arg13 ""}
+    {arg14 ""}
+    {arg15 ""}
+    {arg16 ""}
+    {arg17 ""}
+    {arg18 ""}
+    {arg19 ""}
+    {arg20 ""}
+    {arg21 ""}
+    {arg22 ""}
+    {arg23 ""}
+    {arg24 ""}
+    {arg25 ""}
+    {arg26 ""}
+    {arg27 ""}
+    {arg28 ""}
+    {arg29 ""}
+    {arg30 ""}
+} {
+    Creates a form textarea tag, supplying attributes where nonempty values are supplied.
+    Attribute "label" places a label tag just before textarea tag, instead of wrapping around textarea
+    in order to facilitate practical alignment variations between label and textarea. 
+    To remove label tag, pass label attribute with empty string value.
+} {
+    # use upvar to set form content, set/change defaults
+    # __qf_arr contains last attribute values of tag, indexed by {tag}_attribute, __form_last_id is in __qf_arr(form_id)
+    upvar 1 __form_ids_list __form_ids_list
+    upvar 1 __form_arr __form_arr
+    upvar 1 __qf_remember_attributes __qf_remember_attributes
+    upvar 1 __qf_arr __qf_arr
+    upvar 1 __form_ids_fieldset_open_list __form_ids_fieldset_open_list
+
+    # if proc was passed a list of parameters, parse
+    if { [llength $arg1] > 1 && [llength $arg2] == 0 } {
+        set arg1_list $arg1
+        set lposition 1
+        foreach arg $arg1_list {
+            set arg${lposition} $arg
+            incr lposition
+        }
+        unset arg1_list
+    }
+
+    set attributes_tag_list [list accesskey align class cols id name readonly rows style tabindex title wrap]
+    set attributes_full_list $attributes_tag_list
+    lappend attributes_full_list value label form_id
+    set arg_list [list $arg1 $arg2 $arg3 $arg4 $arg5 $arg6 $arg7 $arg8 $arg9 $arg10 $arg11 $arg12 $arg13 $arg14 $arg15 $arg16 $arg17 $arg18 $arg19 $arg20 $arg21 $arg22 $arg23 $arg24 $arg25 $arg26 $arg27 $arg28 $arg29 $arg30]
+    set attributes_list [list]
+    foreach {attribute value} $arg_list {
+        set attribute_index [lsearch -exact $attributes_full_list $attribute]
+        if { $attribute_index > -1 } {
+            set attributes_arr($attribute) $value
+            if { [lsearch -exact $attributes_tag_list $attribute ] > -1 } {
+                lappend attributes_list $attribute
+            }
+        } elseif { $value eq "" } {
+            # do nothing                  
+        } else {
+            ns_log Error "qf_textarea: $attribute is not a valid attribute. invoke with attribute value pairs. Separate each with a space."
+            ad_script_abort
+        }
+    }
+
+    if { ![info exists __qf_remember_attributes] } {
+        ns_log Error "qf_textarea: invoked before qf_form or used in a different namespace than qf_form.."
+        ad_script_abort
+    }
+    if { ![info exists __form_ids_list] } {
+        ns_log Error "qf_textarea: invoked before qf_form or used in a different namespace than qf_form.."
+        ad_script_abort
+    }
+    # default to last modified form_id
+    set form_id_exists [info exists attributes_arr(form_id)]
+    if { $form_id_exists == 0 || ( $form_id_exists == 1 && $attributes_arr(form_id) eq "" ) } { 
+        set attributes_arr(form_id) $__qf_arr(form_id) 
+    }
+    if { [lsearch $__form_ids_list $attributes_arr(form_id)] == -1 } {
+        ns_log Error "qf_textarea: unknown form_id $attributes_arr(form_id)"
+        ad_script_abort
+    }
+
+    # use previous tag attribute values?
+    if { $__qf_remember_attributes } {
+        foreach attribute $attributes_list {
+            if { $attribute ne "id" && ![info exists attributes_arr($attribute)] && [info exists __qf_arr(textarea_$attribute)] } {
+                set attributes_arr($attribute) $__qf_arr(textarea_$attribute)
+            }
+        }
+    }
+
+    # value defaults to blank
+    if { ![info exists attributes_arr(value) ] } {
+        set attributes_arr(value) ""
+        lappend attributes_list "value"
+    }
+
+    # id defalts to form_id+name if label exists..
+    if { [info exists attributes_arr(label)] && ![info exists attributes_arr(id)] && [info exists attributes_arr(name)] } {
+        set attributes_arr(id) "${attributes_arr(form_id)}-${attributes_arr(name)}"
+        lappend attributes_list id
+    }
+
+    # prepare attributes to process
+    set tag_attributes_list [list]
+    foreach attribute $attributes_list {
+        set __qf_arr(textarea_$attribute) $attributes_arr($attribute)
+        lappend tag_attributes_list $attribute $attributes_arr($attribute)
+    }
+
+    # by default, wrap the input with a label tag for better UI
+    if { [info exists attributes_arr(id) ] && [info exists attributes_arr(label)] && $attributes_arr(label) ne "" } {
+        set tag_html "<label for=\"${attributes_arr(id)}\">${attributes_arr(label)}</label><textarea[qf_insert_attributes $tag_attributes_list]>${attributes_arr(value)}</textarea>"
+    } else {
+        set tag_html "<textarea[qf_insert_attributes $tag_attributes_list]>${attributes_arr(value)}</textarea>"
+    }
+    # set results  __form_arr, we checked form_id above.
+    append __form_arr($attributes_arr(form_id)) "${tag_html}\n"
+     
+}
+
+ad_proc -public qf_select { 
+    {arg1 ""}
+    {arg2 ""}
+    {arg3 ""}
+    {arg4 ""}
+    {arg5 ""}
+    {arg6 ""}
+    {arg7 ""}
+    {arg8 ""}
+    {arg9 ""}
+    {arg10 ""}
+    {arg11 ""}
+    {arg12 ""}
+    {arg13 ""}
+    {arg14 ""}
+    {arg15 ""}
+    {arg16 ""}
+    {arg17 ""}
+    {arg18 ""}
+    {arg19 ""}
+    {arg20 ""}
+    {arg21 ""}
+    {arg22 ""}
+    {arg23 ""}
+    {arg24 ""}
+    {arg25 ""}
+    {arg26 ""}
+    {arg27 ""}
+    {arg28 ""}
+} {
+    Creates a SELECT tag with nested OPTIONS, supplying necessary attributes where nonempty values are supplied. Set "multiple" to 1 to activate multiple attribute.
+    The argument for the "value" attribute is a list_of_lists passed to qf_options, where the list_of_lists represents a list of OPTION tag attribute/value pairs. 
+    Alternate to passing the "value" attribute, you can pass pure html containing literal Option tags as "value_html"
+} {
+    # use upvar to set form content, set/change defaults
+    # __qf_arr contains last attribute values of tag, indexed by {tag}_attribute, __form_last_id is in __qf_arr(form_id)
+    upvar 1 __form_ids_list __form_ids_list
+    upvar 1 __form_arr __form_arr
+    upvar 1 __qf_remember_attributes __qf_remember_attributes
+    upvar 1 __qf_arr __qf_arr
+    upvar 1 __form_ids_select_open_list __form_ids_select_open_list
+
+    # if proc was passed a list of parameters, parse
+    if { [llength $arg1] > 1 && [llength $arg2] == 0 } {
+        set arg1_list $arg1
+        set lposition 1
+        foreach arg $arg1_list {
+            set arg${lposition} $arg
+            incr lposition
+        }
+        unset arg1_list
+    }
+
+    set attributes_tag_list [list accesskey align class cols id name readonly rows style tabindex title wrap]
+    set attributes_full_list $attributes_tag_list
+    lappend attributes_full_list value form_id value_html multiple
+    set arg_list [list $arg1 $arg2 $arg3 $arg4 $arg5 $arg6 $arg7 $arg8 $arg9 $arg10 $arg11 $arg12 $arg13 $arg14 $arg15 $arg16 $arg17 $arg18 $arg19 $arg20 $arg21 $arg22 $arg23 $arg24 $arg25 $arg26 $arg27 $arg28]
+    set attributes_list [list]
+    foreach {attribute value} $arg_list {
+        set attribute_index [lsearch -exact $attributes_full_list $attribute]
+        if { $attribute_index > -1 } {
+            set attributes_arr($attribute) $value
+            if { [lsearch -exact $attributes_tag_list $attribute] > -1 } {
+                lappend attributes_list $attribute
+            }
+        } elseif { $value eq "" } {
+            # do nothing                  
+        } else {
+            ns_log Error "qf_select: [ad_quotehtml [string range $attribute 0 15]] is not a valid attribute. invoke with attribute value pairs. Separate each with a space."
+            ad_script_abort
+        }
+    }
+
+    if { ![info exists __qf_remember_attributes] } {
+        ns_log Error "qf_select: invoked before qf_form or used in a different namespace than qf_form.."
+        ad_script_abort
+    }
+    if { ![info exists __form_ids_list] } {
+        ns_log Error "qf_select: invoked before qf_form or used in a different namespace than qf_form.."
+        ad_script_abort
+    }
+    # default to last modified form_id
+    if { ![info exists attributes_arr(form_id)] || $attributes_arr(form_id) eq "" } { 
+        set attributes_arr(form_id) $__qf_arr(form_id) 
+    }
+    if { [lsearch $__form_ids_list $attributes_arr(form_id)] == -1 } {
+        ns_log Error "qf_select: unknown form_id $attributes_arr(form_id)"
+        ad_script_abort
+    }
+
+    # use previous tag attribute values?
+    if { $__qf_remember_attributes } {
+        foreach attribute $attributes_list {
+            if { $attribute ne "id" && ![info exists attributes_arr($attribute)] && [info exists __qf_arr(select_$attribute)] } {
+                set attributes_arr($attribute) $__qf_arr(select_$attribute)
+            }
+        }
+    }
+
+    # prepare attributes to process
+    set tag_attributes_list [list]
+    foreach attribute $attributes_list {
+            set __qf_arr(select_$attribute) $attributes_arr($attribute)
+            lappend tag_attributes_list $attribute $attributes_arr($attribute)
+    }
+
+    set tag_html ""
+    ## auto closing the select tag has been debrecated because qf_choice and qf_choicesexists.
+    # TO add this feature requires checking other input tags etc too.
+    # This code will be ignored for now, but left in place for future expansion.
+    set previous_select 0
+    # first close any existing selects tag with form_id
+    set __select_open_list_exists [info exists __form_ids_select_open_list]
+    if { $__select_open_list_exists } {
+        if { [lsearch $__form_ids_select_open_list $attributes_arr(form_id)] > -1 } {
+#            append tag_html "</select>\n"
+            set previous_select 1
+        }
+    }
+    # set results __form_ids_select_open_list
+    if { $previous_select } {
+        # no changes needed, "select open" already indicated
+    } else {
+        if { $__select_open_list_exists } {
+            lappend __form_ids_select_open_list $attributes_arr(form_id)
+        } else {
+            set __form_ids_select_open_list [list $attributes_arr(form_id)]
+        }
+    }
+
+    # add options tag
+    if { [info exists attributes_arr(value_html)] } {
+        set value_list_html $attributes_arr(value_html)
+    } else {
+        set value_list_html ""
+    }
+    if { [info exists attributes_arr(value)] } {
+        append value_list_html [qf_options $attributes_arr(value)]
+        
+    }
+
+    append tag_html "<select[qf_insert_attributes $tag_attributes_list]>$value_list_html</select>"
+    # set results  __form_arr, we checked form_id above.
+    append __form_arr($attributes_arr(form_id)) "${tag_html}\n"
+
+}
+
+ad_proc -private qf_options {
+    {options_list_of_lists ""}
+} {
+    Returns the sequence of options tags usually associated with SELECT tag. 
+    Does not append to an open form. These results are usually passed to qf_select that appends an open form.
+    Option tags are added in sequential order. A blank list in a list_of_lists is ignored. 
+    To add a blank option, include the value attribute with a blank/empty value; 
+    The option tag will wrap an attribute called "name".  
+    To indicate "SELECTED" attribute, include the attribute "selected" with the paired value of 1.
+} {
+    # options_list is expected to be a list like this:
+    # \[list \[list attribute1 value attribute2 value attribute3 value attribute4 value attribute5 value...\] \[list {second option tag attribute-value pairs} etc\] \]
+
+    # for this proc, we need to check the individual options for each OPTION tag, to provide the most flexibility.
+    set list_length [llength $options_list_of_lists]
+    # is this a list of lists, or just a list (1 list of list)
+    # test the second row to see if it has multiple list members
+    set multiple_option_tags_p [expr { [llength [lindex $options_list_of_lists 1] ] > 1 } ]
+    if { $list_length > 1 && $multiple_option_tags_p == 0 } {
+        # options_list is malformed, by providing only a list, not list of lists, adjust it:
+        set options_list_of_lists [list $options_list_of_lists]
+    }
+
+    set options_html ""
+    foreach option_tag_attribute_list $options_list_of_lists {
+        append options_html [qf_option $option_tag_attribute_list]
+    }
+    return $options_html
+}
+
+ad_proc -private qf_option {
+    {option_attributes_list ""}
+} {
+    returns an OPTION tag usually associated with SELECT tag. Does not append to an open form. These results are usually passed to qf_select that appends an open form.
+    Creates only one option tag. For multiple OPTION tags, see qf_options
+    To add a blank attribute, include attribute with a blank/empty value; 
+    The option tag will wrap an attribute called "name".  
+    To indicate "SELECTED" or "DISABLED" attribute, include the attribute ("selected" or "disabled") with the paired value of 1.
+} {
+    set attributes_tag_list [list class dir disabled id label lang language selected style title value]
+    set attributes_full_list $attributes_tag_list
+    lappend attributes_full_list label name
+    set arg_list $option_attributes_list
+    set attributes_list [list]
+    foreach {attribute value} $arg_list {
+        set attribute_index [lsearch -exact $attributes_full_list $attribute]
+        if { $attribute_index > -1 } {
+            set attributes_arr($attribute) $value
+            if { [lsearch -exact $attributes_tag_list $attribute] > -1 } {
+                lappend attributes_list $attribute
+            }
+        } elseif { $value eq "" } {
+            # do nothing                  
+        } else {
+            ns_log Error "qf_options: $attribute is not a valid attribute. invoke with attribute value pairs. Separate each with a space."
+            ad_script_abort
+        }
+    }
+
+    # prepare attributes to process
+    set tag_attributes_list [list]
+    foreach attribute $attributes_list {
+        if { $attribute ne "selected" && $attribute ne "disabled" && $attribute ne "checked" } {
+            lappend tag_attributes_list $attribute $attributes_arr($attribute)
+        } 
+    }
+    if { [info exists attributes_arr(label)] } {
+        set name_html $attributes_arr(label)
+    } elseif { [info exists attributes_arr(name)] } {
+        set name_html $attributes_arr(name)
+    } elseif { [info exists attributes_arr(value)] } {
+        set name_html $attributes_arr(value)
+    } else {
+        set name_html ""
+    }
+    if { [info exists attributes_arr(checked)] && ![info exists attributes_arr(selected)] } {
+        set attributes_arr(selected) "1"
+    }
+    if { ([info exists attributes_arr(selected)] && $attributes_arr(selected) eq "1") && $attributes_arr(selected) eq "1" } {
+        set option_html "<option[qf_insert_attributes $tag_attributes_list] selected>$name_html</option>\n"
+    } elseif { [info exists attributes_arr(disabled)] && $attributes_arr(disabled) eq "1" } {
+        set option_html "<option[qf_insert_attributes $tag_attributes_list] disabled>$name_html</option>\n"
+    } else {
+        set option_html "<option[qf_insert_attributes $tag_attributes_list]>$name_html</option>\n"
+    }
+    return $option_html
+}
+
+
+ad_proc -public qf_close { 
+    {arg1 ""}
+    {arg2 ""}
+} {
+    closes a form by appending a close form tag (and fieldset tag if any are open). if id supplied, only closes that referenced form and any fieldsets associated with it.  
+} {
+    # use upvar to set form content, set/change defaults
+    upvar 1 __form_ids_list __form_ids_list
+    upvar 1 __form_arr __form_arr
+    upvar 1 __form_ids_open_list __form_ids_open_list
+    upvar 1 __form_ids_fieldset_open_list __form_ids_fieldset_open_list
+
+    set attributes_full_list [list form_id]
+    set arg_list [list $arg1 $arg2]
+    set attributes_list [list]
+    foreach {attribute value} $arg_list {
+        set attribute_index [lsearch -exact $attributes_full_list $attribute]
+        if { $attribute_index > -1 } {
+            set attributes_arr($attribute) $value
+            lappend attributes_list $attribute
+        } elseif { $value eq "" } {
+            # do nothing                  
+        } else {
+            ns_log Error "qf_close: $attribute is not a valid attribute. invoke with attribute value pairs. Separate each with a space."
+            ad_script_abort
+        }
+    }
+
+    if { ![info exists __form_ids_list] } {
+        ns_log Error "qf_close: invoked before qf_form or used in a different namespace than qf_form.."
+        ad_script_abort
+    }
+    # default to all open form ids
+    if { ![info exists attributes_arr(form_id)] || $attributes_arr(form_id) eq "" } { 
+        set attributes_arr(form_id) $__form_ids_open_list
+        if { [lsearch -exact $attributes_list form_id] == -1 } {
+            lappend attributes_list "form_id"
+        }
+    }
+    # attributes_arr(form_id) might be a list or a single value. Following loop should work either way.
+    # close chosen form_id(s) 
+    set a_fieldset_exists [info exists __form_ids_fieldset_open_list]
+    foreach form_id $attributes_arr(form_id) {
+        # check if form_id is valid
+        set form_id_position [lsearch -exact $__form_ids_open_list $attributes_arr(form_id)]
+        if { $form_id_position == -1 } {
+            ns_log Warning "qf_close: unknown form_id $attributes_arr(form_id)"
+        } else {
+            if { $a_fieldset_exists } {
+                # close fieldset tag if form has an open one.
+                set form_id_fs_position [lsearch -exact $__form_ids_fieldset_open_list $form_id]
+                if { $form_id_fs_position > -1 } {
+                    append __form_arr($form_id) "</fieldset>\n"
+                    # remove form_id from __form_ids_fieldset_open_list
+                    set __form_ids_fieldset_open_list [lreplace $__form_ids_fieldset_open_list $form_id_fs_position $form_id_fs_position]
+                }
+            }
+            # close form
+            append __form_arr($form_id) "</form>\n"    
+            # remove form_id from __form_ids_open_list            
+            set __form_ids_open_list [lreplace $__form_ids_open_list $form_id_position $form_id_position]
+        }
+    }
+}
+
+ad_proc -public qf_read { 
+    {arg1 ""}
+    {arg2 ""}
+} {
+
+    returns the content of forms. If a form is not closed, returns the form in its partial state of completeness. If a form_id is supplied, returns the content of a specific form. Defaults to return all forms in a list.
+} {
+    # use upvar to set form content, set/change defaults
+    upvar 1 __form_ids_list __form_ids_list
+    upvar 1 __form_arr __form_arr
+
+    set attributes_full_list [list form_id]
+    set arg_list [list $arg1 $arg2]
+    set attributes_list [list]
+    foreach {attribute value} $arg_list {
+        set attribute_index [lsearch -exact $attributes_full_list $attribute]
+        if { $attribute_index > -1 } {
+            set attributes_arr($attribute) $value
+            lappend attributes_list $attribute
+        } elseif { $value eq "" } {
+            # do nothing                  
+        } else {
+            ns_log Error "qf_read: $attribute is not a valid attribute. invoke with attribute value pairs. Separate each with a space."
+            ad_script_abort
+        }
+    }
+
+    if { ![info exists __form_ids_list] } {
+        ns_log Error "qf_read: invoked before qf_form or used in a different namespace than qf_form.."
+        ad_script_abort
+    }
+    # normalize code using id instead of form_id
+    if { [info exists attributes_arr(form_id)] } {
+        set attributes_arr(id) $attributes_arr(form_id)
+        unset attributes_arr(form_id)
+    }
+    # defaults to all form ids
+    set form_id_exists [info exists attributes_arr(id)]
+    if { $form_id_exists == 0 || ( $form_id_exists == 1 && $attributes_arr(id) eq "" ) } { 
+        # note, attributes_arr(id) might become a list or a scalar..
+        if { [llength $__form_ids_list ] == 1 } {
+            set specified_1 1
+            set attributes_arr(id) [lindex $__form_ids_list 0]
+        } else {
+            set specified_1 0
+            set attributes_arr(id) $__form_ids_list
+        }
+    } else {
+        set specified_1 1
+    }
+
+    if { $specified_1 } {
+        # a form specified in argument
+        if { ![info exists __form_arr($attributes_arr(id)) ] } {
+            ns_log Warning "qf_read: unknown form_id $attributes_arr(id)"
+        } else {
+             set form_s $__form_arr($attributes_arr(id))
+        }
+    } else {
+        set forms_list [list]
+        foreach form_id $attributes_arr(id) {
+            # check if form_id is valid
+            set form_id_position [lsearch $__form_ids_list $form_id]
+            if { $form_id_position == -1 } {
+                ns_log Warning "qf_read: unknown form_id $form_id"
+            } else {
+                lappend forms_list $__form_arr($form_id)
+            }
+        }
+        set form_s $forms_list
+    }
+    return $form_s
+}
+
+
+ad_proc -public qf_input {
+    {arg1 ""}
+    {arg2 ""}
+    {arg3 ""}
+    {arg4 ""}
+    {arg5 ""}
+    {arg6 ""}
+    {arg7 ""}
+    {arg8 ""}
+    {arg9 ""}
+    {arg10 ""}
+    {arg11 ""}
+    {arg12 ""}
+    {arg13 ""}
+    {arg14 ""}
+    {arg15 ""}
+    {arg16 ""}
+    {arg17 ""}
+    {arg18 ""}
+    {arg19 ""}
+    {arg20 ""}
+    {arg21 ""}
+    {arg22 ""}
+    {arg23 ""}
+    {arg24 ""}
+    {arg25 ""}
+    {arg26 ""}
+    {arg27 ""}
+    {arg28 ""}
+    {arg29 ""}
+    {arg30 ""}
+    {arg31 ""}
+    {arg32 ""}
+} {
+    creates a form input tag, supplying attributes where nonempty values are supplied. when using CHECKED, set the attribute to 1.
+    allowed attributes: type accesskey align alt border checked class id maxlength name readonly size src tabindex value title.
+    other allowed: form_id label. label is used to wrap the input tag with a label tag containing a label that is associated with the input.
+    checkbox and radio inputs present label after input tag, other inputs are preceeded by label. Omit label attribute to not use this feature. Attribute title is associated with label.
+} {
+    # use upvar to set form content, set/change defaults
+    # __qf_arr contains last attribute values of tag, indexed by {tag}_attribute, __form_last_id is in __qf_arr(form_id)
+    upvar 1 __form_ids_list __form_ids_list
+    upvar 1 __form_arr __form_arr
+    upvar 1 __qf_remember_attributes __qf_remember_attributes
+    upvar 1 __qf_arr __qf_arr
+    upvar 1 __form_ids_fieldset_open_list __form_ids_fieldset_open_list
+
+    # if proc was passed a list of parameters, parse
+    if { [llength $arg1] > 1 && [llength $arg2] == 0 } {
+        set arg1_list $arg1
+        set lposition 1
+        foreach arg $arg1_list {
+            set arg${lposition} $arg
+            incr lposition
+        }
+        unset arg1_list
+    }
+
+    set attributes_tag_list [list type accesskey align alt border checked class id maxlength name readonly size src tabindex value]
+    set attributes_full_list $attributes_tag_list
+    lappend attributes_full_list form_id label selected title
+    set arg_list [list $arg1 $arg2 $arg3 $arg4 $arg5 $arg6 $arg7 $arg8 $arg9 $arg10 $arg11 $arg12 $arg13 $arg14 $arg15 $arg16 $arg17 $arg18 $arg19 $arg20 $arg21 $arg22 $arg23 $arg24 $arg25 $arg26 $arg27 $arg28 $arg29 $arg30 $arg31 $arg32]
+
+    set attributes_list [list]
+    foreach {attribute value} $arg_list {
+        set attribute_index [lsearch -exact $attributes_full_list $attribute]
+        if { $attribute_index > -1 } {
+            set attributes_arr($attribute) $value
+            if { [lsearch -exact $attributes_tag_list $attribute] > -1 } {
+                lappend attributes_list $attribute
+            }
+        } elseif { $value eq "" } {
+            # do nothing                  
+        } else {
+            ns_log Error "qf_input: $attribute is not a valid attribute. invoke with attribute value pairs. Separate each with a space."
+        }
+    }
+
+    if { ![info exists __qf_remember_attributes] } {
+        ns_log Notice "qf_input(L801): invoked before qf_form or used in a different namespace than qf_form.."
+        set __qf_remember_attributes 0
+    }
+    if { ![info exists __form_ids_list] } {
+        ns_log Warning "qf_input:(L805) invoked before qf_form or used in a different namespace than qf_form.."
+        set __form_ids_list [list [random]]
+        set __qf_arr(form_id) $__form_ids_list
+    }
+    # default to last modified form_id
+    if { ![info exists attributes_arr(form_id)] || $attributes_arr(form_id) eq "" } { 
+        set attributes_arr(form_id) $__qf_arr(form_id) 
+    }
+    if { [lsearch $__form_ids_list $attributes_arr(form_id)] == -1 } {
+        ns_log Error "qf_input: unknown form_id $attributes_arr(form_id)"
+        ad_script_abort
+    }
+
+    # use previous tag attribute values?
+    if { $__qf_remember_attributes } {
+        foreach attribute $attributes_list {
+            if { $attribute ne "id" && $attribute ne "value" && ![info exists attributes_arr($attribute)] && [info exists __qf_arr(input_$attribute)] } {
+                set attributes_arr($attribute) $__qf_arr(input_$attribute)
+            }
+        }
+    }
+
+    # provide a blank value by default
+    if { ![info exists attributes_arr(value)] } {
+        set attributes_arr(value) ""
+    }
+    # convert a "selected" parameter to checked
+    if { ([info exists attributes_arr(selected)] && $attributes_arr(selected) eq "1") && ![info exists attributes_arr(checked)] } {
+        set attributes_arr(checked) $attributes_arr(selected)
+        lappend attributes_list "checked"
+    }
+
+    # by default, wrap the input with a label tag for better UI, part 1
+    if { [info exists attributes_arr(label)] && [info exists attributes_arr(type) ] && $attributes_arr(type) ne "hidden" } {
+        if { ![info exists attributes_arr(id) ] } {
+            set attributes_arr(id) $attributes_arr(name)
+            append attributes_arr(id) "-[string range [clock clicks -milliseconds] end-3 end]-[string range [random ] 2 end]"
+            lappend attributes_list "id"
+        }
+        if { [info exists attributes_arr(title) ] } {
+            set label_title $attributes_arr(title)
+            unset attributes_arr(title)
+        }
+    }
+    # prepare attributes to process
+    set tag_attributes_list [list]
+    set tag_suffix ""
+    foreach attribute $attributes_list {
+        set __qf_arr(input_$attribute) $attributes_arr($attribute)
+        if { $attribute ne "checked" && $attribute ne "disabled" } {
+            lappend tag_attributes_list $attribute $attributes_arr($attribute)
+        } else {
+            set tag_suffix " ${attribute}"
+           # set to checked or disabled
+        }
+    }
+
+    # by default, wrap the input with a label tag for better UI, part 2
+    if { [info exists attributes_arr(label)] && [info exists attributes_arr(type) ] && $attributes_arr(type) ne "hidden" } {
+        if { $attributes_arr(type) eq "checkbox" || $attributes_arr(type) eq "radio" } {
+            set tag_html "<label for=\"${attributes_arr(id)}\""
+            if { [info exists label_title] } {
+                append tag_html " title=\"${label_title}\""
+            }
+            append tag_html "><input[qf_insert_attributes $tag_attributes_list]${tag_suffix}>${attributes_arr(label)}</label>"
+        } else {
+            set tag_html "<label for=\"${attributes_arr(id)}\""
+            if { [info exists label_title] } {
+                append tag_html " title=\"${label_title}\""
+            }
+            append tag_html ">${attributes_arr(label)}<input[qf_insert_attributes $tag_attributes_list]></label>"
+        }
+    } else {
+        set tag_html "<input[qf_insert_attributes $tag_attributes_list]${tag_suffix}>"
+    }
+
+    # set results  __form_arr, we checked form_id above.
+    append __form_arr($attributes_arr(form_id)) "${tag_html}\n"
+     
+    return "${tag_html}\n"
+}
+
+ad_proc -public qf_append { 
+    {arg1 ""}
+    {arg2 ""}
+    {arg3 ""}
+    {arg4 ""}
+    {arg5 ""}
+    {arg6 ""}
+} {
+    param html required
+    param form_id
+    inserts html in a form by appending supplied html. if form_id supplied, appends form with supplied form_id.
+} {
+    # use upvar to set form content, set/change defaults
+    # __qf_arr contains last attribute values of tag, indexed by {tag}_attribute, __form_last_id is in __qf_arr(form_id)
+    upvar 1 __form_ids_list __form_ids_list
+    upvar 1 __form_arr __form_arr
+    upvar 1 __qf_arr __qf_arr
+    upvar 1 __form_ids_fieldset_open_list __form_ids_fieldset_open_list
+
+    set attributes_full_list [list html form_id]
+    set arg_list [list $arg1 $arg2 $arg3 $arg4 $arg5 $arg6]
+    set attributes_list [list]
+    foreach {attribute value} $arg_list {
+        set attribute_index [lsearch -exact $attributes_full_list $attribute]
+        if { $attribute_index > -1 } {
+            set attributes_arr($attribute) $value
+            lappend attributes_list $attribute
+        } elseif { $value eq "" } {
+            # do nothing                  
+        } else {
+            ns_log Error "qf_append: $attribute is not a valid attribute. invoke with attribute value pairs. Separate each with a space."
+            ad_script_abort
+        }
+    }
+
+    if { ![info exists __form_ids_list] } {
+        ns_log Warning "qf_append:(L1209) invoked before qf_form or used in a different namespace than qf_form.."
+        set __form_ids_list [list [random]]
+        set __qf_arr(form_id) $__form_ids_list
+    }
+    # default to last modified form_id
+    set form_id_exists [info exists attributes_arr(form_id)]
+    if { $form_id_exists == 0 || ( $form_id_exists == 1 && $attributes_arr(form_id) eq "" ) } { 
+        set attributes_arr(form_id) $__qf_arr(form_id) 
+        lappend attributes_list form_id
+    }
+    if { [lsearch $__form_ids_list $attributes_arr(form_id)] == -1 } {
+        ns_log Error "qf_append: unknown form_id $attributes_arr(form_id)"
+        ad_script_abort
+    }
+    if { ![info exists attributes_arr(html)] } {
+        set attributs_arr(html) ""
+        ns_log Notice "qf_append: no argument 'html'"
+        if { [lsearch -exact $attributes_list "html"] == -1 } {
+            set attributes_arr(html) ""
+            lappend attributes_list "html"
+        }
+    }
+
+    # set results  __form_arr, we checked form_id above.
+    append __form_arr($attributes_arr(form_id)) $attributes_arr(html)
+    return $attributes_arr(html)
+}
+
+ad_proc -private qf_insert_attributes {
+    args_list
+} {
+    returns args_list of tag attribute pairs (attribute,value) as html to be inserted into a tag
+} {
+     set args_html ""
+     foreach {attribute value} $args_list {
+         if { [string range $attribute 1 1] eq "-" } {
+             set $attribute [string range $attribute 1 end]
+         }
+         regsub -all -- {\"} $value {\"} value
+         append args_html " $attribute=\"$value\""
+     }
+     return $args_html
+}
+
+ad_proc -public qf_choice {
+    {arg1 ""}
+    {arg2 ""}
+    {arg3 ""}
+    {arg4 ""}
+    {arg5 ""}
+    {arg6 ""}
+    {arg7 ""}
+    {arg8 ""}
+    {arg9 ""}
+    {arg10 ""}
+    {arg11 ""}
+    {arg12 ""}
+    {arg13 ""}
+    {arg14 ""}
+    {arg15 ""}
+    {arg16 ""}
+    {arg17 ""}
+    {arg18 ""}
+    {arg19 ""}
+    {arg20 ""}
+    {arg21 ""}
+    {arg22 ""}
+    {arg23 ""}
+    {arg24 ""}
+} {
+    Returns html of a select/option bar or radio button list (where only 1 value is returned to a posted form).
+    Set "type" to "select" for select bar, or "radio" for radio buttons
+    Required attributes:  name, value
+    "value" argument is a list_of_lists, each list item contains a list of attribute/value pairs for generating a radio or option/bar item.
+    "selected" is not required. Each choice is "unselected" by default. Set "selected" attribute to 1 to indicate item selected.
+    For this proc, "label" refers to the text that labels a radio buttion or select option item. If a "label" attribute/value pair is not included, The tag's value attribute is used for label as well.
+<pre>
+Example usage. This code:
+    set tag_attribute_list [list [list label " label1 " value visa1] [list label " label2 " value visa2] [list label " label3 " value visa3] ]
+    qf_choice type radio name creditcard value $tag_attribute_list
+
+Generates:
+
+"&lt;label>&lt;input type="radio" name="creditcard" value="visa1"> label1 &lt;/label>
+ &lt;label>&lt;input type="radio" name="creditcard" value="visa2"> label2 &lt;/label>
+ &lt;label>&lt;input type="radio" name="creditcard" value="visa3"> label3 &lt;/label>"
+
+By switching type to select like this:
+
+    qf_choice type select name creditcard value $tag_attribute_list
+
+the code generates:
+
+"&lt;select name="creditcard">
+&lt;option value="visa1"> label1 &lt;/option>
+&lt;option value="visa2"> label2 &lt;/option>
+&lt;option value="visa3"> label3 &lt;/option>
+&lt;/select>"
+</pre>
+    <!-- &lt; is used to prevent browser views from rendering the code presented here -->
+} {
+    # use upvar to set form content, set/change defaults
+    # __qf_arr contains last attribute values of tag, indexed by {tag}_attribute, __form_last_id is in __qf_arr(form_id)
+    upvar 1 __form_ids_list __form_ids_list
+    upvar 1 __form_arr __form_arr
+    upvar 1 __qf_remember_attributes __qf_remember_attributes
+    upvar 1 __qf_arr __qf_arr
+    upvar 1 __form_ids_select_open_list __form_ids_select_open_list
+    set attributes_select_list [list value accesskey align class cols name readonly rows style tabindex title wrap]
+    set attributes_full_list $attributes_select_list
+    lappend attributes_full_list type form_id id
+    set arg_list [list $arg1 $arg2 $arg3 $arg4 $arg5 $arg6 $arg7 $arg8 $arg9 $arg10 $arg11 $arg12 $arg13 $arg14 $arg15 $arg16 $arg17 $arg18 $arg19 $arg20 $arg21 $arg22 $arg23 $arg24]
+    set attributes_list [list]
+    set select_list [list]
+    foreach {attribute value} $arg_list {
+        set attribute_index [lsearch -exact $attributes_full_list $attribute]
+        if { $attribute_index > -1 } {
+            set attributes_arr($attribute) $value
+            lappend attributes_list $attribute
+            if { [lsearch -exact $attributes_select_list $attribute] > -1 } {
+                # create a list to pass to qf_select without it balking at unknown parameters
+                lappend select_list $attribute $value
+            } 
+        } elseif { $value eq "" } {
+            # do nothing                  
+        } else {
+            ns_log Error "qf_choice: [string range $attribute 0 15] is not a valid attribute. invoke with attribute value pairs. Separate each with a space."
+            ad_script_abort
+        }
+    }
+    # for passing select_list, we need to pass form_id literally
+    # default to last modified form_id
+    set form_id_exists [info exists attributes_arr(form_id)]
+    if { $form_id_exists == 0 || ( $form_id_exists == 1 && $attributes_arr(form_id) eq "" ) } { 
+        set attributes_arr(form_id) $__qf_arr(form_id) 
+    }
+    if { [lsearch $__form_ids_list $attributes_arr(form_id)] == -1 } {
+        ns_log Error "qf_choice: unknown form_id $attributes_arr(form_id)"
+        ad_script_abort
+    }
+    lappend select_list form_id $attributes_arr(form_id)
+
+ 
+
+    # if attributes_arr(type) = select, then items are option tags wrapped by a select tag
+    # if attributes_arr(type) = radio, then items are input tags, wrapped in a list for now
+    # if needing to paginate radio buttons, build the radio buttons using qf_input directly.
+
+    if { $attributes_arr(type) ne "radio" } {
+        set type "select"
+    } else {
+        set type "radio"
+    }
+    
+    # call qf_select if type is "select" instead of duplicating purpose of that code
+
+    if { $type eq "radio" } {
+        # create wrapping tag
+        set tag_wrapping "ul"
+        set args_html "<${tag_wrapping}"
+        foreach attribute $attributes_list {
+            # ignore proc parameters that are not tag attributes for the tag_wrapping tag
+            if { $attribute eq "id" || $attribute eq "style" || $attribute eq "class"  } {
+                # quoting unquoted double quotes in attribute values, so as to not inadvertently break the tag
+                regsub -all -- {\"} $attributes_arr($attribute) {\"} attributes_arr($attribute)
+                append args_html " $attribute=\"$attributes_arr($attribute)\""
+            }
+        }
+        append args_html ">\n"
+        qf_append form_id $attributes_arr(form_id) html $args_html
+        set args_html ""
+
+        # verify this is a list of lists.
+        set list_length [llength $attributes_arr(value)]
+        # test on the second input, less chance its a special case
+        set second_input_attributes_count [llength [lindex $attributes_arr(value) 1]]
+        if { $list_length > 1 && $second_input_attributes_count < 2 } {
+            # a list was passed instead of a list of lists. Adjust..
+            set attributes_arr(value) [list $attributes_arr(value)]
+        }
+        foreach input_attributes_list $attributes_arr(value) {
+            if { [f::even_p [llength $input_attributes_list]] } {
+                array unset input_arr
+                array set input_arr $input_attributes_list
+                if { ![info exists input_arr(label)] && [info exists input_arr(value)] } {
+                    set input_arr(label) $input_arr(value)
+                } 
+                if { ![info exists input_arr(name)] && [info exists attributes_arr(name)] } {
+                    set input_arr(name) $attributes_arr(name)
+                }
+                set input_attributes_list [array get input_arr]
+                lappend input_attributes_list form_id $attributes_arr(form_id) type radio
+                qf_append form_id $attributes_arr(form_id) html "<li>"
+                qf_input $input_attributes_list
+                qf_append form_id $attributes_arr(form_id) html "</li>"
+            } else {
+                ns_log Notice "qf_choice: list not even number of members, skipping rendering of value attribute with list: $input_attributes_list"
+            }
+        }
+        append args_html "</${tag_wrapping}>"
+        qf_append form_id $attributes_arr(form_id) html $args_html
+
+    } else {
+
+        set args_html [qf_select $select_list]
+
+    }
+    return $args_html
+}
+
+ad_proc -public qf_choices {
+    {arg1 ""}
+    {arg2 ""}
+    {arg3 ""}
+    {arg4 ""}
+    {arg5 ""}
+    {arg6 ""}
+    {arg7 ""}
+    {arg8 ""}
+    {arg9 ""}
+    {arg10 ""}
+    {arg11 ""}
+    {arg12 ""}
+    {arg13 ""}
+    {arg14 ""}
+    {arg15 ""}
+    {arg16 ""}
+    {arg17 ""}
+    {arg18 ""}
+    {arg19 ""}
+    {arg20 ""}
+    {arg21 ""}
+    {arg22 ""}
+    {arg23 ""}
+    {arg24 ""}
+ } {
+    returns html of a select multiple box or list  of checkboxes (where multiple values may be sent with form post).
+     Required attributes:  name, value.  
+     Set "type" to "select" for multi select box, or "checkbox" for checkboxes.
+     The value of the "value" attribute is a list_of_lists, each list item contains attribute/value pairs for a radio or option/bar item.
+     If "label" not provided for tags in the list_of_lists, the value of the "value" attribute is also used for label.
+     Set "selected" attribute to 1 in the value list_of_lists to indicate item selected. Default is unselected (if selected attributed is not included, or its value not 1)..
+ } {
+    # use upvar to set form content, set/change defaults
+    # __qf_arr contains last attribute values of tag, indexed by {tag}_attribute, __form_last_id is in __qf_arr(form_id)
+    upvar 1 __form_ids_list __form_ids_list
+    upvar 1 __form_arr __form_arr
+    upvar 1 __qf_remember_attributes __qf_remember_attributes
+    upvar 1 __qf_arr __qf_arr
+    upvar 1 __form_ids_select_open_list __form_ids_select_open_list
+
+    set attributes_select_list [list value accesskey align class cols name readonly rows style tabindex title wrap]
+    set attributes_full_list $attributes_select_list
+    lappend attributes_full_list type form_id id
+    set arg_list [list $arg1 $arg2 $arg3 $arg4 $arg5 $arg6 $arg7 $arg8 $arg9 $arg10 $arg11 $arg12 $arg13 $arg14 $arg15 $arg16 $arg17 $arg18 $arg19 $arg20 $arg21 $arg22 $arg23 $arg24]
+    set attributes_list [list]
+    set select_list [list]
+    foreach {attribute value} $arg_list {
+        set attribute_index [lsearch -exact $attributes_full_list $attribute]
+        if { $attribute_index > -1 } {
+            set attributes_arr($attribute) $value
+            lappend attributes_list $attribute
+            if { [lsearch -exact $attributes_select_list $attribute ] > -1 } {
+                # create a list to pass to qf_select without it balking at unknown parameters
+                lappend select_list $attribute $value
+            } 
+        } elseif { $value eq "" } {
+            # do nothing                  
+        } else {
+            ns_log Error "qf_choices: [string range $attribute 0 15] is not a valid attribute. invoke with attribute value pairs. Separate each with a space."
+            ad_script_abort
+        }
+    }
+
+    # for passing select_list, we need to pass form_id literally
+    # default to last modified form_id
+    set form_id_exists [info exists attributes_arr(form_id)]
+    if { $form_id_exists == 0 || ( $form_id_exists == 1 && $attributes_arr(form_id) eq "" ) } { 
+        set attributes_arr(form_id) $__qf_arr(form_id) 
+    }
+    if { [lsearch $__form_ids_list $attributes_arr(form_id)] == -1 } {
+        ns_log Error "qf_choice: unknown form_id $attributes_arr(form_id)"
+        ad_script_abort
+    }
+    lappend select_list form_id $attributes_arr(form_id)
+
+    # if attributes_arr(type) = select, then items are option tags wrapped by a select tag
+    # if attributes_arr(type) = checkbox, then items are input tags, wrapped in a list for now
+    # if needing to paginate checkboxes, build the checkboxes using qf_input directly.
+
+    if { $attributes_arr(type) ne "checkbox" } {
+        set type "select"
+    } else {
+        set type "checkbox"
+    }
+    
+    # call qf_select if type is "select" instead of duplicating purpose of that code
+
+    if { $type eq "checkbox" } {
+        # create wrapping tag
+        set tag_wrapping "ul"
+        set args_html "<${tag_wrapping}"
+        foreach attribute $attributes_list {
+            # ignore proc parameters that are not tag attributes
+            if { $attribute eq "id" || $attribute eq "style" || $attribute eq "class"  } {
+                # quoting unquoted double quotes in attribute values, so as to not inadvertently break the tag
+                regsub -all -- {\"} $attributes_arr($attribute) {\"} attributes_arr($attribute)
+                append args_html " $attribute=\"$attributes_arr($attribute)\""
+            }
+        }
+        append args_html ">\n"
+        qf_append form_id $attributes_arr(form_id) html $args_html
+        set args_html ""
+
+        # verify this is a list of lists.
+        set list_length [llength $attributes_arr(value)]
+        # test on the second input, less chance its a special case
+        set second_input_attributes_count [llength [lindex $attributes_arr(value) 1]]
+        if { $list_length > 1 && $second_input_attributes_count < 2 } {
+            # a list was passed instead of a list of lists. Adjust..
+            set attributes_arr(value) [list $attributes_arr(value)]
+        }
+        
+        foreach input_attributes_list $attributes_arr(value) {
+            array unset input_arr
+            array set input_arr $input_attributes_list
+            if { ![info exists input_arr(label)] && [info exists input_arr(value)] } {
+                set input_arr(label) $input_arr(value)
+            } 
+            if { ![info exists input_arr(name)] && [info exists attributes_arr(name)] } {
+                set input_arr(name) $attributes_arr(name)
+            }
+            set input_attributes_list [array get input_arr]
+            lappend input_attributes_list form_id $attributes_arr(form_id) type checkbox
+            qf_append form_id $attributes_arr(form_id) html "<li>"
+            qf_input $input_attributes_list
+            qf_append form_id $attributes_arr(form_id) html "</li>"
+        }
+        qf_append form_id $attributes_arr(form_id) html "</${tag_wrapping}>\n"
+    } else {
+        set args_html [qf_select $select_list]
+    }
+    return $args_html
+}    
Index: openacs-4/packages/q-forms/www/doc/index.adp
===================================================================
RCS file: /usr/local/cvsroot/openacs-4/packages/q-forms/www/doc/index.adp,v
diff -u
--- /dev/null	1 Jan 1970 00:00:00 -0000
+++ openacs-4/packages/q-forms/www/doc/index.adp	14 Nov 2014 18:27:53 -0000	1.1
@@ -0,0 +1,77 @@
+<master>
+<property name="title">@title;noquote@</property>
+<property name="context">@context;noquote@</property>
+<h2>Q-Forms</h2>
+<pre>
+The lastest version of the code is available at the site:
+ http://github.com/dcpm/q-forms
+The development site: http://github.com/tekbasse/q-forms
+</pre>
+<h3>
+introduction
+</h3>
+<p>
+Q-Forms provides procedures for building forms dynamically in OpenACS tcl.
+It is an OpenACS service package that allows convenient building and
+interpreting of web-based forms via tcl in a web page.
+</p><p>
+Q-Forms procedures parallel html's form tags with many
+automatic defaults that remove the tedious nature of building forms 
+via html or an alternate form building context, such as OpenACS form
+builder, ad_form or acs-templating.
+</p>
+<h3>
+license
+</h3>
+<pre>
+Copyright (c) 2013 Benjamin Brink
+po box 20, Marylhurst, OR 97036-0020 usa
+email: tekbasse@yahoo.com
+</pre>
+<p>
+Q-Forms is open source and published under the GNU General Public License, consistent with the OpenACS system: http://www.gnu.org/licenses/gpl.html
+</p><p>
+A local copy is available at <a href="LICENSE.html">q-forms/www/doc/LICENSE.html</a>
+</p><pre>
+    This program is free software: you can redistribute it and/or modify
+    it under the terms of the GNU General Public License as published by
+    the Free Software Foundation, either version 3 of the License, or
+    (at your option) any later version.
+
+    This program is distributed in the hope that it will be useful,
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+    GNU General Public License for more details.
+
+    You should have received a copy of the GNU General Public License
+    along with this program.  If not, see <http://www.gnu.org/licenses/>.
+</pre>
+<h3>
+features
+</h3>
+<ul><li>
+Low learning-curve:<ul><li>Uses tcl context.</li><li>Procedures match tags.</li><li>TCL list friendly.</li></ul>
+</li><li>
+Built-in API defaults: Takes less keystrokes to build a form than typing html manually.
+</li><li>
+Can build multiple forms concurently using Tcl file terminology.
+</li><li>
+No limitations to building dynamic forms with specialized inputs.
+</li><li>
+Form values are retrieved as a tcl array named by the programmer.
+</li><li>
+Form values are automatically quoted, a requirement of secure input handling.
+</li><li>
+Optional, automatic hash generation helps secure form transactions 
+and ignores multiple posts caused from mouse double-clicks and browsing page history.
+</li><li>
+Passing multiple values of same input name can be combined as a list (instead of producing
+a form post error typical of ad_form/ad_page_contract).
+</li><li>
+html can be inserted in any form at any point during the build.
+</li><li>
+No UI javascript is used. Technologies with limited UI, cpu power, or low QOS connection can use it.
+</li><li>
+Integrates with acs-templating features.
+</li></ul>
+
Index: openacs-4/packages/q-forms/www/doc/index.tcl
===================================================================
RCS file: /usr/local/cvsroot/openacs-4/packages/q-forms/www/doc/index.tcl,v
diff -u
--- /dev/null	1 Jan 1970 00:00:00 -0000
+++ openacs-4/packages/q-forms/www/doc/index.tcl	14 Nov 2014 18:27:53 -0000	1.1
@@ -0,0 +1,2 @@
+set title "Documentation"
+set context [list $title]