Merge pull request #9 from misfir3/master
Initial cut-over of backbone port
This commit is contained in:
		
							
								
								
									
										20
									
								
								README.txt
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								README.txt
									
									
									
									
									
								
							| @ -1,3 +1,23 @@ | ||||
| ********** | ||||
| * June 9 2015 | ||||
| * #####  WebGoat Container Migration from AngularJS to Backbone ##### | ||||
| * Why: I believe AngularJS is a little heavy (and acutally a little too secure, imagine that) for the purposes of WebGoat | ||||
| * When: Now | ||||
| * What: Porting current functionality in the 6.0.1 release into the webgoat-container build | ||||
| * How: If you'd like to help.  Fork this repository and contact me (jason.white@owasp.org) as to the current priorities/needs. Once you fork it, | ||||
| * After forking and cloning this repo, you should also fork/clone the lessons repository (https://github.com/WebGoat/WebGoat-Lessons). More on that in a second | ||||
| * in the core/container WebGoat (this) project | ||||
| * $ mvn clean install | ||||
| * Then either package and put the package in a local tomcat OR ... | ||||
| * $ mvn tomcat:run | ||||
| * Then you'll need to switch to the lessons directory and | ||||
| * $ mvn clean packa | ||||
| * Then copy some of the lesson jars into your $TOMCAT_HOME/webapps/{webgoat-container}/plugin_lessons | ||||
| * Now you should have something to work against for the UI work | ||||
| ********** | ||||
|  | ||||
| ##### Original WebGoat ReadMe follows ##### | ||||
|  | ||||
| ********** | ||||
| **********          WebGoat 6.0 | ||||
| **********          August 23, 2014 | ||||
|  | ||||
| @ -551,7 +551,7 @@ public abstract class AbstractLesson extends Screen implements Comparable<Object | ||||
|      * return "*.do"-styled paths.</p> | ||||
|      */ | ||||
|     protected String getPath() { | ||||
|         return "attack"; | ||||
|         return "#attack"; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @ -563,13 +563,9 @@ public abstract class AbstractLesson extends Screen implements Comparable<Object | ||||
|         StringBuffer link = new StringBuffer(); | ||||
|  | ||||
|         // mvc update: | ||||
|         link.append(getPath()).append("?"); | ||||
|         link.append(WebSession.SCREEN); | ||||
|         link.append("="); | ||||
|         link.append(getPath()).append("/"); | ||||
|         link.append(getScreenId()); | ||||
|         link.append("&"); | ||||
|         link.append(WebSession.MENU); | ||||
|         link.append("="); | ||||
|         link.append("/"); | ||||
|         link.append(getCategory().getRanking()); | ||||
|         return link.toString(); | ||||
|     } | ||||
|  | ||||
| @ -198,9 +198,9 @@ public abstract class Screen { | ||||
|  | ||||
|     // hook all the links | ||||
|     public String getContent() { | ||||
|         String makeAllAjax = "<script>goat.utils.makeFormsAjax();goat.utils.ajaxifyAttackHref();</script>"; | ||||
|         //String makeAllAjax = "<script>goat.utils.makeFormsAjax();goat.utils.ajaxifyAttackHref();</script>"; | ||||
|         // need to do this here as some of the lessons render forms after submission of an ajax form | ||||
|         return (content == null) ? "" : content.toString() + makeAllAjax; | ||||
|         return (content == null) ? "" : content.toString();// + makeAllAjax; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|  | ||||
| @ -34,7 +34,7 @@ | ||||
|  | ||||
|             </div><!--toggle navigation end--> | ||||
|             <div class="lessonTitle" > | ||||
|                 <h1 id="lessonTitle">Please login</h1> | ||||
|                  | ||||
|             </div><!--lesson title end--> | ||||
|  | ||||
|         </header> | ||||
|  | ||||
| @ -27,12 +27,7 @@ | ||||
|         <!--  end of CSS --> | ||||
|  | ||||
|         <!-- JS --> | ||||
|         <script src="js/jquery/jquery-1.10.2.min.js"></script> | ||||
|         <script src="js/angular/angular.min.js"></script> | ||||
|         <!-- angular modules --> | ||||
|         <script src="js/angular/angular-animate.min.js"></script> | ||||
|         <script src="js/angular/ui-bootstrap-tpls-0.11.0.min.js"></script> | ||||
|         <!-- Feature detection --> | ||||
|          | ||||
|         <script src="js/modernizr-2.6.2.min.js"></script> | ||||
|         <!-- HTML5 shim and Respond.js IE8 support of HTML5 elements and media queries --> | ||||
|         <!--[if lt IE 9]> | ||||
| @ -40,30 +35,14 @@ | ||||
|         <script src="js/respond.min.js"></script> | ||||
|         <![endif]--> | ||||
|  | ||||
|         <!--Global JS--> | ||||
|  | ||||
|         <script src="js/jquery_form/jquery.form.js"></script>   | ||||
|         <script src="plugins/bootstrap/js/bootstrap.min.js"></script> | ||||
|  | ||||
|         <script src="js/application.js"></script> | ||||
|         <script type="text/javascript"> | ||||
|             var goat = angular.module("goatApp", ['ngAnimate', 'ui.bootstrap']); | ||||
|         </script> | ||||
|         <script type="text/javascript" src="js/goatConstants.js"></script> | ||||
|         <script type="text/javascript" src="js/goatUtil.js"></script> | ||||
|         <script type="text/javascript" src="js/goatData.js"></script> | ||||
|         <script type="text/javascript" src="js/goatLesson.js"></script> | ||||
|         <script type="text/javascript" src="js/goatControllers.js"></script> | ||||
|         <!-- end of JS --> | ||||
|  | ||||
|  | ||||
|         <!-- Require.js used to load js asynchronously --> | ||||
|         <script src="js/libs/require.min.js" data-main="js/main.js"></script> | ||||
|  | ||||
|         <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1" /> | ||||
|         <title>WebGoat</title> | ||||
|     </head> | ||||
|  | ||||
|     <body class="animated fadeIn" ng-app="goatApp"> | ||||
|         <section id="container" ng-controller="goatLesson"> | ||||
|     <body> | ||||
|         <section id="container"> | ||||
|             <header id="header"> | ||||
|                 <!--logo start--> | ||||
|                 <div class="brand"> | ||||
| @ -75,12 +54,12 @@ | ||||
|                         <i class="fa fa-bars"></i> | ||||
|                     </button> | ||||
|                 </div><!--toggle navigation end--> | ||||
|                 <div class="lessonTitle" > | ||||
|                     <h1 id="lessonTitle"></h1> | ||||
|                 <div id="lesson-title-wrapper" > | ||||
|                      | ||||
|                 </div><!--lesson title end--> | ||||
|                 <div class="user-nav pull-right" style="margin-right: 75px;"> | ||||
|                     <div class="dropdown" style="display:inline"> | ||||
|                         <button type="button" class="btn btn-default dropdown-toggle" id="dropdownMenu1" ng-disabled="disabled"> | ||||
|                         <button type="button" class="btn btn-default dropdown-toggle" id="dropdownMenu1" > | ||||
|                             <i class="fa fa-user"></i> <span class="caret"></span> | ||||
|                         </button>                    | ||||
|                         <ul class="dropdown-menu dropdown-menu-left" role="menu" aria-labelledby="dropdownMenu1"> | ||||
| @ -94,7 +73,7 @@ | ||||
|  | ||||
|                         </ul> | ||||
|                     </div> | ||||
|                     <button type="button" class="btn btn-default right_nav_button" ng-click="showAbout()" data-toggle="tooltip" title="About WebGoat"> | ||||
|                     <button type="button" class="btn btn-default right_nav_button" data-toggle="tooltip" title="About WebGoat"> | ||||
|                         <i class="fa fa-info"></i> | ||||
|                     </button> | ||||
|                     <a href="mailto:${contactEmail}?Subject=Webgoat%20feedback" target="_top"> | ||||
| @ -107,47 +86,32 @@ | ||||
|                 </div> | ||||
|             </header> | ||||
|  | ||||
|             <!--sidebar left start--> | ||||
|             <aside class="sidebar" > | ||||
|                 <div id="leftside-navigation" ng-controller="goatMenu" class="nano"> | ||||
|                     <ul class="nano-content"> | ||||
|                         <li class="sub-menu" ng-repeat="item in menuTopics"> | ||||
|                             <a ng-click="accordionMenu(item.id)" href=""><i class="fa {{item.class}}"></i><span>{{item.name}}</span></a><!-- expanded = !expanded--> | ||||
|                             <ul class="slideDown lessonsAndStages {{item.displayClass}}" id="{{item.id}}" isOpen=0> | ||||
|                                 <li ng-repeat="lesson in item.children" class="{{lesson.selectedClass}}"> | ||||
|                                     <a ng-click="renderLesson(lesson.id, lesson.link, {showSource: lesson.showSource, showHints: lesson.showHints})" id="{{lesson.id}}" class="{{lesson.selectedClass}}" title="link to {{lesson.name}}" href="">{{lesson.name}}</a><span class="{{lesson.completeClass}}"></span> | ||||
|                                     <span ng-repeat="stage in lesson.children"> | ||||
|                                         <a ng-click="renderLesson(stage.id, stage.link, {showSource: stage.showSource, showHints: stage.showHints})" class="selectedClass" id="{{stage.id}}"  title="link to {{stage.name}}" href="">{{stage.name}}</a><span class="{{stage.completeClass}}"></span> | ||||
|                                     </span> | ||||
|                                 </li> | ||||
|                             </ul> | ||||
|                         </li> | ||||
|                     </ul>  | ||||
|                 </div> | ||||
|  | ||||
|                 <div id="menu-container"></div> | ||||
|             </aside> | ||||
|             <!--sidebar left end--> | ||||
|  | ||||
|             <!--main content start--> | ||||
|             <section class="main-content-wrapper"> | ||||
|                 <section id="main-content" > <!--ng-controller="goatLesson"--> | ||||
|                     <div class="row"> | ||||
|                         <div class="col-md-8"> | ||||
|                             <div class="col-md-12" align="left"> | ||||
|                                 <div class="panel"> | ||||
|                                     <div class="panel-body"> | ||||
|                                         <button type="button" id="showSourceBtn" ng-show="showSource" class="btn btn-primary btn-xs" ng-click="showLessonSource()">Java [Source]</button> | ||||
|                                         <button type="button" id="showSolutionBtn" class="btn btn-primary btn-xs" ng-click="showLessonSolution()">Solution</button> | ||||
|                                         <button type="button" id="showPlanBtn" class="btn btn-primary btn-xs" ng-click="showLessonPlan()">Lesson Plan</button> | ||||
|                                         <button type="button" id="showHintsBtn" ng-show="showHints" class="btn btn-primary btn-xs"  ng-click="viewHints()">Hints</button> | ||||
|                                         <button type="button" id="restartLessonBtn"  class="btn btn-xs"  ng-click="restartLesson()">Restart Lesson</button> | ||||
|                                 <div class="panel" id="help-controls"> | ||||
|                                     <!-- <div id="help-buttons" class="panel-body"> --> | ||||
|                                         <!-- <button type="button" id="showSourceBtn" ng-show="showSource" class="btn btn-primary btn-xs" onclick="showLessonSource()">Java [Source]</button> | ||||
|                                         <button type="button" id="showSolutionBtn" class="btn btn-primary btn-xs" onclick="showLessonSolution()">Solution</button> | ||||
|                                         <button type="button" id="showPlanBtn" class="btn btn-primary btn-xs" onclick="showLessonPlan()">Lesson Plan</button> | ||||
|                                         <button type="button" id="showHintsBtn" ng-show="showHints" class="btn btn-primary btn-xs"  onclick="viewHints()">Hints</button> | ||||
|                                         <button type="button" id="restartLessonBtn"  class="btn btn-xs"  onclick="restartLesson()">Restart Lesson</button> --> | ||||
|                                     <!-- </div> --> | ||||
|                                 </div> | ||||
|                                 </div> | ||||
|                                 <div class="lessonHelp" id="lesson_hint_row"> | ||||
|                                 <div class="lessonHelp" id="lesson-hint-row"> | ||||
|                                     <h4>Hints</h4> | ||||
|                                     <div class="panel" > | ||||
|                                         <div class="panel-body" id="lesson_hint"> | ||||
|                                             <span class="glyphicon-class glyphicon glyphicon-circle-arrow-left" id="showPrevHintBtn" ng-click="viewPrevHint()"></span> | ||||
|                                             <span class="glyphicon-class glyphicon glyphicon-circle-arrow-right" id="showNextHintBtn" ng-click="viewNextHint()"></span> | ||||
|                                             <span class="glyphicon-class glyphicon glyphicon-circle-arrow-left" id="showPrevHintBtn" onclick="viewPrevHint()"></span> | ||||
|                                             <span class="glyphicon-class glyphicon glyphicon-circle-arrow-right" id="showNextHintBtn" onclick="viewNextHint()"></span> | ||||
|                                             <br/> | ||||
|                                             <span ng-show="showHints" bind-html-unsafe="curHint"></span> | ||||
|                                             <!--<span id="curHintContainer"></span>--> | ||||
| @ -155,12 +119,8 @@ | ||||
|                                     </div> | ||||
|                                 </div> | ||||
|                             </div> | ||||
|  | ||||
|                             <div class="col-md-12"> | ||||
|                                 <div class="panel" > | ||||
|                                     <div class="panel-body" id="lesson_content">     | ||||
|                                         <b>This should default to the "How to Work with Webgoat" lesson</b> | ||||
|                                     </div> | ||||
|                             <div class="col-md-12" align="left"> | ||||
|                                 <div id="lessonContentWrapper" class="panel"> | ||||
|  | ||||
|                                 </div> | ||||
|                             </div> | ||||
| @ -176,10 +136,10 @@ | ||||
|                                         <div id="cookiesAndParamsView"> | ||||
|                                             <div class="cookiesView"> | ||||
|                                                 <h4>Cookies</h4> | ||||
|                                                 <div class="cookieContainer" ng-repeat="cookie in cookies"> | ||||
|                                                 <!-- <div class="cookieContainer" ng-repeat="cookie in cookies"> | ||||
|                                                     <table class="cookieTable table-striped table-nonfluid" > | ||||
|                                                         <thead> | ||||
|                                                             <tr><th class="col-sm-1"></th><th class="col-sm-1"></th></tr> <!-- Field / Value --> | ||||
|                                                             <tr><th class="col-sm-1"></th><th class="col-sm-1"></th></tr>  | ||||
|                                                         </thead> | ||||
|                                                         <tbody> | ||||
|                                                             <tr ng-repeat="(key, value) in cookie"> | ||||
| @ -187,25 +147,11 @@ | ||||
|                                                                 <td>{{value}}</td> | ||||
|                                                             </tr> | ||||
|                                                         </tbody> | ||||
|                                                         <!--<li ng-repeat="(key, value) in cookie">{{key}} :: {{ value}} </td>--> | ||||
|                                                         <!--</ul>--> | ||||
|                                                     </table> | ||||
|                                                 </div> | ||||
|                                                 </div> --> | ||||
|                                             </div> | ||||
|                                             <div id="paramsView"> <!--class="paramsView"--> | ||||
|                                                 <h4>Params</h4> | ||||
|                                                 <table class="paramsTable table-striped table-nonfluid" id="paramsTable"> | ||||
|                                                     <thead> | ||||
|                                                         <tr><th>Param</th><th>Value</th></tr> | ||||
|                                                     </thead> | ||||
|                                                     <tbody> | ||||
|                                                         <tr ng-repeat="param in parameters"> | ||||
|                                                             <td>{{param.name}}</td> | ||||
|                                                             <td>{{param.value}}</td> | ||||
|                                                         </tr>						 | ||||
|                                                     </tbody> | ||||
|                                                 </table> | ||||
|                                                 </ul> | ||||
|                                             </div> | ||||
|                                         </div> | ||||
|                                     </div> | ||||
| @ -214,44 +160,18 @@ | ||||
|                         </div><!--col-md-4 end-->          | ||||
|                     </div> | ||||
|                     <div id="lessonHelpsWrapper"> | ||||
|                         <!--div class="row lessonHelp" id="lesson_hint_row"> | ||||
|                             <div class="col-md-12"> | ||||
|                                 <h4>Hints</h4> | ||||
|                                 <div class="panel" > | ||||
|                                     <div class="panel-body" id="lesson_hint"> | ||||
|                                         <span class="glyphicon-class glyphicon glyphicon-circle-arrow-left" id="showPrevHintBtn" ng-click="viewPrevHint()"></span> | ||||
|                                         <span class="glyphicon-class glyphicon glyphicon-circle-arrow-right" id="showNextHintBtn" ng-click="viewNextHint()"></span> | ||||
|                                         <br/> | ||||
|                                         {{curHint}} | ||||
|                                     </div>                                     | ||||
|                                 </div> | ||||
|                             </div> | ||||
|                         </div--> | ||||
|                         <div class="row lessonHelp" id="lesson_cookies_row"> | ||||
|                         <div class="row lessonHelp" id="lesson-cookies-row"> | ||||
|                             <div class="col-md-12"> | ||||
|                                 <h4>Lesson Parameters and Cookies</h4> | ||||
|                                 <div class="panel" > | ||||
|                                     <div class="panel-body" id="lesson_cookies">	 | ||||
|                                  | ||||
|                             </div> | ||||
|                         </div>    | ||||
|                             </div> | ||||
|                         </div>    | ||||
|                         <div class="row lessonHelp" id="lesson_hint_row"> | ||||
|                             <div class="col-md-12"> | ||||
|                                 <h4>Lesson Hints</h4> | ||||
|                                 <div class="panel" > | ||||
|                                     <div class="panel-body" id="lesson_hint">	 | ||||
|  | ||||
|                                     </div>                                     | ||||
|                                 </div> | ||||
|                             </div> | ||||
|                         </div>                  | ||||
|                         <div class="row lessonHelp" id="lesson_plan_row"> | ||||
|                         <div class="row lessonHelp" id="lesson-plan-row"> | ||||
|                             <div class="col-md-12"> | ||||
|                                 <h4>Lesson Plan</h4> | ||||
|                                 <div class="panel" > | ||||
|                                     <div class="panel-body" id="lesson_plan"> | ||||
|                                     <div class="panel-body" id="lesson-plan"> | ||||
|                                         <!-- allowing jQuery to handle this one --> | ||||
|                                     </div>                                     | ||||
|                                 </div> | ||||
| @ -271,7 +191,7 @@ | ||||
|                                 <h4>Lesson Source Code</h4> | ||||
|                                 <div class="panel"> | ||||
|                                     <div class="panel-body" id="lesson_source"> | ||||
|                                         <pre>{{source}}</pre> | ||||
|                                         <!-- <pre>{{source}}</pre> --> | ||||
|                                     </div>                                     | ||||
|                                 </div> | ||||
|                             </div> | ||||
| @ -280,20 +200,8 @@ | ||||
|                 </section> | ||||
|             </section> | ||||
|  | ||||
|             <!--main content end--> | ||||
|  | ||||
|         </section> | ||||
|  | ||||
|         <!--main content end--> | ||||
|  | ||||
|         </section> | ||||
|  | ||||
|         <script> | ||||
|             $(document).ready(function() { | ||||
|                 //TODO merge appliction.js code into other js files | ||||
|                 app.init(); | ||||
|             }); | ||||
|         </script> | ||||
|  | ||||
|         <!-- About WebGoat Modal --> | ||||
|         <div class="modal fade" id="aboutModal" tabindex="-1" role="dialog" aria-labelledby="myModalLabel" aria-hidden="true"> | ||||
| @ -304,4 +212,6 @@ | ||||
|             </div> | ||||
|         </div> | ||||
|     </body> | ||||
|  | ||||
|  | ||||
| </html> | ||||
|  | ||||
| @ -1,2 +1,2 @@ | ||||
| #lessonTitle {position:absolute;left:94px;top:75px;width:690px;height:22px;z-index:1;float: right;font-size: 20px;color: #FFFFFF;} | ||||
| #lesson-title {position:absolute;left:94px;top:75px;width:690px;height:22px;z-index:1;float: right;font-size: 20px;color: #FFFFFF;} | ||||
| #hMenuBar {position:absolute;left:245px;top:108px;width:538px;height:22px;z-index:2;} | ||||
|  | ||||
| @ -132,7 +132,7 @@ img { | ||||
| #header .user-nav button:hover i { | ||||
|   color: #F6F6F6; | ||||
| } | ||||
| #header .lessonTitle { | ||||
| #header #lesson-title-wrapper { | ||||
| 	display: inline-block; | ||||
| 	margin:0 0 0 20px;	 | ||||
| } | ||||
| @ -747,7 +747,7 @@ cookieContainer { | ||||
|   z-index: 100; | ||||
| } | ||||
|  | ||||
| #leftside-navigation { | ||||
| #menu-container { | ||||
|   overflow-y:scroll; | ||||
|   overflow-x:hidden; | ||||
| } | ||||
| @ -761,18 +761,18 @@ cookieContainer { | ||||
| .sidebar-toggle { | ||||
|   margin-left: -240px; | ||||
| } | ||||
| #leftside-navigation ul, | ||||
| #leftside-navigation ul ul { | ||||
| #menu-container ul, | ||||
| #menu-container ul ul { | ||||
|   margin: -2px 0 0; | ||||
|   padding: 0; | ||||
| } | ||||
| #leftside-navigation ul li { | ||||
| #menu-container ul li { | ||||
|   list-style-type: none; | ||||
|   border-bottom: 1px solid rgba(255, 255, 255, 0.05); | ||||
| } | ||||
|  | ||||
|  | ||||
| #leftside-navigation ul li a { | ||||
| #menu-container ul li a { | ||||
|   color: #aeb2b7; | ||||
|   text-decoration: none; | ||||
|   display: block; | ||||
| @ -785,38 +785,38 @@ cookieContainer { | ||||
|   -ms-transition: all 200ms ease-in; | ||||
|   transition: all 200ms ease-in; | ||||
| } | ||||
| #leftside-navigation ul li a span { | ||||
| #menu-container ul li a span { | ||||
|   display: inline-block; | ||||
| } | ||||
| #leftside-navigation ul ul li { | ||||
| #menu-container ul ul li { | ||||
|   background: #333; | ||||
|   margin-bottom: 0; | ||||
|   margin-left: 0; | ||||
|   margin-right: 0; | ||||
|   border-bottom: none; | ||||
| } | ||||
| #leftside-navigation ul ul li a { | ||||
| #menu-container ul ul li a { | ||||
|   font-size: 11px; | ||||
|   padding-top: 5px; | ||||
|   padding-bottom: 5px; | ||||
|   color: #aeb2b7; | ||||
|   margin-left:8px; | ||||
| } | ||||
| #leftside-navigation ul li a i { | ||||
| #menu-container ul li a i { | ||||
|   width: 20px; | ||||
| } | ||||
| #leftside-navigation ul li a i.fa-angle-right, | ||||
| #leftside-navigation ul li a i.fa-angle-left { | ||||
| #menu-container ul li a i.fa-angle-right, | ||||
| #menu-container ul li a i.fa-angle-left { | ||||
|   padding-top: 3px; | ||||
| } | ||||
| #leftside-navigation ul ul { | ||||
| #menu-container ul ul { | ||||
|   display: none; | ||||
| } | ||||
| #leftside-navigation li.active ul { | ||||
| #menu-container li.active ul { | ||||
|   display: block; | ||||
| } | ||||
| #leftside-navigation ul li a:hover, | ||||
| #leftside-navigation ul li.active > a { | ||||
| #menu-container ul li a:hover, | ||||
| #menu-container ul li.active > a { | ||||
|   color: #e84c3d; | ||||
| } | ||||
|  | ||||
| @ -827,15 +827,15 @@ cookieContainer { | ||||
|     padding-top: 15px; | ||||
| } | ||||
|  | ||||
| #leftside-navigation ul li.selected { | ||||
| #menu-container ul li.selected { | ||||
|     background-color: #555; | ||||
| } | ||||
|  | ||||
| #leftside-navigation ul li.selected a.selected { | ||||
| #menu-container ul li.selected a.selected { | ||||
|   color:white; | ||||
| } | ||||
|  | ||||
| #leftside-navigation ul ul.lessonsAndStages.keepOpen { | ||||
| #menu-container ul ul.lessonsAndStages.keepOpen { | ||||
|   display: block | ||||
| } | ||||
|  | ||||
| @ -849,8 +849,16 @@ cookieContainer { | ||||
|   cursor: pointer; | ||||
| } | ||||
|  | ||||
|  | ||||
| .info { | ||||
| 	color:#e84c3d; | ||||
| 	font-weight: bold;  | ||||
| } | ||||
|  | ||||
| #help-controls { | ||||
|   padding-left: 4px; | ||||
|   padding-top: 4px | ||||
| } | ||||
|  | ||||
| .help-button { | ||||
|   margin-right:4px; | ||||
| } | ||||
							
								
								
									
										1805
									
								
								src/main/webapp/js/backbone/backbone-edge.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1805
									
								
								src/main/webapp/js/backbone/backbone-edge.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										2
									
								
								src/main/webapp/js/backbone/backbone-min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								src/main/webapp/js/backbone/backbone-min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										1608
									
								
								src/main/webapp/js/backbone/backbone.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1608
									
								
								src/main/webapp/js/backbone/backbone.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										6
									
								
								src/main/webapp/js/backbone/underscore-min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								src/main/webapp/js/backbone/underscore-min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										1416
									
								
								src/main/webapp/js/backbone/underscore.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1416
									
								
								src/main/webapp/js/backbone/underscore.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										98
									
								
								src/main/webapp/js/goatApp/controller/LessonController.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								src/main/webapp/js/goatApp/controller/LessonController.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,98 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'libs/backbone', | ||||
