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 | **********          WebGoat 6.0 | ||||||
| **********          August 23, 2014 | **********          August 23, 2014 | ||||||
|  | |||||||
| @ -551,7 +551,7 @@ public abstract class AbstractLesson extends Screen implements Comparable<Object | |||||||
|      * return "*.do"-styled paths.</p> |      * return "*.do"-styled paths.</p> | ||||||
|      */ |      */ | ||||||
|     protected String getPath() { |     protected String getPath() { | ||||||
|         return "attack"; |         return "#attack"; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @ -563,13 +563,9 @@ public abstract class AbstractLesson extends Screen implements Comparable<Object | |||||||
|         StringBuffer link = new StringBuffer(); |         StringBuffer link = new StringBuffer(); | ||||||
|  |  | ||||||
|         // mvc update: |         // mvc update: | ||||||
|         link.append(getPath()).append("?"); |         link.append(getPath()).append("/"); | ||||||
|         link.append(WebSession.SCREEN); |  | ||||||
|         link.append("="); |  | ||||||
|         link.append(getScreenId()); |         link.append(getScreenId()); | ||||||
|         link.append("&"); |         link.append("/"); | ||||||
|         link.append(WebSession.MENU); |  | ||||||
|         link.append("="); |  | ||||||
|         link.append(getCategory().getRanking()); |         link.append(getCategory().getRanking()); | ||||||
|         return link.toString(); |         return link.toString(); | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -198,9 +198,9 @@ public abstract class Screen { | |||||||
|  |  | ||||||
|     // hook all the links |     // hook all the links | ||||||
|     public String getContent() { |     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 |         // 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><!--toggle navigation end--> | ||||||
|             <div class="lessonTitle" > |             <div class="lessonTitle" > | ||||||
|                 <h1 id="lessonTitle">Please login</h1> |                  | ||||||
|             </div><!--lesson title end--> |             </div><!--lesson title end--> | ||||||
|  |  | ||||||
|         </header> |         </header> | ||||||
|  | |||||||
| @ -27,12 +27,7 @@ | |||||||
|         <!--  end of CSS --> |         <!--  end of CSS --> | ||||||
|  |  | ||||||
|         <!-- JS --> |         <!-- 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> |         <script src="js/modernizr-2.6.2.min.js"></script> | ||||||
|         <!-- HTML5 shim and Respond.js IE8 support of HTML5 elements and media queries --> |         <!-- HTML5 shim and Respond.js IE8 support of HTML5 elements and media queries --> | ||||||
|         <!--[if lt IE 9]> |         <!--[if lt IE 9]> | ||||||
| @ -40,30 +35,14 @@ | |||||||
|         <script src="js/respond.min.js"></script> |         <script src="js/respond.min.js"></script> | ||||||
|         <![endif]--> |         <![endif]--> | ||||||
|  |  | ||||||
|         <!--Global JS--> |         <!-- Require.js used to load js asynchronously --> | ||||||
|  |         <script src="js/libs/require.min.js" data-main="js/main.js"></script> | ||||||
|         <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 --> |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|         <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1" /> |         <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1" /> | ||||||
|         <title>WebGoat</title> |         <title>WebGoat</title> | ||||||
|     </head> |     </head> | ||||||
|  |     <body> | ||||||
|     <body class="animated fadeIn" ng-app="goatApp"> |         <section id="container"> | ||||||
|         <section id="container" ng-controller="goatLesson"> |  | ||||||
|             <header id="header"> |             <header id="header"> | ||||||
|                 <!--logo start--> |                 <!--logo start--> | ||||||
|                 <div class="brand"> |                 <div class="brand"> | ||||||
| @ -75,12 +54,12 @@ | |||||||
|                         <i class="fa fa-bars"></i> |                         <i class="fa fa-bars"></i> | ||||||
|                     </button> |                     </button> | ||||||
|                 </div><!--toggle navigation end--> |                 </div><!--toggle navigation end--> | ||||||
|                 <div class="lessonTitle" > |                 <div id="lesson-title-wrapper" > | ||||||
|                     <h1 id="lessonTitle"></h1> |                      | ||||||
|                 </div><!--lesson title end--> |                 </div><!--lesson title end--> | ||||||
|                 <div class="user-nav pull-right" style="margin-right: 75px;"> |                 <div class="user-nav pull-right" style="margin-right: 75px;"> | ||||||
|                     <div class="dropdown" style="display:inline"> |                     <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> |                             <i class="fa fa-user"></i> <span class="caret"></span> | ||||||
|                         </button>                    |                         </button>                    | ||||||
|                         <ul class="dropdown-menu dropdown-menu-left" role="menu" aria-labelledby="dropdownMenu1"> |                         <ul class="dropdown-menu dropdown-menu-left" role="menu" aria-labelledby="dropdownMenu1"> | ||||||
| @ -94,7 +73,7 @@ | |||||||
|  |  | ||||||
|                         </ul> |                         </ul> | ||||||
|                     </div> |                     </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> |                         <i class="fa fa-info"></i> | ||||||
|                     </button> |                     </button> | ||||||
|                     <a href="mailto:${contactEmail}?Subject=Webgoat%20feedback" target="_top"> |                     <a href="mailto:${contactEmail}?Subject=Webgoat%20feedback" target="_top"> | ||||||
| @ -107,47 +86,32 @@ | |||||||
|                 </div> |                 </div> | ||||||
|             </header> |             </header> | ||||||
|  |  | ||||||
|             <!--sidebar left start--> |  | ||||||
|             <aside class="sidebar" > |             <aside class="sidebar" > | ||||||
|                 <div id="leftside-navigation" ng-controller="goatMenu" class="nano"> |                 <div id="menu-container"></div> | ||||||
|                     <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> |  | ||||||
|  |  | ||||||
|             </aside> |             </aside> | ||||||
|             <!--sidebar left end--> |             <!--sidebar left end--> | ||||||
|  |  | ||||||
|             <!--main content start--> |             <!--main content start--> | ||||||
|             <section class="main-content-wrapper"> |             <section class="main-content-wrapper"> | ||||||
|                 <section id="main-content" > <!--ng-controller="goatLesson"--> |                 <section id="main-content" > <!--ng-controller="goatLesson"--> | ||||||
|                     <div class="row"> |                     <div class="row"> | ||||||
|                         <div class="col-md-8"> |                         <div class="col-md-8"> | ||||||
|                             <div class="col-md-12" align="left"> |                             <div class="col-md-12" align="left"> | ||||||
|                                 <div class="panel"> |                                 <div class="panel" id="help-controls"> | ||||||
|                                     <div class="panel-body"> |                                     <!-- <div id="help-buttons" 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="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" ng-click="showLessonSolution()">Solution</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" ng-click="showLessonPlan()">Lesson Plan</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"  ng-click="viewHints()">Hints</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"  ng-click="restartLesson()">Restart Lesson</button> |                                         <button type="button" id="restartLessonBtn"  class="btn btn-xs"  onclick="restartLesson()">Restart Lesson</button> --> | ||||||
|                                     </div> |                                     <!-- </div> --> | ||||||
|                                 </div> |                                 </div> | ||||||
|                                 <div class="lessonHelp" id="lesson_hint_row"> |                                 <div class="lessonHelp" id="lesson-hint-row"> | ||||||
|                                     <h4>Hints</h4> |                                     <h4>Hints</h4> | ||||||
|                                     <div class="panel" > |                                     <div class="panel" > | ||||||
|                                         <div class="panel-body" id="lesson_hint"> |                                         <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-left" id="showPrevHintBtn" onclick="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-right" id="showNextHintBtn" onclick="viewNextHint()"></span> | ||||||
|                                             <br/> |                                             <br/> | ||||||
|                                             <span ng-show="showHints" bind-html-unsafe="curHint"></span> |                                             <span ng-show="showHints" bind-html-unsafe="curHint"></span> | ||||||
|                                             <!--<span id="curHintContainer"></span>--> |                                             <!--<span id="curHintContainer"></span>--> | ||||||
| @ -155,12 +119,8 @@ | |||||||
|                                     </div> |                                     </div> | ||||||
|                                 </div> |                                 </div> | ||||||
|                             </div> |                             </div> | ||||||
|  |                             <div class="col-md-12" align="left"> | ||||||
|                             <div class="col-md-12"> |                                 <div id="lessonContentWrapper" class="panel"> | ||||||
|                                 <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> |                                 </div> | ||||||
|                             </div> |                             </div> | ||||||
| @ -176,10 +136,10 @@ | |||||||
|                                         <div id="cookiesAndParamsView"> |                                         <div id="cookiesAndParamsView"> | ||||||
|                                             <div class="cookiesView"> |                                             <div class="cookiesView"> | ||||||
|                                                 <h4>Cookies</h4> |                                                 <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" > |                                                     <table class="cookieTable table-striped table-nonfluid" > | ||||||
|                                                         <thead> |                                                         <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> |                                                         </thead> | ||||||
|                                                         <tbody> |                                                         <tbody> | ||||||
|                                                             <tr ng-repeat="(key, value) in cookie"> |                                                             <tr ng-repeat="(key, value) in cookie"> | ||||||
| @ -187,25 +147,11 @@ | |||||||
|                                                                 <td>{{value}}</td> |                                                                 <td>{{value}}</td> | ||||||
|                                                             </tr> |                                                             </tr> | ||||||
|                                                         </tbody> |                                                         </tbody> | ||||||
|                                                         <!--<li ng-repeat="(key, value) in cookie">{{key}} :: {{ value}} </td>--> |  | ||||||
|                                                         <!--</ul>--> |  | ||||||
|                                                     </table> |                                                     </table> | ||||||
|                                                 </div> |                                                 </div> --> | ||||||
|                                             </div> |                                             </div> | ||||||
|                                             <div id="paramsView"> <!--class="paramsView"--> |                                             <div id="paramsView"> <!--class="paramsView"--> | ||||||
|                                                 <h4>Params</h4> |                                                 <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> |                                         </div> | ||||||
|                                     </div> |                                     </div> | ||||||
| @ -214,44 +160,18 @@ | |||||||
|                         </div><!--col-md-4 end-->          |                         </div><!--col-md-4 end-->          | ||||||
|                     </div> |                     </div> | ||||||
|                     <div id="lessonHelpsWrapper"> |                     <div id="lessonHelpsWrapper"> | ||||||
|                         <!--div class="row lessonHelp" id="lesson_hint_row"> |                         <div class="row lessonHelp" id="lesson-cookies-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="col-md-12"> |                             <div class="col-md-12"> | ||||||
|                                 <h4>Lesson Parameters and Cookies</h4> |                                 <h4>Lesson Parameters and Cookies</h4> | ||||||
|                                 <div class="panel" > |                                  | ||||||
|                                     <div class="panel-body" id="lesson_cookies">	 |  | ||||||
|  |  | ||||||
|                                     </div>                                     |  | ||||||
|                                 </div> |  | ||||||
|                             </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 class="row lessonHelp" id="lesson-plan-row"> | ||||||
|                                 </div> |  | ||||||
|                             </div> |  | ||||||
|                         </div>                  |  | ||||||
|                         <div class="row lessonHelp" id="lesson_plan_row"> |  | ||||||
|                             <div class="col-md-12"> |                             <div class="col-md-12"> | ||||||
|                                 <h4>Lesson Plan</h4> |                                 <h4>Lesson Plan</h4> | ||||||
|                                 <div class="panel" > |                                 <div class="panel" > | ||||||
|                                     <div class="panel-body" id="lesson_plan"> |                                     <div class="panel-body" id="lesson-plan"> | ||||||
|                                         <!-- allowing jQuery to handle this one --> |                                         <!-- allowing jQuery to handle this one --> | ||||||
|                                     </div>                                     |                                     </div>                                     | ||||||
|                                 </div> |                                 </div> | ||||||
| @ -271,7 +191,7 @@ | |||||||
|                                 <h4>Lesson Source Code</h4> |                                 <h4>Lesson Source Code</h4> | ||||||
|                                 <div class="panel"> |                                 <div class="panel"> | ||||||
|                                     <div class="panel-body" id="lesson_source"> |                                     <div class="panel-body" id="lesson_source"> | ||||||
|                                         <pre>{{source}}</pre> |                                         <!-- <pre>{{source}}</pre> --> | ||||||
|                                     </div>                                     |                                     </div>                                     | ||||||
|                                 </div> |                                 </div> | ||||||
|                             </div> |                             </div> | ||||||
| @ -280,20 +200,8 @@ | |||||||
|                 </section> |                 </section> | ||||||
|             </section> |             </section> | ||||||
|  |  | ||||||
|             <!--main content end--> |  | ||||||
|  |  | ||||||
|         </section> |         </section> | ||||||
|  |  | ||||||
|         <!--main content end--> |  | ||||||
|  |  | ||||||
|         </section> |  | ||||||
|  |  | ||||||
|         <script> |  | ||||||
|             $(document).ready(function() { |  | ||||||
|                 //TODO merge appliction.js code into other js files |  | ||||||
|                 app.init(); |  | ||||||
|             }); |  | ||||||
|         </script> |  | ||||||
|  |  | ||||||
|         <!-- About WebGoat Modal --> |         <!-- About WebGoat Modal --> | ||||||
|         <div class="modal fade" id="aboutModal" tabindex="-1" role="dialog" aria-labelledby="myModalLabel" aria-hidden="true"> |         <div class="modal fade" id="aboutModal" tabindex="-1" role="dialog" aria-labelledby="myModalLabel" aria-hidden="true"> | ||||||
| @ -304,4 +212,6 @@ | |||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
|     </body> |     </body> | ||||||
|  |  | ||||||
|  |  | ||||||
| </html> | </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;} | #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 { | #header .user-nav button:hover i { | ||||||
|   color: #F6F6F6; |   color: #F6F6F6; | ||||||
| } | } | ||||||
| #header .lessonTitle { | #header #lesson-title-wrapper { | ||||||
| 	display: inline-block; | 	display: inline-block; | ||||||
| 	margin:0 0 0 20px;	 | 	margin:0 0 0 20px;	 | ||||||
| } | } | ||||||
| @ -747,7 +747,7 @@ cookieContainer { | |||||||
|   z-index: 100; |   z-index: 100; | ||||||
| } | } | ||||||
|  |  | ||||||
| #leftside-navigation { | #menu-container { | ||||||
|   overflow-y:scroll; |   overflow-y:scroll; | ||||||
|   overflow-x:hidden; |   overflow-x:hidden; | ||||||
| } | } | ||||||
| @ -761,18 +761,18 @@ cookieContainer { | |||||||
| .sidebar-toggle { | .sidebar-toggle { | ||||||
|   margin-left: -240px; |   margin-left: -240px; | ||||||
| } | } | ||||||
| #leftside-navigation ul, | #menu-container ul, | ||||||
| #leftside-navigation ul ul { | #menu-container ul ul { | ||||||
|   margin: -2px 0 0; |   margin: -2px 0 0; | ||||||
|   padding: 0; |   padding: 0; | ||||||
| } | } | ||||||
| #leftside-navigation ul li { | #menu-container ul li { | ||||||
|   list-style-type: none; |   list-style-type: none; | ||||||
|   border-bottom: 1px solid rgba(255, 255, 255, 0.05); |   border-bottom: 1px solid rgba(255, 255, 255, 0.05); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| #leftside-navigation ul li a { | #menu-container ul li a { | ||||||
|   color: #aeb2b7; |   color: #aeb2b7; | ||||||
|   text-decoration: none; |   text-decoration: none; | ||||||
|   display: block; |   display: block; | ||||||
| @ -785,38 +785,38 @@ cookieContainer { | |||||||
|   -ms-transition: all 200ms ease-in; |   -ms-transition: all 200ms ease-in; | ||||||
|   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; |   display: inline-block; | ||||||
| } | } | ||||||
| #leftside-navigation ul ul li { | #menu-container ul ul li { | ||||||
|   background: #333; |   background: #333; | ||||||
|   margin-bottom: 0; |   margin-bottom: 0; | ||||||
|   margin-left: 0; |   margin-left: 0; | ||||||
|   margin-right: 0; |   margin-right: 0; | ||||||
|   border-bottom: none; |   border-bottom: none; | ||||||
| } | } | ||||||
| #leftside-navigation ul ul li a { | #menu-container ul ul li a { | ||||||
|   font-size: 11px; |   font-size: 11px; | ||||||
|   padding-top: 5px; |   padding-top: 5px; | ||||||
|   padding-bottom: 5px; |   padding-bottom: 5px; | ||||||
|   color: #aeb2b7; |   color: #aeb2b7; | ||||||
|   margin-left:8px; |   margin-left:8px; | ||||||
| } | } | ||||||
| #leftside-navigation ul li a i { | #menu-container ul li a i { | ||||||
|   width: 20px; |   width: 20px; | ||||||
| } | } | ||||||
| #leftside-navigation ul li a i.fa-angle-right, | #menu-container ul li a i.fa-angle-right, | ||||||
| #leftside-navigation ul li a i.fa-angle-left { | #menu-container ul li a i.fa-angle-left { | ||||||
|   padding-top: 3px; |   padding-top: 3px; | ||||||
| } | } | ||||||
| #leftside-navigation ul ul { | #menu-container ul ul { | ||||||
|   display: none; |   display: none; | ||||||
| } | } | ||||||
| #leftside-navigation li.active ul { | #menu-container li.active ul { | ||||||
|   display: block; |   display: block; | ||||||
| } | } | ||||||
| #leftside-navigation ul li a:hover, | #menu-container ul li a:hover, | ||||||
| #leftside-navigation ul li.active > a { | #menu-container ul li.active > a { | ||||||
|   color: #e84c3d; |   color: #e84c3d; | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -827,15 +827,15 @@ cookieContainer { | |||||||
|     padding-top: 15px; |     padding-top: 15px; | ||||||
| } | } | ||||||
|  |  | ||||||
| #leftside-navigation ul li.selected { | #menu-container ul li.selected { | ||||||
|     background-color: #555; |     background-color: #555; | ||||||
| } | } | ||||||
|  |  | ||||||
| #leftside-navigation ul li.selected a.selected { | #menu-container ul li.selected a.selected { | ||||||
|   color:white; |   color:white; | ||||||
| } | } | ||||||
|  |  | ||||||
| #leftside-navigation ul ul.lessonsAndStages.keepOpen { | #menu-container ul ul.lessonsAndStages.keepOpen { | ||||||
|   display: block |   display: block | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -849,8 +849,16 @@ cookieContainer { | |||||||
|   cursor: pointer; |   cursor: pointer; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| .info { | .info { | ||||||
| 	color:#e84c3d; | 	color:#e84c3d; | ||||||
| 	font-weight: bold;  | 	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