Index: openacs-4/packages/xowf/COPYRIGHT =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/COPYRIGHT,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/COPYRIGHT 28 Aug 2014 08:24:55 -0000 1.1 @@ -0,0 +1,23 @@ + * xowf + * + * Copyright (C) 2008 Gustaf Neumann, neumann@wu-wien.ac.at + * + * Vienna University of Economics and Business Administration + * Institute of Information Systems and New Media + * A-1090, Augasse 2-6 + * Vienna, Austria + * + * This is a BSD-Style license applicable for the files in this + * directory and below, except when stated explicitly different. + * + * Permission to use, copy, modify, distribute, and sell this + * software and its documentation for any purpose is hereby granted + * without fee, provided that the above copyright notice appear in + * all copies and that both that copyright notice and this permission + * notice appear in supporting documentation. We make no + * representations about the suitability of this software for any + * purpose. It is provided "as is" without express or implied + * warranty. + * + + Index: openacs-4/packages/xowf/xowf.info =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/xowf.info,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/xowf.info 28 Aug 2014 08:24:56 -0000 1.1 @@ -0,0 +1,31 @@ + + + + + XoWiki Workflow + XoWiki Workflows + f + f + f + t + + + Gustaf Neumann + XoWiki Content Flow - an XoWiki based workflow system implementing state-based behavior of wiki pages and forms + WU Vienna + 2014-02-14 + 2 + + + + + + + + + + + + + Index: openacs-4/packages/xowf/catalog/xowf.de_DE.ISO-8859-1.xml =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/catalog/xowf.de_DE.ISO-8859-1.xml,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/catalog/xowf.de_DE.ISO-8859-1.xml 28 Aug 2014 08:24:56 -0000 1.1 @@ -0,0 +1,11 @@ + + + + Richtig! + Falsch. + Beispiel + Angabetext + Bewertungsverfahren + Minuten + + Index: openacs-4/packages/xowf/catalog/xowf.en_US.ISO-8859-1.xml =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/catalog/xowf.en_US.ISO-8859-1.xml,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/catalog/xowf.en_US.ISO-8859-1.xml 28 Aug 2014 08:24:56 -0000 1.1 @@ -0,0 +1,20 @@ + + + + Alternative + That is correct! + Sorry, incorrect! + Columns + Correct? + Exercise + Exercise Text + Multiple Choice Exercise + Open Text Exercise + Single Choice Exercise + Feedback + Feedback Correct + Feedback Incorrect + Grading Schema + Lines + Minutes + Index: openacs-4/packages/xowf/lib/clicker-answer.wf =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/lib/clicker-answer.wf,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/lib/clicker-answer.wf 28 Aug 2014 08:24:56 -0000 1.1 @@ -0,0 +1,31 @@ +# -*- Tcl -*- +# +# Simple clicker answer workflow template. +# +# Template variables: +# @wfID@ +# @wfTitle@ +# @wfQuestion@ + +my set autoname 1 +Property return_url -default "" -allow_query_parameter true +Action allocate -proc activate {obj} { + my set_new_property name ___@wfID@.[::xo::cc set untrusted_user_id] +} +Action initialize -proc activate {obj} { + my set_new_property _title "@wfTitle@" +} +Action submit -next_state done +State initial -actions {submit} -form ../@wfQuestion@ -view_method edit +State done -form ../de:question45 -eval_when_active { + [my object] proc view args { + set return_url [my property return_url] + if {$return_url ne ""} { + ad_returnredirect $return_url + ad_script_abort + } else { + next + } + } +} +if {[::xo::cc mobile]} {::xo::cc set_parameter template_file view-mobile} Index: openacs-4/packages/xowf/lib/online-exam-answer.wf =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/lib/online-exam-answer.wf,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/lib/online-exam-answer.wf 28 Aug 2014 08:24:56 -0000 1.1 @@ -0,0 +1,272 @@ +# -*- Tcl -*- +# +# Workflow template for answering online exams. The workflow is +# typically controlled from a parent workflow that a teacher can use +# to create the exam, to try it out and to publish it +# (oneline-exam.wf). +# +# This workflow is similar to the classical "iterate.wf" but is more +# configurable (the answer forms are passed via template +# variables). The workflow uses a form-loader which renames the input +# fields to avoid potential name clashes. +# +# Template variables: +# @wfTitle@ +# @wfQuestionNames@ +# @wfID@ + + +my set autoname 1 +my set debug 1 + +set object [my set object] +set package_id [$object package_id] +set pages [list @wfQuestionNames@] + +######################################################################## +# +# Properties +# +# pages: the form pages used as inout forms +# position: the current page in the exam +# return_url: when the exam is finished, the user proceeds to this url +# try_out_mode: a teacher can try the exam in this mode +# current_form: used internally to keep the current form +# ip: ip address of the user, kept in the instance attribute as record +# +######################################################################## + +Property pages -default $pages +Property position -default 0 -allow_query_parameter true +Property return_url -default "" -allow_query_parameter true +Property try_out_mode -default 0 -allow_query_parameter true +Property current_form -default "" +Property ip -default [expr {[ns_conn isconnected] ? [ad_conn peeraddr] : "nowhere"}] + +Condition more_ahead \ + -expr {[my property position] < [llength [my property pages]]-1} + +Condition more_before \ + -expr {[my property position] > 0} + +######################################################################## +# +# Action definitions +# +######################################################################## + +Action allocate -proc activate {obj} { + #my msg "allocate $obj" + # Called, when we try to create or use a workflow instance + # via a workflow definition ($obj is a workflow definition) + my set_new_property name ___[::xo::cc set untrusted_user_id] +} + +Action initialize -proc activate {obj} { + # called, after workflow instance was created + + my set_new_property _title "@wfTitle@" + set parent_id [$obj parent_id] + set package_id [$obj package_id] + + # make sure to create the parent (the controlling workflow) + ::xo::db::CrClass get_instance_from_db -item_id $parent_id + set parent_state [$parent_id state] + + # + # Don't allow to enter values when the state of the master workflow + # is not published (e.g. trial mode, or closed) or when try-out-mode + # is not set + # + if {$parent_state ne "published" && [my property try_out_mode 0] == 0} { + #my msg "LOCKED" + set current_state [my property _state] + set lockin_state [expr {$current_state eq "initial" ? "initial" : "done"}] + set lockin_msg(initial) "Die Prüfung ist von der Aufsicht nicht freigegeben!" + set lockin_msg(done) "Die Prüfungszeit ist abgelaufen!" + foreach a [Action info instances] { + if {[namespace tail $a] eq "logout"} continue + $a next_state $lockin_state + $a proc activate {obj} [list util_user_message -message $lockin_msg($lockin_state)] + $a set_property position 0 + $a set_property current_form "" + } + } else { + #my msg "not LOCKED" + } +} + +Action instproc set_page {increment} { + set pages [my property pages] + set position [my property position 0] + incr position $increment + if {$position < 0} { + set position 0 + } elseif {$position >= [llength $pages]} { + set position [expr {[llength $pages] - 1}] + } + my set_property position $position + my set_property current_form [lindex $pages $position] +} + +Action prev \ + -next_state working \ + -label "Vorherige Frage" \ + -proc activate {obj} {my set_page -1} + +Action next \ + -next_state working \ + -label "Nächste Frage" \ + -proc activate {obj} {my set_page 1} + +Action abgabe \ + -next_state done \ + -label "Abgabe" + +Action save \ + -label "Antwort zwischenspeichern" + +Action logout \ + -label "Prüfung verlassen" \ + -proc activate {obj} { + set pid [$obj package_id] + set try_out_mode [my property try_out_mode 0] + set return_url [my property return_url .] + #my msg "tryout $try_out_mode return_url $return_url" + if {$try_out_mode} { + ad_returnredirect $return_url + } else { + ::xo::cc set_parameter return_url /register/logout?return_url=$return_url + } + } + +Action start \ + -next_state working \ + -label Beginnen \ + -proc activate {obj} { + my set_property position 0 + my set_property current_form [lindex [my property pages] 0] + } + +Action start_again \ + -label "Erste Frage" \ + -next_state working -proc activate {obj} { + my set_property position 0 + my set_property current_form [lindex [my property pages] 0] + } + +######################################################################## +# +# State definitions +# +######################################################################## + +State parameter { {view_method edit} + {extra_js { + /resources/xowiki/jquery/jquery.js + ../file:seal.js?m=download + }} + {extra_css { + ../file:seal.js?m=download + }}} + +State working -form [my property current_form] +working proc actions {} { + set actions "" + if {[more_before]} {lappend actions prev} + if {[more_ahead]} {lappend actions next} + lappend actions save abgabe +} + +State initial \ + -actions {start logout} \ + -form "../en:exam-start" + +State done \ + -actions {start_again logout} \ + -form "../en:exam-done" \ + -form_loader summary_form + + +######################################################################## +# +# Helper methods for the workflow context +# +######################################################################## + +# +# Overload default form loader to rename the input fields +# to avoid name clashes +# +my proc default_form_loader {form_name} { + #my msg "renaming_form_loader $form_name" + set form_id [next] + ::xo::db::CrClass get_instance_from_db -item_id $form_id + + set form [$form_id get_property -name form] + set prefix [lindex [split [$form_id name] :] end]-a + set counter 0 + set fc [my get_form_constraints] + dom parse -simple -html $form doc + $doc documentElement root + if {$root ne ""} { + foreach node [$root selectNodes "//textarea|//input"] { + set newName $prefix[incr counter] + $node setAttribute name $newName + #lappend fc $newName:richtext,editor=xinha,slim=true + } + $form_id set_property form [$root asHTML] + } + # Currently, the computation and setting of the form_constraints has + # no effect, when the input field is provided raw in the form + # (e.g. as a handcoded textarea). We set it anyhow here for future + # use + $form_id set_property -new 1 form_constraints $fc + my set_title -question 1 + return $form_id +} + +# +# set title with question and user information +# +my proc set_title {{-question 1}} { + set t [list ] + set object [my object] + set state [$object state] + if {$question && $state eq "working"} {lappend title "Frage [expr {[my property position] + 1}]"} + lappend title \ + "@wfTitle@" \ + "IP: [$object property ip]" + + $object title [join $title " / "] +} + +# +# Form loader for summary +# +my proc summary_form {form_title} { + #my msg "summary_form_loader $form_title" + + my set_title -question 0 + set state [my property _state] + + set summary_form "" + set counter 0 + foreach form_name [my property pages] { + set form_id [my default_form_loader $form_name] + append summary_form

Frage [incr counter]