| 	'goatApp/model/LessonContentData', | ||||
| 	'goatApp/view/LessonContentView', | ||||
| 	'goatApp/view/PlanView', | ||||
| 	'goatApp/view/SourceView', | ||||
| 	'goatApp/view/SolutionView', | ||||
| 	'goatApp/view/LessonHintView', | ||||
| 	'goatApp/view/HelpControlsView' | ||||
| 	],  | ||||
| 	function($, | ||||
| 		_, | ||||
| 		Backbone, | ||||
| 		LessonContentData, | ||||
| 		LessonContentView, | ||||
| 		PlanView, | ||||
| 		SourceView, | ||||
| 		SolutionView, | ||||
| 		LessonHintView, | ||||
| 		HelpControlsView | ||||
| 	) { | ||||
| 		'use strict' | ||||
| 		 | ||||
| 		 | ||||
| 		var Controller = function(options) { | ||||
| 			this.lessonContent = new LessonContentData(); | ||||
| 			this.lessonView = options.lessonView; | ||||
|  | ||||
| 			/*this.planView = new PlanView(); | ||||
| 			this.solutionView = new SolutionView(); | ||||
| 			this.sourceView = new SourceView(); | ||||
| 			*/ | ||||
|  | ||||
| 			_.extend(Controller.prototype,Backbone.Events); | ||||
| 			this.start = function() { | ||||
| 				this.listenTo(this.lessonContent,'contentLoaded',this.onContentLoaded); | ||||
| 			}; | ||||
|  | ||||
| 			//load View, which can pull data | ||||
| 			this.loadLesson = function(scr,menu) { | ||||
| 				this.helpsLoaded = {}; | ||||
| 				this.lessonContent.loadData({ | ||||
| 					'screen': encodeURIComponent(scr), | ||||
| 					'menu': encodeURIComponent(menu), | ||||
| 				}); | ||||
| 				this.planView = {}; | ||||
| 				this.solutionView = {}; | ||||
| 				this.sourceView = {}; | ||||
| 				this.lessonHintView = {}; | ||||
| 				// | ||||
| 				 | ||||
| 			}; | ||||
|  | ||||
| 			this.onContentLoaded = function() { | ||||
| 				//this.lessonView  = new LessonContentView({content:LessonContent.content}); | ||||
| 				this.lessonView.model = this.lessonContent; | ||||
| 				this.lessonView.render(); | ||||
|  | ||||
| 				//load cookies/parameters view | ||||
|  | ||||
| 				//load title view (initially hidden) << currently handled via menu click but need to be able to handle via routed request | ||||
| 				//plan view (initially hidden) | ||||
| 				this.planView = new PlanView(); | ||||
| 				this.listenTo(this.planView,'plan:loaded',this.areHelpsReady); | ||||
| 				//solution view (initially hidden) | ||||
| 				this.solutionView = new SolutionView(); | ||||
| 				this.listenTo(this.solutionView,'solution:loaded',this.areHelpsReady); | ||||
| 				//source (initially hidden) | ||||
| 				this.sourceView = new SourceView(); | ||||
| 				this.listenTo(this.sourceView,'source:loaded',this.areHelpsReady); | ||||
| 				//load help controls view (contextul to what helps are available) | ||||
| 				this.lessonHintView = new LessonHintView(); | ||||
| 				this.listenTo(this.lessonHintView,'hints:loaded',this.areHelpsReady); | ||||
| 				 | ||||
| 			}; | ||||
|  | ||||
| 			this.areHelpsReady = function (curHelp) { | ||||
| 				this.addCurHelpState(curHelp); | ||||
| 				// check if all are ready | ||||
| 				if (this.helpsLoaded['hints'] && this.helpsLoaded['plan'] && this.helpsLoaded['solution'] && this.helpsLoaded['source']) { | ||||
| 					// | ||||
| 					this.helpControlsView = new HelpControlsView({ | ||||
| 						hasPlan:(this.planView.model.get('content') !== null), | ||||
| 						hasSolution:(this.solutionView.model.get('content') !== null), | ||||
| 						hasSource:(this.sourceView.model.get('content') !== null), | ||||
| 						hasHints:(this.lessonHintView.collection.length > 0), | ||||
| 					}); | ||||
| 					this.helpControlsView.render(); | ||||
| 				} | ||||
| 			}; | ||||
|  | ||||
| 			this.addCurHelpState = function (curHelp) { | ||||
| 				this.helpsLoaded[curHelp.helpElement] = curHelp.value; | ||||
| 			};			 | ||||
| 		}; | ||||
| 		return Controller; | ||||
| }); | ||||
							
								
								
									
										25
									
								
								src/main/webapp/js/goatApp/controller/MenuController.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								src/main/webapp/js/goatApp/controller/MenuController.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,25 @@ | ||||
