Merge branch 'master' of ssh://git.onelab.eu/git/myslice
authorJordan Augé <jordan.auge@lip6.fr>
Fri, 18 Oct 2013 15:42:45 +0000 (17:42 +0200)
committerJordan Augé <jordan.auge@lip6.fr>
Fri, 18 Oct 2013 15:42:45 +0000 (17:42 +0200)
15 files changed:
manifold/static/js/manifold.js
plugins/googlemap/static/js/googlemap.js
plugins/hazelnut/static/js/hazelnut.js
plugins/query_editor/__init__.py
plugins/query_editor/static/js/query_editor.js
plugins/senslabmap/__init__.py
plugins/senslabmap/senslabmap.py [deleted file]
plugins/senslabmap/static/css/senslabmap.css
plugins/senslabmap/static/js/senslabmap.js
plugins/senslabmap/static/js/viewer3D.js
plugins/senslabmap/templates/senslabmap.html
portal/accountview.py
portal/registrationview.py
portal/resourceview.py
portal/sliceview.py

index 791e4cf..7da4700 100644 (file)
@@ -904,8 +904,10 @@ var manifold = {
                 manifold.run_query(query_ext.main_query_ext.update_query_ext.query);
                 break;
 
-            case FILTER_ADDED:
-                manifold.raise_query_event(query_uuid, event_type, value);
+            case FILTER_ADDED: 
+// Thierry - this is probably wrong but intended as a hotfix 
+// http://trac.myslice.info/ticket/32
+//                manifold.raise_query_event(query_uuid, event_type, value);
                 break;
             case FILTER_REMOVED:
                 manifold.raise_query_event(query_uuid, event_type, value);
index 8cb652f..424b3aa 100644 (file)
@@ -154,7 +154,7 @@ googlemap_debug_detailed=false;
 
        // this record is *in* the slice
         new_record: function(record) {
-           if (googlemap_debug_detailed) messages.debug ("new_record");
+               if (googlemap_debug_detailed) messages.debug ("new_record");
             if (!(record['latitude'])) return false;
            
             // get the coordinates
@@ -162,33 +162,33 @@ googlemap_debug_detailed=false;
             var longitude=unfold.get_value(record['longitude']);
             var lat_lon = latitude + longitude;
 
-           // check if we've seen anything at that place already
-           // xxx might make sense to allow for some fuzziness, 
-           // i.e. consider 2 places equal if not further away than 300m or so...
-           var marker_s = this.by_lat_lon [lat_lon];
-           if ( marker_s == null ) {
-               marker_s = this.create_marker_struct (this.object, latitude, longitude);
-               this.by_lat_lon [ lat_lon ] = marker_s;
-               this.arm_marker(marker_s.marker, this.map);
-           }
+           // check if we've seen anything at that place already
+           // xxx might make sense to allow for some fuzziness, 
+           // i.e. consider 2 places equal if not further away than 300m or so...
+           var marker_s = this.by_lat_lon [lat_lon];
+           if ( marker_s == null ) {
+                       marker_s = this.create_marker_struct (this.object, latitude, longitude);
+                       this.by_lat_lon [ lat_lon ] = marker_s;
+                       this.arm_marker(marker_s.marker, this.map);
+               }
            
-           // now add a line for this resource in the marker
-           // xxx should compute checked here ?
-           // this is where the checkbox will be appended
-           var ul=marker_s.ul;
-           var checkbox = this.create_record_checkbox (record, ul, false);
-           if ( ! this.key in record ) return;
+           // now add a line for this resource in the marker
+           // xxx should compute checked here ?
+           // this is where the checkbox will be appended
+           var ul=marker_s.ul;
+           var checkbox = this.create_record_checkbox (record, ul, false);
+           if ( ! this.key in record ) return;
             var key_value = record[this.key];
-           // see XXX BACKSLASHES 
-           //var hrn = this.escape_id(key_value).replace(/\\/g, '');
-           var hrn = key_value;
+           // see XXX BACKSLASHES 
+           //var hrn = this.escape_id(key_value).replace(/\\/g, '');
+           var hrn = key_value;
             this.by_hrn[hrn] = {
-               checkbox: checkbox,
-               // xxx Thierry sept 2013
-               // xxx actually we might have just used a domid-based scheme instead of the hash
-               // since at this point we only need to retrieve the checkbox from an hrn
-               // but I was not sure enough that extra needs would not show up so I kept this in place
-               // xxx not sure these are actually useful :
+                   checkbox: checkbox,
+                       // xxx Thierry sept 2013
+                       // xxx actually we might have just used a domid-based scheme instead of the hash
+                       // since at this point we only need to retrieve the checkbox from an hrn
+                       // but I was not sure enough that extra needs would not show up so I kept this in place
+                       // xxx not sure these are actually useful :
                 value: key_value,
                 record: record,
             }
index 249d2b4..bdb2de7 100644 (file)
 
             /* Processing hidden_columns */
             $.each(this.options.hidden_columns, function(i, field) {
-                manifold.raise_event(self.options.query_all_uuid, FIELD_REMOVED, field);
-                //self.hide_column(field);
+                //manifold.raise_event(self.options.query_all_uuid, FIELD_REMOVED, field);
+                self.hide_column(field);
             });
         }, // initialize_table
 
index 0312ef1..2d73053 100644 (file)
@@ -5,6 +5,12 @@ from django.template.loader import render_to_string
 # XXX We need naming helpers in the python Plugin class also, used in template
 
 class QueryEditor(Plugin):
+    def __init__ (self, query, query_all = None, **settings):
+        Plugin.__init__ (self, **settings)
+        self.query=query
+        self.query_uuid = query.query_uuid
+        self.query_all = query_all
+        self.query_all_uuid = query_all.query_uuid if query_all else None
 
     def template_file(self):
         return "query_editor.html"
@@ -22,9 +28,6 @@ class QueryEditor(Plugin):
         }
         return reqs
 
-    def json_settings_list (self):
-        return ['plugin_uuid', 'domid', 'query_uuid']
-
     def export_json_settings (self):
         return True
 