\n + append summary_form [$form_id property form] \n
\n + } + + # disable all input fields and remove wrapping form + regsub -all {\n" + append fc "answer:textarea" + append form "\n" + [my object] set_property -new 1 form $form + [my object] set_property -new 1 form_constraints $fc + set anon_instances true ;# TODO make me configurable + [my object] set_property -new 1 anon_instances $anon_instances + [my object] set_property -new 1 auto_correct [[self class] set auto_correct] + [my object] set_property -new 1 has_solution false + } +} + + +namespace eval ::xowiki::formfield { + + ########################################################### + # + # ::xowiki::formfield::test_section + # + ########################################################### + + Class test_section -superclass {form_page} -parameter { + {multiple true} + } + + test_section instproc pretty_value {v} { + return [[my object] property form ""] + } + + test_section instproc convert_to_internal {} { + # + # Build a complex form composed of the specified form pages names + # contained in the value of this field. The form-fields have to + # be renamed. This affects the input field names in the form and + # the form constaints. We use the item-id contained pages as a the + # prefix for the form-fields. This method must be most likely + # extended for other question types. + # + set form "
\n" + set fc "@categories:off @cr_fields:hidden\n" + set intro_text [[my object] property _text] + append form "$intro_text\n
    \n" + foreach v [my value] { + # TODO: the next two commands should not be necessary to lookup + # again, since the right values are already loaded into the + # options + set item_id [[[my object] package_id] lookup -name $v] + set page [::xo::db::CrClass get_instance_from_db -item_id $item_id] + append form "
  1. [$item_id title]

    \n" + set prefix c$item_id + set __ia [$page set instance_attributes] + # + # If for some reason, we have not form entry, we ignore it. + # TODO: We should deal here with computed forms and with true + # ::xowiki::forms as well... + # + if {![dict exists $__ia form]} { + my msg "$v has no form included" + continue + } + # + # Replace the form-field names in the form + # + dom parse -simple -html [dict get $__ia form] doc + $doc documentElement root + set alt_inputs [list] + set alt_values [list] + foreach html_type {input textarea} { + foreach n [$root selectNodes "//$html_type\[@name != ''\]"] { + set alt_input [$n getAttribute name] + $n setAttribute name $prefix-$alt_input + if {$html_type eq "input"} { + set alt_value [$n getAttribute value] + } else { + set alt_value "" + } + lappend alt_inputs $alt_input + lappend alt_values $alt_value + } + } + # We have to drop the toplevel of the included form + foreach n [$root childNodes] {append form [$n asHTML]} + append form "
  2. \n" + # + # Replace the formfield names in the form constraints + # + foreach f [dict get $__ia form_constraints] { + if {[regexp {^([^:]+):(.*)$} $f _ field_name definition]} { + if {[string match @* $field_name]} continue + # keep all form-constraints for which we have altered the name + #my msg "old fc=$f, [list lsearch -exact $alt_inputs $field_name] => [lsearch -exact $alt_inputs $field_name] $alt_values" + set ff [[my object] create_raw_form_field -name $field_name -spec $definition] + #my msg "ff answer => '[$ff answer]'" + if {$field_name in $alt_inputs} { + lappend fc $prefix-$f + } elseif {[$ff exists answer] && $field_name eq [$ff answer]} { + # this rules is for single choice + lappend fc $prefix-$f + } + } + } + } + append form "