| define(['jquery','underscore','backbone','goatApp/view/MenuView'],  | ||||
| 	function($,_,Backbone,MenuView) { | ||||
| 		 Controller = function(options){ | ||||
| 		 	_.extend(Controller.prototype,Backbone.Events); | ||||
| 			options = options || {}; | ||||
| 			this.menuView = options.menuView; | ||||
| 			this.titleView = options.titleView; | ||||
|  | ||||
| 		 	this.initMenu = function() { | ||||
| 		 		this.listenTo(this.menuView,'lesson:click',this.renderTitle); | ||||
| 		 	} | ||||
|  | ||||
| 		 	this.updateMenu = function() { | ||||
| 		 		 | ||||
| 		 	}, | ||||
|  | ||||
| 		 	//TODO: move title rendering into lessonContent/View pipeline once data can support it | ||||
| 		 	this.renderTitle = function(title) { | ||||
| 		 		this.titleView.render(title); | ||||
| 		 	} | ||||
|  | ||||
| 		 }; | ||||
|  | ||||
| 		 return Controller; | ||||
| }); | ||||
							
								
								
									
										12
									
								
								src/main/webapp/js/goatApp/goatApp.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								src/main/webapp/js/goatApp/goatApp.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | ||||
| define(['jquery','underscore','backbone','goatApp/view/GoatRouter'], | ||||
| 	function($,_,Backbone,Router){ | ||||
| 		'use strict' | ||||
| 		//var goatRouter = new Router(); | ||||
| 		return { | ||||
| 			initApp: function() { | ||||
| 				//TODO: add query/ability to load from where they left off  | ||||
| 				var goatRouter = new Router(); | ||||
| 				goatRouter.init(); | ||||
| 			} | ||||
| 		}; | ||||
| }); | ||||
							
								
								
									
										27
									
								
								src/main/webapp/js/goatApp/model/HTMLContentModel.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								src/main/webapp/js/goatApp/model/HTMLContentModel.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,27 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone'], | ||||
