/**
 * Copyright © 2007 Garmin Ltd. or its subsidiaries.
 *
 * Licensed under the Apache License, Version 2.0 (the 'License')
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an 'AS IS' BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 * 
 * @fileoverview GarminDeviceControlDemo Demonstrates Garmin.DeviceControl.
 * 
 * @author Michael Bina michael.bina.at.garmin.com
 * @version 1.0
 */
var GarminDeviceControlDemo = Class.create();
GarminDeviceControlDemo.prototype = {

	initialize: function(statusDiv, mapId, keysArray) {        
        this.status = $(statusDiv);
        this.mc = new Garmin.MapController(mapId);
        this.keys = keysArray;
        
        this.findDevicesButton = $("findDevicesButton");
        this.cancelFindDevicesButton = $("cancelFindDevicesButton");
        this.deviceSelect = $("deviceSelect");
        this.deviceInfo = $("deviceInfoText");

		this.fileTypeSelect=$("fileTypeSelect");
        this.readDataButton = $("readDataButton");
        this.cancelReadDataButton = $("cancelReadDataButton");
        this.readTracksText = $("readTracksText");
        this.readTracksSelect = $("readTracksSelect");
        this.readWaypointsSelect = $("readWaypointsSelect");
        this.dataString = $("dataString");

        this.writeDataButton = $("writeDataButton");
        this.cancelWriteDataButton = $("cancelWriteDataButton");
        this.writeDataSelect = $("writeDataSelect");
		this.writeDataText = $("writeDataText");
		this.writeDataFilename = $("writeDataFilename");

		this.progressBar = $("progressBar");
		this.progressBarDisplay = $("progressBarDisplay");

		this.garminController = null;
        this.tracks = null;
		this.intializeController();		

		if(this.garminController && this.garminController.isPluginInitialized()) {
	        this.findDevicesButton.disabled = false;
	        this.findDevicesButton.onclick = function() {
	        	this.findDevicesButton.disabled = true;
	        	this.cancelFindDevicesButton.disabled = false;
	        	this.garminController.findDevices();
	        }.bind(this)
		}		
	},
	
	intializeController: function() {
		try {
			this.garminController = new Garmin.DeviceControl();
			this.garminController.register(this);
			
			if(this.garminController.unlock(this.keys)) {
	        	this.setStatus("Plug-in initialized.  Find some devices to get started.");
			} else {
	        	this.setStatus("The plug-in was not unlocked successfully.");
	        	this.garminController = null;
			}
        } catch (e) {
            this.setStatus(e.message);
//            if(e.name == "OutOfDatePluginException") {
//	        	this.setStatus("Your plug-in is out of date.  Please update.");
//            } else if(e.name == "PluginNotInstalledException") { 
//	        	this.setStatus("You'll need to install the Garmin Communicator Plug-In to connect to a device.");
//            } else if(e.name == "BrowserNotSupportedException") { 
//	        	this.setStatus("Your browser is not supported to  use the Garmin Communicator Plug-In.");
//            } else {
//	        	this.setStatus("There was a problem initializing the Garmin Communicator Plug-In.");
//            }
        }
	},

	showProgressBar: function() {
		Element.show(this.progressBar);
	},

	hideProgressBar: function() {
		Element.hide(this.progressBar);
	},

	updateProgressBar: function(value) {
		if (value) {
			var percent = (value <= 100) ? value : 100;
	    	this.progressBarDisplay.style.width = percent + "%";
		}
	},

    onStartFindDevices: function(json) {
        this.setStatus("Looking for connected Garmin devices");
    },

    onFinishFindDevices: function(json) {
       	this.findDevicesButton.disabled = false;
       	this.cancelFindDevicesButton.disabled = true;

        if(json.controller.numDevices > 0) {
            var devices = json.controller.getDevices();
            this.setStatus("Found " + devices.length + " devices.");

			this.listDevices(devices);
			
	        this.cancelReadDataButton.onclick = function() {
	        	this.fileTypeSelect.disabled = false;
	        	this.readDataButton.disabled = false;
	        	this.cancelReadDataButton.disabled = true;
	        	this.writeDataButton.disabled = false;
	        	this.hideProgressBar();
	        	this.garminController.cancelReadFromDevice();
	        }.bind(this)
			
			this.fileTypeSelect.disabled = false;	        
	        this.readDataButton.disabled = false;
	        this.readDataButton.onclick = function() {		
		    	this.readTracksSelect.length = 0;	
		    	this.readWaypointsSelect.length = 0;
				this.mc.map.clearOverlays();
	        	this.fileTypeSelect.disabled = true;
	        	this.readDataButton.disabled = true;
	        	this.cancelReadDataButton.disabled = false;
	        	this.showProgressBar();
	        	if (this.fileTypeSelect.value == 0) {
	        		this.garminController.readFromDevice();
	        		this.writeDataButton.disabled = false;
	        	} else if (this.fileTypeSelect.value == 1) {
	        		this.garminController.readFromDeviceFitness();
	        		this.writeDataButton.disabled = true;
	        	}
	        }.bind(this)


	        this.writeDataSelect.disabled = false;
			this.writeDataSelect.onchange = function() {
				this.loadWriteData(this.writeDataSelect.value);
			}.bind(this)
			this.loadWriteData(this.writeDataSelect.value);

	        this.cancelWriteDataButton.onclick = function() {
	        	this.writeDataButton.disabled = false;
	        	this.cancelWriteDataButton.disabled = true;
	        	this.hideProgressBar();
	        	this.garminController.cancelWriteToDevice();
	        }.bind(this)

	        this.writeDataButton.disabled = false;	        
	        this.writeDataButton.onclick = function() {
	        	this.writeDataButton.disabled = true;
	        	this.cancelWriteDataButton.disabled = false;
	        	this.showProgressBar();
	        	this.garminController.writeToDevice(this.writeDataText.value, this.writeDataFilename.value);
	        }.bind(this);
        } else {
			this.setStatus("No devices found.");
        }
    },

	onCancelFindDevices: function(json) {
    	this.setStatus("Find cancelled");
    },

	listDevices: function(devices) {
		for( var i=0; i < devices.length; i++ ) {
           	this.deviceSelect.options[i] = new Option(devices[i].getDisplayName(),devices[i].getNumber());
           	if(devices[i].getNumber() == this.garminController.deviceNumber) {
           		this.deviceSelect.selectedIndex = i;
           		this.showDeviceInfo(devices[i]);
           	}
		}
		this.deviceSelect.onchange = function() {
			var device = this.garminController.getDevices()[this.deviceSelect.value];
			this.showDeviceInfo(device);
		
			this.garminController.setDeviceNumber(this.deviceSelect.value);
		}.bind(this)
		this.deviceSelect.disabled = false;
	},

	showDeviceInfo: function(device) {
		this.deviceInfo.innerHTML = "Part Number:\t\t" + device.getPartNumber() + "\n";
		this.deviceInfo.innerHTML += "Software Version:\t" + device.getSoftwareVersion() + "\n";
		this.deviceInfo.innerHTML += "Description:\t\t" + device.getDescription() + "\n";
		this.deviceInfo.innerHTML += "Id:\t\t\t" + device.getId() + "\n\n";
		
		var dataTypes = device.getDeviceDataTypes().values();		
		var typeListSize = dataTypes.length;
		for (var i = 0; i < typeListSize; i++) {
			this.deviceInfo.innerHTML += "-DataType---------------\n"
			this.deviceInfo.innerHTML += "  Name:\t\t" + dataTypes[i].getDisplayName() + "\n";
			this.deviceInfo.innerHTML += "  Extension:\t" + dataTypes[i].getFileExtension() + "\n";
			this.deviceInfo.innerHTML += "  Read:\t\t" + dataTypes[i].hasReadAccess() + "\n";
			this.deviceInfo.innerHTML += "  Write:\t" + dataTypes[i].hasWriteAccess() + "\n\n";			
		}
	},

    onProgressReadFromDevice: function(json) {
	  	this.updateProgressBar(json.progress.getPercentage());
    	this.setStatus(json.progress);
    },
    
	onCancelReadFromDevice: function(json) {
    	this.setStatus("Read cancelled");
    },

    onFinishReadFromDevice: function(json) {
    	this.fileTypeSelect.disabled = false;
       	this.readDataButton.disabled = false;
       	this.cancelReadDataButton.disabled = true;
	    this.hideProgressBar();
    	this.setStatus("Data read from device.");
    	var gpsData = json.controller.gpsData;
    	this.dataString.value = json.controller.gpsDataString;

		if (this.fileTypeSelect.value == 0) {
	    	this.setStatus("Parsing track data...");
	    	var factory = new Garmin.GpsDataFactory();
	    	factory.parseGpxDocument(gpsData);
			this.tracks = factory.getTracks();
			this.waypoints = factory.getWaypoints();
	    	this.setStatus(this.tracks.length + " tracks and " + this.waypoints.length + " waypoints found");
	    	this.listTracks(this.tracks);
	    	this.listWayPoints(this.waypoints);			
		}
		
    },
    
    listTracks: function(tracks) {
    	this._clearHtmlSelect(this.readTracksSelect);
    	var tracksDrawable = 0;
		for( var i=0; i < tracks.length; i++ ) {
			var trk = tracks[i];
			if(trk.isDrawable()) {
	           	this.readTracksSelect.options[i] = new Option(trk.getStartDate().getDateString() + " - (" + trk.getDuration() + ")",i);
	           	tracksDrawable++;
	        }
		}
		if(tracksDrawable > 0) {
			if (this.readTracksSelect.selectedIndex == -1) {
				this.readTracksSelect.selectedIndex = 0;
			} 
			this.displayTrack(tracks[this.readTracksSelect.selectedIndex]);				
		}
		this.readTracksSelect.onchange = function() {
			var trk = this.tracks[this.readTracksSelect.selectedIndex];
			this.displayTrack(trk);
		}.bind(this)
		this.readTracksSelect.disabled = false;
    },
    
    listWayPoints: function(waypoints) {
		this._clearHtmlSelect(this.readWaypointsSelect);
		for( var i=0; i < waypoints.length; i++ ) {
			var wpt = waypoints[i];
	        this.readWaypointsSelect.options[i] = new Option(wpt.getName(),i);
		}
		this.readWaypointsSelect.onchange = function() {
			var wpt = this.waypoints[this.readWaypointsSelect.selectedIndex];
			this.displayWayPoint(wpt);
		}.bind(this)
		this.readWaypointsSelect.disabled = false;
    },
    
	/**Sets the size of the select to zero which essentially clears it from 
	 * any values.
	 * @private
	 */
    _clearHtmlSelect: function(select) {
		if(select) {
			select.size = 0;
		}
    },

    displayTrack: function(track) {
		this.mc.map.clearOverlays();
        this.mc.centerAndScale(track.getStartLat(), track.getStartLng());
        this.mc.drawTrack(track);

        var factory = new Garmin.GpsDataFactory();
        var tracks = new Array();
        tracks.push(track);
        this.writeDataText.value = factory.produceGpxString(tracks, null);
    },

    displayWayPoint: function(wpt) {
		this.mc.map.clearOverlays();
        this.mc.centerAndScale(wpt.getLat(), wpt.getLng());
        this.mc.drawWayPoint(wpt);
        
        var factory = new Garmin.GpsDataFactory();
        var waypoints = new Array();
        waypoints.push(wpt);
        this.writeDataText.value = factory.produceGpxString(null, waypoints);
     },

	loadWriteData: function(filepath) {
		new Ajax.Request(filepath, {
			onSuccess: function(resp) {
				this.writeDataText.value = resp.responseText;
			}.bind(this),
			onFailure: function(resp) {
				alert("Error deleting all pending activities");
			}
		});
	},

    onStartWriteToDevice: function(json) { 
    	this.setStatus("Writing data to to the device");
    },

    onCancelWriteToDevice: function(json) { 
    	this.setStatus("Writing cancelled");
    },

    /**
     * The device already has a file with this name on it.  Do we want to override?  1 is yes, 2 is no
     */ 
    onWaitingWriteToDevice: function(json) { 
        if(confirm(json.message.getText())) {
            this.setStatus('Overwriting file');
            json.controller.respondToMessageBox(true);
        } else {
            this.setStatus('Will not be overwriting file');
            json.controller.respondToMessageBox(false);
        }
    },

    onProgressWriteToDevice: function(json) {
	  	this.updateProgressBar(json.progress.getPercentage());
    	this.setStatus(json.progress);
    },

    onFinishWriteToDevice: function(json) {
	    this.hideProgressBar();
    	this.setStatus("Data written to the device.");
	    this.hideProgressBar();
       	this.writeDataButton.disabled = false;
       	this.cancelWriteDataButton.disabled = true;
    },

    onException: function(json) {
	    alert("Exception: " + json.msg.message);
    },

	setStatus: function(statusText) {
	    this.status.innerHTML = statusText;
	}
};