\n" + [my object] set_property -new 1 form $form + [my object] set_property -new 1 form_constraints $fc + set anon_instances true ;# TODO make me configurable + [my object] set_property -new 1 anon_instances $anon_instances + # for mixed test sections (e.g. text interaction and mc), we have + # to combine the values of the items + [my object] set_property -new 1 auto_correct true ;# should be computed + [my object] set_property -new 1 has_solution true ;# should be computed + #my msg "fc=$fc" + } +} Index: openacs-4/packages/xowf/tcl/xowf-form-field-procs.tcl =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/tcl/xowf-form-field-procs.tcl,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/tcl/xowf-form-field-procs.tcl 28 Aug 2014 08:24:56 -0000 1.1 @@ -0,0 +1,354 @@ +::xo::library doc { + XoWiki Workflow - form field procs + + @author Gustaf Neumann + @creation-date 2008-03-05 +} + +::xo::db::require package xowiki + +namespace eval ::xowiki::formfield { + ########################################################### + # + # ::xowiki::formfield::workflow_definition + # + ########################################################### + + Class workflow_definition -superclass textarea -parameter { + {rows 20} + {cols 80} + {dpi 120} + } -extend_slot_default validator workflow + + workflow_definition instproc as_graph {} { + set ctx [::xowf::Context new -destroy_on_cleanup -object [my object] \ + -all_roles true -in_role none \ + -workflow_definition [my value] ] + return [$ctx as_graph -dpi [my dpi] -style "max-width: 35%;"] + } + + workflow_definition instproc check=workflow {value} { + # Do we have a syntax error in the workflow definition? + if {![catch {set ctx [::xowf::Context new -destroy_on_cleanup -object [my object] \ + -all_roles true \ + -workflow_definition [my value]]} errorMsg]} { + unset errorMsg + array set "" [$ctx check] + if {$(rc) == 1} {set errorMsg $(errorMsg)} + } + + if {[info exists errorMsg]} { + #my msg errorMsg=$errorMsg + my uplevel [list set errorMsg $errorMsg] + return 0 + } + return 1 + } + workflow_definition instproc pretty_value {v} { + [my object] do_substitutions 0 + set text [string map [list & "&" < "<" > ">" \" """ ' "'" @ "@"] [my value]] + return "
+
$text
+
[my as_graph]
+ [[my object] include my-refers] + " + } + + + ########################################################### + # + # ::xowiki::formfield::current_state + # + ########################################################### + Class current_state -superclass label -parameter { + {as_graph true} + } + current_state instproc render_input {} { + next + if {[my as_graph]} { + set ctx [::xowf::Context new -destroy_on_cleanup -object [my object] \ + -all_roles true -in_role none \ + -workflow_definition [[my object] wf_property workflow_definition] ] + #set ctx [::xowf::Context require [my object]] + set graph [$ctx as_graph -current_state [my value] -visited [[my object] visited_states]] + ::html::div -style "width: 35%; float: right;" { + ::html::t -disableOutputEscaping $graph + } + } + } + + current_state instproc pretty_value {v} { + set g "" + if {[my as_graph]} { + set ctx [::xowf::Context require [my object]] + set graph [$ctx as_graph -current_state $v -visited [[my object] visited_states]] + set g "
$graph
" + } + return "[next]$g" + } + +} + + +# +# these definitons are only here for the time being +# +namespace eval ::xo::role { + Class Role + Role instproc get_members args { + error "get_members are not implemented for [self]" + } + Role instproc get_object_id {object} { + return [$object package_id] + } + + Role create all + all proc is_member {-user_id:required -package_id} { + return 1 + } + + Role create swa + swa proc is_member {-user_id:required -package_id} { + return [::xo::cc cache [list acs_user::site_wide_admin_p -user_id $user_id]] + } + + Role create registered_user + registered_user proc is_member {-user_id:required -package_id} { + return [expr {$user_id != 0}] + } + registered_user proc get_members {-object_id:required} { + # return just the users with an @ sign, to avoid the users created by automated testing + set members [::xo::dc list_of_lists get_users \ + "select distinct username, user_id from registered_users where username like '%@%'"] + return $members + } + + Role create unregistered_user + unregistered_user proc is_member {-user_id:required -package_id} { + return [expr {$user_id == 0}] + } + + Role create admin + admin proc is_member {-user_id:required -package_id:required} { + return [::xo::cc permission -object_id $package_id -privilege admin -party_id $user_id] + } + admin proc get_members {-object_id:required} { + set members [xo::dc list_of_lists get_admins "select distinct o.title, p.party_id + from acs_object_party_privilege_map p, acs_objects o + where p.object_id = :object_id and p.privilege = 'admin' and o.object_id = p.party_id"] + #my msg members=$members + return $members + } + + Role create creator + creator proc is_member {-user_id:required -package_id -object:required} { + $object instvar creation_user + return [expr {$creation_user == $user_id}] + } + creator proc get_object_id {object} {return [$object item_id]} + creator proc get_members {-object_id:required} { + set creator_id [xo::dc get_value get_owner "select o.creation_user + from acs_objects o + where object_id = :object_id"] + return [list [list [::xo::get_user_name $creator_id] $creator_id]] + } + + Role create app_group_member + app_group_member proc is_member {-user_id:required -package_id} { + return [::xo::cc cache [list application_group::contains_party_p \ + -party_id $user_id \ + -package_id $package_id]] + } + + Role create community_member + community_member proc is_member {-user_id:required -package_id} { + if {[info commands ::dotlrn_community::get_community_id] ne ""} { + set community_id [my cache [list [dotlrn_community::get_community_id -package_id $package_id]]] + if {$community_id ne ""} { + return [my cache [list dotlrn::user_is_community_member_p \ + -user_id $user_id \ + -community_id $community_id]] + } + } + return 0 + } + +} + + + + +namespace eval ::xowiki::formfield { + + ########################################################### + # + # ::xowiki::formfield::role_member + # + ########################################################### + + Class role_member -superclass candidate_box_select -parameter { + role + {online_state off} + } + role_member instproc initialize {} { + next + my set is_party_id 1 + } + role_member instproc render_input {} { + my instvar role + #my msg role=$role,obj=[my object] + if {[info commands ::xo::role::$role] ne ""} { + set object_id [::xo::role::$role get_object_id [my object]] + my set options [::xo::role::$role get_members -object_id $object_id] + } elseif {[set gid [group::get_id -group_name $role]] ne ""} { + my set options [list] + foreach m [group::get_members -group_id $gid] { + my lappend options [list [::xo::get_user_name $m] $m] } + } else { + error "no such role or group '$role'" + } + next + } + + role_member instproc get_entry_label {v} { + set prefix "" + if {[my online_state]} { + set prefix "[::xowiki::utility user_is_active -asHTML true $v] " + } + return $prefix[::xo::get_user_name $v] + } + + role_member instproc pretty_value {v} { + my set options [my get_labels $v] + next + } +} + +namespace eval ::xowiki::formfield { + + ########################################################### + # + # ::xowiki::formfield::mc_exercise + # + ########################################################### + + Class mc_exercise -superclass CompoundField -parameter { + {feedback full} + {inplace true} + } + + mc_exercise instproc initialize {} { + my log "[self class] deprecated, you should switch to test-item procs" + if {[my set __state] ne "after_specs"} return + set javascript [::xowiki::formfield::FormField fc_encode { + xinha_config.toolbar = [ + ['popupeditor', 'bold','italic','createlink','insertimage','separator'], + ['killword','removeformat','htmlmode'] + ]; + }] + my instvar feedback inplace + my create_components [subst { + {text {richtext,required,editor=xinha,height=150px,label=#xowf.exercise-text#,plugins=OacsFs,javascript=$javascript,inplace=$inplace}} + {alt-1 {mc_alternative,feedback=$feedback,label=#xowf.alternative#,inplace=$inplace}} + {alt-2 {mc_alternative,feedback=$feedback,label=#xowf.alternative#,inplace=$inplace}} + {alt-3 {mc_alternative,feedback=$feedback,label=#xowf.alternative#,inplace=$inplace}} + {alt-4 {mc_alternative,feedback=$feedback,label=#xowf.alternative#,inplace=$inplace}} + {alt-5 {mc_alternative,feedback=$feedback,label=#xowf.alternative#,inplace=$inplace}} + }] + my set __initialized 1 + } + + mc_exercise instproc render_input {} { + ::xo::Page requireCSS /resources/xowf/myform.css + next + } + + mc_exercise instproc pretty_value {v} { + return [[my object] property form ""] + } + + mc_exercise instproc convert_to_internal {} { + # + # Build a from from the componets of the exercise on the fly. + # Actually, this methods computes the properties "form" and + # "form_constraints" based on the components of this form field. + # + set form "
\n\n" + set fc "@categories:off @cr_fields:hidden\n" + set intro_text [my get_named_sub_component_value text] + append form "\n" + foreach input_field_name {alt-1 alt-2 alt-3 alt-4 alt-5} { + foreach f {text correct feedback_correct feedback_incorrect} { + set value($f) [my get_named_sub_component_value $input_field_name $f] + } + append form \ + "\n" \ + "\n" + set alt_fc [list] + if {$value(correct)} {lappend alt_fc "answer=on"} else {lappend alt_fc "answer="} + if {$value(feedback_correct) ne ""} { + lappend alt_fc "feedback_answer_correct=[::xowiki::formfield::FormField fc_encode $value(feedback_correct)]" + } + if {$value(feedback_incorrect) ne ""} { + lappend alt_fc "feedback_answer_incorrect=[::xowiki::formfield:::FormField fc_encode $value(feedback_incorrect)]" + } + if {[llength $alt_fc] > 0} {append fc [list $input_field_name:checkbox,[join $alt_fc ,]]\n} + #my msg "$input_field_name .correct = $value(correct)" + } + append form "
$intro_text
$value(text)
\n" + [my object] set_property -new 1 form $form + [my object] set_property -new 1 form_constraints $fc + } + + ########################################################### + # + # ::xowiki::formfield::mc_alternative + # + ########################################################### + + Class mc_alternative -superclass CompoundField -parameter { + {feedback full} + {inplace true} + } + + mc_alternative instproc initialize {} { + my log "[self class] deprecated, you should switch to test-item procs" + if {[my set __state] ne "after_specs"} return + + if {1} { + set javascript [::xowiki::formfield::FormField fc_encode { + xinha_config.toolbar = [ + ['popupeditor', 'bold','italic','createlink','insertimage','separator'], + ['killword','removeformat','htmlmode'] + ]; + }] + set text_config [subst {editor=xinha,height=100px,label=Text,plugins=OacsFs,inplace=[my inplace],javascript=$javascript}] + } else { + set text_config [subst {editor=wym,height=100px,label=Text}] + } + if {[my feedback] eq "full"} { + set feedback_fields { + {feedback_correct {textarea,label=Feedback korrekt}} + {feedback_incorrect {textarea,label=Feedback inkorrekt}} + } + } else { + set feedback_fields "" + } + my create_components [subst { + {text {richtext,$text_config}} + {correct {boolean,horizontal=true,label=Korrekt}} + $feedback_fields + }] + my set __initialized 1 + } + +} + +::xo::library source_dependent + +# +# Local variables: +# mode: tcl +# tcl-indent-level: 2 +# indent-tabs-mode: nil +# End: Index: openacs-4/packages/xowf/tcl/xowf-includelet-procs.tcl =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/tcl/xowf-includelet-procs.tcl,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/tcl/xowf-includelet-procs.tcl 28 Aug 2014 08:24:56 -0000 1.1 @@ -0,0 +1,138 @@ +::xo::library doc { + XoWiki Workflow - includelet definitions + + @author Gustaf Neumann + @creation-date 2008-03-05 +} + +::xo::db::require package xowiki + + +namespace eval ::xowiki::includelet { + # + # Define additional elements for includelets + # + Class form-menu-button-wf-instances -superclass ::xowiki::includelet::form-menu-button-answers + Class form-menu-button-wf -superclass form-menu-button -parameter { + {method view} + } + + # + # Create an includelet called wf-todo, which lists the todo items + # for a user_id from a single or multiple worflows) + # + ::xowiki::IncludeletClass create wf-todo \ + -superclass ::xowiki::Includelet \ + -parameter { + {__decoration plain} + {parameter_declaration { + {-workflow ""} + {-user_id} + {-ical 0} + {-max_entries} + }} + } + + wf-todo instproc initialize {} { + my get_parameters + if {![info exists user_id]} {set user_id [::xo::cc user_id]} + + set sql { + select assignee,xowiki_form_page_id,state,i.publish_status,page_template, + p.creation_date, p.last_modified, p,description, + i2.name as wf_name,p.title,i.name,i.parent_id,o.package_id as pid + from xowiki_form_pagei p,cr_items i, cr_items i2, acs_objects o + where (assignee = :user_id or acs_group__member_p(:user_id,assignee, 'f')) + and i.live_revision = xowiki_form_page_id + and p.page_template = i2.item_id + and o.object_id = xowiki_form_page_id + } + if {$workflow ne ""} { + # The workflow might be of one of the following forms: + # name + # :name + # /path-in-package/:name + # //package/:name + # //package/path-in-package/:name + # + # To address all workflow of a package instance, use + # //package/ + # + if {[regexp {^/(/.*)/$} $workflow _ package]} { + # all workflows from this package + ::xowf::Package initialize -url $package + #my msg "using package_id=$package_id" + append sql " and o.package_id = :package_id" + } else { + if {[regexp {^/(/[^/]+)(/.+)$} $workflow _ package path]} { + ::xowf::Package initialize -url $package + #my msg "using package_id=$package_id" + } else { + set path $workflow + } + set parent_id [[my set __including_page] parent_id] + set wf_page [$package_id get_page_from_item_ref -parent_id $parent_id $path] + if {$wf_page eq ""} { + my msg "cannot resolve page $workflow" + set package_id -1; set page_template -1 + } else { + set page_template [$wf_page item_id] + set package_id [$wf_page package_id] + } + #my msg "page_template=$page_template pkg=$package_id" + append sql " and o.package_id = :package_id and p.page_template = :page_template" + } + } + + append sql " order by p.last_modified desc" + + my set items [::xowiki::FormPage instantiate_objects -sql $sql] + } + + wf-todo instproc render_ical {} { + my instvar items + foreach i [$items children] { + $i instvar wf_name name title state xowiki_form_page_id pid description parent_id + ::xowf::Package initialize -package_id $pid + + $i class ::xo::ical::VTODO + $i configure -uid $pid-$xowiki_form_page_id \ + -url [$pid pretty_link -absolute true $parent_id $name] \ + -summary "$title ($state)" \ + -description "Workflow instance of workflow $wf_name $description" + } + $items mixin ::xo::ical::VCALENDAR + $items configure -prodid "-//WU Wien//NONSGML XoWiki Content Flow//EN" + set text [$items as_ical] + #my log "--ical sending $text" + #ns_return 200 text/calendar $text + ns_return 200 text/plain $text + } + + wf-todo instproc render {} { + my get_parameters + if {$ical} {return [my render_ical]} + + my instvar items + set t [TableWidget new -volatile \ + -columns { + Field package -label Package + AnchorField wf -label Workflow + AnchorField title -label "Todo" + Field state -label [::xowiki::FormPage::slot::state set pretty_name] + }] + foreach i [$items children] { + $i instvar wf_name name title state xowiki_form_page_id pid parent_id + ::xowf::Package initialize -package_id $pid + $t add \ + -wf $wf_name \ + -wf.href [$pid pretty_link -parent_id $parent_id $wf_name] \ + -title $title \ + -title.href [$pid pretty_link -parent_id $parent_id $name] \ + -state $state \ + -package [$pid package_url] + } + return [$t asHTML] + } + +} Index: openacs-4/packages/xowf/tcl/xowf-init.tcl =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/tcl/xowf-init.tcl,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/tcl/xowf-init.tcl 28 Aug 2014 08:24:56 -0000 1.1 @@ -0,0 +1,9 @@ +if {[info commands ::ns_cache_eval] eq ""} {proc ::ns_cache_eval {args} {::ns_cache eval {*}$args}} +# register the dav interface for the todos +::xowf::dav-todo register + +# run the checker for the scheduled at-jobs +ad_schedule_proc -thread t 60 ::xowf::atjob check + +# make sure, we have not missed some at-jobs, while we were down +::xowf::atjob check -with_older true \ No newline at end of file Index: openacs-4/packages/xowf/tcl/xowf-procs.tcl =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/tcl/xowf-procs.tcl,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/tcl/xowf-procs.tcl 28 Aug 2014 08:24:56 -0000 1.1 @@ -0,0 +1,1788 @@ +::xo::library doc { + XoWiki Workflow - main library classes and objects + + @author Gustaf Neumann + @creation-date 2008-03-05 +} + +# Todo: +# - after import, references are not updated +# (same for plain references); after_import methods? +# +# - Roles +# - assignment +# - workflow-assingnment includelet (over multiple workflows and +# package instances) + +::xo::db::require package xowiki +::xo::library require -package xowiki xowiki-procs + +namespace eval ::xowf { + ::xo::PackageMgr create ::xowf::Package \ + -package_key "xowf" -pretty_name "XoWiki Workflow" \ + -superclass ::xowiki::Package + + Package ad_instproc initialize {} { + mixin ::xowf::WorkflowPage to every FormPage + } { + # This method is called, whenever an xowf package is initialized. + next + #my msg "::xowiki::FormPage instmixin add ::xowf::WorkflowPage" + ::xowiki::FormPage instmixin add ::xowf::WorkflowPage + } + + Package instproc call {object method options} { + if {[$object istype ::xowiki::FormPage]} { + if {[$object is_wf_instance]} { + set ctx [::xowf::Context require $object] + #my msg "wfi: creating context form object $object = $ctx, chlds=[$ctx info children]" + #my msg "wfi: o $object has mixins [$object info mixin]" + } elseif {[$object is_wf]} { + set ctx [::xowf::Context require $object] + #my msg "wf: creating context form object $object = $ctx, chlds=[$ctx info children]" + #my msg "wf: o $object has mixins [$object info mixin]" + } + } + next + } + + Package ad_instproc destroy {} { + remove mixin + } { + # + # In general, it is possible, that multiple xowf packages are + # concurrently active in one thread. We want to remove the mixin + # only, when the last instance is deleted. + # + if {[llength [[self class] allinstances]] == 1} { + ::xowiki::FormPage instmixin delete ::xowf::WorkflowPage + } + next + } + + # Package instproc delete {-item_id -name} { + # # Provide a method to delete the foreign key references, when + # # an item for an atjob is deleted. We do here the same magic + # # as in ::xowiki::Package to obtain the item_id + # if {![info exists item_id]} {set item_id [my query_parameter item_id]} + # if {$item_id ne ""} { + # db_dml dbqd..xowf_delete "delete from xowf_atjob where owner_id = :item_id" + # } + # next + # } + + # + # Workflow Context + # + + Class create Context -parameter { + {current_state "[self]::initial"} + workflow_definition + object + {all_roles false} + {default_definition ""} + in_role + } + + # forward property to the workflow object + Context instforward property {%my object} %proc + Context instforward get_property {%my object} %proc + Context instforward set_property {%my object} %proc + Context instforward set_new_property {%my object} set_property -new 1 + + # forward form_constraints, view_method and for the to current state object + Context instforward get_form_constraints {%my current_state} form_constraints + Context instforward get_view_method {%my current_state} view_method + Context instforward form {%my current_state} form + Context instforward form_loader {%my current_state} form_loader + + Context instproc set_current_state {value} { + my current_state [self]::$value + } + Context instproc get_current_state {} { + namespace tail [my current_state] + } + + Context instproc get_actions {} { + set actions [list] + foreach action [[my current_state] get_actions] { + lappend actions [self]::$action + } + #my msg "for [my current_state] actions '$actions" + return $actions + } + Context instproc defined {what} { + set result [list] + foreach c [my info children] {if {[$c istype $what]} {lappend result $c}} + return $result + } + + Context instproc resolve_form_name {-object:required name} { + set package_id [$object package_id] + set parent_id [$object parent_id] + array set "" [$package_id item_ref -normalize_name false \ + -use_package_path 1 \ + -default_lang "" \ + -parent_id $parent_id \ + $name] + return [list form_id $(item_id) name $(prefix):$(stripped_name)] + } + + Context instproc default_form_loader {form_name} { + #my msg "resolving $form_name in state [my current_state] via default form loader" + set form_id 0 + if {$form_name ne ""} { + my instvar object + array set "" [my resolve_form_name -object $object $form_name] + set form_id $(form_id) + #my msg ".... object $object ==> id = $form_id" + } + return $form_id + } + + + Context instproc form_object {object} { + set parent_id [$object parent_id] + # After this method is activated, the form object of the form of + # the current state is created and the instance variable form_id + # is set. + # + # Load the actual form only once for this context. We cache the + # object name of the form in the context. + # + if {[my exists form_id]} {return [my set form_id]} + # + # We have to load the form, maybe via a form loader. If the + # form_loader is set and the method exists, then use the form + # loader instead of the plain lookup. In case the form_loader + # fails, it is supposed to return 0. + # + set loader [my form_loader] + + # TODO why no procsearch instead of "info methods"? + if {$loader eq "" || [my info methods $loader] eq ""} { + set form_id [my default_form_loader [[my current_state] form]] + } else { + #my msg "using custom form loader $loader for [my form]" + set form_id [my $loader [my form]] + } + #my msg form_id=$form_id + set package_id [$object package_id] + + # When no form was found by the form loader ($form_id == 0) we + # create a form on the fly. The created form can be influenced by + # "auto_form_template" and "auto_form_constraints". + if {$form_id == 0} { + set vars [dict keys [$object set instance_attributes]] + if {[my exists auto_form_template]} { + set template [my set auto_form_template] + my log "USE autoform template" + } elseif {[llength $vars] == 0} { + #set template "AUTO form, no instance variables defined,
@_text@" + set template "@_text@" + } else { + set template "@[join $vars @,@]@
@_text@" + } + #my log "USE auto-form template=$template, vars=$vars IA=[$object set instance_attributes], V=[$object info vars] auto [expr {[my exists autoname] ? [my set autoname] : "f"}]" + + if {[my exists auto_form_constraints]} { + set fc [my set auto_form_constraints] + } else { + set fc "" + } + set form_id [::xowiki::Form new -destroy_on_cleanup \ + -package_id $package_id \ + -parent_id [$package_id folder_id] \ + -name "Auto-Form" \ + -anon_instances [expr {[my exists autoname] ? [my set autoname] : "f"}] \ + -form {} \ + -text [list $template text/html] \ + -form_constraints $fc] + } else { + # Be sure, to instantiate the form object + if {![my isobject ::$form_id]} { + ::xo::db::CrClass get_instance_from_db -item_id $form_id + } + #my msg form_id=$form_id,[$form_id info class] + + set form_id ::$form_id + if {[$form_id info class] eq "::xowiki::Form"} { + # The item returned from the form loadeder was a form, + # everything is fine. + } elseif {[$form_id info class] eq "::xowiki::FormPage"} { + # We got an FormPage. This formpage might be already a pseudo + # form (containing property "form"). In this case, we are done as well. + + if {[$form_id property form] eq ""} { + # The FormPage contains no form, so try to provide one. We + # obtain the content by rendering the page_content. In some + # cases it might be more efficient to obtain the content + # from property "_text", but this might lead to unexpected + # cases where the formpage uses _text for partial + # information. + set text [$form_id render_content] + $form_id set_property -new 1 form "
$text
" + #my msg "_text=[$form_id property _text]" + } + } elseif {[$form_id info class] eq "::xowiki::Page"} { + #my msg "creating form" + set form_id [::xowiki::Form new -destroy_on_cleanup \ + -package_id $package_id \ + -parent_id [$package_id folder_id] \ + -name "Auto-Form" \ + -anon_instances [expr {[my exists autoname] ? [my set autoname] : "f"}] \ + -form "
[$form_id get_html_from_content [$form_id text]]
" \ + -text "" \ + -form_constraints ""] + } + } + my set form_id $form_id + } + + Context instproc init {} { + my destroy_on_cleanup + my array set handled_roles {} + # + # We define the classes Action, State and Property per workflow + # instance. This has the advantage that we can provide instprocs + # or parameters per workflow definition without the danger to + # interfere with other Workflows + # + regsub -all \r\n [my workflow_definition] \n workflow_definition + + if {[catch {my contains " + Class create Action -superclass ::xowf::Action + Class create State -superclass ::xowf::State + Class create Condition -superclass ::xowf::Condition + Class create Property -superclass ::xowf::Property -set abstract 1 + [my default_definition] + $workflow_definition"} errorMsg]} { + my msg "Error in workflow definition: $errorMsg" + #my msg [[my object] instance_attributes] + #array set __ia [[my object] instance_attributes] + #catch {unset __ia(workflow_definition)} + #catch {[my object] unset __ia(workflow_definition)} + #catch {[my object] unset __wf(workflow_definition)} + #[my object] instance_attributes [array get __ia] + #[my object] save + #my msg [[my object] serialize] + } + + if {[my all_roles]} { + #my msg want.to.create=[my array names handled_roles] + foreach role [my array names handled_roles] { + Context create [self]-$role -workflow_definition $workflow_definition \ + -in_role $role -object [my object] + } + } + } + + Context proc require {obj} { + set ctx $obj-wfctx + if {![my isobject $ctx]} { + set wfContextClass [$obj wf_property workflow_context_class [self]] + $wfContextClass create $ctx -destroy_on_cleanup -object $obj \ + -workflow_definition [$obj wf_property workflow_definition] + + # set the state to a well defined starting point + set state [$obj state] + if {$state eq ""} {set state initial} + $ctx set_current_state $state + + if {[info commands [$ctx current_state]] eq ""} { + # The state was probably deleted from the workflow definition, + # but the workflow instance does still need it. We complain an + # reset the state to initial, which should be always present. + $obj msg "Workflow instance [$obj name] is in an undefined state $state, reset to initial" + set state initial + $ctx set_current_state $state + } + + set stateObj [$ctx current_state] + catch {$stateObj eval [$stateObj eval_when_active]} + + # set the embedded context to the workflow context, + # used e.g. by "behavior" of form-fields + [[$obj package_id] context] set embedded_context $ctx + + if {[$obj istype ::xowiki::FormPage] && [$obj is_wf_instance]} { + # + # The workflow instance may have the following variables: + # - "debug" + # - "policy" + # - "autoname" + # - "auto_form_constraints" + # - "auto_form_template" + # + if {[$ctx exists debug] && [$ctx set debug]>0} { + $ctx show_debug_info $obj + } + # anything to do with autoname? + #if {[$ctx exists autoname]} { + # my log obj=[$obj info class],obj=$obj + # $obj set_property -new 1 anon_instances t + #} + + if {[$ctx exists policy]} { + set policy [$ctx set policy] + if {![my isobject $policy]} { + my msg "ignore non-existent policy '$policy'" + } else { + [$obj package_id] set policy $policy + #my msg new-pol=[[$obj package_id] set policy ] + } + } + } + } + return $ctx + } + + Context instproc show_debug_info {obj} { + set stateObj [my current_state] + set form [$stateObj form] + set view_method [my get_view_method] + set form_loader [my form_loader] + if {$form eq ""} {set form NONE} + if {$view_method eq ""} {set view_method NONE} + if {$form_loader eq ""} {set form_loader NONE} + + $obj debug_msg "State: [$stateObj name], Form: $form,\ + View method: $view_method, Form loader: $form_loader,\ + Context class: [my info class]" + + set conds [list] + foreach c [my defined Condition] { + lappend conds "[$c name] [$c]" + } + $obj debug_msg "Conditions: [join $conds {, }]" + $obj debug_msg "Instance attributes: [list [$obj instance_attributes]]" + } + + Context instproc draw_arc {from_state next_state action label style} { + if {$next_state eq ""} {set next_state $from_state} + set key transition($from_state,$next_state,$action) + if {[my exists $key]} {return ""} + my set $key 1 + return " state_$from_state -> state_$next_state \[label=\"$label\"$style\];\n" + } + Context instproc draw_transition {from action role} { + #my msg "[self args]" + + if {[$action state_safe]} { + set arc_style {,style="dashed",penwidth=1,color=gray} + } else { + set arc_style "" + } + set cond_values [$action get_cond_values [$action next_state]] + set result "" + if {[llength $cond_values]>2} { + # we have conditional values + set c cond_[$from name]_[my incr condition_count] + append arc_style {,style="setlinewidth(1)",penwidth=1,color=gray} + append result " state_$c \[shape=diamond, fixedsize=1, width=0.2, height=0.2, fixedsize=1,style=solid,color=gray,label=\"\"\];\n" + append result [my draw_arc [$from name] $c [$action name]-1 $role[$action label] ""] + foreach {cond value} $cond_values { + if {$cond ne ""} {set prefix "$cond"} {set prefix "else"} + append result [my draw_arc $c $value [$action name] \[$prefix\] $arc_style] + } + } else { + set prefix "" + append result [my draw_arc [$from name] [lindex $cond_values 1] [$action name] $role$prefix[$action label] $arc_style] + } + return $result + } + + Context instproc as_graph {{-current_state ""} {-visited ""} {-dpi 96} {-style "width:100%"}} { + set dot "" + catch {set dot [::util::which dot]} + # final ressort for cases, where ::util::which is not available + if {$dot eq "" && [file executable /usr/bin/dot]} {set dot /usr/bin/dot} + if {$dot eq ""} {return "Program 'dot' is not available! No graph displayed."} + set obj_id [namespace tail [my object]] + set result [subst {digraph workflow_$obj_id \{ + dpi = $dpi; + node \[shape=doublecircle, margin=0.001, fontsize=8, fixedsize=1, width=0.4, style=filled\]; start; + node \[shape=ellipse, fontname="Courier", color=lightblue2, style=filled, + fixedsize=0, fontsize=10, margin=0.06\]; + edge \[fontname="Courier", fontsize=9\]; + }] + foreach s [my defined State] { + if {[$s name] eq $current_state} { + set color ",color=orange" + } elseif {[$s name] in $visited} { + set color ",color=yellow" + } else { + set color "" + } + append result " state_[$s name] \[label=\"[$s label]\"$color\];\n" + } + if {[my isobject [self]::initialize]} { + append result "start->state_initial \[label=\"[[self]::initialize label]\"\];\n" + } else { + append result "start->state_initial;\n" + } + + my set condition_count 0 + foreach s [my defined State] { + foreach a [$s get_actions -set true] { + append result [my draw_transition $s [self]::$a ""] + set drawn([self]::$a) 1 + } + foreach role [$s set handled_roles] { + set role_ctx [self]-$role + #my msg exists?role=$role->[self]-$role->[info command [self]-$role] + if {[info command ${role_ctx}::[$s name]] ne ""} { + foreach a [${role_ctx}::[$s name] get_actions] { + append result [my draw_transition $s ${role_ctx}::$a "$role:"] + } + } + } + } + # + # State-safe actions might be called from every state. Draw the + # arcs if not done yet. + # + foreach action [my defined Action] { + if {[info exists drawn($action)]} {continue} + if {[$action state_safe]} { + foreach s [my defined State] { + append result [my draw_transition $s $action ""] + } + } + } + + append result "\}\n" + set package_id [[my object] package_id] + set path [acs_package_root_dir [$package_id package_key]]/www/ + set fn $path/g.dot + set ofn dot-$obj_id.png + set f [open $fn w]; fconfigure $f -encoding utf-8; puts $f $result; close $f + if {[catch {exec $dot -Tpng $fn -o $path/$ofn} errorMsg]} { + my msg "Error during execution of $dot: $errorMsg" + } + file delete $fn + return "\n" + } + + Context instproc check {} { + # Check minimal contents + if {![my isobject [self]::initial] || ![[self]::initial istype State]} { + return [list rc 1 errorMsg "No State 'initial' defined"] + } + # ease access to workflow constructs + foreach s [my defined State] {set state([$s name]) $s} + foreach a [my defined Action] {set action([$a name]) $a} + foreach a [my defined Condition] {set condition([$a name]) $a} + array set condition {else 1 true 1 default 1} + # Check actions + foreach a [my defined Action] { + # Are some "next_states" undefined? + foreach {cond value} [$a get_cond_values [$a next_state]] { + if {$cond ne "" && ![info exists condition($cond)]} { + return [list rc 1 errorMsg "Error in action [$a name]: no such condition '$cond' defined \ + (valid: [lsort [array names condition]])"] + } + if {$value ne "" && ![info exists state($value)]} { + return [list rc 1 errorMsg "Error in action [$a name]: no such state '$value' defined \ + (valid: [lsort [array names state]])"] + } + } + } + foreach s [my defined State] { + # Are some "actions" undefined? + foreach {cond actions} [$s get_cond_values [$s actions]] { + foreach a $actions { + if {![info exists action($a)]} { + return [list rc 1 errorMsg "Error in state [$s name]: no such action '$a' defined \ + (valid: [lsort [array names action]])"] + } + } + } + if {[$s form_loader] eq "" && [$s form] ne ""} { + my set forms([$s form]) 1 + } + } + foreach p [my defined ::xowiki::formfield::FormField] { + if {[$p exists parampage]} {my set parampages([$p set parampage]) 1} + } + + #my msg "forms=[my array names forms], parampages=[my array names parampages] in-role [my exists in_role] [my array names handled_roles]" + + if {![my exists in_role]} { + foreach role [my array names handled_roles] { + set role_ctx [self]-$role + if {[my isobject $role_ctx]} { + array set "" [$role_ctx check] + if {$(rc) == 1} {return [array get ""]} + my array set forms [$role_ctx array get forms] + my array set parampage [$role_ctx array get parampage] + } + } + #my msg "forms=[my array names forms], parampages=[my array names parampages]" + set page [my object] + $page set unresolved_references 0 + $page set __unresolved_references [list] + $page set __unresolved_object_type ::xowiki::Form + foreach {type pages} [list wf_form [my array names forms] wf_parampage [my array names parampages]] { + foreach p $pages { + array set "" [my resolve_form_name -object $page $p] + set l [::xowiki::Link new -volatile -page $page -type $type -name $(name) -item_id $(form_id)] + # render does the optional fetch of the names, and maintains the + # variable references of the page object (similar to render). + set link_text [$l render] + } + } + #my msg "-- link_text=$link_text// refs?[$page exists references]" + if {[$page exists references]} { + #my msg "updating references refs=[$page set references]" + $page references_update [lsort -unique [$page set references]] + $page set __extra_references [$page set references] + $page unset references + } + if {[llength [$page set __unresolved_references]] > 0} { + # TODO: we should provide a link to create the missing forms. maybe we + # change unresolved_references to a list..., or maybe we write these into the DB. + my msg -html t "Missing forms: [join [$page set __unresolved_references] {, }]" + } + } + return [list rc 0] + } +} + +namespace eval ::xowf { + # + # WorkflowConstruct, the base class for workflow definitions + # + Class create WorkflowConstruct -parameter { + {handled_roles [list]} + {label "[namespace tail [self]]"} + {name "[namespace tail [self]]"} + } + #WorkflowConstruct ad_instforward property {get property} {%[my info parent] object} %proc + #WorkflowConstruct ad_instforward set_property {set property} {%[my info parent] object} %proc + + WorkflowConstruct instforward property {%[my info parent] object} %proc + WorkflowConstruct instforward set_property {%[my info parent] object} %proc + WorkflowConstruct instforward set_new_property {%[my info parent] object} set_property -new 1 + WorkflowConstruct instforward object {%my info parent} object + + WorkflowConstruct instproc in_role {role configuration} { + set ctx [my info parent] + set obj [$ctx object] + #my msg parent=$obj,cl=[$obj info class],name=[$obj name] + if {[$ctx exists in_role]} { + set success [expr {[$ctx in_role] eq $role}] + } else { + set success [$obj check_role $role] + } + #my msg role-$role->$success + my lappend handled_roles $role + $ctx set handled_roles($role) 1 + if {$success} { + my configure {*}$configuration + } + } + # WorkflowConstruct instproc get_condition {conditional_entry} { + # set e [split $conditional_entry :?] + # if {[llength $e]==2} {return [list cond [lindex $e 0] value [lindex $e 1]]} + # return [list cond "" value $conditional_entry] + # } + + WorkflowConstruct instproc get_cond_values {values} { + if {[lindex $values 0] eq "?"} { + return [lrange $values 1 end] + } elseif {$values eq ""} { + return "" + } else { + if {[regexp {^(.+):([^ ]+) } $values _ cond value]} { + my msg "switch '$values' to new syntax: ? $cond $value ..." + } + return [list "" $values] + } + } + WorkflowConstruct instproc get_value {values} { + foreach {cond value} [my get_cond_values $values] { + if {$cond eq "" || $cond eq "default" || $cond eq "else" || + $cond eq "true"} { + return $value + } elseif {[my $cond]} { + return $value + } + } + } + WorkflowConstruct instproc get_value_set {values} { + set result [list] + foreach {cond value} [my get_cond_values $values] { + foreach v $value {lappend result $v} + } + return [lsort -unique $result] + } +} + +namespace eval ::xowf { + WorkflowConstruct instforward true set "" 1 + WorkflowConstruct instforward false set "" 0 + + proc ? {cmd expected {msg ""}} { + ::xo::Timestamp t1 + set r [uplevel $cmd] + if {$msg eq ""} {set msg $cmd} + if {$r ne $expected} { + regsub -all \# $r "" r + append ::_ "Error: $msg returned \n'$r' ne \n'$expected'\n" + } else { + append ::_ "$msg - passed ([t1 diff] ms)\n" + } + } + + # + # some test cases + # + WorkflowConstruct x + set ::_ "" + ? {x get_value ""} "" + ? {x get_value a} a + ? {x get_value {a b}} {a b} + ? {x get_value "a b"} {a b} + ? {x get_value "? true a default b"} {a} + ? {x get_value "? false a default b"} {b} + ? {x get_value "? true {a b} default {b c}"} {a b} + ? {x get_value "? false {a b} default {b c}"} {b c} + ? {x get_value_set "? false {a b} default {b c}"} {a b c} + ns_log notice "--Test returns $::_" +} + + +namespace eval ::xowf { + + Class create State -superclass WorkflowConstruct -parameter { + {actions ""} + {view_method ""} + {form ""} + {form_loader ""} + {form_constraints ""} + {assigned_to} + {eval_when_active ""} + {extra_js ""} + {extra_css ""} + } + + State instproc get_actions {{-set false}} { + if {!$set} { + return [my get_value [my actions]] + } else { + return [my get_value_set [my actions]] + } + } + State instproc get_all_actions {} { + return [my get_value [my actions]] + } + + Class create Condition -superclass WorkflowConstruct -parameter expr + Condition instproc init {} { + [my info parent]::Action instforward [namespace tail [self]] [self] + [my info parent]::State instforward [namespace tail [self]] [self] + } + Condition instproc defaultmethod {} { + [my info parent] instvar {object obj} + expr [my expr] + } + + #{label "#xowf.form-button-[namespace tail [self]]#"} + Class create Action -superclass WorkflowConstruct -parameter { + {next_state ""} + {roles all} + {state_safe false} + {title} + } + Action instproc activate {obj} {;} + Action instproc get_next_state {} { + return [my get_value [my next_state]] + } + Action instproc invoke {{-attributes ""}} { + set action_name [namespace tail [self]] + set ctx [my info parent] + set object [$ctx object] + set package_id [$object package_id] + my log "--xowf invoke action [self]" + # We fake a work request with the given instance attributes + set last_context [expr {[$package_id exists context] ? [$package_id context] : "::xo::cc"}] + set last_object [$package_id set object] + set cc [::xo::ConnectionContext new -user_id [$last_context user_id]] + $package_id context $cc + $cc array set form_parameter \ + [list __object_name [$object name] \ + _name [$object name] \ + __form_action save-form-data \ + __form_redirect_method __none \ + __action_$action_name $action_name] + #ns_log notice "call_action pushed form_param to $cc: [$cc array get form_parameter]" + $cc array set form_parameter $attributes + + $package_id set object "[$package_id folder_path -parent_id [$object parent_id]][$object name]" + + #my log "call_action calls: ::$package_id invoke -method edit -batch_mode 1 // obj=[$package_id set object]" + if {[catch {::$package_id invoke -method edit -batch_mode 1} errorMsg]} { + my msg "---call_action returns error $errorMsg" + ns_log error "$errorMsg\n$::errorInfo" + error $errorMsg + } + #my log "RESETTING package_id object" + $package_id set object $last_object + $package_id context $last_context + $cc destroy + + #my log "CHECK batch mode: [$package_id exists __batch_mode]" + if {[$package_id exists __batch_mode]} { + my msg "RESETTING BATCH MODE" + my log "RESETTING BATCH MODE" + $package_id unset __batch_mode + } + return "OK" + } + + Class create Property \ + -superclass ::xowiki::formfield::FormField -parameter {{name "[namespace tail [self]]"}} \ + -parameter {{allow_query_parameter false}} + Property set abstract 1 + + Property instproc init {} { + # + # Mostly compatibility fix for XOTcl 2.0. Provide a default + # property for $object, if the property does not exist in the + # instance attributes, but provided in the Property definition. + # + set object [[my info parent] object] + $object instvar instance_attributes + if {[my exists default] && ![dict exists $instance_attributes [my name]]} { + #$object set __ia([my name]) [my default] + dict set instance_attributes [my name] [my default] + #my msg "[self] set default of $object to [my default]" + } + } + + Property instproc get_default_from {page} { + my set parampage $page + my default [[my info parent] get_property -source $page -name [my name] -default ""] + } + #namespace export State Action Property + + + # + # MixinClass for implementing the workflow definition and workflow instance + # + Class create WorkflowPage + + WorkflowPage ad_instproc is_wf {} { + Check, if the current page is a workflow page (page, defining a workflow) + } { + if {[my exists __wf(workflow_definition)]} {return 1} + if {[my property workflow_definition] ne ""} { + my array set __wf [my instance_attributes] + return 1 + } + return 0 + } + + WorkflowPage ad_instproc is_wf_instance {} { + Check, if the current page is a workflow instance (page, refering to a workflow) + } { + # we cannot call get_template_object here, because this will lead + # to a recursive loop. + set pt [my page_template] + if {![my isobject ::$pt]} { + ::xo::db::CrClass get_instance_from_db -item_id $pt + } + if {[my state] ne "" && [$pt istype ::xowiki::FormPage]} { + my array set __wfi [$pt instance_attributes] + return 1 + } + return 0 + } + + WorkflowPage instproc check_role {role} { + if {[::xo::cc info methods role=$role] eq ""} { + my msg "ignoring unknown role '$role'" + return 0 + } + if {$role eq "creator"} { + # hmm, requires additional attibute + return [::xo::cc role=$role \ + -object [self] \ + -user_id [::xo::cc user_id] \ + -package_id [my package_id]] + } else { + return [::xo::cc role=$role \ + -user_id [::xo::cc user_id] \ + -package_id [my package_id]] + } + } + + WorkflowPage ad_instproc render_form_action_buttons {{-CSSclass ""}} { + Render the defined actions in the current state with submit buttons + } { + if {[my is_wf_instance]} { + ::html::div -class form-button { + set ctx [::xowf::Context require [self]] + foreach action [$ctx get_actions] { + set success 0 + foreach role [$action roles] { + set success [my check_role $role] + if {$success} break + } + if {$success} { + set f [::xowiki::formfield::submit_button new -destroy_on_cleanup \ + -name __action_[namespace tail $action] -CSSclass $CSSclass] + if {[$action exists title]} {$f title [$action title]} + #my msg action=$action + $f value [$action label] + $f render_input + } + } + } + } else { + next + } + } + + WorkflowPage ad_instproc post_process_form_fields {form_fields} { + } { + if {[my exists __feedback_mode]} { + # + # Provide feedback for every alternative + # + foreach f $form_fields { + #my msg "[$f name]: correct? [$f answer_is_correct]" + switch -- [$f answer_is_correct] { + 0 { continue } + -1 { set result "incorrect"} + 1 { set result "correct" } + } + $f form_widget_CSSclass $result + $f set evaluated_answer_result $result + + set feedback "" + if {[$f exists feedback_answer_$result]} { + set feedback [$f feedback_answer_$result] + } else { + set feedback [_ xowf.answer_$result] + } + $f help_text $feedback + } + } + } + + WorkflowPage ad_instproc post_process_dom_tree {dom_doc dom_root form_fields} { + post-process form in edit mode to provide feedback in feedback mode + } { + # In feedback mode, we set the CSS class to correct or incorrect + if {[my exists __feedback_mode]} { + my unset __feedback_mode + ::xo::Page requireCSS /resources/xowf/feedback.css + set form [lindex [$dom_root selectNodes "//form"] 0] + $form setAttribute class "[$form getAttribute class] feedback" + + # + # In cases, where the HTML exercise was given, we process the HTML + # to flag the result. + # + # TODO: what should we do with the feedback. util-user-message not optimal... + # + foreach f $form_fields { + if {[$f exists __rendered]} continue + if {[$f exists evaluated_answer_result]} { + set result [$f set evaluated_answer_result] + foreach n [$dom_root selectNodes "//form//*\[@name='[$f name]'\]"] { + set oldCSSClass [expr {[$n hasAttribute class] ? [$n getAttribute class] : ""}] + $n setAttribute class [string trim "$oldCSSClass [$f form_widget_CSSclass]"] + $f form_widget_CSSclass $result + + set helpText [$f help_text] + if {$helpText ne ""} { + set divNode [$dom_doc createElement div] + $divNode setAttribute class [$f form_widget_CSSclass] + $divNode appendChild [$dom_doc createTextNode $helpText] + [$n parentNode] insertBefore $divNode [$n nextSibling] + util_user_message -message "field [$f name], value [$f value]: $helpText" + } + } + } + } + # + # Provide feedback for the whole exercise + # + if {[my answer_is_correct]} { + set feedback [my get_from_template feedback_correct] + } else { + set feedback [my get_from_template feedback_incorrect] + } + if {$feedback ne ""} { + $dom_root appendFromScript { + html::div -class feedback { + html::t -disableOutputEscaping $feedback + } + } + } + } + } + + WorkflowPage instproc debug_msg {msg} { + #util_user_message -message $msg + ns_log notice "--WF $msg" + catch {ds_comment $msg} + } + + WorkflowPage ad_instproc edit args { + Hook for editing workflow pages + } { + if {[my is_wf_instance]} { + set ctx [::xowf::Context require [self]] + set s [$ctx current_state] + my include_header_info -css [$s extra_css] -js [$s extra_js] + } + next + } + + WorkflowPage ad_instproc view {{content ""}} { + Provide additional view modes: + - edit: instead of viewing a page, it is opened in edit mode + - view_user_input: show user the provided input + - view_user_input_with_feedback: show user the provided input with feedback + } { + # The edit method calls view with an HTML content as argument. + # To avoid a loop, when "view" is redirected to "edit", + # we make sure that we only check the redirect on views + # without content. + + #my msg "view [self args]" + + if {[my is_wf_instance] && $content eq ""} { + set ctx [::xowf::Context require [self]] + set method [$ctx get_view_method] + set s [$ctx current_state] + my include_header_info -css [$s extra_css] -js [$s extra_js] + + if {$method ne "" && $method ne "view"} { + my instvar package_id + #my msg "view redirects to $method in state [$ctx get_current_state]" + switch -- $method { + view_user_input { + #my msg "calling edit with disable_input_fields=1" + return [my edit -disable_input_fields 1] + #return [$package_id call [self] edit [list -disable_input_fields 1]] + } + view_user_input_with_feedback { + my set __feedback_mode 1 + #my msg "calling edit with disable_input_fields=1" + return [my edit -disable_input_fields 1] + #return [$package_id call [self] edit [list -disable_input_fields 1]] + } + default { + #my msg "calling $method" + return [$package_id invoke -method $method] + } + } + } + } + next + } + + WorkflowPage instproc get_assignee {assigned_to} { + return [my assignee] + } + + WorkflowPage instproc send_to_assignee { + -subject + -from + -body + {-mime_type text/plain} + {-with_ical:boolean false} + } { + my instvar package_id page_template + set wf_name [$page_template name] + + if {![info exists subject]} { + set subject "\[$wf_name\] [my title] ([my state])" + } + if {![my exists from]} {set from [my creation_user]} + acs_user::get -user_id [my assignee] -array to_info + acs_user::get -user_id $from -array from_info + + set message_id [mime::uniqueID] + set message_date [acs_mail_lite::utils::build_date] + #set tokens [acs_mail_lite::utils::build_body -mime_type text/html $body] + set tokens [mime::initialize \ + -canonical $mime_type \ + -encoding "quoted-printable" -string $body] + + if {$with_ical} { + set items [::xo::OrderedComposite new -destroy_on_cleanup -mixin ::xo::ical::VCALENDAR] + # hmm, mozilla just supports VEVENT, a VTODO would be nice. + $items add [::xo::ical::VEVENT new \ + -creation_date [my set creation_date] \ + -last_modified [my last_modified] \ + -dtstart "now" \ + -uid $package_id-[my revision_id] \ + -url [my pretty_link -absolute true] \ + -summary $subject \ + -description "Workflow instance of workflow $wf_name [my description]"] + $items configure -prodid "-//WU Wien//NONSGML XoWiki Content Flow//EN" -method request + set ical [$items as_ical] + lappend tokens [mime::initialize \ + -canonical text/calendar \ + -param [list method request] \ + -param [list charset UTF-8] \ + -header [list "Content-Disposition" "attachment; filename=\"todo.vcs\""] \ + -encoding "quoted-printable" -string $ical] + lappend tokens [mime::initialize \ + -canonical application/ics -param [list name "invite.ics"] \ + -header [list "Content-Disposition" "attachment; filename=\"todo.ics\""] \ + -encoding "quoted-printable" -string $ical] + #lappend tokens [acs_mail_lite::utils::build_body -mime_type {application/ics; name="invite.ics"} $ical] + } + + if {[llength $tokens]>1} { + set tokens [mime::initialize -canonical "multipart/mixed" -parts $tokens] + } + + set headers_list [list] + lappend headers_list \ + [list From $from_info(email)] \ + [list To $to_info(email)] \ + [list Subject $subject] + + set originator [acs_mail_lite::bounce_address -user_id $from \ + -package_id $package_id \ + -message_id $message_id] + + acs_mail_lite::smtp -multi_token $tokens -headers $headers_list -originator $originator + mime::finalize $tokens -subordinates all + + } + + WorkflowPage instproc activate {{-verbose true} ctx action} { + # Execute action and compute next state + # of action. + set action_command ${ctx}::$action + # Check, if action is defined + if {[info commands $action_command] eq ""} { + # no such action the current context + if {$verbose} {ns_log notice "Warning: [my name] No action $action in workflow context"} + return "" + } + #set next_state [$action_command get_next_state] + # Activate action + if {[catch {$action_command activate [self]} errorMsg]} { + ns_log notice "ACTIVATE [my name] error =>$errorMsg" + set error "error in action '$action' of workflow instance [my name]\ + of workflow [[my page_template] name]:" + if {[[my package_id] exists __batch_mode]} { + [my package_id] set __evaluation_error "$error\n\n$::errorInfo" + incr validation_errors + } else { + my msg -html 1 "$error
$::errorInfo
" + } + ns_log error "--WF: evaluation $error\n$::errorInfo" + return "" + } else { + # We moved get_next_state here to allow an action to infuence the + # conditions in the activation method. + set next_state [$action_command get_next_state] + ns_log notice "ACTIVATE [my name] no error next-state=$next_state" + return $next_state + } + } + + WorkflowPage instproc get_form_data args { + if {[my is_wf_instance]} { + lassign [next] validation_errors category_ids + if {$validation_errors == 0} { + #my msg "validation ok" + set cc [[my package_id] context] + foreach {name value} [$cc get_all_form_parameter] { + if {[regexp {^__action_(.+)$} $name _ action]} { + set ctx [::xowf::Context require [self]] + set next_state [my activate $ctx $action] + my log "after activate next_state=$next_state, current_state=[$ctx get_current_state], [my set instance_attributes]" + if {$next_state ne ""} { + if {[${ctx}::$next_state exists assigned_to]} { + my assignee [my get_assignee [${ctx}::$next_state assigned_to]] + } + $ctx set_current_state $next_state + } + break + } + } + } + return [list $validation_errors $category_ids] + } else { + next + } + } + + WorkflowPage instproc instantiated_form_fields {} { + # Helper method to + # - obtain the field_names from the current form, to + # - create form_field instances from that and to + # - provide the values from the instance attributes into it. + lassign [my field_names_from_form] _ field_names + set form_fields [my create_form_fields $field_names] + my load_values_into_form_fields $form_fields + return $form_fields + } + WorkflowPage ad_instproc solution_set {} { + Compute solution set in form of attribute=value pairs + based on "answer" attribute of form fields. + } { + set solutions [list] + foreach f [my instantiated_form_fields] { + if {![$f exists answer]} continue + lappend solutions [$f name]=[$f answer] + } + return [join [lsort $solutions] ", "] + } + WorkflowPage ad_instproc answer_is_correct {} { + Check, if answer is correct based on "answer" attribute of form fields + and provided user input. + } { + set correct 0 + if {[my get_from_template auto_correct] == true} { + foreach f [my instantiated_form_fields] { + #my msg "checking correctness [$f name] [$f info class] answer?[$f exists answer] -- [my get_from_template auto_correct]" + if {[$f exists answer]} { + if {[$f answer_is_correct] != 1} { + #my msg "checking correctness [$f name] failed ([$f answer_is_correct])" + set correct -1 + break + } + set correct 1 + } + } + } + return $correct + } + WorkflowPage instproc unset_temporary_instance_variables {} { + # never save/cache the following variables + my array unset __wfi + my array unset __wf + next + } + WorkflowPage instproc save_data args { + if {[my is_wf_instance]} { + #array set __ia [my instance_attributes] + set ctx [::xowf::Context require [self]] + my state [$ctx get_current_state] + #my msg "saving ia: [array get __ia]" + #my instance_attributes [array get __ia] + # + # we have to flag currently storing in hstore here, since + # saving removes the temporary variables for properties + # + if {[::xo::dc has_hstore] && [[my package_id] get_parameter use_hstore 0]} {set save_in_hstore 1} + } elseif {[my is_wf]} { + if {[::xo::dc has_hstore] && [[my package_id] get_parameter use_hstore 0]} {set save_in_hstore 1} + } + next + #my msg "save_in_hstore=[info exists save_in_hstore]" + if {[info exists save_in_hstore]} { + # "next" sets the revision_id, by not e.g. page_instance_id + my set page_instance_id [my revision_id] + my save_in_hstore + } + } + + WorkflowPage instproc double_quote {value} { + if {[regexp {[ ,\"\\=>\n\']} $value]} { + set value \"[string map [list \" \\\\\" \\ \\\\ ' ''] $value]\" + } + return $value + } + WorkflowPage instproc save_in_hstore {} { + # experimental code for testing with hstore + # to use it, do for now something like: + # + # /usr/local/pg820/bin/psql -U nsadmin -d dotlrn-test5 < /usr/local/pg820/share/postgresql/contrib/hstore.sql + # alter table xowiki_page_instance add column hkey hstore; + # CREATE INDEX hidx ON xowiki_page_instance using GIST(hkey); + # + set keys [list] + foreach {key value} [my instance_attributes] { + set v [my double_quote $value] + if {$v eq ""} continue + if {$key eq "workflow_definition"} continue + lappend keys [my double_quote $key]=>$v + } + #my msg "hkey='[join $keys ,]'" + xo::dc dml update_hstore "update xowiki_page_instance \ + set hkey = '[join $keys ,]' + where page_instance_id = [my revision_id]" + } + WorkflowPage instproc wf_property {name {default ""}} { + if {[my exists __wf]} {set key __wf($name)} else {set key __wfi($name)} + if {[my exists $key]} { return [my set $key] } + return $default + } + WorkflowPage instproc get_template_object {} { + my instvar page_template + if {[my is_wf_instance]} { + set key __wfi(wf_form_id) + if {![my exists $key]} { + set ctx [::xowf::Context require [self]] + my set $key [$ctx form_object [self]] + } + set form_obj [my set $key] + if {![my isobject $form_obj]} { + set form_id [string trimleft $form_obj :] + ::xo::db::CrClass get_instance_from_db -item_id $form_id + } + return $form_obj + } else { + return [next] + } + } + + WorkflowPage instproc create-or-use args { + #my msg "instance = [my is_wf_instance], wf=[my is_wf]" + if {[my is_wf]} { + my instvar package_id + # + # In a first step, we call "allocate". Allocate is an Action + # defined in a workflow, which is called *before* the workflow + # instance is created. Via allocate, it is e.g. possible to + # provide a computed name for the workflow instance from within + # the workflow definition. + # + set ctx [::xowf::Context require [self]] + my activate $ctx allocate + + # Check, if allocate has provided a name: + set name [my property name ""] + if {$name ne ""} { + # Ok, a name was provided. Check if an instance with this name + # exists in the current folder. + set default_lang [my lang] + set parent_id [my query_parameter "parent_id" [$package_id folder_id]] + $package_id get_lang_and_name -default_lang $default_lang -name $name lang stripped_name + set id [::xo::db::CrClass lookup -name $lang:$stripped_name -parent_id $parent_id] + #my msg "lookup of $lang:$stripped_name returned $id, default-lang([my name])=$default_lang [my nls_language]" + if {$id != 0} { + # The instance exists already + return [$package_id returnredirect \ + [export_vars -base [$package_id pretty_link -parent_id $parent_id $lang:$stripped_name] \ + [list return_url template_file]]] + } else { + if {$lang ne $default_lang} { + set nls_language [my get_nls_language_from_lang $lang] + } else { + set nls_language [my nls_language] + } + #my msg "We want to create $lang:$stripped_name" + return [next -name $lang:$stripped_name -nls_language $nls_language] + } + } + } + next + } + + WorkflowPage instproc initialize_loaded_object {} { + next + if {[my is_wf_instance]} { + my initialize + } + } + + WorkflowPage instproc initialize {} { + #my log "is_wf_instance [my is_wf_instance]" + # + # A fresh workflow page was created (called only once per + # workflow page at initial creation) + # + if {[my is_wf_instance]} { + # + # Get context and call user defined "constructor" + # + set ctx [::xowf::Context require [self]] + my activate -verbose false $ctx initialize + + # Ignore the returned next_state, since the initial state is + # always set to the same value from the ctx (initial) + #my msg "[self] is=[my set instance_attributes]" + } + next + + } + + WorkflowPage instproc default_instance_attributes {} { + # Provide the default list of instance attributes to derived + # FormPages. + if {[my is_wf]} { + # + # We have a workflow page. Get the initial state of the workflow + # instance from the workflow. + # + my instvar instance_attributes + set ctx [::xowf::Context require [self]] + foreach p [$ctx defined ::xowiki::formfield::FormField] { + set name [$p name] + set value [$p default] + if {[::xo::cc exists_query_parameter $name]} { + # never clobber instance attributes from query parameters blindly + #my msg "ignore $name" + continue + } + if {[::xo::cc exists_query_parameter p.$name] + && [$p exists allow_query_parameter]} { + # we allow the value to be taken from the query parameter + set value [::xo::cc query_parameter p.$name] + } + dict set instance_attributes $name $value + set f($name) $p + } + + ## save instance attributes + #set instance_attributes [array get __ia] + #my msg "[self] [my name] setting default parameter" + #my msg ia=$instance_attributes,props=[$ctx defined Property] + + my state [$ctx get_current_state] + #my msg "setting initial state to '[my state]'" + + return $instance_attributes + } else { + next + } + } + WorkflowPage instproc constraints_as_array {c} { + array set __c "" + foreach name_and_spec $c { + regexp {^([^:]+):(.*)$} $name_and_spec _ spec_name short_spec + set __c($spec_name) $short_spec + } + return [array get __c] + } + WorkflowPage instproc merge_constraints {c1 args} { + # Load into the base_constraints c1 the constraints from the argument list. + # The first constraints have the lowest priority + array set __c1 [my constraints_as_array $c1] + foreach c2 $args { + foreach {att value} [my constraints_as_array $c2] { + set key __c1($att) + if {[info exists $key]} {append $key ",$value"} else {set $key $value} + } + } + set result [list] + foreach {att value} [array get __c1] {lappend result $att:$value} + return $result + } + WorkflowPage instproc wfi_merged_form_constraints {constraints_from_form} { + set ctx [::xowf::Context require [self]] + set wf_specific_constraints [[my page_template] property form_constraints] + set m [my merge_constraints $wf_specific_constraints \ + $constraints_from_form [$ctx get_form_constraints]] + #my msg "merged:$m" + return $m + } + WorkflowPage instproc wf_merged_form_constraints {constraints_from_form} { + return $constraints_from_form + #return [my merge_constraints $constraints_from_form [my property form_constraints]] + } + + WorkflowPage instproc get_anon_instances {} { + if {[my istype ::xowiki::FormPage] && [my is_wf_instance]} { + # In case, the workflow has the autoname variable set, it has + # the highest weight of all other sources. + set ctx [::xowf::Context require [self]] + if {[$ctx exists autoname]} { + return [$ctx set autoname] + } + } + next + } + + WorkflowPage instproc get_form_constraints {{-trylocal false}} { + if {[my istype ::xowiki::FormPage] && [my is_wf]} { + #my msg "get_form_constraints is_wf" + return [::xo::cc cache [list [self] wf_merged_form_constraints [next]]] + } elseif {[my istype ::xowiki::FormPage] && [my is_wf_instance]} { + #my msg "get_form_constraints is_wf_instance" + return [::xo::cc cache [list [self] wfi_merged_form_constraints [next]]] + } else { + #my msg "get_form_constraints next" + next + } + } + WorkflowPage instproc visited_states {} { + my instvar item_id + foreach state [xo::dc list history { + select DISTINCT state from xowiki_form_page p, cr_items i, cr_revisions r + where i.item_id = :item_id and r.item_id = i.item_id and xowiki_form_page_id = r.revision_id}] { + set visited($state) 1 + } + #my msg "visisted states of item $item_id = [array names visited]" + return [array names visited] + } + + WorkflowPage ad_instproc footer {} { + Provide a tailored footer for workflow definition pages and + workflow instance pages containing controls for instantiating + forms or providing links to the workflow definition. + } { + if {[my exists __no_form_page_footer]} { + next + } else { + my instvar package_id + set form_item_id [my page_template] + #my msg "is wf page [my is_wf], is wf instance page [my is_wf_instance]" + if {[my is_wf]} { + # + # page containing a work flow definition + # + #set ctx [::xowf::Context require [self]] + set work_flow_form [::xo::db::CrClass get_instance_from_db -item_id $form_item_id] + set work_flow_base [$work_flow_form pretty_link] + + set wf [self] + set wf_base [$wf pretty_link] + set button_objs [list] + + # create new workflow instance button with start form + if {[my parent_id] != [$package_id folder_id]} { + set parent_id [my parent_id] + } + set link [$package_id make_link -link $wf_base $wf create-new parent_id return_url] + lappend button_objs [::xowiki::includelet::form-menu-button-new new -volatile \ + -parent_id [my parent_id] \ + -form $wf -link $link] + + # list workflow instances button + set obj [::xowiki::includelet::form-menu-button-wf-instances new -volatile \ + -package_id $package_id -parent_id [my parent_id] \ + -base $wf_base -form $wf] + if {[info exists return_url]} {$obj return_url $return_url} + lappend button_objs $obj + + # work flow definition button + set obj [::xowiki::includelet::form-menu-button-form new -volatile \ + -package_id $package_id -parent_id [my parent_id] \ + -base $work_flow_base -form $work_flow_form] + if {[info exists return_url]} {$obj return_url $return_url} + lappend button_objs $obj + + # make menu + return [my include [list form-menu -form_item_id [my item_id] -button_objs $button_objs]] + + } elseif {[my is_wf_instance]} { + # + # work flow instance + # + set entry_form_item_id [my wf_property wf_form_id] + set work_flow_form [::xo::db::CrClass get_instance_from_db -item_id $form_item_id] + set work_flow_base [$work_flow_form pretty_link] + set button_objs [list] + + #my msg entry_form_item_id=$entry_form_item_id-exists?=[my isobject $entry_form_item_id] + + # form definition button + if {![my isobject $entry_form_item_id]} { + # In case, the id is a form object, it is a dynamic form, + # that we can't edit; therefore, we provide no link. + # + # Here, we have have an id that we use for fetching... + # + set form [::xo::db::CrClass get_instance_from_db -item_id $entry_form_item_id] + set base [$form pretty_link] + set obj [::xowiki::includelet::form-menu-button-form new -volatile \ + -package_id $package_id -parent_id [my parent_id] \ + -base $base -form $form] + if {[info exists return_url]} {$obj return_url $return_url} + lappend button_objs $obj + } + + # if {[my exists_property form]} { + # lappend button_objs \ + # [::xowiki::includelet::form-menu-button-new new -volatile \ + # -package_id $package_id -parent_id [my parent_id] \ + # -base [my pretty_link] -form [self]] + # lappend button_objs \ + # [::xowiki::includelet::form-menu-button-answers new -volatile \ + # -package_id $package_id -parent_id [my parent_id] \ + # -base [my pretty_link] -form [self]] + # } + # work flow definition button + set obj [::xowiki::includelet::form-menu-button-wf new -volatile \ + -package_id $package_id -parent_id [my parent_id] \ + -base $work_flow_base -form $work_flow_form] + if {[info exists return_url]} {$obj return_url $return_url} + lappend button_objs $obj + # make menu + return [my include [list form-menu -form_item_id [my page_template] -button_objs $button_objs]] + } else { + #return [my include [list form-menu -form_item_id $form_item_id -buttons form]] + next + } + } + } + + WorkflowPage ad_instproc call_action_foreach {-action:required {-attributes ""} page_names} { + Call the specified action for each of the specified pages denoted + by the list of page names + } { + foreach page_name $page_names { + set page [[my package_id] get_page_from_name -parent_id [my parent_id] -name $page_name] + if {$page ne ""} { + $page call_action -action $action -attributes $attributes + } else { + ns_log notice "WF: could not call action $action, since $page_name in [my parent_id] failed" + } + } + } + + + WorkflowPage ad_instproc call_action {-action {-attributes {}}} { + Call the specified action in the current workflow instance. + The specified attributes are provided like form_parameters to + the action of the workflow. + } { + my instvar package_id + if {![my is_wf_instance]} { + error "Page [self] is not a Workflow Instance" + } + set ctx [::xowf::Context require [self]] + my log "CTX of [self] ([my name])= $ctx" + foreach a [$ctx get_actions] { + if {[namespace tail $a] eq "$action"} { + # In the current state, the specified action is allowed + my log "--xowf action $action allowed -- name='[my name]'" + return [$a invoke -attributes $attributes] + } + } + set a ${ctx}::$action + if {[info commands $a] ne "" && [$a state_safe]} { + # The action is defined as state-safe, so if can be called in every state + my log "--xowf action $action state_safe -- name='[my name]'" + return [$a invoke -attributes $attributes] + } + error "\tNo state-safe action '$action' available in workflow instance [self] of \ + [[my page_template] name] in state [$ctx get_current_state] + Available actions: [[$ctx current_state] get_actions]" + } + + # + # Interface to atjobs + # + WorkflowPage ad_instproc schedule_action {-time -party_id -action {-attributes {}}} { + Schedule the specified action for the current workflow instance at the given + time. The specified attributes are provided like form_parameters to + the action of the workflow. + } { + my instvar package_id + if {![my is_wf_instance]} { + error "Page [self] is not a Workflow Instance" + } + if {![info exists party_id]} {set party_id [::xo::cc user_id]} + my schedule_job -time $time -party_id $party_id \ + [list call_action -action $action -attributes $attributes] + } + + WorkflowPage ad_instproc schedule_job {-time:required -party_id cmd} { + Schedule the specified tcl command for the the current package + instance at the given time. + } { + my instvar package_id + my log "-at" + set j [::xowf::atjob new -time $time -party_id $party_id -cmd $cmd -object [self]] + $j persist + } + + ad_proc migrate_from_wf_current_state {} { + # + # Transform the former instance_attributes + # "wf_current_state" to the xowiki::FormPage attribute "state", and + # "wf_assignee" to the xowiki::FormPage attribute "assignee". + # + set count 0 + foreach atts [xo::dc list_of_lists entries { + select p.state,p.assignee,pi.instance_attributes,p.xowiki_form_page_id + from xowiki_form_page p, xowiki_page_instance pi, cr_items i, cr_revisions r + where r.item_id = i.item_id and p.xowiki_form_page_id = r.revision_id and + pi.page_instance_id = r.revision_id + }] { + lassign $atts state assignee instance_attributes xowiki_form_page_id + if {[dict exists $instance_attributes wf_current_state] + && [dict get $instance_attributes wf_current_state] ne $state} { + #Object msg "must update state $state for $xowiki_form_page_id to [dict get $instance_attributes wf_current_state]" + + xo::db dml update_state "update xowiki_form_page \ + set state = '[dict get $instance_attributes wf_current_state]' + where xowiki_form_page_id = :xowiki_form_page_id" + incr count + } + if {[dict exists $instance_attributes wf_assignee] && [dict get $instance_attributes wf_assignee] ne $assignee} { + #Object msg "must update assignee $assignee for $xowiki_form_page_id to [dict get $instance_attributes wf_assignee]" + set wf_assignee [dict get $instance_attributes wf_assignee] + xo::dc dml update_state "update xowiki_form_page set assignee = :wf_assignee \ + where xowiki_form_page_id = :xowiki_form_page_id" + incr count + } + } + return $count + } + + ad_proc update_hstore {package_id} { + update all instance attributes in hstore + } { + if {![::xo::dc has_hstore] && [$package_id get_parameter use_hstore 0] } {return 0} + # + # This proc can be used from ds/shell as follows + # + # ::xowf::Package initialize -url /xowf + # ::xowf::update_hstore $package_id + # + # Check the result + # + # select hkey from xowiki_page_instance where hkey is not null; + # + ::xowf::Package initialize -package_id $package_id + # + # we get all revisions, so use the lower level interface + # + set items [::xowiki::FormPage instantiate_objects \ + -sql "select * from xowiki_form_pagei bt,cr_items i \ + where bt.object_package_id = $package_id and bt.item_id = i.item_id" \ + -object_class ::xowiki::FormPage] + set count 0 + foreach i [$items children] { + #$i msg "working on [$i set xowiki_form_page_id]" + $i save_in_hstore + incr count + } + $items msg "fetched $count objects from parent_id [$package_id folder_id]" + return 1 + } + + # Some example hstore queries (over all revisions) + # select hkey from xowiki_page_instance where hkey is not null; + # select hkey from xowiki_page_instance where defined(hkey, 'team_email'); + # select hkey from xowiki_page_instance where exist(hkey, 'team_email'); + # select hkey from xowiki_page_instance where 'team_email=>neumann@wu-wien.ac.at' <@ hkey; + # select (each(hkey)).key, (each(hkey)).value from xowiki_page_instance; + # select page_instance_id, (each(hkey)).key, (each(hkey)).value from xowiki_page_instance + # where 'assignee=>539,priority=>1' <@ hkey; + # select key, count(*) from (select (each(hkey)).key from xowiki_page_instance) as stat + # group by key order by count desc, key; + # + +} + + + +# +# In order to provide either a REST or a DAV interface, we have to +# switch to basic authentication, since non-openacs packages +# have problems to handle openacs coockies. The basic authentication +# interface can be establised in three steps: +# +# 1) Create a basic authentication handler, Choose a URL and +# define optionally the package to be initialized: +# Example: +# ::xowf::dav create ::xowf::ba -url /ba -package ::xowf::Package +# +# 2) Make sure, the basic authenication handler is initialied during +# startup. Write an -init.tcl file containing a call to the +# created handler. +# Example: +# ::xowf::ba register +# +# 3) Write procs with names such as GET, PUT, POST to handle +# the requests. These procs overload the predefined behavior. +# + +namespace eval ::xowf { + # ::xo::dav should be probably changed to ::xo::ProtocolHandler for release + ::xotcl::Class create ::xowf::dav -superclass ::xo::dav + + ::xowf::dav instproc get_package_id {} { + my instvar uri package wf package_id + if {$uri eq "/"} { + # Take the first package instance + set wf "" + set package_id [lindex [$package instances] 0] + $package initialize -package_id $package_id + } else { + set wf /$uri + $package initialize -url $uri + } + # my log package_id=$package_id + return $package_id + } + + ::xowf::dav instproc call_action {-uri -action -attributes} { + [my package] initialize -url $uri + set object_name [$package_id set object] + set page [$package_id resolve_request -path $object_name method] + if {$page eq ""} { + set errorMsg cannot resolve '$object_name' in package [$package_id package_url] + my log "Error: $errorMsg" + ns_return 406 text/plain "Error: $errorMsg" + } elseif {[catch {set msg [$page call_action \ + -action $action \ + -attributes $attributes]} errorMsg]} { + my log "Error: $uri $action $attributes resulted in\n$errorMsg\n$::errorInfo" + ns_return 406 text/plain "Error: $errorMsg\n" + } else { + ns_return 200 text/plain "Success: $msg\n" + } + } + + + ::xowf::dav create ::xowf::dav-todo -url /dav-todo -package ::xowf::Package + + ::xowf::dav-todo proc GET {} { + my instvar uri wf package_id + set p [::xowiki::Page new -package_id $package_id] + $p include [list wf-todo -ical 1 -workflow $wf] + #ns_return 200 text/plain GET-$uri-XXX-pid=$package_id-wf=$wf-[::xo::cc serialize] + } + + # ::xowf::dav-todo proc GET {} { + # set uri /xowf/153516 + # set uri /xowf/18362 + # set uri /xowf/18205 + # my call_action -uri $uri -action work -attributes [list comment hello3 effort 4] + # } + + proc include {wfName {vars ""}} { + uplevel [::xowf::include_get -level 2 $wfName {*}$vars] + } + + ad_proc include_get {{-level 1} wfName {vars ""}} { + if {![string match "/packages/*/lib/*" $wfName]} { + error "path leading to workflow name must look likw /packages/*/lib/*" + } + set fname [get_server_root]$wfName + + if {![file readable $fname]} { + error "file '$fname' not found" + } + set f [open $fname]; set wfDefinition [read $f]; close $f + #::xotcl::Object log "INCLUDE $wfName [list $vars]" + if {[llength $vars] > 0} { + foreach var $vars { + lappend substMap @$var@ [uplevel $level [list set $var]] + } + set wfDefinition [string map $substMap $wfDefinition] + } + #::xotcl::Object log "AFTER SUBST $wfName [list $vars]\n$wfDefinition" + return [list eval $wfDefinition] + } + +} + +::xo::library source_dependent + +# +# Local variables: +# mode: tcl +# tcl-indent-level: 2 +# indent-tabs-mode: nil +# End: Index: openacs-4/packages/xowf/www/index.vuh =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/www/index.vuh,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/www/index.vuh 28 Aug 2014 08:24:56 -0000 1.1 @@ -0,0 +1,20 @@ +# -*- tcl -*- +::xowf::Package initialize -ad_doc { + + This is the resolver for this package. It turns a request into + an object and executes the object with the computed method + + @author Gustaf Neumann (gustaf.neumann@wu-wien.ac.at) + @creation-date July, 2006 + @cvs-id $Id: index.vuh,v 1.1 2014/08/28 08:24:56 gustafn Exp $ + +} -parameter { + {-m view} + {-folder_id:integer 0} +} +# body=[ns_conn content] +::$package_id log "--starting... [ns_conn url] [ns_conn query] \n\ + form vars = [ns_set array [ns_getform]]" +::$package_id reply_to_user [::$package_id invoke -method $m] +::$package_id log "--i ::$package_id DONE" +ad_script_abort Index: openacs-4/packages/xowf/www/admin/index.vuh =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/www/admin/index.vuh,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/www/admin/index.vuh 28 Aug 2014 08:24:57 -0000 1.1 @@ -0,0 +1,5 @@ +# -*- tcl -*- +::xowf::Package initialize +set l [expr {[lsearch -exact [ns_conn urlv] admin] + 1 }] +set path [join [lrange [ns_conn urlv] $l end] /] +rp_internal_redirect /packages/xowiki/www/admin/$path Index: openacs-4/packages/xowf/www/prototypes/atjob-form.page =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/www/prototypes/Attic/atjob-form.page,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/www/prototypes/atjob-form.page 28 Aug 2014 08:24:57 -0000 1.1 @@ -0,0 +1,18 @@ +# -*- tcl-*- +::xowiki::Form new \ + -set name en:atjob-form \ + -title "At-Job Form" \ + -set anon_instances t \ + -set text {} \ + -set form {{
@_title@ @cmd@ @_publish_date@ @_publish_status@
} text/html} \ + -set form_constraints { +_page_order:hidden _creator:hidden +_title:text,label=#acs-kernel.common_Title# +{_publish_date:date,format=DD_MONTH_YYYY_HH24_MI,label=Scheduled execution time} +{_publish_status:radio,options={waiting production} {done expired},label=Status} +@table_properties:publish_status=all +@table:_name,_publish_status,_last_modified + } + + + Index: openacs-4/packages/xowf/www/prototypes/index.page =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/www/prototypes/index.page,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/www/prototypes/index.page 28 Aug 2014 08:24:57 -0000 1.1 @@ -0,0 +1,27 @@ +::xowiki::Page new -title "Index Page" -text { + +

+This is the default start page of XoWiki. You can edit this page and save it to provide a personalized look of the XoWiki instance. You can as well provide a different index page through configuration. +You can also view the contents of the Wiki in a weblog style. +For more details, consult the [[http://media.wu-wien.ac.at/download/xowiki-doc/|XoWiki documentation]]. +

+

+A user can define notifications +for the whole XoWiki instance (by clicking on the notifications button +in the menu bar or for categories (by clicking on the letter symbol +next to the category entries at the bottom of the page) +

+>>left-col<< +{{recent -max_entries 25}} +>><< +>>right-col<< +{{last-visited -title "Last Visited" -max_entries 10 }} +
+{{most-popular -title "Most Popular" -max_entries 10 }} +>><< + +} + + + + Index: openacs-4/packages/xowf/www/resources/feedback.css =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/www/resources/feedback.css,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/www/resources/feedback.css 28 Aug 2014 08:24:57 -0000 1.1 @@ -0,0 +1,7 @@ +/* .xowiki-content .correct:after {content: " -- ok";}*/ +form.feedback .correct {border: 1px dotted green; background-color: #aaffaa;} + +/* .xowiki-content .incorrect:after {content: "-- wrong";}*/ +form.feedback .incorrect {border: 1px dotted red; background-color: #ffaaaa;} + +form.feedback div.form-required-mark {display:none;} \ No newline at end of file Index: openacs-4/packages/xowf/www/resources/myform.css =================================================================== RCS file: /usr/local/cvsroot/openacs-4/packages/xowf/www/resources/Attic/myform.css,v diff -u --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ openacs-4/packages/xowf/www/resources/myform.css 28 Aug 2014 08:24:57 -0000 1.1 @@ -0,0 +1,11 @@ +.xowiki-content fieldset { + border: 1px solid #ccc; +} +.xowiki-content .margin-form fieldset .form-item-wrapper .form-label { + width: 10em; +} +.xowiki-content .margin-form fieldset .form-item-wrapper .form-widget, +.xowiki-content .margin-form fieldset .form-button, +.xowiki-content .margin-form fieldset div.form-help-text { + margin-left: 11em; +} \ No newline at end of file