| 	function($,_,Backbone) { | ||||
| 	//TODO: make a base class to extend for items with 'traditional data' (e.g. LessonContentData, this ... others?) | ||||
| 	return Backbone.Model.extend({ | ||||
| 		//url:'service/lessonplan.mvc', | ||||
| 		fetch: function (options) { | ||||
| 			options = options || {}; | ||||
| 			return Backbone.Model.prototype.fetch.call(this, _.extend({ dataType: "html"}, options)); | ||||
| 		}, | ||||
| 		loadData: function() { | ||||
| 			var self=this; | ||||
| 			this.fetch().then(function(data) { | ||||
| 				self.setContent(data); | ||||
| 				self.onModelLoaded(); | ||||
| 			}); | ||||
| 		}, | ||||
| 		setContent: function(content) { | ||||
| 			this.set('content',content); | ||||
| 			this.trigger('loaded'); | ||||
| 		}, | ||||
| 		onModelLoaded: function() { | ||||
| 			this.checkNullModel(); | ||||
| 		} | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										40
									
								
								src/main/webapp/js/goatApp/model/LessonContentData.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								src/main/webapp/js/goatApp/model/LessonContentData.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,40 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/model/HTMLContentModel'], | ||||
| 	 function($, | ||||
| 	 	_, | ||||
| 	 	Backbone, | ||||
| 	 	HTMLContentModel){ | ||||
|  | ||||
| 	return HTMLContentModel.extend({ | ||||
| 		urlRoot:null, | ||||
| 		defaults: { | ||||
| 			items:null, | ||||
| 			selectedItem:null | ||||
| 		}, | ||||
| 		initialize: function (options) { | ||||
| 			this.screenParam = null; | ||||
| 			this.menuParam = null; | ||||
| 			this.baseUrlRoot = 'attack?Screen=';// | ||||
| 		}, | ||||
| 		loadData: function(options) { | ||||
| 			this.urlRoot = this.baseUrlRoot  +options.screen + '&menu=' + options.menu; | ||||
| 			this.set('menuParam',options.menu); | ||||
| 			this.set('screenParam',options.screen); | ||||
|  | ||||
| 			var self=this; | ||||
| 			this.fetch().then(function(data) { | ||||
| 				self.setContent(data); | ||||
| 			}); | ||||
| 		}, | ||||
| 		setContent: function(content) { | ||||
| 			this.set('content',content); | ||||
| 			this.trigger('contentLoaded'); | ||||
| 		}, | ||||
| 		fetch: function (options) { | ||||
| 			options = options || {}; | ||||
| 			return Backbone.Model.prototype.fetch.call(this, _.extend({ dataType: "html"}, options)); | ||||
| 		} | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										24
									
								
								src/main/webapp/js/goatApp/model/LessonHintCollection.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								src/main/webapp/js/goatApp/model/LessonHintCollection.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,24 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/model/LessonHintModel'],  | ||||
| 	function($,_,Backbone,LessonHintModel) { | ||||
|  | ||||
| 	return Backbone.Collection.extend({ | ||||
| 		model: LessonHintModel, | ||||
| 		url:'service/hint.mvc', | ||||
| 		initialize: function () { | ||||
| 			var self = this; | ||||
| 			this.fetch().then(function (data) { | ||||
| 				this.models = data; | ||||
| 				self.onDataLoaded(); | ||||
| 			}); | ||||
| 		}, | ||||
| 		onDataLoaded:function() { | ||||
| 			this.trigger('hints:loaded');//copied over as boiler-plate ... use this event trigger? | ||||
| 		}, | ||||
| 		checkNullModel:function() { | ||||
| 			// | ||||
| 		} | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										10
									
								
								src/main/webapp/js/goatApp/model/LessonHintModel.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								src/main/webapp/js/goatApp/model/LessonHintModel.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone'], | ||||
| 	function($, | ||||
| 		_, | ||||
| 		Backbone, | ||||
| 		HTMLContentModel) { | ||||
| 	return Backbone.Model.extend({ | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										18
									
								
								src/main/webapp/js/goatApp/model/LessonPlanModel.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/main/webapp/js/goatApp/model/LessonPlanModel.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,18 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/model/HTMLContentModel'], | ||||
| 	function($, | ||||
| 		_, | ||||
| 		Backbone, | ||||
| 		HTMLContentModel) { | ||||
| 	return HTMLContentModel.extend({ | ||||
| 		url:'service/lessonplan.mvc', | ||||
| 		checkNullModel: function() { | ||||
| 			if (this.get('content').indexOf('Plan is not available for this lesson.') > -1) { | ||||
| 				this.set('content',null); | ||||
| 			} | ||||
| 		} | ||||
| 		 | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										18
									
								
								src/main/webapp/js/goatApp/model/LessonSolutionModel.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/main/webapp/js/goatApp/model/LessonSolutionModel.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,18 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/model/HTMLContentModel'], | ||||
| 	function($, | ||||
| 		_, | ||||
| 		Backbone, | ||||
| 		HTMLContentModel) { | ||||
| 	return HTMLContentModel.extend({ | ||||
| 		url:'service/solution.mvc', | ||||
| 		checkNullModel: function() { | ||||
| 			if (this.get('content').indexOf('Solution  is not available. Contact') === 0) { | ||||
| 				this.set('content',null); | ||||
| 			} | ||||
| 		} | ||||
| 		 | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										19
									
								
								src/main/webapp/js/goatApp/model/LessonSourceModel.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								src/main/webapp/js/goatApp/model/LessonSourceModel.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,19 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/model/HTMLContentModel'], | ||||
| 	function($, | ||||
| 		_, | ||||
| 		Backbone, | ||||
| 		HTMLContentModel) { | ||||
| 	return HTMLContentModel.extend({ | ||||
| 		url:'service/source.mvc', | ||||
| 		checkNullModel: function () { | ||||
| 			//TODO: move this function into HTMLContentModel and make the string a property of this 'child' model | ||||
| 			if (this.get('content').indexOf("No source listing found") > -1) { | ||||
| 				this.set('content',null); | ||||
| 			} | ||||
| 		} | ||||
| 		 | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										21
									
								
								src/main/webapp/js/goatApp/model/MenuCollection.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								src/main/webapp/js/goatApp/model/MenuCollection.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/model/MenuModel'],  | ||||
| 	function($,_,Backbone,MenuModel) { | ||||
|  | ||||
| 	return Backbone.Collection.extend({ | ||||
| 		model: MenuModel, | ||||
| 		url:'service/lessonmenu.mvc', | ||||
| 		initialize: function () { | ||||
| 			var self = this; | ||||
| 			this.fetch().then(function (data) { | ||||
| 				this.models = data; | ||||
| 				self.onDataLoaded(); | ||||
| 			}); | ||||
| 		}, | ||||
| 		onDataLoaded:function() { | ||||
| 			this.trigger('menuData:loaded'); | ||||
| 		} | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										57
									
								
								src/main/webapp/js/goatApp/model/MenuData.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								src/main/webapp/js/goatApp/model/MenuData.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,57 @@ | ||||
| //var goatApp = goatApp || {}; | ||||
|  | ||||
| define(['jquery','underscore','backbone'], function($,_,Backbone) { | ||||
|  | ||||
| var menuData = Backbone.Model.extend({ | ||||
| 		urlRoot:'/webgoat/service/lessonmenu.mvc', | ||||
| 		defaults: { | ||||
| 			items:null, | ||||
| 			selectedItem:null | ||||
| 		}, | ||||
| 		initialize: function () { | ||||
| 			var self = this; | ||||
| 			this.fetch().then(function(menuItems){ | ||||
| 				menuItems = goatUtils.enhanceMenuData(menuItems,this.selectedItem); | ||||
| 				this.setDataItems(menuItems); | ||||
| 			}); | ||||
| 		}, | ||||
| 	 | ||||
| 		update: function() { | ||||
| 			var self = this; | ||||
| 			this.fetch().then(function(menuItems) { | ||||
| 				menuItems = goatUtils.enhanceMenuData(menuItems,this.selectedItem); | ||||
| 				self.setDataItems(menuItems); | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		setDataItems: function (data) { | ||||
| 			this.items = data; | ||||
| 		} | ||||
| 	}); | ||||
|  | ||||
| }); | ||||
|  | ||||
| /* | ||||
| var menuData = Backbone.Model.extend({ | ||||
| 		urlRoot:'/webgoat/service/lessonmenu.mvc', | ||||
| 		defaults: { | ||||
| 			items:null, | ||||
| 			selectedItem:null | ||||
| 		}, | ||||
| 		initialize: function () { | ||||
| 			var self = this; | ||||
| 			this.fetch().then(function(menuItems){ | ||||
| 				menuItems = goatUtils.enhanceMenuData(menuItems,this.selectedItem); | ||||
| 				self.items = menuItems; | ||||
| 			}); | ||||
| 		}, | ||||
| 	 | ||||
| 		update: function() { | ||||
| 			var self = this; | ||||
| 			this.fetch().then(function(data) { | ||||
| 				self.items = data; | ||||
| 				self.render(0); | ||||
| 			}); | ||||
| 		} | ||||
| 	}); | ||||
| */ | ||||
							
								
								
									
										10
									
								
								src/main/webapp/js/goatApp/model/MenuModel.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								src/main/webapp/js/goatApp/model/MenuModel.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone'], | ||||
| 	function($,_,Backbone) { | ||||
|  | ||||
| 	return Backbone.Model.extend({ | ||||
|  | ||||
| 	}); | ||||
|  | ||||
| }); | ||||
							
								
								
									
										178
									
								
								src/main/webapp/js/goatApp/support/GoatUtils.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								src/main/webapp/js/goatApp/support/GoatUtils.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,178 @@ | ||||
| define(['jquery', | ||||
|     'underscore', | ||||
|     'backbone', | ||||
|     'libs/jquery.form' | ||||
|     ],  | ||||
|     function($, | ||||
|         _, | ||||
|         Backbone, | ||||
|         JQueryForm) { | ||||
|             var goatUtils = { | ||||
|                 makeId: function(lessonName) { | ||||
|                     //var id = | ||||
|                     return  lessonName.replace(/\s|\(|\)|\!|\:|\;|\@|\#|\$|\%|\^|\&|\*/g, ''); | ||||
|                 }, | ||||
|                 addMenuClasses: function(arr) { | ||||
|                     for (var i = 0; i < arr.length; i++) { | ||||
|                         var menuItem = arr[i]; | ||||
|                         //console.log(menuItem); | ||||
|                         if (menuItem.type && menuItem.type === 'CATEGORY') { | ||||
|                             menuItem.class = 'fa-angle-right pull-right'; | ||||
|                         } | ||||
|                     } | ||||
|                     return arr; | ||||
|                 }, | ||||
| //                debugFormSubmission: false, | ||||
|                 // pre-submit callback  | ||||
|                 showRequest: function(formData, jqForm, options) { | ||||
|                     if (GoatUtils.debugFormSubmission) { | ||||
|                         // formData is an array; here we use $.param to convert it to a string to display it  | ||||
|                         // but the form plugin does this for you automatically when it submits the data  | ||||
|                         var queryString = $.param(formData); | ||||
|  | ||||
|                         // jqForm is a jQuery object encapsulating the form element.  To access the  | ||||
|                         // DOM element for the form do this:  | ||||
|                         // var formElement = jqForm[0];  | ||||
|  | ||||
|                         alert('About to submit: \n\n' + queryString); | ||||
|                     } | ||||
|                     // here we could return false to prevent the form from being submitted;  | ||||
|                     // returning anything other than false will allow the form submit to continue  | ||||
|                     return true; | ||||
|                 }, | ||||
|                 // post-submit callback  | ||||
|                 showResponse: function(responseText, statusText, xhr, $form) { | ||||
|                     // for normal html responses, the first argument to the success callback  | ||||
|                     // is the XMLHttpRequest object's responseText property  | ||||
|  | ||||
|                     // if the ajaxForm method was passed an Options Object with the dataType  | ||||
|                     // property set to 'xml' then the first argument to the success callback  | ||||
|                     // is the XMLHttpRequest object's responseXML property  | ||||
|  | ||||
|                     // if the ajaxForm method was passed an Options Object with the dataType  | ||||
|                     // property set to 'json' then the first argument to the success callback  | ||||
|                     // is the json data object returned by the server  | ||||
|                     if (GoatUtils.debugFormSubmission) { | ||||
|                         alert('status: ' + statusText + '\n\nresponseText: \n' + responseText + | ||||
|                                 '\n\nThe output div should have already been updated with the responseText.'); | ||||
|                     } | ||||
|                     // update lesson cookies and params | ||||
|                     // make any embedded forms ajaxy | ||||
|                     GoatUtils.showLessonCookiesAndParams(); | ||||
|                     // forms and links are now hooked with each standard lesson render (see Java class Screen.getContent()) | ||||
|                     // but these are safe to call twice | ||||
|                     GoatUtils.makeFormsAjax(); | ||||
|                     GoatUtils.ajaxifyAttackHref(); //TODO find some way to hook scope for current menu. Likely needs larger refactor which is already started/stashed | ||||
|                     //refresh menu | ||||
|                     angular.element($('#leftside-navigation')).scope().renderMenu(); | ||||
|                 }, | ||||
|                 makeFormsAjax: function() { | ||||
|                     // make all forms ajax forms | ||||
|                     var options = { | ||||
|                         target: '#lesson_content', // target element(s) to be updated with server response                      | ||||
|                         beforeSubmit: GoatUtils.showRequest, // pre-submit callback, comment out after debugging  | ||||
|                         success: GoatUtils.showResponse  // post-submit callback, comment out after debugging  | ||||
|  | ||||
|                                 // other available options:  | ||||
|                                 //url:       url         // override for form's 'action' attribute  | ||||
|                                 //type:      type        // 'get' or 'post', override for form's 'method' attribute  | ||||
|                                 //dataType:  null        // 'xml', 'script', or 'json' (expected server response type)  | ||||
|                                 //clearForm: true        // clear all form fields after successful submit  | ||||
|                                 //resetForm: true        // reset the form after successful submit  | ||||
|  | ||||
|                                 // $.ajax options can be used here too, for example:  | ||||
|                                 //timeout:   3000  | ||||
|                     }; | ||||
|                     //console.log("Hooking any lesson forms to make them ajax"); | ||||
|                     $("form").ajaxForm(options); | ||||
|                 }, | ||||
|                 displayButton: function(id, show) { | ||||
|                     if ($('#' + id)) { | ||||
|                         if (show) { | ||||
|                             $('#' + id).show(); | ||||
|                         } else { | ||||
|                             $('#' + id).hide(); | ||||
|                         } | ||||
|                     } | ||||
|                 }, | ||||
|                 showLessonCookiesAndParams: function() { | ||||
|                     $.get(goatConstants.cookieService, {}, function(reply) { | ||||
|                         $("#lesson_cookies").html(reply); | ||||
|                     }, "html"); | ||||
|                 }, | ||||
|                 showLessonHints: function() { | ||||
|                     $('.lessonHelp').hide(); | ||||
|                     $('#lesson_hint').html(); | ||||
|                     $('#lesson_hint_row').show(); | ||||
|                 }, | ||||
|                 showLessonSource: function(source) { | ||||
|                     $('.lessonHelp').hide(); | ||||
|                     //$('#lesson_source').html("<pre>"+goat.lesson.lessonInfo.source+"</pre>"); | ||||
|                     $('#lesson_source_row').show(); | ||||
|                     GoatUtils.scrollToHelp(); | ||||
|                 }, | ||||
|                 showLessonSolution: function() { | ||||
|                     $('.lessonHelp').hide(); | ||||
|                     $('#lesson_solution').html(goat.lesson.lessonInfo.solution); | ||||
|                     $('#lesson_solution_row').show(); | ||||
|                     GoatUtils.scrollToHelp(); | ||||
|                 }, | ||||
|                 showLessonPlan: function(plan) { | ||||
|                     $('.lessonHelp').hide(); | ||||
|                     $("#lesson_plan").html(goat.lesson.lessonInfo.plan); | ||||
|                     $('#lesson_plan_row').show(); | ||||
|                     GoatUtils.scrollToHelp(); | ||||
|                 }, | ||||
|                 scrollToHelp: function() { | ||||
|                     $('#leftside-navigation').height($('#main-content').height() + 15) | ||||
|                     var target = $('#lessonHelpsWrapper'); | ||||
|                     GoatUtils.scrollEasy(target); | ||||
|                 }, | ||||
|                 scrollToTop: function() { | ||||
|                     $('.lessonHelp').hide(); | ||||
|                     var target = $('#container'); | ||||
|                     GoatUtils.scrollEasy(target); | ||||
|                 }, | ||||
|                 scrollEasy: function(target) { | ||||
|                     $('html,body').animate({ | ||||
|                         scrollTop: target.offset().top | ||||
|                     }, 1000); | ||||
|                 }, | ||||
|                 scrapeParams: function(url) { | ||||
|                     if (!url) { | ||||
|                         return; | ||||
|                     } | ||||
|                     var params = url.split('?')[1].split('&'); | ||||
|                     var paramsArr = []; | ||||
|                     for (var i = 0; i < params.length; i++) { | ||||
|                         var paramObj = {}; | ||||
|                         paramObj.name = params[i].split('=')[0]; | ||||
|                         paramObj.value = params[i].split('=')[1]; | ||||
|                         paramsArr.push(paramObj); | ||||
|                     } | ||||
|                     return paramsArr; | ||||
|                 }, | ||||
|                 highlightCurrentLessonMenu: function(id) { | ||||
|                     //TODO: move selectors in first two lines into goatConstants | ||||
|                     $('ul li.selected').removeClass(goatConstants.selectedMenuClass) | ||||
|                     $('ul li.selected a.selected').removeClass(goatConstants.selectedMenuClass) | ||||
|                     $('#' + id).addClass(goatConstants.selectedMenuClass); | ||||
|                     $('#' + id).parent().addClass(goatConstants.selectedMenuClass); | ||||
|                 }, | ||||
|                 ajaxifyAttackHref: function() {  // rewrite any links with hrefs point to relative attack URLs              | ||||
|                  $.each($('a[href^="attack?"]'), | ||||
|         			function(i,el) { | ||||
|         				var url = $(el).attr('href'); | ||||
|         				$(el).unbind('click').attr('href','#').attr('link',url); | ||||
|         				//TODO pull currentMenuId | ||||
|         				$(el).click(function() { | ||||
|         					event.preventDefault(); | ||||
|         					var _url = $(el).attr('link'); | ||||
|         					$.get(_url, {success:showResponse}); | ||||
|         				}); | ||||
|                     }); | ||||
|         		} | ||||
|         }; | ||||
|      | ||||
|         return goatUtils; | ||||
| }); | ||||
							
								
								
									
										39
									
								
								src/main/webapp/js/goatApp/support/goatConstants.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								src/main/webapp/js/goatApp/support/goatConstants.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,39 @@ | ||||
| //goatConstants | ||||
|  | ||||
| var goatConstants = { | ||||
| 	getClasses: function() { | ||||
| 		return { | ||||
| 			categoryClass:'fa-angle-right pull-right', | ||||
| 			lessonCompleteClass:'glyphicon glyphicon-check lessonComplete', | ||||
| 			selectedMenuClass:'selected', | ||||
| 			keepOpenClass:'keepOpen' | ||||
| 		}; | ||||
| 	}, | ||||
| 	getServices: function() { | ||||
| 		return { | ||||
| 			lessonService: 'service/lessonmenu.mvc', | ||||
| 			cookieService: 'service/cookie.mvc', //cookies_widget.mvc | ||||
| 			hintService: 'service/hint.mvc', | ||||
| 			sourceService: 'service/source.mvc', | ||||
| 			solutionService: 'service/solution.mvc', | ||||
| 			lessonPlanService: 'service/lessonplan.mvc', | ||||
| 			menuService: 'service/lessonmenu.mvc', | ||||
| 			lessonTitleService: 'service/lessontitle.mvc', | ||||
| 			restartLessonService: 'service/restartlesson.mvc'	 | ||||
| 		} | ||||
| 	}, | ||||
| 	getMessages: function() { | ||||
| 		return { | ||||
| 			notFound: 'Could not find', | ||||
| 			noHints: 'There are no hints defined.', | ||||
| 			noSourcePulled: 'No source was retrieved for this lesson' | ||||
| 		} | ||||
| 	}, | ||||
| 	getDOMContainers:function() { | ||||
| 		return { | ||||
| 			lessonMenu: '#menu-container' | ||||
| 		} | ||||
| 	} | ||||
| }; | ||||
|  | ||||
|  | ||||
							
								
								
									
										50
									
								
								src/main/webapp/js/goatApp/view/GoatRouter.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								src/main/webapp/js/goatApp/view/GoatRouter.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,50 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/controller/LessonController', | ||||
| 	'goatApp/controller/MenuController', | ||||
| 	'goatApp/view/LessonContentView', | ||||
| 	'goatApp/view/MenuView', | ||||
| 	'goatApp/view/TitleView' | ||||
| 	], function ($,_,Backbone,LessonController,MenuController,LessonContentView,MenuView,TitleView) { | ||||
|  | ||||
| 		var lessonView = new LessonContentView(); | ||||
| 		var menuView = new MenuView();  | ||||
| 		var titleView = new TitleView(); | ||||
|  | ||||
| 		var GoatAppRouter = Backbone.Router.extend({ | ||||
| 			routes: { | ||||
| 				//#.... | ||||
| 				'welcome':'welcomeRoute', | ||||
| 				'attack/:scr/:menu':'attackRoute' //	 | ||||
| 		 	}, | ||||
| 			lessonController: new LessonController({ | ||||
| 				lessonView:lessonView | ||||
| 			}), | ||||
| 			menuController: new MenuController({ | ||||
| 				menuView:menuView, | ||||
| 				titleView:titleView | ||||
| 			}), | ||||
|  | ||||
| 			init:function() { | ||||
| 				goatRouter =  new GoatAppRouter(); | ||||
| 				this.lessonController.start(); | ||||
| 				this.menuController.initMenu(); | ||||
|  | ||||
| 				goatRouter.on('route:attackRoute', function(scr,menu) { | ||||
| 					console.log('attack route'); | ||||
| 					this.lessonController.loadLesson(scr,menu); | ||||
| 					this.menuController.updateMenu(scr,menu); | ||||
| 					//update menu | ||||
| 				}); | ||||
| 				goatRouter.on('route:welcomeRoute', function() { | ||||
| 					alert('welcome route'); | ||||
| 				}); | ||||
| 				 | ||||
| 				Backbone.history.start(); | ||||
| 			} | ||||
| 		}); | ||||
|  | ||||
| 		return GoatAppRouter; | ||||
|  | ||||
| }); | ||||
							
								
								
									
										41
									
								
								src/main/webapp/js/goatApp/view/HelpControlsView.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								src/main/webapp/js/goatApp/view/HelpControlsView.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,41 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone'], | ||||
| function($,_,Backbone) { | ||||
| 	return Backbone.View.extend({ | ||||
| 		el:'#help-controls', //Check this | ||||
| 		helpButtons: { | ||||
| 			//TODO: move this into a template | ||||
| 			showSource:$('<button>',{id:'showSourceBtn','class':'btn btn-primary btn-xs help-button',type:'button',text:'Java [Source]'}), | ||||
| 			showSolution:$('<button>',{id:'showSolutionBtn','class':'btn btn-primary btn-xs help-button',type:'button',text:'Solution'}), | ||||
| 			showPlan:$('<button>',{id:'showPlanBtn','class':'btn btn-primary btn-xs help-button',type:'button',text:'Lesson Plan]'}), | ||||
| 			showHints:$('<button>',{id:'showHintsBtn','class':'btn btn-primary btn-xs help-button',type:'button',text:'Hints'}), | ||||
| 			restartLesson:$('<button>',{id:'restartLessonBtn','class':'btn btn-xs help-button',type:'button',text:'Restart Lesson'}) | ||||
| 		}, | ||||
| 		initialize: function (options) { | ||||
| 			if (!options) { | ||||
| 				return; | ||||
| 			} | ||||
| 			this.hasPlan = options.hasPlan; | ||||
| 			this.hasSolution = options.hasSolution; | ||||
| 			this.hasSource = options.hasSource; | ||||
| 			this.hasHints = options.hasHints; | ||||
| 		}, | ||||
| 		render:function(title) { | ||||
| 			if (this.hasSource) { | ||||
| 				this.$el.append(this.helpButtons.showSource); | ||||
| 			} | ||||
| 			if (this.hasSolution) { | ||||
| 				this.$el.append(this.helpButtons.showSolution); | ||||
| 			} | ||||
| 			if (this.hasPlan) { | ||||
| 				this.$el.append(this.helpButtons.showPlan); | ||||
| 			} | ||||
| 			if (this.hasHints) { | ||||
| 				this.$el.append(this.helpButtons.showHints); | ||||
| 			} | ||||
| 			// | ||||
| 			this.$el.append(this.helpButtons.restartLesson); | ||||
| 		} | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										23
									
								
								src/main/webapp/js/goatApp/view/HelpView.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								src/main/webapp/js/goatApp/view/HelpView.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,23 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/model/LessonSourceModel'], | ||||
| function($, | ||||
| 	_, | ||||
| 	Backbone, | ||||
| 	LessonSourceModel) { | ||||
| 	return Backbone.View.extend({ | ||||
| 		el:'#lessonHelpWrapper .lessonHelp.lessonPlan', //Check this | ||||
| 		initialize: function() { | ||||
| 			this.model = new LessonSourceModel(); | ||||
| 			this.listenTo(this.model,'loaded',this.onModelLoaded); | ||||
| 			this.model.loadData(); | ||||
| 		}, | ||||
| 		render:function(title) { | ||||
| 			 | ||||
| 		}, | ||||
| 		onModelLoaded: function() { | ||||
| 			this.trigger(this.loadedMessage,this.helpElement); | ||||
| 		} | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										39
									
								
								src/main/webapp/js/goatApp/view/LessonContentView.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								src/main/webapp/js/goatApp/view/LessonContentView.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,39 @@ | ||||
| //LessonContentView | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'libs/jquery.form', | ||||
| 	'goatApp/model/LessonContentData'],  | ||||
| function($,_,Backbone,JQueryForm,LessonData) { | ||||
| 	return Backbone.View.extend({ | ||||
| 		el:'#lessonContentWrapper', //TODO << get this fixed up in DOM | ||||
| 		initialize: function(options) { | ||||
| 			options = options || {}; | ||||
| 		}, | ||||
| 		render: function() { | ||||
| 			this.$el.html(this.model.get('content')); | ||||
| 			this.makeFormsAjax(); | ||||
| 		}, | ||||
| 		//TODO: reimplement this in custom fashion maybe? | ||||
| 		makeFormsAjax: function () { | ||||
| 			var options = { | ||||
| 			    //target: '#lesson_content', // target element(s) to be updated with server response                      | ||||
| 			    //beforeSubmit: GoatUtils.showRequest, // pre-submit callback, comment out after debugging  | ||||
| 			    //success: GoatUtils.showResponse  // post-submit callback, comment out after debugging  | ||||
| 			    success:this.reLoadView.bind(this), | ||||
| 			    url:'attack?Screen=' + this.model.get('screenParam') + '&menu=' + this.model.get('menuParam'), | ||||
| 			    type:'GET' | ||||
| 	            // $.ajax options can be used here too, for example:  | ||||
| 	            //timeout:   3000  | ||||
| 			}; | ||||
| 			//hook forms //TODO: clarify form selectors later | ||||
| 		    $("form").ajaxForm(options); | ||||
|         }, | ||||
|         reLoadView: function(content) { | ||||
|         	this.model.setContent(content); | ||||
|         	this.render(); | ||||
|         } | ||||
| 	}); | ||||
|  | ||||
| 	 | ||||
| }); | ||||
							
								
								
									
										22
									
								
								src/main/webapp/js/goatApp/view/LessonHintView.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/main/webapp/js/goatApp/view/LessonHintView.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,22 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/model/LessonHintCollection'], | ||||
| function($, | ||||
| 	_, | ||||
| 	Backbone, | ||||
| 	LessonHintCollection) { | ||||
| 	return Backbone.View.extend({ | ||||
| 		el:'#lessonHelpWrapper .lessonHelp.lessonHint', | ||||
| 		initialize: function() { | ||||
| 			this.collection = new LessonHintCollection(); | ||||
| 			this.listenTo(this.collection,'hints:loaded',this.onModelLoaded); | ||||
| 		}, | ||||
| 		render:function(title) { | ||||
| 			 | ||||
| 		}, | ||||
| 		onModelLoaded: function() { | ||||
| 			this.trigger('hints:loaded',{'helpElement':'hints','value':true}) | ||||
| 		} | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										30
									
								
								src/main/webapp/js/goatApp/view/MenuItemView.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								src/main/webapp/js/goatApp/view/MenuItemView.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,30 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/support/GoatUtils', | ||||
| 	'goatApp/view/MenuItemView'], | ||||
| 	function( | ||||
| 		$, | ||||
| 		_, | ||||
| 		Backbone, | ||||
| 		GoatUtils, | ||||
| 		MenuItemView) { | ||||
|  | ||||
| 	return Backbone.View.extend({ | ||||
| 		initialize: function(options) { | ||||
| 			options = options || {}; | ||||
| 			this.items = options.items; | ||||
| 		}, | ||||
| 		render: function() { | ||||
| 			var viewItems = []; | ||||
| 			for (var i=0;i<this.items.length;i++) { | ||||
| 				var listItem = $('<li>',{text:this.items[i].name}); | ||||
| 				//viewItems | ||||
| 				viewItems.push(listItem); | ||||
| 			} | ||||
| 			return viewItems; | ||||
| 		} | ||||
|  | ||||
| 	}); | ||||
|  | ||||
| }); | ||||
							
								
								
									
										14
									
								
								src/main/webapp/js/goatApp/view/MenuStageView.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								src/main/webapp/js/goatApp/view/MenuStageView.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,14 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone'], function($,_,Backbone) { | ||||
|  | ||||
| 	return Backbone.View.extend({ | ||||
| 		initialize: function(options) { | ||||
| 			options = options || {}; | ||||
|  | ||||
| 		} | ||||
|  | ||||
|  | ||||
| 	}); | ||||
|  | ||||
| }); | ||||
							
								
								
									
										96
									
								
								src/main/webapp/js/goatApp/view/MenuView.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								src/main/webapp/js/goatApp/view/MenuView.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,96 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/model/MenuCollection', | ||||
| 	'goatApp/view/MenuItemView', | ||||
| 	'goatApp/support/GoatUtils'],  | ||||
| 	function( | ||||
| 		$, | ||||
| 		_, | ||||
| 		Backbone, | ||||
| 		MenuCollection, | ||||
| 		MenuItemView, | ||||
| 		GoatUtils) { | ||||
| 	return  Backbone.View.extend({ | ||||
| 		el:'#menu-container', | ||||
| 		//TODO: set template | ||||
| 		initialize: function() { | ||||
| 			this.collection = new MenuCollection(); | ||||
| 			this.listenTo(this.collection,'menuData:loaded',this.render); | ||||
| 			this.listenTo(this,'menu:click',this.accordionMenu); | ||||
| 		}, | ||||
| 		// rendering top level menu | ||||
| 		render: function (model){ | ||||
| 			//for now, just brute force | ||||
| 			//TODO: refactor into sub-views/components | ||||
| 			var items, catItems, stages; | ||||
| 			items = this.collection.models; // top level items | ||||
| 			var menuMarkup = ''; | ||||
| 			var menuUl = $('<ul>',{class:'nano-content'}); | ||||
| 			for(var i=0;i<items.length;i++) { //CATEGORY LEVEL | ||||
| 				var catId, category, catLink, catArrow, catLinkText, lessonName, stageName; | ||||
| 				catId = GoatUtils.makeId(items[i].get('name')); | ||||
| 				category = $('<li>',{class:'sub-menu ng-scope'}); | ||||
| 				catLink = $('<a>',{'category':catId}); | ||||
| 				catArrow = $('<i>',{class:'fa fa-angle-right pull-right'}); | ||||
| 				catLinkText = $('<span>',{text:items[i].get('name')}); | ||||
|  | ||||
| 				catLink.append(catArrow); | ||||
| 				catLink.append(catLinkText); | ||||
| 				//TODO: refactor this along with sub-views/components | ||||
| 				var self = this; | ||||
| 				catLink.click(_.bind(this.expandCategory,this,catId)); | ||||
| 				category.append(catLink); | ||||
| 				// lesson level (first children level) | ||||
| 				//var lessons = new MenuItemView({items:items[i].get('children')}).render(); | ||||
| 				var lessons=items[i].get('children'); | ||||
| 				if (lessons) { | ||||
| 					var categoryLessonList = $('<ul>',{class:'slideDown lessonsAndStages',id:catId}); //keepOpen | ||||
| 					for (var j=0; j < lessons.length;j++) { | ||||
| 						var lessonItem = $('<li>'); | ||||
| 						lessonName = lessons[j].name; | ||||
| 						var lessonLink = $('<a>',{href:lessons[j].link,text:lessonName,id:lessonName}); | ||||
| 						lessonLink.click(_.bind(this.triggerTitleRender,this,lessonName)); | ||||
| 						lessonItem.append(lessonLink); | ||||
| 						//check for lab/stages | ||||
| 						categoryLessonList.append(lessonItem); | ||||
| 						var stages = lessons[j].children; | ||||
| 						for (k=0; k < stages.length; k++) { | ||||
| 							var stageName = stages[k].name; | ||||
| 							var stageSpan = $('<span>'); | ||||
| 							var stageLink = $('<a>',{href:stages[k].link,text:stageName,id:GoatUtils.makeId(stageName)}); | ||||
| 							stageSpan.append(stageLink); | ||||
| 							categoryLessonList.append(stageSpan); | ||||
| 						} | ||||
| 					} | ||||
| 					category.append(categoryLessonList); | ||||
| 				} | ||||
|  | ||||
| 				menuUl.append(category); | ||||
| 			} | ||||
| 			this.$el.append(menuUl); | ||||
| 			//if we need to keep a menu open | ||||
| 			if (this.openMenu) { | ||||
| 				this.accordionMenu(this.openMenu); | ||||
| 			} | ||||
| 		}, | ||||
| 		triggerTitleRender: function (title) { | ||||
| 			this.trigger('lesson:click',title); | ||||
| 		}, | ||||
| 		expandCategory: function (id) { | ||||
| 			if (id) { | ||||
| 				this.accordionMenu(id); | ||||
| 			} | ||||
| 		}, | ||||
| 		accordionMenu: function(id) { | ||||
| 	        if (this.openMenu !== id) { | ||||
| 	        	this.$el.find('#' + id).slideDown(300); | ||||
| 	        	this.openMenu = id; | ||||
| 	        } else { //it's open | ||||
| 	            this.$el.find('#' + id).slideUp(300).attr('isOpen', 0); | ||||
| 	            this.openMenu = null; | ||||
| 	            return; | ||||
| 	        } | ||||
| 		} | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										23
									
								
								src/main/webapp/js/goatApp/view/PlanView.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								src/main/webapp/js/goatApp/view/PlanView.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,23 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/model/LessonPlanModel'], | ||||
| function($, | ||||
| 	_, | ||||
| 	Backbone, | ||||
| 	LessonPlanModel) { | ||||
| 	return Backbone.View.extend({ | ||||
| 		el:'#lessonHelpWrapper .lessonHelp.lessonPlan', //Check this | ||||
| 		initialize: function() { | ||||
| 			this.model = new LessonPlanModel(); | ||||
| 			this.listenTo(this.model,'loaded',this.onModelLoaded); | ||||
| 			this.model.loadData(); | ||||
| 		}, | ||||
| 		render:function(title) { | ||||
| 			 | ||||
| 		}, | ||||
| 		onModelLoaded: function() { | ||||
| 			this.trigger('plan:loaded',{'helpElement':'plan','value':true}); | ||||
| 		} | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										22
									
								
								src/main/webapp/js/goatApp/view/SolutionView.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/main/webapp/js/goatApp/view/SolutionView.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,22 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/model/LessonSolutionModel'], | ||||
| 	//TODO: create a base 'HelpView class' | ||||
| function($,_,Backbone,LessonSolutionModel) { | ||||
| 	return Backbone.View.extend({ | ||||
| 		el:'#lessonHelpWrapper .lessonHelp.lessonSolution', //Check this | ||||
| 		initialize: function() { | ||||
| 			this.model = new LessonSolutionModel(); | ||||
| 			this.listenTo(this.model,'loaded',this.onModelLoaded); | ||||
| 			this.model.loadData(); | ||||
| 			//TODO: handle error cases | ||||
| 		}, | ||||
| 		render:function(title) { | ||||
| 			 | ||||
| 		}, | ||||
| 		onModelLoaded: function() { | ||||
| 			this.trigger('solution:loaded',{'helpElement':'solution','value':true}); | ||||
| 		} | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										21
									
								
								src/main/webapp/js/goatApp/view/SourceView.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								src/main/webapp/js/goatApp/view/SourceView.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/model/LessonSourceModel', | ||||
| 	'goatApp/view/HelpView'], | ||||
| function($, | ||||
| 	_, | ||||
| 	Backbone, | ||||
| 	LessonSourceModel, | ||||
| 	HelpView) { | ||||
| 	return HelpView.extend({ | ||||
| 		helpElement:{'helpElement':'source','value':true}, | ||||
| 		loadedMessage:'source:loaded', | ||||
| 		el:'#lessonHelpWrapper .lessonHelp.lessonPlan', //Check this | ||||
| 		initialize: function() { | ||||
| 			this.model = new LessonSourceModel(); | ||||
| 			this.listenTo(this.model,'loaded',this.onModelLoaded); | ||||
| 			this.model.loadData(); | ||||
| 		} | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										13
									
								
								src/main/webapp/js/goatApp/view/TitleView.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								src/main/webapp/js/goatApp/view/TitleView.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,13 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone'], | ||||
| function($,_,Backbone) { | ||||
| 	return Backbone.View.extend({ | ||||
| 		el:'#header #lesson-title-wrapper', | ||||
| 		render:function(title) { | ||||
| 			var lessonTitleEl = $('<h1>',{id:'lesson-title',text:title}); | ||||
| 			this.$el.html(lessonTitleEl); | ||||
| 			//this.$el.append(lessonTitleEl); | ||||
| 		} | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										44
									
								
								src/main/webapp/js/goatApp/view/goatRouter.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								src/main/webapp/js/goatApp/view/goatRouter.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,44 @@ | ||||
| define(['jquery', | ||||
| 	'underscore', | ||||
| 	'backbone', | ||||
| 	'goatApp/controller/LessonController', | ||||
| 	'goatApp/controller/MenuController', | ||||
| 	'goatApp/view/LessonContentView', | ||||
| 	'goatApp/view/MenuView' | ||||
| 	], function ($,_,Backbone,LessonController,MenuController,LessonContentView,MenuView) { | ||||
|  | ||||
| 		var lessonView = new LessonContentView(); | ||||
| 		var menuView = new MenuView();  | ||||
| 		var GoatAppRouter = Backbone.Router.extend({ | ||||
| 			routes: { | ||||
| 				//#.... | ||||
| 				'welcome':'welcomeRoute', | ||||
| 				'attack/:scr/:menu':'attackRoute' //	 | ||||
| 			}, | ||||
| 			lessoonController: lessoonController({ | ||||
| 				lessonView:lessonView | ||||
| 			}), | ||||
| 			menuView: new MenuController({ | ||||
| 				menuView:menuView | ||||
| 			}) | ||||
| 		}); | ||||
|  | ||||
| 		var init = function() { | ||||
| 			goatRouter =  new GoatAppRouter(); | ||||
|  | ||||
| 			goatRouter.on('route:attackRoute', function(scr,menu) { | ||||
| 				this.lessonController.loadLesson(scr,menu); | ||||
| 				//update menu | ||||
| 			}); | ||||
| 			goatRouter.on('route:welcomeRoute', function() { | ||||
| 				alert('welcome route'); | ||||
| 			}); | ||||
| 			// init the history/router | ||||
| 			Backbone.history.start(); | ||||
| 		} | ||||
|  | ||||
| 		return { | ||||
| 			init:init | ||||
| 		}; | ||||
|  | ||||
| }); | ||||
| @ -1,189 +0,0 @@ | ||||
| goat.utils = { | ||||
|     //TODO add recursion to handle arr[i].children objects | ||||
|     // ... in case lower-level's need classes as well ... don't right now | ||||
|     addMenuClasses: function(arr) { | ||||
|         for (var i = 0; i < arr.length; i++) { | ||||
|             var menuItem = arr[i]; | ||||
|             //console.log(menuItem); | ||||
|             if (menuItem.type && menuItem.type === 'CATEGORY') { | ||||
|                 menuItem.class = 'fa-angle-right pull-right'; | ||||
|             } | ||||
|         } | ||||
|         return arr; | ||||
|     }, | ||||
|     debugFormSubmission: false, | ||||
|     // pre-submit callback  | ||||
|     showRequest: function(formData, jqForm, options) { | ||||
|         if (goat.utils.debugFormSubmission) { | ||||
|             // formData is an array; here we use $.param to convert it to a string to display it  | ||||
|             // but the form plugin does this for you automatically when it submits the data  | ||||
|             var queryString = $.param(formData); | ||||
|  | ||||
|             // jqForm is a jQuery object encapsulating the form element.  To access the  | ||||
|             // DOM element for the form do this:  | ||||
|             // var formElement = jqForm[0];  | ||||
|  | ||||
|             alert('About to submit: \n\n' + queryString); | ||||
|         } | ||||
|         // here we could return false to prevent the form from being submitted;  | ||||
|         // returning anything other than false will allow the form submit to continue  | ||||
|         return true; | ||||
|     }, | ||||
|     // post-submit callback  | ||||
|     showResponse: function(responseText, statusText, xhr, $form) { | ||||
|         // for normal html responses, the first argument to the success callback  | ||||
|         // is the XMLHttpRequest object's responseText property  | ||||
|  | ||||
|         // if the ajaxForm method was passed an Options Object with the dataType  | ||||
|         // property set to 'xml' then the first argument to the success callback  | ||||
|         // is the XMLHttpRequest object's responseXML property  | ||||
|  | ||||
|         // if the ajaxForm method was passed an Options Object with the dataType  | ||||
|         // property set to 'json' then the first argument to the success callback  | ||||
|         // is the json data object returned by the server  | ||||
|         if (goat.utils.debugFormSubmission) { | ||||
|             alert('status: ' + statusText + '\n\nresponseText: \n' + responseText + | ||||
|                     '\n\nThe output div should have already been updated with the responseText.'); | ||||
|         } | ||||
|         // update lesson cookies and params | ||||
|         // make any embedded forms ajaxy | ||||
|         goat.utils.showLessonCookiesAndParams(); | ||||
|         // forms and links are now hooked with each standard lesson render (see Java class Screen.getContent()) | ||||
|         // but these are safe to call twice | ||||
|         goat.utils.makeFormsAjax(); | ||||
|         goat.utils.ajaxifyAttackHref(); //TODO find some way to hook scope for current menu. Likely needs larger refactor which is already started/stashed | ||||
|         //refresh menu | ||||
|         angular.element($('#leftside-navigation')).scope().renderMenu(); | ||||
|     }, | ||||
|     makeFormsAjax: function() { | ||||
|         // make all forms ajax forms | ||||
|         var options = { | ||||
|             target: '#lesson_content', // target element(s) to be updated with server response                      | ||||
|             beforeSubmit: goat.utils.showRequest, // pre-submit callback, comment out after debugging  | ||||
|             success: goat.utils.showResponse  // post-submit callback, comment out after debugging  | ||||
|  | ||||
|                     // other available options:  | ||||
|                     //url:       url         // override for form's 'action' attribute  | ||||
|                     //type:      type        // 'get' or 'post', override for form's 'method' attribute  | ||||
|                     //dataType:  null        // 'xml', 'script', or 'json' (expected server response type)  | ||||
|                     //clearForm: true        // clear all form fields after successful submit  | ||||
|                     //resetForm: true        // reset the form after successful submit  | ||||
|  | ||||
|                     // $.ajax options can be used here too, for example:  | ||||
|                     //timeout:   3000  | ||||
|         }; | ||||
|         //console.log("Hooking any lesson forms to make them ajax"); | ||||
|         $("form").ajaxForm(options); | ||||
|     }, | ||||
|     displayButton: function(id, show) { | ||||
|         if ($('#' + id)) { | ||||
|             if (show) { | ||||
|                 $('#' + id).show(); | ||||
|             } else { | ||||
|                 $('#' + id).hide(); | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|     showLessonCookiesAndParams: function() { | ||||
|         $.get(goatConstants.cookieService, {}, function(reply) { | ||||
|             $("#lesson_cookies").html(reply); | ||||
|         }, "html"); | ||||
|     }, | ||||
|     showLessonHints: function() { | ||||
|         $('.lessonHelp').hide(); | ||||
|         $('#lesson_hint').html(); | ||||
|         $('#lesson_hint_row').show(); | ||||
|     }, | ||||
|     showLessonSource: function(source) { | ||||
|         $('.lessonHelp').hide(); | ||||
|         //$('#lesson_source').html("<pre>"+goat.lesson.lessonInfo.source+"</pre>"); | ||||
|         $('#lesson_source_row').show(); | ||||
|         goat.utils.scrollToHelp(); | ||||
|     }, | ||||
|     showLessonSolution: function() { | ||||
|         $('.lessonHelp').hide(); | ||||
|         $('#lesson_solution').html(goat.lesson.lessonInfo.solution); | ||||
|         $('#lesson_solution_row').show(); | ||||
|         goat.utils.scrollToHelp(); | ||||
|     }, | ||||
|     showLessonPlan: function(plan) { | ||||
|         $('.lessonHelp').hide(); | ||||
|         $("#lesson_plan").html(goat.lesson.lessonInfo.plan); | ||||
|         $('#lesson_plan_row').show(); | ||||
|         goat.utils.scrollToHelp(); | ||||
|     }, | ||||
|     scrollToHelp: function() { | ||||
|         $('#leftside-navigation').height($('#main-content').height() + 15) | ||||
|         var target = $('#lessonHelpsWrapper'); | ||||
|         goat.utils.scrollEasy(target); | ||||
|     }, | ||||
|     scrollToTop: function() { | ||||
|         $('.lessonHelp').hide(); | ||||
|         var target = $('#container'); | ||||
|         goat.utils.scrollEasy(target); | ||||
|     }, | ||||
|     scrollEasy: function(target) { | ||||
|         $('html,body').animate({ | ||||
|             scrollTop: target.offset().top | ||||
|         }, 1000); | ||||
|     }, | ||||
|     scrapeParams: function(url) { | ||||
|         if (!url) { | ||||
|             return; | ||||
|         } | ||||
|         var params = url.split('?')[1].split('&'); | ||||
|         var paramsArr = []; | ||||
|         for (var i = 0; i < params.length; i++) { | ||||
|             var paramObj = {}; | ||||
|             paramObj.name = params[i].split('=')[0]; | ||||
|             paramObj.value = params[i].split('=')[1]; | ||||
|             paramsArr.push(paramObj); | ||||
|         } | ||||
|         return paramsArr; | ||||
|     }, | ||||
|     highlightCurrentLessonMenu: function(id) { | ||||
|         //TODO: move selectors in first two lines into goatConstants | ||||
|         $('ul li.selected').removeClass(goatConstants.selectedMenuClass) | ||||
|         $('ul li.selected a.selected').removeClass(goatConstants.selectedMenuClass) | ||||
|         $('#' + id).addClass(goatConstants.selectedMenuClass); | ||||
|         $('#' + id).parent().addClass(goatConstants.selectedMenuClass); | ||||
|     }, | ||||
|     makeId: function(lessonName) { | ||||
|         return lessonName.replace(/\s|\(|\)|\!|\:|\;|\@|\#|\$|\%|\^|\&|\*/g, '');//TODO move the replace routine into util function | ||||
|     }, | ||||
|     ajaxifyAttackHref: function() { | ||||
|         /* Jason I commented this implementation out | ||||
|          * I think we should show the attack link on the lessons that need it by modifying the lesson | ||||
|          * itself or we could add a new button up top for "show lesson link"       | ||||
|          $.each($('a[href^="attack?"]'), | ||||
|          function(i,el) { | ||||
|          var url = $(el).attr('href'); | ||||
|          $(el).attr('href','#'); | ||||
|          $(el).attr('link',url); | ||||
|          //TODO pull currentMenuId | ||||
|          $(el).click( | ||||
|          function() { | ||||
|          var _url = $(el).attr('link'); | ||||
|          $.get(_url, {success:showResponse}); | ||||
|          } | ||||
|          ) | ||||
|          } | ||||
|          ); | ||||
|          */ | ||||
|         // alternate implementation | ||||
|         // unbind any bound events so we are safe to be called twice | ||||
|         $('#lesson_content a').unbind('click'); | ||||
|         $('#lesson_content a').bind('click', function(event) { | ||||
|             event.preventDefault(); | ||||
|             $.get(this.href, {}, function(response) { | ||||
|                 $('#lesson_content').html(response); | ||||
|             }); | ||||
|         }); | ||||
|     } | ||||
| }; | ||||
|  | ||||
|  | ||||
| $(window).resize(function() { | ||||
|     //$('#leftside-navigation').css('height',$('div.panel-body').height()); | ||||
|     console.log($(window).height()); | ||||
| }); | ||||
							
								
								
									
										2
									
								
								src/main/webapp/js/libs/backbone-min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								src/main/webapp/js/libs/backbone-min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										1608
									
								
								src/main/webapp/js/libs/backbone.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1608
									
								
								src/main/webapp/js/libs/backbone.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										6
									
								
								src/main/webapp/js/libs/jquery-1.10.2.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								src/main/webapp/js/libs/jquery-1.10.2.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										1277
									
								
								src/main/webapp/js/libs/jquery.form.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1277
									
								
								src/main/webapp/js/libs/jquery.form.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										2083
									
								
								src/main/webapp/js/libs/require.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2083
									
								
								src/main/webapp/js/libs/require.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										36
									
								
								src/main/webapp/js/libs/require.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								src/main/webapp/js/libs/require.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,36 @@ | ||||
| /* | ||||
|  RequireJS 2.1.16 Copyright (c) 2010-2015, The Dojo Foundation All Rights Reserved. | ||||
|  Available via the MIT or new BSD license. | ||||
|  see: http://github.com/jrburke/requirejs for details | ||||
| */ | ||||
| var requirejs,require,define; | ||||
| (function(ba){function G(b){return"[object Function]"===K.call(b)}function H(b){return"[object Array]"===K.call(b)}function v(b,c){if(b){var d;for(d=0;d<b.length&&(!b[d]||!c(b[d],d,b));d+=1);}}function T(b,c){if(b){var d;for(d=b.length-1;-1<d&&(!b[d]||!c(b[d],d,b));d-=1);}}function t(b,c){return fa.call(b,c)}function m(b,c){return t(b,c)&&b[c]}function B(b,c){for(var d in b)if(t(b,d)&&c(b[d],d))break}function U(b,c,d,e){c&&B(c,function(c,g){if(d||!t(b,g))e&&"object"===typeof c&&c&&!H(c)&&!G(c)&&!(c instanceof | ||||
| RegExp)?(b[g]||(b[g]={}),U(b[g],c,d,e)):b[g]=c});return b}function u(b,c){return function(){return c.apply(b,arguments)}}function ca(b){throw b;}function da(b){if(!b)return b;var c=ba;v(b.split("."),function(b){c=c[b]});return c}function C(b,c,d,e){c=Error(c+"\nhttp://requirejs.org/docs/errors.html#"+b);c.requireType=b;c.requireModules=e;d&&(c.originalError=d);return c}function ga(b){function c(a,k,b){var f,l,c,d,e,g,i,p,k=k&&k.split("/"),h=j.map,n=h&&h["*"];if(a){a=a.split("/");l=a.length-1;j.nodeIdCompat&& | ||||
| Q.test(a[l])&&(a[l]=a[l].replace(Q,""));"."===a[0].charAt(0)&&k&&(l=k.slice(0,k.length-1),a=l.concat(a));l=a;for(c=0;c<l.length;c++)if(d=l[c],"."===d)l.splice(c,1),c-=1;else if(".."===d&&!(0===c||1==c&&".."===l[2]||".."===l[c-1])&&0<c)l.splice(c-1,2),c-=2;a=a.join("/")}if(b&&h&&(k||n)){l=a.split("/");c=l.length;a:for(;0<c;c-=1){e=l.slice(0,c).join("/");if(k)for(d=k.length;0<d;d-=1)if(b=m(h,k.slice(0,d).join("/")))if(b=m(b,e)){f=b;g=c;break a}!i&&(n&&m(n,e))&&(i=m(n,e),p=c)}!f&&i&&(f=i,g=p);f&&(l.splice(0, | ||||
| g,f),a=l.join("/"))}return(f=m(j.pkgs,a))?f:a}function d(a){z&&v(document.getElementsByTagName("script"),function(k){if(k.getAttribute("data-requiremodule")===a&&k.getAttribute("data-requirecontext")===i.contextName)return k.parentNode.removeChild(k),!0})}function e(a){var k=m(j.paths,a);if(k&&H(k)&&1<k.length)return k.shift(),i.require.undef(a),i.makeRequire(null,{skipMap:!0})([a]),!0}function n(a){var k,c=a?a.indexOf("!"):-1;-1<c&&(k=a.substring(0,c),a=a.substring(c+1,a.length));return[k,a]}function p(a, | ||||
| k,b,f){var l,d,e=null,g=k?k.name:null,j=a,p=!0,h="";a||(p=!1,a="_@r"+(K+=1));a=n(a);e=a[0];a=a[1];e&&(e=c(e,g,f),d=m(r,e));a&&(e?h=d&&d.normalize?d.normalize(a,function(a){return c(a,g,f)}):-1===a.indexOf("!")?c(a,g,f):a:(h=c(a,g,f),a=n(h),e=a[0],h=a[1],b=!0,l=i.nameToUrl(h)));b=e&&!d&&!b?"_unnormalized"+(O+=1):"";return{prefix:e,name:h,parentMap:k,unnormalized:!!b,url:l,originalName:j,isDefine:p,id:(e?e+"!"+h:h)+b}}function s(a){var k=a.id,b=m(h,k);b||(b=h[k]=new i.Module(a));return b}function q(a, | ||||
| k,b){var f=a.id,c=m(h,f);if(t(r,f)&&(!c||c.defineEmitComplete))"defined"===k&&b(r[f]);else if(c=s(a),c.error&&"error"===k)b(c.error);else c.on(k,b)}function w(a,b){var c=a.requireModules,f=!1;if(b)b(a);else if(v(c,function(b){if(b=m(h,b))b.error=a,b.events.error&&(f=!0,b.emit("error",a))}),!f)g.onError(a)}function x(){R.length&&(ha.apply(A,[A.length,0].concat(R)),R=[])}function y(a){delete h[a];delete V[a]}function F(a,b,c){var f=a.map.id;a.error?a.emit("error",a.error):(b[f]=!0,v(a.depMaps,function(f, | ||||
| d){var e=f.id,g=m(h,e);g&&(!a.depMatched[d]&&!c[e])&&(m(b,e)?(a.defineDep(d,r[e]),a.check()):F(g,b,c))}),c[f]=!0)}function D(){var a,b,c=(a=1E3*j.waitSeconds)&&i.startTime+a<(new Date).getTime(),f=[],l=[],g=!1,h=!0;if(!W){W=!0;B(V,function(a){var i=a.map,j=i.id;if(a.enabled&&(i.isDefine||l.push(a),!a.error))if(!a.inited&&c)e(j)?g=b=!0:(f.push(j),d(j));else if(!a.inited&&(a.fetched&&i.isDefine)&&(g=!0,!i.prefix))return h=!1});if(c&&f.length)return a=C("timeout","Load timeout for modules: "+f,null, | ||||
| f),a.contextName=i.contextName,w(a);h&&v(l,function(a){F(a,{},{})});if((!c||b)&&g)if((z||ea)&&!X)X=setTimeout(function(){X=0;D()},50);W=!1}}function E(a){t(r,a[0])||s(p(a[0],null,!0)).init(a[1],a[2])}function I(a){var a=a.currentTarget||a.srcElement,b=i.onScriptLoad;a.detachEvent&&!Y?a.detachEvent("onreadystatechange",b):a.removeEventListener("load",b,!1);b=i.onScriptError;(!a.detachEvent||Y)&&a.removeEventListener("error",b,!1);return{node:a,id:a&&a.getAttribute("data-requiremodule")}}function J(){var a; | ||||
| for(x();A.length;){a=A.shift();if(null===a[0])return w(C("mismatch","Mismatched anonymous define() module: "+a[a.length-1]));E(a)}}var W,Z,i,L,X,j={waitSeconds:7,baseUrl:"./",paths:{},bundles:{},pkgs:{},shim:{},config:{}},h={},V={},$={},A=[],r={},S={},aa={},K=1,O=1;L={require:function(a){return a.require?a.require:a.require=i.makeRequire(a.map)},exports:function(a){a.usingExports=!0;if(a.map.isDefine)return a.exports?r[a.map.id]=a.exports:a.exports=r[a.map.id]={}},module:function(a){return a.module? | ||||
| a.module:a.module={id:a.map.id,uri:a.map.url,config:function(){return m(j.config,a.map.id)||{}},exports:a.exports||(a.exports={})}}};Z=function(a){this.events=m($,a.id)||{};this.map=a;this.shim=m(j.shim,a.id);this.depExports=[];this.depMaps=[];this.depMatched=[];this.pluginMaps={};this.depCount=0};Z.prototype={init:function(a,b,c,f){f=f||{};if(!this.inited){this.factory=b;if(c)this.on("error",c);else this.events.error&&(c=u(this,function(a){this.emit("error",a)}));this.depMaps=a&&a.slice(0);this.errback= | ||||
| c;this.inited=!0;this.ignore=f.ignore;f.enabled||this.enabled?this.enable():this.check()}},defineDep:function(a,b){this.depMatched[a]||(this.depMatched[a]=!0,this.depCount-=1,this.depExports[a]=b)},fetch:function(){if(!this.fetched){this.fetched=!0;i.startTime=(new Date).getTime();var a=this.map;if(this.shim)i.makeRequire(this.map,{enableBuildCallback:!0})(this.shim.deps||[],u(this,function(){return a.prefix?this.callPlugin():this.load()}));else return a.prefix?this.callPlugin():this.load()}},load:function(){var a= | ||||
| this.map.url;S[a]||(S[a]=!0,i.load(this.map.id,a))},check:function(){if(this.enabled&&!this.enabling){var a,b,c=this.map.id;b=this.depExports;var f=this.exports,l=this.factory;if(this.inited)if(this.error)this.emit("error",this.error);else{if(!this.defining){this.defining=!0;if(1>this.depCount&&!this.defined){if(G(l)){if(this.events.error&&this.map.isDefine||g.onError!==ca)try{f=i.execCb(c,l,b,f)}catch(d){a=d}else f=i.execCb(c,l,b,f);this.map.isDefine&&void 0===f&&((b=this.module)?f=b.exports:this.usingExports&& | ||||
| (f=this.exports));if(a)return a.requireMap=this.map,a.requireModules=this.map.isDefine?[this.map.id]:null,a.requireType=this.map.isDefine?"define":"require",w(this.error=a)}else f=l;this.exports=f;if(this.map.isDefine&&!this.ignore&&(r[c]=f,g.onResourceLoad))g.onResourceLoad(i,this.map,this.depMaps);y(c);this.defined=!0}this.defining=!1;this.defined&&!this.defineEmitted&&(this.defineEmitted=!0,this.emit("defined",this.exports),this.defineEmitComplete=!0)}}else this.fetch()}},callPlugin:function(){var a= | ||||
| this.map,b=a.id,d=p(a.prefix);this.depMaps.push(d);q(d,"defined",u(this,function(f){var l,d;d=m(aa,this.map.id);var e=this.map.name,P=this.map.parentMap?this.map.parentMap.name:null,n=i.makeRequire(a.parentMap,{enableBuildCallback:!0});if(this.map.unnormalized){if(f.normalize&&(e=f.normalize(e,function(a){return c(a,P,!0)})||""),f=p(a.prefix+"!"+e,this.map.parentMap),q(f,"defined",u(this,function(a){this.init([],function(){return a},null,{enabled:!0,ignore:!0})})),d=m(h,f.id)){this.depMaps.push(f); | ||||
| if(this.events.error)d.on("error",u(this,function(a){this.emit("error",a)}));d.enable()}}else d?(this.map.url=i.nameToUrl(d),this.load()):(l=u(this,function(a){this.init([],function(){return a},null,{enabled:!0})}),l.error=u(this,function(a){this.inited=!0;this.error=a;a.requireModules=[b];B(h,function(a){0===a.map.id.indexOf(b+"_unnormalized")&&y(a.map.id)});w(a)}),l.fromText=u(this,function(f,c){var d=a.name,e=p(d),P=M;c&&(f=c);P&&(M=!1);s(e);t(j.config,b)&&(j.config[d]=j.config[b]);try{g.exec(f)}catch(h){return w(C("fromtexteval", | ||||
| "fromText eval for "+b+" failed: "+h,h,[b]))}P&&(M=!0);this.depMaps.push(e);i.completeLoad(d);n([d],l)}),f.load(a.name,n,l,j))}));i.enable(d,this);this.pluginMaps[d.id]=d},enable:function(){V[this.map.id]=this;this.enabling=this.enabled=!0;v(this.depMaps,u(this,function(a,b){var c,f;if("string"===typeof a){a=p(a,this.map.isDefine?this.map:this.map.parentMap,!1,!this.skipMap);this.depMaps[b]=a;if(c=m(L,a.id)){this.depExports[b]=c(this);return}this.depCount+=1;q(a,"defined",u(this,function(a){this.defineDep(b, | ||||
| a);this.check()}));this.errback?q(a,"error",u(this,this.errback)):this.events.error&&q(a,"error",u(this,function(a){this.emit("error",a)}))}c=a.id;f=h[c];!t(L,c)&&(f&&!f.enabled)&&i.enable(a,this)}));B(this.pluginMaps,u(this,function(a){var b=m(h,a.id);b&&!b.enabled&&i.enable(a,this)}));this.enabling=!1;this.check()},on:function(a,b){var c=this.events[a];c||(c=this.events[a]=[]);c.push(b)},emit:function(a,b){v(this.events[a],function(a){a(b)});"error"===a&&delete this.events[a]}};i={config:j,contextName:b, | ||||
| registry:h,defined:r,urlFetched:S,defQueue:A,Module:Z,makeModuleMap:p,nextTick:g.nextTick,onError:w,configure:function(a){a.baseUrl&&"/"!==a.baseUrl.charAt(a.baseUrl.length-1)&&(a.baseUrl+="/");var b=j.shim,c={paths:!0,bundles:!0,config:!0,map:!0};B(a,function(a,b){c[b]?(j[b]||(j[b]={}),U(j[b],a,!0,!0)):j[b]=a});a.bundles&&B(a.bundles,function(a,b){v(a,function(a){a!==b&&(aa[a]=b)})});a.shim&&(B(a.shim,function(a,c){H(a)&&(a={deps:a});if((a.exports||a.init)&&!a.exportsFn)a.exportsFn=i.makeShimExports(a); | ||||
| b[c]=a}),j.shim=b);a.packages&&v(a.packages,function(a){var b,a="string"===typeof a?{name:a}:a;b=a.name;a.location&&(j.paths[b]=a.location);j.pkgs[b]=a.name+"/"+(a.main||"main").replace(ia,"").replace(Q,"")});B(h,function(a,b){!a.inited&&!a.map.unnormalized&&(a.map=p(b))});if(a.deps||a.callback)i.require(a.deps||[],a.callback)},makeShimExports:function(a){return function(){var b;a.init&&(b=a.init.apply(ba,arguments));return b||a.exports&&da(a.exports)}},makeRequire:function(a,e){function j(c,d,m){var n, | ||||
| q;e.enableBuildCallback&&(d&&G(d))&&(d.__requireJsBuild=!0);if("string"===typeof c){if(G(d))return w(C("requireargs","Invalid require call"),m);if(a&&t(L,c))return L[c](h[a.id]);if(g.get)return g.get(i,c,a,j);n=p(c,a,!1,!0);n=n.id;return!t(r,n)?w(C("notloaded",'Module name "'+n+'" has not been loaded yet for context: '+b+(a?"":". Use require([])"))):r[n]}J();i.nextTick(function(){J();q=s(p(null,a));q.skipMap=e.skipMap;q.init(c,d,m,{enabled:!0});D()});return j}e=e||{};U(j,{isBrowser:z,toUrl:function(b){var d, | ||||
| e=b.lastIndexOf("."),k=b.split("/")[0];if(-1!==e&&(!("."===k||".."===k)||1<e))d=b.substring(e,b.length),b=b.substring(0,e);return i.nameToUrl(c(b,a&&a.id,!0),d,!0)},defined:function(b){return t(r,p(b,a,!1,!0).id)},specified:function(b){b=p(b,a,!1,!0).id;return t(r,b)||t(h,b)}});a||(j.undef=function(b){x();var c=p(b,a,!0),e=m(h,b);d(b);delete r[b];delete S[c.url];delete $[b];T(A,function(a,c){a[0]===b&&A.splice(c,1)});e&&(e.events.defined&&($[b]=e.events),y(b))});return j},enable:function(a){m(h,a.id)&& | ||||
| s(a).enable()},completeLoad:function(a){var b,c,d=m(j.shim,a)||{},g=d.exports;for(x();A.length;){c=A.shift();if(null===c[0]){c[0]=a;if(b)break;b=!0}else c[0]===a&&(b=!0);E(c)}c=m(h,a);if(!b&&!t(r,a)&&c&&!c.inited){if(j.enforceDefine&&(!g||!da(g)))return e(a)?void 0:w(C("nodefine","No define call for "+a,null,[a]));E([a,d.deps||[],d.exportsFn])}D()},nameToUrl:function(a,b,c){var d,e,h;(d=m(j.pkgs,a))&&(a=d);if(d=m(aa,a))return i.nameToUrl(d,b,c);if(g.jsExtRegExp.test(a))d=a+(b||"");else{d=j.paths; | ||||
| a=a.split("/");for(e=a.length;0<e;e-=1)if(h=a.slice(0,e).join("/"),h=m(d,h)){H(h)&&(h=h[0]);a.splice(0,e,h);break}d=a.join("/");d+=b||(/^data\:|\?/.test(d)||c?"":".js");d=("/"===d.charAt(0)||d.match(/^[\w\+\.\-]+:/)?"":j.baseUrl)+d}return j.urlArgs?d+((-1===d.indexOf("?")?"?":"&")+j.urlArgs):d},load:function(a,b){g.load(i,a,b)},execCb:function(a,b,c,d){return b.apply(d,c)},onScriptLoad:function(a){if("load"===a.type||ja.test((a.currentTarget||a.srcElement).readyState))N=null,a=I(a),i.completeLoad(a.id)}, | ||||
| onScriptError:function(a){var b=I(a);if(!e(b.id))return w(C("scripterror","Script error for: "+b.id,a,[b.id]))}};i.require=i.makeRequire();return i}var g,x,y,D,I,E,N,J,s,O,ka=/(\/\*([\s\S]*?)\*\/|([^:]|^)\/\/(.*)$)/mg,la=/[^.]\s*require\s*\(\s*["']([^'"\s]+)["']\s*\)/g,Q=/\.js$/,ia=/^\.\//;x=Object.prototype;var K=x.toString,fa=x.hasOwnProperty,ha=Array.prototype.splice,z=!!("undefined"!==typeof window&&"undefined"!==typeof navigator&&window.document),ea=!z&&"undefined"!==typeof importScripts,ja= | ||||
| z&&"PLAYSTATION 3"===navigator.platform?/^complete$/:/^(complete|loaded)$/,Y="undefined"!==typeof opera&&"[object Opera]"===opera.toString(),F={},q={},R=[],M=!1;if("undefined"===typeof define){if("undefined"!==typeof requirejs){if(G(requirejs))return;q=requirejs;requirejs=void 0}"undefined"!==typeof require&&!G(require)&&(q=require,require=void 0);g=requirejs=function(b,c,d,e){var n,p="_";!H(b)&&"string"!==typeof b&&(n=b,H(c)?(b=c,c=d,d=e):b=[]);n&&n.context&&(p=n.context);(e=m(F,p))||(e=F[p]=g.s.newContext(p)); | ||||
| n&&e.configure(n);return e.require(b,c,d)};g.config=function(b){return g(b)};g.nextTick="undefined"!==typeof setTimeout?function(b){setTimeout(b,4)}:function(b){b()};require||(require=g);g.version="2.1.16";g.jsExtRegExp=/^\/|:|\?|\.js$/;g.isBrowser=z;x=g.s={contexts:F,newContext:ga};g({});v(["toUrl","undef","defined","specified"],function(b){g[b]=function(){var c=F._;return c.require[b].apply(c,arguments)}});if(z&&(y=x.head=document.getElementsByTagName("head")[0],D=document.getElementsByTagName("base")[0]))y= | ||||
| x.head=D.parentNode;g.onError=ca;g.createNode=function(b){var c=b.xhtml?document.createElementNS("http://www.w3.org/1999/xhtml","html:script"):document.createElement("script");c.type=b.scriptType||"text/javascript";c.charset="utf-8";c.async=!0;return c};g.load=function(b,c,d){var e=b&&b.config||{};if(z)return e=g.createNode(e,c,d),e.setAttribute("data-requirecontext",b.contextName),e.setAttribute("data-requiremodule",c),e.attachEvent&&!(e.attachEvent.toString&&0>e.attachEvent.toString().indexOf("[native code"))&& | ||||
| !Y?(M=!0,e.attachEvent("onreadystatechange",b.onScriptLoad)):(e.addEventListener("load",b.onScriptLoad,!1),e.addEventListener("error",b.onScriptError,!1)),e.src=d,J=e,D?y.insertBefore(e,D):y.appendChild(e),J=null,e;if(ea)try{importScripts(d),b.completeLoad(c)}catch(m){b.onError(C("importscripts","importScripts failed for "+c+" at "+d,m,[c]))}};z&&!q.skipDataMain&&T(document.getElementsByTagName("script"),function(b){y||(y=b.parentNode);if(I=b.getAttribute("data-main"))return s=I,q.baseUrl||(E=s.split("/"), | ||||
| s=E.pop(),O=E.length?E.join("/")+"/":"./",q.baseUrl=O),s=s.replace(Q,""),g.jsExtRegExp.test(s)&&(s=I),q.deps=q.deps?q.deps.concat(s):[s],!0});define=function(b,c,d){var e,g;"string"!==typeof b&&(d=c,c=b,b=null);H(c)||(d=c,c=null);!c&&G(d)&&(c=[],d.length&&(d.toString().replace(ka,"").replace(la,function(b,d){c.push(d)}),c=(1===d.length?["require"]:["require","exports","module"]).concat(c)));if(M){if(!(e=J))N&&"interactive"===N.readyState||T(document.getElementsByTagName("script"),function(b){if("interactive"=== | ||||
| b.readyState)return N=b}),e=N;e&&(b||(b=e.getAttribute("data-requiremodule")),g=F[e.getAttribute("data-requirecontext")])}(g?g.defQueue:R).push([b,c,d])};define.amd={jQuery:!0};g.exec=function(b){return eval(b)};g(q)}})(this); | ||||
							
								
								
									
										6
									
								
								src/main/webapp/js/libs/underscore-min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								src/main/webapp/js/libs/underscore-min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										1416
									
								
								src/main/webapp/js/libs/underscore.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1416
									
								
								src/main/webapp/js/libs/underscore.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										34
									
								
								src/main/webapp/js/main.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								src/main/webapp/js/main.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,34 @@ | ||||
| //main.js | ||||
| /* | ||||
| /js | ||||
| js/main.js << main file for require.js | ||||
| --/libs/(jquery,backbone,etc.) << base libs | ||||
| --/goatApp/ << base dir for goat application, js-wise | ||||
| --/goatApp/model | ||||
| --/goatApp/view | ||||
| --/goatApp/support | ||||
| --/goatApp/controller | ||||
| */ | ||||
|  | ||||
| require.config({ | ||||
|   baseUrl: "js/", | ||||
|   paths: { | ||||
|     jquery: 'libs/jquery-1.10.2.min', | ||||
|     underscore: 'libs/underscore-min', | ||||
|     backbone: 'libs/backbone-min' | ||||
|   } | ||||
| , | ||||
|   shim: { | ||||
|     underscore: { | ||||
|       exports: "_" | ||||
|     }, | ||||
|     backbone: { | ||||
|       deps: ['underscore', 'jquery'], | ||||
|       exports: 'Backbone' | ||||
|     } | ||||
|   } | ||||
| }); | ||||
|  | ||||
| require(['jquery','underscore','backbone','goatApp/goatApp'], function($,_,Backbone,Goat){ | ||||
|   Goat.initApp(); | ||||
| }); | ||||
		Reference in New Issue
	
	Block a user