diff --git a/js/pianos4.js b/js/pianos4.js index c3aa3a530c9251f7f842ec48824f658e622fdbc6..28903a4906f14a53b6223ef78a681d7f93e73cb9 100644 --- a/js/pianos4.js +++ b/js/pianos4.js @@ -88,7 +88,21 @@ function Pianos4( params ) tutor: false }; if( typeof ( params.hotSwitchPhase ) == 'undefined' ) - params.hotSwitchPhase = false; + params.hotSwitchPhase = false; // This, in theory, enables one to + // activate the phase directly, without + // reload. In the current state, this + // doesn't work as the UI is created + // based on the said phase. For it to + // work, the application must be + // re-designed so that the whole UI is + // always created, and its element are + // shown/hidden based on the mode. The + // non-hotSwitchPhase is nevertheless + // still better than in PiAnoS 3 as the + // results are cached - the "reload" is + // just about deleting the pianos object + // and recreating it. No data will be + // reloaded from the server. if( !params.initData ) params.initData = null; if( !params.linkToPhases ) @@ -215,128 +229,135 @@ function Pianos4( params ) this.init = function() { - var activetools = [ - new pianos4tooledit( { + var activeTools = [ + new Pianos4ToolEdit( { v3: params.v3, - hotkey: params.edittoolhotkey, - markcanhavecenter: params.markcanhavecenter + hotkey: params.editToolHotkey, + markCanHaveCenter: params.markCanHaveCenter } ), ]; if( !params.v3 ) { - if( params.usequalitytool ) + if( params.useQualityTool ) { - activetools.push( new pianos4toolridge( { + activeTools.push( new Pianos4ToolRidge( { type: 'quality', - intersectscript: params.intersectscript, - qualitydefaultansinist: false, - qualitytypeselector: 'dropdown', + intersectScript: params.intersectScript, + qualityDefaultAnsiNist: false, + qualityTypeSelector: 'dropdown', } ) ); } } - activetools.push( new pianos4toolminutia( { + activeTools.push( new Pianos4ToolMinutia( { v3: params.v3, - autopairing: params.minutiatoolautopairing, + autoPairing: params.minutiaToolAutoPairing, phase: params.phase, - singleclickdeselects: params.inferedittool, - markcanhavecenter: params.markcanhavecenter, + singleClickDeselects: params.inferEditTool, + markCanHaveCenter: params.markCanHaveCenter, } ) ); - activetools.push( new pianos4toolridge() ); + activeTools.push( new Pianos4ToolRidge() ); if( !params.v3 ) { - if( params.useareatool ) - activetools.push( new pianos4toolridge( { + if( params.useAreaTool ) + activeTools.push( new Pianos4ToolRidge( { type: 'area' } ) ); - if( params.usebrushtool ) - activetools.push( new pianos4toolbrush() ); - if( params.usecommenttool ) - activetools.push( new pianos4toolcomment() ); - if( params.usearctool ) - activetools.push( new pianos4toolarc() ); - if( params.usewcstool ) - activetools.push( new pianos4toolridge( { + if( params.useBrushTool ) + activeTools.push( new Pianos4ToolBrush() ); + if( params.useCommentTool ) + activeTools.push( new Pianos4ToolComment() ); + if( params.useArcTool ) + activeTools.push( new Pianos4ToolArc() ); + if( params.useWCSTool ) + activeTools.push( new Pianos4ToolRidge( { type: 'wcs' } ) ); - if( params.usegrouptool ) - activetools.push( new pianos4toolgroup() ); + if( params.useGroupTool ) + activeTools.push( new Pianos4ToolGroup() ); if( params.phase != 'analysis' ) - activetools.push( new pianos4toolpairing() ); + activeTools.push( new Pianos4ToolPairing() ); } - activetools.push( new pianos4toolpan( { + activeTools.push( new Pianos4ToolPan( { phase: params.phase } ) ); - if( typeof( tps_generate ) !== "undefined" && params.phase != 'analysis' ) - activetools.push( new pianos4tooltps() ); + if( typeof( TPS_generate ) !== "undefined" && params.phase != 'analysis' ) + activeTools.push( new Pianos4ToolTPS() ); - activetools.push( new pianos4toolmeasure() ); + activeTools.push( new Pianos4ToolMeasure() ); - if( typeof( pianos4toolimageeditor ) !== "undefined" ) - activetools.push( new pianos4toolimageeditor() ); + if( typeof( Pianos4ToolImageEditor ) !== "undefined" ) + activeTools.push( new Pianos4ToolImageEditor() ); if( params.user.tutor ) { - activetools.push( new pianos4toolreview() ); + activeTools.push( new Pianos4ToolReview() ); } - for( var tool in activetools ) + for( var tool in activeTools ) { - var tool = activetools[ tool ]; - tools[ tool.getname() ] = tool; + var tool = activeTools[ tool ]; + tools[ tool.getName() ] = tool; } - activetool = params.initialtool; - buildui(); + activeTool = params.initialTool; + buildUI(); - if( typeof ( params.screenresolution ) == 'object' ) + if( typeof ( params.screenResolution ) == 'object' ) { - screenresolution = calculatescreenresolution( params.screenresolution ); - } else if( typeof ( params.screenresolution ) != 'number' ) + screenResolution = calculateScreenResolution( params.screenResolution ); + } else if( typeof ( params.screenResolution ) != 'number' ) { - screenresolution = parsefloat( params.screenresolution ); + screenResolution = parseFloat( params.screenResolution ); } else { - screenresolution = params.screenresolution; + screenResolution = params.screenResolution; } - self.setzoomfit(); + self.setZoomFit(); - if( params.usehotkeys ) - registerhotkeys(); + if( params.useHotkeys ) + registerHotkeys(); + /* + * self.registerHandler('contextmenu', function(e) { if + * (skipNextContextMenu) { skipNextContextMenu = false; + * e.preventDefault(); return false; } }); + */ - if( params.initdata ) + if( params.initData ) { - ondataloaded( params.initdata ); - updateui(); - } else if( params.loadresults ) + onDataLoaded( params.initData ); + updateUI(); + } else if( params.loadResults ) { + // load() will call ajax() which will in turn call updateUI() self.load(); } else { - onconclusionupdated( true ); - updateui(); + onConclusionUpdated( true ); // This will also be called after + // load() + updateUI(); } - activetool = null; - setactivetool( params.initialtool ); + activeTool = null; + setActiveTool( params.initialTool ); - if( params.startswithconclusionopen ) - toggleconclusion(); + if( params.startsWithConclusionOpen ) + toggleConclusion(); - if( params.onexercisestartup ) - params.onexercisestartup( self, params ); + if( params.onExerciseStartup ) + params.onExerciseStartup( self, params ); - // preload some images from the analysis view + // Preload some images from the analysis view if( params.phase === "analysis" ) { $( document ).ready( function() { - settimeout( function() + setTimeout( function() { preloader( 'css/location/main.png' ); preloader( params.print.href, 120, 15 ); @@ -408,9 +429,15 @@ function Pianos4( params ) var closeButtonLabel = 'Close'; var fields = getConclusionFields(); - // This will map unanswered questions to their numbered labels to help the users answer the appropriate question(s). - // This mapping should make it into the 'fields' variable above but this implies refactoring, which I have no time for. - // TODO: merge the numbers to the fields, make the script apply numbers to the questions (they are applied in buildStatusBar() for the moment) and get rid of 'uglyMapping' + // This will map unanswered questions to their numbered labels + // to help the users answer + // the appropriate question(s). This mapping should make it into + // the 'fields' variable + // above but this implies refactoring, which I have no time for. + // TODO: merge the numbers to the fields, make the script apply + // numbers to the questions + // (they are applied in buildStatusBar() for the moment) and get + // rid of 'uglyMapping' if( ! params.skip_conclusion_check ) { @@ -793,11 +820,15 @@ function Pianos4( params ) } else { action.redo(); - // Cut the stack where the pointer currently is - this will loose track of all modifications that were rolled back to that point (in other words, the chain of actions available for "redo" is destroyed). + // Cut the stack where the pointer currently is - this will loose + // track of all modifications that were rolled back to that point + // (in other words, the chain of actions available for "redo" is + // destroyed). if( stack.length > 0 && stackPtr < stack.length - 1 ) { if( lastSaveData.stackPtr > stackPtr ) - // Clear the lastSaveData's stack ptr since it points to an area of the stack that has been invalidated. + // Clear the lastSaveData's stack ptr since it points to an area + // of the stack that has been invalidated. lastSaveData.stackPtr = -1 stack = stack.slice( 0, stackPtr + 1 ); } @@ -952,8 +983,10 @@ function Pianos4( params ) if( !params.user.tutor ) { data.observations = getConclusions(); - delete data.observations.length; - delete data.observations.isOK; + delete data.observations.length; // 'length' is set by + // getConclusions() but isn't + // actual data + delete data.observations.isOK; // Same remark delete data.observations.mandatoryOK; } @@ -1099,6 +1132,9 @@ function Pianos4( params ) case 'dropdown': var valueToSet = field[ 2 ] in data ? data[ field[ 2 ] ] : -1; field[ 1 ].val( valueToSet ); + // The call above can actually fail if the dropdown does not + // contain the key + // that was passed in, so the assert() below makes sense. assert( field[ 1 ].val() == valueToSet ); break; case 'data': @@ -1141,9 +1177,12 @@ function Pianos4( params ) } // Call when the selection has changed to update the properties panel. - // Pass-in the view on which the latest event occurred. - // If null is given, clears the focus out of both views and display 'No properties to show', as if nothing was selected in the current view. - // Pass a boolean true value to re-use the current view with focus + // Pass-in the view + // on which the latest event occurred. If null is given, clears the focus + // out of both + // views and display 'No properties to show', as if nothing was selected in + // the current + // view. Pass a boolean true value to re-use the current view with focus this.onSelectionChanged = function( view ) { if( typeof ( view ) == 'boolean' && view ) @@ -1159,6 +1198,8 @@ function Pianos4( params ) if( view ) { var sel = view.getSelection(); + // log('Selection changed to'); + // log(sel); var firstType = null; var multiTypes = false; @@ -1232,6 +1273,14 @@ function Pianos4( params ) ] ); } catch( e ) { + // May fail if the quality ID was set to + // e.g. 5 in ansi_nist mode, then the mode + // was changed to standard + // Don't print anything in this case - not + // too much harm anyway + // The actual fix will be an automated + // translation of ALL quality areas when the + // mode is changed. } } } @@ -1423,6 +1472,9 @@ function Pianos4( params ) return false; } + // Returns the currently logged-in user (if 'id' argument is not provided) + // or the user whose id is given - this user must exist within the current + // results working set. this.getUser = function( id ) { if( typeof ( id ) == 'undefined' ) @@ -1492,9 +1544,13 @@ function Pianos4( params ) layoutObj.destroy(); $( 'body' ).html( '
' ); pianos.unregisterHandlers(); + // $('*').unbind(); + // $(document).unbind(); + // $(window).unbind(); delete pianos; - // Create the new instance and run it jQueryUIResetMouseWidgets(); + // Create the new instance and run it + // jQueryUIResetMouseWidgets(); pianos = new Pianos4( p ); pianos.init(); }, 1 ); @@ -1685,6 +1741,7 @@ function Pianos4( params ) for( var pane in layoutObj.panes ) { + // var pane = layoutObj.panes[i]; if( pane == 'center' ) continue; if( wanted ) @@ -1696,16 +1753,34 @@ function Pianos4( params ) isFullscreen = wanted; } - // This functionality is a "RAII" version of $(document/window).bind() that will remember the handler so that it can be removed when the pianos.unregisterHandlers() method is called. - // This is because one cannot simply do a $(document).unbind() and expect things to work - jQuery UI (esp. jquery.ui.mouse) and maybe other libraries/plugins do register global, static callbacks that would be unbound, without possibility of re-registering them again. - // For the record, we want to unbind everything when the pianos object is destroyed/recreated, so that handlers are not registered twice. - // Because of this, we could either expect jQuery UI to have a reset() function of some kind - which it hasn't and will probably never have - or do the things cleanly on our end. - // This is the goal of this function. + // This functionality is a "RAII" version of $(document/window).bind() that + // will remember the + // handler so that it can be removed when the pianos.unregisterHandlers() + // method is called. This is because one cannot + // simply do a $(document).unbind() and expect things to work - jQuery UI + // (esp. jquery.ui.mouse) and + // maybe other libraries/plugins do register global, static callbacks that + // would be unbound, without + // possibility of re-registering them again. For the record, we want to + // unbind everything when the + // pianos object is destroyed/recreated, so that handlers are not registered + // twice. Because of this, + // we could either expect jQuery UI to have a reset() function of some kind + // - which it hasn't and will + // probably never have - or do the things cleanly on our end. This is the + // goal of this function. // Note: - // This fixes the "cannot move slider after step change" bug - this is because jquery.ui.mouse has a 'mouseup' handler that sets 'mouseHandled' to false appropriately, and would not work after the global unbind(), thus preventing any plugin that relied on the mouse to work, e.g. sliders, but also the layout, resizables, etc. + // This fixes the "cannot move slider after step change" bug - this is + // because jquery.ui.mouse + // has a 'mouseup' handler that sets 'mouseHandled' to false appropriately, + // and would not work after + // the global unbind(), thus preventing any plugin that relied on the mouse + // to work, e.g. sliders, but + // also the layout, resizables, etc. // Usage: // will call $(document/window).bind(event,handler) if event is a string - // will call $(document/window).bind(event[0], event[1], handler) if event is an array of length 2 + // will call $(document/window).bind(event[0], event[1], handler) if event + // is an array of length 2 // The second form is useful for e.g. keydown events. this.registerHandler = function( event, handler, isWindow ) { @@ -1824,10 +1899,12 @@ function Pianos4( params ) var drawTutorSVG = null; var tutorData = {}; var tutorDataReady = false; + // var skipNextContextMenu = false; // The zoom slider (v) goes from 0 to 100 // The actual zoom factor (z) goes from 2% to 3200% - // We want a value of 100% zoom when the slider is at the middle point (v=50) + // We want a value of 100% zoom when the slider is at the middle point + // (v=50) // Using MATLAB to compute a quadratic logarithmic regression, i.e. // > v = [0,50,100] // > z = [2,100,3200] @@ -1839,7 +1916,8 @@ function Pianos4( params ) var sliderQuadRegVals = [ -8.925742052568407e-05, 0.082703331134847, 0.693147180559945 ]; - // Inverse function: after c' = polyfit(Z,v,2), g(x) = c(1)*log(x)^2+c(2)*log(x)+c(3) + // Inverse function: after c' = polyfit(Z,v,2), g(x) = + // c(1)*log(x)^2+c(2)*log(x)+c(3) var sliderQuadRegValsInv = [ 0.223081222576626, 11.599155816038060, -8.147102196497766 ]; @@ -1944,6 +2022,7 @@ function Pianos4( params ) inferEditTool: params.inferEditTool, showModifiedMinutiae: params.showModifiedMinutiae, updatedMinutiaeTimestamp: params.updatedMinutiaeTimestamp, + // useH5Canvas: params.useBrushTool, } ); allViews.push( mark ); @@ -1963,6 +2042,7 @@ function Pianos4( params ) editToolHotkey: params.editToolHotkey, useOtherFeaturesLayer: params.useWCSTool, inferEditTool: params.inferEditTool, + // useH5Canvas: params.useBrushTool, } ); allViews.push( print ); @@ -1990,6 +2070,7 @@ function Pianos4( params ) spacing_open: params.layoutBorderWidth, spacing_closed: params.layoutBorderWidth, initClosed: params.startsFullscreen, + // applyDefaultStyles: true, }, center: { onresize: function() @@ -2030,6 +2111,7 @@ function Pianos4( params ) var radio = $( '' ) .attr( 'title', tool.getLabel() ); var label = $( '