/********************************************************************************** * $URL: https://source.sakaiproject.org/svn/sam/trunk/samigo-app/src/java/org/sakaiproject/tool/assessment/ui/listener/evaluation/StudentScoreUpdateListener.java $ * $Id: StudentScoreUpdateListener.java 121258 2013-03-15 15:03:36Z ottenhoff@longsight.com $ *********************************************************************************** * * Copyright (c) 2004, 2005, 2006, 2007, 2008, 2009 The Sakai Foundation * * Licensed under the Educational Community License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.opensource.org/licenses/ECL-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * **********************************************************************************/ package org.sakaiproject.tool.assessment.ui.listener.evaluation; import java.util.ArrayList; import java.util.Date; import java.util.HashMap; import java.util.HashSet; import java.util.Iterator; import java.util.List; import java.util.Set; import javax.faces.application.FacesMessage; import javax.faces.context.FacesContext; import javax.faces.event.AbortProcessingException; import javax.faces.event.ActionEvent; import javax.faces.event.ActionListener; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.apache.commons.math.util.MathUtils; import org.sakaiproject.event.cover.EventTrackingService; import org.sakaiproject.tool.assessment.data.dao.grading.AssessmentGradingData; import org.sakaiproject.tool.assessment.data.dao.grading.ItemGradingAttachment; import org.sakaiproject.tool.assessment.data.dao.grading.ItemGradingData; import org.sakaiproject.tool.assessment.facade.AgentFacade; import org.sakaiproject.tool.assessment.services.GradebookServiceException; import org.sakaiproject.tool.assessment.services.GradingService; import org.sakaiproject.tool.assessment.ui.bean.delivery.DeliveryBean; import org.sakaiproject.tool.assessment.ui.bean.delivery.ItemContentsBean; import org.sakaiproject.tool.assessment.ui.bean.delivery.SectionContentsBean; import org.sakaiproject.tool.assessment.ui.bean.evaluation.StudentScoresBean; import org.sakaiproject.tool.assessment.ui.bean.evaluation.TotalScoresBean; import org.sakaiproject.tool.assessment.ui.listener.util.ContextUtil; import org.sakaiproject.tool.assessment.util.TextFormat; /** * <p> * This handles the updating of the Student Score page. * </p> * <p>Description: Action Listener Evaluation Updating Student Score page</p> * <p>Copyright: Copyright (c) 2004</p> * <p>Organization: Sakai Project</p> * @author Rachel Gollub * @version $Id: StudentScoreUpdateListener.java 121258 2013-03-15 15:03:36Z ottenhoff@longsight.com $ */ public class StudentScoreUpdateListener implements ActionListener { private static Log log = LogFactory.getLog(StudentScoreUpdateListener.class); private static ContextUtil cu; /** * Standard process action method. * @param ae ActionEvent * @throws AbortProcessingException */ public void processAction(ActionEvent ae) throws AbortProcessingException { log.debug("Student Score Update LISTENER."); StudentScoresBean bean = (StudentScoresBean) cu.lookupBean("studentScores"); TotalScoresBean tbean = (TotalScoresBean) cu.lookupBean("totalScores"); tbean.setAssessmentGradingHash(tbean.getPublishedAssessment().getPublishedAssessmentId()); DeliveryBean delivery = (DeliveryBean) cu.lookupBean("delivery"); log.debug("Calling saveStudentScores."); try { if (!saveStudentScores(bean, tbean, delivery)) { throw new RuntimeException("failed to call saveStudentScores."); } } catch (GradebookServiceException ge) { FacesContext context = FacesContext.getCurrentInstance(); String err=(String)cu.getLocalizedString("org.sakaiproject.tool.assessment.bundle.AuthorMessages", "gradebook_exception_error"); context.addMessage(null, new FacesMessage(err)); } } /** * Persist the results from the ActionForm in the student page. * @param bean StudentScoresBean bean * @return true if successful */ public boolean saveStudentScores(StudentScoresBean bean, TotalScoresBean tbean, DeliveryBean delivery) { GradingService delegate = new GradingService(); HashSet itemGradingSet = new HashSet(); AssessmentGradingData adata = null; try { ArrayList parts = delivery.getPageContents().getPartsContents(); Iterator iter = parts.iterator(); boolean updateFlag = false; while (iter.hasNext()) { ArrayList items = ((SectionContentsBean) iter.next()).getItemContents(); Iterator iter2 = items.iterator(); while (iter2.hasNext()) { ItemContentsBean question = (ItemContentsBean) iter2.next(); ArrayList gradingarray = question.getItemGradingDataArray(); log.debug("****1. pub questionId = " + question.getItemData().getItemId()); log.debug("****2. Gradingarray length = " + gradingarray.size()); // Create a new one if we need it. if (gradingarray.isEmpty() && (question.getExactPoints() > 0 || (question.getGradingComment() != null && !question.getGradingComment().trim().equals("")) )) { // this is another mystery, no idea why review is involved here - daiyf question.setReview(false); // This creates an itemgradingdata gradingarray = question.getItemGradingDataArray(); } int fibFinNumCorrect = 0; if (question.getItemData().getTypeId().equals(Long.valueOf(8)) || question.getItemData().getTypeId().equals(Long.valueOf(11))) { Iterator itemGradingIter = gradingarray.iterator(); while (itemGradingIter.hasNext()){ Object obj = itemGradingIter.next(); ItemGradingData data = (ItemGradingData) obj; if (Boolean.TRUE.equals(data.getIsCorrect())) { fibFinNumCorrect++; } } } log.debug("****3a Gradingarray length2 = " + gradingarray.size()); log.debug("****3b set points = " + question.getExactPoints() + ", comments to " + question.getGradingComment()); Iterator iter3 = gradingarray.iterator(); while (iter3.hasNext()) { ItemGradingData data = (ItemGradingData) iter3.next(); if (adata == null && data.getAssessmentGradingId() != null){ adata = delegate.load(data.getAssessmentGradingId().toString()); } if (data.getAgentId() == null) { // this is a skipped question, set submittedDate=null data.setSubmittedDate(null); data.setAgentId(bean.getStudentId()); } double newAutoScore = 0; if ((question.getItemData().getTypeId().equals(Long.valueOf(8)) || question.getItemData().getTypeId().equals(Long.valueOf(11))) && fibFinNumCorrect != 0) { if (Boolean.TRUE.equals(data.getIsCorrect())) { newAutoScore = (question.getExactPoints() / (double) fibFinNumCorrect); } } else { newAutoScore = (question.getExactPoints() / (double) gradingarray.size()); } double oldAutoScore = 0; if (data.getAutoScore() !=null) { oldAutoScore=data.getAutoScore().doubleValue(); } String newComments = TextFormat.convertPlaintextToFormattedTextNoHighUnicode(log, question.getGradingComment()); if (newComments != null) { newComments = newComments.trim(); } else { newComments = ""; } String oldComments = data.getComments(); if (oldComments != null) { oldComments = oldComments.trim(); } else { oldComments = ""; } // if newAutoScore != oldAutoScore then updateScore = true boolean updateScore = !(MathUtils.equalsIncludingNaN(newAutoScore, oldAutoScore, 0.0001)); boolean updateComments = !newComments.equals(oldComments); StringBuffer logString = new StringBuffer(); logString.append("gradedBy="); logString.append(AgentFacade.getAgentString()); logString.append(", itemGradingId="); logString.append(data.getItemGradingId()); if (updateScore) { data.setAutoScore(Double.valueOf(newAutoScore)); logString.append(", newAutoScore="); logString.append(newAutoScore); logString.append(", oldAutoScore="); logString.append(oldAutoScore); } if (updateComments) { data.setComments(newComments); logString.append(", newComments="); logString.append(newComments); logString.append(", oldComments="); logString.append(oldComments); } if (updateScore || updateComments) { updateFlag = true; data.setGradedBy(AgentFacade.getAgentString()); data.setGradedDate(new Date()); String targetString = "siteId=" + AgentFacade.getCurrentSiteId() + ", " + logString.toString(); String safeString = targetString.length() > 255 ? targetString.substring(0, 255) : targetString; EventTrackingService.post(EventTrackingService.newEvent("sam.student.score.update", safeString, true)); log.debug("****4 itemGradingId="+data.getItemGradingId()); log.debug("****5 set points = " + data.getAutoScore() + ", comments to " + data.getComments()); } data.setAnswerText(ContextUtil.stringWYSIWYG(data.getAnswerText())); itemGradingSet.add(data); } } if (adata==null){ // this is for cases when studnet submitted an assessment but skipped all teh questions // when we won't be able to get teh assessmentGrading based on itemGrdaing ('cos there is none). String assessmentGradingId = cu.lookupParam("gradingData"); adata = delegate.load(assessmentGradingId); } adata.setItemGradingSet(itemGradingSet); } if (adata == null) return true; // Nothing to save. String newComments = TextFormat.convertPlaintextToFormattedTextNoHighUnicode(log, bean.getComments()); if (newComments != null) { newComments = newComments.trim(); } else { newComments = ""; } String oldComments = adata.getComments(); if (oldComments != null) { oldComments = oldComments.trim(); } else { oldComments = ""; } if (!newComments.equals(oldComments)) { updateFlag = true; adata.setComments(newComments); adata.setGradedBy(AgentFacade.getAgentString()); adata.setGradedDate(new Date()); StringBuffer logString = new StringBuffer(); logString.append("gradedBy="); logString.append(AgentFacade.getAgentString()); logString.append(", assessmentGradingId="); logString.append(adata.getAssessmentGradingId()); logString.append(", newComments="); logString.append(newComments); logString.append(", oldComments="); logString.append(oldComments); EventTrackingService.post(EventTrackingService.newEvent("sam.student.score.update", logString.toString(), true)); } if (updateFlag) { delegate.updateAssessmentGradingScore(adata, tbean.getPublishedAssessment()); } log.debug("Saved student scores."); updateAttachment(delivery); } catch (GradebookServiceException ge) { FacesContext context = FacesContext.getCurrentInstance(); String err=(String)cu.getLocalizedString("org.sakaiproject.tool.assessment.bundle.AuthorMessages", "gradebook_exception_error"); context.addMessage(null, new FacesMessage(err)); } catch (Exception e) { e.printStackTrace(); return false; } return true; } public void updateAttachment(DeliveryBean delivery){ ArrayList parts = delivery.getPageContents().getPartsContents(); Iterator iter = parts.iterator(); List attachmentList = new ArrayList(); while (iter.hasNext()) { ArrayList items = ((SectionContentsBean) iter.next()).getItemContents(); Iterator iter2 = items.iterator(); while (iter2.hasNext()) { ItemContentsBean question = (ItemContentsBean) iter2.next(); ArrayList gradingarray = question.getItemGradingDataArray(); log.debug("Gradingarray length2 = " + gradingarray.size()); Iterator iter3 = gradingarray.iterator(); while (iter3.hasNext()) { ItemGradingData itemGradingData = (ItemGradingData) iter3.next(); List oldList = itemGradingData.getItemGradingAttachmentList(); List newList = question.getItemGradingAttachmentList(); if ((oldList == null || oldList.size() == 0 ) && (newList == null || newList.size() == 0)) { continue; } HashMap map = getAttachmentIdHash(oldList); for (int i=0; i<newList.size(); i++){ ItemGradingAttachment itemGradingAttachment = (ItemGradingAttachment) newList.get(i); if (map.get(itemGradingAttachment.getAttachmentId()) != null){ // exist already, remove it from map map.remove(itemGradingAttachment.getAttachmentId()); } else{ // new attachments itemGradingAttachment.setItemGrading(itemGradingData); attachmentList.add(itemGradingAttachment); } } // save new ones GradingService gradingService = new GradingService(); if (attachmentList.size() > 0) { gradingService.saveOrUpdateAttachments(attachmentList); EventTrackingService.post(EventTrackingService.newEvent("sam.student.score.update", "siteId=" + AgentFacade.getCurrentSiteId() + ", Adding " + attachmentList.size() + " attachments for itemGradingData id = " + itemGradingData.getItemGradingId(), true)); } // remove old ones Set set = map.keySet(); Iterator iter4 = set.iterator(); while (iter4.hasNext()){ Long attachmentId = (Long)iter4.next(); gradingService.removeItemGradingAttachment(attachmentId.toString()); EventTrackingService.post(EventTrackingService.newEvent("sam.student.score.update", "siteId=" + AgentFacade.getCurrentSiteId() + ", Removing attachmentId = " + attachmentId, true)); } } } } } private HashMap getAttachmentIdHash(List list){ HashMap map = new HashMap(); for (int i=0; i<list.size(); i++){ ItemGradingAttachment a = (ItemGradingAttachment)list.get(i); map.put(a.getAttachmentId(), a); } return map; } }