Index: lams_build/3rdParty.userlibraries =================================================================== diff -u -r8bc318f44b193b1d926fcc62211ca1b4b18d9b30 -recc49cd6851b43f37ef02c2ddb85257096e2cf49 --- lams_build/3rdParty.userlibraries (.../3rdParty.userlibraries) (revision 8bc318f44b193b1d926fcc62211ca1b4b18d9b30) +++ lams_build/3rdParty.userlibraries (.../3rdParty.userlibraries) (revision ecc49cd6851b43f37ef02c2ddb85257096e2cf49) @@ -1,48 +1,54 @@ - - - - - - + + + + + + - - + + - + - - + - - - - - - - - - - - + + + + + + + + + + + + - + + + + + + + Index: lams_build/build.xml =================================================================== diff -u -rcf258bae4dd72eab7a6f5bc895b96966caafaf06 -recc49cd6851b43f37ef02c2ddb85257096e2cf49 --- lams_build/build.xml (.../build.xml) (revision cf258bae4dd72eab7a6f5bc895b96966caafaf06) +++ lams_build/build.xml (.../build.xml) (revision ecc49cd6851b43f37ef02c2ddb85257096e2cf49) @@ -86,11 +86,11 @@ - + --> @@ -123,6 +123,7 @@ + @@ -286,6 +287,7 @@ + --> @@ -400,6 +402,22 @@ + + + + + + + + + + + + + + @@ -408,6 +426,15 @@ + + + + + + + + @@ -424,14 +451,15 @@ - - - + + + - + + @@ -510,6 +538,16 @@ + + + + + + + + + @@ -573,6 +611,7 @@ + + + + + + + + + + + + + + + + + + + @@ -763,6 +822,7 @@ + @@ -854,7 +914,6 @@ - Index: lams_central/conf/security/Owasp.CsrfGuard.properties =================================================================== diff -u -rcf258bae4dd72eab7a6f5bc895b96966caafaf06 -recc49cd6851b43f37ef02c2ddb85257096e2cf49 --- lams_central/conf/security/Owasp.CsrfGuard.properties (.../Owasp.CsrfGuard.properties) (revision cf258bae4dd72eab7a6f5bc895b96966caafaf06) +++ lams_central/conf/security/Owasp.CsrfGuard.properties (.../Owasp.CsrfGuard.properties) (revision ecc49cd6851b43f37ef02c2ddb85257096e2cf49) @@ -1,8 +1,12 @@ # Only check POST forms. If we need to, we can add GET and other HTTP methods org.owasp.csrfguard.ProtectedMethods=POST -# By default do not check anything. ignoreAll is the same as filter coverage in web.xml -org.owasp.csrfguard.unprotected.ignoreAll=*.do +# Do not check anything except for pages which are explicitly marked as protected +org.owasp.csrfguard.Protect = true + +# Mandatory field for stateful applications like LAMS +org.owasp.csrfguard.LogicalSessionExtractor = org.owasp.csrfguard.session.SessionTokenKeyExtractor + # List of actions to check # Each key goes into a separate line prefixed with org.owasp.csrfguard.protected. # A key suffix must not contain a dot "." character @@ -59,6 +63,8 @@ org.owasp.csrfguard.protected.centralPortraitDelete=/lams/saveportrait/deletePortrait.do org.owasp.csrfguard.protected.centralPortraitSave=/lams/saveportrait.do org.owasp.csrfguard.protected.centralPasswordChange=/lams/passwordChanged.do +org.owasp.csrfguard.protected.centralForgotPassword=/lams/ForgotPasswordRequest +org.owasp.csrfguard.protected.centralLogin=/lams/j_security_check #QB org.owasp.csrfguard.protected.centralSaveQuestion=/lams/qb/edit/saveOrUpdateQuestion.do @@ -127,6 +133,7 @@ org.owasp.csrfguard.protected.assessmentMonitoringExportExcel=/lams/tool/laasse10/monitoring/exportSummary.do org.owasp.csrfguard.protected.assessmentMonitoringAllocateUserAnswer=/lams/tool/laasse10/monitoring/allocateUserAnswer.do org.owasp.csrfguard.protected.assessmentMonitoringSetActivityEvaluation=/lams/tool/laasse10/monitoring/setActivityEvaluation.do +org.owasp.csrfguard.protected.assessmentMonitoringChangeLeader=/lams/tool/laasse10/monitoring/changeLeaderForGroup.do org.owasp.csrfguard.protected.assessmentSaveUserGrade=/lams/tool/laasse10/monitoring/saveUserGrade.do org.owasp.csrfguard.protected.assessmentUpdateTimeLimit=/lams/tool/laasse10/monitoring/updateTimeLimit.do org.owasp.csrfguard.protected.assessmentUpdateIndividualTimeLimit=/lams/tool/laasse10/monitoring/updateIndividualTimeLimit.do @@ -141,6 +148,9 @@ org.owasp.csrfguard.protected.dokuAuthoringSave=/lams/tool/ladoku11/authoring/update.do org.owasp.csrfguard.protected.dokuAuthoringDefineLater=/lams/tool/ladoku11/authoring/definelater.do org.owasp.csrfguard.protected.dokuMonitoringUpdateLearnerMark=/lams/tool/ladoku11/monitoring/updateLearnerMark.do +org.owasp.csrfguard.protected.dokuMonitoringChangeLeader=/lams/tool/ladoku11/monitoring/changeLeaderForGroup.do +org.owasp.csrfguard.protected.dokuMonitoringUpdateTimeLimit=/lams/tool/ladoku11/monitoring/updateTimeLimit.do +org.owasp.csrfguard.protected.dokuMonitoringUpdateIndividualTimeLimit=/lams/tool/ladoku11/monitoring/updateIndividualTimeLimit.do org.owasp.csrfguard.protected.forumAuthoringSave=/lams/tool/lafrum11/authoring/update.do org.owasp.csrfguard.protected.forumAuthoringDefineLater=/lams/tool/lafrum11/authoring/definelater.do @@ -164,7 +174,7 @@ org.owasp.csrfguard.protected.leaderAuthoringSave=/lams/tool/lalead11/authoring/updateContent.do org.owasp.csrfguard.protected.leaderAuthoringDefineLater=/lams/tool/lalead11/authoring/definelater.do org.owasp.csrfguard.protected.leaderSaveLeaders=/lams/tool/lalead11/monitoring/saveLeaders.do -org.owasp.csrfguard.protected.leaderTblmonitoringChangeLeader=/lams/tool/lalead11/tblmonitoring/changeLeader.do +org.owasp.csrfguard.protected.leaderMonitoringChangeLeader=/lams/tool/lalead11/monitoring/changeLeader.do org.owasp.csrfguard.protected.laqaAuthoringSave=/lams/tool/laqa11/authoring/submitAllContent.do org.owasp.csrfguard.protected.laqaAuthoringDefineLater=/lams/tool/laqa11/authoring/definelater.do @@ -173,6 +183,7 @@ org.owasp.csrfguard.protected.laqaAuthoringSaveOrUpdateCondition=/lams/tool/laqa11/authoringConditions/saveOrUpdateCondition.do org.owasp.csrfguard.protected.laqaAuthoringRemoveCondition=/lams/tool/laqa11/authoringConditions/removeCondition.do org.owasp.csrfguard.protected.laqaMonitoringSubmissionDeadline=/lams/tool/laqa11/monitoring/setSubmissionDeadline.do +org.owasp.csrfguard.protected.laqaMonitoringChangeLeader=/lams/tool/laqa11/monitoring/changeLeaderForGroup.do org.owasp.csrfguard.protected.larsrcAuthoringSave=/lams/tool/larsrc11/authoring/update.do org.owasp.csrfguard.protected.larsrcAuthoringDefineLater=/lams/tool/larsrc11/authoring/definelater.do @@ -204,6 +215,7 @@ org.owasp.csrfguard.protected.sbmtAuthoringSave=/lams/tool/lasbmt11/authoring/updateContent.do org.owasp.csrfguard.protected.sbmtAuthoringDefineLater=/lams/tool/lasbmt11/authoring/definelater.do org.owasp.csrfguard.protected.sbmtMonitoringSubmissionDeadline=/lams/tool/lasbmt11/monitoring/setSubmissionDeadline.do +org.owasp.csrfguard.protected.sbmtMonitoringChangeLeader=/lams/tool/lasbmt11/monitoring/changeLeaderForGroup.do org.owasp.csrfguard.protected.sbmtUpdateMark=/lams/tool/lasbmt11/mark/updateMark.do org.owasp.csrfguard.protected.sbmtReleaseMarks=/lams/tool/lasbmt11/monitoring/releaseMarks.do org.owasp.csrfguard.protected.sbmtDownloadMarks=/lams/tool/lasbmt11/monitoring/downloadMarks.do @@ -218,8 +230,11 @@ org.owasp.csrfguard.protected.scratchieAuthoringRemoveItem=/lams/tool/lascrt11/authoring/removeItem.do org.owasp.csrfguard.protected.scratchieMonitoringExportExcel=/lams/tool/lascrt11/monitoring/exportExcel.do org.owasp.csrfguard.protected.scratchieMonitoringSubmissionDeadline=/lams/tool/lascrt11/monitoring/setSubmissionDeadline.do +org.owasp.csrfguard.protected.scratchieMonitoringChangeLeader=/lams/tool/lascrt11/monitoring/changeLeaderForGroup.do org.owasp.csrfguard.protected.scratchieTblMonitoringExportExcel=/lams/tool/lascrt11/tblmonitoring/exportExcel.do org.owasp.csrfguard.protected.scratchieSaveUserMark=/lams/tool/lascrt11/monitoring/saveUserMark.do +org.owasp.csrfguard.protected..scratchieUpdateTimeLimit=/lams/tool/lascrt11/monitoring/updateTimeLimit.do +org.owasp.csrfguard.protected..scratchieUpdateIndividualTimeLimit=/lams/tool/lascrt11/monitoring/updateIndividualTimeLimit.do org.owasp.csrfguard.protected.spreadsheetAuthoringSave=/lams/tool/lasprd10/authoring/updateContent.do org.owasp.csrfguard.protected.spreadsheetAuthoringDefineLater=/lams/tool/lasprd10/authoring/definelater.do @@ -240,6 +255,7 @@ org.owasp.csrfguard.protected.voteAuthoringAddNomination=/lams/tool/lavote11/authoring/addSingleNomination.do org.owasp.csrfguard.protected.voteAuthoringRemoveNomination=/lams/tool/lavote11/authoring/removeNomination.do org.owasp.csrfguard.protected.voteMonitoringSubmissionDeadline=/lams/tool/lavote11/monitoring/setSubmissionDeadline.do +org.owasp.csrfguard.protected.voteMonitoringChangeLeader=/lams/tool/lavote11/monitoring/changeLeaderForGroup.do org.owasp.csrfguard.protected.voteHideOpenVote=/lams/tool/lavote11/monitoring/hideOpenVote.do org.owasp.csrfguard.protected.voteShowOpenVote=/lams/tool/lavote11/monitoring/showOpenVote.do Index: lams_central/src/java/org/lamsfoundation/lams/authoring/template/web/LdTemplateController.java =================================================================== diff -u -rcf258bae4dd72eab7a6f5bc895b96966caafaf06 -recc49cd6851b43f37ef02c2ddb85257096e2cf49 --- lams_central/src/java/org/lamsfoundation/lams/authoring/template/web/LdTemplateController.java (.../LdTemplateController.java) (revision cf258bae4dd72eab7a6f5bc895b96966caafaf06) +++ lams_central/src/java/org/lamsfoundation/lams/authoring/template/web/LdTemplateController.java (.../LdTemplateController.java) (revision ecc49cd6851b43f37ef02c2ddb85257096e2cf49) @@ -41,6 +41,7 @@ import javax.servlet.http.HttpSession; import org.apache.log4j.Logger; +import org.lamsfoundation.lams.authoring.service.AuthoringService; import org.lamsfoundation.lams.authoring.service.IAuthoringFullService; import org.lamsfoundation.lams.authoring.template.AssessMCAnswer; import org.lamsfoundation.lams.authoring.template.Assessment; @@ -58,10 +59,8 @@ import org.lamsfoundation.lams.questions.Question; import org.lamsfoundation.lams.questions.QuestionParser; import org.lamsfoundation.lams.rest.RestTags; -import org.lamsfoundation.lams.rest.ToolRestManager; import org.lamsfoundation.lams.tool.Tool; import org.lamsfoundation.lams.tool.dao.IToolDAO; -import org.lamsfoundation.lams.tool.exception.ToolException; import org.lamsfoundation.lams.tool.service.ILamsCoreToolService; import org.lamsfoundation.lams.usermanagement.dto.UserDTO; import org.lamsfoundation.lams.util.AuthoringJsonTags; @@ -110,7 +109,7 @@ NumberFormat groupNumberFormatter = new DecimalFormat("00"); private static Logger log = Logger.getLogger(LdTemplateController.class); - public static final int MAX_OPTION_COUNT = 6; + public static final int MAX_OPTION_COUNT = 20; public static final int MAX_FLOATING_ACTIVITY_OPTIONS = 6; // Hardcoded in the Flash client public static final String PARENT_ACTIVITY_TYPE = "parentActivityType"; // used to work out transitions - not used by the authoring module @@ -130,37 +129,39 @@ // icon strings found in the lams_learningdesign_activity table protected static final String ASSESSMENT_TOOL_SIGNATURE = "laasse10"; - protected static final String ASSESSMENT_ICON = "tool/laasse10/images/icon_assessment.swf"; + protected static final String ASSESSMENT_ICON = "tool/laasse10/images/icon_assessment.svg"; protected static final String ASSESSMENT_TOOL_OUTPUT_DEFINITION = "learner.total.score"; protected static final String CHAT_TOOL_SIGNATURE = "lachat11"; - protected static final String CHAT_ICON = "tool/lachat11/images/icon_chat.swf"; + protected static final String CHAT_ICON = "tool/lachat11/images/icon_chat.svg"; + protected static final String DOKU_TOOL_SIGNATURE = "ladoku11"; + protected static final String DOKU_ICON = "tool/ladoku11/images/icon_dokumaran.svg"; protected static final String FORUM_TOOL_SIGNATURE = "lafrum11"; - protected static final String FORUM_ICON = "tool/lafrum11/images/icon_forum.swf"; + protected static final String FORUM_ICON = "tool/lafrum11/images/icon_forum.svg"; protected static final String LEADER_TOOL_SIGNATURE = "lalead11"; protected static final String LEADER_ICON = "tool/lalead11/images/icon_leaderselection.swf"; protected static final String NOTEBOOK_TOOL_SIGNATURE = "lantbk11"; - protected static final String NOTEBOOK_ICON = "tool/lantbk11/images/icon_notebook.swf"; + protected static final String NOTEBOOK_ICON = "tool/lantbk11/images/icon_notebook.svg"; protected static final String NOTICEBOARD_TOOL_SIGNATURE = "lanb11"; - protected static final String NOTICEBOARD_ICON = "tool/lanb11/images/icon_htmlnb.swf"; + protected static final String NOTICEBOARD_ICON = "tool/lanb11/images/icon_htmlnb.svg"; protected static final String QA_TOOL_SIGNATURE = "laqa11"; - protected static final String QA_ICON = "tool/laqa11/images/icon_questionanswer.swf"; + protected static final String QA_ICON = "tool/laqa11/images/icon_questionanswer.svg"; protected static final String SHARE_RESOURCES_TOOL_SIGNATURE = "larsrc11"; - protected static final String SHARE_RESOURCES_ICON = "tool/larsrc11/images/icon_rsrc.swf"; + protected static final String SHARE_RESOURCES_ICON = "tool/larsrc11/images/icon_rsrc.svg"; protected static final String SCRATCHIE_TOOL_SIGNATURE = "lascrt11"; - protected static final String SCRATCHIE_ICON = "tool/lascrt11/images/icon_scratchie.swf"; + protected static final String SCRATCHIE_ICON = "tool/lascrt11/images/icon_scratchie.svg"; protected static final String SCRATCHIE_TOOL_OUTPUT_DEFINITION = "learner.mark"; protected static final String SCRIBE_TOOL_SIGNATURE = "lascrb11"; - protected static final String SCRIBE_ICON = "tool/lascrb11/images/icon_scribe.swf"; + protected static final String SCRIBE_ICON = "tool/lascrb11/images/icon_scribe.svg"; protected static final String SUBMIT_TOOL_SIGNATURE = "lasbmt11"; - protected static final String SUBMIT_ICON = "tool/lasbmt11/images/icon_reportsubmission.swf"; + protected static final String SUBMIT_ICON = "tool/lasbmt11/images/icon_reportsubmission.svg"; protected static final String SURVEY_TOOL_SIGNATURE = "lasurv11"; - protected static final String SURVEY_ICON = "tool/lasurv11/images/icon_survey.swf"; + protected static final String SURVEY_ICON = "tool/lasurv11/images/icon_survey.svg"; protected static final String WIKI_TOOL_SIGNATURE = "lawiki10"; - protected static final String WIKI_ICON = "tool/lawiki10/images/icon_wiki.swf"; + protected static final String WIKI_ICON = "tool/lawiki10/images/icon_wiki.svg"; protected static final String MINDMAP_TOOL_SIGNATURE = "lamind10"; - protected static final String MINDMAP_ICON = "tool/lamind10/images/icon_mindmap.swf"; + protected static final String MINDMAP_ICON = "tool/lamind10/images/icon_mindmap.svg"; protected static final String VOTE_TOOL_SIGNATURE = "lavote11"; - protected static final String VOTE_ICON = "tool/lavote11/images/icon_ranking.swf"; + protected static final String VOTE_ICON = "tool/lavote11/images/icon_ranking.svg"; protected static final String PEER_REVIEW_TOOL_SIGNATURE = "laprev11"; protected static final String PEER_REVIEW_ICON = "tool/laprev11/images/icon_peerreview.svg"; @@ -286,10 +287,10 @@ return transitions; } - protected static final int rowHeightSpace = 100; - protected static final int activityWidthSpace = 185; - protected static final int gateHeightOffset = 5; - protected static final int gateWidthOffset = 50; + protected static final int rowHeightSpace = 120; + protected static final int activityWidthSpace = 240; + protected static final int gateHeightOffset = 20; + protected static final int gateWidthOffset = 70; /** * Calculate where to draw an activity. Aim for 4 activities per line. Returns Integer[x,y] @@ -399,7 +400,7 @@ /* ************************************** Non-Tool Activity methods ******************************************** */ protected ObjectNode createGateActivity(AtomicInteger uiid, int order, Integer[] layoutCoords, String activityTitle, - String activityDescription) { + String activityDescription, boolean gateStopAtPrecedingActivity) { ObjectNode activityJSON = JsonNodeFactory.instance.objectNode(); Integer[] pos = layoutCoords; @@ -421,14 +422,16 @@ } activityJSON.put(AuthoringJsonTags.ACTIVITY_TYPE_ID, Activity.PERMISSION_GATE_ACTIVITY_TYPE); activityJSON.put(AuthoringJsonTags.GATE_ACTIVITY_LEVEL_ID, GateActivity.LEARNER_GATE_LEVEL); + activityJSON.put(AuthoringJsonTags.GATE_STOP_AT_PRECEDING_ACTIVITY, gateStopAtPrecedingActivity); return activityJSON; } protected ObjectNode createScheduledGateActivity(AtomicInteger uiid, int order, Integer[] layoutCoords, - String activityTitle, String activityDescription, Long startOffset) { + String activityTitle, String activityDescription, Long startOffset, boolean gateStopAtPrecedingActivity) { - ObjectNode activityJSON = createGateActivity(uiid, order, layoutCoords, activityTitle, activityDescription); + ObjectNode activityJSON = createGateActivity(uiid, order, layoutCoords, activityTitle, activityDescription, + gateStopAtPrecedingActivity); activityJSON.put(AuthoringJsonTags.ACTIVITY_TYPE_ID, Activity.SCHEDULE_GATE_ACTIVITY_TYPE); activityJSON.put(AuthoringJsonTags.GATE_START_OFFSET, startOffset); @@ -662,29 +665,6 @@ return activityJSON; } - /* ************************************** Tool related methods ********************************************** */ - /** General method to create a tool content. All calls to create tool content should go through this method */ - protected Long createToolContent(UserDTO user, String toolSignature, ObjectNode toolContentJSON) - throws IOException { - try { - Tool tool = getTool(toolSignature); - Long toolContentID = authoringService.insertToolContentID(tool.getToolId()); - - // Tools' services implement an interface for processing REST requests - ToolRestManager toolRestService = (ToolRestManager) lamsCoreToolService.findToolService(tool); - toolRestService.createRestToolContent(user.getUserID(), toolContentID, toolContentJSON); - - return toolContentID; - } catch (Exception e) { - log.error("Unable to create tool content for " + toolSignature + " with details " + toolContentJSON - + ". \nThe tool probably threw an exception - check the server logs for more details.\n" - + "If the exception is \"Servlet.service() for servlet ToolContentRestServlet threw exception java.lang.ClassCastException: com.sun.proxy.$ProxyXXX cannot be cast to org.lamsfoundation.lams.rest.ToolRestManager)\"" - + " then the tool doesn't support the LDTemplate service calls (ie has not implemented the ToolRestManager interface / createRestToolContent() method."); - throw new ToolException( - "Unable to create tool content for " + toolSignature + " with details " + toolContentJSON); - } - } - /** * General method to create tool activity. All calls to create an activity relating to a tool should go through this * method @@ -732,87 +712,7 @@ return activityJSON; } - /** Sets up the standard fields that are used by many tools! */ - protected ObjectNode createStandardToolContent(String title, String instructions, String reflectionInstructions, - Boolean lockWhenFinished, Boolean allowRichTextEditor, UserDTO user) { - ObjectNode toolContentJSON = JsonNodeFactory.instance.objectNode(); - toolContentJSON.put(RestTags.TITLE, title != null ? title : ""); - toolContentJSON.put(RestTags.INSTRUCTIONS, instructions != null ? instructions : ""); - - if (reflectionInstructions != null) { - toolContentJSON.put(RestTags.REFLECT_ON_ACTIVITY, true); - toolContentJSON.put(RestTags.REFLECT_INSTRUCTIONS, reflectionInstructions); - } - - toolContentJSON.put(RestTags.LOCK_WHEN_FINISHED, lockWhenFinished); - toolContentJSON.put(RestTags.ALLOW_RICH_TEXT_EDITOR, allowRichTextEditor); - - if (user != null) { - toolContentJSON.put("firstName", user.getFirstName()); - toolContentJSON.put("lastName", user.getLastName()); - toolContentJSON.put("loginName", user.getLogin()); - } - return toolContentJSON; - } - /** - * Helper method to create a Assessment tool content. Assessment is one of the unusuals tool in that it caches - * user's login names and - * first/last names Mandatory fields in toolContentJSON: title, instructions, resources, user fields firstName, - * lastName and loginName. - * - * Required fields in toolContentJSON: "title", "instructions", "questions", "firstName", "lastName", "lastName", - * "questions" and "references". - * - * The questions entry should be ArrayNode containing JSON objects, which in turn must contain - * "questionTitle", "questionText", "displayOrder" (Integer), "type" (Integer). If the type is Multiple Choice, - * Numerical or Matching Pairs - * then a ArrayNode "answers" is required. - * - * The answers entry should be ArrayNode - * containing JSON objects, which in turn must contain "answerText" or "answerFloat", "displayOrder" (Integer), - * "grade" (Integer). - * - * For the templates, all the questions that are created will be set up as references, therefore the questions in - * the assessment == the bank of questions. - * So references entry will be a ArrayNode containing JSON objects, which in turn must contain "displayOrder" - * (Integer), - * "questionDisplayOrder" (Integer - to match to the question). If default grade or random questions are needed then - * this method needs - * to be expanded. - */ - protected Long createAssessmentToolContent(UserDTO user, String title, String instructions, - String reflectionInstructions, boolean selectLeaderToolOutput, boolean enableNumbering, - boolean enableConfidenceLevels, boolean allowDiscloseAnswers, boolean allowAnswerJustification, - ArrayNode questions) throws IOException { - - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, reflectionInstructions, null, null, - user); - toolContentJSON.put(RestTags.USE_SELECT_LEADER_TOOL_OUTPUT, selectLeaderToolOutput); - toolContentJSON.put(RestTags.ENABLE_CONFIDENCE_LEVELS, enableConfidenceLevels); - toolContentJSON.put("numbered", enableNumbering); - toolContentJSON.put("displaySummary", Boolean.TRUE); - toolContentJSON.put("allowDiscloseAnswers", allowDiscloseAnswers); - toolContentJSON.put("allowAnswerJustification", allowAnswerJustification); - - toolContentJSON.set(RestTags.QUESTIONS, questions); - - ArrayNode references = JsonNodeFactory.instance.arrayNode(); - for (int i = 0; i < questions.size(); i++) { - ObjectNode question = (ObjectNode) questions.get(i); - question.put("answerRequired", true); - - Integer questionDisplayOrder = question.get(RestTags.DISPLAY_ORDER).asInt(); - Integer defaultGrade = JsonUtil.optInt(question, "defaultGrade", 1); - references.add(JsonNodeFactory.instance.objectNode().put(RestTags.DISPLAY_ORDER, questionDisplayOrder) - .put("questionDisplayOrder", questionDisplayOrder).put("defaultGrade", defaultGrade)); - } - toolContentJSON.set("references", references); - - return createToolContent(user, LdTemplateController.ASSESSMENT_TOOL_SIGNATURE, toolContentJSON); - } - - /** * Creates a forum activity's JSON details. */ protected ObjectNode createAssessmentActivity(AtomicInteger uiid, int order, Integer[] layoutCoords, @@ -833,10 +733,10 @@ protected Long createChatToolContent(UserDTO user, String title, String instructions, boolean lockWhenFinished, String filterKeywords, String reflectionInstructions) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, reflectionInstructions, - lockWhenFinished, null, null); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, + reflectionInstructions, lockWhenFinished, null, null); toolContentJSON.put("filterKeywords", filterKeywords); - return createToolContent(user, LdTemplateController.CHAT_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.CHAT_TOOL_SIGNATURE, toolContentJSON); } /** @@ -872,8 +772,8 @@ boolean allowRichTextEditor, boolean allowNewTopic, boolean allowRateMessages, boolean allowUpload, boolean limitedMaxCharacters, Integer maxCharacters, ArrayNode topics) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, null, lockWhenFinished, - allowRichTextEditor, user); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, null, + lockWhenFinished, allowRichTextEditor, user); toolContentJSON.set("topics", topics); toolContentJSON.put("allowNewTopic", allowNewTopic); toolContentJSON.put("allowRateMessages", allowRateMessages); @@ -882,7 +782,7 @@ if (limitedMaxCharacters && maxCharacters != null) { toolContentJSON.put("maxCharacters", maxCharacters); } - return createToolContent(user, LdTemplateController.FORUM_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.FORUM_TOOL_SIGNATURE, toolContentJSON); } /** @@ -903,8 +803,9 @@ */ protected Long createLeaderSelectionToolContent(UserDTO user, String title, String instructions) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, null, null, null, null); - return createToolContent(user, LdTemplateController.LEADER_TOOL_SIGNATURE, toolContentJSON); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, null, null, null, + null); + return authoringService.createToolContent(user, LdTemplateController.LEADER_TOOL_SIGNATURE, toolContentJSON); } /** @@ -926,9 +827,9 @@ protected Long createNotebookToolContent(UserDTO user, String title, String instructions, boolean lockWhenFinished, boolean allowRichTextEditor) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, null, lockWhenFinished, - allowRichTextEditor, null); - return createToolContent(user, LdTemplateController.NOTEBOOK_TOOL_SIGNATURE, toolContentJSON); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, null, + lockWhenFinished, allowRichTextEditor, null); + return authoringService.createToolContent(user, LdTemplateController.NOTEBOOK_TOOL_SIGNATURE, toolContentJSON); } /** @@ -950,9 +851,11 @@ protected Long createNoticeboardToolContent(UserDTO user, String title, String content, String reflectionInstructions) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, null, reflectionInstructions, null, null, null); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, null, reflectionInstructions, + null, null, null); toolContentJSON.put("content", content != null ? content : ""); - return createToolContent(user, LdTemplateController.NOTICEBOARD_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.NOTICEBOARD_TOOL_SIGNATURE, + toolContentJSON); } /** @@ -975,13 +878,13 @@ boolean allowRichTextEditor, boolean oneQuestionPerPage, boolean showOtherLearnersAnswers, boolean showOtherLearnersNames, ArrayNode questions) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, null, lockWhenFinished, - allowRichTextEditor, null); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, null, + lockWhenFinished, allowRichTextEditor, null); toolContentJSON.set(RestTags.QUESTIONS, questions); toolContentJSON.put("questionsSequenced", oneQuestionPerPage); toolContentJSON.put("showOtherAnswers", showOtherLearnersAnswers); toolContentJSON.put("usernameVisible", showOtherLearnersNames); - return createToolContent(user, LdTemplateController.QA_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.QA_TOOL_SIGNATURE, toolContentJSON); } /** @@ -1002,9 +905,10 @@ protected Long createMindmapToolContent(UserDTO user, String title, String instructions, boolean lockWhenFinished, boolean multiUserMode, String reflectionInstruction) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, null, lockWhenFinished, null, null); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, null, + lockWhenFinished, null, null); toolContentJSON.put("multiUserMode", multiUserMode); - return createToolContent(user, LdTemplateController.MINDMAP_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.MINDMAP_TOOL_SIGNATURE, toolContentJSON); } /** @@ -1031,8 +935,8 @@ boolean notifyInstructors, Integer minResourcesToView, String reflectionInstructions, ArrayNode resources) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, reflectionInstructions, - lockWhenFinished, null, user); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, + reflectionInstructions, lockWhenFinished, null, user); toolContentJSON.put("allowAddFiles", allowLearnerAddFile); toolContentJSON.put("allowAddUrls", allowLearnerAddURL); toolContentJSON.put("notifyTeachersOnAssigmentSumbit", notifyInstructors); @@ -1041,7 +945,8 @@ toolContentJSON.put("minViewResourceNumber", minResourcesToView); } toolContentJSON.set("resources", resources); - return createToolContent(user, LdTemplateController.SHARE_RESOURCES_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.SHARE_RESOURCES_TOOL_SIGNATURE, + toolContentJSON); } // resource type - copied from ResourceConstants @@ -1122,21 +1027,24 @@ * full details of questions). Other fields are optional. */ protected Long createScratchieToolContent(UserDTO user, String title, String instructions, - boolean useSelectLeaderToolOuput, Integer confidenceLevelsActivityUiid, ArrayNode questions) - throws IOException { + boolean useSelectLeaderToolOuput, boolean enableDiscussionSentiment, Integer confidenceLevelsActivityUiid, + ArrayNode questions) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, null, null, null, null); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, null, null, null, + null); toolContentJSON.set(RestTags.QUESTIONS, questions); if (confidenceLevelsActivityUiid != null) { toolContentJSON.put(RestTags.CONFIDENCE_LEVELS_ACTIVITY_UIID, confidenceLevelsActivityUiid); } + toolContentJSON.put(RestTags.ENABLE_DISCUSSION_SENTIMENT, enableDiscussionSentiment); + for (int i = 0; i < questions.size(); i++) { ObjectNode question = (ObjectNode) questions.get(i); question.put("answerRequired", true); } - return createToolContent(user, LdTemplateController.SCRATCHIE_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.SCRATCHIE_TOOL_SIGNATURE, toolContentJSON); } /** @@ -1160,12 +1068,12 @@ boolean autoSelectScribe, boolean showAggregatedReports, String reflectionInstructions, ArrayNode questions) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, reflectionInstructions, - lockWhenFinished, null, null); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, + reflectionInstructions, lockWhenFinished, null, null); toolContentJSON.set(RestTags.QUESTIONS, questions); toolContentJSON.put("autoSelectScribe", autoSelectScribe); toolContentJSON.put("showAggregatedReports", showAggregatedReports); - return createToolContent(user, LdTemplateController.SCRIBE_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.SCRIBE_TOOL_SIGNATURE, toolContentJSON); } /** @@ -1188,13 +1096,13 @@ protected Long createSubmitToolContent(UserDTO user, String title, String instructions, boolean lockWhenFinished, Boolean limitUpload, Integer limitUploadNumber, String reflectionInstructions) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, reflectionInstructions, - lockWhenFinished, null, user); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, + reflectionInstructions, lockWhenFinished, null, user); if (limitUploadNumber != null) { toolContentJSON.put("limitUpload", limitUpload != null ? limitUpload : true); toolContentJSON.put("limitUploadNumber", limitUploadNumber); } - return createToolContent(user, LdTemplateController.SUBMIT_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.SUBMIT_TOOL_SIGNATURE, toolContentJSON); } /** @@ -1216,9 +1124,10 @@ protected Long createSurveyToolContent(UserDTO user, String title, String instructions, Boolean lockWhenFinished, ArrayNode questions) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, null, lockWhenFinished, null, user); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, null, + lockWhenFinished, null, user); toolContentJSON.set("questions", questions); - return createToolContent(user, LdTemplateController.SURVEY_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.SURVEY_TOOL_SIGNATURE, toolContentJSON); } /** @@ -1240,10 +1149,11 @@ protected Long createVoteToolContent(UserDTO user, String title, String instructions, ArrayNode answers, Boolean showResults) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, null, null, null, null); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, null, null, null, + null); toolContentJSON.set(RestTags.ANSWERS, answers); toolContentJSON.put("showResults", showResults); - return createToolContent(user, LdTemplateController.VOTE_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.VOTE_TOOL_SIGNATURE, toolContentJSON); } /** @@ -1264,9 +1174,10 @@ protected Long createWikiToolContent(UserDTO user, String title, String instructions, boolean lockWhenFinished, String reflectionInstruction, ArrayNode pages) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, null, lockWhenFinished, null, null); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, null, + lockWhenFinished, null, null); toolContentJSON.set("pages", pages); - return createToolContent(user, LdTemplateController.WIKI_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.WIKI_TOOL_SIGNATURE, toolContentJSON); } /** @@ -1291,10 +1202,11 @@ protected Long createPeerReviewToolContent(UserDTO user, String title, String instructions, String reflectionInstructions, ArrayNode criterias) throws IOException { - ObjectNode toolContentJSON = createStandardToolContent(title, instructions, reflectionInstructions, null, null, - user); + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, instructions, + reflectionInstructions, null, null, user); toolContentJSON.set("criterias", criterias); - return createToolContent(user, LdTemplateController.PEER_REVIEW_TOOL_SIGNATURE, toolContentJSON); + return authoringService.createToolContent(user, LdTemplateController.PEER_REVIEW_TOOL_SIGNATURE, + toolContentJSON); } /** @@ -1310,6 +1222,36 @@ } /** + * Helper method to create a doKumaran tool content. + */ + protected Long createDokumaranToolContent(UserDTO user, String title, String description, String instructions, + boolean selectLeaderToolOutput, boolean galleryWalkEnabled, boolean galleryWalkReadOnly, + String galleryWalkInstructions, String reflectionInstructions) throws IOException { + + ObjectNode toolContentJSON = AuthoringService.createStandardToolContent(title, description, + reflectionInstructions, null, null, user); + toolContentJSON.put(RestTags.USE_SELECT_LEADER_TOOL_OUTPUT, selectLeaderToolOutput); + toolContentJSON.put(RestTags.LOCK_WHEN_FINISHED, true); + toolContentJSON.put("etherpadInstructions", instructions); + toolContentJSON.put("galleryWalkEnabled", galleryWalkEnabled); + toolContentJSON.put("galleryWalkReadOnly", galleryWalkReadOnly); + toolContentJSON.put("galleryWalkInstructions", galleryWalkInstructions); + return authoringService.createToolContent(user, LdTemplateController.DOKU_TOOL_SIGNATURE, toolContentJSON); + } + + /** + * Creates a doKumaran activity's JSON details. + */ + protected ObjectNode createDokumaranActivity(AtomicInteger uiid, int order, Integer[] layoutCoords, + Long toolContentID, String contentFolderID, Integer groupingUIID, Integer parentUIID, + Integer parentActivityType, String activityTitle) { + + return createToolActivity(uiid, order, layoutCoords, LdTemplateController.DOKU_TOOL_SIGNATURE, + LdTemplateController.DOKU_ICON, toolContentID, contentFolderID, groupingUIID, parentUIID, + parentActivityType, activityTitle != null ? activityTitle : "doKumaran"); + } + + /** * * /* ************************************** Service related methods ********************************************** */ @@ -1389,7 +1331,7 @@ model.addAttribute(AttributeNames.PARAM_CONTENT_FOLDER_ID, qbQuestion.getContentFolderId()); model.addAttribute("question", question); - if (question.getType() == Assessment.ASSESSMENT_QUESTION_TYPE_MULTIPLE_CHOICE) { + if (question.getType() == QbQuestion.TYPE_MULTIPLE_CHOICE) { Set answers = question.getAnswers(); for (QbOption qbOption : qbQuestion.getQbOptions()) { AssessMCAnswer answer = new AssessMCAnswer(qbOption.getDisplayOrder(), qbOption.getName(), @@ -1410,10 +1352,11 @@ QbQuestion qbQuestion = qbService.getQuestionByUid(qbQuestionUid); Assessment question = new Assessment(); - question.setType(Assessment.ASSESSMENT_QUESTION_TYPE_MULTIPLE_CHOICE); + question.setType(qbQuestion.getType()); question.setTitle(qbQuestion.getName()); question.setText(qbQuestion.getDescription()); question.setUuid(qbQuestion.getUuid().toString()); + question.setDefaultGrade(qbQuestion.getMaxMark()); Set answers = question.getAnswers(); for (QbOption qbOption : qbQuestion.getQbOptions()) { @@ -1439,16 +1382,17 @@ assessments.add(assessment); boolean isMultipleChoice = Question.QUESTION_TYPE_MULTIPLE_CHOICE.equals(question.getType()); + boolean isMarkHedging = Question.QUESTION_TYPE_MARK_HEDGING.equals(question.getType()); boolean isMultipleResponse = Question.QUESTION_TYPE_MULTIPLE_RESPONSE.equals(question.getType()); - int defaultGrade = 1; + Integer defaultGrade = question.getScore(); assessment.setText(QuestionParser.processHTMLField(question.getText(), false, contentFolderID, question.getResourcesFolderPath())); assessment.setTitle(question.getTitle()); assessment.setUuid(question.getQbUUID()); - if (isMultipleChoice) { - assessment.setType(Assessment.ASSESSMENT_QUESTION_TYPE_MULTIPLE_CHOICE); + if (isMultipleChoice || isMarkHedging) { + assessment.setType(isMultipleChoice ? QbQuestion.TYPE_MULTIPLE_CHOICE : QbQuestion.TYPE_MARK_HEDGING); assessment.setMultipleAnswersAllowed(false); String correctAnswer = null; @@ -1467,8 +1411,10 @@ if ((answer.getScore() != null) && (answer.getScore() > 0)) { if (correctAnswer == null) { - // whatever the correct answer holds, it becomes the question score - defaultGrade = Double.valueOf(Math.ceil(answer.getScore())).intValue(); + if (defaultGrade == null) { + // if grade not explicitly set, whatever the correct answer holds, it becomes the question score + defaultGrade = Double.valueOf(Math.ceil(answer.getScore())).intValue(); + } // 100% goes to the correct answer newAnswer.setGrade(1F); correctAnswer = answerText; @@ -1488,7 +1434,7 @@ } } else if (isMultipleResponse) { - assessment.setType(Assessment.ASSESSMENT_QUESTION_TYPE_MULTIPLE_CHOICE); + assessment.setType(QbQuestion.TYPE_MULTIPLE_CHOICE); assessment.setMultipleAnswersAllowed(true); if (question.getAnswers() != null) { @@ -1499,7 +1445,10 @@ totalScore += answer.getScore(); } } - defaultGrade = Double.valueOf(Math.round(totalScore)).intValue(); + if (defaultGrade == null) { + // if grade not explicitly set, user total score as grade + defaultGrade = Double.valueOf(Math.round(totalScore)).intValue(); + } int displayOrder = 1; for (Answer answer : question.getAnswers()) { @@ -1519,10 +1468,12 @@ } } } else { - assessment.setType(Assessment.ASSESSMENT_QUESTION_TYPE_ESSAY); + assessment.setType(QbQuestion.TYPE_ESSAY); } - assessment.setDefaultGrade(defaultGrade); + assessment.setDefaultGrade(defaultGrade == null ? 1 : defaultGrade); + + assessment.setLearningOutcomes(question.getLearningOutcomes()); } return assessments; Index: lams_central/src/java/org/lamsfoundation/lams/web/qb/SearchQBController.java =================================================================== diff -u -rfffb5d73e9c9bf785cbf01613e6972065cf88bc6 -recc49cd6851b43f37ef02c2ddb85257096e2cf49 --- lams_central/src/java/org/lamsfoundation/lams/web/qb/SearchQBController.java (.../SearchQBController.java) (revision fffb5d73e9c9bf785cbf01613e6972065cf88bc6) +++ lams_central/src/java/org/lamsfoundation/lams/web/qb/SearchQBController.java (.../SearchQBController.java) (revision ecc49cd6851b43f37ef02c2ddb85257096e2cf49) @@ -103,8 +103,9 @@ questionTypesAvailable.append(QbQuestion.TYPE_MULTIPLE_CHOICE); questionTypesAvailable.append(","); questionTypesAvailable.append(QbQuestion.TYPE_VERY_SHORT_ANSWERS); + questionTypesAvailable.append(","); + questionTypesAvailable.append(QbQuestion.TYPE_MARK_HEDGING); - //CommonConstants.TOOL_SIGNATURE_SURVEY } else if ("lasurv11".equals(toolSignature)) { questionTypesAvailable.append(QbQuestion.TYPE_MULTIPLE_CHOICE); questionTypesAvailable.append(","); @@ -130,7 +131,14 @@ //CommonConstants.TOOL_SIGNATURE_QA } else if ("laqa11".equals(toolSignature)) { questionTypeDefault = QbQuestion.TYPE_ESSAY; + } else if ("tblIrat".equals(toolSignature)) { + // this is a special tool "signature" which allows mcq and mark hedging + // used by iRAT page in TBL template + questionTypesAvailable.append(QbQuestion.TYPE_MULTIPLE_CHOICE); + questionTypesAvailable.append(","); + questionTypesAvailable.append(QbQuestion.TYPE_MARK_HEDGING); } + request.setAttribute("questionType", questionTypeDefault); request.setAttribute("questionTypesAvailable", questionTypesAvailable.toString()); //let jsp know it's Scratchie, so we can disable VSA questions not compatible with TBL @@ -224,8 +232,8 @@ @RequestMapping("/getPagedQuestions") @ResponseBody private String getPagedQuestions(HttpServletRequest request, HttpServletResponse response, - @RequestParam(required = false) String questionTypes, - @RequestParam(required = false) String collectionUids) { + @RequestParam(required = false) String questionTypes, @RequestParam(required = false) String collectionUids, + @RequestParam(required = false) Long toolContentID) { if (StringUtils.isEmpty(questionTypes)) { questionTypes = null; } @@ -244,8 +252,8 @@ String searchString = WebUtil.readStrParam(request, "searchString", true); // Get the user list from the db - List questions = qbService.getPagedQuestions(questionTypes, collectionUids, page - 1, rowLimit, - sortBy, sortOrder, searchString); + List questions = qbService.getPagedQuestions(questionTypes, collectionUids, toolContentID, page - 1, + rowLimit, sortBy, sortOrder, searchString); int countQuestions = qbService.getCountQuestions(questionTypes, collectionUids, searchString); int totalPages = Double.valueOf(Math.ceil(Double.valueOf(countQuestions) / Double.valueOf(rowLimit))) .intValue(); @@ -291,8 +299,7 @@ boolean isScratchie = WebUtil.readBooleanParam(request, "isScratchie", false); request.setAttribute("isScratchie", isScratchie); - List otherVersions = qbService.getQuestionsByQuestionId(qbQuestion.getQuestionId()); - request.setAttribute("otherVersions", otherVersions); + qbService.fillVersionMap(qbQuestion); return "qb/qbQuestionDetails"; } Index: lams_central/src/java/org/lamsfoundation/lams/webservice/xml/LessonManagerServlet.java =================================================================== diff -u -rb5d2a6851ee4797e641081bd5e6a1f28f78c30a6 -recc49cd6851b43f37ef02c2ddb85257096e2cf49 --- lams_central/src/java/org/lamsfoundation/lams/webservice/xml/LessonManagerServlet.java (.../LessonManagerServlet.java) (revision b5d2a6851ee4797e641081bd5e6a1f28f78c30a6) +++ lams_central/src/java/org/lamsfoundation/lams/webservice/xml/LessonManagerServlet.java (.../LessonManagerServlet.java) (revision ecc49cd6851b43f37ef02c2ddb85257096e2cf49) @@ -928,8 +928,8 @@ if (StringUtils.isNotBlank(userName)) { // integrationService.addExtUserToLesson(extServer, IntegrationConstants.METHOD_LEARNER, lessonId, // userName, firstName, lastName, email, courseId, countryIsoCode, langIsoCode); - integrationService.addExtUserToCourseAndLesson(extServer, IntegrationConstants.METHOD_LEARNER, lessonId, - userName, firstName, lastName, email, courseId, country, locale); + integrationService.addExtUserToCourseAndLesson(extServer, IntegrationConstants.METHOD_LEARNER, + lessonId, userName, firstName, lastName, email, courseId, country, locale, true); } i++; } @@ -946,8 +946,8 @@ } if (StringUtils.isNotBlank(userName)) { - integrationService.addExtUserToCourseAndLesson(extServer, IntegrationConstants.METHOD_MONITOR, lessonId, - userName, firstName, lastName, email, courseId, country, locale); + integrationService.addExtUserToCourseAndLesson(extServer, IntegrationConstants.METHOD_MONITOR, + lessonId, userName, firstName, lastName, email, courseId, country, locale, true); } i++; } @@ -1018,8 +1018,8 @@ Element lessonElement = document.createElement(CentralConstants.ELEM_LESSON); lessonElement.setAttribute(CentralConstants.ATTR_LESSON_ID, "" + lessonId); lessonElement.setAttribute("lessonName", lesson.getLessonName()); - String createDateTime = lesson.getCreateDateTime().toString(); - lessonElement.setAttribute("createDateTime", StringUtils.isBlank(createDateTime) ? "" : createDateTime); + String createDateTime = lesson.getCreateDateTime().toString(); + lessonElement.setAttribute("createDateTime", StringUtils.isBlank(createDateTime) ? "" : createDateTime); // calculate lesson's MaxPossibleMark Long lessonMaxPossibleMark = lamsCoreToolService.getLessonMaxPossibleMark(lesson); @@ -1179,8 +1179,8 @@ toolOutputsElement.setAttribute(CentralConstants.ATTR_LESSON_ID, "" + lessonId); toolOutputsElement.setAttribute("name", lesson.getLessonName()); - String createDateTime = lesson.getCreateDateTime().toString(); - toolOutputsElement.setAttribute("createDateTime", StringUtils.isBlank(createDateTime) ? "" : createDateTime); + String createDateTime = lesson.getCreateDateTime().toString(); + toolOutputsElement.setAttribute("createDateTime", StringUtils.isBlank(createDateTime) ? "" : createDateTime); List learnerProgresses = lessonService.getUserProgressForLesson(lesson.getLessonId()); List toolSessions = lamsCoreToolService.getToolSessionsByLesson(lesson); @@ -1369,7 +1369,8 @@ ToolOutputDefinition definition = toolOutputDefinitions.get(outputName); if (isAuthoredToolOutputs) { - ActivityEvaluation evaluation = activity.getEvaluation(); + ActivityEvaluation evaluation = (ActivityEvaluation) userManagementService + .findById(ActivityEvaluation.class, activity.getActivityId()); if (evaluation != null) { if (outputName.equals(evaluation.getToolOutputDefinition())) { ToolOutput toolOutput = lamsCoreToolService.getOutputFromTool(outputName, toolSession, Index: lams_central/web/css/free.ui.jqgrid.custom.css =================================================================== diff -u -r9d7b90d94d48b27360701e1ae04497958f6eb259 -recc49cd6851b43f37ef02c2ddb85257096e2cf49 --- lams_central/web/css/free.ui.jqgrid.custom.css (.../free.ui.jqgrid.custom.css) (revision 9d7b90d94d48b27360701e1ae04497958f6eb259) +++ lams_central/web/css/free.ui.jqgrid.custom.css (.../free.ui.jqgrid.custom.css) (revision ecc49cd6851b43f37ef02c2ddb85257096e2cf49) @@ -121,4 +121,33 @@ font-size: 1rem; padding: 0; text-align: center; +} + +.ui-jqgrid.ui-jqgrid-bootstrap .ui-jqgrid-btable td > a { + text-decoration: none; +} + +.ui-jqgrid.ui-jqgrid-bootstrap .sgbutton > .fa-solid { + font-weight: 1000; + color: #3C42E0; +} + +.ui-jqgrid.ui-jqgrid-bootstrap .ui-pg-button span { + margin: 5px 5px 0 0; +} + +.ui-jqgrid.ui-jqgrid-bootstrap .ui-pg-table .ui-pg-selbox { + padding-left: 5px; +} + +.ui-jqgrid.ui-jqgrid-bootstrap .ui-search-toolbar .ui-search-table td { + border-bottom: none; +} + +.ui-jqgrid.ui-jqgrid-bootstrap .ui-search-toolbar .ui-search-table .ui-search-clear > a { + font-size: 30px; +} + +.ui-jqgrid.ui-jqgrid-bootstrap .ui-search-toolbar .ui-search-table .ui-search-input { + width: 90%; } \ No newline at end of file Index: lams_central/web/includes/javascript/authoring/authoringActivity.js =================================================================== diff -u -r7a6cf37e3bfd077514f47ff4f4f37522b72a33da -recc49cd6851b43f37ef02c2ddb85257096e2cf49 --- lams_central/web/includes/javascript/authoring/authoringActivity.js (.../authoringActivity.js) (revision 7a6cf37e3bfd077514f47ff4f4f37522b72a33da) +++ lams_central/web/includes/javascript/authoring/authoringActivity.js (.../authoringActivity.js) (revision ecc49cd6851b43f37ef02c2ddb85257096e2cf49) @@ -1,48 +1,8 @@ -/** +/** * This file contains methods for Activity definition and manipulation on canvas. */ /** - * Stores different Activity types structures. - */ -var ActivityIcons = { - 'grouping' : '', - 'bin' : '', - - 'Assessment' : '', - 'Bbb' : '', - 'Chat' : '', - 'Data Collection' : '', - 'doKumaran' : '', - 'Forum' : '', - 'Gmap' : '', - 'Share imageGallery' : '', - 'Share commonCartridge' : '', - 'MCQ' : '', - 'Question and Answer' : '', - 'Share resources' : '', - 'Leaderselection' : '', - 'Mindmap' : '', - 'Noticeboard' : '', - 'Notebook' : '', - 'Peerreview' : '', - 'Pixlr' : '', - 'Submit file' : '', - 'Scratchie' : '', - 'Scribe' : '', - 'Spreadsheet' : '', - 'Survey' : '', - 'Share taskList' : '', - 'Voting' : '', - 'Wiki' : '', - 'Kaltura' : '', - 'Zoom' : '', - 'Resources and Forum' : '', - 'Chat and Scribe' : '', - 'Forum and Scribe' : '' -}, - -/** * For colouring. See LDEV-5058 * CATEGORY_SYSTEM = 1; CATEGORY_COLLABORATION = 2; @@ -64,7 +24,7 @@ 'MCQ' : 3, 'Question and Answer' : 6, 'Share resources' : 4, - 'Leaderselection' : 6, + 'Leaderselection' : 2, 'Mindmap' : 6, 'Noticeboard' : 4, 'Notebook' : 6, @@ -77,12 +37,19 @@ 'Survey' : 6, 'Share taskList' : 4, 'Voting' : 6, + 'Whiteboard' : 2, 'Wiki' : 2, 'Kaltura' : 2, 'Zoom' : 2, 'Resources and Forum' : 5, 'Chat and Scribe' : 5, - 'Forum and Scribe' : 5 + 'Forum and Scribe' : 5, + + 'grouping' : 1, + 'gate' : 1, + 'branching': 1, + 'optional' : 1, + 'floating' : 1 }, ActivityDefs = { @@ -174,13 +141,16 @@ /** * Constructor for a Gate Activity. */ - GateActivity : function(id, uiid, x, y, title, description, readOnly, gateType, startTimeOffset, gateActivityCompletionBased, password) { + GateActivity : function(id, uiid, x, y, title, description, readOnly, gateType, startTimeOffset, + gateActivityCompletionBased, gateStopAtPrecedingActivity, password) { this.id = +id || null; this.uiid = +uiid || ++layout.ld.maxUIID; this.title = title; this.description = description; this.readOnly = readOnly; this.gateType = gateType || 'permission'; + this.gateStopAtPrecedingActivity = gateStopAtPrecedingActivity; + if (gateType == 'schedule') { var day = 24*60; this.offsetDay = Math.floor(startTimeOffset/day); @@ -193,6 +163,7 @@ if (gateType == 'password') { this.password = password; } + // mapping between tool output and gate states ("branches"), if applicable this.conditionsToBranches = []; this.transitions = { @@ -223,6 +194,10 @@ this.groupingType = groupingType || 'random'; this.groupDivide = groupDivide || 'groups'; this.groupCount = +groupCount || layout.conf.defaultGroupingGroupCount; + if (groups && groups.length > this.groupCount) { + // when opening a run sequence, groups created in monitoring can be more numerous then the original setting + this.groupCount = groups.length; + } this.learnerCount = +learnerCount || layout.conf.defaultGroupingLearnerCount; this.equalSizes = equalSizes || false; this.viewLearners = viewLearners || false; @@ -378,20 +353,35 @@ this.items.remove(); } - x = GeneralLib.snapToGrid(x); - y = GeneralLib.snapToGrid(y); + // make the icon more centred + x = GeneralLib.snapToGrid(x - 40) + 40; + y = GeneralLib.snapToGrid(y - 20) + 20; // create activity SVG elements - var shape = paper.circle(x, y, 20) - .addClass('svg-branching svg-branching-' + (this.isStart ? 'start' : 'end')); - - this.items = paper.g(shape); + var shape = paper.circle(x + 20, y + 20, 20) + .addClass('svg-branching svg-shadow svg-branching-' + (this.isStart ? 'start' : 'end')), + icon = ActivityLib.getActivityIcon(this.isStart ? 'branching' : 'branchingEnd'); + icon.select('svg').attr({ + 'x' : x + 5, + 'y' : y + 5, + 'width' : '30px', + 'height': '30px' + }); + + this.items = paper.g(shape, icon); + this.items.addClass('svg-activity svg-activity-branching'); if (this.readOnly && !isReadOnlyMode) { this.items.attr('filter', layout.conf.readOnlyFilter); } if (this.isStart) { - // uiid is needed in Monitoring - this.items.attr('uiid', this.branchingActivity.uiid); + // these are needed in monitoring + this.items.attr({ + 'uiid' : this.branchingActivity.uiid, + 'data-x' : x, + 'data-y' : y, + 'data-width' : 40, + 'data-height': 40 + }); } this.items.shape = shape; @@ -437,57 +427,61 @@ this.drawContainer(x, y, box.x2 + layout.conf.containerActivityPadding, - box.y2 + layout.conf.containerActivityPadding, - layout.colors.optionalActivity, layout.colors.optionalActivityBorder, 0.5); + box.y2 + layout.conf.containerActivityPadding); } else { this.drawContainer(x, y, x + layout.conf.containerActivityEmptyWidth, - y + layout.conf.containerActivityEmptyHeight, - layout.colors.optionalActivity, layout.colors.optionalActivityBorder, 0.5); + y + layout.conf.containerActivityEmptyHeight); } this.items.data('parentObject', this); + this.items.addClass('svg-activity svg-activity-floating svg-shadow'); }, /** * Draws a Gate activity */ gate : function(x, y) { + if (x == undefined || y == undefined) { x = this.items.getBBox().x; y = this.items.getBBox().y; } + + x = Math.round(x); + y = Math.round(y); if (this.items) { this.items.remove(); } x = GeneralLib.snapToGrid(x); + // make the icon more centred y = GeneralLib.snapToGrid(y); // create activity SVG elements - var shape = paper.path(Snap.format('M {x} {y} l-9 9 v16 l9 9 h16 l9 -9 v-16 l-9 -9 z', - { - 'x' : x + 9, - 'y' : y - }) - ) - .attr({ - 'stroke' : layout.colors.gateBorder, - 'stroke-width' : '0.5', - 'fill' : layout.colors.gate - }), - label = paper.text(x + 17, y + 20, LABELS.GATE_ACTIVITY_LABEL) - .attr(layout.defaultTextAttributes) - .attr('stroke', layout.colors.gateText); - - this.items = paper.g(shape, label); + var shape = ActivityLib.getActivityIcon('gate'); + shape.select('svg').attr({ + 'x' : x, + 'y' : y, + 'width' : '40px', + 'height': '40px' + }); + + this.items = paper.g(shape); + this.items.addClass('svg-activity svg-activity-gate svg-shadow'); if (this.readOnly && !isReadOnlyMode) { this.items.attr('filter', layout.conf.readOnlyFilter); } - // uiid is needed in Monitoring - this.items.attr('uiid', this.uiid); + // these are needed in monitoring + this.items.attr({ + 'uiid' : this.uiid, + 'data-x' : x, + 'data-y' : y, + 'data-width' : 40, + 'data-height': 40 + }); this.items.shape = shape; ActivityLib.activityHandlersInit(this); @@ -499,11 +493,11 @@ */ grouping : function(x, y) { if (x == undefined || y == undefined) { - // just redraw the activity + // if no new coordinates are given, just redraw the activity x = this.items.getBBox().x; y = this.items.getBBox().y; } - + if (this.items) { this.items.remove(); } @@ -512,28 +506,41 @@ y = GeneralLib.snapToGrid(y); // create activity SVG elements - var shape = paper.path(Snap.format('M {x} {y} h 125 v 50 h -125 z', - { - 'x' : x, - 'y' : y - }) - ) - .attr({ - 'stroke' : layout.colors.groupingBorder, - 'stroke-width' : '0.5', - 'fill' : layout.colors.grouping - }), - icon = paper.image(ActivityIcons.grouping, x + 47, y - 3, 32, 45), - label = paper.text(x + 62, y + 42, ActivityLib.shortenActivityTitle(this.title)) - .attr(layout.defaultTextAttributes); + var curve = layout.activity.borderCurve, + width = layout.activity.width, + height = layout.activity.height, + shapePath = ' M ' + (x + curve) + ' ' + y + ' h ' + (width - 2 * curve) + ' q ' + curve + ' 0 ' + curve + ' ' + curve + + ' v ' + (height - 2 * curve) + ' q 0 ' + curve + ' ' + -curve + ' ' + curve + + ' h ' + (-width + 2 * curve) + ' q ' + -curve + ' 0 ' + -curve + ' ' + -curve + + ' v ' + (-height + 2 * curve) + ' q 0 ' + -curve + ' ' + curve + ' ' + -curve, + shape = paper.path(shapePath) + .addClass('svg-tool-activity-background svg-shadow'), + shapeBorder = paper.path(shapePath) + .addClass('svg-tool-activity-border'), + // check for icon in the library + icon = ActivityLib.getActivityIcon('grouping'), + label = ActivityLib.getActivityTitle(this.title, x, y); + + icon.select('svg').attr({ + 'x' : x + 20, + 'y' : y + 15, + 'width' : '50px', + 'height': '50px' + }); - this.items = paper.g(shape, icon, label); - // uiid is needed in Monitoring + this.items = paper.g(shape, shapeBorder, label, icon); + this.items.attr({ + 'uiid' : this.uiid, + 'data-x' : x, + 'data-y' : y, + 'data-width' : width, + 'data-height': height + }); + + this.items.addClass('svg-activity svg-activity-grouping'); if (this.readOnly && !isReadOnlyMode) { this.items.attr('filter', layout.conf.readOnlyFilter); } - // uiid is needed in Monitoring - this.items.attr('uiid', this.uiid); this.items.shape = shape; ActivityLib.activityHandlersInit(this); @@ -558,6 +565,9 @@ this.childActivities = childActivities; } + var width = null, + height = null; + if (this.childActivities && this.childActivities.length > 0) { // draw one by one, vertically var activityY = y + layout.conf.containerActivityPadding + 10, @@ -578,14 +588,16 @@ // area containing all drawn child activities box = allElements.getBBox(); - this.drawContainer(x, y, - box.x2 + layout.conf.containerActivityPadding, - box.y2 + layout.conf.containerActivityPadding, + width = box.x2 + layout.conf.containerActivityPadding - x; + height = box.y2 + layout.conf.containerActivityPadding - y; + + this.drawContainer(x, y, x + width, y + height, layout.colors.optionalActivity, layout.colors.optionalActivityBorder, 0.5); } else { - this.drawContainer(x, y, - x + layout.conf.containerActivityEmptyWidth, - y + layout.conf.containerActivityEmptyHeight, + width = layout.conf.containerActivityEmptyWidth; + height = layout.conf.containerActivityEmptyHeight; + + this.drawContainer(x, y, x + width, y + height, layout.colors.optionalActivity, layout.colors.optionalActivityBorder, 0.5); } @@ -595,6 +607,15 @@ } this.items.data('parentObject', this); + this.items.addClass('svg-activity svg-activity-optional svg-shadow'); + // these are needed in monitoring + this.items.attr({ + 'uiid' : this.uiid, + 'data-x' : x, + 'data-y' : y, + 'data-width' : width, + 'data-height': height + }); }, @@ -612,7 +633,10 @@ x = GeneralLib.snapToGrid(x); y = GeneralLib.snapToGrid(y); - + + var width = null, + height = null; + if (this.childActivities && this.childActivities.length > 0) { // draw one by one, vertically var activityY = y + layout.conf.containerActivityPadding + 10, @@ -628,14 +652,16 @@ // area containing all drawn child activities var box = allElements.getBBox(); - this.drawContainer(x, y, - box.x2 + layout.conf.containerActivityPadding, - box.y2 + layout.conf.containerActivityPadding, + width = box.x2 + layout.conf.containerActivityPadding - x; + height = box.y2 + layout.conf.containerActivityPadding - y; + + this.drawContainer(x, y, x + width, y + height, layout.colors.optionalActivity, layout.colors.optionalActivityBorder, 0.5); } else { - this.drawContainer(x, y, - x + layout.conf.containerActivityEmptyWidth, - y + layout.conf.containerActivityEmptyHeight, + width = layout.conf.containerActivityEmptyWidth; + height = layout.conf.containerActivityEmptyHeight; + + this.drawContainer(x, y, x + width, y + height, layout.colors.optionalActivity, layout.colors.optionalActivityBorder, 0.5); } @@ -649,6 +675,15 @@ } this.items.data('parentObject', this); + this.items.addClass('svg-activity svg-activity-parallel svg-shadow'); + // these are needed in monitoring + this.items.attr({ + 'uiid' : this.uiid, + 'data-x' : x, + 'data-y' : y, + 'data-width' : width, + 'data-height': height + }); }, @@ -675,40 +710,50 @@ var curve = layout.activity.borderCurve, width = layout.activity.width, height = layout.activity.height, - bannerWidth = layout.activity.bannerWidth, bannerPath = 'M ' + (x + curve) + ' ' + (y + height) + ' q ' + -curve + ' 0 ' + -curve + ' ' + -curve + ' v ' + (-height + 2 * curve) + ' q 0 ' + -curve + ' ' + curve + ' ' + -curve, + // by default the wide banner is displayed, + // but when there are learners in monitoring, the narrow one is shown instead + bannerWidePath = bannerPath + ' h ' + layout.activity.bannerWideWidth + ' v ' + height + ' z', + bannerNarrowPath = bannerPath + ' h ' + layout.activity.bannerNarrowWidth + ' v ' + height + ' z', + bannerWide = paper.path(bannerWidePath) + .addClass('svg-tool-banner-wide svg-tool-activity-category-' + layout.toolMetadata[this.learningLibraryID].activityCategoryID), + bannerNarrow = paper.path(bannerNarrowPath) + .addClass('svg-tool-banner-narrow svg-tool-activity-category-' + layout.toolMetadata[this.learningLibraryID].activityCategoryID), shapePath = bannerPath + ' h ' + (width - 2 * curve) + ' q ' + curve + ' 0 ' + curve + ' ' + curve + ' v ' + (height - 2 * curve) + ' q 0 ' + curve + ' ' + -curve + ' ' + curve + ' z', shape = paper.path(shapePath) - .addClass('svg-tool-activity-background'), + .addClass('svg-tool-activity-background ' + (this.grouping ? '' : 'svg-shadow')), shapeBorder = paper.path(shapePath) .addClass('svg-tool-activity-border' + (this.requireGrouping ? '-require-grouping' : '')), - // check for icon in the library - imageData = null, // ActivityIcons[this.learningLibraryID], - icon = imageData ? paper.image(imageData, x + 5, y + 3, 20, 20) : null, - label = paper.text(x + 30, y + 18, ActivityLib.shortenActivityTitle(this.title)) - .attr(layout.defaultTextAttributes) - .attr({ - 'id' : 'toolActivityTitle', - 'fill' : layout.colors.activityText, - 'text-anchor' : 'start' - }); + label = ActivityLib.getActivityTitle(this.title, x, y), + icon = ActivityLib.getActivityIcon(this.learningLibraryID); + + $(bannerNarrow.node).hide(); + this.items = paper.g(shape, bannerWide, bannerNarrow, shapeBorder, label); - bannerPath += ' h ' + bannerWidth + ' v ' + height + ' z'; - var banner = paper.path(bannerPath) - .addClass('svg-tool-activity-category-' + layout.toolMetadata[this.learningLibraryID].activityCategoryID); - this.items = paper.g(shape, banner, shapeBorder, label); - if (icon) { - this.items.add(icon); + icon.select('svg').attr({ + 'x' : x + 15, + 'y' : y + 20, + 'width' : '40px', + 'height': '40px' + }); + this.items.add(icon); } + if (this.readOnly && !isReadOnlyMode) { this.items.attr('filter', layout.conf.readOnlyFilter); } - // uiid is needed in Monitoring - this.items.attr('uiid', this.uiid); - this.items.attr('id' , 'toolActivity'); + // these are needed in monitoring + this.items.attr({ + 'uiid' : this.uiid, + 'data-x' : x, + 'data-y' : y, + 'data-width' : width, + 'data-height': height + }); + this.items.addClass('svg-activity svg-activity-tool'); this.items.shape = shape; if (this.grouping) { @@ -728,7 +773,8 @@ } this.items = paper.g(); - var isBranching = this.fromActivity instanceof ActivityDefs.BranchingEdgeActivity || this.toActivity instanceof ActivityDefs.BranchingEdgeActivity, + var isBranching = (this.fromActivity instanceof ActivityDefs.BranchingEdgeActivity && this.fromActivity.isStart) || + (this.toActivity instanceof ActivityDefs.BranchingEdgeActivity && !this.toActivity.isStart), points = ActivityLib.findTransitionPoints(this.fromActivity, this.toActivity), curve = layout.transition.curve, straightLineThreshold = 2 * curve + 2; @@ -937,7 +983,7 @@ activityBox.width, activityBox.height, 5, 5) - .addClass('svg-tool-activity-border'); + .addClass('svg-tool-activity-border svg-tool-activity-border-grouped svg-shadow'); activity.items.prepend(activity.items.groupingEffect); @@ -1244,7 +1290,16 @@ GeneralLib.setModified(true); return transition; }, - + + adjustTransitionPoint : function(bottomLimit, topLimit, target) { + bottomLimit = Math.round(bottomLimit); + topLimit = Math.round(topLimit); + target = Math.round(target); + // find a good point inside the grid, then make sure it is within bounds + return Math.max(bottomLimit + layout.transition.adjustStep, Math.min(topLimit - layout.transition.adjustStep, + Math.floor(target / layout.transition.adjustStep) * layout.transition.adjustStep)); + }, + /** * It is run from authoringConfirm.jsp * It closes the dialog with activity authoring @@ -1368,13 +1423,6 @@ return nestedBranching; }, - adjustTransitionPoint : function(bottomLimit, topLimit, target) { - // find a good point inside the grid, then make sure it is within bounds - return Math.max(bottomLimit + layout.transition.adjustStep, Math.min(topLimit - layout.transition.adjustStep, - Math.floor(target / layout.transition.adjustStep) * layout.transition.adjustStep + layout.snapToGrid.offset)); - }, - - /** * Calculates start, middle and end points of a line between two activities. */ @@ -1393,8 +1441,7 @@ if (direction === 'vertical') { if (fromActivityBox.cy < toActivityBox.cy) { points = { - 'startX' : ActivityLib.adjustTransitionPoint(fromActivityBox.x, fromActivityBox.x2, toActivityBox.x + toActivityBox.width / 2) - - (fromActivity.items.groupingEffect ? 0.5 * layout.conf.groupingEffectPadding : 0), + 'startX' : ActivityLib.adjustTransitionPoint(fromActivityBox.x, fromActivityBox.x2, toActivityBox.x + toActivityBox.width / 2), 'startY' : fromActivityBox.y2 + layout.transition.dotRadius, 'endY' : toActivityBox.y, 'direction' : 'down', @@ -1409,15 +1456,13 @@ 'direction' : 'up', 'arrowAngle': 0 }; - points.endX = ActivityLib.adjustTransitionPoint(toActivityBox.x, toActivityBox.x2, points.startX) - - (toActivity.items.groupingEffect ? 0.5 * layout.conf.groupingEffectPadding : 0); + points.endX = ActivityLib.adjustTransitionPoint(toActivityBox.x, toActivityBox.x2, points.startX); } } else { if (fromActivityBox.cx < toActivityBox.cx) { points = { 'startX' : fromActivityBox.x2 + layout.transition.dotRadius, - 'startY' : ActivityLib.adjustTransitionPoint(fromActivityBox.y, fromActivityBox.y2, toActivityBox.y + toActivityBox.height / 2) - - (fromActivity.items.groupingEffect ? 0.5 * layout.conf.groupingEffectPadding : 0), + 'startY' : ActivityLib.adjustTransitionPoint(fromActivityBox.y, fromActivityBox.y2, toActivityBox.y + toActivityBox.height / 2), 'endX' : toActivityBox.x, 'direction' : 'right', 'arrowAngle': 90 @@ -1432,8 +1477,7 @@ 'direction' : 'left', 'arrowAngle': 270 }; - points.endY = ActivityLib.adjustTransitionPoint(toActivityBox.y, toActivityBox.y2, points.startY) - - (toActivity.items.groupingEffect ? 0.5 * layout.conf.groupingEffectPadding : 0); + points.endY = ActivityLib.adjustTransitionPoint(toActivityBox.y, toActivityBox.y2, points.startY); } } @@ -1446,18 +1490,49 @@ return points; }, + getActivityIcon : function(activityName) { + // check for icon SVG cache in the library + var iconData = layout.toolMetadata[activityName].iconData; + if (!iconData) { + if (!layout.toolMetadata[activityName].iconPath) { + return; + } + // if SVG is not cached, get it synchronously + $.ajax({ + url : LAMS_URL + layout.toolMetadata[activityName].iconPath, + async : false, + dataType : 'text', + success : function(response) { + iconData = response; + layout.toolMetadata[activityName].iconData = iconData; + } + }); + } + + if (iconData) { + // build a SVG fragment + var fragment = Snap.parse(iconData); + return Snap(fragment.node); + } + }, + /** * Finds activity/region this shape is bound with. */ getParentObject : function(item) { - var parentObject = item.data('parentObject'); - if (!parentObject) { - var parentNode = item.parent(); - if (parentNode.type == 'g') { - parentObject = parentNode.data('parentObject'); + var parentObject = null; + + while (!parentObject) { + parentObject = item.data('parentObject'); + if (!parentObject) { + item = item.parent(); + if (!item || item.attr('id') == 'canvas') { + break; + } } } + return parentObject; }, @@ -1639,23 +1714,21 @@ * Deletes the given activity. */ removeActivity : function(activity, forceRemove) { - var coreActivity = activity.branchingActivity || this; + var coreActivity = activity.branchingActivity || activity; if (!forceRemove && activity instanceof ActivityDefs.BranchingEdgeActivity){ // user removes one of the branching edges, so remove the whole activity - if (confirm(LABELS.REMOVE_ACTIVITY_CONFIRM)){ - var otherEdge = activity.isStart ? coreActivity.end - : coreActivity.start; - ActivityLib.removeActivity(otherEdge, true); - } else { + if (!confirm(LABELS.REMOVE_ACTIVITY_CONFIRM)){ return; } + var otherEdge = activity.isStart ? coreActivity.end + : coreActivity.start; + ActivityLib.removeActivity(otherEdge, true); } if (activity instanceof ActivityDefs.FloatingActivity) { layout.floatingActivity = null; // re-enable the button, as the only possible Floating Activity is gone now - $('#floatingActivityButton').attr('disabled', null) - .css('opacity', 1); + $('.template[learningLibraryId="floating"]').slideDown(); } else { // remove the transitions // need to use slice() to copy the array as it gets modified in removeTransition() @@ -1674,11 +1747,14 @@ // find references of this activity as grouping or input $.each(layout.activities, function(){ - if (activity == coreActivity.grouping) { - coreActivity.grouping = null; + var candidate = this.branchingActivity || this; + if (candidate.grouping == coreActivity) { + candidate.grouping = null; + this.propertiesContent = null; this.draw(); - } else if (activity == coreActivity.input) { - coreActivity.input = null; + } else if (candidate.input == coreActivity) { + candidate.input = null; + this.propertiesContent = null; } }); } @@ -1700,7 +1776,23 @@ GeneralLib.setModified(true); }, + /** + * Deletes an item (activity, annotation etc.) as a result of user pressing a button on properties box + */ + removeItemWithButton : function(item) { + if ((item instanceof ActivityDefs.BranchingEdgeActivity) || confirm(LABELS.REMOVE_BUTTON_CONFIRM)) { + ActivityLib.removeSelectEffect(item); + if (item instanceof DecorationDefs.Label) { + DecorationLib.removeLabel(item); + } else if (item instanceof DecorationDefs.Region) { + DecorationLib.removeRegion(item); + } else { + ActivityLib.removeActivity(item); + } + } + }, + /** * Deselects an activity/transition/annotation */ @@ -1717,7 +1809,7 @@ // different effects for different types of objects if (object instanceof DecorationDefs.Region) { object.items.shape.attr({ - 'stroke' : layout.colors.activityBorder, + 'stroke' : 'black', 'stroke-dasharray' : null }); object.items.fitButton.attr('display','none'); @@ -1784,7 +1876,7 @@ /** * Removes the given transition. */ - removeTransition : function(transition, redraw) { + removeTransition : function(transition) { // find the transition and remove it var transitions = transition.fromActivity.transitions.from; transitions.splice(transitions.indexOf(transition), 1); @@ -1806,17 +1898,7 @@ GeneralLib.setModified(true); }, - /** - * Reduce length of activity's title so it fits in its SVG shape. - */ - shortenActivityTitle : function(title) { - if (title.length > 18) { - title = title.substring(0, 17) + '...'; - } - return title; - }, - /** * Crawles through branches setting their lengths and finding the longest one. */ @@ -1848,5 +1930,22 @@ }); branchingActivity.longestBranchLength = longestBranchLength; + }, + + getActivityTitle : function(title, x, y) { + + if (title.length > 35) { + title = title.substring(0, 35) + '...'; + } + var label = $('
').addClass('svg-activity-title-label svg-tool-activity-title-box') + .attr('xmlns', 'http://www.w3.org/1999/xhtml') + .text(title), + wrapper = $('').append(label).attr({ + 'x' : x + 75, + 'y' : y, + 'width' : layout.activity.width - 75, + 'height': layout.activity.height + }); + return Snap.parse(wrapper[0].outerHTML); } -}; +}; Index: lams_central/web/includes/javascript/authoring/authoringGeneral.js =================================================================== diff -u -rdcb81d1b2bd66f65b38ae9c167bad8d743e03ab0 -recc49cd6851b43f37ef02c2ddb85257096e2cf49 --- lams_central/web/includes/javascript/authoring/authoringGeneral.js (.../authoringGeneral.js) (revision dcb81d1b2bd66f65b38ae9c167bad8d743e03ab0) +++ lams_central/web/includes/javascript/authoring/authoringGeneral.js (.../authoringGeneral.js) (revision ecc49cd6851b43f37ef02c2ddb85257096e2cf49) @@ -8,6 +8,23 @@ */ $(document).ready(function() { GeneralInitLib.initAll(); + + // we display authoring in separate window if LAMS acts as a LTI 1.3 Tool + if (isLtiContentSelection || initRelaunchMonitorLessonID) { + window.onbeforeunload = function(e) { + if (window.opener && typeof window.opener.refreshSeqList === 'function') { + window.opener.refreshSeqList(); + } + + if (!GeneralLib.canClose()) { + e.preventDefault(); + e.returnValue = LABELS.NAVIGATE_AWAY_CONFIRM; + return LABELS.NAVIGATE_AWAY_CONFIRM; + } + + delete e['returnValue']; + } + } }); @@ -32,22 +49,22 @@ 'addBranchingStart' : null, // list of all dialogs, so they can be easily closed all at once 'dialogs' : [], - // icons for special activities + // for storing icons and other activity metadata 'toolMetadata': { - 'gate' : { - 'iconPath' : '../images/stop.gif' + 'branchingEnd' : { + 'iconPath' : 'images/svg/branchingEnd.svg' }, - 'grouping' : { - 'iconPath' : '../images/grouping.png' + 'bin' : { + 'iconPath' : 'images/svg/authoringBin.svg' } }, // graphics constants 'conf' : { - 'arrangeHorizontalSpace' : 200, - 'arrangeVerticalSpace' : 100, - 'arrangeHorizontalPadding' : 35, - 'arrangeVerticalPadding' : 50, + 'arrangeHorizontalSpace' : 240, + 'arrangeVerticalSpace' : 120, + 'arrangeHorizontalPadding' : 40, + 'arrangeVerticalPadding' : 40, 'dragStartThreshold' : 300, @@ -58,8 +75,8 @@ 'defaultGroupingGroupCount' : 2, 'defaultGroupingLearnerCount' : 1, - 'containerActivityEmptyWidth' : 50, - 'containerActivityEmptyHeight' : 70, + 'containerActivityEmptyWidth' : 200, + 'containerActivityEmptyHeight' : 100, 'containerActivityPadding' : 20, 'containerActivityChildrenPadding' : 10, 'regionEmptyWidth' : 20, @@ -80,15 +97,16 @@ 'snapToGrid' : { // snapping grid step when dragging an activity 'step' : 40, - 'padding' : 20, - 'offset' : 10 + // distance from canvas border so activities are not on edge + 'padding' : 40 }, 'activity' : { 'width' : 200, 'height' : 80, 'borderCurve' : 5, - 'bannerWidth' : 10 + 'bannerNarrowWidth' : 10, + 'bannerWideWidth' : 60 }, 'transition' : { @@ -99,8 +117,6 @@ }, 'colors' : { - 'activityBorder' : 'black', - 'activityText' : 'black', // default region colour 'annotation' : '#CCFF99', // region colours to choose from @@ -110,14 +126,6 @@ // when mouse hovers over rubbish bin 'binSelect' : 'red', - - 'gate' : 'red', - 'gateBorder' : '#801515', - 'gateText' : 'white', - 'grouping' : '#caddfb', - 'groupingBorder' : '#00007f', - 'optionalActivity' : '#caddfb', - 'optionalActivityBorder' : '#00007f', // dashed border around a selected activity 'selectEffect' : 'black', // highlight TBL activities which should be grouped @@ -175,9 +183,11 @@ * Draw boxes with Tool Activity templates in the panel on the left. */ initTemplates : function(){ + var templateContainerCell = $('#templateContainerCell'); + // store some template data in JS structures - $('.template').each(function(){ - var learningLibraryID = +$(this).attr('learningLibraryId'), + $('.template', templateContainerCell).each(function(){ + var learningLibraryID = $(this).attr('learningLibraryId'), learningLibraryTitle = $(this).attr('learningLibraryTitle'), activityCategoryID = ActivityCategories[learningLibraryTitle], parallelChildActivityDefs = null; @@ -197,10 +207,6 @@ }); } - // assign icons' data uris to their learning library IDs instead of labels - ActivityIcons[learningLibraryID] = ActivityIcons[learningLibraryTitle]; - delete ActivityIcons[learningLibraryTitle]; - $('').attr('src', ActivityIcons[learningLibraryID]).appendTo(".img-"+learningLibraryID); // register tool properties so they are later easily accessible layout.toolMetadata[learningLibraryID] = { 'iconPath' : $(this).attr('iconPath'), @@ -209,75 +215,45 @@ 'activityCategoryID' : activityCategoryID, 'parallelChildActivityDefs' : parallelChildActivityDefs }; - }); if (!isReadOnlyMode){ // store the initial window height now as on iPad the iframe grows when templates are show, // reporting incorrect window height to the first resizePaper() run layout.initWindowHeight = $(window).height(); - // create list of learning libraries for each group - var templateContainerCell = $('#templateContainerCell'), - learningLibraryGroupSelect = $('select', templateContainerCell), - allGroup = $('option', learningLibraryGroupSelect), - allTemplates = $('#templateContainerCell .templateContainer').show(); - - learningLibraryGroupSelect.change(function(){ - $('.templateContainer').hide(); - // show DIV with the selected learning libraries group - $('option:selected', this).data('templates').show(); - }); - allGroup.data('templates', allTemplates); - $.each(learningLibraryGroups, function(){ - var learningLibraries = this.learningLibraries; - if (!learningLibraries) { - return true; - } - - var templates = allTemplates.clone().appendTo(templateContainerCell); - // cloned everything, now remove ones that are not in the list - $('.template', templates).each(function(){ - var learningLibraryId = $(this).attr('learningLibraryId'), - found = false; - $.each(learningLibraries, function(){ - if (learningLibraryId == this) { - found = true; - return false; - } - }); + $('.template', templateContainerCell).each(function(){ + let learningLibraryID = $(this).attr('learningLibraryId'), + isFlowActivity = isNaN(+learningLibraryID), + activityCategoryID = layout.toolMetadata[learningLibraryID].activityCategoryID; - if (!found) { - $(this).remove(); - } - }); + $('#collapse-tool-category-' + activityCategoryID, templateContainerCell).append(this); - $('