001    /**
002     * Copyright 2005-2012 The Kuali Foundation
003     *
004     * Licensed under the Educational Community License, Version 2.0 (the "License");
005     * you may not use this file except in compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     * http://www.opensource.org/licenses/ecl2.php
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed under the License is distributed on an "AS IS" BASIS,
012     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013     * See the License for the specific language governing permissions and
014     * limitations under the License.
015     */
016    package org.kuali.rice.krms.impl.ui;
017    
018    import org.apache.commons.collections.CollectionUtils;
019    import org.apache.commons.lang.StringUtils;
020    import org.kuali.rice.core.api.util.tree.Node;
021    import org.kuali.rice.krad.maintenance.MaintenanceDocument;
022    import org.kuali.rice.krad.service.KRADServiceLocator;
023    import org.kuali.rice.krad.service.SequenceAccessorService;
024    import org.kuali.rice.krad.uif.UifParameters;
025    import org.kuali.rice.krad.util.GlobalVariables;
026    import org.kuali.rice.krad.util.ObjectUtils;
027    import org.kuali.rice.krad.web.controller.MaintenanceDocumentController;
028    import org.kuali.rice.krad.web.form.MaintenanceForm;
029    import org.kuali.rice.krad.web.form.UifFormBase;
030    import org.kuali.rice.krms.api.KrmsApiServiceLocator;
031    import org.kuali.rice.krms.api.engine.expression.ComparisonOperatorService;
032    import org.kuali.rice.krms.api.repository.rule.RuleDefinition;
033    import org.kuali.rice.krms.api.repository.term.TermDefinition;
034    import org.kuali.rice.krms.api.repository.term.TermResolverDefinition;
035    import org.kuali.rice.krms.api.repository.term.TermSpecificationDefinition;
036    import org.kuali.rice.krms.api.repository.type.KrmsAttributeDefinition;
037    import org.kuali.rice.krms.impl.repository.ActionBo;
038    import org.kuali.rice.krms.api.repository.LogicalOperator;
039    import org.kuali.rice.krms.api.repository.proposition.PropositionType;
040    import org.kuali.rice.krms.impl.repository.AgendaBo;
041    import org.kuali.rice.krms.impl.repository.AgendaItemBo;
042    import org.kuali.rice.krms.impl.repository.ContextBoService;
043    import org.kuali.rice.krms.impl.repository.KrmsAttributeDefinitionService;
044    import org.kuali.rice.krms.impl.repository.KrmsRepositoryServiceLocator;
045    import org.kuali.rice.krms.impl.repository.PropositionBo;
046    import org.kuali.rice.krms.impl.repository.RuleBo;
047    import org.kuali.rice.krms.impl.repository.RuleBoService;
048    import org.kuali.rice.krms.impl.repository.TermBo;
049    import org.kuali.rice.krms.impl.rule.AgendaEditorBusRule;
050    import org.kuali.rice.krms.impl.util.KRMSPropertyConstants;
051    import org.kuali.rice.krms.impl.util.KrmsImplConstants;
052    import org.springframework.stereotype.Controller;
053    import org.springframework.validation.BindingResult;
054    import org.springframework.web.bind.annotation.ModelAttribute;
055    import org.springframework.web.bind.annotation.RequestMapping;
056    import org.springframework.web.servlet.ModelAndView;
057    
058    import javax.servlet.http.HttpServletRequest;
059    import javax.servlet.http.HttpServletResponse;
060    import java.util.ArrayList;
061    import java.util.Collection;
062    import java.util.Collections;
063    import java.util.Enumeration;
064    import java.util.HashMap;
065    import java.util.List;
066    import java.util.Map;
067    import java.util.UUID;
068    
069    /**
070     * Controller for the Test UI Page
071     * @author Kuali Rice Team (rice.collab@kuali.org)
072     */
073    @Controller
074    @RequestMapping(value = org.kuali.rice.krms.impl.util.KrmsImplConstants.WebPaths.AGENDA_EDITOR_PATH)
075    public class AgendaEditorController extends MaintenanceDocumentController {
076    
077        private SequenceAccessorService sequenceAccessorService;
078    
079        /**
080         * This overridden method does extra work on refresh to update the namespace when the context has been changed.
081         *
082         * @see org.kuali.rice.krad.web.controller.UifControllerBase#refresh(org.kuali.rice.krad.web.form.UifFormBase, org.springframework.validation.BindingResult, javax.servlet.http.HttpServletRequest, javax.servlet.http.HttpServletResponse)
083         */
084        @RequestMapping(params = "methodToCall=" + "refresh")
085        @Override
086        public ModelAndView refresh(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
087                HttpServletRequest request, HttpServletResponse response) throws Exception {
088            ModelAndView modelAndView = super.refresh(form, result, request, response);
089    
090            // handle return from context lookup
091            MaintenanceForm maintenanceForm = (MaintenanceForm) form;
092            AgendaEditor agendaEditor = ((AgendaEditor) maintenanceForm.getDocument().getNewMaintainableObject().getDataObject());
093            AgendaEditorBusRule rule = new AgendaEditorBusRule();
094            if (rule.validContext(agendaEditor) && rule.validAgendaName(agendaEditor)) {
095                // update the namespace on all agenda related objects if the contest has been changed
096                if (!StringUtils.equals(agendaEditor.getOldContextId(), agendaEditor.getAgenda().getContextId())) {
097                    agendaEditor.setOldContextId(agendaEditor.getAgenda().getContextId());
098    
099                    String namespace = "";
100                    if (!StringUtils.isBlank(agendaEditor.getAgenda().getContextId())) {
101                        namespace = getContextBoService().getContextByContextId(agendaEditor.getAgenda().getContextId()).getNamespace();
102                    }
103    
104                    for (AgendaItemBo agendaItem : agendaEditor.getAgenda().getItems()) {
105                        agendaItem.getRule().setNamespace(namespace);
106                        for (ActionBo action : agendaItem.getRule().getActions()) {
107                            action.setNamespace(namespace);
108                        }
109                    }
110                }
111            }
112            return modelAndView;
113        }
114    
115        /**
116         * This method updates the existing rule in the agenda.
117         */
118        @RequestMapping(params = "methodToCall=" + "goToAddRule")
119        public ModelAndView goToAddRule(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
120                HttpServletRequest request, HttpServletResponse response) throws Exception {
121            setAgendaItemLine(form, null);
122            AgendaEditor agendaEditor = getAgendaEditor(form);
123            agendaEditor.setAddRuleInProgress(true);
124            form.getActionParameters().put(UifParameters.NAVIGATE_TO_PAGE_ID, "AgendaEditorView-AddRule-Page");
125            return super.navigate(form, result, request, response);
126        }
127    
128        /**
129         * This method sets the agendaItemLine for adding/editing AgendaItems.
130         * The agendaItemLine is a copy of the agendaItem so that changes are not applied when
131         * they are abandoned.  If the agendaItem is null a new empty agendaItemLine is created.
132         *
133         * @param form
134         * @param agendaItem
135         */
136        private void setAgendaItemLine(UifFormBase form, AgendaItemBo agendaItem) {
137            AgendaEditor agendaEditor = getAgendaEditor(form);
138            if (agendaItem == null) {
139                RuleBo rule = new RuleBo();
140                rule.setId(getSequenceAccessorService().getNextAvailableSequenceNumber("KRMS_RULE_S")
141                        .toString());
142                if (StringUtils.isBlank(agendaEditor.getAgenda().getContextId())) {
143                    rule.setNamespace("");
144                } else {
145                    rule.setNamespace(getContextBoService().getContextByContextId(agendaEditor.getAgenda().getContextId()).getNamespace());
146                }
147                agendaItem = new AgendaItemBo();
148                agendaItem.setRule(rule);
149                agendaEditor.setAgendaItemLine(agendaItem);
150            } else {
151                // TODO: Add a copy not the reference
152                agendaEditor.setAgendaItemLine((AgendaItemBo) ObjectUtils.deepCopy(agendaItem));
153            }
154    
155    
156            if (agendaItem.getRule().getActions().isEmpty()) {
157                ActionBo actionBo = new ActionBo();
158                actionBo.setTypeId("");
159                actionBo.setNamespace(agendaItem.getRule().getNamespace());
160                actionBo.setRuleId(agendaItem.getRule().getId());
161                actionBo.setSequenceNumber(1);
162                agendaEditor.setAgendaItemLineRuleAction(actionBo);
163            } else {
164                agendaEditor.setAgendaItemLineRuleAction(agendaItem.getRule().getActions().get(0));
165            }
166    
167            agendaEditor.setCustomRuleActionAttributesMap(agendaEditor.getAgendaItemLineRuleAction().getAttributes());
168            agendaEditor.setCustomRuleAttributesMap(agendaEditor.getAgendaItemLine().getRule().getAttributes());
169        }
170    
171        /**
172         * This method returns the id of the selected agendaItem.
173         *
174         * @param form
175         * @return selectedAgendaItemId
176         */
177        private String getSelectedAgendaItemId(UifFormBase form) {
178            AgendaEditor agendaEditor = getAgendaEditor(form);
179            return agendaEditor.getSelectedAgendaItemId();
180        }
181    
182        /**
183         * This method sets the id of the cut agendaItem.
184         *
185         * @param form
186         * @param cutAgendaItemId
187         */
188        private void setCutAgendaItemId(UifFormBase form, String cutAgendaItemId) {
189            AgendaEditor agendaEditor = getAgendaEditor(form);
190            agendaEditor.setCutAgendaItemId(cutAgendaItemId);
191        }
192    
193        /**
194         * This method returns the id of the cut agendaItem.
195         *
196         * @param form
197         * @return cutAgendaItemId
198         */
199        private String getCutAgendaItemId(UifFormBase form) {
200            AgendaEditor agendaEditor = getAgendaEditor(form);
201            return agendaEditor.getCutAgendaItemId();
202        }
203    
204        /**
205         * This method updates the existing rule in the agenda.
206         */
207        @RequestMapping(params = "methodToCall=" + "goToEditRule")
208        public ModelAndView goToEditRule(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
209                HttpServletRequest request, HttpServletResponse response) throws Exception {
210    
211            AgendaEditor agendaEditor = getAgendaEditor(form);
212            agendaEditor.setAddRuleInProgress(false);
213            // this is the root of the tree:
214            AgendaItemBo firstItem = getFirstAgendaItem(agendaEditor.getAgenda());
215            String selectedItemId = agendaEditor.getSelectedAgendaItemId();
216            AgendaItemBo node = getAgendaItemById(firstItem, selectedItemId);
217    
218            setAgendaItemLine(form, node);
219    
220            form.getActionParameters().put(UifParameters.NAVIGATE_TO_PAGE_ID, "AgendaEditorView-EditRule-Page");
221            return super.navigate(form, result, request, response);
222        }
223    
224        /**
225         *  This method adds the newly create rule to the agenda.
226         */
227        @RequestMapping(params = "methodToCall=" + "addRule")
228        public ModelAndView addRule(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
229                HttpServletRequest request, HttpServletResponse response) throws Exception {
230    
231            AgendaEditor agendaEditor = getAgendaEditor(form);
232            AgendaBo agenda = agendaEditor.getAgenda();
233            AgendaItemBo newAgendaItem = agendaEditor.getAgendaItemLine();
234    
235            if (!validateProposition(newAgendaItem.getRule().getProposition(), newAgendaItem.getRule().getNamespace())) {
236                form.getActionParameters().put(UifParameters.NAVIGATE_TO_PAGE_ID, "AgendaEditorView-AddRule-Page");
237                // NOTICE short circuit method on invalid proposition
238                return super.navigate(form, result, request, response);
239            }
240    
241            newAgendaItem.getRule().setAttributes(agendaEditor.getCustomRuleAttributesMap());
242            updateRuleAction(agendaEditor);
243    
244            if (agenda.getItems() == null) {
245                agenda.setItems(new ArrayList<AgendaItemBo>());
246            }
247    
248            AgendaEditorBusRule rule = new AgendaEditorBusRule();
249            MaintenanceForm maintenanceForm = (MaintenanceForm) form;
250            MaintenanceDocument document = maintenanceForm.getDocument();
251            if (rule.processAgendaItemBusinessRules(document)) {
252                newAgendaItem.setId(getSequenceAccessorService().getNextAvailableSequenceNumber("KRMS_AGENDA_ITM_S")
253                        .toString());
254                newAgendaItem.setAgendaId(getCreateAgendaId(agenda));
255                if (agenda.getFirstItemId() == null) {
256                    agenda.setFirstItemId(newAgendaItem.getId());
257                } else {
258                    // insert agenda in tree
259                    String selectedAgendaItemId = getSelectedAgendaItemId(form);
260                    if (StringUtils.isBlank(selectedAgendaItemId)) {
261                        // add after the last root node
262                        AgendaItemBo node = getFirstAgendaItem(agenda);
263                        while (node.getAlways() != null) {
264                            node = node.getAlways();
265                        }
266                        node.setAlwaysId(newAgendaItem.getId());
267                        node.setAlways(newAgendaItem);
268                    } else {
269                        // add after selected node
270                        AgendaItemBo firstItem = getFirstAgendaItem(agenda);
271                        AgendaItemBo node = getAgendaItemById(firstItem, selectedAgendaItemId);
272                        newAgendaItem.setAlwaysId(node.getAlwaysId());
273                        newAgendaItem.setAlways(node.getAlways());
274                        node.setAlwaysId(newAgendaItem.getId());
275                        node.setAlways(newAgendaItem);
276                    }
277                }
278                // add it to the collection on the agenda too
279                agenda.getItems().add(newAgendaItem);
280                agendaEditor.setAddRuleInProgress(false);
281                form.getActionParameters().put(UifParameters.NAVIGATE_TO_PAGE_ID, "AgendaEditorView-Agenda-Page");
282            } else {
283                form.getActionParameters().put(UifParameters.NAVIGATE_TO_PAGE_ID, "AgendaEditorView-AddRule-Page");
284            }
285            return super.navigate(form, result, request, response);
286        }
287    
288        /**
289         * Validate the given proposition and its children.  Note that this method is side-effecting,
290         * when errors are detected with the proposition, errors are added to the error map.
291         * @param proposition the proposition to validate
292         * @param namespace the namespace of the parent rule
293         * @return true if the proposition and its children (if any) are considered valid
294         */
295        // TODO also wire up to proposition for faster feedback to the user
296        private boolean validateProposition(PropositionBo proposition, String namespace) {
297            boolean result = true;
298    
299            if (proposition != null) { // Null props are allowed.
300    
301                if (StringUtils.isBlank(proposition.getCompoundOpCode())) {
302                    // then this is a simple proposition, validate accordingly
303    
304                    result &= validateSimpleProposition(proposition, namespace);
305    
306                } else {
307                    // this is a compound proposition (or it should be)
308                    List<PropositionBo> compoundComponents = proposition.getCompoundComponents();
309    
310                    if (!CollectionUtils.isEmpty(proposition.getParameters())) {
311                        GlobalVariables.getMessageMap().putError(KRMSPropertyConstants.Rule.PROPOSITION_TREE_GROUP_ID,
312                                "error.rule.proposition.compound.invalidParameter", proposition.getDescription());
313                        result &= false;
314                    }
315    
316                    // recurse
317                    if (!CollectionUtils.isEmpty(compoundComponents)) for (PropositionBo childProp : compoundComponents) {
318                        result &= validateProposition(childProp, namespace);
319                    }
320                }
321            }
322    
323            return result;
324        }
325    
326        /**
327         * Validate the given simple proposition.  Note that this method is side-effecting,
328         * when errors are detected with the proposition, errors are added to the error map.
329         * @param proposition the proposition to validate
330         * @param namespace the namespace of the parent rule
331         * @return true if the proposition is considered valid
332         */
333        private boolean validateSimpleProposition(PropositionBo proposition, String namespace) {
334            boolean result = true; 
335            
336            String propConstant = null;
337            if (proposition.getParameters().get(1) != null) {
338                propConstant = proposition.getParameters().get(1).getValue();
339            }
340            String operator = null;
341            if (proposition.getParameters().get(2) != null) {
342                operator = proposition.getParameters().get(2).getValue();
343            }
344    
345            String termId = null;
346            if (proposition.getParameters().get(0) != null) {
347                termId = proposition.getParameters().get(0).getValue();
348            }
349            // Simple proposition requires all of propConstant, termId and operator to be specified
350            if (StringUtils.isBlank(termId)) {
351                GlobalVariables.getMessageMap().putErrorWithoutFullErrorPath(KRMSPropertyConstants.Rule.PROPOSITION_TREE_GROUP_ID,
352                        "error.rule.proposition.simple.blankField", proposition.getDescription(), "Term");
353                result &= false;
354            } else {
355                result = validateTerm(proposition, namespace);
356            }
357            if (StringUtils.isBlank(operator)) {
358                GlobalVariables.getMessageMap().putErrorWithoutFullErrorPath(KRMSPropertyConstants.Rule.PROPOSITION_TREE_GROUP_ID,
359                        "error.rule.proposition.simple.blankField", proposition.getDescription(), "Operator");
360                result &= false;
361            }
362            if (StringUtils.isBlank(propConstant) && !operator.endsWith("null")) { // ==null and !=null operators have blank values.
363                GlobalVariables.getMessageMap().putErrorForSectionId(KRMSPropertyConstants.Rule.PROPOSITION_TREE_GROUP_ID,
364                        "error.rule.proposition.simple.blankField", proposition.getDescription(), "Value");
365                result &= false;
366            }  else if (operator.endsWith("null")) { // ==null and !=null operators have blank values.
367                if (propConstant != null) {
368                    proposition.getParameters().get(1).setValue(null);
369                }
370            } else if (!StringUtils.isBlank(termId)) {
371                // validate that the constant value is comparable against the term
372                String termType = lookupTermType(termId);
373                ComparisonOperatorService comparisonOperatorService = KrmsApiServiceLocator.getComparisonOperatorService();
374                if (comparisonOperatorService.canCoerce(termType, propConstant)) {
375                    if (comparisonOperatorService.coerce(termType, propConstant) == null) { // HMM, what if we wanted a rule that
376                    // checked a null value?
377                        GlobalVariables.getMessageMap().putError(KRMSPropertyConstants.Rule.PROPOSITION_TREE_GROUP_ID,
378                                "error.rule.proposition.simple.invalidValue", proposition.getDescription(), propConstant);
379                        result &= false;
380                    }
381                }
382            }
383    
384            if (!CollectionUtils.isEmpty(proposition.getCompoundComponents())) {
385                GlobalVariables.getMessageMap().putError(KRMSPropertyConstants.Rule.PROPOSITION_TREE_GROUP_ID,
386                        "error.rule.proposition.simple.hasChildren", proposition.getDescription());
387                result &= false; // simple prop should not have compound components
388            }
389            return result;
390        }
391    
392        /**
393         * Validate the term in the given simple proposition.  Note that this method is side-effecting,
394         * when errors are detected with the proposition, errors are added to the error map.
395         * @param proposition the proposition with the term to validate
396         * @param namespace the namespace of the parent rule
397         * @return true if the proposition's term is considered valid
398         */
399        private boolean validateTerm(PropositionBo proposition, String namespace) {
400            boolean result = true;
401    
402            String termId = proposition.getParameters().get(0).getValue();
403            if (termId.startsWith(KrmsImplConstants.PARAMETERIZED_TERM_PREFIX)) {
404                // validate parameterized term
405    
406                // is the term name non-blank
407                if (StringUtils.isBlank(proposition.getNewTermDescription())) {
408                    GlobalVariables.getMessageMap().putErrorWithoutFullErrorPath(KRMSPropertyConstants.Rule.PROPOSITION_TREE_GROUP_ID,
409                            "error.rule.proposition.simple.emptyTermName", proposition.getDescription());
410                    result &= false;
411                } else { // check if the term name is unique
412    
413                    Map<String, String> criteria = new HashMap<String, String>();
414    
415                    criteria.put("description", proposition.getNewTermDescription());
416                    criteria.put("specification.namespace", namespace);
417    
418                    Collection<TermBo> matchingTerms =
419                            KRADServiceLocator.getBusinessObjectService().findMatching(TermBo.class, criteria);
420    
421                    if (!CollectionUtils.isEmpty(matchingTerms)) {
422                        // this is a Warning -- maybe it should be an error?
423                        GlobalVariables.getMessageMap().putWarningWithoutFullErrorPath(KRMSPropertyConstants.Rule.PROPOSITION_TREE_GROUP_ID,
424                                "warning.rule.proposition.simple.duplicateTermName", proposition.getDescription());
425                    }
426                }
427    
428                String termSpecificationId = termId.substring(KrmsImplConstants.PARAMETERIZED_TERM_PREFIX.length());
429    
430                TermResolverDefinition termResolverDefinition =
431                        AgendaEditorMaintainable.getSimplestTermResolver(termSpecificationId, namespace);
432    
433                if (termResolverDefinition == null) {
434                    GlobalVariables.getMessageMap().putErrorWithoutFullErrorPath(KRMSPropertyConstants.Rule.PROPOSITION_TREE_GROUP_ID,
435                            "error.rule.proposition.simple.invalidTerm", proposition.getDescription());
436                    result &= false;
437                } else {
438                    List<String> parameterNames = new ArrayList<String>(termResolverDefinition.getParameterNames());
439                    Collections.sort(parameterNames);
440                    for (String parameterName : parameterNames) {
441                        if (!proposition.getTermParameters().containsKey(parameterName) || 
442                                StringUtils.isBlank(proposition.getTermParameters().get(parameterName))) {
443                            GlobalVariables.getMessageMap().putErrorWithoutFullErrorPath(KRMSPropertyConstants.Rule.PROPOSITION_TREE_GROUP_ID,
444                                    "error.rule.proposition.simple.missingTermParameter", proposition.getDescription());
445                            result &= false;
446                            break;
447                        }
448                    }
449                }
450    
451            } else {
452                //validate normal term
453                TermDefinition termDefinition = KrmsRepositoryServiceLocator.getTermBoService().getTerm(termId);
454                if (termDefinition == null) {
455                    GlobalVariables.getMessageMap().putErrorWithoutFullErrorPath(KRMSPropertyConstants.Rule.PROPOSITION_TREE_GROUP_ID,
456                            "error.rule.proposition.simple.invalidTerm", proposition.getDescription());
457                } else if (!namespace.equals(termDefinition.getSpecification().getNamespace())) {
458                    GlobalVariables.getMessageMap().putErrorWithoutFullErrorPath(KRMSPropertyConstants.Rule.PROPOSITION_TREE_GROUP_ID,
459                            "error.rule.proposition.simple.invalidTerm", proposition.getDescription());
460                }
461            }
462            return result;
463        }
464    
465        /**
466         * Lookup the {@link org.kuali.rice.krms.api.repository.term.TermSpecificationDefinitionContract} type.
467         * @param key krms_term_t key
468         * @return String the krms_term_spec_t TYP for the given krms_term_t key given
469         */
470        private String lookupTermType(String key) {
471            TermSpecificationDefinition termSpec = null;
472            if (key.startsWith(KrmsImplConstants.PARAMETERIZED_TERM_PREFIX)) {
473                String termSpecificationId = key.substring(KrmsImplConstants.PARAMETERIZED_TERM_PREFIX.length());
474                termSpec = KrmsRepositoryServiceLocator.getTermBoService().getTermSpecificationById(termSpecificationId);
475            } else {
476                TermDefinition term = KrmsRepositoryServiceLocator.getTermBoService().getTerm(key);
477                if (term != null) {
478                    termSpec = term.getSpecification();
479                }
480            }
481            if (termSpec != null) {
482                return termSpec.getType();
483            } else {
484                return null;
485            }
486        }
487    
488    
489        /**
490         * This method returns the agendaId of the given agenda.  If the agendaId is null a new id will be created.
491         */
492        private String getCreateAgendaId(AgendaBo agenda) {
493            if (agenda.getId() == null) {
494                agenda.setId(getSequenceAccessorService().getNextAvailableSequenceNumber("KRMS_AGENDA_S").toString());
495            }
496            return agenda.getId();
497        }
498    
499        private void updateRuleAction(AgendaEditor agendaEditor) {
500            agendaEditor.getAgendaItemLine().getRule().setActions(new ArrayList<ActionBo>());
501            if (StringUtils.isNotBlank(agendaEditor.getAgendaItemLineRuleAction().getTypeId())) {
502                agendaEditor.getAgendaItemLineRuleAction().setAttributes(agendaEditor.getCustomRuleActionAttributesMap());
503                agendaEditor.getAgendaItemLine().getRule().getActions().add(agendaEditor.getAgendaItemLineRuleAction());
504            }
505        }
506    
507        /**
508         * Build a map from attribute name to attribute definition from all the defined attribute definitions for the
509         * specified rule action type
510         * @param actionTypeId
511         * @return
512         */
513        private Map<String, KrmsAttributeDefinition> buildAttributeDefinitionMap(String actionTypeId) {
514            KrmsAttributeDefinitionService attributeDefinitionService =
515                KrmsRepositoryServiceLocator.getKrmsAttributeDefinitionService();
516    
517            // build a map from attribute name to definition
518            Map<String, KrmsAttributeDefinition> attributeDefinitionMap = new HashMap<String, KrmsAttributeDefinition>();
519    
520            List<KrmsAttributeDefinition> attributeDefinitions =
521                    attributeDefinitionService.findAttributeDefinitionsByType(actionTypeId);
522    
523            for (KrmsAttributeDefinition attributeDefinition : attributeDefinitions) {
524                attributeDefinitionMap.put(attributeDefinition.getName(), attributeDefinition);
525            }
526            return attributeDefinitionMap;
527        }
528    
529        /**
530         * This method updates the existing rule in the agenda.
531         */
532        @RequestMapping(params = "methodToCall=" + "editRule")
533        public ModelAndView editRule(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
534                HttpServletRequest request, HttpServletResponse response) throws Exception {
535            AgendaEditor agendaEditor = getAgendaEditor(form);
536            // this is the root of the tree:
537            AgendaItemBo firstItem = getFirstAgendaItem(agendaEditor.getAgenda());
538            AgendaItemBo node = getAgendaItemById(firstItem, getSelectedAgendaItemId(form));
539            AgendaItemBo agendaItemLine = agendaEditor.getAgendaItemLine();
540    
541            if (!validateProposition(agendaItemLine.getRule().getProposition(), agendaItemLine.getRule().getNamespace())) {
542                form.getActionParameters().put(UifParameters.NAVIGATE_TO_PAGE_ID, "AgendaEditorView-EditRule-Page");
543                // NOTICE short circuit method on invalid proposition
544                return super.navigate(form, result, request, response);
545            }
546    
547            agendaItemLine.getRule().setAttributes(agendaEditor.getCustomRuleAttributesMap());
548            updateRuleAction(agendaEditor);
549    
550            AgendaEditorBusRule rule = new AgendaEditorBusRule();
551            MaintenanceForm maintenanceForm = (MaintenanceForm) form;
552            MaintenanceDocument document = maintenanceForm.getDocument();
553            if (rule.processAgendaItemBusinessRules(document)) {
554                node.setRule(agendaItemLine.getRule());
555                form.getActionParameters().put(UifParameters.NAVIGATE_TO_PAGE_ID, "AgendaEditorView-Agenda-Page");
556            } else {
557                form.getActionParameters().put(UifParameters.NAVIGATE_TO_PAGE_ID, "AgendaEditorView-EditRule-Page");
558            }
559            return super.navigate(form, result, request, response);
560        }
561    
562        /**
563         * @return the ALWAYS {@link AgendaItemInstanceChildAccessor} for the last ALWAYS child of the instance accessed by the parameter.
564         * It will by definition refer to null.  If the instanceAccessor parameter refers to null, then it will be returned.  This is useful
565         * for adding a youngest child to a sibling group.
566         */
567        private AgendaItemInstanceChildAccessor getLastChildsAlwaysAccessor(AgendaItemInstanceChildAccessor instanceAccessor) {
568            AgendaItemBo next = instanceAccessor.getChild();
569            if (next == null) return instanceAccessor;
570            while (next.getAlways() != null) { next = next.getAlways(); };
571            return new AgendaItemInstanceChildAccessor(AgendaItemChildAccessor.always, next);
572        }
573    
574        /**
575         * @return the accessor to the child with the given agendaItemId under the given parent.  This method will search both When TRUE and 
576         * When FALSE sibling groups.  If the instance with the given id is not found, null is returned.
577         * @see AgendaItemChildAccessor for nomenclature explanation
578         */
579        private AgendaItemInstanceChildAccessor getInstanceAccessorToChild(AgendaItemBo parent, String agendaItemId) {
580    
581            // first try When TRUE, then When FALSE via AgendaItemChildAccessor.levelOrderChildren
582            for (AgendaItemChildAccessor levelOrderChildAccessor : AgendaItemChildAccessor.children) {
583    
584                AgendaItemBo next = levelOrderChildAccessor.getChild(parent);
585                
586                // if the first item matches, return the accessor from the parent
587                if (next != null && agendaItemId.equals(next.getId())) return new AgendaItemInstanceChildAccessor(levelOrderChildAccessor, parent);
588    
589                // otherwise walk the children
590                while (next != null && next.getAlwaysId() != null) {
591                    if (next.getAlwaysId().equals(agendaItemId)) return new AgendaItemInstanceChildAccessor(AgendaItemChildAccessor.always, next);
592                    // move down
593                    next = next.getAlways();
594                }
595            }
596            
597            return null;
598        }
599    
600        @RequestMapping(params = "methodToCall=" + "ajaxRefresh")
601        public ModelAndView ajaxRefresh(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
602                HttpServletRequest request, HttpServletResponse response)
603                throws Exception {
604            // call the super method to avoid the agenda tree being reloaded from the db
605            return getUIFModelAndView(form);
606        }
607    
608        @RequestMapping(params = "methodToCall=" + "moveUp")
609        public ModelAndView moveUp(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
610                HttpServletRequest request, HttpServletResponse response)
611                throws Exception {
612            moveSelectedSubtreeUp(form);
613    
614            return super.refresh(form, result, request, response);
615        }
616    
617        @RequestMapping(params = "methodToCall=" + "ajaxMoveUp")
618        public ModelAndView ajaxMoveUp(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
619                HttpServletRequest request, HttpServletResponse response)
620                throws Exception {
621            moveSelectedSubtreeUp(form);
622    
623            // call the super method to avoid the agenda tree being reloaded from the db
624            return getUIFModelAndView(form);
625        }
626    
627        /**
628         *
629         * @param form
630         * @see AgendaItemChildAccessor for nomenclature explanation
631         */
632        private void moveSelectedSubtreeUp(UifFormBase form) {
633    
634            /* Rough algorithm for moving a node up.  This is a "level order" move.  Note that in this tree,
635             * level order means something a bit funky.  We are defining a level as it would be displayed in the browser,
636             * so only the traversal of When FALSE or When TRUE links increments the level, since ALWAYS linked nodes are
637             * considered siblings.
638             *
639             * find the following:
640             *   node := the selected node
641             *   parent := the selected node's parent, its containing node (via when true or when false relationship)
642             *   parentsOlderCousin := the parent's level-order predecessor (sibling or cousin)
643             *
644             * if (node is first child in sibling group)
645             *     if (node is in When FALSE group)
646             *         move node to last position in When TRUE group
647             *     else
648             *         find youngest child of parentsOlderCousin and put node after it
649             * else
650             *     move node up within its sibling group
651             */
652    
653            AgendaEditor agendaEditor = getAgendaEditor(form);
654            // this is the root of the tree:
655            AgendaItemBo firstItem = getFirstAgendaItem(agendaEditor.getAgenda());
656    
657            String selectedItemId = agendaEditor.getSelectedAgendaItemId();
658            AgendaItemBo node = getAgendaItemById(firstItem, selectedItemId);
659            AgendaItemBo parent = getParent(firstItem, selectedItemId);
660            AgendaItemBo parentsOlderCousin = (parent == null) ? null : getNextOldestOfSameGeneration(firstItem, parent);
661    
662            StringBuilder ruleEditorMessage = new StringBuilder();
663            AgendaItemChildAccessor childAccessor = getOldestChildAccessor(node, parent);
664            if (childAccessor != null) { // node is first child in sibling group
665                if (childAccessor == AgendaItemChildAccessor.whenFalse) {
666                    // move node to last position in When TRUE group
667                    AgendaItemInstanceChildAccessor youngestWhenTrueSiblingInsertionPoint =
668                            getLastChildsAlwaysAccessor(new AgendaItemInstanceChildAccessor(AgendaItemChildAccessor.whenTrue, parent));
669                    youngestWhenTrueSiblingInsertionPoint.setChild(node);
670                    AgendaItemChildAccessor.whenFalse.setChild(parent, node.getAlways());
671                    AgendaItemChildAccessor.always.setChild(node, null);
672    
673                    ruleEditorMessage.append("Moved ").append(node.getRule().getName()).append(" up ");
674                    ruleEditorMessage.append("to last position in When TRUE group of ").append(parent.getRule().getName());
675                } else if (parentsOlderCousin != null) {
676                    // find youngest child of parentsOlderCousin and put node after it
677                    AgendaItemInstanceChildAccessor youngestWhenFalseSiblingInsertionPoint =
678                            getLastChildsAlwaysAccessor(new AgendaItemInstanceChildAccessor(AgendaItemChildAccessor.whenFalse, parentsOlderCousin));
679                    youngestWhenFalseSiblingInsertionPoint.setChild(node);
680                    AgendaItemChildAccessor.whenTrue.setChild(parent, node.getAlways());
681                    AgendaItemChildAccessor.always.setChild(node, null);
682                    ruleEditorMessage.append("Moved ").append(node.getRule().getName()).append(" up ");
683                    ruleEditorMessage.append("to When FALSE group of ").append(parentsOlderCousin.getRule().getName());
684                }
685            } else if (!selectedItemId.equals(firstItem.getId())) { // conditional to miss special case of first node
686    
687                AgendaItemBo bogusRootNode = null;
688                if (parent == null) {
689                    // special case, this is a top level sibling. rig up special parent node
690                    bogusRootNode = new AgendaItemBo();
691                    AgendaItemChildAccessor.whenTrue.setChild(bogusRootNode, firstItem);
692                    parent = bogusRootNode;
693                }
694    
695                // move node up within its sibling group
696                AgendaItemInstanceChildAccessor accessorToSelectedNode = getInstanceAccessorToChild(parent, node.getId());
697                AgendaItemBo olderSibling = accessorToSelectedNode.getInstance();
698                AgendaItemInstanceChildAccessor accessorToOlderSibling = getInstanceAccessorToChild(parent, olderSibling.getId());
699    
700                accessorToOlderSibling.setChild(node);
701                accessorToSelectedNode.setChild(node.getAlways());
702                AgendaItemChildAccessor.always.setChild(node, olderSibling);
703    
704                ruleEditorMessage.append("Moved ").append(node.getRule().getName()).append(" up ");
705    
706                if (bogusRootNode != null) {
707                    // clean up special case with bogus root node
708                    agendaEditor.getAgenda().setFirstItemId(bogusRootNode.getWhenTrueId());
709                    ruleEditorMessage.append(" to ").append(getFirstAgendaItem(agendaEditor.getAgenda()).getRule().getName()).append(" When TRUE group");
710                } else {
711                    ruleEditorMessage.append(" within its sibling group, above " + olderSibling.getRule().getName());
712                }
713            }
714            agendaEditor.setRuleEditorMessage(ruleEditorMessage.toString());
715        }
716    
717        @RequestMapping(params = "methodToCall=" + "moveDown")
718        public ModelAndView moveDown(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
719                HttpServletRequest request, HttpServletResponse response)
720                throws Exception {
721            moveSelectedSubtreeDown(form);
722            
723            return super.refresh(form, result, request, response);
724        }
725    
726        @RequestMapping(params = "methodToCall=" + "ajaxMoveDown")
727        public ModelAndView ajaxMoveDown(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
728                HttpServletRequest request, HttpServletResponse response)
729                throws Exception {
730            moveSelectedSubtreeDown(form);
731    
732            // call the super method to avoid the agenda tree being reloaded from the db
733            return getUIFModelAndView(form);
734        }
735    
736        /**
737         *
738         * @param form
739         * @see AgendaItemChildAccessor for nomenclature explanation
740         */
741        private void moveSelectedSubtreeDown(UifFormBase form) {
742    
743            /* Rough algorithm for moving a node down.  This is a "level order" move.  Note that in this tree,
744             * level order means something a bit funky.  We are defining a level as it would be displayed in the browser,
745             * so only the traversal of When FALSE or When TRUE links increments the level, since ALWAYS linked nodes are
746             * considered siblings.
747             *
748             * find the following:
749             *   node := the selected node
750             *   parent := the selected node's parent, its containing node (via when true or when false relationship)
751             *   parentsYoungerCousin := the parent's level-order successor (sibling or cousin)
752             *
753             * if (node is last child in sibling group)
754             *     if (node is in When TRUE group)
755             *         move node to first position in When FALSE group
756             *     else
757             *         move to first child of parentsYoungerCousin
758             * else
759             *     move node down within its sibling group
760             */
761    
762            AgendaEditor agendaEditor = getAgendaEditor(form);
763            // this is the root of the tree:
764            AgendaItemBo firstItem = getFirstAgendaItem(agendaEditor.getAgenda());
765    
766            String selectedItemId = agendaEditor.getSelectedAgendaItemId();
767            AgendaItemBo node = getAgendaItemById(firstItem, selectedItemId);
768            AgendaItemBo parent = getParent(firstItem, selectedItemId);
769            AgendaItemBo parentsYoungerCousin = (parent == null) ? null : getNextYoungestOfSameGeneration(firstItem, parent);
770    
771            StringBuilder ruleEditorMessage = new StringBuilder();
772            if (node.getAlways() == null && parent != null) { // node is last child in sibling group
773                // set link to selected node to null
774                if (parent.getWhenTrue() != null && isSiblings(parent.getWhenTrue(), node)) { // node is in When TRUE group
775                    // move node to first child under When FALSE
776                    AgendaItemInstanceChildAccessor accessorToSelectedNode = getInstanceAccessorToChild(parent, node.getId());
777                    accessorToSelectedNode.setChild(null);
778    
779                    AgendaItemBo parentsFirstChild = parent.getWhenFalse();
780                    AgendaItemChildAccessor.whenFalse.setChild(parent, node);
781                    AgendaItemChildAccessor.always.setChild(node, parentsFirstChild);
782    
783                    ruleEditorMessage.append("Moved ").append(node.getRule().getName()).append(" down ");
784                    ruleEditorMessage.append("to first child under When FALSE group of ").append(parent.getRule().getName());
785                } else if (parentsYoungerCousin != null) { // node is in the When FALSE group
786                    // move to first child of parentsYoungerCousin under When TRUE
787                    AgendaItemInstanceChildAccessor accessorToSelectedNode = getInstanceAccessorToChild(parent, node.getId());
788                    accessorToSelectedNode.setChild(null);
789    
790                    AgendaItemBo parentsYoungerCousinsFirstChild = parentsYoungerCousin.getWhenTrue();
791                    AgendaItemChildAccessor.whenTrue.setChild(parentsYoungerCousin, node);
792                    AgendaItemChildAccessor.always.setChild(node, parentsYoungerCousinsFirstChild);
793    
794                    ruleEditorMessage.append("Moved ").append(node.getRule().getName()).append(" down ");
795                    ruleEditorMessage.append("to first child under When TRUE group of ").append(parentsYoungerCousin.getRule().getName());
796                }
797            } else if (node.getAlways() != null) { // move node down within its sibling group
798    
799                AgendaItemBo bogusRootNode = null;
800                if (parent == null) {
801                    // special case, this is a top level sibling. rig up special parent node
802    
803                    bogusRootNode = new AgendaItemBo();
804                    AgendaItemChildAccessor.whenFalse.setChild(bogusRootNode, firstItem);
805                    parent = bogusRootNode;
806                }
807    
808                // move node down within its sibling group
809                AgendaItemInstanceChildAccessor accessorToSelectedNode = getInstanceAccessorToChild(parent, node.getId());
810                AgendaItemBo youngerSibling = node.getAlways();
811                accessorToSelectedNode.setChild(youngerSibling);
812                AgendaItemChildAccessor.always.setChild(node, youngerSibling.getAlways());
813                AgendaItemChildAccessor.always.setChild(youngerSibling, node);
814    
815                if (bogusRootNode != null) {
816                    // clean up special case with bogus root node
817                    agendaEditor.getAgenda().setFirstItemId(bogusRootNode.getWhenFalseId());
818                }
819                ruleEditorMessage.append("Moved ").append(node.getRule().getName()).append(" down ");
820                ruleEditorMessage.append(" within its sibling group, below ").append(youngerSibling.getRule().getName());
821            } // falls through if already bottom-most
822            agendaEditor.setRuleEditorMessage(ruleEditorMessage.toString());
823        }
824    
825        @RequestMapping(params = "methodToCall=" + "moveLeft")
826        public ModelAndView moveLeft(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
827                HttpServletRequest request, HttpServletResponse response)
828                throws Exception {
829            moveSelectedSubtreeLeft(form);
830            
831            return super.refresh(form, result, request, response);
832        }
833    
834        @RequestMapping(params = "methodToCall=" + "ajaxMoveLeft")
835        public ModelAndView ajaxMoveLeft(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
836                HttpServletRequest request, HttpServletResponse response)
837                throws Exception {
838    
839            moveSelectedSubtreeLeft(form);
840    
841            // call the super method to avoid the agenda tree being reloaded from the db
842            return getUIFModelAndView(form);
843        }
844    
845        /**
846         *
847         * @param form
848         * @see AgendaItemChildAccessor for nomenclature explanation
849         */
850        private void moveSelectedSubtreeLeft(UifFormBase form) {
851    
852            /*
853             * Move left means make it a younger sibling of it's parent.
854             */
855    
856            AgendaEditor agendaEditor = getAgendaEditor(form);
857            // this is the root of the tree:
858            AgendaItemBo firstItem = getFirstAgendaItem(agendaEditor.getAgenda());
859    
860            String selectedItemId = agendaEditor.getSelectedAgendaItemId();
861            AgendaItemBo node = getAgendaItemById(firstItem, selectedItemId);
862            AgendaItemBo parent = getParent(firstItem, selectedItemId);
863    
864            if (parent != null) {
865                AgendaItemInstanceChildAccessor accessorToSelectedNode = getInstanceAccessorToChild(parent, node.getId());
866                accessorToSelectedNode.setChild(node.getAlways());
867                AgendaItemChildAccessor.always.setChild(node, parent.getAlways());
868                AgendaItemChildAccessor.always.setChild(parent, node);
869    
870                StringBuilder ruleEditorMessage = new StringBuilder();
871                ruleEditorMessage.append("Moved ").append(node.getRule().getName()).append(" left to be a sibling of its parent ").append(parent.getRule().getName());
872                agendaEditor.setRuleEditorMessage(ruleEditorMessage.toString());
873            }
874        }
875    
876    
877        @RequestMapping(params = "methodToCall=" + "moveRight")
878        public ModelAndView moveRight(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
879                HttpServletRequest request, HttpServletResponse response)
880                throws Exception {
881    
882            moveSelectedSubtreeRight(form);
883    
884            return super.refresh(form, result, request, response);
885        }
886    
887        @RequestMapping(params = "methodToCall=" + "ajaxMoveRight")
888        public ModelAndView ajaxMoveRight(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
889                HttpServletRequest request, HttpServletResponse response)
890                throws Exception {
891    
892            moveSelectedSubtreeRight(form);
893    
894            // call the super method to avoid the agenda tree being reloaded from the db
895            return getUIFModelAndView(form);
896        }
897    
898        /**
899         *
900         * @param form
901         * @see AgendaItemChildAccessor for nomenclature explanation
902         */
903        private void moveSelectedSubtreeRight(UifFormBase form) {
904    
905            /*
906             * Move right prefers moving to bottom of upper sibling's When FALSE branch
907             * ... otherwise ..
908             * moves to top of lower sibling's When TRUE branch
909             */
910    
911            AgendaEditor agendaEditor = getAgendaEditor(form);
912            // this is the root of the tree:
913            AgendaItemBo firstItem = getFirstAgendaItem(agendaEditor.getAgenda());
914    
915            String selectedItemId = agendaEditor.getSelectedAgendaItemId();
916            AgendaItemBo node = getAgendaItemById(firstItem, selectedItemId);
917            AgendaItemBo parent = getParent(firstItem, selectedItemId);
918    
919            AgendaItemBo bogusRootNode = null;
920            if (parent == null) {
921                // special case, this is a top level sibling. rig up special parent node
922                bogusRootNode = new AgendaItemBo();
923                AgendaItemChildAccessor.whenFalse.setChild(bogusRootNode, firstItem);
924                parent = bogusRootNode;
925            }
926    
927            AgendaItemInstanceChildAccessor accessorToSelectedNode = getInstanceAccessorToChild(parent, node.getId());
928            AgendaItemBo olderSibling = (accessorToSelectedNode.getInstance() == parent) ? null : accessorToSelectedNode.getInstance();
929    
930            StringBuilder ruleEditorMessage = new StringBuilder();
931            if (olderSibling != null) {
932                accessorToSelectedNode.setChild(node.getAlways());
933                AgendaItemInstanceChildAccessor yougestWhenFalseSiblingInsertionPoint =
934                        getLastChildsAlwaysAccessor(new AgendaItemInstanceChildAccessor(AgendaItemChildAccessor.whenFalse, olderSibling));
935                yougestWhenFalseSiblingInsertionPoint.setChild(node);
936                AgendaItemChildAccessor.always.setChild(node, null);
937    
938                ruleEditorMessage.append("Moved ").append(node.getRule().getName()).append(" right to ");
939                ruleEditorMessage.append(olderSibling.getRule().getName()).append(" When FALSE group.");
940            } else if (node.getAlways() != null) { // has younger sibling
941                accessorToSelectedNode.setChild(node.getAlways());
942                AgendaItemBo childsWhenTrue = node.getAlways().getWhenTrue();
943                AgendaItemChildAccessor.whenTrue.setChild(node.getAlways(), node);
944                AgendaItemChildAccessor.always.setChild(node, childsWhenTrue);
945    
946                ruleEditorMessage.append("Moved ").append(node.getRule().getName()).append(" right to ");
947                if (childsWhenTrue != null) { // childsWhenTrue is null if the topmost rule is moved right see bogusRootNode below
948                    ruleEditorMessage.append(childsWhenTrue.getRule().getName()).append(" When TRUE group");
949                }
950            } // falls through if node is already the rightmost.
951    
952            if (bogusRootNode != null) {
953                // clean up special case with bogus root node
954                agendaEditor.getAgenda().setFirstItemId(bogusRootNode.getWhenFalseId());
955                ruleEditorMessage.append(getFirstAgendaItem(agendaEditor.getAgenda()).getRule().getName()).append(" When TRUE group");
956            }
957            agendaEditor.setRuleEditorMessage(ruleEditorMessage.toString());
958        }
959    
960        /**
961         *
962         * @param cousin1
963         * @param cousin2
964         * @return
965         * @see AgendaItemChildAccessor for nomenclature explanation
966         */
967        private boolean isSiblings(AgendaItemBo cousin1, AgendaItemBo cousin2) {
968            if (cousin1.equals(cousin2)) return true; // this is a bit abusive
969            
970            // can you walk to c1 from ALWAYS links of c2?
971            AgendaItemBo candidate = cousin2;
972            while (null != (candidate = candidate.getAlways())) {
973                if (candidate.equals(cousin1)) return true;
974            }
975            // can you walk to c2 from ALWAYS links of c1?
976            candidate = cousin1;
977            while (null != (candidate = candidate.getAlways())) {
978                if (candidate.equals(cousin2)) return true;
979            }
980            return false;
981        }
982    
983        /**
984         * This method returns the level order accessor (getWhenTrue or getWhenFalse) that relates the parent directly 
985         * to the child.  If the two nodes don't have such a relationship, null is returned. 
986         * Note that this only finds accessors for oldest children, not younger siblings.
987         * @see AgendaItemChildAccessor for nomenclature explanation
988         */
989        private AgendaItemChildAccessor getOldestChildAccessor(
990                AgendaItemBo child, AgendaItemBo parent) {
991            AgendaItemChildAccessor levelOrderChildAccessor = null;
992            
993            if (parent != null) {
994                for (AgendaItemChildAccessor childAccessor : AgendaItemChildAccessor.children) {
995                    if (child.equals(childAccessor.getChild(parent))) {
996                        levelOrderChildAccessor = childAccessor;
997                        break;
998                    }
999                }
1000            }
1001            return levelOrderChildAccessor;
1002        }
1003        
1004        /**
1005         * This method finds and returns the first agenda item in the agenda, or null if there are no items presently
1006         * 
1007         * @param agenda
1008         * @return
1009         */
1010        private AgendaItemBo getFirstAgendaItem(AgendaBo agenda) {
1011            AgendaItemBo firstItem = null;
1012            if (agenda != null && agenda.getItems() != null) for (AgendaItemBo agendaItem : agenda.getItems()) {
1013                if (agenda.getFirstItemId().equals(agendaItem.getId())) {
1014                    firstItem = agendaItem;
1015                    break;
1016                }
1017            }
1018            return firstItem;
1019        }
1020        
1021        /**
1022         * @return the closest younger sibling of the agenda item with the given ID, and if there is no such sibling, the closest younger cousin.
1023         * If there is no such cousin either, then null is returned.
1024         * @see AgendaItemChildAccessor for nomenclature explanation
1025         */
1026        private AgendaItemBo getNextYoungestOfSameGeneration(AgendaItemBo root, AgendaItemBo agendaItem) {
1027    
1028            int genNumber = getAgendaItemGenerationNumber(0, root, agendaItem.getId());
1029            List<AgendaItemBo> genList = new ArrayList<AgendaItemBo>();
1030            buildAgendaItemGenerationList(genList, root, 0, genNumber);
1031    
1032            int itemIndex = genList.indexOf(agendaItem);
1033            if (genList.size() > itemIndex + 1) return genList.get(itemIndex + 1);
1034    
1035            return null;
1036        }
1037    
1038        /**
1039         *
1040         * @param currentLevel
1041         * @param node
1042         * @param agendaItemId
1043         * @return
1044         * @see AgendaItemChildAccessor for nomenclature explanation
1045         */
1046        private int getAgendaItemGenerationNumber(int currentLevel, AgendaItemBo node, String agendaItemId) {
1047            int result = -1;
1048            if (agendaItemId.equals(node.getId())) {
1049                result = currentLevel;
1050            } else {
1051                for (AgendaItemChildAccessor childAccessor : AgendaItemChildAccessor.linkedNodes) {
1052                    AgendaItemBo child = childAccessor.getChild(node);
1053                    if (child != null) {
1054                        int nextLevel = currentLevel;
1055                        // we don't change the level order parent when we traverse ALWAYS links
1056                        if (childAccessor != AgendaItemChildAccessor.always) {
1057                            nextLevel = currentLevel +1;
1058                        }
1059                        result = getAgendaItemGenerationNumber(nextLevel, child, agendaItemId);
1060                        if (result != -1) break;
1061                    }
1062                }
1063            }
1064            return result;
1065        }
1066    
1067        /**
1068         *
1069         * @param genList
1070         * @param node
1071         * @param currentLevel
1072         * @param generation
1073         * @see AgendaItemChildAccessor for nomenclature explanation
1074         */
1075        private void buildAgendaItemGenerationList(List<AgendaItemBo> genList, AgendaItemBo node, int currentLevel, int generation) {
1076            if (currentLevel == generation) {
1077                genList.add(node);
1078            }
1079    
1080            if (currentLevel > generation) return;
1081    
1082            for (AgendaItemChildAccessor childAccessor : AgendaItemChildAccessor.linkedNodes) {
1083                AgendaItemBo child = childAccessor.getChild(node);
1084                if (child != null) {
1085                    int nextLevel = currentLevel;
1086                    // we don't change the level order parent when we traverse ALWAYS links
1087                    if (childAccessor != AgendaItemChildAccessor.always) {
1088                        nextLevel = currentLevel +1;
1089                    }
1090                    buildAgendaItemGenerationList(genList, child, nextLevel, generation);
1091                }
1092            }
1093        }
1094    
1095        /**
1096         * @return the closest older sibling of the agenda item with the given ID, and if there is no such sibling, the closest older cousin.
1097         * If there is no such cousin either, then null is returned.
1098         * @see AgendaItemChildAccessor for nomenclature explanation
1099         */
1100        private AgendaItemBo getNextOldestOfSameGeneration(AgendaItemBo root, AgendaItemBo agendaItem) {
1101    
1102            int genNumber = getAgendaItemGenerationNumber(0, root, agendaItem.getId());
1103            List<AgendaItemBo> genList = new ArrayList<AgendaItemBo>();
1104            buildAgendaItemGenerationList(genList, root, 0, genNumber);
1105    
1106            int itemIndex = genList.indexOf(agendaItem);
1107            if (itemIndex >= 1) return genList.get(itemIndex - 1);
1108    
1109            return null;
1110        }
1111        
1112    
1113        /**
1114         * returns the parent of the item with the passed in id.  Note that {@link AgendaItemBo}s related by ALWAYS relationships are considered siblings.
1115         * @see AgendaItemChildAccessor for nomenclature explanation
1116         */
1117        private AgendaItemBo getParent(AgendaItemBo root, String agendaItemId) {
1118            return getParentHelper(root, null, agendaItemId);
1119        }
1120    
1121        /**
1122         *
1123         * @param node
1124         * @param levelOrderParent
1125         * @param agendaItemId
1126         * @return
1127         * @see AgendaItemChildAccessor for nomenclature explanation
1128         */
1129        private AgendaItemBo getParentHelper(AgendaItemBo node, AgendaItemBo levelOrderParent, String agendaItemId) {
1130            AgendaItemBo result = null;
1131            if (agendaItemId.equals(node.getId())) {
1132                result = levelOrderParent;
1133            } else {
1134                for (AgendaItemChildAccessor childAccessor : AgendaItemChildAccessor.linkedNodes) {
1135                    AgendaItemBo child = childAccessor.getChild(node);
1136                    if (child != null) {
1137                        // we don't change the level order parent when we traverse ALWAYS links 
1138                        AgendaItemBo lop = (childAccessor == AgendaItemChildAccessor.always) ? levelOrderParent : node;
1139                        result = getParentHelper(child, lop, agendaItemId);
1140                        if (result != null) break;
1141                    }
1142                }
1143            }
1144            return result;
1145        }
1146    
1147        /**
1148         * Search the tree for the agenda item with the given id.
1149         */
1150        private AgendaItemBo getAgendaItemById(AgendaItemBo node, String agendaItemId) {
1151            if (node == null) throw new IllegalArgumentException("node must be non-null");
1152    
1153            AgendaItemBo result = null;
1154            
1155            if (agendaItemId.equals(node.getId())) {
1156                result = node;
1157            } else {
1158                for (AgendaItemChildAccessor childAccessor : AgendaItemChildAccessor.linkedNodes) {
1159                    AgendaItemBo child = childAccessor.getChild(node);
1160                    if (child != null) {
1161                        result = getAgendaItemById(child, agendaItemId);
1162                        if (result != null) break;
1163                    }
1164                }
1165            } 
1166            return result;
1167        }
1168    
1169        /**
1170         * @param form
1171         * @return the {@link AgendaEditor} from the form
1172         */
1173        private AgendaEditor getAgendaEditor(UifFormBase form) {
1174            MaintenanceForm maintenanceForm = (MaintenanceForm) form;
1175            return ((AgendaEditor)maintenanceForm.getDocument().getDocumentDataObject());
1176        }
1177    
1178        private void treeToInOrderList(AgendaItemBo agendaItem, List<AgendaItemBo> listToBuild) {
1179            listToBuild.add(agendaItem);
1180            for (AgendaItemChildAccessor childAccessor : AgendaItemChildAccessor.linkedNodes) {
1181                AgendaItemBo child = childAccessor.getChild(agendaItem);
1182                if (child != null) treeToInOrderList(child, listToBuild);
1183            }
1184        }
1185    
1186        
1187        @RequestMapping(params = "methodToCall=" + "delete")
1188        public ModelAndView delete(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1189                HttpServletRequest request, HttpServletResponse response)
1190                throws Exception {
1191    
1192            deleteSelectedSubtree(form);
1193    
1194            return super.refresh(form, result, request, response);
1195        }
1196    
1197        @RequestMapping(params = "methodToCall=" + "ajaxDelete")
1198        public ModelAndView ajaxDelete(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1199                HttpServletRequest request, HttpServletResponse response)
1200                throws Exception {
1201    
1202            deleteSelectedSubtree(form);
1203    
1204            // call the super method to avoid the agenda tree being reloaded from the db
1205            return getUIFModelAndView(form);
1206        }
1207    
1208        
1209        private void deleteSelectedSubtree(UifFormBase form) {
1210            AgendaEditor agendaEditor = getAgendaEditor(form);
1211            AgendaItemBo firstItem = getFirstAgendaItem(agendaEditor.getAgenda());
1212    
1213            if (firstItem != null) {
1214                String agendaItemSelected = agendaEditor.getSelectedAgendaItemId();
1215                AgendaItemBo selectedItem = getAgendaItemById(firstItem, agendaItemSelected);
1216    
1217                // need to handle the first item here, our recursive method won't handle it.
1218                if (agendaItemSelected.equals(firstItem.getId())) {
1219                    agendaEditor.getAgenda().setFirstItemId(firstItem.getAlwaysId());
1220                } else {
1221                    deleteAgendaItem(firstItem, agendaItemSelected);
1222                }
1223    
1224                StringBuilder ruleEditorMessage = new StringBuilder();
1225                ruleEditorMessage.append("Deleted ").append(selectedItem.getRule().getName());
1226                // remove agenda item and its whenTrue & whenFalse children from the list of agendaItems of the agenda
1227                if (selectedItem.getWhenTrue() != null) {
1228                    removeAgendaItem(agendaEditor.getAgenda().getItems(), selectedItem.getWhenTrue());
1229                    ruleEditorMessage.append(" and its When TRUE ").append(selectedItem.getWhenTrue().getRule().getName());
1230                }
1231                if (selectedItem.getWhenFalse() != null) {
1232                    removeAgendaItem(agendaEditor.getAgenda().getItems(), selectedItem.getWhenFalse());
1233                    ruleEditorMessage.append(" and its When FALSE ").append(selectedItem.getWhenFalse().getRule().getName());
1234                }
1235                agendaEditor.getAgenda().getItems().remove(selectedItem);
1236                agendaEditor.setRuleEditorMessage(ruleEditorMessage.toString());
1237            }
1238        }
1239    
1240        private void deleteAgendaItem(AgendaItemBo root, String agendaItemIdToDelete) {
1241            if (deleteAgendaItem(root, AgendaItemChildAccessor.whenTrue, agendaItemIdToDelete) || 
1242                    deleteAgendaItem(root, AgendaItemChildAccessor.whenFalse, agendaItemIdToDelete) || 
1243                    deleteAgendaItem(root, AgendaItemChildAccessor.always, agendaItemIdToDelete)); // TODO: this is confusing, refactor
1244        }
1245        
1246        private boolean deleteAgendaItem(AgendaItemBo agendaItem, AgendaItemChildAccessor childAccessor, String agendaItemIdToDelete) {
1247            if (agendaItem == null || childAccessor.getChild(agendaItem) == null) return false;
1248            if (agendaItemIdToDelete.equals(childAccessor.getChild(agendaItem).getId())) {
1249                // delete the child in such a way that any ALWAYS children don't get lost from the tree
1250                AgendaItemBo grandchildToKeep = childAccessor.getChild(agendaItem).getAlways();
1251                childAccessor.setChild(agendaItem, grandchildToKeep);
1252                return true;
1253            } else {
1254                AgendaItemBo child = childAccessor.getChild(agendaItem);
1255                // recurse
1256                for (AgendaItemChildAccessor nextChildAccessor : AgendaItemChildAccessor.linkedNodes) {
1257                    if (deleteAgendaItem(child, nextChildAccessor, agendaItemIdToDelete)) return true;
1258                }
1259            }
1260            return false;
1261        }
1262    
1263        /**
1264         * Recursively delete the agendaItem and its children from the agendaItemBo list.
1265         * @param items, the list of agendaItemBo that the agenda holds
1266         * @param removeAgendaItem, the agendaItemBo to be removed
1267         */
1268        private void removeAgendaItem(List<AgendaItemBo> items, AgendaItemBo removeAgendaItem) {
1269            if (removeAgendaItem.getWhenTrue() != null) {
1270                removeAgendaItem(items, removeAgendaItem.getWhenTrue());
1271            }
1272            if (removeAgendaItem.getWhenFalse() != null) {
1273                removeAgendaItem(items, removeAgendaItem.getWhenFalse());
1274            }
1275            if (removeAgendaItem.getAlways() != null) {
1276                removeAgendaItem(items, removeAgendaItem.getAlways());
1277            }
1278            items.remove(removeAgendaItem);
1279        }
1280    
1281        @RequestMapping(params = "methodToCall=" + "ajaxCut")
1282        public ModelAndView ajaxCut(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1283                HttpServletRequest request, HttpServletResponse response) throws Exception {
1284    
1285            AgendaEditor agendaEditor = getAgendaEditor(form);
1286            // this is the root of the tree:
1287            AgendaItemBo firstItem = getFirstAgendaItem(agendaEditor.getAgenda());
1288            String selectedItemId = agendaEditor.getSelectedAgendaItemId();
1289    
1290            AgendaItemBo selectedAgendaItem = getAgendaItemById(firstItem, selectedItemId);
1291            setCutAgendaItemId(form, selectedItemId);
1292    
1293            StringBuilder ruleEditorMessage = new StringBuilder();
1294            ruleEditorMessage.append("Marked ").append(selectedAgendaItem.getRule().getName()).append(" for cutting.");
1295            agendaEditor.setRuleEditorMessage(ruleEditorMessage.toString());
1296            // call the super method to avoid the agenda tree being reloaded from the db
1297            return getUIFModelAndView(form);
1298        }
1299    
1300        @RequestMapping(params = "methodToCall=" + "ajaxPaste")
1301        public ModelAndView ajaxPaste(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1302                HttpServletRequest request, HttpServletResponse response) throws Exception {
1303    
1304            AgendaEditor agendaEditor = getAgendaEditor(form);
1305            // this is the root of the tree:
1306            AgendaItemBo firstItem = getFirstAgendaItem(agendaEditor.getAgenda());
1307            String selectedItemId = agendaEditor.getSelectedAgendaItemId();
1308    
1309            String agendaItemId = getCutAgendaItemId(form);
1310            if (StringUtils.isNotBlank(selectedItemId) && StringUtils.isNotBlank(agendaItemId)) {
1311                StringBuilder ruleEditorMessage = new StringBuilder();
1312                AgendaItemBo node = getAgendaItemById(firstItem, agendaItemId);
1313                AgendaItemBo orgRefNode = getReferringNode(firstItem, agendaItemId);
1314                AgendaItemBo newRefNode = getAgendaItemById(firstItem, selectedItemId);
1315    
1316                if (isSameOrChildNode(node, newRefNode)) {
1317                    // note if the cut agenda item is not cleared, then the javascript on the AgendaEditorView will need to be
1318                    // updated to deal with a paste that doesn't paste.  As the ui disables the paste button after it is clicked
1319                    ruleEditorMessage.append("Cannot paste ").append(node.getRule().getName()).append(" to itself.");
1320                } else {
1321                    // remove node
1322                    if (orgRefNode == null) {
1323                        agendaEditor.getAgenda().setFirstItemId(node.getAlwaysId());
1324                    } else {
1325                        // determine if true, false or always
1326                        // do appropriate operation
1327                        if (node.getId().equals(orgRefNode.getWhenTrueId())) {
1328                            orgRefNode.setWhenTrueId(node.getAlwaysId());
1329                            orgRefNode.setWhenTrue(node.getAlways());
1330                        } else if(node.getId().equals(orgRefNode.getWhenFalseId())) {
1331                            orgRefNode.setWhenFalseId(node.getAlwaysId());
1332                            orgRefNode.setWhenFalse(node.getAlways());
1333                        } else {
1334                            orgRefNode.setAlwaysId(node.getAlwaysId());
1335                            orgRefNode.setAlways(node.getAlways());
1336                        }
1337                    }
1338    
1339                    // insert node
1340                    node.setAlwaysId(newRefNode.getAlwaysId());
1341                    node.setAlways(newRefNode.getAlways());
1342                    newRefNode.setAlwaysId(node.getId());
1343                    newRefNode.setAlways(node);
1344    
1345                    ruleEditorMessage.append(" Pasted ").append(node.getRule().getName());
1346                    ruleEditorMessage.append(" to ").append(newRefNode.getRule().getName());
1347                    agendaEditor.setRuleEditorMessage(ruleEditorMessage.toString());
1348    
1349                }
1350                setCutAgendaItemId(form, null);
1351            }
1352    
1353    
1354            // call the super method to avoid the agenda tree being reloaded from the db
1355            return getUIFModelAndView(form);
1356        }
1357    
1358        /**
1359         * Updates to the category call back to this method to set the categoryId appropriately
1360         * TODO: shouldn't this happen automatically?  We're taking it out of the form by hand here
1361         */
1362        @RequestMapping(params = "methodToCall=" + "ajaxCategoryChangeRefresh")
1363        public ModelAndView ajaxCategoryChangeRefresh(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1364                HttpServletRequest request, HttpServletResponse response)
1365                throws Exception {
1366    
1367            String categoryParamName = null;
1368            Enumeration paramNames = request.getParameterNames();
1369            while (paramNames.hasMoreElements()) {
1370                String paramName = paramNames.nextElement().toString();
1371                if (paramName.endsWith("categoryId")) {
1372                    categoryParamName = paramName;
1373                    break;
1374                }
1375            }
1376    
1377            if (categoryParamName != null) {
1378                String categoryId = request.getParameter(categoryParamName);
1379    
1380                if (StringUtils.isBlank(categoryId)) { categoryId = null; }
1381    
1382                AgendaEditor agendaEditor = getAgendaEditor(form);
1383                RuleBo rule = agendaEditor.getAgendaItemLine().getRule();
1384                String selectedPropId = agendaEditor.getSelectedPropositionId();
1385    
1386                // TODO: This should work even if the prop isn't selected!!!  Find the node in edit mode?
1387                if (!StringUtils.isBlank(selectedPropId)) {
1388                    Node<RuleTreeNode, String> selectedPropositionNode =
1389                            findPropositionTreeNode(rule.getPropositionTree().getRootElement(), selectedPropId);
1390                    selectedPropositionNode.getData().getProposition().setCategoryId(categoryId);
1391                }
1392            }
1393    
1394            return ajaxRefresh(form, result, request, response);
1395        }
1396    
1397        /**
1398         * This method checks if the node is the same as the new parent node or a when-true/when-fase
1399         * child of the new parent node.
1400         *
1401         * @param node - the node to be checked if it's the same or a child
1402         * @param newParent - the parent node to check against
1403         * @return true if same or child, false otherwise
1404         * @see AgendaItemChildAccessor for nomenclature explanation
1405         */
1406        private boolean isSameOrChildNode(AgendaItemBo node, AgendaItemBo newParent) {
1407            return isSameOrChildNodeHelper(node, newParent, AgendaItemChildAccessor.children);
1408        }
1409    
1410        private boolean isSameOrChildNodeHelper(AgendaItemBo node, AgendaItemBo newParent, AgendaItemChildAccessor[] childAccessors) {
1411            boolean result = false;
1412            if (newParent == null || node == null) {
1413                return false;
1414            }
1415            if (StringUtils.equals(node.getId(), newParent.getId())) {
1416                result = true;
1417            } else {
1418                for (AgendaItemChildAccessor childAccessor : childAccessors) {
1419                    AgendaItemBo child = childAccessor.getChild(node);
1420                    if (child != null) {
1421                        result = isSameOrChildNodeHelper(child, newParent, AgendaItemChildAccessor.linkedNodes);
1422                        if (result == true) break;
1423                    }
1424                }
1425            }
1426            return result;
1427        }
1428    
1429        /**
1430         * This method returns the node that points to the specified agendaItemId.
1431         * (returns the next older sibling or the parent if no older sibling exists)
1432         *
1433         * @param root - the first agenda item of the agenda
1434         * @param agendaItemId - agenda item id of the agenda item whose referring node is to be returned
1435         * @return AgendaItemBo that points to the specified agenda item
1436         * @see AgendaItemChildAccessor for nomenclature explanation
1437         */
1438        private AgendaItemBo getReferringNode(AgendaItemBo root, String agendaItemId) {
1439            return getReferringNodeHelper(root, null, agendaItemId);
1440        }
1441    
1442        private AgendaItemBo getReferringNodeHelper(AgendaItemBo node, AgendaItemBo referringNode, String agendaItemId) {
1443            AgendaItemBo result = null;
1444            if (agendaItemId.equals(node.getId())) {
1445                result = referringNode;
1446            } else {
1447                for (AgendaItemChildAccessor childAccessor : AgendaItemChildAccessor.linkedNodes) {
1448                    AgendaItemBo child = childAccessor.getChild(node);
1449                    if (child != null) {
1450                        result = getReferringNodeHelper(child, node, agendaItemId);
1451                        if (result != null) break;
1452                    }
1453                }
1454            }
1455            return result;
1456        }
1457    
1458        /**
1459         *  return the sequenceAssessorService
1460         */
1461        private SequenceAccessorService getSequenceAccessorService() {
1462            if ( sequenceAccessorService == null ) {
1463                sequenceAccessorService = KRADServiceLocator.getSequenceAccessorService();
1464            }
1465            return sequenceAccessorService;
1466        }
1467    
1468        /**
1469         * return the contextBoService
1470         */
1471        private ContextBoService getContextBoService() {
1472            return KrmsRepositoryServiceLocator.getContextBoService();
1473        }
1474    
1475        /**
1476         * return the contextBoService
1477         */
1478        private RuleBoService getRuleBoService() {
1479            return KrmsRepositoryServiceLocator.getRuleBoService();
1480        }
1481    
1482        /**
1483         * binds a child accessor to an AgendaItemBo instance.  An {@link AgendaItemInstanceChildAccessor} allows you to
1484         * get and set the referent
1485         */
1486        private static class AgendaItemInstanceChildAccessor {
1487            
1488            private final AgendaItemChildAccessor accessor;
1489            private final AgendaItemBo instance;
1490    
1491            public AgendaItemInstanceChildAccessor(AgendaItemChildAccessor accessor, AgendaItemBo instance) {
1492                this.accessor = accessor;
1493                this.instance = instance;
1494            }
1495            
1496            public void setChild(AgendaItemBo child) {
1497                accessor.setChild(instance, child);
1498            }
1499            
1500            public AgendaItemBo getChild() {
1501                return accessor.getChild(instance);
1502            }
1503            
1504            public AgendaItemBo getInstance() { return instance; }
1505        }
1506        
1507        /**
1508         * <p>This class abstracts getting and setting a child of an AgendaItemBo, making some recursive operations
1509         * require less boiler plate.</p>
1510         *
1511         * <p>The word 'child' in AgendaItemChildAccessor means child in the strict data structures sense, in that the
1512         * instance passed in holds a reference to some other node (or null).  However, when discussing the agenda tree
1513         * and algorithms for manipulating it, the meaning of 'child' is somewhat different, and there are notions of
1514         * 'sibling' and 'cousin' that are tossed about too. It's probably worth explaining that somewhat here:</p>
1515         *
1516         * <p>General principals of relationships when talking about the agenda tree:
1517         * <ul>
1518         * <li>Generation boundaries (parent to child) are across 'When TRUE' and 'When FALSE' references.</li>
1519         * <li>"Age" among siblings & cousins goes from top (oldest) to bottom (youngest).</li>
1520         * <li>siblings are related by 'Always' references.</li>
1521         * </ul>
1522         * </p>
1523         * <p>This diagram of an agenda tree and the following examples seek to illustrate these principals:</p>
1524         * <img src="doc-files/AgendaEditorController-1.png" alt="Example Agenda Items"/>
1525         * <p>Examples:
1526         * <ul>
1527         * <li>A is the parent of B, C, & D</li>
1528         * <li>E is the younger sibling of A</li>
1529         * <li>B is the older cousin of C</li>
1530         * <li>C is the older sibling of D</li>
1531         * <li>F is the younger cousin of D</li>
1532         * </ul>
1533         * </p>
1534         */
1535        protected static class AgendaItemChildAccessor {
1536            
1537            private enum Child { WHEN_TRUE, WHEN_FALSE, ALWAYS };
1538            
1539            private static final AgendaItemChildAccessor whenTrue = new AgendaItemChildAccessor(Child.WHEN_TRUE); 
1540            private static final AgendaItemChildAccessor whenFalse = new AgendaItemChildAccessor(Child.WHEN_FALSE); 
1541            private static final AgendaItemChildAccessor always = new AgendaItemChildAccessor(Child.ALWAYS); 
1542    
1543            /**
1544             * Accessors for all linked items
1545             */
1546            private static final AgendaItemChildAccessor [] linkedNodes = { whenTrue, whenFalse, always };
1547            
1548            /**
1549             * Accessors for children (so ALWAYS is omitted);
1550             */
1551            private static final AgendaItemChildAccessor [] children = { whenTrue, whenFalse };
1552            
1553            private final Child whichChild;
1554            
1555            private AgendaItemChildAccessor(Child whichChild) {
1556                if (whichChild == null) throw new IllegalArgumentException("whichChild must be non-null");
1557                this.whichChild = whichChild;
1558            }
1559            
1560            /**
1561             * @return the referenced child
1562             */
1563            public AgendaItemBo getChild(AgendaItemBo parent) {
1564                switch (whichChild) {
1565                case WHEN_TRUE: return parent.getWhenTrue();
1566                case WHEN_FALSE: return parent.getWhenFalse();
1567                case ALWAYS: return parent.getAlways();
1568                default: throw new IllegalStateException();
1569                }
1570            }
1571            
1572            /**
1573             * Sets the child reference and the child id 
1574             */
1575            public void setChild(AgendaItemBo parent, AgendaItemBo child) {
1576                switch (whichChild) {
1577                case WHEN_TRUE: 
1578                    parent.setWhenTrue(child);
1579                    parent.setWhenTrueId(child == null ? null : child.getId());
1580                    break;
1581                case WHEN_FALSE:
1582                    parent.setWhenFalse(child);
1583                    parent.setWhenFalseId(child == null ? null : child.getId());
1584                    break;
1585                case ALWAYS:
1586                    parent.setAlways(child);
1587                    parent.setAlwaysId(child == null ? null : child.getId());
1588                    break;
1589                default: throw new IllegalStateException();
1590                }
1591            }
1592        }
1593        //
1594        // Rule Editor Controller methods
1595        //
1596        @RequestMapping(params = "methodToCall=" + "copyRule")
1597        public ModelAndView copyRule(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1598                HttpServletRequest request, HttpServletResponse response) throws Exception {
1599    
1600            AgendaEditor agendaEditor = getAgendaEditor(form);
1601            String name = agendaEditor.getCopyRuleName();
1602            String namespace = agendaEditor.getNamespace();
1603            // fetch existing rule and copy fields to new rule
1604            RuleDefinition oldRuleDefinition = getRuleBoService().getRuleByNameAndNamespace(name, namespace);
1605            RuleBo oldRule = RuleBo.from(oldRuleDefinition);
1606            RuleBo newRule = RuleBo.copyRule(oldRule);
1607            agendaEditor.getAgendaItemLine().setRule( newRule );
1608            // hack to set ui action object to first action in the list
1609            if (!newRule.getActions().isEmpty()) {
1610                agendaEditor.setAgendaItemLineRuleAction( newRule.getActions().get(0));
1611            }
1612            return super.refresh(form, result, request, response);
1613        }
1614    
1615    
1616        /**
1617         * This method starts an edit proposition.
1618         */
1619        @RequestMapping(params = "methodToCall=" + "goToEditProposition")
1620        public ModelAndView goToEditProposition(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1621                HttpServletRequest request, HttpServletResponse response) throws Exception {
1622    
1623            // open the selected node for editing
1624            AgendaEditor agendaEditor = getAgendaEditor(form);
1625            RuleBo rule = agendaEditor.getAgendaItemLine().getRule();
1626            String selectedPropId = agendaEditor.getSelectedPropositionId();
1627    
1628            Node<RuleTreeNode,String> root = rule.getPropositionTree().getRootElement();
1629            PropositionBo propositionToToggleEdit = null;
1630            boolean newEditMode = true;
1631    
1632            // find parent
1633            Node<RuleTreeNode,String> parent = findParentPropositionNode( root, selectedPropId);
1634            if (parent != null){
1635                List<Node<RuleTreeNode,String>> children = parent.getChildren();
1636                for( int index=0; index< children.size(); index++){
1637                    Node<RuleTreeNode,String> child = children.get(index);
1638                    if (propIdMatches(child, selectedPropId)){
1639                        PropositionBo prop = child.getData().getProposition();
1640                        propositionToToggleEdit = prop;
1641                        newEditMode =  !prop.getEditMode();
1642                        break;
1643                    } else {
1644                        child.getData().getProposition().setEditMode(false);
1645                    }
1646                }
1647            }
1648    
1649            resetEditModeOnPropositionTree(root);
1650            if (propositionToToggleEdit != null) {
1651                propositionToToggleEdit.setEditMode(newEditMode);
1652                //refresh the tree
1653                rule.refreshPropositionTree(null);
1654            }
1655    
1656            return getUIFModelAndView(form);
1657        }
1658    
1659        @RequestMapping(params = "methodToCall=" + "addProposition")
1660        public ModelAndView addProposition(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1661                HttpServletRequest request, HttpServletResponse response) throws Exception {
1662    
1663            AgendaEditor agendaEditor = getAgendaEditor(form);
1664            RuleBo rule = agendaEditor.getAgendaItemLine().getRule();
1665            String selectedPropId = agendaEditor.getSelectedPropositionId();
1666    
1667    
1668            // find parent
1669            Node<RuleTreeNode,String> root = agendaEditor.getAgendaItemLine().getRule().getPropositionTree().getRootElement();
1670            Node<RuleTreeNode,String> parent = findParentPropositionNode( root, selectedPropId);
1671    
1672            resetEditModeOnPropositionTree(root);
1673    
1674            // add new child at appropriate spot
1675            if (parent != null){
1676                List<Node<RuleTreeNode,String>> children = parent.getChildren();
1677                for( int index=0; index< children.size(); index++){
1678                    Node<RuleTreeNode,String> child = children.get(index);
1679    
1680                    // if our selected node is a simple proposition, add a new one after
1681                    if (propIdMatches(child, selectedPropId)){
1682                        // handle special case of adding to a lone simple proposition.
1683                        // in this case, we need to change the root level proposition to a compound proposition
1684                        // move the existing simple proposition as the first compound component,
1685                        // then add a new blank simple prop as the second compound component.
1686                        if (parent == root &&
1687                            (SimplePropositionNode.NODE_TYPE.equalsIgnoreCase(child.getNodeType()) ||
1688                            SimplePropositionEditNode.NODE_TYPE.equalsIgnoreCase(child.getNodeType()))){
1689    
1690                            // create a new compound proposition
1691                            PropositionBo compound = PropositionBo.createCompoundPropositionBoStub(child.getData().getProposition(), true);
1692                            // don't set compound.setEditMode(true) as the Simple Prop in the compound prop is the only prop in edit mode
1693                            rule.setProposition(compound);
1694                            rule.refreshPropositionTree(null);
1695                        }
1696                        // handle regular case of adding a simple prop to an existing compound prop
1697                        else if(SimplePropositionNode.NODE_TYPE.equalsIgnoreCase(child.getNodeType()) ||
1698                           SimplePropositionEditNode.NODE_TYPE.equalsIgnoreCase(child.getNodeType())){
1699    
1700                            // build new Blank Proposition
1701                            PropositionBo blank = PropositionBo.createSimplePropositionBoStub(child.getData().getProposition(),PropositionType.SIMPLE.getCode());
1702                            //add it to the parent
1703                            PropositionBo parentProp = parent.getData().getProposition();
1704                            parentProp.getCompoundComponents().add(((index/2)+1), blank);
1705    
1706                            rule.refreshPropositionTree(true);
1707                        }
1708    
1709                        break;
1710                    }
1711                }
1712            } else {
1713                // special case, if root has no children, add a new simple proposition
1714                // todo: how to add compound proposition. - just add another to the firs simple
1715                if (root.getChildren().isEmpty()){
1716                    PropositionBo blank = PropositionBo.createSimplePropositionBoStub(null,PropositionType.SIMPLE.getCode());
1717                    blank.setRuleId(rule.getId());
1718                    rule.setPropId(blank.getId());
1719                    rule.setProposition(blank);
1720                    rule.refreshPropositionTree(true);
1721                }
1722            }
1723            return getUIFModelAndView(form);
1724        }
1725    
1726        /**
1727         *
1728         * This method adds an opCode Node to separate components in a compound proposition.
1729         *
1730         * @param currentNode
1731         * @param prop
1732         * @return
1733         */
1734        private void addOpCodeNode(Node currentNode, PropositionBo prop, int index){
1735            String opCodeLabel = "";
1736    
1737            if (LogicalOperator.AND.getCode().equalsIgnoreCase(prop.getCompoundOpCode())){
1738                opCodeLabel = "AND";
1739            } else if (LogicalOperator.OR.getCode().equalsIgnoreCase(prop.getCompoundOpCode())){
1740                opCodeLabel = "OR";
1741            }
1742            Node<RuleTreeNode, String> aNode = new Node<RuleTreeNode, String>();
1743            aNode.setNodeLabel("");
1744            aNode.setNodeType("ruleTreeNode compoundOpCodeNode");
1745            aNode.setData(new CompoundOpCodeNode(prop));
1746            currentNode.insertChildAt(index, aNode);
1747        }
1748    
1749    
1750        private boolean propIdMatches(Node<RuleTreeNode, String> node, String propId){
1751            if (propId!=null && node != null && node.getData() != null && propId.equalsIgnoreCase(node.getData().getProposition().getId())) {
1752                return true;
1753            }
1754            return false;
1755        }
1756    
1757        /**
1758         * disable edit mode for all Nodes beneath and including the passed in Node
1759         * @param currentNode
1760         */
1761        private void resetEditModeOnPropositionTree(Node<RuleTreeNode, String> currentNode){
1762            if (currentNode.getData() != null){
1763                RuleTreeNode dataNode = currentNode.getData();
1764                dataNode.getProposition().setEditMode(false);
1765            }
1766            List<Node<RuleTreeNode,String>> children = currentNode.getChildren();
1767            for( Node<RuleTreeNode,String> child : children){
1768                  resetEditModeOnPropositionTree(child);
1769            }
1770        }
1771    
1772        private Node<RuleTreeNode, String> findPropositionTreeNode(Node<RuleTreeNode, String> currentNode, String selectedPropId){
1773            Node<RuleTreeNode,String> bingo = null;
1774            if (currentNode.getData() != null){
1775                RuleTreeNode dataNode = currentNode.getData();
1776                if (selectedPropId.equalsIgnoreCase(dataNode.getProposition().getId())){
1777                    return currentNode;
1778                }
1779            }
1780            List<Node<RuleTreeNode,String>> children = currentNode.getChildren();
1781            for( Node<RuleTreeNode,String> child : children){
1782                  bingo = findPropositionTreeNode(child, selectedPropId);
1783                  if (bingo != null) break;
1784            }
1785            return bingo;
1786        }
1787    
1788        private Node<RuleTreeNode, String> findParentPropositionNode(Node<RuleTreeNode, String> currentNode, String selectedPropId){
1789            Node<RuleTreeNode,String> bingo = null;
1790            if (selectedPropId != null) {
1791                // if it's in children, we have the parent
1792                List<Node<RuleTreeNode,String>> children = currentNode.getChildren();
1793                for( Node<RuleTreeNode,String> child : children){
1794                    RuleTreeNode dataNode = child.getData();
1795                    if (selectedPropId.equalsIgnoreCase(dataNode.getProposition().getId()))
1796                        return currentNode;
1797                }
1798    
1799                // if not found check grandchildren
1800                for( Node<RuleTreeNode,String> kid : children){
1801                      bingo = findParentPropositionNode(kid, selectedPropId);
1802                      if (bingo != null) break;
1803                }
1804            }
1805            return bingo;
1806        }
1807    
1808        /**
1809         * This method return the index of the position of the child that matches the id
1810         * @param parent
1811         * @param propId
1812         * @return index if found, -1 if not found
1813         */
1814        private int findChildIndex(Node<RuleTreeNode,String> parent, String propId){
1815            int index;
1816            List<Node<RuleTreeNode,String>> children = parent.getChildren();
1817            for(index=0; index< children.size(); index++){
1818                Node<RuleTreeNode,String> child = children.get(index);
1819                // if our selected node is a simple proposition, add a new one after
1820                if (propIdMatches(child, propId)){
1821                    return index;
1822                }
1823            }
1824            return -1;
1825        }
1826    
1827        @RequestMapping(params = "methodToCall=" + "movePropositionUp")
1828        public ModelAndView movePropositionUp(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1829                HttpServletRequest request, HttpServletResponse response)
1830                throws Exception {
1831            moveSelectedProposition(form, true);
1832    
1833            return getUIFModelAndView(form);
1834        }
1835    
1836        @RequestMapping(params = "methodToCall=" + "movePropositionDown")
1837        public ModelAndView movePropositionDown(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1838                HttpServletRequest request, HttpServletResponse response)
1839                throws Exception {
1840            moveSelectedProposition(form, false);
1841    
1842            return getUIFModelAndView(form);
1843        }
1844    
1845        private void moveSelectedProposition(UifFormBase form, boolean up) {
1846    
1847            /* Rough algorithm for moving a node up.
1848             *
1849             * find the following:
1850             *   node := the selected node
1851             *   parent := the selected node's parent, its containing node (via when true or when false relationship)
1852             *   parentsOlderCousin := the parent's level-order predecessor (sibling or cousin)
1853             *
1854             */
1855            AgendaEditor agendaEditor = getAgendaEditor(form);
1856            RuleBo rule = agendaEditor.getAgendaItemLine().getRule();
1857            String selectedPropId = agendaEditor.getSelectedPropositionId();
1858    
1859            // find parent
1860            Node<RuleTreeNode,String> parent = findParentPropositionNode(rule.getPropositionTree().getRootElement(), selectedPropId);
1861    
1862            // add new child at appropriate spot
1863            if (parent != null){
1864                List<Node<RuleTreeNode,String>> children = parent.getChildren();
1865                for( int index=0; index< children.size(); index++){
1866                    Node<RuleTreeNode,String> child = children.get(index);
1867                    // if our selected node is a simple proposition, add a new one after
1868                    if (propIdMatches(child, selectedPropId)){
1869                        if(SimplePropositionNode.NODE_TYPE.equalsIgnoreCase(child.getNodeType()) ||
1870                           SimplePropositionEditNode.NODE_TYPE.equalsIgnoreCase(child.getNodeType()) ||
1871                           RuleTreeNode.COMPOUND_NODE_TYPE.equalsIgnoreCase(child.getNodeType()) ){
1872    
1873                            if (((index > 0) && up) || ((index <(children.size() - 1)&& !up))){
1874                                //remove it from its current spot
1875                                PropositionBo parentProp = parent.getData().getProposition();
1876                                PropositionBo workingProp = parentProp.getCompoundComponents().remove(index/2);
1877                                if (up){
1878                                    parentProp.getCompoundComponents().add((index/2)-1, workingProp);
1879                                }else{
1880                                    parentProp.getCompoundComponents().add((index/2)+1, workingProp);
1881                                }
1882    
1883                                // insert it in the new spot
1884                                // redisplay the tree (editMode = true)
1885                                boolean editMode = (SimplePropositionEditNode.NODE_TYPE.equalsIgnoreCase(child.getNodeType()));
1886                                rule.refreshPropositionTree(editMode);
1887                            }
1888                        }
1889    
1890                        break;
1891                    }
1892                }
1893            }
1894        }
1895        @RequestMapping(params = "methodToCall=" + "movePropositionLeft")
1896        public ModelAndView movePropositionLeft(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1897                HttpServletRequest request, HttpServletResponse response)
1898                throws Exception {
1899    
1900            /* Rough algorithm for moving a node up.
1901             *
1902             * find the following:
1903             *   node := the selected node
1904             *   parent := the selected node's parent, its containing node (via when true or when false relationship)
1905             *   parentsOlderCousin := the parent's level-order predecessor (sibling or cousin)
1906             *
1907             */
1908            AgendaEditor agendaEditor = getAgendaEditor(form);
1909            RuleBo rule = agendaEditor.getAgendaItemLine().getRule();
1910            String selectedPropId = agendaEditor.getSelectedPropositionId();
1911    
1912            // find agendaEditor.getAgendaItemLine().getRule().getPropositionTree().getRootElement()parent
1913            Node<RuleTreeNode,String> root = rule.getPropositionTree().getRootElement();
1914            Node<RuleTreeNode,String> parent = findParentPropositionNode(root, selectedPropId);
1915            if ((parent != null) && (RuleTreeNode.COMPOUND_NODE_TYPE.equalsIgnoreCase(parent.getNodeType()))){
1916                Node<RuleTreeNode,String> granny = findParentPropositionNode(root,parent.getData().getProposition().getId());
1917                if (granny != root){
1918                    int oldIndex = findChildIndex(parent, selectedPropId);
1919                    int newIndex = findChildIndex(granny, parent.getData().getProposition().getId());
1920                    if (oldIndex >= 0 && newIndex >= 0){
1921                        PropositionBo prop = parent.getData().getProposition().getCompoundComponents().remove(oldIndex/2);
1922                        granny.getData().getProposition().getCompoundComponents().add((newIndex/2)+1, prop);
1923                        rule.refreshPropositionTree(false);
1924                    }
1925                } else {
1926                    // TODO: do we allow moving up to the root?
1927                    // we could add a new top level compound node, with current root as 1st child,
1928                    // and move the node to the second child.
1929                }
1930            }
1931            return getUIFModelAndView(form);
1932        }
1933    
1934        @RequestMapping(params = "methodToCall=" + "movePropositionRight")
1935        public ModelAndView movePropositionRight(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1936                HttpServletRequest request, HttpServletResponse response)
1937                throws Exception {
1938            /* Rough algorithm for moving a node Right
1939             * if the selected node is above a compound proposition, move it into the compound proposition as the first child
1940             * if the node is above a simple proposition, do nothing.
1941             * find the following:
1942             *   node := the selected node
1943             *   parent := the selected node's parent, its containing node
1944             *   nextSibling := the node after the selected node
1945             *
1946             */
1947            AgendaEditor agendaEditor = getAgendaEditor(form);
1948            RuleBo rule = agendaEditor.getAgendaItemLine().getRule();
1949            String selectedPropId = agendaEditor.getSelectedPropositionId();
1950    
1951            // find parent
1952            Node<RuleTreeNode,String> parent = findParentPropositionNode(
1953                    rule.getPropositionTree().getRootElement(), selectedPropId);
1954            if (parent != null){
1955                int index = findChildIndex(parent, selectedPropId);
1956                // if we are the last child, do nothing, otherwise
1957                if (index >= 0 && index+1 < parent.getChildren().size()){
1958                    Node<RuleTreeNode,String> child = parent.getChildren().get(index);
1959                    Node<RuleTreeNode,String> nextSibling = parent.getChildren().get(index+2);
1960                    // if selected node above a compound node, move it into it as first child
1961                    if(RuleTreeNode.COMPOUND_NODE_TYPE.equalsIgnoreCase(nextSibling.getNodeType()) ){
1962                        // remove selected node from it's current spot
1963                        PropositionBo prop = parent.getData().getProposition().getCompoundComponents().remove(index/2);
1964                        // add it to it's siblings children
1965                        nextSibling.getData().getProposition().getCompoundComponents().add(0, prop);
1966                        rule.refreshPropositionTree(false);
1967                    }
1968                }
1969            }
1970            return getUIFModelAndView(form);
1971        }
1972    
1973        /**
1974         * introduces a new compound proposition between the selected proposition and its parent.
1975         * Additionally, it puts a new blank simple proposition underneath the compound proposition
1976         * as a sibling to the selected proposition.
1977         */
1978        @RequestMapping(params = "methodToCall=" + "togglePropositionSimpleCompound")
1979        public ModelAndView togglePropositionSimpleCompound(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
1980                HttpServletRequest request, HttpServletResponse response)
1981                throws Exception {
1982    
1983            AgendaEditor agendaEditor = getAgendaEditor(form);
1984            RuleBo rule = agendaEditor.getAgendaItemLine().getRule();
1985            String selectedPropId = agendaEditor.getSelectedPropositionId();
1986    
1987            resetEditModeOnPropositionTree(rule.getPropositionTree().getRootElement());
1988    
1989            if (!StringUtils.isBlank(selectedPropId)) {
1990                // find parent
1991                Node<RuleTreeNode,String> parent = findParentPropositionNode(
1992                        rule.getPropositionTree().getRootElement(), selectedPropId);
1993                if (parent != null){
1994    
1995                    int index = findChildIndex(parent, selectedPropId);
1996    
1997                    PropositionBo propBo = parent.getChildren().get(index).getData().getProposition();
1998    
1999                    // create a new compound proposition
2000                    PropositionBo compound = PropositionBo.createCompoundPropositionBoStub(propBo, true);
2001                    compound.setDescription("New Compound Proposition " + UUID.randomUUID().toString());
2002                    compound.setEditMode(false);
2003    
2004                    if (parent.getData() == null) { // SPECIAL CASE: this is the only proposition in the tree
2005                        rule.setProposition(compound);
2006                    } else {
2007                        PropositionBo parentBo = parent.getData().getProposition();
2008                        List<PropositionBo> siblings = parentBo.getCompoundComponents();
2009    
2010                        int propIndex = -1;
2011                        for (int i=0; i<siblings.size(); i++) {
2012                            if (propBo.getId().equals(siblings.get(i).getId())) {
2013                                propIndex = i;
2014                                break;
2015                            }
2016                        }
2017    
2018                        parentBo.getCompoundComponents().set(propIndex, compound);
2019                    }
2020                }
2021            }
2022    
2023            agendaEditor.getAgendaItemLine().getRule().refreshPropositionTree(true);
2024            return getUIFModelAndView(form);
2025        }
2026    
2027    
2028        @RequestMapping(params = "methodToCall=" + "cutProposition")
2029        public ModelAndView cutProposition(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
2030                HttpServletRequest request, HttpServletResponse response)
2031                throws Exception {
2032    
2033            AgendaEditor agendaEditor = getAgendaEditor(form);
2034            String selectedPropId = agendaEditor.getSelectedPropositionId();
2035            agendaEditor.setCutPropositionId(selectedPropId);
2036    
2037            return getUIFModelAndView(form);
2038        }
2039    
2040        @RequestMapping(params = "methodToCall=" + "pasteProposition")
2041        public ModelAndView pasteProposition(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
2042                HttpServletRequest request, HttpServletResponse response)
2043                throws Exception {
2044    
2045            AgendaEditor agendaEditor = getAgendaEditor(form);
2046            RuleBo rule = agendaEditor.getAgendaItemLine().getRule();
2047    
2048            // get selected id
2049            String cutPropId = agendaEditor.getCutPropositionId();
2050            String selectedPropId = agendaEditor.getSelectedPropositionId();
2051    
2052            if (StringUtils.isNotBlank(selectedPropId) && selectedPropId.equals(cutPropId)) {
2053                    // do nothing; can't paste to itself
2054            } else {
2055    
2056                // proposition tree root
2057                Node<RuleTreeNode, String> root = rule.getPropositionTree().getRootElement();
2058    
2059                if (StringUtils.isNotBlank(selectedPropId) && StringUtils.isNotBlank(cutPropId)) {
2060                    Node<RuleTreeNode,String> parentNode = findParentPropositionNode(root, selectedPropId);
2061                    PropositionBo newParent;
2062                    if (parentNode == root){
2063                        // special case
2064                        // build new top level compound proposition,
2065                        // add existing as first child
2066                        // then paste cut node as 2nd child
2067                        newParent = PropositionBo.createCompoundPropositionBoStub2(
2068                                root.getChildren().get(0).getData().getProposition());
2069                        newParent.setEditMode(true);
2070                        rule.setProposition(newParent);
2071                    } else {
2072                        newParent = parentNode.getData().getProposition();
2073                    }
2074                    PropositionBo oldParent = findParentPropositionNode(root, cutPropId).getData().getProposition();
2075    
2076                    PropositionBo workingProp = null;
2077                    // cut from old
2078                    if (oldParent != null){
2079                        List <PropositionBo> children = oldParent.getCompoundComponents();
2080                        for( int index=0; index< children.size(); index++){
2081                            if (cutPropId.equalsIgnoreCase(children.get(index).getId())){
2082                                workingProp = oldParent.getCompoundComponents().remove(index);
2083                                break;
2084                            }
2085                        }
2086                    }
2087    
2088                    // add to new
2089                    if (newParent != null && workingProp != null){
2090                        List <PropositionBo> children = newParent.getCompoundComponents();
2091                        for( int index=0; index< children.size(); index++){
2092                            if (selectedPropId.equalsIgnoreCase(children.get(index).getId())){
2093                                children.add(index+1, workingProp);
2094                                break;
2095                            }
2096                        }
2097                    }
2098                    // TODO: determine edit mode.
2099    //                boolean editMode = (SimplePropositionEditNode.NODE_TYPE.equalsIgnoreCase(child.getNodeType()));
2100                    rule.refreshPropositionTree(false);
2101                }
2102            }
2103            agendaEditor.setCutPropositionId(null);
2104            // call the super method to avoid the agenda tree being reloaded from the db
2105            return getUIFModelAndView(form);
2106        }
2107    
2108        @RequestMapping(params = "methodToCall=" + "deleteProposition")
2109        public ModelAndView deleteProposition(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
2110                HttpServletRequest request, HttpServletResponse response)
2111                throws Exception {
2112            AgendaEditor agendaEditor = getAgendaEditor(form);
2113            String selectedPropId = agendaEditor.getSelectedPropositionId();
2114            Node<RuleTreeNode, String> root = agendaEditor.getAgendaItemLine().getRule().getPropositionTree().getRootElement();
2115    
2116            Node<RuleTreeNode, String> parentNode = findParentPropositionNode(root, selectedPropId);
2117    
2118            // what if it is the root?
2119            if (parentNode != null && parentNode.getData() != null) { // it is not the root as there is a parent w/ a prop
2120                PropositionBo parent = parentNode.getData().getProposition();
2121                if (parent != null){
2122                    List <PropositionBo> children = parent.getCompoundComponents();
2123                    for( int index=0; index< children.size(); index++){
2124                        if (selectedPropId.equalsIgnoreCase(children.get(index).getId())){
2125                            parent.getCompoundComponents().remove(index);
2126                            break;
2127                        }
2128                    }
2129                }
2130            } else { // no parent, it is the root
2131                parentNode.getChildren().clear();
2132                agendaEditor.getAgendaItemLine().getRule().getPropositionTree().setRootElement(null);
2133                agendaEditor.getAgendaItemLine().getRule().setPropId(null);
2134                agendaEditor.getAgendaItemLine().getRule().setProposition(null);
2135            }
2136    
2137            agendaEditor.getAgendaItemLine().getRule().refreshPropositionTree(false);
2138            return getUIFModelAndView(form);
2139        }
2140    
2141        @RequestMapping(params = "methodToCall=" + "updateCompoundOperator")
2142        public ModelAndView updateCompoundOperator(@ModelAttribute("KualiForm") UifFormBase form, BindingResult result,
2143                HttpServletRequest request, HttpServletResponse response)
2144                throws Exception {
2145    
2146            AgendaEditor agendaEditor = getAgendaEditor(form);
2147            RuleBo rule = agendaEditor.getAgendaItemLine().getRule();
2148            rule.refreshPropositionTree(false);
2149    
2150            return getUIFModelAndView(form);
2151        }
2152    
2153    }