@@ -82,4 +85,7 @@ class QueryEditor(Plugin):
                 'checked':       md_field['name'] in self.query.get_select()
             })
         #return { 'fields': fields, 'hidden_columns': hidden_columns }
+        #return { 'fields': fields , 'query_uuid': self.query_uuid, 'query_all_uuid': self.query_all_uuid }
         return { 'fields': fields }
+
+    def json_settings_list (self): return ['plugin_uuid', 'domid', 'query_uuid', 'query_all_uuid', ]
index 9858818..17b9bf7 100644 (file)
         init: function(options, element) {
             this._super(options, element);
             this.listen_query(options.query_uuid);
+            // this one is the complete list of resources
+            // and will be bound to callbacks like on_all_new_record
+            this.listen_query(options.query_all_uuid, 'all');
+
 
             this.elts('queryeditor-auto-filter').change(this.event_filter_added('='));
             this.elts('queryeditor-filter').change(this.event_filter_added('='));
@@ -88,6 +92,7 @@
 //                    { 'sWidth': '8px', 'aTargets': [ 4 ] } // XXX NB OF COLS
 //                ]
             });
+            this.table = metaTable;
 
             // Actions on the newly added fields
             this.elmt('table tbody td span').on('click', function() {
             //console.log("Query_Editor: query_done!");
             //console.log(this.availableTags);
         },
-        on_new_record: function(record)
+        /* Autocomplete based on query_all to get all the fields, where query get only the fields selected  */
+        on_all_new_record: function(record)
         {
-            //console.log("Query_Editor: new_record!");
-            //console.log(record);
             availableTags = this.availableTags;           
             jQuery.each(record,function(key,value){
                 value = unfold.get_value(value);
             
             jQuery.each(availableTags, function(key, value){
                 value.sort();
-                jQuery("#"+domid+"__field__"+key).autocomplete({
+                // using dataTables's $ to search also in nodes that are not currently displayed
+                var element = self.table.$("#"+domid+"__field__"+key);
+
+                element.autocomplete({
                             source: value,
                             selectFirst: true,
                             minLength: 0, // allows to browse items with no value typed in
index e69de29..7540edb 100644 (file)
@@ -0,0 +1,33 @@
+from unfold.plugin import Plugin
+
+class SensLabMap (Plugin):
+
+    # set checkboxes if a final column with checkboxes is desired
+    # pass columns as the initial set of columns
+    #   if None then this is taken from the query's fields
+    def __init__ (self, query, query_all, **settings):
+        Plugin.__init__ (self, **settings)
+        self.query = query
+        self.query_all = query_all
+        self.query_all_uuid = query_all.query_uuid
+
+    def template_file (self):
+        return "senslabmap.html"
+
+    def template_env (self, request):
+        env={}
+        return env
+
+    def requirements (self):
+        reqs = {
+            'js_files' : [ "js/senslabmap.js",
+                           "js/spin.presets.js", "js/spin.min.js", "js/jquery.spin.js",
+                           "js/three.js", "js/grenoble.js", "js/viewer3D.js",
+                           ] ,
+            'css_files': [ "css/senslabmap.css" ,
+                           ],
+            }
+        return reqs
+
+    # the list of things passed to the js plugin
+    def json_settings_list (self): return ['plugin_uuid', 'dom_id', 'query_uuid', 'query_all_uuid']
diff --git a/plugins/senslabmap/senslabmap.py b/plugins/senslabmap/senslabmap.py
deleted file mode 100644 (file)
index 73c767d..0000000
+++ /dev/null
@@ -1,31 +0,0 @@
-from unfold.plugin import Plugin
-
-class SensLabMap (Plugin):
-
-    # set checkboxes if a final column with checkboxes is desired
-    # pass columns as the initial set of columns
-    #   if None then this is taken from the query's fields
-    def __init__ (self, query, **settings):
-        Plugin.__init__ (self, **settings)
-        self.query=query
-
-    def template_file (self):
-        return "senslabmap.html"
-
-    def template_env (self, request):
-        env={}
-        return env
-
-    def requirements (self):
-        reqs = {
-            'js_files' : [ "js/senslabmap.js",
-                           "js/spin.presets.js", "js/spin.min.js", "js/jquery.spin.js",
-                           "js/three.js", "js/grenoble.js", "js/viewer3D.js",
-                           ] ,
-            'css_files': [ "css/senslabmap.css" , 
-                           ],
-            }
-        return reqs
-
-    # the list of things passed to the js plugin
-    def json_settings_list (self): return ['plugin_uuid','query_uuid']
index cfd2d94..86f3638 100644 (file)
@@ -1,11 +1,40 @@
-.senslab-map {
-  width: 800px;
-  height: 500px;
-  height: 100%;
+.senslabmap {
+  margin: 10px;
+  font-family: "Helvetica Neue", Helvetica, Arial, sans-serif;
+  font-size: 13px;
+  line-height: 18px;
+  color: #333333;
+  background-color: #ffffff;
 }
 
 #div3d {
-    width: 100%; 
-    height: 500px; 
-    background-color: #202020;
+  height: 500px;
+  background-color: #202020;
+  z-index:-1
+}
+
+#selectionbox {
+  text-align: center;
+  padding:2px
+}
+
+#infobox {
+  text-align: center
+}
+
+#node_alive {
+  background-image: url('');
+  
+}
+
+#node_down {
+  background-image: url('');
+}
+
+#node_selected {
+  background-image: url('');
+}
+
+#node_used {
+  background-image: url('');
 }
index 8075be9..1fe7810 100644 (file)
@@ -1,80 +1,51 @@
-/**
- * Description: SensLab display of 3D-geolocated data
- * Copyright (c) 2012 UPMC Sorbonne Universite - INRIA
- * License: GPLv3
- */
-
-// xxx TODO -- this plugin has never been tested 
-// update_map looks suspiciously empty...
-// in addition it could use a bit of cleaning like what was done for the first plugins
-// especially wrt using 'instance' and 'data' in such a confusing way
-
-(function( $ ){
-
-    $.fn.SensLabMap = function( method ) {
-       /* Method calling logic */
-       if ( methods[method] ) {
-           return methods[ method ].apply( this, Array.prototype.slice.call( arguments, 1 ));
-       } else if ( typeof method === 'object' || ! method ) {
-           return methods.init.apply( this, arguments );
-       } else {
-           $.error( 'Method ' +  method + ' does not exist on $.SensLabMap' );
-       }    
-    };
-
-    var methods = {
-       init : function( options ) {
-           return this.each(function(){
-               var $this = $(this),
-               data = $this.data('SensLabMap'), SensLabMap = $('<div />', { text : $this.attr('title') });
-               // If the plugin hasn't been initialized yet
-               if ( ! data ) {
-                   /* Plugin initialization */
-                   //google.load('maps', '3', { other_params: 'sensor=false' });
-                   //google.setOnLoadCallback(initialize);
-                   init();
-                   /* End of plugin initialization */
-                   $(this).data('SensLabMap', {
-                       plugin_uuid: options.plugin_uuid,
-                       query_uuid: options.query_uuid,
-                       target : $this,
-                       SensLabMap : SensLabMap
-                   });
-                   /* Subscribe to query updates */
-                   $.subscribe('/results/' + options.query_uuid + '/changed', {instance: $this}, update_map);
-               }
-           });
-       },
-       destroy : function( ) {
-            return this.each(function(){
-               var $this = $(this), data = $this.data('SensLabMap');
-               $(window).unbind('SensLabMap');
-               data.SensLabMap.remove();
-               $this.removeData('SensLabMap');
-           });
-       },
-/*
-    reposition : function( ) { // ... },
-    show : function( ) { // ... },
-    hide : function( ) { // ... },
-*/
-       update : function( content ) { 
-            // should be made a private function
-            set3dsize()
-       },
-  };
-
-    /* Private methods */
-    function update_map(e, rows) {
-        var $plugindiv = e.data.instance;
-       $plugindiv.closest('.need-spin').spin(false);
-       
-        if(rows) {
-           /* TODO rendering */   
-        } else {
-           alert('error');
-        }
-       
+(function($){
+  var SensLabMap = Plugin.extend({
+    init: function(options, element) {
+      this._super(options, element);
+      
+      this.elmt().on('show', this, this.on_show);
+      
+      this.method = query.object;
+      
+      /* Setup query and record handlers */
+      this.listen_query(options.query_uuid);
+      this.listen_query(options.query_all_uuid, 'all');
+      
+      /* GUI setup and event binding */
+      this.initialize_map();
+    },
+    
+    initialize_map: function() {
+      this.nodes = [];
+      this.id = 0;
+      init();
+    },
+    
+    refresh: function() {
+      console.log("refresh");
+      myrender();
+    },
+    
+    on_show: function(e) {
+      e.data.refresh();
+    },
+    
+    on_all_new_record: function(n) {
+      // format is : [name, x, y, z, uid, state]
+      // state = "Busy", "Alive" or "Suspected"
+      if (n.x == null || n.y == null || n.z == null) {
+        console.log("Warning: no coord for " + n.hrn);
+        return;
+      }
+      this.id++;
+      node = [this.id, n.x, n.y, n.z, this.id, n.boot_state];
+      this.nodes.push(node);
+    },
+    
+    on_all_query_done: function() {
+      drawNodes(this.nodes);
+      parseNodebox();
     }
-
-})( jQuery );
+  });
+  $.plugin('SensLabMap', SensLabMap);
+})(jQuery);
index cf70e81..33ae659 100644 (file)
@@ -1,17 +1,8 @@
-\r
-// Various global variables\r
-var mouseX = 0, mouseY = 0,\r
-                       camera, scene, renderer, projector;\r
-var sTestEventType = 'mousedown';\r
-//var window3DWidth = window.innerWidth * 0.8;\r
-//var window3DHeight=window.innerHeight*0.75;\r
-var window3DWidth, window3DHeight;\r
-var offX, offY;\r
+// Various global variables\r
+var mouseX = 0, mouseY = 0, camera, scene, renderer, projector;\r
+\r
 // Camera parameters\r
-phi = -100, theta = 0, distance = 150;\r
-var rcount = 0;\r
-// Text display\r
-var info, nodelist, help;\r
+var phi = -100, theta = 0, distance = 130;\r
 \r
 // graphical nodes\r
 var objects = [];\r
@@ -21,322 +12,331 @@ var selectedNodes = [];
 \r
 var div3d, nodebox, infobox;\r
 \r
-function init() {\r
-    var particles, particle;\r
-\r
-    // jordan : modifier document.getElementById\r
-    div3d = jQuery('#div3d');\r
-    nodebox = jQuery('#nodebox');\r
-    infobox = jQuery('#infobox');\r
-    titlebox = jQuery('#titlebox');\r
-    // offset also\r
-    var offset = div3d.offset();\r
-    offY = offset.top;\r
-    offX = offset.left;\r
-\r
-    titlebox.innerHTML = 'Grenoble Site ' + nodes_gre.length + " nodes";\r
-    infobox.innerHTML = 'Node info : ';\r
-\r
-    nodebox.value = "";\r
-\r
-    camera = new THREE.PerspectiveCamera(75, window3DWidth / window3DHeight, 1, 10000);\r
-\r
-    scene = new THREE.Scene();\r
+function onResize() {\r
+  renderer.setSize(div3d.offsetWidth, div3d.offsetHeight);\r
+  camera.aspect = div3d.offsetWidth / div3d.offsetHeight;\r
+  camera.updateProjectionMatrix();\r
+  myrender();\r
+}\r
 \r
-    renderer = new THREE.CanvasRenderer();\r
+function getCenter(nodes) {\r
+  var xmin = 0, ymin = 0, zmin = 0;\r
+  var xmax = 0, ymax = 0, zmax = 0;\r
+  \r
+  for (var i = 0; i < nodes.length; i++) {\r
+    if (nodes[i][1] > xmax) xmax = nodes[i][1];\r
+    if (nodes[i][1] < xmin) xmin = nodes[i][1];\r
+    if (nodes[i][2] > ymax) ymax = nodes[i][2];\r
+    if (nodes[i][2] < ymin) ymin = nodes[i][2];\r
+    if (nodes[i][3] > zmax) zmax = nodes[i][3];\r
+    if (nodes[i][3] < zmin) zmin = nodes[i][3];\r
+  }\r
+  return {x: (xmax + xmin) / 2, y: (ymax + ymin) / 2, z: (zmax + zmin) / 2};\r
+}\r
 \r
-    // jordan XXX div3d.appendChild(renderer.domElement);\r
-    div3d.append(renderer.domElement);\r
+function debugaxis(axisLength) {\r
+  var v = function(x, y, z) {\r
+    return new THREE.Vertex(new THREE.Vector3(x, y, z));\r
+  }\r
+  \r
+  var createAxis = function (p1, p2, color) {\r
+    var lineGeometry = new THREE.Geometry();\r
+    var lineMat = new THREE.LineBasicMaterial({ color: color, lineWidth: 2 });\r
+    lineGeometry.vertices.push(p1, p2);\r
+    var line = new THREE.Line(lineGeometry, lineMat);\r
+    scene.add(line);\r
+  }\r
+  \r
+  createAxis(v(0, 0, 0), v(axisLength, 0, 0), 0xFF0000);\r
+  createAxis(v(0, 0, 0), v(0, axisLength, 0), 0x00FF00);\r
+  createAxis(v(0, 0, 0), v(0, 0, axisLength), 0x0000FF);\r
+};\r
 \r
-    window.addEventListener('resize', set3dsize, false);\r
-    scene.add(camera);\r
-    set3dsize();\r
+function drawNodes(nodes) {\r
+  var geometry = new THREE.Geometry();\r
+  var center = getCenter(nodes);\r
+  \r
+  onResize();\r
+  for (var i = 0; i < nodes.length; i++) {\r
+    var material = new THREE.ParticleCanvasMaterial({\r
+      color: 0xffffff,\r
+      program: function (context) {\r
+        context.beginPath();\r
+        context.arc(0, 0, 1, 0, Math.PI * 2, true);\r
+        context.closePath();\r
+        context.fill();\r
+      }\r
+    });\r
+    var particle = new THREE.Particle(material);\r
+    particle.name = nodes[i][0];\r
+    particle.position.x = nodes[i][1] - center.x;\r
+    particle.position.y = nodes[i][2] - center.y;\r
+    particle.position.z = nodes[i][3] - center.z;\r
+    particle.uid = nodes[i][4];\r
+    particle.state = nodes[i][5];\r
+    particle.position.multiplyScalar(10);\r
+    particle.scale.x = particle.scale.y = 1;\r
+    scene.add(particle);\r
+    \r
+    var vertex = new THREE.Vertex(particle.position);\r
+    geometry.vertices.push(vertex);\r
+    objects.push(particle)\r
     \r
-\r
-    var PI2 = Math.PI * 2;\r
-    var geometry = new THREE.Geometry();\r
-\r
-    // let's find the center of the nodes\r
-    xmin = ymin = zmin = 0;\r
-    xmax = ymax = zmax = 0;\r
-\r
-    for (var i = 0; i < nodes_gre.length; i++) {\r
-        if (nodes_gre[i][1] > xmax) xmax = nodes_gre[i][1];\r
-        if (nodes_gre[i][1] < xmin) xmin = nodes_gre[i][1];\r
-        if (nodes_gre[i][2] > ymax) ymax = nodes_gre[i][2];\r
-        if (nodes_gre[i][2] < ymin) ymin = nodes_gre[i][2];\r
-        if (nodes_gre[i][3] > zmax) zmax = nodes_gre[i][3];\r
-        if (nodes_gre[i][3] < zmin) zmin = nodes_gre[i][3];\r
-    }\r
-\r
-    xcenter = (xmax + xmin) / 2;\r
-    ycenter = (ymax + ymin) / 2;\r
-    zcenter = (zmax + zmin) / 2;\r
-\r
-    // display nodes as TREE particles\r
-    for (var i = 0; i < nodes_gre.length; i++) {\r
-        material = new THREE.ParticleCanvasMaterial({\r
-            color: 0xffffff,\r
-            program: function (context) {\r
-                context.beginPath();\r
-                context.arc(0, 0, 1, 0, PI2, true);\r
-                context.closePath();\r
-                context.fill();\r
-            }\r
-        });\r
-        particle = new THREE.Particle(material);\r
-        particle.name = nodes_gre[i][0];\r
-        particle.position.x = nodes_gre[i][1] - xcenter;\r
-        particle.position.y = nodes_gre[i][2] - ycenter;\r
-        particle.position.z = nodes_gre[i][3] - zcenter;\r
-        particle.position.multiplyScalar(10);\r
-        particle.scale.x = particle.scale.y = 1;\r
-        scene.add(particle);\r
-        v = new THREE.Vertex(particle.position);\r
-        geometry.vertices.push(v);\r
-        objects.push(particle)\r
-    }\r
-\r
     debugaxis(10);\r
-    // a projector is needed to find which node is under the mouse\r
-    projector = new THREE.Projector();\r
-\r
-    nodebox.onkeyup = parseNodebox;\r
-    // set mouse event handlers // TODO jordan\r
-    document.onmousedown = OnMouseDown;\r
-    document.onmouseup = OnMouseUp;\r
-    document.onmousemove = displayNodeInfo;\r
-    if (window.addEventListener)\r
-    /** DOMMouseScroll is for mozilla. */\r
-        window.addEventListener('DOMMouseScroll', wheel, false);\r
-    /** IE/Opera. */\r
-    window.onmousewheel = document.onmousewheel = wheel;\r
     myrender();\r
+  }\r
 }\r
 \r
-function set3dsize() {\r
-    // jordan\r
-    var offset = div3d.offset();\r
-    offY = offset.top;\r
-    offX = offset.left;\r
-    window3DWidth = div3d.width();\r
-    window3DHeight = div3d.height();\r
-    renderer.setSize(window3DWidth, window3DHeight);\r
-\r
-    camera.aspect = window3DWidth / window3DHeight;\r
-    camera.updateProjectionMatrix();\r
-\r
-    myrender();\r
+function init(nodes) {\r
+  div3d = document.getElementById('div3d');\r
+  nodebox = document.getElementById('nodebox');\r
+  infobox = document.getElementById('infobox');\r
+  \r
+  infobox.innerHTML = 'Node info : ';\r
+  nodebox.value = "";\r
+  \r
+  camera = new THREE.PerspectiveCamera(75, div3d.offsetWidth / div3d.offsetHeight, 1, 10000);\r
+  scene = new THREE.Scene();\r
+  renderer = new THREE.CanvasRenderer();\r
+  projector = new THREE.Projector();\r
+  \r
+  div3d.appendChild(renderer.domElement);\r
+  \r
+  window.addEventListener('resize', onResize, false);\r
+  scene.add(camera);\r
+  \r
+  nodebox.onkeyup = parseNodebox;\r
+  \r
+  div3d.onmousedown = OnMouseDown;\r
+  div3d.onmouseup = OnMouseUp;\r
+  div3d.onmousemove = displayNodeInfo;\r
+  if (div3d.addEventListener) {\r
+    div3d.addEventListener('DOMMouseScroll', wheel, false); // mozilla\r
+  }\r
+  div3d.onmousewheel = wheel; // IE & Opera\r
 }\r
 \r
 function sortfunction(a, b) {\r
-    return (a - b) //causes an array to be sorted numerically and ascending\r
+  return (a - b) //causes an array to be sorted numerically and ascending\r
 }\r
 \r
 // factorize the expanded list in selectedNode to produce dash intervals\r
 // 1,2,3,5,9  -> 1-3,5,9\r
 function factorize() {\r
-    var fact = [];\r
-    var previous = 0;\r
-    var intervalStart = 0;\r
-    selectedNodes.sort(sortfunction);\r
-    for (j = 0; j < selectedNodes.length; j++) {\r
-        if (intervalStart) {\r
-            // we are in an interval\r
-            if (selectedNodes[j] == previous + 1) {\r
-                // interval grows\r
-                previous += 1;\r
-            } else {\r
-                // end of interval\r
-                fact.push(intervalStart + "-" + previous);\r
-                intervalStart = 0;\r
-                previous = selectedNodes[j];\r
-            }\r
-        } else {\r
-            // we are not in an interval\r
-            if (selectedNodes[j] == previous + 1) {\r
-                // let's begin an interval\r
-                intervalStart = previous;\r
-                previous += 1;\r
-            } else {\r
-                if (previous) fact.push(previous);\r
-                previous = selectedNodes[j];\r
-            }\r
-        }\r
-    } // for end \r
-    // at the end of the loop, two cases\r
-    // we were still in an interval, then add it to the result\r
-    // we were not in an interval, then add the last selectednode (previous)\r
+  var fact = [];\r
+  var previous = 0;\r
+  var intervalStart = 0;\r
+  selectedNodes.sort(sortfunction);\r
+  for (j = 0; j < selectedNodes.length; j++) {\r
     if (intervalStart) {\r
+      // we are in an interval\r
+      if (selectedNodes[j] == previous + 1) {\r
+        // interval grows\r
+        previous += 1;\r
+      } else {\r
+        // end of interval\r
         fact.push(intervalStart + "-" + previous);\r
+        intervalStart = 0;\r
+        previous = selectedNodes[j];\r
+      }\r
     } else {\r
-        if (previous) fact.push(previous);\r
+      // we are not in an interval\r
+      if (selectedNodes[j] == previous + 1) {\r
+        // let's begin an interval\r
+        intervalStart = previous;\r
+        previous += 1;\r
+      } else {\r
+        if (previous) {\r
+          fact.push(previous);\r
+        }\r
+        previous = selectedNodes[j];\r
+      }\r
     }\r
-    return fact;\r
+  } // for end\r
+  // at the end of the loop, two cases\r
+  // we were still in an interval, then add it to the result\r
+  // we were not in an interval, then add the last selectednode (previous)\r
+  if (intervalStart) {\r
+    fact.push(intervalStart + "-" + previous);\r
+  } else if (previous) {\r
+    fact.push(previous);\r
+  }\r
+  return fact;\r
 }\r
 \r
 // expand a list of nodes containing dash intervals\r
 // 1-3,5,9 -> 1,2,3,5,9\r
-function expand(factExp) {\r
-    exp = [];\r
-    for (i = 0; i < factExp.length; i++) {\r
-        dashExpression = factExp[i].split("-");\r
-        if (dashExpression.length == 2) {\r
-            for (j = parseInt(dashExpression[0]); j < (parseInt(dashExpression[1]) + 1); j++)\r
-                exp.push(j);\r
-        } else exp.push(parseInt(factExp[i]));\r
+function expand(input) {\r
+  var factorized = input.split(",");\r
+  var expanded = [];\r
+  for (var i = 0; i < factorized.length; i++) {\r
+    var d = factorized[i].split("-");\r
+    if (d.length == 2) {\r
+      for (var j = parseInt(d[0]); j < parseInt(d[1]) + 1; j++) {\r
+        expanded.push(j);\r
+      }\r
+    } else {\r
+      expanded.push(parseInt(factorized[i]));\r
+    }\r
+  }\r
+  expanded.sort(sortfunction);\r
+  for (var i = 1; i < expanded.length; i++) {\r
+    if (expanded[i] == expanded[i - 1]) {\r
+      expanded.splice(i--, 1);\r
     }\r
-    exp.sort(sortfunction);\r
-    for (var i = 1; i < exp.length; i++) { if (exp[i] == exp[i - 1]) { exp.splice(i--, 1); } }\r
-    return exp;\r
+  }\r
+  return expanded;\r
 }\r
 \r
 function parseNodebox() {\r
-    input = nodebox.value;\r
-    selectedNodes = expand(input.split(","));\r
-    myrender();\r
+  selectedNodes = expand(nodebox.value);\r
+  myrender();\r
 }\r
 \r
-function debugaxis(axisLength) {\r
-    //Shorten the vertex function\r
-    function v(x, y, z) {\r
-        return new THREE.Vertex(new THREE.Vector3(x, y, z));\r
-    }\r
-\r
-    //Create axis (point1, point2, colour)\r
-    function createAxis(p1, p2, color) {\r
-        var line, lineGeometry = new THREE.Geometry(),\r
-                    lineMat = new THREE.LineBasicMaterial({ color: color, lineWidth: 2 });\r
-        lineGeometry.vertices.push(p1, p2);\r
-        line = new THREE.Line(lineGeometry, lineMat);\r
-        scene.add(line);\r
-    }\r
-\r
-    createAxis(v(0, 0, 0), v(axisLength, 0, 0), 0xFF0000);\r
-    createAxis(v(0, 0, 0), v(0, axisLength, 0), 0x00FF00);\r
-    createAxis(v(0, 0, 0), v(0, 0, axisLength), 0x0000FF);\r
-};\r
-\r
 // set the camera position according two angles theta and phi and the distance\r
 // and display the scene\r
 function myrender() {\r
-    //infobox.innerHTML = " Cam Pos = " + camera.position.x + "," + camera.position.y + "," + camera.position.z\r
-    //                + " - " + theta + "," + phi + ","+ distance;\r
-    //infobox.innerHTML = selectedNodes;\r
-    camera.position.x = distance * Math.sin(theta * Math.PI / 360) * Math.cos(phi * Math.PI / 360);\r
-    camera.position.y = distance * Math.sin(this.phi * Math.PI / 360);\r
-    camera.position.z = distance * Math.cos(this.theta * Math.PI / 360) * Math.cos(this.phi * Math.PI / 360);\r
-    camera.lookAt(scene.position);\r
-    camera.updateMatrix();\r
-\r
-    for (i = 0; i < objects.length; i++)\r
-        if (selectedNodes.indexOf(objects[i].name) != -1) objects[i].material.color.setHex(0xff0000);\r
-        else objects[i].material.color.setHex(0xffffff);\r
-    renderer.render(scene, camera);\r
+  camera.position.x = distance * Math.sin(theta * Math.PI / 360) * Math.cos(phi * Math.PI / 360);\r
+  camera.position.y = distance * Math.sin(this.phi * Math.PI / 360);\r
+  camera.position.z = distance * Math.cos(this.theta * Math.PI / 360) * Math.cos(this.phi * Math.PI / 360);\r
+  camera.lookAt(scene.position);\r
+  camera.updateMatrix();\r
+  \r
+  for (i = 0; i < objects.length; i++) {\r
+    if (selectedNodes.indexOf(objects[i].name) != -1) {\r
+      objects[i].material.color.setHex(0x0099CC);\r
+    } else {\r
+      if (objects[i].state == "Busy") {\r
+        objects[i].material.color.setHex(0x9943BE);\r
+      } else if (objects[i].state == "Alive") {\r
+        objects[i].material.color.setHex(0x7FFF00);\r
+      } else {\r
+        objects[i].material.color.setHex(0xFF3030);\r
+      }\r
+    }\r
+  }\r
+  renderer.render(scene, camera);\r
 }\r
 \r
-// rigthbutton is used for rotations \r
+// rigthbutton is used for rotations\r
 function onDocumentMouseMoveRot(event) {\r
-    NewmouseX = event.clientX;\r
-    NewmouseY = event.clientY;\r
-    DeltaX = NewmouseX - mouseX;\r
-    DeltaY = NewmouseY - mouseY;\r
-\r
-    mouseX = NewmouseX;\r
-    mouseY = NewmouseY;\r
-\r
-    theta -= DeltaX;\r
-    phi += DeltaY;\r
-    if (phi > 180) phi = 180;\r
-    if (phi < -180) phi = -180;\r
-    myrender();\r
+  var NewmouseX = event.clientX;\r
+  var NewmouseY = event.clientY;\r
+  var DeltaX = NewmouseX - mouseX;\r
+  var DeltaY = NewmouseY - mouseY;\r
+  \r
+  mouseX = NewmouseX;\r
+  mouseY = NewmouseY;\r
+  \r
+  theta -= DeltaX;\r
+  phi += DeltaY;\r
+  if (phi > 180) phi = 180;\r
+  if (phi < -180) phi = -180;\r
+  myrender();\r
 }\r
 // mousewheel is used for zoom\r
 function Zoom(delta) {\r
-    distance += delta * 5;\r
-    myrender();\r
+  distance += delta * 5;\r
+  myrender();\r
 }\r
 \r
 // return the graphical node under the mouse\r
 function findNodeUnderMouse(event) {\r
-    var vector = new THREE.Vector3(((event.clientX - offX) / window3DWidth) * 2 - 1, -((event.clientY - offY) / window3DHeight) * 2 + 1, 0.5);\r
-    projector.unprojectVector(vector, camera);\r
-    var ray = new THREE.Ray(camera.position, vector.subSelf(camera.position).normalize());\r
-    var intersects = ray.intersectObjects(objects);\r
-    if (intersects.length > 0) {\r
-        return intersects[0];\r
-    } else return null;\r
+  var x, y, z;\r
+  \r
+  event.preventDefault();\r
+  \r
+  x = ((event.clientX - div3d.offsetLeft) / div3d.offsetWidth) * 2 - 1;\r
+  y = -((event.clientY - div3d.offsetTop) / div3d.offsetHeight) * 2 + 1;\r
+  z = 0.5;\r
+  \r
+  var vector = new THREE.Vector3(x, y, z);\r
+  projector.unprojectVector(vector, camera);\r
+  \r
+  var ray = new THREE.Ray(camera.position, vector.subSelf(camera.position).normalize());\r
+  var intersects = ray.intersectObjects(objects);\r
+  \r
+  if (intersects.length > 0) {\r
+    return intersects[0];\r
+  } else {\r
+    return null;\r
+  }\r
 }\r
 \r
 // display some info of the node under the mouse\r
 function displayNodeInfo(e) {\r
-    obj = findNodeUnderMouse(e);\r
-    if (obj) infobox.innerHTML = 'Node info : number ' + obj.object.name + " with id " + nodes_gre[obj.object.id][4];\r
-    //    else infobox.innerHTML = e.clientX + "," + e.clientY + " - " + offX + "," + offY;\r
+  obj = findNodeUnderMouse(e);\r
+  if (obj) {\r
+    infobox.innerHTML = 'Node info : number ' + obj.object.name + " with id " + obj.object.uid;\r
+  }\r
 }\r
 \r
 // select/unselect nodes\r
 function toggleNode(obj) {\r
-    nodeId = obj.object.name;\r
-    i = selectedNodes.indexOf(nodeId);\r
-    if (i == -1) selectedNodes.push(nodeId);\r
-    else selectedNodes.splice(i, 1);\r
-    nodebox.value = factorize().join(",");\r
-    myrender();\r
+  var nodeId = obj.object.name;\r
+  var index = selectedNodes.indexOf(nodeId);\r
+  \r
+  if (index == -1) {\r
+    selectedNodes.push(nodeId);\r
+  } else {\r
+    selectedNodes.splice(index, 1);\r
+  }\r
+  nodebox.value = factorize().join(",");\r
+  myrender();\r
 }\r
 \r
 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\r
 // Mouse handling functions\r
 //\r
 function OnMouseDown(e) {\r
-    var clickType = 'LEFT';\r
-    if (e.type != sTestEventType) return true\r
+  var clickType = 'LEFT';\r
+  if (e.type != 'mousedown') return true\r
     if (e.which) {\r
-        if (e.which == 3) clickType = 'RIGHT';\r
-        if (e.which == 2) clickType = 'MIDDLE';\r
-        //infobox.innerHTML = clickType + " - Cam Pos = " + camera.position.x + "," + camera.position.y + "," + camera.position.z;\r
+      if (e.which == 3) clickType = 'RIGHT';\r
+      if (e.which == 2) clickType = 'MIDDLE';\r
+      //infobox.innerHTML = clickType + " - Cam Pos = " + camera.position.x + "," + camera.position.y + "," + camera.position.z;\r
     }\r
     mouseX = e.clientX;\r
-    mouseY = e.clientY;\r
-    if (clickType == 'RIGHT') {\r
-        document.onmousemove = onDocumentMouseMoveRot;\r
-    }\r
-    if (clickType == 'LEFT') {\r
-        obj = findNodeUnderMouse(e);\r
-        if (obj != null) toggleNode(obj);\r
-    }\r
+  mouseY = e.clientY;\r
+  if (clickType == 'RIGHT') {\r
+    document.onmousemove = onDocumentMouseMoveRot;\r
+  }\r
+  if (clickType == 'LEFT') {\r
+    obj = findNodeUnderMouse(e);\r
+    if (obj != null) toggleNode(obj);\r
+  }\r
 }\r
 \r
 function OnMouseUp(e) {\r
-    document.onmousemove = displayNodeInfo;\r
+  document.onmousemove = displayNodeInfo;\r
 }\r
 \r
 // This function was taken here : http://www.adomas.org/javascript-mouse-wheel/\r
 // Event handler for mouse wheel event.\r
 \r
 function wheel(event) {\r
-    var delta = 0;\r
-    if (!event) /* For IE. */\r
-        event = window.event;\r
-    if (event.wheelDelta) { /* IE/Opera. */\r
-        delta = event.wheelDelta / 120;\r
-    } else if (event.detail) { /** Mozilla case. */\r
-        /** In Mozilla, sign of delta is different than in IE.\r
-        * Also, delta is multiple of 3.\r
-        */\r
-        delta = -event.detail / 3;\r
-    }\r
-    /** If delta is nonzero, handle it.\r
-    * Basically, delta is now positive if wheel was scrolled up,\r
-    * and negative, if wheel was scrolled down.\r
-    */\r
-    if (delta)\r
-        Zoom(delta);\r
-    /** Prevent default actions caused by mouse wheel.\r
-    * That might be ugly, but we handle scrolls somehow\r
-    * anyway, so don't bother here..\r
-    */\r
-    if (event.preventDefault)\r
-        event.preventDefault();\r
-    event.returnValue = false;\r
-}          \r
+  var delta = 0;\r
+  if (!event) /* For IE. */\r
+    event = window.event;\r
+  if (event.wheelDelta) { /* IE/Opera. */\r
+    delta = event.wheelDelta / 120;\r
+  } else if (event.detail) { /** Mozilla case. */\r
+    /** In Mozilla, sign of delta is different than in IE.\r
+     * Also, delta is multiple of 3.\r
+     */\r
+    delta = -event.detail / 3;\r
+  }\r
+  /** If delta is nonzero, handle it.\r
+   * Basically, delta is now positive if wheel was scrolled up,\r
+   * and negative, if wheel was scrolled down.\r
+   */\r
+  if (delta)\r
+    Zoom(delta);\r
+  /** Prevent default actions caused by mouse wheel.\r
+   * That might be ugly, but we handle scrolls somehow\r
+   * anyway, so don't bother here..\r
+   */\r
+  if (event.preventDefault)\r
+    event.preventDefault();\r
+  event.returnValue = false;\r
+}\r
index c1c6599..c00ea64 100644 (file)
@@ -1 +1,18 @@
-<div class='senslab-map'><div id='div3d'></div></div>
+<div class="senslabmap">
+  <div class="row" id="maps">
+    <div class="span8 offset2">
+      <div id="selectionbox">
+        Selected Nodes <input type="text" placeholder="1-10,24,25" id="nodebox" />
+        <button class="btn" id="btnSave" value="Save">Save</button>
+      </div>
+      <div id="div3d" oncontextmenu="return false;"></div>
+      <div id="infobox"></div>
+      <div style="text-align:right">
+        <div id="node_alive"></div> Alive -
+        <div id="node_down"></div> Down -
+        <div id="node_selected"></div> Selected -
+        <div id="node_used"></div> Busy
+      </div>
+    </div>
+  </div>
+</div>
\ No newline at end of file
index d97126f..d4b68f4 100644 (file)
@@ -142,19 +142,26 @@ def account_process(request):
         messages.success(request, 'Sucess: Password Updated.')
         return HttpResponseRedirect("/portal/account/")
 
+# XXX TODO: Factorize with portal/registrationview.py
+
     elif 'generate' in request.POST:
         for account_detail in account_details:
             for platform_detail in platform_details:
                 if platform_detail['platform_id'] == account_detail['platform_id']:
                     if 'myslice' in platform_detail['platform']:
+                        from Crypto.PublicKey import RSA
+                        private = RSA.generate(1024)
+                        private_key = json.dumps(private.exportKey())
+                        public  = private.publickey()
+                        public_key = json.dumps(public.exportKey(format='OpenSSH'))
                         # Generate public and private keys using SFA Library
-                        from sfa.trust.certificate  import Keypair
-                        k = Keypair(create=True)
-                        public_key = k.get_pubkey_string()
-                        private_key = k.as_pem()
-                        private_key = ''.join(private_key.split())
-                        public_key = "ssh-rsa " + public_key
-                        keypair = '{"user_public_key":"'+ public_key + '", "user_private_key":"'+ private_key + '"}'
+#                        from sfa.trust.certificate  import Keypair
+#                        k = Keypair(create=True)
+#                        public_key = k.get_pubkey_string()
+#                        private_key = k.as_pem()
+#                        private_key = ''.join(private_key.split())
+#                        public_key = "ssh-rsa " + public_key
+                        keypair = '{"user_public_key":'+ public_key + ', "user_private_key":'+ private_key + '}'
 #                       keypair = re.sub("\r", "", keypair)
 #                       keypair = re.sub("\n", "\\n", keypair)
 #                       #keypair = keypair.rstrip('\r\n')
index 1f8ab8b..4c7c8f2 100644 (file)
@@ -68,16 +68,24 @@ class RegistrationView (View):
             # XXX validate authority hrn !!
             if PendingUser.objects.filter(email__iexact=reg_email):
                 errors.append('Email already registered.Please provide a new email address.')
+
+# XXX TODO: Factorize with portal/accountview.py
             if 'generate' in request.POST['question']:
-                # Generate public and private keys using SFA Library
-                from sfa.trust.certificate  import Keypair
-                k = Keypair(create=True)
-                public_key = k.get_pubkey_string()
-                private_key = k.as_pem()
-                private_key = ''.join(private_key.split())
-                public_key = "ssh-rsa " + public_key
+                from Crypto.PublicKey import RSA
+                private = RSA.generate(1024)
+                private_key = json.dumps(private.exportKey())
+                public  = private.publickey()
+                public_key = json.dumps(public.exportKey(format='OpenSSH'))
+
+#                # Generate public and private keys using SFA Library
+#                from sfa.trust.certificate  import Keypair
+#                k = Keypair(create=True)
+#                public_key = k.get_pubkey_string()
+#                private_key = k.as_pem()
+#                private_key = ''.join(private_key.split())
+#                public_key = "ssh-rsa " + public_key
                 # Saving to DB
-                keypair = '{"user_public_key":"'+ public_key + '", "user_private_key":"'+ private_key + '"}'
+                keypair = '{"user_public_key":'+ public_key + ', "user_private_key":'+ private_key + '}'
                 #keypair = re.sub("\r", "", keypair)
                 #keypair = re.sub("\n", "\\n", keypair)
                 #keypair = keypair.rstrip('\r\n')
index 1923670..65b369a 100644 (file)
@@ -47,7 +47,7 @@ class ResourceView(TemplateView):
             title      = 'Geographic view',
             domid      = 'resources-map',
             # tab's sons preferably turn this off
-            togglable  = False,
+            togglable  = True,
             query      = resource_query,
             query_all  = resource_query,
             checkboxes = False,
index 9b96d4f..2f22789 100644 (file)
@@ -14,7 +14,7 @@ from plugins.tabs                    import Tabs
 from plugins.hazelnut                import Hazelnut 
 from plugins.resources_selected      import ResourcesSelected
 from plugins.googlemap               import GoogleMap
-from plugins.senslabmap.senslabmap   import SensLabMap
+from plugins.senslabmap              import SensLabMap
 from plugins.querycode               import QueryCode
 from plugins.query_editor            import QueryEditor
 from plugins.active_filters          import ActiveFilters
@@ -117,7 +117,8 @@ class SliceView (LoginRequiredAutoLogoutView):
        
         filter_query_editor = QueryEditor(
             page  = page,
-            query = query_resource_all,
+            query = sq_resource, 
+            query_all = query_resource_all,
             title = "Select Columns",
             domid = 'select-columns',
             )
@@ -132,8 +133,7 @@ class SliceView (LoginRequiredAutoLogoutView):
             domid               = 'filters',
             sons                = [filter_query_editor, filter_active_filters],
             togglable           = True,
-            # start turned off, it will open up itself when stuff comes in
-            toggled             = False,
+            toggled             = 'persistent',
             outline_complete    = True, 
         )
         main_stack.insert (filters_area)
@@ -160,6 +160,7 @@ class SliceView (LoginRequiredAutoLogoutView):
         resources_as_list = Hazelnut( 
             page       = page,
             domid      = 'resources-list',
+            title      = 'List view',
             # this is the query at the core of the slice list
             query      = sq_resource,
             query_all  = query_resource_all,
@@ -171,23 +172,9 @@ class SliceView (LoginRequiredAutoLogoutView):
                 },
             )
 
-        # List area itself is a Stack with hazelnut on top,
-        # and a togglable tabs for customization plugins 
-        resources_as_list_area = Stack(
-            page        = page,
-            title       = 'Resources as a List',
-            domid       = 'resources-list-area',
-            sons= [ resources_as_list, 
-                    Tabs ( page=page,
-                           title="Customize Resources layout",
-                           togglable=True,
-                           toggled='persistent',
-                           domid="customize-resources",
-                           outline_complete=True,
-                           #sons = [ resources_query_editor, resources_active_filters, ],
-                           ),
-                    ],
-            )
+        # with the new 'Filter' stuff on top, no need for anything but the hazelnut
+        resources_as_list_area = resources_as_list 
+
         resources_area = Tabs ( page=page, 
                                 domid="resources",
                                 togglable=True,